New Upstream Snapshot - r-cran-popepi

Ready changes

Summary

Merged new upstream version: 0.4.10+git20221003.1.6b03392+dfsg (was: 0.4.10+dfsg).

Resulting package

Built on 2023-01-19T14:08 (took 18m54s)

The resulting binary packages can be installed (if you have the apt repository enabled) by running one of:

apt install -t fresh-snapshots r-cran-popepi

Diff

diff --git a/DESCRIPTION b/DESCRIPTION
index 6137c4c..99c6d5a 100644
--- a/DESCRIPTION
+++ b/DESCRIPTION
@@ -7,9 +7,9 @@ Authors@R: c(
     person("Matti", "Rantanen", , "matti.rantanen@statfinn.com", role = "aut"),
     person("Karri", "Seppa", , "karri.seppa@cancer.fi", role = "ctb")
   )
-Author: Joonas Miettinen [cre, aut] (<https://orcid.org/0000-0001-8624-6754>),
-  Matti Rantanen [aut],
-  Karri Seppa [ctb]
+Author: Joonas Miettinen [cre, aut]
+    (<https://orcid.org/0000-0001-8624-6754>), Matti Rantanen [aut], Karri
+    Seppa [ctb]
 Maintainer: Joonas Miettinen <joonas.miettinen@cancer.fi>
 Description: Enables computation of epidemiological statistics, including
     those where counts or mortality rates of the reference population are
@@ -36,6 +36,4 @@ Language: en-GB
 LazyData: true
 RoxygenNote: 7.2.1
 NeedsCompilation: no
-Packaged: 2022-09-16 07:49:50 UTC; joonas.miettinen
-Repository: CRAN
-Date/Publication: 2022-09-25 16:40:02 UTC
+Packaged: 2023-01-19 13:55:24 UTC; root
diff --git a/MD5 b/MD5
deleted file mode 100644
index 1ad3f2e..0000000
--- a/MD5
+++ /dev/null
@@ -1,142 +0,0 @@
-ba4af2506083a32f3dc8ec56140c4305 *DESCRIPTION
-535ba352a75bd85f478f8c2e4731209c *NAMESPACE
-c1c981feb4077469e5c4f5dbe5976ef0 *NEWS.md
-7ac196a0ba2e0576911f5b19ff80b45f *R/S3_definitions.R
-5e3990e28fd56322cd0fc4261606c593 *R/Surv.R
-f0e3448c2e62238304ba9d936888cc27 *R/aggregating.R
-338a72fadf7a048a2fb99aa9a1d453c4 *R/data_document.R
-f6e8887c2c30620dbac2331be695cdb9 *R/direct_adjusting.R
-3b224d4a39adac5dcae5e23d902be746 *R/evaluation.R
-c2f087ec5aea7166d7b6f2f96d4b997e *R/flexyargs.R
-6d1ef35703f9670b04a2a2073f3cbe87 *R/fractional_years.R
-431a22e6aad6ce0c4df2bbedb5ee4c9a *R/incidence_rates.R
-338d3049dc118d009a92cc21a834c35a *R/incidence_rates_utils.R
-84437008fb3afc5e62a5cd163113d599 *R/lexpand.R
-66037734d577055aaad5c85486333e1b *R/long_df_and_array.R
-62a4fd54fac7f7ea505f37debf5bbb8c *R/ltable.R
-74dddb2fad46ba8a6062b3ba43f80019 *R/mean_survival.R
-4a811e31873177de5ebbbfda97626407 *R/popEpi_package.r
-0d57cae6ad721b6326074cad0707b8cf *R/pophaz.R
-f22959fac4b0cec375aca8c9b38c51e9 *R/prevalence.R
-9c756502961d4f96e2cf291d59ddb5a2 *R/relative_poisson.R
-edb847575f8fe5a83e69a6b86ca10c31 *R/relative_poisson_net_survival.R
-96dabe89cd610ffd8f696ad0079a9342 *R/sir.R
-40ec7c438f8f0da21f9d38057c788312 *R/sir_utils.R
-3672931fdd079b2b7bc64d9c009e116a *R/splitLexisDT.R
-82cadd5ee91fd0322aa233b2ac25fb86 *R/splitMulti.R
-ad2ca67705b62eb3ba846735991df48a *R/splitting_utility_functions.R
-2395e9993f6139db23707b45b6ccb9af *R/startup_message.R
-f8aea8b79b2c6da3d071d93f7f6ff13f *R/survival_aggregated.R
-ea41b8eaa3bc9cce84d9af4e7e3a6aa6 *R/survival_lexis.R
-22ed929d72c918ca92f4f6f2bba4867d *R/survival_utility_functions.R
-be31aacb98b9ff25a28a2afdfef8c0ea *R/utility_functions.R
-03be45e738826a355bf72ee2da9720d9 *R/weighted_table.R
-4eff267b8695980db96cb03fdc065b04 *README.md
-a93c6b14146112e0036831c8c8882cd5 *build/partial.rdb
-393896cf26615010bd75e5e4c62a4d63 *build/vignette.rds
-42a2e450ebc81d8cf32d7eb9f3efc406 *data/ICSS.rdata
-d15a881aeee84aa9307933f84acbe2ce *data/meanpop_fi.rda
-ba6bcd1044f5a25c868742359b4338ab *data/popmort.rdata
-9e5330442b29336fd3701ffe5127e4a4 *data/sibr.rdata
-acb6b4afdc41b51495ca354054899b43 *data/sire.rdata
-1f3ba42ee14403d7e17eff9ee19d9adc *data/stdpop101.rdata
-905a241496ce00690f2d227753b62364 *data/stdpop18.rdata
-b79094b0fa30222f908d4624b3ef7ee2 *inst/WORDLIST
-5fadf738581382aba8f336c577b3f007 *inst/doc/sir.R
-cd1fbd52e290ec8e854b9087a33d533d *inst/doc/sir.Rmd
-fb740191d15aaff0276dc7dc03c31258 *inst/doc/sir.html
-632dbb732de09105b715e22df0167217 *inst/doc/survtab_examples.R
-e3fcf1e6d707b9a160daaddef78ea27e *inst/doc/survtab_examples.Rmd
-2089bbfbbc7d2cda6d23c49b2da87e69 *inst/doc/survtab_examples.html
-c13a6c22ce194ec8b2f01b2cabda5d1a *man/ICSS.Rd
-335f3b865f5d689816cea1b073c44596 *man/Lexis_fpa.Rd
-a57913a664cabeabeafe5aa0d014a1ec *man/RPL.Rd
-ef36c1f7855e56c186e8d5e6cfd6694c *man/Surv.Rd
-b0ec6686fe92c944616bebdc5a5274ff *man/adjust.Rd
-68cbaad567b4a4b9c9c3ad326fe6ebcf *man/aggre.Rd
-37ced5d82fce2a80f149a6b9ff21f959 *man/all_names_present.Rd
-50782923c789b77771ce61618f5c6e58 *man/array_df_ratetable_utils.Rd
-85701435d8e55ac2bfd9b41645479031 *man/as.Date.yrs.Rd
-206f63d1784456e627ef52c17979789c *man/as.aggre.Rd
-4f64ac7af84cc7481cf795a7b4f5b22d *man/cast_simple.Rd
-8424da25d2d4664a03e9f3353f706bc3 *man/cut_bound.Rd
-9d87b572c13f3b5cf79e31fd3f7b10e1 *man/direct_standardization.Rd
-a3c7c6566b8c031e8d60db60935f3f61 *man/fac2num.Rd
-8cdc3edc12c8f6055d0a9f3593ec75e1 *man/flexible_argument.Rd
-00dc9a2bda50a073d117e85b82bc5dda *man/get.yrs.Rd
-adc3f6ad88a58248ae1f92b71fe2d860 *man/is.Date.Rd
-041347f1969b595f85cf0d24654b5248 *man/is_leap_year.Rd
-3a94c35655e9f048af39a10f0effd6e7 *man/lexpand.Rd
-fdf37774de766dcd30093dbc71bee607 *man/lines.sirspline.Rd
-a34febc8e4845cabd474e010d259d717 *man/lines.survmean.Rd
-56e27653855222fa8acb0d5996beff61 *man/lines.survtab.Rd
-97a7024ad28bee8899ccb5f458127ac7 *man/lower_bound.Rd
-c33adb55f917f147c592bed69a69b6fe *man/ltable.Rd
-4d64d1bd71ed9e5a510dc07841b3f8e5 *man/meanpop_fi.Rd
-5d547328b5e356eb2d23ab579ff94b59 *man/na2zero.Rd
-e8a763bbb8bffacd54fa044089b73bcd *man/plot.rate.Rd
-220e841bb017cb441242d3c2138115fb *man/plot.sir.Rd
-4ef693e3fa1b4ec77ad3489af1c35f85 *man/plot.sirspline.Rd
-a48ad93449b1aa8aa985c1bdf883fce0 *man/plot.survmean.Rd
-739eaa715f5dc73758530bee135a036b *man/plot.survtab.Rd
-c48c4652161a4671ee4cf50628a005e4 *man/poisson.ci.Rd
-77ca8adad5035322270e52f1c2e9139b *man/popEpi.Rd
-a7b9600e477a7e86b381b3f7d1e274f3 *man/pophaz.Rd
-661a6bbec3a708a0cc5367af5094afa1 *man/popmort.Rd
-64ea0b4ecb4249ac08c0b334b61859ef *man/prepExpo.Rd
-1f6db7311fbf4477fc510f9207499b11 *man/print.aggre.Rd
-b5edcc6693f6285e395a939eaed92ee1 *man/print.rate.Rd
-f3f079f48206e63832a271194166a3f1 *man/print.survtab.Rd
-c59d6ef5ed587a7616fdf06a312bc799 *man/rate.Rd
-e79bb61cf2b2e84368e08e33efb91f9f *man/rate_ratio.Rd
-d4d6543b3ed7b831a629c5975e85c1cc *man/relpois.Rd
-ba6e109dbda1259076a78417eabe9f56 *man/relpois_ag.Rd
-e971c89e11d25768c0a2b7f7c5972018 *man/robust_values.Rd
-b0b000728004c4bd01ab964230a1a0ec *man/rpcurve.Rd
-3e2457d5489c2fc1591c5a3c303f3e07 *man/setaggre.Rd
-5a97ddc2e9b68a65552e70c3eb88ef67 *man/setclass.Rd
-aec9591da20b5c92ac513eb7f0deea11 *man/setcolsnull.Rd
-2082f435648c0e0637e2b74232a87c69 *man/sibr.Rd
-65ed6829d48e2f25d6d01bfba90f655c *man/sir.Rd
-3c0fb1c0183446a6daab340590cd7a72 *man/sir_exp.Rd
-2e6e257cad3b538550a60e62f2ac670b *man/sir_ratio.Rd
-61c5f17ca120f073371552925c0b6776 *man/sire.Rd
-ad4d7c22ba9696ac0560340369ff95c5 *man/sirspline.Rd
-77d3a49456cb62be745a0710cfdf1bbf *man/splitLexisDT.Rd
-d46a324a333fa8afb79aebfcefc25a0b *man/splitMulti.Rd
-c1a63e1ecc9215d04abedd5005541923 *man/stdpop101.Rd
-5f5c79ba200cf05c56f7dbca62e22440 *man/stdpop18.Rd
-8555dd3fec52fad9b1e2c215c33bbaea *man/summary.aggre.Rd
-b0e3f88653965958a872d51bd97f41bf *man/summary.survtab.Rd
-9c3c6ba441327d6cd88b684decf79dbf *man/survmean.Rd
-81c18b1fdf771038fa0eba9bbc81cec9 *man/survtab.Rd
-757faa9fdd0b7546dff9c9a58f47e8f5 *man/survtab_ag.Rd
-ca267f6405095228829fc56031bf6cb5 *man/try2int.Rd
-5eb66e1ed29f5b085c6705a825f4fec1 *tests/testthat.R
-00119beca1146e39ca2a46935068915e *tests/testthat/Rplots.pdf
-7649bcb74c39a065e146c7af95db3bdf *tests/testthat/survmean_test_data_01.rds
-4cf13ed3f2ee2ad8cdddeb4feaeb70c1 *tests/testthat/test_Surv.R
-74f0a458ed04d0c586e50c77bab5ecb7 *tests/testthat/test_aggre.R
-d57c5f9349d55d889b97fc6b659c81f1 *tests/testthat/test_epi.R
-997a01976d8039d083b7e22e9f3f0b7f *tests/testthat/test_expo.R
-1a95164a315e190c37ead4e6b606cef7 *tests/testthat/test_lexpand.R
-24181e8c256b26fd5c91e33f43ad2c43 *tests/testthat/test_prevtab.R
-2d7c824c6765652ff843c92f24af2ff5 *tests/testthat/test_rate.R
-0d7d8a1b889eb37f56eeefb4e6eb9a09 *tests/testthat/test_relpois_mean_curve.R
-98bc582f455f7820698b6621c7acaa6e *tests/testthat/test_sir.R
-280c74b49ad712d28c65790ea53a2908 *tests/testthat/test_splitLexisDT.R
-181ff6622b7e91570dc02c4dcae102df *tests/testthat/test_splitMulti.R
-8aef60a92539712ed5a5575759893366 *tests/testthat/test_splitting_attributes.R
-fc7548c2dea21aee0c4c3cfcfe4f1d72 *tests/testthat/test_splitting_breaks.R
-544ba7d91fcd1820b41e8bbadfe48fda *tests/testthat/test_splitting_randomly_on_fixed_data.R
-564b3975dcceb0b2b099a49c64805af0 *tests/testthat/test_splitting_randomly_on_random_data.R
-e99bfa9755bf8fcc612cc2a3363b4c14 *tests/testthat/test_survmean.R
-82c4d07ac62512757883bc144e09e18a *tests/testthat/test_survtab_adjusted.R
-81256094125f368f9ecc1cc6e9cd178f *tests/testthat/test_survtab_bad_surv_ints.R
-0d42fb7d69144b00740d82b406392f6b *tests/testthat/test_survtab_observed.R
-c23a371e345f98d23accfbc0e5460aa6 *tests/testthat/test_survtab_relative.R
-b76e84d95ac49c21cd4980a1527d20da *tests/testthat/test_survtab_usage.R
-1da18d17a357c1c333e9ed3eb4b52844 *tests/testthat/test_utils.R
-011ce65ee701ccb0caab204ece04b5a7 *tests/testthat/test_weighter.R
-cd1fbd52e290ec8e854b9087a33d533d *vignettes/sir.Rmd
-e3fcf1e6d707b9a160daaddef78ea27e *vignettes/survtab_examples.Rmd
diff --git a/NAMESPACE b/NAMESPACE
index a00bb12..ebae18d 100644
--- a/NAMESPACE
+++ b/NAMESPACE
@@ -1,101 +1,101 @@
-# Generated by roxygen2: do not edit by hand
-
-S3method("[",aggre)
-S3method("[",rate)
-S3method("[",sir)
-S3method("[",survmean)
-S3method("[",survtab)
-S3method("[",yrs)
-S3method(as.Date,yrs)
-S3method(as.aggre,data.frame)
-S3method(as.aggre,data.table)
-S3method(as.aggre,default)
-S3method(coef,sir)
-S3method(confint,sir)
-S3method(formula,survmean)
-S3method(formula,survtab)
-S3method(getCall,rate)
-S3method(getCall,sir)
-S3method(getCall,survmean)
-S3method(getCall,survtab)
-S3method(lines,sirspline)
-S3method(lines,survmean)
-S3method(lines,survtab)
-S3method(plot,rate)
-S3method(plot,sir)
-S3method(plot,sirspline)
-S3method(plot,survmean)
-S3method(plot,survtab)
-S3method(print,aggre)
-S3method(print,rate)
-S3method(print,sir)
-S3method(print,sirspline)
-S3method(print,survtab)
-S3method(print,yrs)
-S3method(subset,aggre)
-S3method(subset,rate)
-S3method(subset,survmean)
-S3method(subset,survtab)
-S3method(summary,aggre)
-S3method(summary,survtab)
-export(Lexis_fpa)
-export(RPL)
-export(Surv)
-export(adjust)
-export(aggre)
-export(all_names_present)
-export(array_to_long_df)
-export(array_to_long_dt)
-export(array_to_ratetable)
-export(as.aggre)
-export(cast_simple)
-export(cut_bound)
-export(expr.by.cj)
-export(fac2num)
-export(get.yrs)
-export(is.Date)
-export(is_leap_year)
-export(lexpand)
-export(long_df_to_array)
-export(long_df_to_ratetable)
-export(long_dt_to_array)
-export(long_dt_to_ratetable)
-export(lower_bound)
-export(ltable)
-export(na2zero)
-export(poisson.ci)
-export(prepExpo)
-export(rate)
-export(rate_ratio)
-export(ratetable_to_array)
-export(ratetable_to_long_df)
-export(ratetable_to_long_dt)
-export(relpois)
-export(relpois_ag)
-export(robust_values)
-export(rpcurve)
-export(setaggre)
-export(setcolsnull)
-export(sir)
-export(sir_ag)
-export(sir_exp)
-export(sir_lex)
-export(sir_ratio)
-export(sirspline)
-export(splitLexisDT)
-export(splitMulti)
-export(survmean)
-export(survtab)
-export(survtab_ag)
-export(try2int)
-import(Epi)
-import(data.table)
-import(grDevices)
-import(graphics)
-import(splines)
-import(stats)
-importFrom(data.table,copy)
-importFrom(data.table,is.data.table)
-importFrom(data.table,setDT)
-importFrom(data.table,setattr)
-importFrom(survival,Surv)
+# Generated by roxygen2: do not edit by hand
+
+S3method("[",aggre)
+S3method("[",rate)
+S3method("[",sir)
+S3method("[",survmean)
+S3method("[",survtab)
+S3method("[",yrs)
+S3method(as.Date,yrs)
+S3method(as.aggre,data.frame)
+S3method(as.aggre,data.table)
+S3method(as.aggre,default)
+S3method(coef,sir)
+S3method(confint,sir)
+S3method(formula,survmean)
+S3method(formula,survtab)
+S3method(getCall,rate)
+S3method(getCall,sir)
+S3method(getCall,survmean)
+S3method(getCall,survtab)
+S3method(lines,sirspline)
+S3method(lines,survmean)
+S3method(lines,survtab)
+S3method(plot,rate)
+S3method(plot,sir)
+S3method(plot,sirspline)
+S3method(plot,survmean)
+S3method(plot,survtab)
+S3method(print,aggre)
+S3method(print,rate)
+S3method(print,sir)
+S3method(print,sirspline)
+S3method(print,survtab)
+S3method(print,yrs)
+S3method(subset,aggre)
+S3method(subset,rate)
+S3method(subset,survmean)
+S3method(subset,survtab)
+S3method(summary,aggre)
+S3method(summary,survtab)
+export(Lexis_fpa)
+export(RPL)
+export(Surv)
+export(adjust)
+export(aggre)
+export(all_names_present)
+export(array_to_long_df)
+export(array_to_long_dt)
+export(array_to_ratetable)
+export(as.aggre)
+export(cast_simple)
+export(cut_bound)
+export(expr.by.cj)
+export(fac2num)
+export(get.yrs)
+export(is.Date)
+export(is_leap_year)
+export(lexpand)
+export(long_df_to_array)
+export(long_df_to_ratetable)
+export(long_dt_to_array)
+export(long_dt_to_ratetable)
+export(lower_bound)
+export(ltable)
+export(na2zero)
+export(poisson.ci)
+export(prepExpo)
+export(rate)
+export(rate_ratio)
+export(ratetable_to_array)
+export(ratetable_to_long_df)
+export(ratetable_to_long_dt)
+export(relpois)
+export(relpois_ag)
+export(robust_values)
+export(rpcurve)
+export(setaggre)
+export(setcolsnull)
+export(sir)
+export(sir_ag)
+export(sir_exp)
+export(sir_lex)
+export(sir_ratio)
+export(sirspline)
+export(splitLexisDT)
+export(splitMulti)
+export(survmean)
+export(survtab)
+export(survtab_ag)
+export(try2int)
+import(Epi)
+import(data.table)
+import(grDevices)
+import(graphics)
+import(splines)
+import(stats)
+importFrom(data.table,copy)
+importFrom(data.table,is.data.table)
+importFrom(data.table,setDT)
+importFrom(data.table,setattr)
+importFrom(survival,Surv)
diff --git a/NEWS.md b/NEWS.md
index a623a23..30b965c 100644
--- a/NEWS.md
+++ b/NEWS.md
@@ -1,194 +1,194 @@
-# Changes in 0.4.10
-
--   No changes for user — “Suggests” packages now used conditionally in
-    vignettes/tests/examples as per Writing R Extensions.
-
-# Changes in 0.4.9
-
--   `popEpi::aggre` bugfix: `aggre` now correctly infers which
-    stratifying variables use Lexis time scales when output is cartesian
-    and argument `by` is of type character.
-
-# Changes in 0.4.8
-
--   small internal fixes due to upcoming new package survival release
--   popEpi now includes a wrapper for survival::Surv, so you don’t need
-    to do library(“survival”) when using Surv() in formulae (e.g. in
-    survtab)
-
-# Changes in 0.4.7
-
--   implemented small internal changes due to upcoming R 3.6.0
-
-# Changes in 0.4.6
-
--   implemented small internal changes due to upcoming data.table
-    version
-
-# Changes in 0.4.5
-
--   fixed errors arising from new data.table version
-
-# Changes in 0.4.4
-
--   splitLexisDT/splitMulti bug fix: splitting along multiple time
-    scales *sometimes* produced duplicate transitions (e.g. alive -\>
-    dead in the last two rows). see
-    <https://github.com/FinnishCancerRegistry/popEpi/issues/138> for
-    details.
--   splitLexisDT/splitMulti now retain time.since attribute; this
-    attribute plays a role in cutLexis
--   known issue: splitLexisDT/splitMulti not guaranteed to work
-    identically to splitLexis from Epi when there are NA values in the
-    time scale one is splitting along.
-
-# Changes in 0.4.3
-
--   survtab adjusting was broken with older versions of data.table
-    (tested 1.9.6); therefore popEpi now requires the newest version of
-    data.table!
-
-# Changes in 0.4.2
-
--   **`survtab()` bug fix: standard errors were mis-specified for
-    adjusted curves, e.g. age-adjusted Ederer II estimates. This
-    resulted in too wide confidence intervals! SEE HERE FOR EXAMPLE:
-    [#135](https://github.com/FinnishCancerRegistry/popEpi/issues/135)**.
-    The standard errors and confidence intervals of non-adjusted curves
-    have always been correct.
--   `survtab()` bug fix: confidence level was always 95 % regardless of
-    `conf.level`
-    [#134](https://github.com/FinnishCancerRegistry/popEpi/issues/134)
-
-# Changes in 0.4.1
-
--   `lexpand()` bug fixed (#120): observations were dropped if their
-    entry by age was smaller than the smallest age value, though entry
-    at exit is correct and used now.
--   `sir()` rewrite (#118, #122). New more consistent output, updates on
-    plotting and minor changes in arguments. Introduce very simple
-    `coef()` and `confint()` methods for sir class.
--   new functions in sir family: `sir_ag()`, `sir_lex()` and `sir_exp()`
-    for extracting SMRs from `aggre` and `Lexis` objects.
--   fixed issue in internal test brought by pkg survival version 2.39.5;
-    no changes in functions were needed (#125)
--   robustified `aggre()`; there were issues with Epi pkg dev version
-    which are definitely avoided (#119)
-
-# Changes in 0.4.0
-
--   removed previously deprecated shift.var (#35)
--   popEpi no longer depends on package data.table but imports it - this
-    means the user will have to do library(data.table) separately to
-    make data.table’s functions become usable. Formerly popEpi
-    effectively did library(data.table) when you did library(popEpi).
--   summary.survtab: args t and q behaviour changed
--   survtab: internal weights now based on counts of subjects in
-    follow-up at the start of follow-up (used to be sum of counts/pyrs
-    over all of follow-up)
--   new functions: `rate_ratio()`, `sir_ratio()`
--   small internal changes in preparation for data.table 1.9.8
-
-# Changes in 0.3.1
-
-This is a hotfix. survtab() was causing warnings in certain situations,
-which this update fixes. Also fixed plotting survtab objects so that
-multiple strata are plotted correctly when one or more curves end before
-the longest one as well other small fixes: See Github issues #89, #90,
-#91, and #92.
-
-# Changes in 0.3.0
-
-## Adjusting
-
-Direct adjusting (computing weighted averages of estimates) has been
-generalized. Functions such as `survtab` and `survmean` allow for using
-`adjust()` mini function within formulas, or a separate `adjust`
-argument. Weights are passed separately. See the examples in the next
-chapter. See also `?direct_adjusting`.
-
-## Estimating functions of survival time
-
-The `survtab` function computes observed, net/relative and
-cause-specific survivals as well as cumulative incidence functions for
-`Lexis` data. Any of the supported survival time functions can be easily
-adjusted by any number of categorical variables if needed.
-
-One can also use `survtab_ag` for aggregated data. This means the data
-does not have to be on the subject-level to compute survival time
-function estimates.
-
-``` r
-## prep data
-data(sibr)
-sire$cancer <- "rectal"
-sibr$cancer <- "breast"
-sr <- rbind(sire, sibr)
-
-sr$cancer <- factor(sr$cancer)
-sr <- sr[sr$dg_date < sr$ex_date, ]
-
-sr$status <- factor(sr$status, levels = 0:2, 
-                    labels = c("alive", "canD", "othD"))
-
-## create Lexis object
-library(Epi)
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sr,
-           exit.status = status)
-#> NOTE: entry.status has been set to "alive" for all.
-
-## population hazards file - see ?pophaz for general instructions
-data(popmort)
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-
-## simple usage - uses lex.Xst as status variable
-st <- survtab(FUT ~ cancer, data = x,
-              breaks = list(FUT = seq(0, 5, 1/12)),
-              surv.type = "surv.rel", pophaz = pm)
-
-## more explicit usage
-st <- survtab(Surv(FUT, event = lex.Xst) ~ cancer, data = x,
-              breaks = list(FUT = seq(0, 5, 1/12)),
-              surv.type = "surv.rel", pophaz = pm)
-
-
-## adjusting
-x$agegr <- cut(x$dg_age, c(0,55,65,75,Inf))
-w <- as.numeric(table(x$agegr))
-st <- survtab(Surv(FUT, event = lex.Xst) ~ cancer + adjust(agegr), 
-              data = x,
-              breaks = list(FUT = seq(0, 5, 1/12)),
-              surv.type = "surv.rel", 
-              pophaz = pm, weights = w)
-```
-
-## Rates
-
-The new `rate` function enables easy calculation of e.g. standardized
-incidence rates:
-
-``` r
-## dummy data
-
-a <- merge(0:1, 1:18)
-names(a) <- c("sex", "agegroup")
-set.seed(1)
-a$obs <- rbinom(nrow(a), 100, 0.5)
-set.seed(1)
-a$pyrs <- rbinom(nrow(a), 1e4, 0.75)
-
-## so called "world" standard rates (weighted to hypothetical world pop in 2000)
-r <- rate(data = a, obs = obs, pyrs = pyrs, print = sex, 
-          adjust = agegroup, weights = 'world_2000_18of5')
-#> Warning in pyrJjCscXlsrH * pyrJjCscXlsrH: NAs produced by integer overflow
-
-#> Warning in pyrJjCscXlsrH * pyrJjCscXlsrH: NAs produced by integer overflow
-```
-
-| sex | obs |   pyrs |  rate.adj | SE.rate.adj | rate.adj.lo | rate.adj.hi |      rate | SE.rate |   rate.lo |   rate.hi |
-|---:|---:|-----:|-------:|--------:|--------:|--------:|-------:|-----:|-------:|-------:|
-|   0 | 933 | 134986 | 0.0069947 |   0.0002541 |   0.0065140 |   0.0075108 | 0.0069118 |      NA | 0.0064822 | 0.0073699 |
-|   1 | 875 | 134849 | 0.0064453 |   0.0002429 |   0.0059865 |   0.0069394 | 0.0064887 |      NA | 0.0060727 | 0.0069332 |
+# Changes in 0.4.10
+
+-   No changes for user — “Suggests” packages now used conditionally in
+    vignettes/tests/examples as per Writing R Extensions.
+
+# Changes in 0.4.9
+
+-   `popEpi::aggre` bugfix: `aggre` now correctly infers which
+    stratifying variables use Lexis time scales when output is cartesian
+    and argument `by` is of type character.
+
+# Changes in 0.4.8
+
+-   small internal fixes due to upcoming new package survival release
+-   popEpi now includes a wrapper for survival::Surv, so you don’t need
+    to do library(“survival”) when using Surv() in formulae (e.g. in
+    survtab)
+
+# Changes in 0.4.7
+
+-   implemented small internal changes due to upcoming R 3.6.0
+
+# Changes in 0.4.6
+
+-   implemented small internal changes due to upcoming data.table
+    version
+
+# Changes in 0.4.5
+
+-   fixed errors arising from new data.table version
+
+# Changes in 0.4.4
+
+-   splitLexisDT/splitMulti bug fix: splitting along multiple time
+    scales *sometimes* produced duplicate transitions (e.g. alive -\>
+    dead in the last two rows). see
+    <https://github.com/FinnishCancerRegistry/popEpi/issues/138> for
+    details.
+-   splitLexisDT/splitMulti now retain time.since attribute; this
+    attribute plays a role in cutLexis
+-   known issue: splitLexisDT/splitMulti not guaranteed to work
+    identically to splitLexis from Epi when there are NA values in the
+    time scale one is splitting along.
+
+# Changes in 0.4.3
+
+-   survtab adjusting was broken with older versions of data.table
+    (tested 1.9.6); therefore popEpi now requires the newest version of
+    data.table!
+
+# Changes in 0.4.2
+
+-   **`survtab()` bug fix: standard errors were mis-specified for
+    adjusted curves, e.g. age-adjusted Ederer II estimates. This
+    resulted in too wide confidence intervals! SEE HERE FOR EXAMPLE:
+    [#135](https://github.com/FinnishCancerRegistry/popEpi/issues/135)**.
+    The standard errors and confidence intervals of non-adjusted curves
+    have always been correct.
+-   `survtab()` bug fix: confidence level was always 95 % regardless of
+    `conf.level`
+    [#134](https://github.com/FinnishCancerRegistry/popEpi/issues/134)
+
+# Changes in 0.4.1
+
+-   `lexpand()` bug fixed (#120): observations were dropped if their
+    entry by age was smaller than the smallest age value, though entry
+    at exit is correct and used now.
+-   `sir()` rewrite (#118, #122). New more consistent output, updates on
+    plotting and minor changes in arguments. Introduce very simple
+    `coef()` and `confint()` methods for sir class.
+-   new functions in sir family: `sir_ag()`, `sir_lex()` and `sir_exp()`
+    for extracting SMRs from `aggre` and `Lexis` objects.
+-   fixed issue in internal test brought by pkg survival version 2.39.5;
+    no changes in functions were needed (#125)
+-   robustified `aggre()`; there were issues with Epi pkg dev version
+    which are definitely avoided (#119)
+
+# Changes in 0.4.0
+
+-   removed previously deprecated shift.var (#35)
+-   popEpi no longer depends on package data.table but imports it - this
+    means the user will have to do library(data.table) separately to
+    make data.table’s functions become usable. Formerly popEpi
+    effectively did library(data.table) when you did library(popEpi).
+-   summary.survtab: args t and q behaviour changed
+-   survtab: internal weights now based on counts of subjects in
+    follow-up at the start of follow-up (used to be sum of counts/pyrs
+    over all of follow-up)
+-   new functions: `rate_ratio()`, `sir_ratio()`
+-   small internal changes in preparation for data.table 1.9.8
+
+# Changes in 0.3.1
+
+This is a hotfix. survtab() was causing warnings in certain situations,
+which this update fixes. Also fixed plotting survtab objects so that
+multiple strata are plotted correctly when one or more curves end before
+the longest one as well other small fixes: See Github issues #89, #90,
+#91, and #92.
+
+# Changes in 0.3.0
+
+## Adjusting
+
+Direct adjusting (computing weighted averages of estimates) has been
+generalized. Functions such as `survtab` and `survmean` allow for using
+`adjust()` mini function within formulas, or a separate `adjust`
+argument. Weights are passed separately. See the examples in the next
+chapter. See also `?direct_adjusting`.
+
+## Estimating functions of survival time
+
+The `survtab` function computes observed, net/relative and
+cause-specific survivals as well as cumulative incidence functions for
+`Lexis` data. Any of the supported survival time functions can be easily
+adjusted by any number of categorical variables if needed.
+
+One can also use `survtab_ag` for aggregated data. This means the data
+does not have to be on the subject-level to compute survival time
+function estimates.
+
+``` r
+## prep data
+data(sibr)
+sire$cancer <- "rectal"
+sibr$cancer <- "breast"
+sr <- rbind(sire, sibr)
+
+sr$cancer <- factor(sr$cancer)
+sr <- sr[sr$dg_date < sr$ex_date, ]
+
+sr$status <- factor(sr$status, levels = 0:2, 
+                    labels = c("alive", "canD", "othD"))
+
+## create Lexis object
+library(Epi)
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sr,
+           exit.status = status)
+#> NOTE: entry.status has been set to "alive" for all.
+
+## population hazards file - see ?pophaz for general instructions
+data(popmort)
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+
+## simple usage - uses lex.Xst as status variable
+st <- survtab(FUT ~ cancer, data = x,
+              breaks = list(FUT = seq(0, 5, 1/12)),
+              surv.type = "surv.rel", pophaz = pm)
+
+## more explicit usage
+st <- survtab(Surv(FUT, event = lex.Xst) ~ cancer, data = x,
+              breaks = list(FUT = seq(0, 5, 1/12)),
+              surv.type = "surv.rel", pophaz = pm)
+
+
+## adjusting
+x$agegr <- cut(x$dg_age, c(0,55,65,75,Inf))
+w <- as.numeric(table(x$agegr))
+st <- survtab(Surv(FUT, event = lex.Xst) ~ cancer + adjust(agegr), 
+              data = x,
+              breaks = list(FUT = seq(0, 5, 1/12)),
+              surv.type = "surv.rel", 
+              pophaz = pm, weights = w)
+```
+
+## Rates
+
+The new `rate` function enables easy calculation of e.g. standardized
+incidence rates:
+
+``` r
+## dummy data
+
+a <- merge(0:1, 1:18)
+names(a) <- c("sex", "agegroup")
+set.seed(1)
+a$obs <- rbinom(nrow(a), 100, 0.5)
+set.seed(1)
+a$pyrs <- rbinom(nrow(a), 1e4, 0.75)
+
+## so called "world" standard rates (weighted to hypothetical world pop in 2000)
+r <- rate(data = a, obs = obs, pyrs = pyrs, print = sex, 
+          adjust = agegroup, weights = 'world_2000_18of5')
+#> Warning in pyrJjCscXlsrH * pyrJjCscXlsrH: NAs produced by integer overflow
+
+#> Warning in pyrJjCscXlsrH * pyrJjCscXlsrH: NAs produced by integer overflow
+```
+
+| sex | obs |   pyrs |  rate.adj | SE.rate.adj | rate.adj.lo | rate.adj.hi |      rate | SE.rate |   rate.lo |   rate.hi |
+|---:|---:|-----:|-------:|--------:|--------:|--------:|-------:|-----:|-------:|-------:|
+|   0 | 933 | 134986 | 0.0069947 |   0.0002541 |   0.0065140 |   0.0075108 | 0.0069118 |      NA | 0.0064822 | 0.0073699 |
+|   1 | 875 | 134849 | 0.0064453 |   0.0002429 |   0.0059865 |   0.0069394 | 0.0064887 |      NA | 0.0060727 | 0.0069332 |
diff --git a/R/S3_definitions.R b/R/S3_definitions.R
index 54d67a5..46a301c 100644
--- a/R/S3_definitions.R
+++ b/R/S3_definitions.R
@@ -1,1514 +1,1514 @@
-
-
-#' @export
-print.sir <- function(x, subset = NULL, ...) {
-  
-  at <- attributes(x)$sir.meta
-  PF <- parent.frame(1L)
-  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
-  x <- x[subset, ]
-  setDT(x)
-  
-  t1 <- paste0("SIR (adjusted by ", paste(at$adjust, collapse = ', '),')',
-               ' with ', at$conf.level*100, '% ', 'confidence intervals (', at$conf.type,')')
-  
-  
-  # cat
-  t3 <- paste0(' Total sir: ', round(at$pooled.sir$sir,2),' (', 
-               round(at$pooled.sir$sir.lo,2),'-', round(at$pooled.sir$sir.hi, 2),')\n',
-               ' Total observed: ', at$pooled.sir$observed, '\n',
-               ' Total expected: ', round(at$pooled.sir$expected,2), '\n',
-               ' Total person-years: ', round(at$pooled.sir$pyrs))
-
-  rv <- intersect(names(x), c('sir','sir.lo','sir.hi','observed','expected','pyrs'))
-  if (length(rv)) {
-    x[, (rv) := lapply(.SD, round, digits = 2L),  .SDcols = rv]
-  }
-  
-  rv <- intersect(names(x), c('p_value'))
-  if (length(rv)) {
-    x[, (rv) := lapply(.SD, round, digits = 4),  .SDcols = rv]
-  }
-
-  
-  if(is.null(at$lrt.test)) {
-    d <- paste("Could not test", at$lrt.test.type)
-  } else {
-    if(at$lrt.test.type == 'homogeneity') {
-      d <- paste("Test for homogeneity: p", p.round( c(at$lrt.test)))
-    }
-    if(at$lrt.test.type == 'trend') {
-      d <- paste("Test for trend: p", p.round( c(at$lrt.test)))
-    }
-  }
-  #b <- round(c(ta$total$sir, ta$total$sir.lo, ta$total$sir.hi), 2)
-  # cat('\n',"Total observed", ta$total$observed, '\n',
-  #     "Total expected:", ta$total$expected, '\n',
-  #     "SIR:", paste0(b[1], ' (',b[2], '-',b[3],')'), '\n',
-  #     "Person-years:", ta$total$pyrs, '\n',
-  #     fill=TRUE)
-  
-  cat(t1, '\n')
-  if(x[,.N] > 1) {
-    cat(d, '\n')
-  }
-  cat(fill=TRUE)
-  cat(t3, '\n', fill=TRUE)  
-  
-  print(data.table(x), ...)
-  return(invisible())
-}
-
-#' @export 
-`[.sir` <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "sir.meta", attr(x, "sir.meta"))
-  }
-  y
-}
-
-
-#' @import grDevices
-#' @export
-print.sirspline <- function(x, ...) {
-  if ( x$spline.dependent ) {
-    if( any( !is.na(x$p.values))) {
-      cat( 'global p-value:', p.round(x$p.values[1]),'\n' )
-      cat( 'level p-value:', p.round(x$p.values[2]) , fill= TRUE)      
-    } else {
-      cat( 'No models compared.', fill= TRUE)
-    }
-    cat('---', '\n')
-    cat('Colour codes:', '\n', fill=TRUE)
-  } else {
-    
-    for(i in 1:length(x$p.values)) {
-      cat( x$spline[i] ,': p ', p.round( x$p.values[[i]] ), '\n', sep = '')
-    }
-    cat(fill=TRUE)
-    
-  }
-  # Print colour codes:
-  cols <- unique(x$spline.est.A[,1])
-  col.length <- length(cols)
-  print( data.frame(levels = cols, colour = palette()[1:col.length]), include.rownames = FALSE)
-  
-  # Print p-values
-  return(invisible())
-}
-    
-
-#' Plot method for sir-object
-#' 
-#' Plot SIR estimates with error bars
-#' 
-#' @seealso \code{\link{sir}},  \code{\link{sirspline}}
-#' 
-#' @import graphics
-#' 
-#' @author Matti Rantanen
-#' 
-#' @param x an object returned by function \code{sir}
-#' @param conf.int default TRUE draws confidence intervals
-#' @param xlab overwrites default x-axis label
-#' @param ylab overwrites default y-axis label
-#' @param xlim x-axis minimum and maximum values
-#' @param main optional plot title
-#' @param abline logical; draws a grey line in SIR = 1
-#' @param log logical; SIR is not in log scale by default
-#' @param eps error bar vertical bar height (works only in 'model' or 'univariate')
-#' @param left.margin adjust left marginal of the plot to fit long variable names
-#' @param ... arguments passed on to plot(), segment and lines()
-#' 
-#' 
-#' @details Plot SIR estimates and confidence intervals 
-#' \itemize{
-#'  \item univariate - plots SIR with univariate confidence intervals
-#'  \item model - plots SIR with Poisson modelled confidence intervals
-#' }
-#' 
-#' \strong{Customize}
-#' Normal plot parameters can be passed to \code{plot}. These can be a vector when plotting error bars:
-#' \itemize{
-#'  \item \code{pch} - point type
-#'  \item \code{lty} - line type
-#'  \item \code{col} - line/point colour 
-#'  \item \code{lwd} - point/line size
-#' }
-#'
-#' \strong{Tips for plotting splines}
-#' It's possible to use \code{plot} to first draw the 
-#' confidence intervals using specific line type or colour and then plotting 
-#' again the estimate using \code{lines(... , conf.int = FALSE)} with different 
-#' settings. This works only when \code{plot.type} is 'splines'.
-#' 
-#' 
-#' @examples 
-#' \donttest{
-#' # Plot SIR estimates
-#'# plot(sir.by.gender, col = c(4,2), log=FALSE, eps=0.2, lty=1, lwd=2, pch=19,  
-#'#      main = 'SIR by gender', abline=TRUE)
-#' }
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-#' @export
-
-plot.sir <- function(x, conf.int = TRUE, ylab, xlab, xlim, main, 
-                     eps=0.2, abline = TRUE, log = FALSE, left.margin, ...) {
-  
-  a <- data.table(x)
-  at <- attributes(x)$sir.meta
-  level_names <- at$print
-  
-  if(is.null(level_names)) {
-    levels <- 'Crude'
-    level_names <- levels
-  }
-  else {
-    q <- paste0('paste(', paste(level_names, collapse=', '),', sep = ":")' )
-    q <- parse(text = q)
-    levels <- a[, eval(q)]
-  }
-  
-  # predefined parameters
-  if( missing(main) ){
-    main <- NA
-  }
-  if( missing(xlab) ){
-    xlab <- 'SIR'
-  }
-  if( missing(ylab) ){
-    ylab <- NA
-  }
-  if( missing(xlim) ) {
-    xlimit <- c(min(a$sir.lo[a$sir.lo <Inf]), max(a$sir.hi[a$sir.hi <Inf]))
-    xlimit <- xlimit + diff(xlimit)*c(-.3,.3)
-  } 
-  else {
-    xlimit <- xlim
-  }
-  
-  # par options
-  old_mar <- par("mar")
-  on.exit(par(mar = old_mar))
-  new.margin <- old_mar
-  if(missing(left.margin)) {
-    new.margin[2] <- 4.1 + sqrt( max(nchar(as.character(level_names))) )*2
-  } 
-  else {
-    new.margin[2] <- left.margin
-  }
-  par(mar = new.margin)
-  
-  # plot frame, estimates and CI (optional abline)
-  logarithm <- ''
-  if(log){
-    logarithm <- 'x'
-    if(xlimit[1]==0) xlimit[1] <- xlimit[1] + 0.01
-  }
-  y.axis.levels <- 1:length(levels)
-  plot(c(xlimit), c(min(y.axis.levels)-0.5, max(y.axis.levels)+0.5), 
-       type='n', yaxt = 'n', xlab=xlab, ylab=ylab, log=logarithm, main = main, ...)
-  axis(side = 2, at = y.axis.levels, labels = levels, las=1)
-  
-  if(abline) {
-    abline(v=1, col = 'darkgray')
-  }
-  
-  points(a$sir, factor(y.axis.levels, labels=levels), ...) 
-  if(conf.int) {
-    segments(a$sir.lo, y.axis.levels , a$sir.hi, y.axis.levels, ...)
-    segments(a$sir.lo, y.axis.levels - eps, a$sir.lo, y.axis.levels +eps, ... )
-    segments(a$sir.hi, y.axis.levels - eps, a$sir.hi, y.axis.levels +eps, ... )
-  }
-  
-  return(invisible(NULL))
-}
-
-#' @title \code{plot} method for sirspline-object
-#' 
-#' @description Plot SIR splines using R base graphics.
-#' 
-#' 
-#' @import graphics
-#' 
-#' @author Matti Rantanen
-#' 
-#' @param x an object returned by function sirspline
-#' @param conf.int logical; default TRUE draws also the 95 confidence intervals
-#' @param xlab overwrites default x-axis label; can be a vector if multiple splines fitted
-#' @param ylab overwrites default y-axis label; can be a vector if multiple splines fitted
-#' @param log logical; default FALSE. Should the y-axis be in log scale
-#' @param abline logical; draws a reference line where SIR = 1
-#' @param type select \code{type = 'n'} to plot only figure frames 
-#' @param ... arguments passed on to plot()
-#' 
-#' @details
-#' In \code{plot.sirspline} almost every graphical parameter are user
-#' adjustable, such as \code{ylim}, \code{xlim}.
-#' \code{plot.sirsplines} calls \code{lines.splines} to add lines.
-#' 
-#' The plot axis without lines can be plotted using option \code{type = 'n'}. 
-#' On top of the frame it's then possible to add a \code{grid}, 
-#' \code{abline} or text before plotting the lines (see: \code{sirspline}).
-#' @export
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-#' @family sir functions
-plot.sirspline <- function(x, conf.int=TRUE, abline = TRUE, log = FALSE, type, ylab, xlab,  ...) {
-
-  #print(list(...))
-  
-  ## premilinary checks  
-  if (is.null(x$spline.seq.A)) stop('No splines found.')
-  
-  ## prepare dimension and par
-  plotdim <- as.numeric(c( !is.null( x$spline.seq.A ),
-                           !is.null( x$spline.seq.B ),
-                           !is.null( x$spline.seq.C ) ))
-  
-  if(sum(plotdim) > 1) {
-    old_mfrow <- par("mfrow")
-    on.exit(par(mfrow = old_mfrow))
-    new_mfrow <- c(1,sum(plotdim))
-    par(mfrow = new_mfrow)
-    type <- 'l'
-  }
-  
-  ## set labels
-  if ( missing(xlab) ) {
-    xlab <- x$spline
-  }
-  
-  if ( missing(ylab) ) {
-    ylab <- rep('SIR',sum(plotdim))
-    if(log){
-      ylab <- rep('log(SIR)', sum(plotdim))
-    }
-    if(x$spline.dependent & sum(plotdim) > 1) {
-      ylab <- c(ylab[1], paste(ylab[2:sum(plotdim)], 'ratio'))
-    }
-  }
-  else{
-    if( length(ylab) < sum(plotdim))
-      ylab <- rep(ylab, sum(plotdim))
-    if(length(ylab) > sum(plotdim)) {
-      warning('set ylabs in a vector length of num of plots (',sum(plotdim),')')
-    }
-  }
-  
-  ## set scale
-  if(!is.logical(log)) stop('log should be a logical value.')
-  log.bin <- ifelse(log, 'y', '')
-  
-  ## remove infinite values
-  #rm_inf <- function(est){
-  #  x[[est]][ is.finite(x[[est]][[2]]) & is.finite(x[[est]][[3]]) & is.finite(x[[est]][[4]]), ]
-  #}
-  
-  spl <- c('spline.seq.A', 'spline.seq.B', 'spline.seq.C')[1:sum(plotdim)]
-  est <- gsub("seq", "est", spl)
-  
-  for (i in 1:sum(plotdim)) {  # age, per, fot, 
-    # empty plot
-    max_x <- range(x[[spl[i]]])
-    max_y <- range( x[[est[i]]][, 2:4] )
-    plot(max_x, max_y, type = 'n', ylab = ylab[i], xlab = xlab[i], log = log.bin, ...) 
-    if(abline) abline(h = 1)
-    
-    # plot lines
-    if (missing(type) || type != 'n') {
-      lines.sirspline(x, conf.int = conf.int, select.spline = i, ...)
-    }
-  }
-  return(invisible(NULL))
-}
-
-
-
-#' @title lines method for sirspline-object
-#' @description Plot SIR spline lines with R base graphics
-#' 
-#' 
-#' @author Matti Rantanen
-#' 
-#' @param x an object returned by function sirspline
-#' @param conf.int logical; default TRUE draws also the 95 confidence intervals
-#' @param print.levels name(s) to be plotted. Default plots all levels.
-#' @param select.spline select which spline variable (a number or a name) is plotted.
-#' @param ... arguments passed on to lines()
-#' 
-#' @details  In \code{lines.sirspline} most of graphical parameters is user 
-#' adjustable.
-#' Desired spline variable can be selected with \code{select.spline} and only one
-#' can be plotted at a time. The spline variable can include 
-#' several levels, e.g. gender (these are the levels of \code{print}
-#' from \code{sirspline}). All levels are printed by default, but a
-#' specific level can be selected using argument
-#' \code{print.levels}. Printing the levels separately enables  e.g. to
-#' give different colours for each level.
-#' 
-#' @family sir functions
-#' 
-#' @import graphics
-#' @export
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-
-lines.sirspline <- function(x, conf.int = TRUE, print.levels = NA, select.spline, ... ){
-  ## input: sirspline object, with only one spline var (spline.est.A)
-  ## input: print levels can be > 1.
-  
-  ## subset splines
-  if( length(x$spline) > 1 ) {
-    if ( missing(select.spline) ) {
-      stop(paste('select what spline to plot in select.spline:', paste(x$spline, collapse = ', ')))
-    }
-    else {
-      if(is.numeric(select.spline)) {
-        k <- select.spline
-      }
-      else {
-        k <- which(x$spline == select.spline)
-      }
-      if(length(k) == 0 | length(x$spline) < k) stop('select.spline name/number is incorrect')
-    }
-  } 
-  else {
-    k <- 1
-  }
-  
-  spl <- c('spline.seq.A', 'spline.seq.B', 'spline.seq.C')[k]
-  est <- gsub("seq", "est", spl)
-  
-  ## remove infinite values
-  # x[[h]] <- rm_inf(est=h)
-  
-  # get print levels
-  if(missing(print.levels)) {
-    print.levels <- NA
-  }
-  pl <- unique(x$spline.est.A[,1])
-  if(any( is.null(print.levels), is.na(print.levels))) {
-    print.levels <- pl
-  }
-  pl <- pl[ pl %in% print.levels]
-  
-  ## get conf.int
-  if( !is.logical(conf.int) ) stop('conf.int is not logical')
-  n <- c(2,4)[c(!conf.int, conf.int)]
-  
-  
-  ## draw lines
-  for( l in pl ){
-    # loop through print.levels
-    index <- which(x$spline.est.A$i == l)
-    
-    for(m in 2:n) {
-      # loop through estiamte and confidence intervals
-      lines(x = x[[spl]], y = x[[est]][index, m], ...)
-    }
-  }
-  return(invisible(NULL))
-}
-
-#' @title Print an rate object
-#' @author Matti Rantanen
-#' @description Print method function for \code{rate} objects; see
-#' \code{\link{rate}}.
-#' @param x an \code{rate} object
-#' @param subset a logical condition to subset results table by
-#' before printing; use this to limit to a certain stratum. E.g.
-#' \code{subset = sex == "female"}
-#' @param ... arguments for data.tables print method, e.g. row.names = FALSE suppresses row numbers.
-#' @export
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-print.rate <- function(x, subset = NULL, ...) {
-  
-  ra <- attributes(x)$rate.meta
-  PF <- parent.frame(1L)
-  TF <- environment()
-  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
-  x <- x[subset, ]
-  
-  # pre texts:
-  cat('\n')
-  if(!is.null(ra$adjust)){
-    if(is.character(ra$weights)) {
-      a <- paste(ra$weights, collapse = ',')
-    }
-    if(all(is.numeric(ra$weights))) {
-      a <- length(ra$weights)
-    }
-    if(is.list(ra$weights)) {
-      a <- sapply(ra$weights, length)
-    }
-    
-    b <- paste(ra$adjust,a, collapse = ', ', sep = '; ')
-    cat('Adjusted rates (', b,') ', sep = '')
-  }
-  else{
-    cat('Crude rates ')
-  }
-  cat('and', '95%', 'confidence intervals:', fill=TRUE)
-  cat('\n')
-  # table itself
-
-  
-  setDT(x)
-  print(x, ...)
-  return(invisible(NULL))
-}
-
-
-#' @title plot method for rate object
-#' @description Plot rate estimates with confidence intervals lines using R base graphics
-#' @author Matti Rantanen
-#' 
-#' @param x a rate object (see \code{\link{rate}})
-#' @param conf.int logical; default TRUE draws the confidence intervals
-#' @param eps is the height of the ending of the error bars
-#' @param left.margin set a custom left margin for long variable names. Function
-#' tries to do it by default.
-#' @param xlim change the x-axis location
-#' @param ... arguments passed on to graphical functions points and segment 
-#' (e.g. \code{col}, \code{lwd}, \code{pch} and \code{cex})
-#' 
-#' @details This is limited explanatory tool but most graphical 
-#' parameters are user adjustable. 
-#' 
-#' @import graphics
-#' @export
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-plot.rate <- function(x, conf.int = TRUE, eps = 0.2, left.margin, xlim, ...) {
-  
-  ra <- attributes(x)$rate.meta
-  varcol <- ra$print
-  
-  if(is.null(varcol)) {
-    lvl.name <- 'Crude'
-  }
-  else {
-    pp <- paste0('paste(', paste(varcol, collapse=','),',sep = ":")')
-    q <- parse(text=pp)
-    lvl.name <- x[,eval(q)]
-  }
-  lvls <- 1:length(lvl.name)
-  
-  # WHICH RATE:
-  if('rate.adj' %in% names(x)) {
-    r <- x$rate.adj
-    hi <- x$rate.adj.hi
-    lo <- x$rate.adj.lo
-  }
-  else {
-    r <- x$rate
-    hi <- x$rate.hi
-    lo <- x$rate.lo
-  }
-  # X-AXIS LIMITs
-  if(missing(xlim)) {
-    t <- range(na.omit(c(lo , r, hi)))
-    t0 <- (t[2]-t[1])/4
-    xlimit <- c(pmax(t[1]-t0, 0), t[2] + t0)
-  } 
-  else {
-    xlimit <- xlim
-  }
-  
-  # MARGINS  
-  old_mar <- par("mar")
-  on.exit(par(mar = old_mar))
-  if(missing(left.margin)) {
-   new.margin <- par("mar")
-   new.margin[2] <- 4.1 + sqrt( max(nchar(as.character(lvl.name))) )*2
-  }
-  else {
-   new.margin[2] <- left.margin
-  }
-  par(mar = new.margin)
-  
-  plot(c(xlimit), c(min(lvls)-0.5, max(lvls)+0.5), type='n', yaxt = 'n', ylab = '', xlab='')
-  axis(side = 2, at = lvls, labels = lvl.name, las = 1)
-  points(r, lvls, ...)
-  
-  if(conf.int) {
-    segments(lo, lvls, hi, lvls, ...)
-    segments(lo, lvls - eps, lo, lvls + eps, ...)
-    segments(hi, lvls - eps, hi, lvls + eps, ...)
-  }
-  return(invisible(NULL))
-}
-
-
-#' @export 
-print.yrs <- function(x, ...) {
-  print(as.numeric(x))
-}
-
-
-#' @export 
-`[.yrs` <- function(x, ...) {
-  yl <- attr(x, "year.length")
-  structure(NextMethod(), year.length = yl, class = c("yrs", "numeric"))
-}
-
-
-#' @export 
-`[.aggre` <- function(x, ...) {
-  xa <- attributes(x)
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", xa$class)
-    setattr(y, "aggre.meta", xa$aggre.meta)
-    setattr(y, "breaks", xa$breaks)
-  }
-  y
-}
-
-#' @export 
-subset.aggre <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "aggre.meta", attr(x, "aggre.meta"))
-    setattr(y, "breaks", attr(x, "breaks"))
-  }
-  y
-}
-
-preface_survtab.print <- function(x) {
-  surv.int <- NULL ## APPEASE R CMD CHECK
-  at <- attributes(x)$survtab.meta
-  arg <- at$arguments
-  
-  cat("\n")
-  cat("Call: \n", oneWhitespace(deparse(at$call)), "\n")
-  cat("\n")
-  cat("Type arguments: \n surv.type:", as.character(arg$surv.type), 
-      "--- surv.method:", as.character(arg$surv.method))
-  if (as.character(arg$surv.type) == "surv.rel")
-    cat(" --- relsurv.method:", as.character(arg$relsurv.method))
-  cat("\n \n")
-  cat("Confidence interval arguments: \n level:", 
-      as.character(arg$conf.level*100), "%")
-  cat(" --- transformation:",
-      as.character(arg$conf.type))
-  cat("\n \n")
-  cat("Totals:") 
-  totCat <- paste0("\n person-time:", round(sum(x$pyrs)))
-  if (arg$surv.method == "lifetable") {
-    totCat <- paste0("\n at-risk at T=0: ", round(sum(x[surv.int == 1L]$n)))
-  }
-    
-  cat(totCat)
-  cat(" --- events:", sum(x$d))
-  cat("\n \n")
-  if (length(at$print.vars) > 0L) {
-    cat("Stratified by:", paste0("'", at$print.vars, "'", collapse = ", "))
-    if (length(at$adjust.vars) > 0L) cat(" --- ")
-  }
-  if (length(at$adjust.vars) > 0L) {
-    cat("Adjusted by:", paste0("'", at$adjust.vars, "'", collapse = ", "))
-  }
-  cat("\n")
-  invisible()
-}
-
-
-#' @title Print an \code{aggre} Object
-#' @author Joonas Miettinen
-#' @description Print method function for \code{aggre} objects; see
-#' \code{\link{as.aggre}} and \code{\link{aggre}}.
-#' @param x an \code{aggre} object
-#' @param subset a logical condition to subset results table by
-#' before printing; use this to limit to a certain stratum. E.g.
-#' \code{subset = sex == "male"}
-#' @param ... arguments passed to \code{print.data.table}; try e.g.
-#' \code{top = 2} for numbers of rows in head and tail printed 
-#' if the table is large, 
-#' \code{nrow = 100} for number of rows to print, etc.
-#' @export
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-print.aggre <- function(x, subset = NULL, ...) {
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  sa <- attributes(x)$aggre.meta
-  
-  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
-  x <- x[subset, ]
-  setDT(x)
-  
-  print(x, ...)
-  return(invisible(NULL))
-}
-
-#' @title Summarize an \code{aggre} Object
-#' @author Joonas Miettinen
-#' @description \code{summary} method function for \code{aggre} objects; see
-#' \code{\link{as.aggre}} and \code{\link{aggre}}.
-#' @param object an \code{aggre} object
-#' @param by list of columns to summarize by - e.g. \code{list(V1, V2)}
-#' where \code{V1} and \code{V2} are columns in the data.
-#' @param subset a logical condition to subset results table by
-#' before summarizing; use this to limit to a certain stratum. E.g.
-#' \code{subset = sex == "male"}
-#' @param ... unused
-#' @export
-#' @family aggregation functions
-#' @return
-#' Returns a `data.table` --- a further aggregated version of `object`.
-summary.aggre <- function(object, by = NULL, subset = NULL, ...) {
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  x <- object
-  sa <- attributes(x)$aggre.meta
-  
-  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
-  x <- x[subset, ]
-  setDT(x)
-  
-  bys <- substitute(by)
-  bye <- evalPopArg(x, bys, enclos = environment(), types = c("list", "NULL"))
-  
-  vals <- sa$values
-  vals <- intersect(names(x), vals)
-  if (!length(vals)) {
-    cat("No originally created value columns appear to be left in data.")
-  }
-  r <- x[, lapply(.SD, sum), by = eval(bye), .SDcols = vals]
-  r
-}
-
-#' @title Print a survtab Object
-#' @author Joonas Miettinen
-#' @description Print method function for \code{survtab} objects; see
-#' \code{\link{survtab_ag}}.
-#' @param x a \code{survtab} object
-#' @param subset a logical condition to subset results table by
-#' before printing; use this to limit to a certain stratum. E.g.
-#' \code{subset = sex == "male"}
-#' @param ... arguments passed to \code{print.data.table}; try e.g.
-#' \code{top = 2} for numbers of rows in head and tail printed 
-#' if the table is large, 
-#' \code{nrow = 100} for number of rows to print, etc.
-#' @export
-#' @family survtab functions
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-print.survtab <- function(x, subset = NULL, ...) {
-  
-  Tstart <- Tstop <- NULL ## APPEASE R CMD CHECK
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  sa <- attributes(x)$survtab.meta
-  
-  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
-  x <- x[subset, ]
-  
-  preface_survtab.print(x)
-  
-  setDT(x)
-  
-  if (nrow(x) == 0L) {
-    print(x)
-    return(invisible())
-  }
-  
-  pv <- as.character(sa$print.vars)
-  if (length(pv) == 0L) pv <- NULL
-  
-  magicMedian <- function(x) {
-    if (length(x) %% 2L == 0L) median(x[-1L], na.rm = TRUE) else
-      median(x, na.rm = TRUE)
-  }
-  
-  ## to avoid e.g. 'factor(V1, 1:2)' going bonkers
-  pv_orig <- pv
-  if (length(pv) > 0L) {
-    pv <- makeTempVarName(x, pre = paste0("print_", 1:length(pv)))
-    setnames(x, pv_orig, pv)
-  }
-  
-  medmax <- x[, list(Tstop = c(magicMedian(c(min(Tstart),Tstop)), max(Tstop))), keyby = eval(pv)]
-  
-  setkeyv(medmax, c(pv, "Tstop"))
-  setkeyv(x, c(pv, "Tstop"))
-  x <- x[medmax]
-  
-  rv <- intersect(names(x), c(sa$est.vars, sa$CI.vars, sa$misc.vars))
-  if (length(rv)) {
-    x[, (rv) := lapply(.SD, round, digits = 4L),  .SDcols = rv]
-  }
-  
-  sv <- intersect(names(x), sa$SE.vars)
-  if (length(sv > 0L)) {
-    x[, c(sv) := lapply(.SD, signif, digits = 4L), .SDcols = sv]
-  }
-  
-  
-  setcolsnull(x, keep = c(pv, "Tstop", sa$surv.vars), colorder = TRUE)
-  if (length(pv)) setnames(x, pv, pv_orig)
-  print(data.table(x), ...)
-  return(invisible(NULL))
-}
-
-#' @title Summarize a survtab Object
-#' @author Joonas Miettinen
-#' @description Summary method function for \code{survtab} objects; see
-#' \code{\link{survtab_ag}}. Returns estimates at given time points
-#' or all time points if \code{t} and \code{q} are both \code{NULL}.
-#' @param object a \code{survtab} object
-#' @param t a vector of times at which time points (actually intervals that
-#' contain t) to print summary table of survival function estimates by strata;
-#' values not existing in any interval cause rows containing only \code{NAs} to
-#' be returned. 
-#' @param q a named \code{list} of quantiles to include in returned data set,
-#' where names must match to estimates in \code{object};
-#' returns intervals where the quantiles are reached first;
-#' e.g. \code{list(surv.obs = 0.5)} finds the interval where \code{surv.obs}
-#' is 0.45 and 0.55 at the beginning and end of the interval, respectively;
-#' returns rows with \code{NA} values for quantiles not reached in estimates
-#' (e.g. if \code{q = list(surv.obs = 0.5)} but lowest estimate is 0.6);
-#' see Examples.
-#' @param subset a logical condition to subset results table by
-#' before printing; use this to limit to a certain stratum. E.g.
-#' \code{subset = sex == "male"}
-#' @param ... unused; required for congruence with other \code{summary} methods
-#' 
-#' @details
-#' Note that this function returns the intervals and NOT the time points
-#' corresponding to quantiles / estimates corresponding to time points.
-#' If you want precise estimates at time points that are not interval breaks,
-#' add the time points as breaks and re-estimate the survival time function.
-#' In interval-based estimation, the estimates denote e.g. probability of 
-#' dying \emph{during} the interval, so time points within the intervals
-#' are not usually considered at all. See e.g. Seppa, Dyba, and Hakulinen 
-#' (2015).
-#' 
-#' @references
-#' Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
-#' Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
-#' \doi{10.1016/B978-0-12-801238-3.02745-8}
-#' 
-#' @examples 
-#' 
-#' library(Epi)
-#' 
-#' ## NOTE: recommended to use factor status variable
-#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-#'            exit = list(CAL = get.yrs(ex_date)), 
-#'            data = sire[sire$dg_date < sire$ex_date, ],
-#'            exit.status = factor(status, levels = 0:2, 
-#'            labels = c("alive", "canD", "othD")), 
-#'            merge = TRUE)
-#' ## pretend some are male
-#' set.seed(1L)
-#' x$sex <- rbinom(nrow(x), 1, 0.5)
-#' ## observed survival
-#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-#'                   surv.type = "cif.obs",
-#'                   breaks = list(FUT = seq(0, 5, 1/12)))
-#' 
-#' ## estimates at full years of follow-up
-#' summary(st, t = 1:5)
-#' 
-#' ## interval estimate closest to 75th percentile, i.e.
-#' ## first interval where surv.obs < 0.75 at end
-#' ## (just switch 0.75 to 0.5 for median survival, etc.)
-#' summary(st, q = list(surv.obs = 0.75))
-#' ## multiple quantiles
-#' summary(st, q = list(surv.obs = c(0.75, 0.90), CIF_canD = 0.20))
-#' 
-#' ## if you want all estimates in a new data.frame, you can also simply do
-#' 
-#' x <- as.data.frame(st)
-#' @return
-#' A `data.table`: a slice from `object` based on `t`, `subset`, and `q`.
-#' @export
-#' @family survtab functions
-summary.survtab <- function(object, t = NULL, subset = NULL, q = NULL, ...) {
-  
-  PF <- parent.frame(1L)
-  at <- copy(attr(object, "survtab.meta"))
-  subr <- copy(at$surv.breaks)
-  
-  if (!is.null(t) && !is.null(q)) {
-    stop("Only supply either t or q.")
-  } 
-  
-  sb <- substitute(subset)
-  subset <- evalLogicalSubset(object, sb, enclos = PF)
-  x <- object[subset, ]
-  
-  ## to avoid e.g. 'factor(V1, 1:2)' going bonkers
-  pv_orig <- pv <- at$print.vars
-  if (length(pv) > 0L) {
-    pv <- makeTempVarName(x, pre = paste0("print_", 1:length(pv)))
-    setnames(x, pv_orig, pv)
-  }
-  
-  setDT(x)
-  
-  ## quantile detection --------------------------------------------------------
-  if (!is.null(q)) {
-    bn <- setdiff(names(q), at$est.vars)
-    if (length(bn) > 0L) {
-      stop("No survival time function estimates named ",
-           paste0("'", bn, "'", collapse = ", "), 
-           " found in supplied survtab object. Available ",
-           "survival time function estimates: ", 
-           paste0("'", at$est.vars, "'", collapse = ", "))
-    }
-    
-    lapply(q, function(x) {
-      if (min(x <= 0L) || max(x >= 1L)) {
-        stop("Quantiles must be expressed as numbers between 0 and 1, ",
-             "e.g. surv.obs = 0.5.")
-      }
-    })
-    
-    
-    m <- x[, .SD[1, ], keyby = eval(pv)][, c(pv, "Tstop"), with = FALSE]
-    setDF(m)
-    
-    rollVars <- makeTempVarName(x, pre = names(q))
-    x[, c(rollVars) := lapply(.SD, copy), .SDcols = names(q)]
-    
-    m <- lapply(seq_along(q), function(i) {
-      m <- merge(m, q[[i]])
-      setnames(m, "y", rollVars[i])
-      if (length(pv)) setorderv(m, pv)
-      m[, c(pv, rollVars[i]), drop = FALSE]
-    })
-    names(m) <- names(q)
-    
-    l <- vector("list", length(q))
-    names(l) <- names(q)
-    for (k in names(q)) {
-      
-      l[[k]] <- setDT(x[m[[k]], on = names(m[[k]]), roll = 1L])
-      
-    }
-    l <- rbindlist(l)
-    set(l, j = rollVars, value = NULL)
-    if (length(pv)) setkeyv(l, pv)
-    x <- l
-  }
-  
-  ## time point detection ------------------------------------------------------
-  
-  if (!is.null(t)) {
-    
-    tcutv <- makeTempVarName(x, pre = "cut_time_")
-    
-    set(x, j = tcutv, value = cut(x$Tstop, breaks = subr, right = TRUE, 
-                                  include.lowest = FALSE))
-    cutt <- cut(t, breaks = subr, right = TRUE, include.lowest = FALSE)
-    
-    l <- list(cutt)
-    names(l) <- tcutv
-    if (length(pv)) {
-      pvdt <- setDF(unique(x, by = pv))[, pv, drop = FALSE]
-      l <- setDT(merge(pvdt, as.data.frame(l)))
-      setkeyv(l, pv)
-    }
-    
-    x <- x[l, on = c(pv, tcutv)]
-    
-    set(x, j = tcutv, value = NULL)
-    if (length(pv)) setkeyv(x, pv)
-  }
-  
-  
-  ## final touches -------------------------------------------------------------
-  if (length(pv) > 0L) setnames(x, pv, pv_orig)
-  
-  if (!return_DT()) setDFpe(x)
-  
-  x
-}
-
-#' @export 
-`[.survtab` <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "survtab.meta", attr(x, "survtab.meta"))
-  }
-  y
-}
-
-#' @export 
-subset.survtab <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "survtab.meta", attr(x, "survtab.meta"))
-  }
-  y
-}
-
-
-#' @export 
-`[.survmean` <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "survmean.mean", attr(x, "survmean.mean"))
-  }
-  y
-}
-
-
-#' @export 
-subset.survmean <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "survmean.mean", attr(x, "survmean.mean"))
-  }
-  y
-}
-
-#' @export 
-`[.rate` <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "rate.meta", attr(x, "rate.meta"))
-  }
-  y
-}
-
-#' @export 
-subset.rate <- function(x, ...) {
-  y <- NextMethod()
-  if (is.data.frame(y)) {
-    setattr(y, "class", class(x))
-    setattr(y, "rate.meta", attr(x, "rate.meta"))
-  }
-  y
-}
-
-
-
-
-
-prep_plot_survtab <- function(x, 
-                              y = NULL, 
-                              subset = NULL, 
-                              conf.int = TRUE, 
-                              enclos = parent.frame(1L), 
-                              ...) {
-  
-  ## subsetting ----------------------------------------------------------------
-  subset <- evalLogicalSubset(data = x, substiset = substitute(subset), 
-                              enclos = environment())
-  
-  attrs <- attributes(x)
-  
-  if (!inherits(x, "survtab")) stop("x is not a survtab object")
-  if (is.null(attrs$survtab.meta)) {
-    stop("Missing meta information (attributes) in survtab object; ",
-         "have you tampered with it after estimation?")
-  }
-  strata.vars <- attrs$survtab.meta$print.vars
-  x <- copy(x)
-  setDT(x)
-  x <- x[subset, ]
-  
-  ## detect survival variables in data -----------------------------------------
-  surv_vars <- c("surv.obs","CIF.rel","CIF_","r.e2","r.pp")
-  wh <- NULL
-  for (k in surv_vars) {
-    wh <- c(wh, which(substr(names(x), 1, nchar(k)) == k))
-  }
-  surv_vars <- names(x)[wh]
-  surv_vars <- surv_vars[!substr(surv_vars, nchar(surv_vars)-1, nchar(surv_vars)) %in% c("hi","lo")]
-  if (length(surv_vars) == 0) {
-    stop("x does not appear to have any survival variables; ",
-         "did you tamper with it after estimation?")
-  }
-  
-  
-  ## getting y -----------------------------------------------------------------
-  if (!is.null(y)) {
-    if (!is.character(y)) {
-      stop("please supply y as a character string indicating ",
-           "the name of a variable in x")
-    }
-    if (length(y) > 1) stop("y must be of length 1 or NULL")
-    if (!all_names_present(x, y, stops = FALSE)) {
-      stop("Given survival variable in argument 'y' ",
-           "not present in survtab object ('", y, "')")
-    }
-  } else {
-    y <- surv_vars[length(surv_vars)]
-    if (length(surv_vars) > 1L) message("y was NULL; chose ", y, " automatically")
-  }
-  rm(surv_vars)
-  
-  if (substr(y, 1, 3) == "CIF" && conf.int) {
-    stop("No confidence intervals currently supported for CIFs. ",
-         "Hopefully they will be added in a future version; ",
-         "meanwhile use conf.int = FALSE when plotting CIFs.")
-  }
-  
-  
-  ## confidence intervals ------------------------------------------------------
-  y.lo <- y.hi <- y.ci <- NULL
-  if (conf.int) {
-    
-    y.lo <- paste0(y, ".lo")
-    y.hi <- paste0(y, ".hi")
-    y.ci <- c(y.lo, y.hi)
-    
-    badCIvars <- setdiff(y.ci, names(x))
-    if (sum(length(badCIvars))) {
-      stop("conf.int = TRUE, but missing confidence interval ",
-           "variables in data for y = '", y, "' (could not detect ",
-           "variables named", paste0("'", badCIvars, "'", collapse = ", ") ,")")
-    }
-    
-  } 
-  
-  list(x = x, y = y, y.ci = y.ci, y.lo = y.lo, y.hi = y.hi, 
-       strata = strata.vars, attrs = attrs)
-  
-}
-
-
-
-
-
-#' \code{plot} method for survtab objects
-#' 
-#' Plotting for \code{survtab} objects
-#' 
-#' @import graphics
-#' 
-#' @author Joonas Miettinen
-#' 
-#' @param x a \code{survtab} output object
-#' @param y survival a character vector of a variable names to plot;
-#' e.g. \code{y = "r.e2"}
-#' @param subset a logical condition; \code{obj} is subset accordingly 
-#' before plotting; use this for limiting to specific strata, 
-#' e.g. \code{subset = sex == "male"}
-#' @param conf.int logical; if \code{TRUE}, also plots any confidence intervals
-#' present in \code{obj} for variables in \code{y}
-#' @param col line colour; one value for each stratum; will be recycled
-#' @param lty line type; one value for each stratum; will be recycled
-#' @param ylab label for Y-axis
-#' @param xlab label for X-axis
-#' @param ... additional arguments passed on to \code{plot} and 
-#' \code{lines.survtab}; e.g. \code{ylim} can be defined this way
-#' @examples 
-#' data(sire)
-#' data(sibr)
-#' si <- rbind(sire, sibr)
-#' si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
-#' si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
-#' x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
-#'              status = status %in% 1:2, 
-#'              fot = 0:5, aggre = list(cancer, period, fot))
-#' st <- survtab_ag(fot ~ cancer + period, data = x, 
-#'                  surv.method = "lifetable", surv.type = "surv.obs")
-#' 
-#' plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
-#' lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
-#' 
-#' ## or
-#' plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
-#' @export
-#' @family survtab functions
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-plot.survtab <- function(x, y = NULL, subset=NULL, conf.int=TRUE, col=NULL,lty=NULL, ylab = NULL, xlab = NULL, ...) {
-  
-  Tstop <- delta <- NULL ## APPEASE R CMD CHECK
-  ## prep ----------------------------------------------------------------------
-  PF <- parent.frame(1L)
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = x, subset, enclos = PF)
-  
-  l <- prep_plot_survtab(x = x, y = y, subset = subset, 
-                         conf.int = conf.int, enclos = PF)
-  x <- l$x
-  y <- l$y
-  y.ci <- l$y.ci
-  y.lo <- l$y.lo
-  y.hi <- l$y.hi
-  
-  ## figure out limits, etc. to pass to plot() ---------------------------------
-  
-  min_y <- do.call("min", c(mget(c(y, y.lo), as.environment(x)), na.rm = TRUE))
-  min_y <- max(min_y, 0)
-  max_y <- max(x[[y]], na.rm=TRUE)
-  
-  if (substr(y, 1, 3) == "CIF") {
-    min_y <- 0.0
-  } else {
-    max_y <- max(1.0, max_y)
-  }
-  
-  max_x <- max(x[, Tstop])
-  min_x <- min(x[, Tstop-delta])
-  
-  if (is.null(ylab)) {
-    ylab <- "Observed survival"
-    if (substr(y[1], 1,4) %in% c("r.e2", "r.pp")) ylab <- "Net survival"
-    if (substr(y[1], 1,4) == "CIF_") ylab <- "Absolute risk"
-    if (substr(y[1], 1,6) == "CIF.rel") ylab <- "Absolute risk"
-  }
-  if (is.null(xlab)) xlab <- "Time from entry"
- 
-  ## attributes insurance to pass to lines.survtab
-  setattr(x, "survtab.meta", l$attrs$survtab.meta)
-  setattr(x, "class", c("survtab", "data.table", "data.frame"))
-  
-  ## plotting ------------------------------------------------------------------
-  plot(I(c(min_y,max_y))~I(c(min_x,max_x)), data=x, type="n", 
-       xlab = xlab, ylab = ylab, ...)
-  
-  
-  lines.survtab(x, subset = NULL, y = y, conf.int=conf.int,
-                col=col, lty=lty, ...)
-  
-  return(invisible(NULL))
-}
-
-
-#' \code{lines} method for survtab objects
-#' 
-#' Plot \code{lines} from a \code{survtab} object
-#' 
-#' @import graphics
-#' 
-#' @author Joonas Miettinen
-#' 
-#' @param x a \code{survtab} output object
-#' @param y a variable to plot; a quoted name of a variable
-#' in \code{x}; e.g. \code{y = "surv.obs"};
-#' if \code{NULL}, picks last survival variable column in order in \code{x}
-#' @param subset a logical condition; \code{obj} is subset accordingly 
-#' before plotting; use this for limiting to specific strata, 
-#' e.g. \code{subset = sex == "male"}
-#' @param conf.int logical; if \code{TRUE}, also plots any confidence intervals
-#' present in \code{obj} for variables in \code{y}
-#' @param col line colour passed to \code{matlines}
-#' @param lty line type passed to \code{matlines}
-#' @param ... additional arguments passed on to to a \code{matlines} call;
-#' e.g. \code{lwd} can be defined this way
-#' @examples 
-#' data(sire)
-#' data(sibr)
-#' si <- rbind(sire, sibr)
-#' si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
-#' si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
-#' x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
-#'              status = status %in% 1:2, 
-#'              fot = 0:5, aggre = list(cancer, period, fot))
-#' st <- survtab_ag(fot ~ cancer + period, data = x, 
-#'                  surv.method = "lifetable", surv.type = "surv.obs")
-#' 
-#' plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
-#' lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
-#' 
-#' ## or
-#' plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
-#' @export
-#' @family survtab functions
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-lines.survtab <- function(x, y = NULL, subset = NULL, 
-                          conf.int = TRUE, col=NULL, lty=NULL, ...) {
-  Tstop <- NULL ## APPEASE R CMD CHECK
-  ## prep ----------------------------------------------------------------------
-  PF <- parent.frame(1L)
-  global_breaks <- attr(x, "survtab.meta")$surv.breaks
-  
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = x, subset, enclos = PF)
-  
-  
-  l <- prep_plot_survtab(x = x, y = y, subset = subset, 
-                         conf.int = conf.int, enclos = environment())
-  x <- l$x
-  y <- l$y
-  y.ci <- l$y.ci
-  y.lo <- l$y.lo
-  y.hi <- l$y.hi
-  strata <- l$strata ## character vector of var names
-  
-  
-  ## impute first values (time = 0, surv = 1 / cif = 0) ------------------------
-  
-  is_CIF <- if (substr(y, 1, 3) == "CIF") TRUE else FALSE
-  setkeyv(x, c(strata, "Tstop"))
-  first <- x[1, ]
-  if (length(strata)) first <- unique(x, by = strata)
-  first[, c(y) := ifelse(is_CIF, 0, 1)]
-  first$Tstop <- min(global_breaks)
-  
-  if (length(y.ci) > 0) first[, (y.ci) := get(y) ]
-  x <- rbindlist(list(first, x[, ]), use.names = TRUE)
-  setkeyv(x, c(strata, "Tstop"))
-  
-  ## plotting ------------------------------------------------------------------
-  
-  if (is.null(lty)) {
-    lty <- list(c(1,2,2))
-    if (!length(y.ci)) lty <- list(1)
-  }
-  
-  lines_by(x = "Tstop", y = c(y, y.ci), 
-           strata.vars = strata, 
-           data = x, col = col, lty = lty, ...)
-  
-  return(invisible(NULL))
-}
-
-
-
-lines_by <- function(x, y, strata.vars = NULL, data, col, lty, ...) {
-  ## INTENTION: plots lines separately by strata,
-  ## which may have different colours / linetypes.
-  ## @param x a variable to plot y by; a character string
-  ## @param y a character vector of variables to plot by x;
-  ## e.g. the estimate and confidence interval variables
-  ## @param strata.vars a character string vector; variables
-  ## to add lines by, which may have different colours etc for identification
-  ## @param data a data.frame where x, y, and strata.vars are found
-  ## @param col a vector of colors passed to lines(); if vector length 1,
-  ## used for each level of strata. If vector length > 1, 
-  ## has to match to total number of strata. If list, must match
-  ## to number of strata by length and contain elements of length
-  ## length(y).
-  ## @param see col; line type passed to lines().
-  ## @param ... other arguments passed on to lines().
-  
-  TF <- environment()
-  PF <- parent.frame(1L)
-  
-  stopifnot(is.data.frame(data))
-  stopifnot(is.character(x) && length(x) == 1L)
-  stopifnot(is.character(y) && length(y) > 0L)
-  stopifnot(is.character(strata.vars) || is.null(strata.vars))
-  all_names_present(data, c(x,y,strata.vars))
-  
-  d <- mget(c(strata.vars, y, x), envir = as.environment(data))
-  setDT(d)
-  setkeyv(d, c(strata.vars, x))
-  
-  ## create list of datas
-  l <- list(d)
-  inter <- 1L
-  if (length(strata.vars)) {
-    inter <- do.call(interaction, d[, strata.vars, with = FALSE])
-    l <- vector("list", uniqueN(inter))
-    l <- split(d, f = inter, drop = TRUE)
-  }
-  
-  l <- lapply(l, function(tab) {
-    setDT(tab)
-    setcolsnull(tab, keep = c(x, y))
-    tab
-  })
-  
-  
-  ## figure out colours and ltys
-  for (objname in c("col", "lty")) {
-    obj <- TF[[objname]]
-    
-    if (missing(obj) || !length(obj)) obj <- 1
-    if (!length(obj) %in% c(1, length(l))) {
-      stop("Argument ", objname, " is not of length 1 or ",
-           "of length equal to total number of strata (",
-           length(l), ").")
-    }
-    
-    ol <- unlist(lapply(obj, length))
-    if (length(y) > 1 && is.list(obj) && !all(ol %in% c(1, length(y)))) {
-      stop("Argument y is of length > 1, and you passed ",
-           objname, " as a list of values, but at least one element is not ",
-           "of length 1 or length(y).")
-    }
-    
-    ## NOTE: rep works for vector and list just the same
-    if (length(obj) == 1) obj <- rep(obj, length(l))
-    obj <- as.list(obj)
-    
-    assign(x = objname, value = obj)
-  }
-  
-  lapply(seq_along(l), function(i) {
-    
-    tab <- l[[i]]
-    cols <- col[[i]]
-    ltys <- lty[[i]]
-    
-    matlines(x = tab[[x]], y = tab[, y, with = FALSE], 
-             col = cols, lty = ltys, ...)
-    
-  })
-  
-  invisible(NULL)
-}
-
-
-
-
-#' @title Graphically Inspect Curves Used in Mean Survival Computation
-#' @description Plots the observed (with extrapolation) and expected survival
-#' curves for all strata in an object created by \code{\link{survmean}}
-#' @author Joonas Miettinen
-#' @param x a \code{survmean} object
-#' @param ... arguments passed (ultimately) to \code{matlines}; you
-#' may, therefore, supply e.g. \code{xlab} through this, though arguments
-#' such as \code{lty} and \code{col} will not work
-#' @details 
-#' 
-#' For examples see \code{\link{survmean}}. This function is intended only
-#' for graphically inspecting that the observed survival curves with extrapolation
-#' and the expected survival curves have been sensibly computed in \code{survmean}.
-#' 
-#' If you want finer control over the plotted curves, extract the curves from
-#' the \code{survmean} output using 
-#' 
-#' \code{attr(x, "curves")}
-#' 
-#' where \code{x} is a \code{survmean} object.
-#' @export
-#' @family survmean functions
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-plot.survmean <- function(x, ...) {
-  at <- attr(x, "survmean.meta")
-  curves <- at$curves
-  if (is.null(curves)) {
-    stop("no curves information in x; sometimes lost if x ",
-         "altered after using survmean")
-  }
-  
-  by.vars <- at$tprint
-  by.vars <- c(by.vars, at$tadjust)
-  by.vars <- intersect(by.vars, names(curves))
-  if (!length(by.vars)) by.vars <- NULL
-  
-  plot(curves$surv ~ curves$Tstop, type="n",
-       xlab = "Time from entry", ylab = "Survival")
-  lines.survmean(x, ...)
-  
-  subr <- at$breaks[[at$survScale]]
-  abline(v = max(subr), lty=2, col="grey")
-  
-  if (length(by.vars)) {
-    ## add legend denoting colors
-    Stratum <- curves[, unique(interaction(.SD)), .SDcols = eval(by.vars)]
-    legend(x = "topright", legend = Stratum, col = seq_along(Stratum), lty = 1)
-  }
-  return(invisible(NULL))
-}
-
-#' @title Graphically Inspect Curves Used in Mean Survival Computation
-#' @description Plots the observed (with extrapolation) and expected survival
-#' curves for all strata in an object created by \code{\link{survmean}}
-#' @author Joonas Miettinen
-#' @param x a \code{survmean} object
-#' @param ... arguments passed (ultimately) to \code{matlines}; you
-#' may, therefore, supply e.g. \code{lwd} through this, though arguments
-#' such as \code{lty} and \code{col} will not work
-#' @details 
-#' 
-#' This function is intended to be a workhorse for \code{\link{plot.survmean}}.
-#' If you want finer control over the plotted curves, extract the curves from
-#' the \code{survmean} output using 
-#' 
-#' \code{attr(x, "curves")}
-#' 
-#' where \code{x} is a \code{survmean} object.
-#' @export
-#' @family survmean functions
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-lines.survmean <- function(x, ...) {
-  at <- copy(attr(x, "survmean.meta"))
-  curves <- at$curves
-  if (is.null(curves)) stop("no curves information in x; usually lost if x altered after using survmean")
-  
-  by.vars <- at$tprint
-  by.vars <- c(by.vars, at$tadjust)
-  by.vars <- c("survmean_type", by.vars)
-  by.vars <- intersect(by.vars, names(curves))
-  if (!length(by.vars)) by.vars <- NULL
-  
-  curves <- data.table(curves)
-  setkeyv(curves, c(by.vars, "Tstop"))
-  
-  type_levs <- length(levels(interaction(curves[, c(by.vars), with=FALSE])))/2L
-  other_levs <- 1L
-  if (length(by.vars) > 1) {
-    other_levs <- length(levels(interaction(curves[, setdiff(by.vars, "survmean_type"), with=FALSE])))
-  }
-  
-  curves <- cast_simple(curves, columns = by.vars, rows = "Tstop", values = "surv")
-  matlines(x=curves$Tstop, y=curves[, setdiff(names(curves), "Tstop"), with=FALSE],  
-           lty = rep(1:2, each=type_levs), col = 1:other_levs, ...)
-  return(invisible(NULL))
-}
-
-
-
-
-
-
-
-#' @export
-getCall.survtab <- function(x, ...) {
-  attributes(x)$survtab.meta$call
-}
-
-
-#' @export
-formula.survtab <- function(x, ...) {
-  attr(x, "survtab.meta")$arguments$formula
-}
-
-
-
-
-
-#' @export
-getCall.survmean <- function(x, ...) {
-  attributes(x)$survmean.meta$call
-}
-
-
-#' @export
-formula.survmean <- function(x, ...) {
-  attr(x, "survmean.meta")$formula
-}
-
-
-
+
+
+#' @export
+print.sir <- function(x, subset = NULL, ...) {
+  
+  at <- attributes(x)$sir.meta
+  PF <- parent.frame(1L)
+  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
+  x <- x[subset, ]
+  setDT(x)
+  
+  t1 <- paste0("SIR (adjusted by ", paste(at$adjust, collapse = ', '),')',
+               ' with ', at$conf.level*100, '% ', 'confidence intervals (', at$conf.type,')')
+  
+  
+  # cat
+  t3 <- paste0(' Total sir: ', round(at$pooled.sir$sir,2),' (', 
+               round(at$pooled.sir$sir.lo,2),'-', round(at$pooled.sir$sir.hi, 2),')\n',
+               ' Total observed: ', at$pooled.sir$observed, '\n',
+               ' Total expected: ', round(at$pooled.sir$expected,2), '\n',
+               ' Total person-years: ', round(at$pooled.sir$pyrs))
+
+  rv <- intersect(names(x), c('sir','sir.lo','sir.hi','observed','expected','pyrs'))
+  if (length(rv)) {
+    x[, (rv) := lapply(.SD, round, digits = 2L),  .SDcols = rv]
+  }
+  
+  rv <- intersect(names(x), c('p_value'))
+  if (length(rv)) {
+    x[, (rv) := lapply(.SD, round, digits = 4),  .SDcols = rv]
+  }
+
+  
+  if(is.null(at$lrt.test)) {
+    d <- paste("Could not test", at$lrt.test.type)
+  } else {
+    if(at$lrt.test.type == 'homogeneity') {
+      d <- paste("Test for homogeneity: p", p.round( c(at$lrt.test)))
+    }
+    if(at$lrt.test.type == 'trend') {
+      d <- paste("Test for trend: p", p.round( c(at$lrt.test)))
+    }
+  }
+  #b <- round(c(ta$total$sir, ta$total$sir.lo, ta$total$sir.hi), 2)
+  # cat('\n',"Total observed", ta$total$observed, '\n',
+  #     "Total expected:", ta$total$expected, '\n',
+  #     "SIR:", paste0(b[1], ' (',b[2], '-',b[3],')'), '\n',
+  #     "Person-years:", ta$total$pyrs, '\n',
+  #     fill=TRUE)
+  
+  cat(t1, '\n')
+  if(x[,.N] > 1) {
+    cat(d, '\n')
+  }
+  cat(fill=TRUE)
+  cat(t3, '\n', fill=TRUE)  
+  
+  print(data.table(x), ...)
+  return(invisible())
+}
+
+#' @export 
+`[.sir` <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "sir.meta", attr(x, "sir.meta"))
+  }
+  y
+}
+
+
+#' @import grDevices
+#' @export
+print.sirspline <- function(x, ...) {
+  if ( x$spline.dependent ) {
+    if( any( !is.na(x$p.values))) {
+      cat( 'global p-value:', p.round(x$p.values[1]),'\n' )
+      cat( 'level p-value:', p.round(x$p.values[2]) , fill= TRUE)      
+    } else {
+      cat( 'No models compared.', fill= TRUE)
+    }
+    cat('---', '\n')
+    cat('Colour codes:', '\n', fill=TRUE)
+  } else {
+    
+    for(i in 1:length(x$p.values)) {
+      cat( x$spline[i] ,': p ', p.round( x$p.values[[i]] ), '\n', sep = '')
+    }
+    cat(fill=TRUE)
+    
+  }
+  # Print colour codes:
+  cols <- unique(x$spline.est.A[,1])
+  col.length <- length(cols)
+  print( data.frame(levels = cols, colour = palette()[1:col.length]), include.rownames = FALSE)
+  
+  # Print p-values
+  return(invisible())
+}
+    
+
+#' Plot method for sir-object
+#' 
+#' Plot SIR estimates with error bars
+#' 
+#' @seealso \code{\link{sir}},  \code{\link{sirspline}}
+#' 
+#' @import graphics
+#' 
+#' @author Matti Rantanen
+#' 
+#' @param x an object returned by function \code{sir}
+#' @param conf.int default TRUE draws confidence intervals
+#' @param xlab overwrites default x-axis label
+#' @param ylab overwrites default y-axis label
+#' @param xlim x-axis minimum and maximum values
+#' @param main optional plot title
+#' @param abline logical; draws a grey line in SIR = 1
+#' @param log logical; SIR is not in log scale by default
+#' @param eps error bar vertical bar height (works only in 'model' or 'univariate')
+#' @param left.margin adjust left marginal of the plot to fit long variable names
+#' @param ... arguments passed on to plot(), segment and lines()
+#' 
+#' 
+#' @details Plot SIR estimates and confidence intervals 
+#' \itemize{
+#'  \item univariate - plots SIR with univariate confidence intervals
+#'  \item model - plots SIR with Poisson modelled confidence intervals
+#' }
+#' 
+#' \strong{Customize}
+#' Normal plot parameters can be passed to \code{plot}. These can be a vector when plotting error bars:
+#' \itemize{
+#'  \item \code{pch} - point type
+#'  \item \code{lty} - line type
+#'  \item \code{col} - line/point colour 
+#'  \item \code{lwd} - point/line size
+#' }
+#'
+#' \strong{Tips for plotting splines}
+#' It's possible to use \code{plot} to first draw the 
+#' confidence intervals using specific line type or colour and then plotting 
+#' again the estimate using \code{lines(... , conf.int = FALSE)} with different 
+#' settings. This works only when \code{plot.type} is 'splines'.
+#' 
+#' 
+#' @examples 
+#' \donttest{
+#' # Plot SIR estimates
+#'# plot(sir.by.gender, col = c(4,2), log=FALSE, eps=0.2, lty=1, lwd=2, pch=19,  
+#'#      main = 'SIR by gender', abline=TRUE)
+#' }
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+#' @export
+
+plot.sir <- function(x, conf.int = TRUE, ylab, xlab, xlim, main, 
+                     eps=0.2, abline = TRUE, log = FALSE, left.margin, ...) {
+  
+  a <- data.table(x)
+  at <- attributes(x)$sir.meta
+  level_names <- at$print
+  
+  if(is.null(level_names)) {
+    levels <- 'Crude'
+    level_names <- levels
+  }
+  else {
+    q <- paste0('paste(', paste(level_names, collapse=', '),', sep = ":")' )
+    q <- parse(text = q)
+    levels <- a[, eval(q)]
+  }
+  
+  # predefined parameters
+  if( missing(main) ){
+    main <- NA
+  }
+  if( missing(xlab) ){
+    xlab <- 'SIR'
+  }
+  if( missing(ylab) ){
+    ylab <- NA
+  }
+  if( missing(xlim) ) {
+    xlimit <- c(min(a$sir.lo[a$sir.lo <Inf]), max(a$sir.hi[a$sir.hi <Inf]))
+    xlimit <- xlimit + diff(xlimit)*c(-.3,.3)
+  } 
+  else {
+    xlimit <- xlim
+  }
+  
+  # par options
+  old_mar <- par("mar")
+  on.exit(par(mar = old_mar))
+  new.margin <- old_mar
+  if(missing(left.margin)) {
+    new.margin[2] <- 4.1 + sqrt( max(nchar(as.character(level_names))) )*2
+  } 
+  else {
+    new.margin[2] <- left.margin
+  }
+  par(mar = new.margin)
+  
+  # plot frame, estimates and CI (optional abline)
+  logarithm <- ''
+  if(log){
+    logarithm <- 'x'
+    if(xlimit[1]==0) xlimit[1] <- xlimit[1] + 0.01
+  }
+  y.axis.levels <- 1:length(levels)
+  plot(c(xlimit), c(min(y.axis.levels)-0.5, max(y.axis.levels)+0.5), 
+       type='n', yaxt = 'n', xlab=xlab, ylab=ylab, log=logarithm, main = main, ...)
+  axis(side = 2, at = y.axis.levels, labels = levels, las=1)
+  
+  if(abline) {
+    abline(v=1, col = 'darkgray')
+  }
+  
+  points(a$sir, factor(y.axis.levels, labels=levels), ...) 
+  if(conf.int) {
+    segments(a$sir.lo, y.axis.levels , a$sir.hi, y.axis.levels, ...)
+    segments(a$sir.lo, y.axis.levels - eps, a$sir.lo, y.axis.levels +eps, ... )
+    segments(a$sir.hi, y.axis.levels - eps, a$sir.hi, y.axis.levels +eps, ... )
+  }
+  
+  return(invisible(NULL))
+}
+
+#' @title \code{plot} method for sirspline-object
+#' 
+#' @description Plot SIR splines using R base graphics.
+#' 
+#' 
+#' @import graphics
+#' 
+#' @author Matti Rantanen
+#' 
+#' @param x an object returned by function sirspline
+#' @param conf.int logical; default TRUE draws also the 95 confidence intervals
+#' @param xlab overwrites default x-axis label; can be a vector if multiple splines fitted
+#' @param ylab overwrites default y-axis label; can be a vector if multiple splines fitted
+#' @param log logical; default FALSE. Should the y-axis be in log scale
+#' @param abline logical; draws a reference line where SIR = 1
+#' @param type select \code{type = 'n'} to plot only figure frames 
+#' @param ... arguments passed on to plot()
+#' 
+#' @details
+#' In \code{plot.sirspline} almost every graphical parameter are user
+#' adjustable, such as \code{ylim}, \code{xlim}.
+#' \code{plot.sirsplines} calls \code{lines.splines} to add lines.
+#' 
+#' The plot axis without lines can be plotted using option \code{type = 'n'}. 
+#' On top of the frame it's then possible to add a \code{grid}, 
+#' \code{abline} or text before plotting the lines (see: \code{sirspline}).
+#' @export
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+#' @family sir functions
+plot.sirspline <- function(x, conf.int=TRUE, abline = TRUE, log = FALSE, type, ylab, xlab,  ...) {
+
+  #print(list(...))
+  
+  ## premilinary checks  
+  if (is.null(x$spline.seq.A)) stop('No splines found.')
+  
+  ## prepare dimension and par
+  plotdim <- as.numeric(c( !is.null( x$spline.seq.A ),
+                           !is.null( x$spline.seq.B ),
+                           !is.null( x$spline.seq.C ) ))
+  
+  if(sum(plotdim) > 1) {
+    old_mfrow <- par("mfrow")
+    on.exit(par(mfrow = old_mfrow))
+    new_mfrow <- c(1,sum(plotdim))
+    par(mfrow = new_mfrow)
+    type <- 'l'
+  }
+  
+  ## set labels
+  if ( missing(xlab) ) {
+    xlab <- x$spline
+  }
+  
+  if ( missing(ylab) ) {
+    ylab <- rep('SIR',sum(plotdim))
+    if(log){
+      ylab <- rep('log(SIR)', sum(plotdim))
+    }
+    if(x$spline.dependent & sum(plotdim) > 1) {
+      ylab <- c(ylab[1], paste(ylab[2:sum(plotdim)], 'ratio'))
+    }
+  }
+  else{
+    if( length(ylab) < sum(plotdim))
+      ylab <- rep(ylab, sum(plotdim))
+    if(length(ylab) > sum(plotdim)) {
+      warning('set ylabs in a vector length of num of plots (',sum(plotdim),')')
+    }
+  }
+  
+  ## set scale
+  if(!is.logical(log)) stop('log should be a logical value.')
+  log.bin <- ifelse(log, 'y', '')
+  
+  ## remove infinite values
+  #rm_inf <- function(est){
+  #  x[[est]][ is.finite(x[[est]][[2]]) & is.finite(x[[est]][[3]]) & is.finite(x[[est]][[4]]), ]
+  #}
+  
+  spl <- c('spline.seq.A', 'spline.seq.B', 'spline.seq.C')[1:sum(plotdim)]
+  est <- gsub("seq", "est", spl)
+  
+  for (i in 1:sum(plotdim)) {  # age, per, fot, 
+    # empty plot
+    max_x <- range(x[[spl[i]]])
+    max_y <- range( x[[est[i]]][, 2:4] )
+    plot(max_x, max_y, type = 'n', ylab = ylab[i], xlab = xlab[i], log = log.bin, ...) 
+    if(abline) abline(h = 1)
+    
+    # plot lines
+    if (missing(type) || type != 'n') {
+      lines.sirspline(x, conf.int = conf.int, select.spline = i, ...)
+    }
+  }
+  return(invisible(NULL))
+}
+
+
+
+#' @title lines method for sirspline-object
+#' @description Plot SIR spline lines with R base graphics
+#' 
+#' 
+#' @author Matti Rantanen
+#' 
+#' @param x an object returned by function sirspline
+#' @param conf.int logical; default TRUE draws also the 95 confidence intervals
+#' @param print.levels name(s) to be plotted. Default plots all levels.
+#' @param select.spline select which spline variable (a number or a name) is plotted.
+#' @param ... arguments passed on to lines()
+#' 
+#' @details  In \code{lines.sirspline} most of graphical parameters is user 
+#' adjustable.
+#' Desired spline variable can be selected with \code{select.spline} and only one
+#' can be plotted at a time. The spline variable can include 
+#' several levels, e.g. gender (these are the levels of \code{print}
+#' from \code{sirspline}). All levels are printed by default, but a
+#' specific level can be selected using argument
+#' \code{print.levels}. Printing the levels separately enables  e.g. to
+#' give different colours for each level.
+#' 
+#' @family sir functions
+#' 
+#' @import graphics
+#' @export
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+
+lines.sirspline <- function(x, conf.int = TRUE, print.levels = NA, select.spline, ... ){
+  ## input: sirspline object, with only one spline var (spline.est.A)
+  ## input: print levels can be > 1.
+  
+  ## subset splines
+  if( length(x$spline) > 1 ) {
+    if ( missing(select.spline) ) {
+      stop(paste('select what spline to plot in select.spline:', paste(x$spline, collapse = ', ')))
+    }
+    else {
+      if(is.numeric(select.spline)) {
+        k <- select.spline
+      }
+      else {
+        k <- which(x$spline == select.spline)
+      }
+      if(length(k) == 0 | length(x$spline) < k) stop('select.spline name/number is incorrect')
+    }
+  } 
+  else {
+    k <- 1
+  }
+  
+  spl <- c('spline.seq.A', 'spline.seq.B', 'spline.seq.C')[k]
+  est <- gsub("seq", "est", spl)
+  
+  ## remove infinite values
+  # x[[h]] <- rm_inf(est=h)
+  
+  # get print levels
+  if(missing(print.levels)) {
+    print.levels <- NA
+  }
+  pl <- unique(x$spline.est.A[,1])
+  if(any( is.null(print.levels), is.na(print.levels))) {
+    print.levels <- pl
+  }
+  pl <- pl[ pl %in% print.levels]
+  
+  ## get conf.int
+  if( !is.logical(conf.int) ) stop('conf.int is not logical')
+  n <- c(2,4)[c(!conf.int, conf.int)]
+  
+  
+  ## draw lines
+  for( l in pl ){
+    # loop through print.levels
+    index <- which(x$spline.est.A$i == l)
+    
+    for(m in 2:n) {
+      # loop through estiamte and confidence intervals
+      lines(x = x[[spl]], y = x[[est]][index, m], ...)
+    }
+  }
+  return(invisible(NULL))
+}
+
+#' @title Print an rate object
+#' @author Matti Rantanen
+#' @description Print method function for \code{rate} objects; see
+#' \code{\link{rate}}.
+#' @param x an \code{rate} object
+#' @param subset a logical condition to subset results table by
+#' before printing; use this to limit to a certain stratum. E.g.
+#' \code{subset = sex == "female"}
+#' @param ... arguments for data.tables print method, e.g. row.names = FALSE suppresses row numbers.
+#' @export
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+print.rate <- function(x, subset = NULL, ...) {
+  
+  ra <- attributes(x)$rate.meta
+  PF <- parent.frame(1L)
+  TF <- environment()
+  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
+  x <- x[subset, ]
+  
+  # pre texts:
+  cat('\n')
+  if(!is.null(ra$adjust)){
+    if(is.character(ra$weights)) {
+      a <- paste(ra$weights, collapse = ',')
+    }
+    if(all(is.numeric(ra$weights))) {
+      a <- length(ra$weights)
+    }
+    if(is.list(ra$weights)) {
+      a <- sapply(ra$weights, length)
+    }
+    
+    b <- paste(ra$adjust,a, collapse = ', ', sep = '; ')
+    cat('Adjusted rates (', b,') ', sep = '')
+  }
+  else{
+    cat('Crude rates ')
+  }
+  cat('and', '95%', 'confidence intervals:', fill=TRUE)
+  cat('\n')
+  # table itself
+
+  
+  setDT(x)
+  print(x, ...)
+  return(invisible(NULL))
+}
+
+
+#' @title plot method for rate object
+#' @description Plot rate estimates with confidence intervals lines using R base graphics
+#' @author Matti Rantanen
+#' 
+#' @param x a rate object (see \code{\link{rate}})
+#' @param conf.int logical; default TRUE draws the confidence intervals
+#' @param eps is the height of the ending of the error bars
+#' @param left.margin set a custom left margin for long variable names. Function
+#' tries to do it by default.
+#' @param xlim change the x-axis location
+#' @param ... arguments passed on to graphical functions points and segment 
+#' (e.g. \code{col}, \code{lwd}, \code{pch} and \code{cex})
+#' 
+#' @details This is limited explanatory tool but most graphical 
+#' parameters are user adjustable. 
+#' 
+#' @import graphics
+#' @export
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+plot.rate <- function(x, conf.int = TRUE, eps = 0.2, left.margin, xlim, ...) {
+  
+  ra <- attributes(x)$rate.meta
+  varcol <- ra$print
+  
+  if(is.null(varcol)) {
+    lvl.name <- 'Crude'
+  }
+  else {
+    pp <- paste0('paste(', paste(varcol, collapse=','),',sep = ":")')
+    q <- parse(text=pp)
+    lvl.name <- x[,eval(q)]
+  }
+  lvls <- 1:length(lvl.name)
+  
+  # WHICH RATE:
+  if('rate.adj' %in% names(x)) {
+    r <- x$rate.adj
+    hi <- x$rate.adj.hi
+    lo <- x$rate.adj.lo
+  }
+  else {
+    r <- x$rate
+    hi <- x$rate.hi
+    lo <- x$rate.lo
+  }
+  # X-AXIS LIMITs
+  if(missing(xlim)) {
+    t <- range(na.omit(c(lo , r, hi)))
+    t0 <- (t[2]-t[1])/4
+    xlimit <- c(pmax(t[1]-t0, 0), t[2] + t0)
+  } 
+  else {
+    xlimit <- xlim
+  }
+  
+  # MARGINS  
+  old_mar <- par("mar")
+  on.exit(par(mar = old_mar))
+  if(missing(left.margin)) {
+   new.margin <- par("mar")
+   new.margin[2] <- 4.1 + sqrt( max(nchar(as.character(lvl.name))) )*2
+  }
+  else {
+   new.margin[2] <- left.margin
+  }
+  par(mar = new.margin)
+  
+  plot(c(xlimit), c(min(lvls)-0.5, max(lvls)+0.5), type='n', yaxt = 'n', ylab = '', xlab='')
+  axis(side = 2, at = lvls, labels = lvl.name, las = 1)
+  points(r, lvls, ...)
+  
+  if(conf.int) {
+    segments(lo, lvls, hi, lvls, ...)
+    segments(lo, lvls - eps, lo, lvls + eps, ...)
+    segments(hi, lvls - eps, hi, lvls + eps, ...)
+  }
+  return(invisible(NULL))
+}
+
+
+#' @export 
+print.yrs <- function(x, ...) {
+  print(as.numeric(x))
+}
+
+
+#' @export 
+`[.yrs` <- function(x, ...) {
+  yl <- attr(x, "year.length")
+  structure(NextMethod(), year.length = yl, class = c("yrs", "numeric"))
+}
+
+
+#' @export 
+`[.aggre` <- function(x, ...) {
+  xa <- attributes(x)
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", xa$class)
+    setattr(y, "aggre.meta", xa$aggre.meta)
+    setattr(y, "breaks", xa$breaks)
+  }
+  y
+}
+
+#' @export 
+subset.aggre <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "aggre.meta", attr(x, "aggre.meta"))
+    setattr(y, "breaks", attr(x, "breaks"))
+  }
+  y
+}
+
+preface_survtab.print <- function(x) {
+  surv.int <- NULL ## APPEASE R CMD CHECK
+  at <- attributes(x)$survtab.meta
+  arg <- at$arguments
+  
+  cat("\n")
+  cat("Call: \n", oneWhitespace(deparse(at$call)), "\n")
+  cat("\n")
+  cat("Type arguments: \n surv.type:", as.character(arg$surv.type), 
+      "--- surv.method:", as.character(arg$surv.method))
+  if (as.character(arg$surv.type) == "surv.rel")
+    cat(" --- relsurv.method:", as.character(arg$relsurv.method))
+  cat("\n \n")
+  cat("Confidence interval arguments: \n level:", 
+      as.character(arg$conf.level*100), "%")
+  cat(" --- transformation:",
+      as.character(arg$conf.type))
+  cat("\n \n")
+  cat("Totals:") 
+  totCat <- paste0("\n person-time:", round(sum(x$pyrs)))
+  if (arg$surv.method == "lifetable") {
+    totCat <- paste0("\n at-risk at T=0: ", round(sum(x[surv.int == 1L]$n)))
+  }
+    
+  cat(totCat)
+  cat(" --- events:", sum(x$d))
+  cat("\n \n")
+  if (length(at$print.vars) > 0L) {
+    cat("Stratified by:", paste0("'", at$print.vars, "'", collapse = ", "))
+    if (length(at$adjust.vars) > 0L) cat(" --- ")
+  }
+  if (length(at$adjust.vars) > 0L) {
+    cat("Adjusted by:", paste0("'", at$adjust.vars, "'", collapse = ", "))
+  }
+  cat("\n")
+  invisible()
+}
+
+
+#' @title Print an \code{aggre} Object
+#' @author Joonas Miettinen
+#' @description Print method function for \code{aggre} objects; see
+#' \code{\link{as.aggre}} and \code{\link{aggre}}.
+#' @param x an \code{aggre} object
+#' @param subset a logical condition to subset results table by
+#' before printing; use this to limit to a certain stratum. E.g.
+#' \code{subset = sex == "male"}
+#' @param ... arguments passed to \code{print.data.table}; try e.g.
+#' \code{top = 2} for numbers of rows in head and tail printed 
+#' if the table is large, 
+#' \code{nrow = 100} for number of rows to print, etc.
+#' @export
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+print.aggre <- function(x, subset = NULL, ...) {
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  sa <- attributes(x)$aggre.meta
+  
+  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
+  x <- x[subset, ]
+  setDT(x)
+  
+  print(x, ...)
+  return(invisible(NULL))
+}
+
+#' @title Summarize an \code{aggre} Object
+#' @author Joonas Miettinen
+#' @description \code{summary} method function for \code{aggre} objects; see
+#' \code{\link{as.aggre}} and \code{\link{aggre}}.
+#' @param object an \code{aggre} object
+#' @param by list of columns to summarize by - e.g. \code{list(V1, V2)}
+#' where \code{V1} and \code{V2} are columns in the data.
+#' @param subset a logical condition to subset results table by
+#' before summarizing; use this to limit to a certain stratum. E.g.
+#' \code{subset = sex == "male"}
+#' @param ... unused
+#' @export
+#' @family aggregation functions
+#' @return
+#' Returns a `data.table` --- a further aggregated version of `object`.
+summary.aggre <- function(object, by = NULL, subset = NULL, ...) {
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  x <- object
+  sa <- attributes(x)$aggre.meta
+  
+  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
+  x <- x[subset, ]
+  setDT(x)
+  
+  bys <- substitute(by)
+  bye <- evalPopArg(x, bys, enclos = environment(), types = c("list", "NULL"))
+  
+  vals <- sa$values
+  vals <- intersect(names(x), vals)
+  if (!length(vals)) {
+    cat("No originally created value columns appear to be left in data.")
+  }
+  r <- x[, lapply(.SD, sum), by = eval(bye), .SDcols = vals]
+  r
+}
+
+#' @title Print a survtab Object
+#' @author Joonas Miettinen
+#' @description Print method function for \code{survtab} objects; see
+#' \code{\link{survtab_ag}}.
+#' @param x a \code{survtab} object
+#' @param subset a logical condition to subset results table by
+#' before printing; use this to limit to a certain stratum. E.g.
+#' \code{subset = sex == "male"}
+#' @param ... arguments passed to \code{print.data.table}; try e.g.
+#' \code{top = 2} for numbers of rows in head and tail printed 
+#' if the table is large, 
+#' \code{nrow = 100} for number of rows to print, etc.
+#' @export
+#' @family survtab functions
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+print.survtab <- function(x, subset = NULL, ...) {
+  
+  Tstart <- Tstop <- NULL ## APPEASE R CMD CHECK
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  sa <- attributes(x)$survtab.meta
+  
+  subset <- evalLogicalSubset(x, substitute(subset), enclos = PF)
+  x <- x[subset, ]
+  
+  preface_survtab.print(x)
+  
+  setDT(x)
+  
+  if (nrow(x) == 0L) {
+    print(x)
+    return(invisible())
+  }
+  
+  pv <- as.character(sa$print.vars)
+  if (length(pv) == 0L) pv <- NULL
+  
+  magicMedian <- function(x) {
+    if (length(x) %% 2L == 0L) median(x[-1L], na.rm = TRUE) else
+      median(x, na.rm = TRUE)
+  }
+  
+  ## to avoid e.g. 'factor(V1, 1:2)' going bonkers
+  pv_orig <- pv
+  if (length(pv) > 0L) {
+    pv <- makeTempVarName(x, pre = paste0("print_", 1:length(pv)))
+    setnames(x, pv_orig, pv)
+  }
+  
+  medmax <- x[, list(Tstop = c(magicMedian(c(min(Tstart),Tstop)), max(Tstop))), keyby = eval(pv)]
+  
+  setkeyv(medmax, c(pv, "Tstop"))
+  setkeyv(x, c(pv, "Tstop"))
+  x <- x[medmax]
+  
+  rv <- intersect(names(x), c(sa$est.vars, sa$CI.vars, sa$misc.vars))
+  if (length(rv)) {
+    x[, (rv) := lapply(.SD, round, digits = 4L),  .SDcols = rv]
+  }
+  
+  sv <- intersect(names(x), sa$SE.vars)
+  if (length(sv > 0L)) {
+    x[, c(sv) := lapply(.SD, signif, digits = 4L), .SDcols = sv]
+  }
+  
+  
+  setcolsnull(x, keep = c(pv, "Tstop", sa$surv.vars), colorder = TRUE)
+  if (length(pv)) setnames(x, pv, pv_orig)
+  print(data.table(x), ...)
+  return(invisible(NULL))
+}
+
+#' @title Summarize a survtab Object
+#' @author Joonas Miettinen
+#' @description Summary method function for \code{survtab} objects; see
+#' \code{\link{survtab_ag}}. Returns estimates at given time points
+#' or all time points if \code{t} and \code{q} are both \code{NULL}.
+#' @param object a \code{survtab} object
+#' @param t a vector of times at which time points (actually intervals that
+#' contain t) to print summary table of survival function estimates by strata;
+#' values not existing in any interval cause rows containing only \code{NAs} to
+#' be returned. 
+#' @param q a named \code{list} of quantiles to include in returned data set,
+#' where names must match to estimates in \code{object};
+#' returns intervals where the quantiles are reached first;
+#' e.g. \code{list(surv.obs = 0.5)} finds the interval where \code{surv.obs}
+#' is 0.45 and 0.55 at the beginning and end of the interval, respectively;
+#' returns rows with \code{NA} values for quantiles not reached in estimates
+#' (e.g. if \code{q = list(surv.obs = 0.5)} but lowest estimate is 0.6);
+#' see Examples.
+#' @param subset a logical condition to subset results table by
+#' before printing; use this to limit to a certain stratum. E.g.
+#' \code{subset = sex == "male"}
+#' @param ... unused; required for congruence with other \code{summary} methods
+#' 
+#' @details
+#' Note that this function returns the intervals and NOT the time points
+#' corresponding to quantiles / estimates corresponding to time points.
+#' If you want precise estimates at time points that are not interval breaks,
+#' add the time points as breaks and re-estimate the survival time function.
+#' In interval-based estimation, the estimates denote e.g. probability of 
+#' dying \emph{during} the interval, so time points within the intervals
+#' are not usually considered at all. See e.g. Seppa, Dyba, and Hakulinen 
+#' (2015).
+#' 
+#' @references
+#' Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
+#' Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
+#' \doi{10.1016/B978-0-12-801238-3.02745-8}
+#' 
+#' @examples 
+#' 
+#' library(Epi)
+#' 
+#' ## NOTE: recommended to use factor status variable
+#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+#'            exit = list(CAL = get.yrs(ex_date)), 
+#'            data = sire[sire$dg_date < sire$ex_date, ],
+#'            exit.status = factor(status, levels = 0:2, 
+#'            labels = c("alive", "canD", "othD")), 
+#'            merge = TRUE)
+#' ## pretend some are male
+#' set.seed(1L)
+#' x$sex <- rbinom(nrow(x), 1, 0.5)
+#' ## observed survival
+#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+#'                   surv.type = "cif.obs",
+#'                   breaks = list(FUT = seq(0, 5, 1/12)))
+#' 
+#' ## estimates at full years of follow-up
+#' summary(st, t = 1:5)
+#' 
+#' ## interval estimate closest to 75th percentile, i.e.
+#' ## first interval where surv.obs < 0.75 at end
+#' ## (just switch 0.75 to 0.5 for median survival, etc.)
+#' summary(st, q = list(surv.obs = 0.75))
+#' ## multiple quantiles
+#' summary(st, q = list(surv.obs = c(0.75, 0.90), CIF_canD = 0.20))
+#' 
+#' ## if you want all estimates in a new data.frame, you can also simply do
+#' 
+#' x <- as.data.frame(st)
+#' @return
+#' A `data.table`: a slice from `object` based on `t`, `subset`, and `q`.
+#' @export
+#' @family survtab functions
+summary.survtab <- function(object, t = NULL, subset = NULL, q = NULL, ...) {
+  
+  PF <- parent.frame(1L)
+  at <- copy(attr(object, "survtab.meta"))
+  subr <- copy(at$surv.breaks)
+  
+  if (!is.null(t) && !is.null(q)) {
+    stop("Only supply either t or q.")
+  } 
+  
+  sb <- substitute(subset)
+  subset <- evalLogicalSubset(object, sb, enclos = PF)
+  x <- object[subset, ]
+  
+  ## to avoid e.g. 'factor(V1, 1:2)' going bonkers
+  pv_orig <- pv <- at$print.vars
+  if (length(pv) > 0L) {
+    pv <- makeTempVarName(x, pre = paste0("print_", 1:length(pv)))
+    setnames(x, pv_orig, pv)
+  }
+  
+  setDT(x)
+  
+  ## quantile detection --------------------------------------------------------
+  if (!is.null(q)) {
+    bn <- setdiff(names(q), at$est.vars)
+    if (length(bn) > 0L) {
+      stop("No survival time function estimates named ",
+           paste0("'", bn, "'", collapse = ", "), 
+           " found in supplied survtab object. Available ",
+           "survival time function estimates: ", 
+           paste0("'", at$est.vars, "'", collapse = ", "))
+    }
+    
+    lapply(q, function(x) {
+      if (min(x <= 0L) || max(x >= 1L)) {
+        stop("Quantiles must be expressed as numbers between 0 and 1, ",
+             "e.g. surv.obs = 0.5.")
+      }
+    })
+    
+    
+    m <- x[, .SD[1, ], keyby = eval(pv)][, c(pv, "Tstop"), with = FALSE]
+    setDF(m)
+    
+    rollVars <- makeTempVarName(x, pre = names(q))
+    x[, c(rollVars) := lapply(.SD, copy), .SDcols = names(q)]
+    
+    m <- lapply(seq_along(q), function(i) {
+      m <- merge(m, q[[i]])
+      setnames(m, "y", rollVars[i])
+      if (length(pv)) setorderv(m, pv)
+      m[, c(pv, rollVars[i]), drop = FALSE]
+    })
+    names(m) <- names(q)
+    
+    l <- vector("list", length(q))
+    names(l) <- names(q)
+    for (k in names(q)) {
+      
+      l[[k]] <- setDT(x[m[[k]], on = names(m[[k]]), roll = 1L])
+      
+    }
+    l <- rbindlist(l)
+    set(l, j = rollVars, value = NULL)
+    if (length(pv)) setkeyv(l, pv)
+    x <- l
+  }
+  
+  ## time point detection ------------------------------------------------------
+  
+  if (!is.null(t)) {
+    
+    tcutv <- makeTempVarName(x, pre = "cut_time_")
+    
+    set(x, j = tcutv, value = cut(x$Tstop, breaks = subr, right = TRUE, 
+                                  include.lowest = FALSE))
+    cutt <- cut(t, breaks = subr, right = TRUE, include.lowest = FALSE)
+    
+    l <- list(cutt)
+    names(l) <- tcutv
+    if (length(pv)) {
+      pvdt <- setDF(unique(x, by = pv))[, pv, drop = FALSE]
+      l <- setDT(merge(pvdt, as.data.frame(l)))
+      setkeyv(l, pv)
+    }
+    
+    x <- x[l, on = c(pv, tcutv)]
+    
+    set(x, j = tcutv, value = NULL)
+    if (length(pv)) setkeyv(x, pv)
+  }
+  
+  
+  ## final touches -------------------------------------------------------------
+  if (length(pv) > 0L) setnames(x, pv, pv_orig)
+  
+  if (!return_DT()) setDFpe(x)
+  
+  x
+}
+
+#' @export 
+`[.survtab` <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "survtab.meta", attr(x, "survtab.meta"))
+  }
+  y
+}
+
+#' @export 
+subset.survtab <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "survtab.meta", attr(x, "survtab.meta"))
+  }
+  y
+}
+
+
+#' @export 
+`[.survmean` <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "survmean.mean", attr(x, "survmean.mean"))
+  }
+  y
+}
+
+
+#' @export 
+subset.survmean <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "survmean.mean", attr(x, "survmean.mean"))
+  }
+  y
+}
+
+#' @export 
+`[.rate` <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "rate.meta", attr(x, "rate.meta"))
+  }
+  y
+}
+
+#' @export 
+subset.rate <- function(x, ...) {
+  y <- NextMethod()
+  if (is.data.frame(y)) {
+    setattr(y, "class", class(x))
+    setattr(y, "rate.meta", attr(x, "rate.meta"))
+  }
+  y
+}
+
+
+
+
+
+prep_plot_survtab <- function(x, 
+                              y = NULL, 
+                              subset = NULL, 
+                              conf.int = TRUE, 
+                              enclos = parent.frame(1L), 
+                              ...) {
+  
+  ## subsetting ----------------------------------------------------------------
+  subset <- evalLogicalSubset(data = x, substiset = substitute(subset), 
+                              enclos = environment())
+  
+  attrs <- attributes(x)
+  
+  if (!inherits(x, "survtab")) stop("x is not a survtab object")
+  if (is.null(attrs$survtab.meta)) {
+    stop("Missing meta information (attributes) in survtab object; ",
+         "have you tampered with it after estimation?")
+  }
+  strata.vars <- attrs$survtab.meta$print.vars
+  x <- copy(x)
+  setDT(x)
+  x <- x[subset, ]
+  
+  ## detect survival variables in data -----------------------------------------
+  surv_vars <- c("surv.obs","CIF.rel","CIF_","r.e2","r.pp")
+  wh <- NULL
+  for (k in surv_vars) {
+    wh <- c(wh, which(substr(names(x), 1, nchar(k)) == k))
+  }
+  surv_vars <- names(x)[wh]
+  surv_vars <- surv_vars[!substr(surv_vars, nchar(surv_vars)-1, nchar(surv_vars)) %in% c("hi","lo")]
+  if (length(surv_vars) == 0) {
+    stop("x does not appear to have any survival variables; ",
+         "did you tamper with it after estimation?")
+  }
+  
+  
+  ## getting y -----------------------------------------------------------------
+  if (!is.null(y)) {
+    if (!is.character(y)) {
+      stop("please supply y as a character string indicating ",
+           "the name of a variable in x")
+    }
+    if (length(y) > 1) stop("y must be of length 1 or NULL")
+    if (!all_names_present(x, y, stops = FALSE)) {
+      stop("Given survival variable in argument 'y' ",
+           "not present in survtab object ('", y, "')")
+    }
+  } else {
+    y <- surv_vars[length(surv_vars)]
+    if (length(surv_vars) > 1L) message("y was NULL; chose ", y, " automatically")
+  }
+  rm(surv_vars)
+  
+  if (substr(y, 1, 3) == "CIF" && conf.int) {
+    stop("No confidence intervals currently supported for CIFs. ",
+         "Hopefully they will be added in a future version; ",
+         "meanwhile use conf.int = FALSE when plotting CIFs.")
+  }
+  
+  
+  ## confidence intervals ------------------------------------------------------
+  y.lo <- y.hi <- y.ci <- NULL
+  if (conf.int) {
+    
+    y.lo <- paste0(y, ".lo")
+    y.hi <- paste0(y, ".hi")
+    y.ci <- c(y.lo, y.hi)
+    
+    badCIvars <- setdiff(y.ci, names(x))
+    if (sum(length(badCIvars))) {
+      stop("conf.int = TRUE, but missing confidence interval ",
+           "variables in data for y = '", y, "' (could not detect ",
+           "variables named", paste0("'", badCIvars, "'", collapse = ", ") ,")")
+    }
+    
+  } 
+  
+  list(x = x, y = y, y.ci = y.ci, y.lo = y.lo, y.hi = y.hi, 
+       strata = strata.vars, attrs = attrs)
+  
+}
+
+
+
+
+
+#' \code{plot} method for survtab objects
+#' 
+#' Plotting for \code{survtab} objects
+#' 
+#' @import graphics
+#' 
+#' @author Joonas Miettinen
+#' 
+#' @param x a \code{survtab} output object
+#' @param y survival a character vector of a variable names to plot;
+#' e.g. \code{y = "r.e2"}
+#' @param subset a logical condition; \code{obj} is subset accordingly 
+#' before plotting; use this for limiting to specific strata, 
+#' e.g. \code{subset = sex == "male"}
+#' @param conf.int logical; if \code{TRUE}, also plots any confidence intervals
+#' present in \code{obj} for variables in \code{y}
+#' @param col line colour; one value for each stratum; will be recycled
+#' @param lty line type; one value for each stratum; will be recycled
+#' @param ylab label for Y-axis
+#' @param xlab label for X-axis
+#' @param ... additional arguments passed on to \code{plot} and 
+#' \code{lines.survtab}; e.g. \code{ylim} can be defined this way
+#' @examples 
+#' data(sire)
+#' data(sibr)
+#' si <- rbind(sire, sibr)
+#' si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
+#' si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
+#' x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
+#'              status = status %in% 1:2, 
+#'              fot = 0:5, aggre = list(cancer, period, fot))
+#' st <- survtab_ag(fot ~ cancer + period, data = x, 
+#'                  surv.method = "lifetable", surv.type = "surv.obs")
+#' 
+#' plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
+#' lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
+#' 
+#' ## or
+#' plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
+#' @export
+#' @family survtab functions
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+plot.survtab <- function(x, y = NULL, subset=NULL, conf.int=TRUE, col=NULL,lty=NULL, ylab = NULL, xlab = NULL, ...) {
+  
+  Tstop <- delta <- NULL ## APPEASE R CMD CHECK
+  ## prep ----------------------------------------------------------------------
+  PF <- parent.frame(1L)
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = x, subset, enclos = PF)
+  
+  l <- prep_plot_survtab(x = x, y = y, subset = subset, 
+                         conf.int = conf.int, enclos = PF)
+  x <- l$x
+  y <- l$y
+  y.ci <- l$y.ci
+  y.lo <- l$y.lo
+  y.hi <- l$y.hi
+  
+  ## figure out limits, etc. to pass to plot() ---------------------------------
+  
+  min_y <- do.call("min", c(mget(c(y, y.lo), as.environment(x)), na.rm = TRUE))
+  min_y <- max(min_y, 0)
+  max_y <- max(x[[y]], na.rm=TRUE)
+  
+  if (substr(y, 1, 3) == "CIF") {
+    min_y <- 0.0
+  } else {
+    max_y <- max(1.0, max_y)
+  }
+  
+  max_x <- max(x[, Tstop])
+  min_x <- min(x[, Tstop-delta])
+  
+  if (is.null(ylab)) {
+    ylab <- "Observed survival"
+    if (substr(y[1], 1,4) %in% c("r.e2", "r.pp")) ylab <- "Net survival"
+    if (substr(y[1], 1,4) == "CIF_") ylab <- "Absolute risk"
+    if (substr(y[1], 1,6) == "CIF.rel") ylab <- "Absolute risk"
+  }
+  if (is.null(xlab)) xlab <- "Time from entry"
+ 
+  ## attributes insurance to pass to lines.survtab
+  setattr(x, "survtab.meta", l$attrs$survtab.meta)
+  setattr(x, "class", c("survtab", "data.table", "data.frame"))
+  
+  ## plotting ------------------------------------------------------------------
+  plot(I(c(min_y,max_y))~I(c(min_x,max_x)), data=x, type="n", 
+       xlab = xlab, ylab = ylab, ...)
+  
+  
+  lines.survtab(x, subset = NULL, y = y, conf.int=conf.int,
+                col=col, lty=lty, ...)
+  
+  return(invisible(NULL))
+}
+
+
+#' \code{lines} method for survtab objects
+#' 
+#' Plot \code{lines} from a \code{survtab} object
+#' 
+#' @import graphics
+#' 
+#' @author Joonas Miettinen
+#' 
+#' @param x a \code{survtab} output object
+#' @param y a variable to plot; a quoted name of a variable
+#' in \code{x}; e.g. \code{y = "surv.obs"};
+#' if \code{NULL}, picks last survival variable column in order in \code{x}
+#' @param subset a logical condition; \code{obj} is subset accordingly 
+#' before plotting; use this for limiting to specific strata, 
+#' e.g. \code{subset = sex == "male"}
+#' @param conf.int logical; if \code{TRUE}, also plots any confidence intervals
+#' present in \code{obj} for variables in \code{y}
+#' @param col line colour passed to \code{matlines}
+#' @param lty line type passed to \code{matlines}
+#' @param ... additional arguments passed on to to a \code{matlines} call;
+#' e.g. \code{lwd} can be defined this way
+#' @examples 
+#' data(sire)
+#' data(sibr)
+#' si <- rbind(sire, sibr)
+#' si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
+#' si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
+#' x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
+#'              status = status %in% 1:2, 
+#'              fot = 0:5, aggre = list(cancer, period, fot))
+#' st <- survtab_ag(fot ~ cancer + period, data = x, 
+#'                  surv.method = "lifetable", surv.type = "surv.obs")
+#' 
+#' plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
+#' lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
+#' 
+#' ## or
+#' plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
+#' @export
+#' @family survtab functions
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+lines.survtab <- function(x, y = NULL, subset = NULL, 
+                          conf.int = TRUE, col=NULL, lty=NULL, ...) {
+  Tstop <- NULL ## APPEASE R CMD CHECK
+  ## prep ----------------------------------------------------------------------
+  PF <- parent.frame(1L)
+  global_breaks <- attr(x, "survtab.meta")$surv.breaks
+  
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = x, subset, enclos = PF)
+  
+  
+  l <- prep_plot_survtab(x = x, y = y, subset = subset, 
+                         conf.int = conf.int, enclos = environment())
+  x <- l$x
+  y <- l$y
+  y.ci <- l$y.ci
+  y.lo <- l$y.lo
+  y.hi <- l$y.hi
+  strata <- l$strata ## character vector of var names
+  
+  
+  ## impute first values (time = 0, surv = 1 / cif = 0) ------------------------
+  
+  is_CIF <- if (substr(y, 1, 3) == "CIF") TRUE else FALSE
+  setkeyv(x, c(strata, "Tstop"))
+  first <- x[1, ]
+  if (length(strata)) first <- unique(x, by = strata)
+  first[, c(y) := ifelse(is_CIF, 0, 1)]
+  first$Tstop <- min(global_breaks)
+  
+  if (length(y.ci) > 0) first[, (y.ci) := get(y) ]
+  x <- rbindlist(list(first, x[, ]), use.names = TRUE)
+  setkeyv(x, c(strata, "Tstop"))
+  
+  ## plotting ------------------------------------------------------------------
+  
+  if (is.null(lty)) {
+    lty <- list(c(1,2,2))
+    if (!length(y.ci)) lty <- list(1)
+  }
+  
+  lines_by(x = "Tstop", y = c(y, y.ci), 
+           strata.vars = strata, 
+           data = x, col = col, lty = lty, ...)
+  
+  return(invisible(NULL))
+}
+
+
+
+lines_by <- function(x, y, strata.vars = NULL, data, col, lty, ...) {
+  ## INTENTION: plots lines separately by strata,
+  ## which may have different colours / linetypes.
+  ## @param x a variable to plot y by; a character string
+  ## @param y a character vector of variables to plot by x;
+  ## e.g. the estimate and confidence interval variables
+  ## @param strata.vars a character string vector; variables
+  ## to add lines by, which may have different colours etc for identification
+  ## @param data a data.frame where x, y, and strata.vars are found
+  ## @param col a vector of colors passed to lines(); if vector length 1,
+  ## used for each level of strata. If vector length > 1, 
+  ## has to match to total number of strata. If list, must match
+  ## to number of strata by length and contain elements of length
+  ## length(y).
+  ## @param see col; line type passed to lines().
+  ## @param ... other arguments passed on to lines().
+  
+  TF <- environment()
+  PF <- parent.frame(1L)
+  
+  stopifnot(is.data.frame(data))
+  stopifnot(is.character(x) && length(x) == 1L)
+  stopifnot(is.character(y) && length(y) > 0L)
+  stopifnot(is.character(strata.vars) || is.null(strata.vars))
+  all_names_present(data, c(x,y,strata.vars))
+  
+  d <- mget(c(strata.vars, y, x), envir = as.environment(data))
+  setDT(d)
+  setkeyv(d, c(strata.vars, x))
+  
+  ## create list of datas
+  l <- list(d)
+  inter <- 1L
+  if (length(strata.vars)) {
+    inter <- do.call(interaction, d[, strata.vars, with = FALSE])
+    l <- vector("list", uniqueN(inter))
+    l <- split(d, f = inter, drop = TRUE)
+  }
+  
+  l <- lapply(l, function(tab) {
+    setDT(tab)
+    setcolsnull(tab, keep = c(x, y))
+    tab
+  })
+  
+  
+  ## figure out colours and ltys
+  for (objname in c("col", "lty")) {
+    obj <- TF[[objname]]
+    
+    if (missing(obj) || !length(obj)) obj <- 1
+    if (!length(obj) %in% c(1, length(l))) {
+      stop("Argument ", objname, " is not of length 1 or ",
+           "of length equal to total number of strata (",
+           length(l), ").")
+    }
+    
+    ol <- unlist(lapply(obj, length))
+    if (length(y) > 1 && is.list(obj) && !all(ol %in% c(1, length(y)))) {
+      stop("Argument y is of length > 1, and you passed ",
+           objname, " as a list of values, but at least one element is not ",
+           "of length 1 or length(y).")
+    }
+    
+    ## NOTE: rep works for vector and list just the same
+    if (length(obj) == 1) obj <- rep(obj, length(l))
+    obj <- as.list(obj)
+    
+    assign(x = objname, value = obj)
+  }
+  
+  lapply(seq_along(l), function(i) {
+    
+    tab <- l[[i]]
+    cols <- col[[i]]
+    ltys <- lty[[i]]
+    
+    matlines(x = tab[[x]], y = tab[, y, with = FALSE], 
+             col = cols, lty = ltys, ...)
+    
+  })
+  
+  invisible(NULL)
+}
+
+
+
+
+#' @title Graphically Inspect Curves Used in Mean Survival Computation
+#' @description Plots the observed (with extrapolation) and expected survival
+#' curves for all strata in an object created by \code{\link{survmean}}
+#' @author Joonas Miettinen
+#' @param x a \code{survmean} object
+#' @param ... arguments passed (ultimately) to \code{matlines}; you
+#' may, therefore, supply e.g. \code{xlab} through this, though arguments
+#' such as \code{lty} and \code{col} will not work
+#' @details 
+#' 
+#' For examples see \code{\link{survmean}}. This function is intended only
+#' for graphically inspecting that the observed survival curves with extrapolation
+#' and the expected survival curves have been sensibly computed in \code{survmean}.
+#' 
+#' If you want finer control over the plotted curves, extract the curves from
+#' the \code{survmean} output using 
+#' 
+#' \code{attr(x, "curves")}
+#' 
+#' where \code{x} is a \code{survmean} object.
+#' @export
+#' @family survmean functions
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+plot.survmean <- function(x, ...) {
+  at <- attr(x, "survmean.meta")
+  curves <- at$curves
+  if (is.null(curves)) {
+    stop("no curves information in x; sometimes lost if x ",
+         "altered after using survmean")
+  }
+  
+  by.vars <- at$tprint
+  by.vars <- c(by.vars, at$tadjust)
+  by.vars <- intersect(by.vars, names(curves))
+  if (!length(by.vars)) by.vars <- NULL
+  
+  plot(curves$surv ~ curves$Tstop, type="n",
+       xlab = "Time from entry", ylab = "Survival")
+  lines.survmean(x, ...)
+  
+  subr <- at$breaks[[at$survScale]]
+  abline(v = max(subr), lty=2, col="grey")
+  
+  if (length(by.vars)) {
+    ## add legend denoting colors
+    Stratum <- curves[, unique(interaction(.SD)), .SDcols = eval(by.vars)]
+    legend(x = "topright", legend = Stratum, col = seq_along(Stratum), lty = 1)
+  }
+  return(invisible(NULL))
+}
+
+#' @title Graphically Inspect Curves Used in Mean Survival Computation
+#' @description Plots the observed (with extrapolation) and expected survival
+#' curves for all strata in an object created by \code{\link{survmean}}
+#' @author Joonas Miettinen
+#' @param x a \code{survmean} object
+#' @param ... arguments passed (ultimately) to \code{matlines}; you
+#' may, therefore, supply e.g. \code{lwd} through this, though arguments
+#' such as \code{lty} and \code{col} will not work
+#' @details 
+#' 
+#' This function is intended to be a workhorse for \code{\link{plot.survmean}}.
+#' If you want finer control over the plotted curves, extract the curves from
+#' the \code{survmean} output using 
+#' 
+#' \code{attr(x, "curves")}
+#' 
+#' where \code{x} is a \code{survmean} object.
+#' @export
+#' @family survmean functions
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+lines.survmean <- function(x, ...) {
+  at <- copy(attr(x, "survmean.meta"))
+  curves <- at$curves
+  if (is.null(curves)) stop("no curves information in x; usually lost if x altered after using survmean")
+  
+  by.vars <- at$tprint
+  by.vars <- c(by.vars, at$tadjust)
+  by.vars <- c("survmean_type", by.vars)
+  by.vars <- intersect(by.vars, names(curves))
+  if (!length(by.vars)) by.vars <- NULL
+  
+  curves <- data.table(curves)
+  setkeyv(curves, c(by.vars, "Tstop"))
+  
+  type_levs <- length(levels(interaction(curves[, c(by.vars), with=FALSE])))/2L
+  other_levs <- 1L
+  if (length(by.vars) > 1) {
+    other_levs <- length(levels(interaction(curves[, setdiff(by.vars, "survmean_type"), with=FALSE])))
+  }
+  
+  curves <- cast_simple(curves, columns = by.vars, rows = "Tstop", values = "surv")
+  matlines(x=curves$Tstop, y=curves[, setdiff(names(curves), "Tstop"), with=FALSE],  
+           lty = rep(1:2, each=type_levs), col = 1:other_levs, ...)
+  return(invisible(NULL))
+}
+
+
+
+
+
+
+
+#' @export
+getCall.survtab <- function(x, ...) {
+  attributes(x)$survtab.meta$call
+}
+
+
+#' @export
+formula.survtab <- function(x, ...) {
+  attr(x, "survtab.meta")$arguments$formula
+}
+
+
+
+
+
+#' @export
+getCall.survmean <- function(x, ...) {
+  attributes(x)$survmean.meta$call
+}
+
+
+#' @export
+formula.survmean <- function(x, ...) {
+  attr(x, "survmean.meta")$formula
+}
+
+
+
diff --git a/R/Surv.R b/R/Surv.R
index 52f26ce..c83d3f2 100644
--- a/R/Surv.R
+++ b/R/Surv.R
@@ -1,70 +1,70 @@
-
-
-
-
-
-#' @md
-#' @title Survival Objects
-#' @description
-#' Wrapper for [survival::Surv].
-#' @param time see  [survival::Surv]
-#' @param time2 see  [survival::Surv]
-#' @param event see  [survival::Surv]
-#' @param type see  [survival::Surv]
-#' @param origin see [survival::Surv]
-#' @section Surv in survival vs. in popEpi:
-#' `popEpi::Surv` is a wrapper for [survival::Surv].
-#' Therefore you don't need to to do `library("survival")` when using `Surv` 
-#' with e.g.
-#' \code{\link{survtab}}. Remember that if you do `library("survival")` after
-#' `library("popEpi")`, the `Surv` from \pkg{survival} is used instead of
-#' from \pkg{popEpi} (`R` throws a warning about this) when an expression
-#' such as `Surv(my_times, my_events)` is evaluated. You can avoid such
-#' conflicts by writing e.g. `popEpi::Surv(my_times, my_events)` instead.
-#' However, `popEpi::Surv` is designed in such a way that this should not
-#' become a problem and you should be able to use the two interchangeably.
-#' @export
-#' @family main functions
-#' @family survtab functions
-#' @family survmean functions
-#' @importFrom survival Surv
-#' @return
-#' See `[survival::Surv]`.
-#' 
-Surv <- function(
-  time, 
-  time2, 
-  event, 
-  type = c("right", "left", "interval", "counting", "interval2", "mstate"), 
-  origin = 0
-) {
-  
-  pf <- parent.frame(1)
-  arg_nms <- names(formals(Surv))
-  test_env <- environment()
-  is_missing <- vapply(arg_nms, function(arg_nm) {
-    eval(substitute(
-      missing(OBJ),
-      list(OBJ = parse(text = arg_nm)[[1]])
-    ), envir = test_env)
-  }, logical(1))
-  pass_arg_nms <- arg_nms[!is_missing]
-  pass_arg_nms <- intersect(pass_arg_nms, names(formals(survival::Surv)))
-  
-  pass_arg_list <- mget(pass_arg_nms)
-  eval_env <- as.environment(pass_arg_list)
-  parent.env(eval_env) <- pf
-  expr_args <- lapply(pass_arg_nms, function(stri) {
-    parse(text = stri)[[1]]
-  })
-  names(expr_args) <- pass_arg_nms
-  surv_expr <- as.call(c(
-    quote(survival::Surv),
-    expr_args
-  ))
-  eval(surv_expr, envir = eval_env)
-}
-
-
-
-
+
+
+
+
+
+#' @md
+#' @title Survival Objects
+#' @description
+#' Wrapper for [survival::Surv].
+#' @param time see  [survival::Surv]
+#' @param time2 see  [survival::Surv]
+#' @param event see  [survival::Surv]
+#' @param type see  [survival::Surv]
+#' @param origin see [survival::Surv]
+#' @section Surv in survival vs. in popEpi:
+#' `popEpi::Surv` is a wrapper for [survival::Surv].
+#' Therefore you don't need to to do `library("survival")` when using `Surv` 
+#' with e.g.
+#' \code{\link{survtab}}. Remember that if you do `library("survival")` after
+#' `library("popEpi")`, the `Surv` from \pkg{survival} is used instead of
+#' from \pkg{popEpi} (`R` throws a warning about this) when an expression
+#' such as `Surv(my_times, my_events)` is evaluated. You can avoid such
+#' conflicts by writing e.g. `popEpi::Surv(my_times, my_events)` instead.
+#' However, `popEpi::Surv` is designed in such a way that this should not
+#' become a problem and you should be able to use the two interchangeably.
+#' @export
+#' @family main functions
+#' @family survtab functions
+#' @family survmean functions
+#' @importFrom survival Surv
+#' @return
+#' See `[survival::Surv]`.
+#' 
+Surv <- function(
+  time, 
+  time2, 
+  event, 
+  type = c("right", "left", "interval", "counting", "interval2", "mstate"), 
+  origin = 0
+) {
+  
+  pf <- parent.frame(1)
+  arg_nms <- names(formals(Surv))
+  test_env <- environment()
+  is_missing <- vapply(arg_nms, function(arg_nm) {
+    eval(substitute(
+      missing(OBJ),
+      list(OBJ = parse(text = arg_nm)[[1]])
+    ), envir = test_env)
+  }, logical(1))
+  pass_arg_nms <- arg_nms[!is_missing]
+  pass_arg_nms <- intersect(pass_arg_nms, names(formals(survival::Surv)))
+  
+  pass_arg_list <- mget(pass_arg_nms)
+  eval_env <- as.environment(pass_arg_list)
+  parent.env(eval_env) <- pf
+  expr_args <- lapply(pass_arg_nms, function(stri) {
+    parse(text = stri)[[1]]
+  })
+  names(expr_args) <- pass_arg_nms
+  surv_expr <- as.call(c(
+    quote(survival::Surv),
+    expr_args
+  ))
+  eval(surv_expr, envir = eval_env)
+}
+
+
+
+
diff --git a/R/aggregating.R b/R/aggregating.R
index 237db86..15cce53 100644
--- a/R/aggregating.R
+++ b/R/aggregating.R
@@ -1,622 +1,622 @@
-
-#' @title Set \code{aggre} attributes to an object by modifying in place
-#' @author Joonas Miettinen
-#' @description Coerces an R object to an \code{aggre} object, identifying
-#' the object as one containing aggregated counts, person-years and other
-#' information. \code{setaggre} modifies in place without taking any copies.
-#' Retains all other attributes.
-#' @param x a \code{data.frame} or \code{data.table}
-#' @param values a character string vector; the names of value variables
-#' @param by a character string vector; the names of variables by which 
-#' \code{values} have been tabulated
-#' @param breaks a list of breaks, where each element is a breaks vector
-#' as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
-#' fully named, with the names corresponding to time scales at the aggregate
-#' level in your data. Every unique value in a time scale variable in data must
-#' also exist in the corresponding vector in the breaks list.
-#' @details 
-#' 
-#' \code{setaggre} sets \code{x} to the \code{aggre} class in place 
-#' without taking a copy as e.g. \code{as.data.frame.XXX} functions do; see e.g. 
-#' \code{\link[data.table]{setDT}}.
-#' 
-#' @family aggregation functions
-#' @return
-#' Returns `x` invisibly after setting attributes to it without taking a copy.
-#' This function is called for its side effects.
-#' @export setaggre
-#' @examples 
-#' df <- data.frame(sex = rep(c("male", "female"), each = 5), 
-#'                  obs = rpois(10, rep(7,5, each=5)), 
-#'                  pyrs = rpois(10, lambda = 10000))
-#' ## without any breaks
-#' setaggre(df, values = c("obs", "pyrs"), by = "sex")
-#' df <- data.frame(df)
-#' df$FUT <- 0:4
-#' ## with breaks list
-#' setaggre(df, values = c("obs", "pyrs"), by = "sex", breaks = list(FUT = 0:5))
-setaggre <- function(x, values = NULL, by = NULL, breaks = NULL) {
-  ## input: aggregated data in data.frame or data.table format
-  ## intention: any user can define their data as an aggregated data set
-  ## which will be usable by survtab / sir / other
-  ## output: no need to do x <- setaggre(x); instead modifies attributes in place;
-  ## sets "aggre.meta" attribute, a list of names of various variables.
-  ## survtab for aggregated data will need this attribute to work.
-  all_names_present(x, c(values, by))
-  
-  if (!length(by) && length(values)) by <- setdiff(names(x), values)
-  if (length(by) && !length(values)) values <- setdiff(names(x), by)
-  
-  if (!inherits(x, "aggre")) {
-    cl <- class(x)
-    wh <- which(cl %in% c("data.table", "data.frame"))
-    wh <- min(wh)
-    
-    ## yes, from zero: in case only one class
-    cl <- c(cl[0:(wh-1)], "aggre", cl[wh:length(cl)])
-    setattr(x, "class", cl)
-  }
-  
-  
-  setattr(x, "aggre.meta", list(values = values, by = by, breaks = breaks))
-  setattr(x, "breaks", breaks)
-  invisible(x)
-}
-
-#' @title Coercion to Class \code{aggre}
-#' @author Joonas Miettinen
-#' @description Coerces an R object to an \code{aggre} object, identifying
-#' the object as one containing aggregated counts, person-years and other
-#' information. 
-#' @inheritParams setaggre
-#' @param ... arguments passed to or from methods
-#' @family aggregation functions
-#' 
-#' 
-#' @examples 
-#' library("data.table")
-#' df <- data.frame(sex = rep(c("male", "female"), each = 5), 
-#'                  obs = rpois(10, rep(7,5, each=5)), 
-#'                  pyrs = rpois(10, lambda = 10000))
-#' dt <- as.data.table(df)
-#' 
-#' df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex")
-#' dt <- as.aggre(dt, values = c("pyrs", "obs"), by = "sex")
-#' 
-#' class(df)
-#' class(dt)
-#' 
-#' BL <- list(fot = 0:5)
-#' df <- data.frame(df)
-#' df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex", breaks = BL)
-#' 
-#' @return
-#' Returns a copy of `x` with attributes set to those of an object of class
-#' `"aggre"`.
-#' @export
-as.aggre <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
-  UseMethod("as.aggre", x)
-}
-
-#' @describeIn as.aggre Coerces a \code{data.frame} to an \code{aggre} object
-#' @export
-as.aggre.data.frame <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
-  x <- copy(x)
-  setaggre(x, values = values, by = by, breaks = breaks, ...)
-  setattr(x, "class", c("aggre", "data.frame"))
-  x[]
-}
-
-#' @describeIn as.aggre Coerces a \code{data.table} to an \code{aggre} object
-#' @export
-as.aggre.data.table <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
-  x <- copy(x)
-  setaggre(x, values = values, by = by, breaks = breaks, ...)
-  setattr(x, "class", c("aggre", "data.table", "data.frame"))
-  x[]
-}
-
-#' @describeIn as.aggre Default method for \code{as.aggre} (stops computations
-#' if no class-specific method found)
-#' @export
-as.aggre.default <- function(x, ...) {
-  stop(gettextf("cannot coerce class \"%s\" to 'aggre'", deparse(class(x))), 
-       domain = NA)
-}
-
-
-
-#' @title Aggregation of split \code{Lexis} data
-#' @author Joonas Miettinen
-#' @description Aggregates a split \code{Lexis} object by given variables 
-#' and / or expressions into a long-format table of person-years and 
-#' transitions / end-points. Automatic aggregation over time scales
-#' by which data has been split if the respective time scales are mentioned
-#' in the aggregation argument to e.g. intervals of calendar time, follow-up time
-#' and/or age.
-#' @param lex a \code{Lexis} object split with e.g. 
-#' \code{\link[Epi]{splitLexis}} or \code{\link{splitMulti}}
-#' @param by variables to tabulate (aggregate) by.
-#' \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{by = c("V1", "V2")}. See Details and Examples.
-#' @param type determines output levels to which data is aggregated varying
-#' from returning only rows with \code{pyrs > 0} (\code{"unique"}) to
-#' returning all possible combinations of variables given in \code{aggre} even
-#' if those combinations are not represented in data (\code{"full"}); 
-#' see Details
-#' @param sum.values optional: additional variables to sum by argument
-#'  \code{by}. \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{sum.values = c("V1", "V2")}
-#' @param subset a logical condition to subset by before computations;
-#' e.g. \code{subset = area \%in\% c("A", "B")}
-#' @param verbose \code{logical}; if \code{TRUE}, the function returns timings
-#' and some information useful for debugging along the aggregation process
-#' @details 
-#' 
-#' \strong{Basics}
-#' 
-#' \code{aggre} is intended for aggregation of split \code{Lexis} data only.
-#' See \code{\link[Epi]{Lexis}} for forming \code{Lexis} objects by hand
-#' and e.g. \code{\link[Epi]{splitLexis}}, \code{\link{splitLexisDT}}, and
-#' \code{\link{splitMulti}} for splitting the data. \code{\link{lexpand}}
-#' may be used for simple data sets to do both steps as well as aggregation
-#' in the same function call.
-#' 
-#' Here aggregation refers to computing person-years and the appropriate events
-#' (state transitions and end points in status) for the subjects in the data.
-#' Hence, it computes e.g. deaths (end-point and state transition) and 
-#' censorings (end-point) as well as events in a multi-state setting
-#' (state transitions).
-#' 
-#' The result is a long-format \code{data.frame} or \code{data.table}
-#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi})
-#' with the columns \code{pyrs} and the appropriate transitions named as
-#' \code{fromXtoY}, e.g. \code{from0to0} and \code{from0to1} depending
-#' on the values of \code{lex.Cst} and \code{lex.Xst}.
-#' 
-#' 
-#' \strong{The by argument}
-#' 
-#' The \code{by} argument determines the length of the table, i.e.
-#' the combinations of variables to which data is aggregated.  
-#' \code{by} is relatively flexible, as it can be supplied as
-#' 
-#' \itemize{
-#'  \item{a character string vector, e.g. \code{c("sex", "area")}, 
-#'  naming variables existing in \code{lex}}
-#'  \item{an expression, e.g. \code{factor(sex, 0:1, c("m", "f"))} 
-#'  using any variable found in \code{lex}}
-#'  \item{a list (fully or partially named) of expressions, e.g. 
-#'  \code{list(gender = factor(sex, 0:1, c("m", "f"), area)}}
-#' }
-#' 
-#' Note that expressions effectively allow a variable to be supplied simply as
-#' e.g. \code{by = sex} (as a symbol/name in R lingo).
-#' 
-#' The data is then aggregated to the levels of the given variables 
-#' or expression(s). Variables defined to be time scales in the supplied 
-#' \code{Lexis} are processed in a special way: If any are mentioned in the
-#' \code{by} argument, intervals of them are formed based on the breaks
-#' used to split the data: e.g. if \code{age} was split using the breaks 
-#' \code{c(0, 50, Inf)}, mentioning \code{age} in \code{by} leads to
-#' creating the \code{age} intervals \code{[0, 50)} and \code{[50, Inf)}
-#' and aggregating to them. The intervals are identified in the output
-#' as the lower bounds of the appropriate intervals.
-#' 
-#' The order of multiple time scales mentioned in \code{by} matters,
-#' as the last mentioned time scale is assumed to be a survival time scale
-#' for when computing event counts. E.g. when the data is split by the breaks
-#' \code{list(FUT = 0:5, CAL = c(2008,2010))}, time lines cut short at
-#' \code{CAL = 2010} are considered to be censored, but time lines cut short at
-#' \code{FUT = 5} are not. See Return.
-#' 
-#' \strong{Aggregation types (styles)}
-#' 
-#' It is almost always enough to aggregate the data to variable levels
-#' that are actually represented in the data 
-#' (default \code{aggre = "unique"}; alias \code{"non-empty"}). 
-#' For certain uses it may be useful
-#' to have also "empty" levels represented (resulting in some rows in output
-#' with zero person-years and events); in these cases supplying
-#' \code{aggre = "full"} (alias \code{"cartesian"}) causes \code{aggre}
-#' to determine the Cartesian product of all the levels of the supplied 
-#' \code{by} variables or expressions and aggregate to them. As an example
-#' of a Cartesian product, try
-#' 
-#' \code{merge(1:2, 1:5)}.
-#' 
-#' @return 
-#' A long \code{data.frame} or \code{data.table} of aggregated person-years 
-#' (\code{pyrs}), numbers of subjects at risk (\code{at.risk}), and events
-#' formatted \code{fromXtoY}, where \code{X} and \code{X} are states 
-#' transitioning from and to or states at the end of each \code{lex.id}'s 
-#' follow-up (implying \code{X} = \code{Y}). Subjects at risk are computed 
-#' in the beginning of an interval defined by any Lexis time scales and 
-#' mentioned in \code{by}, but events occur at any point within an interval.
-#' 
-#' When the data has been split along multiple time scales, the last
-#' time scale mentioned in \code{by} is considered to be the survival time 
-#' scale with regard to computing events. Time lines cut short by the
-#' extrema of non-survival-time-scales are considered to be censored
-#' ("transitions" from the current state to the current state).
-#' 
-#' @seealso \code{\link{aggregate}} for a similar base R solution,
-#' and \code{\link{ltable}} for a \code{data.table} based aggregator. Neither
-#' are directly applicable to split \code{Lexis} data.
-#' 
-#' @family aggregation functions
-#' 
-#' 
-#' 
-#' @examples 
-#' 
-#' ## form a Lexis object
-#' library(Epi)
-#' data(sibr)
-#' x <- sibr[1:10,]
-#' x[1:5,]$sex <- 0 ## pretend some are male
-#' x <- Lexis(data = x,
-#'            entry = list(AGE = dg_age, CAL = get.yrs(dg_date)),
-#'            exit = list(CAL = get.yrs(ex_date)),
-#'            entry.status=0, exit.status = status)
-#' x <- splitMulti(x, breaks = list(CAL = seq(1993, 2013, 5), 
-#'                                  AGE = seq(0, 100, 50)))
-#' 
-#' ## these produce the same results (with differing ways of determining aggre)
-#' a1 <- aggre(x, by = list(gender = factor(sex, 0:1, c("m", "f")), 
-#'              agegroup = AGE, period = CAL))
-#' 
-#' a2 <- aggre(x, by = c("sex", "AGE", "CAL"))
-#' 
-#' a3 <- aggre(x, by = list(sex, agegroup = AGE, CAL))
-#' 
-#' ## returning also empty levels
-#' a4 <- aggre(x, by = c("sex", "AGE", "CAL"), type = "full")
-#' 
-#' ## computing also expected numbers of cases
-#' x <- lexpand(sibr[1:10,], birth = bi_date, entry = dg_date,
-#'              exit = ex_date, status = status %in% 1:2, 
-#'              pophaz = popmort, fot = 0:5, age = c(0, 50, 100))
-#' x$d.exp <- with(x, lex.dur*pop.haz)
-#' ## these produce the same result
-#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = list(d.exp))
-#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = "d.exp")
-#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = d.exp)
-#' ## same result here with custom name
-#' a5 <- aggre(x, by = c("sex", "age", "fot"), 
-#'              sum.values = list(expCases = d.exp))
-#'              
-#' ## computing pohar-perme weighted figures
-#' x$d.exp.pp <- with(x, lex.dur*pop.haz*pp)
-#' a6 <- aggre(x, by = c("sex", "age", "fot"), 
-#'              sum.values = c("d.exp", "d.exp.pp"))
-#' ## or equivalently e.g. sum.values = list(expCases = d.exp, expCases.p = d.exp.pp).
-#' @export
-aggre <- function(lex, by = NULL, type = c("unique", "full"), sum.values = NULL, subset = NULL, verbose = FALSE) {
-  
-  allTime <- proc.time()
-  
-  lex.Cst <- lex.Xst <- lex.id <- at.risk <- NULL ## APPEASE R CMD CHECK
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  
-  type <- match.arg(type[1], c("non-empty", "unique", "full", "cartesian"))
-  if (type == "cartesian") type <- "full"
-  if (type == "non-empty") type <- "unique"
-  
-  if (verbose) cat("Aggregation type: '", type, "' \n", sep = "")
-  
-  checkLexisData(lex)
-  
-  breaks <- copy(attr(lex, "breaks"))
-  checkBreaksList(lex, breaks)
-  
-  allScales <- copy(attr(lex, "time.scales"))
-  if (length(allScales) == 0 ) {
-    stop("could not determine names of time scales; ",
-         "is the data a Lexis object?")
-  }
-  
-  ## subset --------------------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(lex, subset)
-  
-  ## check sum.values ----------------------------------------------------------
-  sumSub <- substitute(sum.values)
-  sum.values <- evalPopArg(lex[1:min(nrow(lex), 20L), ], arg = sumSub, 
-                     enclos = PF, recursive = TRUE, DT = TRUE)
-  sumType <- attr(sum.values, "arg.type")
-  sumVars <- attr(sum.values, "all.vars")
-  sumSub <- attr(sum.values, "quoted.arg")
-  if (is.null(sum.values)) {
-    sumType <- "NULL"
-    sumVars <- NULL
-    sumSub <- quote(list())
-  }
-  badSum <- names(sum.values)[!sapply(sum.values, is.numeric)]
-  if (length(badSum) > 0L) {
-    badSum <- paste0("'", badSum, "'", collapse = ", ")
-    stop("Following variables resulting from evaluating supplied sum.values ",
-         "argument are not numeric and cannot be summed: ", badSum, 
-         ". Evaluated sum.values: ", deparse(sumSub))
-  }
-  
-  
-  ## by argument type -------------------------------------------------------
-  ## NOTE: need to eval by AFTER cutting time scales!
-  
-  ags <- substitute(by)
-  if (verbose) cat("Used by argument:", paste0(deparse(ags)),"\n")
-  
-  ## NOTE: with recursive = TRUE, evalPopArg digs deep enough to find
-  ## the actual expression (substituted only once) and returns that and other
-  ## things in attributes. Useful if arg substituted multiple times.
-  by <- evalPopArg(data = lex[1:min(nrow(lex), 20),], 
-                      arg = ags, DT = TRUE, enclos = PF, recursive = TRUE)
-  ags <- attr(by, "quoted.arg") 
-  av <- attr(by, "all.vars")
-  argType <- attr(by, "arg.type")
-  
-  if (is.null(by)) {
-    ags <- substitute(list())
-    av <- NULL
-    argType <- "NULL"
-    type <- "unique"
-  }
-  if (verbose) cat("Type of by argument:", argType, "\n")
-  
-  ## take copy of lex ----------------------------------------------------------
-  ## if lex is a data.table, this function gets really complicated.
-  ## if copy is taken only of necessary vars, it should be fine.
-  keepVars <- unique(c("lex.id", allScales, "lex.dur", 
-                       "lex.Cst", "lex.Xst", av, sumVars))
-  lex.orig <- lex
-  lex <- subsetDTorDF(lex, subset = subset, select = keepVars)
-  lex <- data.table(lex)
-  forceLexisDT(lex, breaks = breaks, allScales = allScales, key = FALSE)
-  
-  
-  ## ensure no observations outside breaks limits are left in
-  lex <- intelliDrop(lex, breaks = breaks)
-  
-  setkeyv(lex, c("lex.id", allScales[1]))
-  setcolsnull(lex, delete = setdiff(allScales, names(breaks)))
-  
-  
-  ## cut time scales for aggregating if needed ---------------------------------
-  aggScales <- intersect(av, allScales)
-  if (any(!aggScales %in% names(breaks))) {
-    aggScales <- paste0("'", setdiff(aggScales, names(breaks)), "'", collapse = ", ")
-    stop("Requested aggregating by time scale(s) by which data ",
-         "has not been split: ", aggScales)
-  }
-  
-  ## before cutting, find out which rows count towards "at.risk" figure:
-  ## of all scales in aggScales, the last one (or the only one) is assumed
-  ## to be the survival time scale.
-  tmpAtRisk <- makeTempVarName(lex, pre = "at.risk_")
-  set(lex, j = tmpAtRisk, value = TRUE)
-  survScale <- NULL
-  
-  
-  if (length(aggScales) > 0) {
-    cutTime <- proc.time()
-    ## "at.risk" counts subjects at risk in the beginning of the survival
-    ## time scale interval.
-    survScale <- aggScales[length(aggScales)]
-    lex[, c(tmpAtRisk) := lex[[survScale]] %in% breaks[[survScale]] ]
-    catAggScales <- paste0("'", aggScales, "'", collapse = ", ")
-    if (verbose) {
-      cat("Following time scales mentioned in by argument and will be",
-          "categorized into intervals (defined by breaks in object",
-          "attributes) for aggregation:", catAggScales, "\n")
-    }
-    
-    ## NEW METHOD: use a copy of lex and just modify in place.
-    
-    for (sc in aggScales) {
-      set(lex, j = sc, value = cutLow(lex[[sc]], breaks = breaks[[sc]]))
-    }
-    
-    if (verbose) cat("Time taken by cut()'ting time scales: ", timetaken(cutTime), "\n")
-  }
-  
-  othVars <- setdiff(av, aggScales)
-  if (verbose && length(othVars) > 0) {
-    catOthVars <- paste0("'", othVars, "'", collapse = ", ")
-    cat("Detected the following non-time-scale variables to be utilized in aggregating:", catOthVars, "\n")
-  }
-  
-  ## eval by -------------------------------------------------------------------
-  ## NOTE: needed to eval by AFTER cutting time scales!
-  by <- evalPopArg(data = lex, arg = ags, DT = TRUE, enclos = PF, recursive = TRUE)
-  byNames <- names(by)
-  
-  ## computing pyrs ------------------------------------------------------------
-  ## final step in determining at.risk:
-  ## a lex.id is at.risk only once per by-level
-  pyrsTime <- proc.time()
-  vdt <- data.table(pyrs = lex$lex.dur, at.risk = lex[[tmpAtRisk]], 
-                    lex.id = lex$lex.id)
-  pyrs <- vdt[, .(pyrs = sum(pyrs), 
-                  at.risk = sum(!duplicated(lex.id) & at.risk)), 
-              keyby = by]
-  setDT(pyrs)
-  
-  rm(vdt)
-  sumNames <- NULL
-  if (sumType != "NULL") {
-    if (sumType == "character") {
-      sumNames <- evalPopArg(lex, sumSub, n = 1L, DT = FALSE, recursive = TRUE, enclos = PF)
-      sum.values <- lex[, lapply(.SD, sum), keyby = by, .SDcols = c(sumNames)]
-    } else {
-      sum.values <- evalPopArg(lex, sumSub, n = 1L, enclos = PF)
-      sumNames <- names(sum.values)
-      sumTmpNames <- makeTempVarName(lex, pre = sumNames)
-      set(lex, j = sumTmpNames, value = sum.values)
-      sum.values <- lex[, lapply(.SD, sum), keyby = by, .SDcols = sumTmpNames]
-      setnames(sum.values, sumTmpNames, sumNames)
-      setcolsnull(lex, sumTmpNames)
-    }
-    
-    setDT(sum.values)
-    pyrs <- merge(pyrs, sum.values, all = TRUE)
-    rm(sum.values)
-  }
-  
-  
-  if (verbose) cat("Time taken by aggregating pyrs: ", timetaken(pyrsTime), "\n")
-  
-  valVars <- setdiff(names(pyrs), byNames) ## includes pyrs and anything created by sum
-  
-  pyrs[is.na(pyrs), pyrs := 0]
-  pyrs <- pyrs[pyrs > 0]
-  
-  aggPyrs <- pyrs[, sum(pyrs)] 
-  lexPyrs <- sum(lex.orig$lex.dur[subset])
-  pyrsDiff <- aggPyrs - lexPyrs
-  if (!isTRUE(all.equal(aggPyrs, lexPyrs, scale = NULL))) {
-    warning("Found discrepancy of ", abs(round(pyrsDiff, 4)), " ",
-            "in total aggregated pyrs compared to ",
-            "sum(lex$lex.dur); compare results by hand and make sure ",
-            "settings are right \n")
-  }
-  rm(subset, aggPyrs, lexPyrs)
-  
-  ## cartesian output ----------------------------------------------------------
-  if (type == "full") {
-    carTime <- proc.time()
-    
-    varsUsingScales <- NULL
-    
-    ## which variables used one time scale? and which one?
-    ## will only be used in cartesian stuff.
-    if (argType == "character") {
-      varsUsingScales <- intersect(names(by), aggScales)
-      whScaleUsed <- varsUsingScales
-    } else if (argType != "NULL") {
-      ## note: ags a substitute()'d list at this point always if not char
-      whScaleUsed <- lapply(ags[-1], function(x) intersect(all.vars(x), aggScales))
-      ## only one time scale should be used in a variable!
-      oneScaleTest <- any(sapply(whScaleUsed, function(x) length(x) > 1L))
-      if (oneScaleTest) stop("Only one Lexis time scale can be used in any one variable in by argument!")
-      varsUsingScales <- byNames[sapply(whScaleUsed, function (x) length(x) == 1L)]
-      whScaleUsed <- unlist(whScaleUsed)
-    }
-    
-    ceejay <- lapply(by, function(x) if (is.factor(x)) levels(x) else sort(unique(x)))
-    if (length(aggScales) > 0) {
-      ## which variables in ceejay used the Lexis time scales from lex?
-      ceejay[varsUsingScales] <- lapply(breaks[whScaleUsed], function(x) x[-length(x)])
-    }
-    
-    ceejay <- do.call(CJ, ceejay)
-    setkeyv(ceejay, byNames)
-    setkeyv(pyrs, byNames)
-    
-    pyrs <- pyrs[ceejay]
-    rm(ceejay)
-    
-    if (verbose) cat("Time taken by making aggregated data large in the cartesian product sense: ", timetaken(carTime), "\n")
-  }
-  
-  
-  ## computing events ----------------------------------------------------------
-  
-  transTime <- proc.time()
-  
-  if (is.null(by) || (is.data.table(by) && nrow(by) == 0L)) {
-    
-    by <- quote(list(lex.Cst, lex.Xst))
-    
-  } else {
-    for (var in c("lex.Cst", "lex.Xst")) {
-      set(by, j = var, value = lex[[var]])
-    }
-  }
-  
-  
-  ## NOTE: this will ensure correct detection of censorings:
-  ## observations cut short by e.g. period window's edge
-  ## will be considered a censoring if the breaks along that time scale
-  ## are not passed to detectEvents (assuming the survival time scale is
-  ## used in by). If no time scale mentioned in by, then all endings
-  ## of observations are either censorings or events.
-  detBr <- breaks[survScale]
-  if (!length(survScale)) detBr <- NULL
-  hasEvent <- detectEvents(lex, breaks = detBr, by = "lex.id") %in% 1:2
-  ## is language if user supplied by = NULL 
-  if (!is.language(by)) by <- by[hasEvent]
-  
-  trans <- lex[hasEvent, list(obs = .N), keyby = by]
-  
-  rm(by, lex)
-  
-  
-  if (verbose) cat("Time taken by aggregating events: ", timetaken(transTime), "\n")
-  
-  ## casting & merging ---------------------------------------------------------
-  
-  mergeTime <- proc.time()
-  setDT(trans)
-  setDT(pyrs)
-  
-  ## tmpTr to be used in casting
-  tmpTr <- makeTempVarName(trans, pre = "trans_")
-  trans[, c(tmpTr) := paste0("from", lex.Cst, "to", lex.Xst)]
-  transitions <- sort(unique(trans[[tmpTr]]))
-  trans[, c("lex.Cst", "lex.Xst") := NULL]
-  
-  ## note: need tmpDum if by = NULL for correct casting & merging
-  tmpDum <- makeTempVarName(trans)
-  byNames <- c(byNames, tmpDum)
-  byNames <- setdiff(byNames, c("lex.Cst", "lex.Xst"))
-  trans[, c(tmpDum) := 1L]
-  pyrs[, c(tmpDum) := 1L]
-  
-  valVars <- unique(c(valVars, transitions))
-  
-  trans <- cast_simple(trans, rows = byNames, columns = tmpTr, values = "obs")
-  
-  setkeyv(trans, NULL); setkeyv(pyrs, NULL) ## dcast.data.table seems to keep key but row order may be funky; this avoids a warning
-  setkeyv(trans, byNames); setkeyv(pyrs, byNames)
-  trans <- trans[pyrs]; rm(pyrs)
-  
-  trans[, c(tmpDum) := NULL]
-  byNames <- setdiff(byNames, tmpDum)
-  setcolorder(trans, c(byNames, valVars))
-  
-  if (verbose) cat("Time taken by merging pyrs & transitions: ", timetaken(mergeTime), "\n")
-  
-  if (length(valVars) > 0L) {
-    trans[, c(valVars) := lapply(.SD, function(x) {
-      x[is.na(x)] <- 0
-      x
-    }), .SDcols = c(valVars)]
-  }
-  
-  
-  ## final touch ---------------------------------------------------------------
-  trans <- data.table(trans)
-  setaggre(trans, values = c("pyrs", "at.risk", transitions, sumNames), 
-           by = byNames, breaks = breaks)
-  if (!return_DT()) setDFpe(trans)
-  if (verbose) cat("Time taken by aggre(): ", timetaken(allTime), "\n")
-  
-  
-  
-  trans[]
-}
-
-
-
-
-
-
-
-
-
-
-
-
+
+#' @title Set \code{aggre} attributes to an object by modifying in place
+#' @author Joonas Miettinen
+#' @description Coerces an R object to an \code{aggre} object, identifying
+#' the object as one containing aggregated counts, person-years and other
+#' information. \code{setaggre} modifies in place without taking any copies.
+#' Retains all other attributes.
+#' @param x a \code{data.frame} or \code{data.table}
+#' @param values a character string vector; the names of value variables
+#' @param by a character string vector; the names of variables by which 
+#' \code{values} have been tabulated
+#' @param breaks a list of breaks, where each element is a breaks vector
+#' as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
+#' fully named, with the names corresponding to time scales at the aggregate
+#' level in your data. Every unique value in a time scale variable in data must
+#' also exist in the corresponding vector in the breaks list.
+#' @details 
+#' 
+#' \code{setaggre} sets \code{x} to the \code{aggre} class in place 
+#' without taking a copy as e.g. \code{as.data.frame.XXX} functions do; see e.g. 
+#' \code{\link[data.table]{setDT}}.
+#' 
+#' @family aggregation functions
+#' @return
+#' Returns `x` invisibly after setting attributes to it without taking a copy.
+#' This function is called for its side effects.
+#' @export setaggre
+#' @examples 
+#' df <- data.frame(sex = rep(c("male", "female"), each = 5), 
+#'                  obs = rpois(10, rep(7,5, each=5)), 
+#'                  pyrs = rpois(10, lambda = 10000))
+#' ## without any breaks
+#' setaggre(df, values = c("obs", "pyrs"), by = "sex")
+#' df <- data.frame(df)
+#' df$FUT <- 0:4
+#' ## with breaks list
+#' setaggre(df, values = c("obs", "pyrs"), by = "sex", breaks = list(FUT = 0:5))
+setaggre <- function(x, values = NULL, by = NULL, breaks = NULL) {
+  ## input: aggregated data in data.frame or data.table format
+  ## intention: any user can define their data as an aggregated data set
+  ## which will be usable by survtab / sir / other
+  ## output: no need to do x <- setaggre(x); instead modifies attributes in place;
+  ## sets "aggre.meta" attribute, a list of names of various variables.
+  ## survtab for aggregated data will need this attribute to work.
+  all_names_present(x, c(values, by))
+  
+  if (!length(by) && length(values)) by <- setdiff(names(x), values)
+  if (length(by) && !length(values)) values <- setdiff(names(x), by)
+  
+  if (!inherits(x, "aggre")) {
+    cl <- class(x)
+    wh <- which(cl %in% c("data.table", "data.frame"))
+    wh <- min(wh)
+    
+    ## yes, from zero: in case only one class
+    cl <- c(cl[0:(wh-1)], "aggre", cl[wh:length(cl)])
+    setattr(x, "class", cl)
+  }
+  
+  
+  setattr(x, "aggre.meta", list(values = values, by = by, breaks = breaks))
+  setattr(x, "breaks", breaks)
+  invisible(x)
+}
+
+#' @title Coercion to Class \code{aggre}
+#' @author Joonas Miettinen
+#' @description Coerces an R object to an \code{aggre} object, identifying
+#' the object as one containing aggregated counts, person-years and other
+#' information. 
+#' @inheritParams setaggre
+#' @param ... arguments passed to or from methods
+#' @family aggregation functions
+#' 
+#' 
+#' @examples 
+#' library("data.table")
+#' df <- data.frame(sex = rep(c("male", "female"), each = 5), 
+#'                  obs = rpois(10, rep(7,5, each=5)), 
+#'                  pyrs = rpois(10, lambda = 10000))
+#' dt <- as.data.table(df)
+#' 
+#' df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex")
+#' dt <- as.aggre(dt, values = c("pyrs", "obs"), by = "sex")
+#' 
+#' class(df)
+#' class(dt)
+#' 
+#' BL <- list(fot = 0:5)
+#' df <- data.frame(df)
+#' df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex", breaks = BL)
+#' 
+#' @return
+#' Returns a copy of `x` with attributes set to those of an object of class
+#' `"aggre"`.
+#' @export
+as.aggre <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
+  UseMethod("as.aggre", x)
+}
+
+#' @describeIn as.aggre Coerces a \code{data.frame} to an \code{aggre} object
+#' @export
+as.aggre.data.frame <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
+  x <- copy(x)
+  setaggre(x, values = values, by = by, breaks = breaks, ...)
+  setattr(x, "class", c("aggre", "data.frame"))
+  x[]
+}
+
+#' @describeIn as.aggre Coerces a \code{data.table} to an \code{aggre} object
+#' @export
+as.aggre.data.table <- function(x, values = NULL, by = NULL, breaks = NULL, ...) {
+  x <- copy(x)
+  setaggre(x, values = values, by = by, breaks = breaks, ...)
+  setattr(x, "class", c("aggre", "data.table", "data.frame"))
+  x[]
+}
+
+#' @describeIn as.aggre Default method for \code{as.aggre} (stops computations
+#' if no class-specific method found)
+#' @export
+as.aggre.default <- function(x, ...) {
+  stop(gettextf("cannot coerce class \"%s\" to 'aggre'", deparse(class(x))), 
+       domain = NA)
+}
+
+
+
+#' @title Aggregation of split \code{Lexis} data
+#' @author Joonas Miettinen
+#' @description Aggregates a split \code{Lexis} object by given variables 
+#' and / or expressions into a long-format table of person-years and 
+#' transitions / end-points. Automatic aggregation over time scales
+#' by which data has been split if the respective time scales are mentioned
+#' in the aggregation argument to e.g. intervals of calendar time, follow-up time
+#' and/or age.
+#' @param lex a \code{Lexis} object split with e.g. 
+#' \code{\link[Epi]{splitLexis}} or \code{\link{splitMulti}}
+#' @param by variables to tabulate (aggregate) by.
+#' \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{by = c("V1", "V2")}. See Details and Examples.
+#' @param type determines output levels to which data is aggregated varying
+#' from returning only rows with \code{pyrs > 0} (\code{"unique"}) to
+#' returning all possible combinations of variables given in \code{aggre} even
+#' if those combinations are not represented in data (\code{"full"}); 
+#' see Details
+#' @param sum.values optional: additional variables to sum by argument
+#'  \code{by}. \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{sum.values = c("V1", "V2")}
+#' @param subset a logical condition to subset by before computations;
+#' e.g. \code{subset = area \%in\% c("A", "B")}
+#' @param verbose \code{logical}; if \code{TRUE}, the function returns timings
+#' and some information useful for debugging along the aggregation process
+#' @details 
+#' 
+#' \strong{Basics}
+#' 
+#' \code{aggre} is intended for aggregation of split \code{Lexis} data only.
+#' See \code{\link[Epi]{Lexis}} for forming \code{Lexis} objects by hand
+#' and e.g. \code{\link[Epi]{splitLexis}}, \code{\link{splitLexisDT}}, and
+#' \code{\link{splitMulti}} for splitting the data. \code{\link{lexpand}}
+#' may be used for simple data sets to do both steps as well as aggregation
+#' in the same function call.
+#' 
+#' Here aggregation refers to computing person-years and the appropriate events
+#' (state transitions and end points in status) for the subjects in the data.
+#' Hence, it computes e.g. deaths (end-point and state transition) and 
+#' censorings (end-point) as well as events in a multi-state setting
+#' (state transitions).
+#' 
+#' The result is a long-format \code{data.frame} or \code{data.table}
+#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi})
+#' with the columns \code{pyrs} and the appropriate transitions named as
+#' \code{fromXtoY}, e.g. \code{from0to0} and \code{from0to1} depending
+#' on the values of \code{lex.Cst} and \code{lex.Xst}.
+#' 
+#' 
+#' \strong{The by argument}
+#' 
+#' The \code{by} argument determines the length of the table, i.e.
+#' the combinations of variables to which data is aggregated.  
+#' \code{by} is relatively flexible, as it can be supplied as
+#' 
+#' \itemize{
+#'  \item{a character string vector, e.g. \code{c("sex", "area")}, 
+#'  naming variables existing in \code{lex}}
+#'  \item{an expression, e.g. \code{factor(sex, 0:1, c("m", "f"))} 
+#'  using any variable found in \code{lex}}
+#'  \item{a list (fully or partially named) of expressions, e.g. 
+#'  \code{list(gender = factor(sex, 0:1, c("m", "f"), area)}}
+#' }
+#' 
+#' Note that expressions effectively allow a variable to be supplied simply as
+#' e.g. \code{by = sex} (as a symbol/name in R lingo).
+#' 
+#' The data is then aggregated to the levels of the given variables 
+#' or expression(s). Variables defined to be time scales in the supplied 
+#' \code{Lexis} are processed in a special way: If any are mentioned in the
+#' \code{by} argument, intervals of them are formed based on the breaks
+#' used to split the data: e.g. if \code{age} was split using the breaks 
+#' \code{c(0, 50, Inf)}, mentioning \code{age} in \code{by} leads to
+#' creating the \code{age} intervals \code{[0, 50)} and \code{[50, Inf)}
+#' and aggregating to them. The intervals are identified in the output
+#' as the lower bounds of the appropriate intervals.
+#' 
+#' The order of multiple time scales mentioned in \code{by} matters,
+#' as the last mentioned time scale is assumed to be a survival time scale
+#' for when computing event counts. E.g. when the data is split by the breaks
+#' \code{list(FUT = 0:5, CAL = c(2008,2010))}, time lines cut short at
+#' \code{CAL = 2010} are considered to be censored, but time lines cut short at
+#' \code{FUT = 5} are not. See Return.
+#' 
+#' \strong{Aggregation types (styles)}
+#' 
+#' It is almost always enough to aggregate the data to variable levels
+#' that are actually represented in the data 
+#' (default \code{aggre = "unique"}; alias \code{"non-empty"}). 
+#' For certain uses it may be useful
+#' to have also "empty" levels represented (resulting in some rows in output
+#' with zero person-years and events); in these cases supplying
+#' \code{aggre = "full"} (alias \code{"cartesian"}) causes \code{aggre}
+#' to determine the Cartesian product of all the levels of the supplied 
+#' \code{by} variables or expressions and aggregate to them. As an example
+#' of a Cartesian product, try
+#' 
+#' \code{merge(1:2, 1:5)}.
+#' 
+#' @return 
+#' A long \code{data.frame} or \code{data.table} of aggregated person-years 
+#' (\code{pyrs}), numbers of subjects at risk (\code{at.risk}), and events
+#' formatted \code{fromXtoY}, where \code{X} and \code{X} are states 
+#' transitioning from and to or states at the end of each \code{lex.id}'s 
+#' follow-up (implying \code{X} = \code{Y}). Subjects at risk are computed 
+#' in the beginning of an interval defined by any Lexis time scales and 
+#' mentioned in \code{by}, but events occur at any point within an interval.
+#' 
+#' When the data has been split along multiple time scales, the last
+#' time scale mentioned in \code{by} is considered to be the survival time 
+#' scale with regard to computing events. Time lines cut short by the
+#' extrema of non-survival-time-scales are considered to be censored
+#' ("transitions" from the current state to the current state).
+#' 
+#' @seealso \code{\link{aggregate}} for a similar base R solution,
+#' and \code{\link{ltable}} for a \code{data.table} based aggregator. Neither
+#' are directly applicable to split \code{Lexis} data.
+#' 
+#' @family aggregation functions
+#' 
+#' 
+#' 
+#' @examples 
+#' 
+#' ## form a Lexis object
+#' library(Epi)
+#' data(sibr)
+#' x <- sibr[1:10,]
+#' x[1:5,]$sex <- 0 ## pretend some are male
+#' x <- Lexis(data = x,
+#'            entry = list(AGE = dg_age, CAL = get.yrs(dg_date)),
+#'            exit = list(CAL = get.yrs(ex_date)),
+#'            entry.status=0, exit.status = status)
+#' x <- splitMulti(x, breaks = list(CAL = seq(1993, 2013, 5), 
+#'                                  AGE = seq(0, 100, 50)))
+#' 
+#' ## these produce the same results (with differing ways of determining aggre)
+#' a1 <- aggre(x, by = list(gender = factor(sex, 0:1, c("m", "f")), 
+#'              agegroup = AGE, period = CAL))
+#' 
+#' a2 <- aggre(x, by = c("sex", "AGE", "CAL"))
+#' 
+#' a3 <- aggre(x, by = list(sex, agegroup = AGE, CAL))
+#' 
+#' ## returning also empty levels
+#' a4 <- aggre(x, by = c("sex", "AGE", "CAL"), type = "full")
+#' 
+#' ## computing also expected numbers of cases
+#' x <- lexpand(sibr[1:10,], birth = bi_date, entry = dg_date,
+#'              exit = ex_date, status = status %in% 1:2, 
+#'              pophaz = popmort, fot = 0:5, age = c(0, 50, 100))
+#' x$d.exp <- with(x, lex.dur*pop.haz)
+#' ## these produce the same result
+#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = list(d.exp))
+#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = "d.exp")
+#' a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = d.exp)
+#' ## same result here with custom name
+#' a5 <- aggre(x, by = c("sex", "age", "fot"), 
+#'              sum.values = list(expCases = d.exp))
+#'              
+#' ## computing pohar-perme weighted figures
+#' x$d.exp.pp <- with(x, lex.dur*pop.haz*pp)
+#' a6 <- aggre(x, by = c("sex", "age", "fot"), 
+#'              sum.values = c("d.exp", "d.exp.pp"))
+#' ## or equivalently e.g. sum.values = list(expCases = d.exp, expCases.p = d.exp.pp).
+#' @export
+aggre <- function(lex, by = NULL, type = c("unique", "full"), sum.values = NULL, subset = NULL, verbose = FALSE) {
+  
+  allTime <- proc.time()
+  
+  lex.Cst <- lex.Xst <- lex.id <- at.risk <- NULL ## APPEASE R CMD CHECK
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  
+  type <- match.arg(type[1], c("non-empty", "unique", "full", "cartesian"))
+  if (type == "cartesian") type <- "full"
+  if (type == "non-empty") type <- "unique"
+  
+  if (verbose) cat("Aggregation type: '", type, "' \n", sep = "")
+  
+  checkLexisData(lex)
+  
+  breaks <- copy(attr(lex, "breaks"))
+  checkBreaksList(lex, breaks)
+  
+  allScales <- copy(attr(lex, "time.scales"))
+  if (length(allScales) == 0 ) {
+    stop("could not determine names of time scales; ",
+         "is the data a Lexis object?")
+  }
+  
+  ## subset --------------------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(lex, subset)
+  
+  ## check sum.values ----------------------------------------------------------
+  sumSub <- substitute(sum.values)
+  sum.values <- evalPopArg(lex[1:min(nrow(lex), 20L), ], arg = sumSub, 
+                     enclos = PF, recursive = TRUE, DT = TRUE)
+  sumType <- attr(sum.values, "arg.type")
+  sumVars <- attr(sum.values, "all.vars")
+  sumSub <- attr(sum.values, "quoted.arg")
+  if (is.null(sum.values)) {
+    sumType <- "NULL"
+    sumVars <- NULL
+    sumSub <- quote(list())
+  }
+  badSum <- names(sum.values)[!sapply(sum.values, is.numeric)]
+  if (length(badSum) > 0L) {
+    badSum <- paste0("'", badSum, "'", collapse = ", ")
+    stop("Following variables resulting from evaluating supplied sum.values ",
+         "argument are not numeric and cannot be summed: ", badSum, 
+         ". Evaluated sum.values: ", deparse(sumSub))
+  }
+  
+  
+  ## by argument type -------------------------------------------------------
+  ## NOTE: need to eval by AFTER cutting time scales!
+  
+  ags <- substitute(by)
+  if (verbose) cat("Used by argument:", paste0(deparse(ags)),"\n")
+  
+  ## NOTE: with recursive = TRUE, evalPopArg digs deep enough to find
+  ## the actual expression (substituted only once) and returns that and other
+  ## things in attributes. Useful if arg substituted multiple times.
+  by <- evalPopArg(data = lex[1:min(nrow(lex), 20),], 
+                      arg = ags, DT = TRUE, enclos = PF, recursive = TRUE)
+  ags <- attr(by, "quoted.arg") 
+  av <- attr(by, "all.vars")
+  argType <- attr(by, "arg.type")
+  
+  if (is.null(by)) {
+    ags <- substitute(list())
+    av <- NULL
+    argType <- "NULL"
+    type <- "unique"
+  }
+  if (verbose) cat("Type of by argument:", argType, "\n")
+  
+  ## take copy of lex ----------------------------------------------------------
+  ## if lex is a data.table, this function gets really complicated.
+  ## if copy is taken only of necessary vars, it should be fine.
+  keepVars <- unique(c("lex.id", allScales, "lex.dur", 
+                       "lex.Cst", "lex.Xst", av, sumVars))
+  lex.orig <- lex
+  lex <- subsetDTorDF(lex, subset = subset, select = keepVars)
+  lex <- data.table(lex)
+  forceLexisDT(lex, breaks = breaks, allScales = allScales, key = FALSE)
+  
+  
+  ## ensure no observations outside breaks limits are left in
+  lex <- intelliDrop(lex, breaks = breaks)
+  
+  setkeyv(lex, c("lex.id", allScales[1]))
+  setcolsnull(lex, delete = setdiff(allScales, names(breaks)))
+  
+  
+  ## cut time scales for aggregating if needed ---------------------------------
+  aggScales <- intersect(av, allScales)
+  if (any(!aggScales %in% names(breaks))) {
+    aggScales <- paste0("'", setdiff(aggScales, names(breaks)), "'", collapse = ", ")
+    stop("Requested aggregating by time scale(s) by which data ",
+         "has not been split: ", aggScales)
+  }
+  
+  ## before cutting, find out which rows count towards "at.risk" figure:
+  ## of all scales in aggScales, the last one (or the only one) is assumed
+  ## to be the survival time scale.
+  tmpAtRisk <- makeTempVarName(lex, pre = "at.risk_")
+  set(lex, j = tmpAtRisk, value = TRUE)
+  survScale <- NULL
+  
+  
+  if (length(aggScales) > 0) {
+    cutTime <- proc.time()
+    ## "at.risk" counts subjects at risk in the beginning of the survival
+    ## time scale interval.
+    survScale <- aggScales[length(aggScales)]
+    lex[, c(tmpAtRisk) := lex[[survScale]] %in% breaks[[survScale]] ]
+    catAggScales <- paste0("'", aggScales, "'", collapse = ", ")
+    if (verbose) {
+      cat("Following time scales mentioned in by argument and will be",
+          "categorized into intervals (defined by breaks in object",
+          "attributes) for aggregation:", catAggScales, "\n")
+    }
+    
+    ## NEW METHOD: use a copy of lex and just modify in place.
+    
+    for (sc in aggScales) {
+      set(lex, j = sc, value = cutLow(lex[[sc]], breaks = breaks[[sc]]))
+    }
+    
+    if (verbose) cat("Time taken by cut()'ting time scales: ", timetaken(cutTime), "\n")
+  }
+  
+  othVars <- setdiff(av, aggScales)
+  if (verbose && length(othVars) > 0) {
+    catOthVars <- paste0("'", othVars, "'", collapse = ", ")
+    cat("Detected the following non-time-scale variables to be utilized in aggregating:", catOthVars, "\n")
+  }
+  
+  ## eval by -------------------------------------------------------------------
+  ## NOTE: needed to eval by AFTER cutting time scales!
+  by <- evalPopArg(data = lex, arg = ags, DT = TRUE, enclos = PF, recursive = TRUE)
+  byNames <- names(by)
+  
+  ## computing pyrs ------------------------------------------------------------
+  ## final step in determining at.risk:
+  ## a lex.id is at.risk only once per by-level
+  pyrsTime <- proc.time()
+  vdt <- data.table(pyrs = lex$lex.dur, at.risk = lex[[tmpAtRisk]], 
+                    lex.id = lex$lex.id)
+  pyrs <- vdt[, .(pyrs = sum(pyrs), 
+                  at.risk = sum(!duplicated(lex.id) & at.risk)), 
+              keyby = by]
+  setDT(pyrs)
+  
+  rm(vdt)
+  sumNames <- NULL
+  if (sumType != "NULL") {
+    if (sumType == "character") {
+      sumNames <- evalPopArg(lex, sumSub, n = 1L, DT = FALSE, recursive = TRUE, enclos = PF)
+      sum.values <- lex[, lapply(.SD, sum), keyby = by, .SDcols = c(sumNames)]
+    } else {
+      sum.values <- evalPopArg(lex, sumSub, n = 1L, enclos = PF)
+      sumNames <- names(sum.values)
+      sumTmpNames <- makeTempVarName(lex, pre = sumNames)
+      set(lex, j = sumTmpNames, value = sum.values)
+      sum.values <- lex[, lapply(.SD, sum), keyby = by, .SDcols = sumTmpNames]
+      setnames(sum.values, sumTmpNames, sumNames)
+      setcolsnull(lex, sumTmpNames)
+    }
+    
+    setDT(sum.values)
+    pyrs <- merge(pyrs, sum.values, all = TRUE)
+    rm(sum.values)
+  }
+  
+  
+  if (verbose) cat("Time taken by aggregating pyrs: ", timetaken(pyrsTime), "\n")
+  
+  valVars <- setdiff(names(pyrs), byNames) ## includes pyrs and anything created by sum
+  
+  pyrs[is.na(pyrs), pyrs := 0]
+  pyrs <- pyrs[pyrs > 0]
+  
+  aggPyrs <- pyrs[, sum(pyrs)] 
+  lexPyrs <- sum(lex.orig$lex.dur[subset])
+  pyrsDiff <- aggPyrs - lexPyrs
+  if (!isTRUE(all.equal(aggPyrs, lexPyrs, scale = NULL))) {
+    warning("Found discrepancy of ", abs(round(pyrsDiff, 4)), " ",
+            "in total aggregated pyrs compared to ",
+            "sum(lex$lex.dur); compare results by hand and make sure ",
+            "settings are right \n")
+  }
+  rm(subset, aggPyrs, lexPyrs)
+  
+  ## cartesian output ----------------------------------------------------------
+  if (type == "full") {
+    carTime <- proc.time()
+    
+    varsUsingScales <- NULL
+    
+    ## which variables used one time scale? and which one?
+    ## will only be used in cartesian stuff.
+    if (argType == "character") {
+      varsUsingScales <- intersect(names(by), aggScales)
+      whScaleUsed <- varsUsingScales
+    } else if (argType != "NULL") {
+      ## note: ags a substitute()'d list at this point always if not char
+      whScaleUsed <- lapply(ags[-1], function(x) intersect(all.vars(x), aggScales))
+      ## only one time scale should be used in a variable!
+      oneScaleTest <- any(sapply(whScaleUsed, function(x) length(x) > 1L))
+      if (oneScaleTest) stop("Only one Lexis time scale can be used in any one variable in by argument!")
+      varsUsingScales <- byNames[sapply(whScaleUsed, function (x) length(x) == 1L)]
+      whScaleUsed <- unlist(whScaleUsed)
+    }
+    
+    ceejay <- lapply(by, function(x) if (is.factor(x)) levels(x) else sort(unique(x)))
+    if (length(aggScales) > 0) {
+      ## which variables in ceejay used the Lexis time scales from lex?
+      ceejay[varsUsingScales] <- lapply(breaks[whScaleUsed], function(x) x[-length(x)])
+    }
+    
+    ceejay <- do.call(CJ, ceejay)
+    setkeyv(ceejay, byNames)
+    setkeyv(pyrs, byNames)
+    
+    pyrs <- pyrs[ceejay]
+    rm(ceejay)
+    
+    if (verbose) cat("Time taken by making aggregated data large in the cartesian product sense: ", timetaken(carTime), "\n")
+  }
+  
+  
+  ## computing events ----------------------------------------------------------
+  
+  transTime <- proc.time()
+  
+  if (is.null(by) || (is.data.table(by) && nrow(by) == 0L)) {
+    
+    by <- quote(list(lex.Cst, lex.Xst))
+    
+  } else {
+    for (var in c("lex.Cst", "lex.Xst")) {
+      set(by, j = var, value = lex[[var]])
+    }
+  }
+  
+  
+  ## NOTE: this will ensure correct detection of censorings:
+  ## observations cut short by e.g. period window's edge
+  ## will be considered a censoring if the breaks along that time scale
+  ## are not passed to detectEvents (assuming the survival time scale is
+  ## used in by). If no time scale mentioned in by, then all endings
+  ## of observations are either censorings or events.
+  detBr <- breaks[survScale]
+  if (!length(survScale)) detBr <- NULL
+  hasEvent <- detectEvents(lex, breaks = detBr, by = "lex.id") %in% 1:2
+  ## is language if user supplied by = NULL 
+  if (!is.language(by)) by <- by[hasEvent]
+  
+  trans <- lex[hasEvent, list(obs = .N), keyby = by]
+  
+  rm(by, lex)
+  
+  
+  if (verbose) cat("Time taken by aggregating events: ", timetaken(transTime), "\n")
+  
+  ## casting & merging ---------------------------------------------------------
+  
+  mergeTime <- proc.time()
+  setDT(trans)
+  setDT(pyrs)
+  
+  ## tmpTr to be used in casting
+  tmpTr <- makeTempVarName(trans, pre = "trans_")
+  trans[, c(tmpTr) := paste0("from", lex.Cst, "to", lex.Xst)]
+  transitions <- sort(unique(trans[[tmpTr]]))
+  trans[, c("lex.Cst", "lex.Xst") := NULL]
+  
+  ## note: need tmpDum if by = NULL for correct casting & merging
+  tmpDum <- makeTempVarName(trans)
+  byNames <- c(byNames, tmpDum)
+  byNames <- setdiff(byNames, c("lex.Cst", "lex.Xst"))
+  trans[, c(tmpDum) := 1L]
+  pyrs[, c(tmpDum) := 1L]
+  
+  valVars <- unique(c(valVars, transitions))
+  
+  trans <- cast_simple(trans, rows = byNames, columns = tmpTr, values = "obs")
+  
+  setkeyv(trans, NULL); setkeyv(pyrs, NULL) ## dcast.data.table seems to keep key but row order may be funky; this avoids a warning
+  setkeyv(trans, byNames); setkeyv(pyrs, byNames)
+  trans <- trans[pyrs]; rm(pyrs)
+  
+  trans[, c(tmpDum) := NULL]
+  byNames <- setdiff(byNames, tmpDum)
+  setcolorder(trans, c(byNames, valVars))
+  
+  if (verbose) cat("Time taken by merging pyrs & transitions: ", timetaken(mergeTime), "\n")
+  
+  if (length(valVars) > 0L) {
+    trans[, c(valVars) := lapply(.SD, function(x) {
+      x[is.na(x)] <- 0
+      x
+    }), .SDcols = c(valVars)]
+  }
+  
+  
+  ## final touch ---------------------------------------------------------------
+  trans <- data.table(trans)
+  setaggre(trans, values = c("pyrs", "at.risk", transitions, sumNames), 
+           by = byNames, breaks = breaks)
+  if (!return_DT()) setDFpe(trans)
+  if (verbose) cat("Time taken by aggre(): ", timetaken(allTime), "\n")
+  
+  
+  
+  trans[]
+}
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/R/data_document.R b/R/data_document.R
index 7ca38bc..314f340 100644
--- a/R/data_document.R
+++ b/R/data_document.R
@@ -1,175 +1,175 @@
-
-
-
-# sire - simulated survival data ------------------------------------------
-
-#' sire - a simulated cohort of Finnish female rectal cancer patients
-#'
-#' \code{sire} is a simulated cohort pertaining female Finnish rectal cancer patients
-#' diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
-#' via modest randomization within several time windows. 
-#'
-#' The closing date for the pertinent data was 2012-12-31, meaning status information was
-#' available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
-#'
-#' @source The Finnish Cancer Registry
-#' @format data.table with columns
-#' \itemize{
-#'  \item sex - gender of the patient (1 = female)
-#'  \item bi_date - date of birth
-#'  \item dg_date - date of cancer diagnosis
-#'  \item ex_date - date of exit from follow-up (death or censoring)
-#'  \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
-#'  \item dg_age  - age at diagnosis expressed as fractional years
-#' }
-#' @author Karri Seppa
-#' @name sire
-#' @family popEpi data
-#' @family survival data
-NULL
-
-
-# sibr - simulated survival data ------------------------------------------
-
-#' sibr - a simulated cohort of Finnish female breast cancer patients
-#'
-#' \code{sibr} is a simulated cohort pertaining female Finnish breast cancer patients
-#' diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
-#' via modest randomization within several time windows. The dataset is additionally
-#' a random sample of 10 000 cases from the pertaining time window. 
-#'
-#' The closing date for the pertinent data was 2012-12-31, meaning status information was
-#' available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
-#'
-#' @source The Finnish Cancer Registry
-#' @format data.table with columns
-#' \itemize{
-#'  \item sex - gender of the patient (1 = female)
-#'  \item bi_date - date of birth
-#'  \item dg_date - date of cancer diagnosis
-#'  \item ex_date - date of exit from follow-up (death or censoring)
-#'  \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
-#'  \item dg_age  - age at diagnosis expressed as fractional years
-#' }
-#' @author Karri Seppa
-#' @name sibr
-#' @family popEpi data
-#' @family survival data
-NULL
-
-
-
-
-# International standard weights ------------------------------------------
-#' Age standardisation weights from the ICSS scheme.
-#'
-#' Contains three sets age-standardisation weights for age-standardized survival (net, relative or observed).
-#' 
-#'
-#' @source 
-#' \href{https://seer.cancer.gov/stdpopulations/survival.html}{ICSS weights (US National Cancer Institute website)}
-#' 
-#' Corazziari, Isabella, Mike Quinn, and Riccardo Capocaccia. "Standard cancer patient population for age standardising survival ratios." European Journal of Cancer 40.15 (2004): 2307-2316.
-#' @format data.table with columns
-#' \itemize{
-#'  \item age - lower bound of the age group
-#'  \item ICSS1 - first set of weights, sums to 100 000
-#'  \item ICSS2 - second set of weights, sums to 100 000
-#'  \item ICSS3 - third set of weights, sums to 100 000
-#' }
-#' @name ICSS
-#' @family popEpi data
-#' @family weights
-#' @examples 
-#' ## aggregate weights to a subset of age groups
-#' data(ICSS)
-#' cut <- c(0, 30, 50, 70, Inf)
-#' agegr <- cut(ICSS$age, cut, right = FALSE)
-#' aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-NULL
-
-
-# popmort -------------------------------------------------------------------
-#' Population mortality rates in Finland 1951 - 2013 in 101 age groups and
-#' by gender. This is an example of a population hazard table as used in 
-#' \pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.
-#'
-#'
-#' @source Statistics Finland
-#' @format \code{data.table} with columns
-#' \itemize{
-#'  \item \code{sex} gender coded as male, female (0, 1)
-#'  \item \code{year} calendar year
-#'  \item \code{agegroup} - coded 0 to 100; one-year age groups
-#'  \item \code{haz} the average population mortality rate per person-year 
-#'  (d/(pyrs), where d is the number of deaths and pyrs is the person-years)
-#' }
-#' @name popmort
-#' @family popEpi data
-#' @seealso \code{\link{pophaz}}
-NULL
-
-
-
-
-# stdpop18 ------------------------------------------------------------------
-
-#' Standard populations from 2000: world, Europe and Nordic.
-#'
-#' World, European, and Nordic standard populations by 18 age categories. 
-#' Sums to 100000.
-#'
-#' @source Nordcan, 2000
-#' @format data.table with columns
-#' \itemize{
-#'  \item \code{agegroup}, age group in 18 categories (character)
-#'  \item \code{world}, World 2000 standard population (numeric)
-#'  \item \code{europe}, European standard population (numeric)
-#'  \item \code{nordic}, Nordic standard population (numeric)
-#' }
-#' @name stdpop18
-#' @family popEpi data
-#' @family weights
-NULL
-
-
-# stdpop101 -----------------------------------------------------------------
-
-#' World standard population by 1 year age groups from 1 to 101. Sums to 100 000.
-#'
-#'
-#' @source Standard population is from:
-#' \href{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}{world standard population "101of1"}
-#' 
-#' @format data.table with columns
-#' \itemize{
-#'  \item \code{world_std} weight that sums to 100000 (numeric)
-#'  \item \code{agegroup} age group from 1 to 101 (numeric)
-#' }
-#' @name stdpop101
-#' @family popEpi data
-#' @family weights
-NULL
-
-
-
-
-
-# meanpop_fi -------------------------------------------------------------------
-#' Mean population counts in Finland year, sex, and age group.
-#'
-#' @source Statistics Finland
-#' @format \code{data.table} with columns
-#' \itemize{
-#'  \item \code{sex} gender coded as male, female (0, 1)
-#'  \item \code{year} calendar year 1981-2016
-#'  \item \code{agegroup} - coded 0 to 100; one-year age groups
-#'  \item \code{meanpop} the mean population count; that is, the mean of the
-#'  annual population counts of two consecutive years; e.g. for 1990 
-#'  \code{meanpop} is the mean of population counts for 1990 and 1991 
-#'  (counted at 1990-01-01 and 1991-01-01, respectively)
-#' }
-#' @name meanpop_fi
-#' @family popEpi data
-NULL
-
+
+
+
+# sire - simulated survival data ------------------------------------------
+
+#' sire - a simulated cohort of Finnish female rectal cancer patients
+#'
+#' \code{sire} is a simulated cohort pertaining female Finnish rectal cancer patients
+#' diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
+#' via modest randomization within several time windows. 
+#'
+#' The closing date for the pertinent data was 2012-12-31, meaning status information was
+#' available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
+#'
+#' @source The Finnish Cancer Registry
+#' @format data.table with columns
+#' \itemize{
+#'  \item sex - gender of the patient (1 = female)
+#'  \item bi_date - date of birth
+#'  \item dg_date - date of cancer diagnosis
+#'  \item ex_date - date of exit from follow-up (death or censoring)
+#'  \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
+#'  \item dg_age  - age at diagnosis expressed as fractional years
+#' }
+#' @author Karri Seppa
+#' @name sire
+#' @family popEpi data
+#' @family survival data
+NULL
+
+
+# sibr - simulated survival data ------------------------------------------
+
+#' sibr - a simulated cohort of Finnish female breast cancer patients
+#'
+#' \code{sibr} is a simulated cohort pertaining female Finnish breast cancer patients
+#' diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
+#' via modest randomization within several time windows. The dataset is additionally
+#' a random sample of 10 000 cases from the pertaining time window. 
+#'
+#' The closing date for the pertinent data was 2012-12-31, meaning status information was
+#' available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
+#'
+#' @source The Finnish Cancer Registry
+#' @format data.table with columns
+#' \itemize{
+#'  \item sex - gender of the patient (1 = female)
+#'  \item bi_date - date of birth
+#'  \item dg_date - date of cancer diagnosis
+#'  \item ex_date - date of exit from follow-up (death or censoring)
+#'  \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
+#'  \item dg_age  - age at diagnosis expressed as fractional years
+#' }
+#' @author Karri Seppa
+#' @name sibr
+#' @family popEpi data
+#' @family survival data
+NULL
+
+
+
+
+# International standard weights ------------------------------------------
+#' Age standardisation weights from the ICSS scheme.
+#'
+#' Contains three sets age-standardisation weights for age-standardized survival (net, relative or observed).
+#' 
+#'
+#' @source 
+#' \href{https://seer.cancer.gov/stdpopulations/survival.html}{ICSS weights (US National Cancer Institute website)}
+#' 
+#' Corazziari, Isabella, Mike Quinn, and Riccardo Capocaccia. "Standard cancer patient population for age standardising survival ratios." European Journal of Cancer 40.15 (2004): 2307-2316.
+#' @format data.table with columns
+#' \itemize{
+#'  \item age - lower bound of the age group
+#'  \item ICSS1 - first set of weights, sums to 100 000
+#'  \item ICSS2 - second set of weights, sums to 100 000
+#'  \item ICSS3 - third set of weights, sums to 100 000
+#' }
+#' @name ICSS
+#' @family popEpi data
+#' @family weights
+#' @examples 
+#' ## aggregate weights to a subset of age groups
+#' data(ICSS)
+#' cut <- c(0, 30, 50, 70, Inf)
+#' agegr <- cut(ICSS$age, cut, right = FALSE)
+#' aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+NULL
+
+
+# popmort -------------------------------------------------------------------
+#' Population mortality rates in Finland 1951 - 2013 in 101 age groups and
+#' by gender. This is an example of a population hazard table as used in 
+#' \pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.
+#'
+#'
+#' @source Statistics Finland
+#' @format \code{data.table} with columns
+#' \itemize{
+#'  \item \code{sex} gender coded as male, female (0, 1)
+#'  \item \code{year} calendar year
+#'  \item \code{agegroup} - coded 0 to 100; one-year age groups
+#'  \item \code{haz} the average population mortality rate per person-year 
+#'  (d/(pyrs), where d is the number of deaths and pyrs is the person-years)
+#' }
+#' @name popmort
+#' @family popEpi data
+#' @seealso \code{\link{pophaz}}
+NULL
+
+
+
+
+# stdpop18 ------------------------------------------------------------------
+
+#' Standard populations from 2000: world, Europe and Nordic.
+#'
+#' World, European, and Nordic standard populations by 18 age categories. 
+#' Sums to 100000.
+#'
+#' @source Nordcan, 2000
+#' @format data.table with columns
+#' \itemize{
+#'  \item \code{agegroup}, age group in 18 categories (character)
+#'  \item \code{world}, World 2000 standard population (numeric)
+#'  \item \code{europe}, European standard population (numeric)
+#'  \item \code{nordic}, Nordic standard population (numeric)
+#' }
+#' @name stdpop18
+#' @family popEpi data
+#' @family weights
+NULL
+
+
+# stdpop101 -----------------------------------------------------------------
+
+#' World standard population by 1 year age groups from 1 to 101. Sums to 100 000.
+#'
+#'
+#' @source Standard population is from:
+#' \href{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}{world standard population "101of1"}
+#' 
+#' @format data.table with columns
+#' \itemize{
+#'  \item \code{world_std} weight that sums to 100000 (numeric)
+#'  \item \code{agegroup} age group from 1 to 101 (numeric)
+#' }
+#' @name stdpop101
+#' @family popEpi data
+#' @family weights
+NULL
+
+
+
+
+
+# meanpop_fi -------------------------------------------------------------------
+#' Mean population counts in Finland year, sex, and age group.
+#'
+#' @source Statistics Finland
+#' @format \code{data.table} with columns
+#' \itemize{
+#'  \item \code{sex} gender coded as male, female (0, 1)
+#'  \item \code{year} calendar year 1981-2016
+#'  \item \code{agegroup} - coded 0 to 100; one-year age groups
+#'  \item \code{meanpop} the mean population count; that is, the mean of the
+#'  annual population counts of two consecutive years; e.g. for 1990 
+#'  \code{meanpop} is the mean of population counts for 1990 and 1991 
+#'  (counted at 1990-01-01 and 1991-01-01, respectively)
+#' }
+#' @name meanpop_fi
+#' @family popEpi data
+NULL
+
diff --git a/R/direct_adjusting.R b/R/direct_adjusting.R
index ec3bca8..4b58984 100644
--- a/R/direct_adjusting.R
+++ b/R/direct_adjusting.R
@@ -1,155 +1,155 @@
-
-
-
-#' @title Direct Adjusting in \pkg{popEpi} Using Weights
-#' @author Joonas Miettinen
-#' @name direct_standardization
-#' @aliases direct_adjusting
-#' @description 
-#' 
-#' Several functions in \pkg{popEpi} have support for direct standardization
-#' of estimates. This document explains the usage of weighting with those
-#' functions.
-#' 
-#' @details 
-#' 
-#' Direct standardization is performed by computing estimates of
-#' \code{E}
-#' by the set of adjusting variables \code{A}, to which a set of weights 
-#' \code{W} is applicable. The weighted average over \code{A} is then the
-#' direct-adjusted estimate of \code{E} (\code{E*}).
-#' 
-#' To enable both quick and easy as well as more rigorous usage of direct
-#' standardization with weights, the weights arguments in \pkg{popEpi}
-#' can be supplied in several ways. Ability to use the different
-#' ways depends on the number of adjusting variables.
-#' 
-#' The weights are always handled internally to sum to 1, so they do not
-#' need to be scaled in this manner when they are supplied. E.g.
-#' counts of subjects in strata may be passed.
-#' 
-#' @section Basic usage - one adjusting variable:
-#' 
-#' In the simple case where we are adjusting by only one variable 
-#' (e.g. by age group), one can simply supply a vector of weights:
-#' 
-#' \code{FUN(weights = c(0.1, 0.25, 0.25, 0.2, 0.2))}
-#' 
-#' which may be stored in advance:
-#' 
-#' \code{w <- c(0.1, 0.25, 0.25, 0.2, 0.2)}
-#' 
-#' \code{FUN(weights = w)}
-#' 
-#' The order of the weights matters. \pkg{popEpi} functions with direct
-#' adjusting enabled match the supplied weights to the adjusting variables
-#' as follows: If the adjusting variable is a \code{factor}, the order
-#' of the levels is used. Otherwise, the alphabetic order of the unique
-#' values is used (try \code{sort} to see how it works). For clarity
-#' and certainty we recommend using \code{factor} or \code{numeric} variables
-#' when possible. \code{character} variables should be avoided: to see why,
-#' try \code{sort(15:9)} and \code{sort(as.character(15:9))}.
-#' 
-#' It is also possible to supply a \code{character} string corresponding
-#' to one of the age group standardization schemes integrated into \pkg{popEpi}:
-#' 
-#' \itemize{
-#' \item \code{'europe_1976_18of5'} - European std. population (1976), 18 age groups
-#' \item \code{'nordic_2000_18of5'} - Nordic std. population (2000), 18 age groups
-#' \item \code{'world_1966_18of5'} - world standard (1966), 18 age groups
-#' \item \code{'world_2000_18of5'} - world standard (2000), 18 age groups
-#' \item \code{'world_2000_20of5'} - world standard (2000), 20 age groups 
-#' \item \code{'world_2000_101of1'} - world standard (2000), 101 age groups
-#' }
-#' 
-#' Additionally, \code{\link{ICSS}} contains international weights used in 
-#' cancer survival analysis, but they are not currently usable by passing
-#' a string to \code{weights} and must be supplied by hand.
-#' 
-#' You may also supply \code{weights = "internal"} to use internally
-#' computed weights, i.e. usually simply the counts of subjects / person-time
-#' experienced in each stratum. E.g.
-#' 
-#' \code{FUN(weights = "world_2000_18of5")}
-#' 
-#' will use the world standard population from 2000 as 
-#' weights for 18 age groups, that your adjusting variable is 
-#' assumed to contain. The adjusting variable must be coded in this case as 
-#' a numeric variable containing \code{1:18} or as a \code{factor} with
-#' 18 levels (coded from the youngest to the oldest age group). 
-#' 
-#' @section More than one adjusting variable:
-#' 
-#' In the case that you employ more than one adjusting variable, separate
-#' weights should be passed to match to the levels of the different adjusting
-#' variables. When supplied correctly, "grand" weights are formed based on
-#' the variable-specific weights by multiplying over the variable-specific
-#' weights (e.g. if men have \code{w = 0.5} and the age group 0-4 has 
-#' \code{w = 0.1}, the "grand" weight for men aged 0-4 is \code{0.5*0.1}).
-#' The "grand" weights are then used for adjusting after ensuring they
-#' sum to one.
-#' 
-#' When using multiple adjusting variables, you
-#' are allowed to pass either a named \code{list} of 
-#' weights or a \code{data.frame} of weights. E.g.
-#' 
-#' \code{WL <- list(agegroup = age_w, sex = sex_w)}
-#' 
-#' \code{FUN(weights = WL)}
-#' 
-#' where \code{age_w} and \code{sex_w} are numeric vectors. Given the 
-#' conditions explained in the previous section are satisfied, you may also do
-#' e.g.
-#' 
-#' \code{WL <- list(agegroup = "world_2000_18of", sex = sex_w)}
-#' 
-#' \code{FUN(weights = WL)}
-#' 
-#' and the world standard pop is used as weights for the age groups as outlined
-#' in the previous section.
-#' 
-#' Sometimes using a \code{data.frame} can be clearer (and it is fool-proof
-#' as well). To do this, form a \code{data.frame} that repeats the levels
-#' of your adjusting variables by each level of every other adjusting variable,
-#' and assign the weights as a column named \code{"weights"}. E.g.
-#' 
-#' \code{wdf <- data.frame(sex = rep(0:1, each = 18), agegroup = rep(1:18, 2))}
-#' 
-#' \code{wdf$weights <- rbinom(36, size = 100, prob = 0.25)}
-#' 
-#' \code{FUN(weights = wdf)}
-#' 
-#' If you want to use the counts of subjects in strata as the weights,
-#' one way to do this is by e.g.
-#' 
-#' \code{wdf <- as.data.frame(x$V1, x$V2, x$V3)}
-#' \code{names(wdf) <- c("V1", "V2", "V3", "weights")}
-#' 
-#' @family weights
-#' @family popEpi argument evaluation docs
-#' 
-#' @references
-#' Source of the Nordic standard population in 5-year age groups 
-#' (also contains European & 1966 world standards):
-#' \url{https://www-dep.iarc.fr/NORDCAN/english/glossary.htm}
-#' 
-#' Source of the 1976 European standard population: 
-#' 
-#' Waterhouse, J.,Muir, C.S.,Correa, P.,Powell, J., eds (1976). 
-#' Cancer Incidence in Five Continents, Vol. III. 
-#' IARC Scientific Publications, No. 15, Lyon, IARC.
-#' ISBN: 9789283211150
-#' 
-#' Source of 2000 world standard population in 1-year age groups:
-#' \url{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}
-
-NULL
-
-
-
-
-
-
-
-
-
+
+
+
+#' @title Direct Adjusting in \pkg{popEpi} Using Weights
+#' @author Joonas Miettinen
+#' @name direct_standardization
+#' @aliases direct_adjusting
+#' @description 
+#' 
+#' Several functions in \pkg{popEpi} have support for direct standardization
+#' of estimates. This document explains the usage of weighting with those
+#' functions.
+#' 
+#' @details 
+#' 
+#' Direct standardization is performed by computing estimates of
+#' \code{E}
+#' by the set of adjusting variables \code{A}, to which a set of weights 
+#' \code{W} is applicable. The weighted average over \code{A} is then the
+#' direct-adjusted estimate of \code{E} (\code{E*}).
+#' 
+#' To enable both quick and easy as well as more rigorous usage of direct
+#' standardization with weights, the weights arguments in \pkg{popEpi}
+#' can be supplied in several ways. Ability to use the different
+#' ways depends on the number of adjusting variables.
+#' 
+#' The weights are always handled internally to sum to 1, so they do not
+#' need to be scaled in this manner when they are supplied. E.g.
+#' counts of subjects in strata may be passed.
+#' 
+#' @section Basic usage - one adjusting variable:
+#' 
+#' In the simple case where we are adjusting by only one variable 
+#' (e.g. by age group), one can simply supply a vector of weights:
+#' 
+#' \code{FUN(weights = c(0.1, 0.25, 0.25, 0.2, 0.2))}
+#' 
+#' which may be stored in advance:
+#' 
+#' \code{w <- c(0.1, 0.25, 0.25, 0.2, 0.2)}
+#' 
+#' \code{FUN(weights = w)}
+#' 
+#' The order of the weights matters. \pkg{popEpi} functions with direct
+#' adjusting enabled match the supplied weights to the adjusting variables
+#' as follows: If the adjusting variable is a \code{factor}, the order
+#' of the levels is used. Otherwise, the alphabetic order of the unique
+#' values is used (try \code{sort} to see how it works). For clarity
+#' and certainty we recommend using \code{factor} or \code{numeric} variables
+#' when possible. \code{character} variables should be avoided: to see why,
+#' try \code{sort(15:9)} and \code{sort(as.character(15:9))}.
+#' 
+#' It is also possible to supply a \code{character} string corresponding
+#' to one of the age group standardization schemes integrated into \pkg{popEpi}:
+#' 
+#' \itemize{
+#' \item \code{'europe_1976_18of5'} - European std. population (1976), 18 age groups
+#' \item \code{'nordic_2000_18of5'} - Nordic std. population (2000), 18 age groups
+#' \item \code{'world_1966_18of5'} - world standard (1966), 18 age groups
+#' \item \code{'world_2000_18of5'} - world standard (2000), 18 age groups
+#' \item \code{'world_2000_20of5'} - world standard (2000), 20 age groups 
+#' \item \code{'world_2000_101of1'} - world standard (2000), 101 age groups
+#' }
+#' 
+#' Additionally, \code{\link{ICSS}} contains international weights used in 
+#' cancer survival analysis, but they are not currently usable by passing
+#' a string to \code{weights} and must be supplied by hand.
+#' 
+#' You may also supply \code{weights = "internal"} to use internally
+#' computed weights, i.e. usually simply the counts of subjects / person-time
+#' experienced in each stratum. E.g.
+#' 
+#' \code{FUN(weights = "world_2000_18of5")}
+#' 
+#' will use the world standard population from 2000 as 
+#' weights for 18 age groups, that your adjusting variable is 
+#' assumed to contain. The adjusting variable must be coded in this case as 
+#' a numeric variable containing \code{1:18} or as a \code{factor} with
+#' 18 levels (coded from the youngest to the oldest age group). 
+#' 
+#' @section More than one adjusting variable:
+#' 
+#' In the case that you employ more than one adjusting variable, separate
+#' weights should be passed to match to the levels of the different adjusting
+#' variables. When supplied correctly, "grand" weights are formed based on
+#' the variable-specific weights by multiplying over the variable-specific
+#' weights (e.g. if men have \code{w = 0.5} and the age group 0-4 has 
+#' \code{w = 0.1}, the "grand" weight for men aged 0-4 is \code{0.5*0.1}).
+#' The "grand" weights are then used for adjusting after ensuring they
+#' sum to one.
+#' 
+#' When using multiple adjusting variables, you
+#' are allowed to pass either a named \code{list} of 
+#' weights or a \code{data.frame} of weights. E.g.
+#' 
+#' \code{WL <- list(agegroup = age_w, sex = sex_w)}
+#' 
+#' \code{FUN(weights = WL)}
+#' 
+#' where \code{age_w} and \code{sex_w} are numeric vectors. Given the 
+#' conditions explained in the previous section are satisfied, you may also do
+#' e.g.
+#' 
+#' \code{WL <- list(agegroup = "world_2000_18of", sex = sex_w)}
+#' 
+#' \code{FUN(weights = WL)}
+#' 
+#' and the world standard pop is used as weights for the age groups as outlined
+#' in the previous section.
+#' 
+#' Sometimes using a \code{data.frame} can be clearer (and it is fool-proof
+#' as well). To do this, form a \code{data.frame} that repeats the levels
+#' of your adjusting variables by each level of every other adjusting variable,
+#' and assign the weights as a column named \code{"weights"}. E.g.
+#' 
+#' \code{wdf <- data.frame(sex = rep(0:1, each = 18), agegroup = rep(1:18, 2))}
+#' 
+#' \code{wdf$weights <- rbinom(36, size = 100, prob = 0.25)}
+#' 
+#' \code{FUN(weights = wdf)}
+#' 
+#' If you want to use the counts of subjects in strata as the weights,
+#' one way to do this is by e.g.
+#' 
+#' \code{wdf <- as.data.frame(x$V1, x$V2, x$V3)}
+#' \code{names(wdf) <- c("V1", "V2", "V3", "weights")}
+#' 
+#' @family weights
+#' @family popEpi argument evaluation docs
+#' 
+#' @references
+#' Source of the Nordic standard population in 5-year age groups 
+#' (also contains European & 1966 world standards):
+#' \url{https://www-dep.iarc.fr/NORDCAN/english/glossary.htm}
+#' 
+#' Source of the 1976 European standard population: 
+#' 
+#' Waterhouse, J.,Muir, C.S.,Correa, P.,Powell, J., eds (1976). 
+#' Cancer Incidence in Five Continents, Vol. III. 
+#' IARC Scientific Publications, No. 15, Lyon, IARC.
+#' ISBN: 9789283211150
+#' 
+#' Source of 2000 world standard population in 1-year age groups:
+#' \url{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}
+
+NULL
+
+
+
+
+
+
+
+
+
diff --git a/R/evaluation.R b/R/evaluation.R
index 82575a7..790a033 100644
--- a/R/evaluation.R
+++ b/R/evaluation.R
@@ -1,993 +1,993 @@
-
-
-is_expression <- function(e) {
-  
-  ifelse(any(c("call", "name") %in% class(e)), TRUE, FALSE)
-  
-}
-
-is_list_expression <- function(e) {
-  
-  ifelse(is_expression(e) && deparse(as.list(e)[[1]]) == "list", TRUE, FALSE)
-  
-}
-
-is_dollar_expression <- function(e) {
-  
-  ifelse(is_expression(e) && deparse(as.list(e)[[1]]) == "$", TRUE, FALSE)
-  
-}
-
-
-is_variable <- function(x) {
-  
-  varModes <- c("numeric", "complex", "logical", "character", "raw")
-  
-  ifelse(mode(x) %in% varModes, TRUE, FALSE)
-  
-}
-
-
-evalArg <- function(arg, env, enc, ...) {
-  UseMethod("evalArg")
-}
-
-
-evalArg.default <- function(arg, env, enc) {
-  if (is.list(arg)) {
-    arg <- as.list(arg)
-  } else if (is_variable(arg)) {
-    l <- list(arg)
-  }
-  return(arg)
-}
-
-evalArg.name <- function(arg, env, enc) {
-  d <- deparse(arg)
-  out <- try(get(d, envir = env, inherits = FALSE), silent = TRUE)
-  if (inherits(out, "try-error")) {
-    out <- try(get(d, envir = enc, inherits = FALSE), silent = TRUE)
-  }
-  if (inherits(out, "try-error")) {
-    stop("Could not find object ", d, ".")
-  }
-  out <- list(out)
-  names(out) <- d
-  out
-}
-
-evalArg.call <- function(arg, env, enc) {
-  
-  out <- eval(arg, envir = env, enclos = enc)
-  
-  if (is.list(out)) {
-    out <- as.list(out)
-  } else if (is_variable(out)) {
-    out <- list(out)
-    names(out) <- paste0(deparse(arg), collapse = "")
-  }
-  out
-}
-
-evalArg.character <- function(arg, env, enc) {
-  ## NOTE: enc unused
-  se <- substitute(env)
-  out <- lapply(arg, function(stri) {
-    try({
-      get(stri, envir = as.environment(env), inherits = FALSE)
-    }, silent = TRUE)
-  })
-  
-  notFound <- arg[sapply(out, inherits, "try-error")]
-  
-  if (length(notFound)) {
-    if (length(notFound) > 5) notFound <- notFound[1:5]
-    stop("Could not find object(s): ", 
-         paste0(notFound, collapse = ", "), ".")
-  }
-  names(out) <- arg
-  out
-}
-
-evalArg.formula <- function(arg, env, enc) {
-  
-  rhsl <- as.list(RHS2DT(arg, data = env, enclos = enc))
-  
-  rhsl
-  
-}
-
-
-method_classes <- function(f) {
-  
-  stopifnot(is.character(f))
-  e <- utils::methods(f)
-  e <- unlist(lapply(e, as.character))
-  e <- unlist(lapply(e, sub, pattern = paste0(f, "."), replacement = ""))
-  setdiff(e, "default")
-  
-}
-
-
-do_evalPopArg <- function(arg, env, enc) {
-  eam <- method_classes("evalArg")
-  
-  ne <- list(env = env, enc = enc)
-  de <- list(env = enc, enc = baseenv())
-  
-  r <- arg
-  tick <- 1L
-  while (any(class(r) %in% eam)) {
-    
-    envs <-  if (is_dollar_expression(r)) de else ne
-    r <- evalArg(arg = r, env = envs$env, enc = envs$enc)
-    
-    tick <- tick + 1L
-    if (tick == 100L) stop("No result after 100 evaluations")
-  }
-  
-  r
-}
-
-
-
-argType <- function(arg) {
-  
-  tl <- list("NULL" = "NULL", character = "character",
-             list = "call", formula = "formula", 
-             expression = c("call", "name"))
-  
-  tl <- sapply(tl, function(ch) {
-    t <- tryCatch(inherits(arg, ch), 
-                  error = function(e) e,
-                  warning = function(w) w)
-    isTRUE(t)
-  })
-  if (!any(tl)) tl[names(tl) == "expression"] <- TRUE
-  if (tl["list"]) tl["list"] <- substr(deparse(arg), 1, 5) == "list("
-  names(tl)[tl & !duplicated(tl)]
-  
-}
-
-
-
-
-
-evalPopArg2 <- function(data, arg, enclos, DT = TRUE, 
-                        types = c("NULL","character", "list", "expression")) {
-  
-  allowed_types <- c("NULL", "character", "list", "expression", "formula")
-  types <- match.arg(types, allowed_types, 
-                     several.ok = TRUE)
-  if (!argType(arg) %in% types) {
-    stop("Supplied argument not allowed type. Current type: ",
-         argType(arg), ". Allowed types: ",
-         paste0(types, collapse = ", "))
-  }
-  
-  l <- do_evalPopArg(arg = arg, env = data, enc = enclos)
-  l
-}
-
-
-
-
-evalPopArg <- function(
-  data, 
-  arg, 
-  n = 1L, 
-  DT = TRUE, 
-  enclos = NULL, 
-  recursive = TRUE, 
-  types = c("NULL","character", "list", "expression"), 
-  naming = c("DT", "model")
-) {
-  ## arg: an unevaluated AND substitute()'d argument within a function, which may be
-  ## * an expression
-  ## * a list of expressions
-  ## * a character vector of variable names (in a given data set)
-  ## n: steps upstream as in parent.frame(n); 0L refers to calling environment
-  ## of evalPopArg, 1L to calling environment of e.g. sir which uses evalPopArg, etc.
-  ## hence n = 1L should be almost always the right way to go.
-  ## ALTERNATIVELY supply an environment by hand via enclos.
-  ## enclos will override n.
-  ## recursive: if TRUE, evals arg as many times as it is of type language.
-  ## output:
-  ## * vector as a result of an expression
-  ## * list as a result of a list
-  ## * character vector of names
-  ## OR with DT = TRUE, a data.table based on aforementioned results.
-  ## intention: output to be used in by argument of data.table.
-  ## a data.table output is directly usable in by.
-  ## if column names cannot be easily found, BV1, BV2, ... are imputed
-  ## for missing names (unrobustly: such names may already exist, resulting in duplicates)
-  
-  ## naming: DT style uses first element of all.names() where 
-  ## a name has to be created; model style keeps the whole deparsed
-  ## expression. Only applied when DT = TRUE
-  naming <- match.arg(naming[1L], c("DT", "model"))
-  
-  ## types: allowed popArg types of arguments.
-  types <- match.arg(types, c("NULL","character", "list", "expression", "formula"), several.ok = TRUE)
-  
-  if (!is.null(enclos) && !is.environment(enclos)) {
-    stop("enclos must be NULL or an environment")
-  }
-  if (!is.environment(enclos)) enclos <- parent.frame(n + 1L)
-  
-  ## used data may change if expression uses dollar operator, hence
-  ## arg should not be evaluated within data but only its surroundings.
-  use_data <- data
-  use_enc <- enclos
-  dataNames <- names(data)
-  
-  if (uses_dollar(arg, data.names = dataNames)) {
-    use_data <- enclos
-    use_enc <- baseenv()
-  }
-  e <- eval(arg, envir = use_data, enclos = use_enc)
-  if (is.language(e) && !inherits(e, "formula")) {
-    if (!recursive) stop("arg is of type language after evaluating, and recursive = FALSE")
-    
-    tick <- 1L
-    while (is.language(e) && !inherits(e, "formula") && tick < 100L) {
-      arg <- e
-      use_data <- data
-      use_enc <- enclos
-      if (uses_dollar(arg, data.names = dataNames)) {
-        use_data <- enclos
-        use_enc <- baseenv()
-      }
-      e <- eval(arg, envir = use_data, enclos = use_enc)
-      tick <- tick + 1L
-    }
-    if (tick == 100L) stop("arg was of type language even after 100 evaluations. Something went wrong here...")
-    
-    
-    
-  } 
-  argType <- "NULL"
-  if (is.list(e)) argType <- "list" else 
-    if (is.character(e)) argType <- "character" else 
-      if (mode(e) == "numeric" || is.vector(e) || is.factor(e)) argType <- "expression" else 
-        if (inherits(e, "formula")) argType <- "formula"
-  
-  if (!argType %in% types) {
-    stop(
-      "popArg type of evaluated arg not one of the allowed types ",
-      "(set via argument types). Detected type: '", deparse(argType), 
-      "'. Allowed types: ", deparse(types))
-  }
-  if (argType == "NULL") return(NULL)
-  
-  av <- all.vars(arg)
-  if (argType == "character") av <- e
-  
-  ## byNames: names of columns resulting from aggre argument, by which
-  ## pyrs and such are aggregated. same functionality
-  ## as in results seen in e.g.
-  ## dt[, .N, by = list(factor(x), y, z = w)] ## factor, y, z
-  ## note: first object in ags with list or expression aggre is "list"
-  byNames <- NULL
-  arg_substr5 <- substr(paste0(deparse(arg), collapse = " "), 1, 5)
-  if (is.character(e)) {
-    byNames <- e
-  } else if (argType == "list" && arg_substr5 == "list(") {
-    byNames <- sapply(arg[-1], function(x) all.names(x)[1]) 
-  } else if (argType == "expression") {
-    byNames <- all.names(arg)[1]
-  }
-  
-  badNames <- c("$", ":")
-  
-  byNames[byNames %in% badNames] <- paste0("BV", 1:length(byNames))[byNames %in% badNames]
-  
-  if (argType == "formula") {
-    arg <- e
-    use_data <- data
-    use_enc <- enclos
-    e <- RHS2DT(formula = e, data = use_data, enclos = use_enc)
-    if (ncol(e) == 0L || nrow(e) == 0L) e <- data.table() ## e.g. y ~ 1
-    
-  } else if (is.character(e)) {
-    all_names_present(data, e)
-    if (DT) {
-      ## note: e contains variable names in character strings,
-      ## ergo fully named list & DT created
-      l <- lapply(e, function(x) data[[x]])
-      setattr(l, "names", e)
-      setDT(l)
-      e <- l; rm(l)
-    }
-  } else if (is.list(e)) {
-    ## note: fully unnamed list has NULL names()
-    ## partially named list has some "" names
-    ne <- names(e)
-    
-    if (DT && any(sapply(e, is.null))) stop("at least one object in list arg is NULL; cannot form data.table with such list")
-    
-    if (is.null(ne)) ne <- rep("", length(e))
-    
-    
-    wh_bad <- which(ne == "")
-    if (length(wh_bad) > 0) {
-      if (is.null(byNames)) {
-        byNames <- paste0("BV", 1:length(e))
-      }
-      
-      ne[wh_bad] <- byNames[wh_bad]
-      setattr(e, "names", ne)
-    }
-    
-    if (DT) {
-      ## NOTE: used to be setDT, but length of different elements
-      ## in list may differ, which as.data.table handles correctly
-      e <- as.data.table(e)
-    }
-  } else if (mode(e) == "numeric" || is.vector(e) || is.factor(e)) {
-    ## is e.g. a numeric vector or a factor
-    if (DT) {
-      e <- data.table(V1 = e)
-      setnames(e, 1, byNames)
-    }
-  }
-  
-  ## NOTE: e may be of type language at this point if arg was double-quoted
-  ## and recursive = FALSE
-  
-  if (DT) {
-    setDT(e)
-    setattr(e, "all.vars", av)
-    setattr(e, "quoted.arg", arg)
-    setattr(e, "arg.type", argType)
-    if (naming == "model" && ncol(e) > 0L) setnames(e, 1:ncol(e), popArg2ModelNames(arg, type = argType))
-  }
-  e
-}
-
-
-popArgType <- function(arg, data = NULL, n = 1L, enclos = NULL, recursive = TRUE) {
-  ## input: a substitute()'d expression / argument
-  ## NOTE: recursive useful when arg might be quoted twice and want the eventual
-  ## result; need to supply data for it though
-  ## output: type of thingie that was substitute()'d
-  ##  * list (of expressions)
-  ##  * character string vector
-  ##  * an expression (includes symbol)
-  av <- all.vars(arg, unique = TRUE) ## all variables
-  av <- setdiff(av, c("$", "T", "F"))
-  an <- all.names(arg, unique = TRUE) ## all variables and functions
-  af <- setdiff(an, av) ## all functions used
-  
-  a <- deparse(arg)
-  a <- paste0(a, collapse = "") ## lists may somehow produce length > 1 here
-  if (substr(a, 1, 5) == "list(") return("list")
-  if (a == "NULL") return("NULL")
-  ## detection of character arguments is not easy and should not be considered
-  ## fool proof since user may pass e.g. a vector of character strings as a 
-  ## symbol, which can only really be interpreted as an expression
-  if (sum(grep('\\"', a)) && length(setdiff(af, "c")) == 0) return("character")
-  
-  if (is.data.frame(data)) {
-    if (is.symbol(arg) && a %in% names(data)) return("expression")
-    if (length(av) == 1L && av %in% names(data)) return("expression")
-    e <- eval(arg, envir = data[1:min(nrow(data), 20L), ], 
-              enclos = if (is.environment(enclos)) enclos else parent.frame(n + 1L))
-    if (inherits(e, "formula")) return("formula")
-    if (is.null(e)) return("NULL")
-    if (is.list(e)) return("list")
-    if (is.character(e) && all(e %in% names(data))) return("character")
-    if (is.vector(e) || is.factor(e)) return("expression")
-    
-    if (recursive && is.language(e)) return(popArgType(e, data = data, n = n + 1L, enclos = enclos))
-  }
-  
-  "expression"
-  
-}
-popArg2ModelNames <- function(arg, type) {
-  ## INTENTION: given a quoted/substituted expression,
-  ## digs out the expression(s) creating a/multiple column(s)
-  ## and returns the deparsed expression(s) to be used as names
-  ## of columns the same way that models such as lm() display
-  ## the names of expressions used within formula
-  
-  ## some exceptions
-  if (is.data.frame(arg)) return(names(arg))
-  if (is.character(arg)) return(arg)
-  
-  type <- match.arg(type[1L], c("NULL", "character", "list", "expression", "formula"))
-  
-  lang <- NULL
-  lang <- try(is.language(arg) || inherits(arg, "formula"), silent = TRUE)
-  
-  
-  if (inherits(lang, "try-error") || !lang) stop("arg must be a quoted or substituted expression or a formula. Error message: ", lang, ". type of arg: ", typeof(arg), ". Class: ", class(arg), ". Mode: ", mode(arg), ".")
-  
-  d <- oneWhitespace(paste0(deparse(arg)))
-  
-  if (type == "expression") return(d) else 
-    if (type == "NULL") return(NULL) else 
-      if (type == "character") return(eval(arg)) else 
-        if (type == "list") {
-          d <- substr(d, 6, nchar(d)-1L) ## removes "list(" and ")"
-          d <- strsplit(d, ", ")
-          return(unlist(d))
-        } else if (type == "formula") {
-          arg <- eval(arg)
-          d <- names(RHS2list(arg))
-          if (length(d) == 0L) return(NULL) ## e.g. y ~ 1
-          return(d)
-        }
-  stop("could not determine deparsed-expression-names")
-}
-
-
-
-
-
-uses_dollar <- function(q, data.names) {
-  ## INTENTION: determine whether q is an expressions that is evaluated
-  ## outside a data.frame, i.e. one that uses the dollar operator.
-  ## e.g. TF$V1 should not be evaluated in a data.frame even if it has
-  ## the variables TF and V1 since it wont work and was not intended.
-  if (!is.language(q) || inherits(q, "formula")) {
-    return(FALSE)
-  }
-  
-  d <- deparse(q)
-  ## sometimes d is of length > 1 for some reason...
-  d <- paste0(d, collapse = "")
-  d <- oneWhitespace(d)
-  
-  if (substr(d, 1, 4) == "list") {
-    ## lists are not allowed to work in this manner for now.
-    return(FALSE)
-  }
-  
-  if (!grepl(x = d, pattern = "\\$")) {
-    ## does not use dollar operator.
-    return(FALSE)
-  }
-  
-  ## detect if word$word is used in d
-  t <- regexec(pattern = "\\w+\\$\\w+", text = d)
-  if (t != -1) {
-    ## ok, used word$word
-    ## is there are variable with that name in data.names?
-    m <- unlist(regmatches(d, t))
-    if (m %in% data.names) {
-      return(FALSE)
-    }
-    ## if not, it should be evaluated outside the data.
-    return(TRUE)
-  } 
-  
-  return(FALSE)
-}
-
-
-
-
-`%.:%` <- function(x, y) {
-  ## INTENTION: hacking formula calls using `:`
-  ## which is apparently normally evaluated in C... (in model.matrix.default)
-  ## USAGE: e.g. c(1,2) %.:% c(3,4) = c(3, 8)
-  ## (instead of getting warning)
-  if (length(x) > 1L && length(y) > 1L && is.numeric(x) && is.numeric(y)) {
-    return(x*y)
-  } else if (length(x) == 1L && length(y) == 1L && is.numeric(x) && is.numeric(y)) {
-    return(x:y)
-  }
-  as.factor(x):as.factor(y)
-  
-}
-
-
-
-RHS2list <- function(formula, handle.adjust=TRUE) {
-  ## INTENTION: turns the right-hand side of a formula
-  ## into a list of substituted expressions;
-  ## each element in list is an expressions separated
-  ## by a '+' in the formula. needs to be eval()'d,
-  ## preferably using the appropriate data set.
-  if (!inherits(formula, "formula")) stop("not a formula")
-  
-  ## no response
-  formula <- formula[c(1, length(formula))]
-  
-  te <- terms(formula)
-  tl <- attr(te, "term.labels")
-  
-  ## handle adjusting variables (e.g. adjust(V1, V2) -> c("V1", "V2"))
-  adj <- tl[substr(tl, 1, 7) == "adjust("]
-  if (length(adj) == 0L) adj <- NULL
-  if (handle.adjust && !is.null(adj)) {
-    tl <- setdiff(tl, adj)
-    adj <- unlist(lapply(adj, function(stri) {
-      e <- parse(text = stri)[[1]]
-      e <- as.list(e)[-1]
-      unlist(lapply(e, deparse))
-    }))
-    
-    tl <- c(tl, adj)
-  }
-  
-  ## to avoid e.g. c(1,2):c(3,4) NOT evaluating as c(3, 8)
-  l <- lapply(tl, function(x) gsub(pattern = ":", x = x, replacement = "%.:%"))
-  l <- lapply(l, function(x) parse(text = x)[[1L]])
-  
-  names(l) <- tl
-  
-  setattr(l, "adjust", adj)
-  
-  l
-}
-
-uses_colon <- function(e) {
-  if (is.character(e)) e <- parse(text = e)[[1]]
-  stopifnot(is.call(e) || is.name(e))
-  stopifnot(!inherits(e, "formula"))
-  
-  l <- as.list(e)
-  if (deparse(l[[1]]) %in% c(":", "%.:%") && length(l) == 3) {
-    return(TRUE)
-  }
-  FALSE
-}
-
-replace_colon <- function(e) {
-  stopifnot(is.call(e) || is.name(e))
-  stopifnot(!inherits(e, "formula"))
-  
-  if (!uses_colon(e)) return(e)
-  
-  l <- as.list(e)
-  l[[1]] <- parse(text = "popEpi:::`%.:%`")[[1]]
-  as.call(l)
-}
-
-RHS2DT <- function(formula, data = data.frame(), enclos = parent.frame(1L)) {
-  l <- RHS2list(formula)
-  if (length(l) == 0L) return(data.table())
-  adj <- attr(l, "adjust")
-  
-  ## foolproofing in case data contains column named e.g. "a:b" and that is 
-  ## intended to be used
-  dana <- names(data)
-  dana <- gsub(x=dana, pattern=" %.:% ", replacement = ":")
-  dana <- gsub(x=dana, pattern="%.:%", replacement = ":")
-  
-  
-  ld <- lapply(l, deparse)
-  ld <- lapply(ld, function(ch) {
-    ch <- gsub(x=ch, pattern=" %.:% ", replacement = ":")
-    ch <- gsub(x=ch, pattern="%.:%", replacement = ":")
-    int <- if (ch %in% dana) which(dana %in% ch) else ch
-    if (is.integer(int)) data[[names(data)[int]]] else NULL
-  })
-  ld[which(sapply(ld, is.null))] <- NULL
-  l[names(ld)] <- ld
-  
-  ## foolproofs use of function %.:% by explicit referral, i.e. :::
-  ## (this avoids scoping problem)
-  l <- lapply(l, function(elem) {
-    if (!is.call(elem)) return(elem)
-    if (!uses_colon(elem)) return(elem)
-    
-    replace_colon(elem)
-    
-  })
-  
-  l <- lapply(l, function(elem) {
-    eval(expr = elem, envir = data, enclos = enclos)
-  })
-  
-  l <- as.data.table(l)
-  setattr(l, "adjust", adj)
-  l
-}
-
-
-
-
-
-Surv2DT <- function(Surv) {
-  sa <- attributes(Surv)
-  type <- sa$type
-  
-  dt <- as.data.table(lapply(sa$dimnames[[2]], function(col_nm) {
-    Surv[, col_nm]
-  }))
-  setnames(dt, names(dt), sa$dimnames[[2]])
-  
-  statNA <- sum(is.na(dt$status))
-  if (statNA) {
-    stop("Some status indicators (", statNA  ," values in total) were NA. ",
-         "Usual suspects: original status variable ",
-         "has NA values, or you have numeric status variable with more than ",
-         "two levels and you did not assign e.g. type = 'mstate' (e.g. ",
-         "Surv(time = c(1,1,1), event = c(0,1,2), type = 'mstate') works).")
-    
-  }
-  
-  
-  setattr(dt, "type", type)
-  
-  label_sources <- c("time2", "event")
-  lapply(label_sources, function(lbl_src) {
-    if (identical(sa$inputAttributes[[lbl_src]]$class, "factor")) {
-      set(
-        dt, j = "status", 
-        value = factor(dt$status, labels = sa$inputAttributes[[lbl_src]]$levels)
-      )
-    }
-    NULL
-  })
-  
-  dt[]
-}
-
-
-
-
-
-evalPopFormula <- function(formula, data = data.frame(), enclos = parent.frame(2L), subset = NULL, Surv.response = TRUE) {
-  
-  ## INTENTION: given a formula object, returns a DT where each column
-  ## is an evaluated expression from formula (separated by  + )
-  
-  fe <- environment(formula)
-  
-  either <- FALSE
-  if (is.character(Surv.response)) {
-    Surv.response <- match.arg(Surv.response, "either")
-    Surv.response <- TRUE
-    either <- TRUE
-  } else if (!is.logical(Surv.response)) {
-    stop("Surv.response must be either logical or 'either'")
-  }
-  
-  ## subset if needed ----------------------------------------------------------
-  if (!is.null(subset) && !is.logical(subset)) {
-    stop("subset must be NULL or a logical vector and not an expression at ",
-         "this point. If you see this, complain to the package maintainer.")
-  }
-  if (!is.null(subset)) {
-    keepVars <- c(all.vars(formula), "lex.Xst")
-    data <- subsetDTorDF(data, subset = subset, select = keepVars)
-  }
-  
-  
-  ## formula -------------------------------------------------------------------
-  if (!inherits(formula, "formula")) {
-    stop("formula is not of class 'formula'; supply it as e.g. y ~ x")
-  }
-  if (length(formula) < 3L) {
-    stop("formula appears to be one-sided, which is not supported; ",
-         "supply it as e.g. y ~ x")
-  }
-  
-  ## response
-  y <- eval(formula[[2L]], envir = data, enclos = enclos)
-  if (inherits(y, "Surv") && !either && !Surv.response) {
-    stop("Response is a result of using Surv(), which is not allowed in ",
-         "this context.")
-  }
-  
-  if (!inherits(y, "Surv") && !either && Surv.response) {
-    stop("The response of the formula must be a Surv object; ",
-         "see ?Surv (in package survival).")
-  }
-  
-  if (inherits(y, "Surv")) {
-    y <- Surv2DT(y)
-    setcolsnull(y, keep = c("time", "start", "status"), colorder = TRUE)
-    if (!any(c("time", "start") %in% names(y))) {
-      stop("You must supply function Surv a value to the 'time' ",
-           "argument. See ?Surv")
-    }
-    setnames(y, names(y), c("time", "status")[1:ncol(y)])
-  } else {
-    y <- data.table(y)
-    setnames(y, 1, deparse(formula[[2L]]))
-    if (either && inherits(data, "Lexis")) {
-      ## we assume the unmodified lex.Xst to be a useful status variable.
-      if (!"lex.Xst" %in% names(data)) {
-        stop("Supplied a formula without using Surv(), and data was a Lexis ",
-             "object, so assumed you intended to use 'lex.Xst' in data as the ",
-             "status variable in this context, but that column was missing ",
-             "from data.")
-      }
-      setnames(y, 1, "time")
-      y[, "status"] <- data$lex.Xst
-      
-    }
-  }
-  
-  
-  ## RHS
-  l <- RHS2DT(formula, data = data, enclos = enclos)
-  adj <- attr(l, "adjust")
-  
-  ## combine
-  l <- if (length(l) > 0L) cbind(y, l) else y
-  
-  setattr(l, "adjust.names", adj)
-  setattr(l, "print.names", setdiff(names(l), c(adj, names(y))))
-  setattr(l, "Surv.names", names(y))
-  setattr(l, "formula", formula)
-  
-  l
-}
-
-
-evalRecursive <- function(arg, env, enc, max.n = 100L) {
-  ## INTENTION: digs out actual evaluatable value and expression
-  
-  if (missing(env)) env <- environment()
-  if (missing(enc)) enc <- parent.frame(1L)
-  
-  if (is.data.frame(env)) {
-    na <- names(env)
-    env <- env[1:(min(10L, nrow(env))), ]
-    
-    env <- data.frame(env)
-    setattr(env, "names", na)
-    
-  }
-  argSub <- arg
-  
-  tick <- 1L
-  while (!inherits(arg, "try-error") && is.language(arg) && 
-         !inherits(arg, "formula") && tick < max.n) {
-    
-    argSub <- arg
-    arg <- try(eval(argSub, envir = env, enclos = enc), silent = TRUE)
-    
-    tick <- tick + 1L
-  }
-  
-  if (tick == max.n) {
-    stop("evaluated expression ", max.n, 
-         " times and still could not find underlying expression")
-  }
-  if (!is.language(argSub)) argSub <- substitute(arg)
-  list(arg = arg, argSub = argSub, all.vars = all.vars(argSub))
-}
-
-
-usePopFormula <- function(form = NULL, adjust = NULL, data = data.frame(), 
-                          enclos, Surv.response = TRUE) {
-  ## INTENTION: evaluates form and combines with adjust appropriately
-  ## returns a list of the elements dug out from the formula and adjust 
-  ## arguments.
-  # formSub <- substitute(form)
-  al <- evalRecursive(arg = form, env = data, enc = enclos)
-  
-  if (!inherits(al$arg, "formula")) stop("'form' is not a formula object")
-  
-  dt <- evalPopFormula(formula = al$arg, data = data, enclos = enclos, 
-                       Surv.response = Surv.response)
-  adNames <- attr(dt, "adjust.names")
-  prNames <- attr(dt, "print.names")
-  suNames <- attr(dt, "Surv.names")
-  
-  adjust <- evalPopArg(data, adjust, DT = TRUE, recursive = TRUE, 
-                       enclos = new.env(), naming = "model",
-                       types = c("NULL", "character", "list", "expression"))
-  
-  if (is.data.frame(adjust) && (nrow(adjust) == 0L || ncol(adjust) == 0L)) {
-    stop("adjust evaluated to an empty data.frame")
-  }
-  if (!is.null(adjust) && ncol(adjust) > 0L && length(adNames) > 0L) {
-    stop("Cannot both use argument 'adjust' AND use an adjust() term within ",
-         "the formula argument. Please only use one.")
-  }
-  if (is.null(adjust) && length(adNames) > 0L) {
-    adjust <- dt[, .SD, .SDcols = c(adNames)]
-  }
-  
-  
-  print <- NULL
-  if (length(prNames > 0L)) print <- dt[, .SD, .SDcols = eval(prNames)]
-  
-  list(y = dt[, .SD, .SDcols = c(suNames)], 
-       print = print, 
-       adjust = adjust, formula = al$arg)
-}
-
-
-
-
-
-#' @title Adjust Estimates by Categorical Variables
-#' @description This function is only intended to be used within a formula
-#' when supplied to e.g. \code{\link{survtab_ag}} and should not be
-#' used elsewhere. 
-#' @param ... variables to adjust by, e.g. \code{adjust(factor(v1), v2, v3)}
-#' @return Returns a list of promises of the variables supplied which can be
-#' evaluated.
-#' @examples 
-#' 
-#' y ~ x + adjust(z)
-#' @export
-adjust <- function(...) {
-  
-  call <- sys.call(1L)
-  call <- as.list(call)[1L]
-
-  if (deparse(call) %in% c("adjust", "list(adjust)")) {
-    stop("Function adjust() only intended to be used within the formulas of ",
-         "certain functions of package popEpi. See e.g. ?survtab_ag for usage.")
-  }
-  
-  mc <- as.list(match.call())[-1L]
-  if (is.list(mc) && length(mc) == 1) mc <- mc[[1]]
-  mc
-}
-
-
-
-
-
-parse_adjust_formula <- function(f) {
-  
-  env <- environment(f)
-  t <- attr(terms(f), "term.labels")
-  
-  l <- sapply(t, grepl, pattern = "adjust(", fixed = TRUE)
-  
-  a <- t[l]
-  
-  if (!any(l)) return(f)
-  
-  f <- deparse(f)
-  f <- paste0(f, collapse = "")
-  f <- oneWhitespace(f)
-  
-  for (k in seq_along(a)) {
-    f <- sub(x = f, pattern = paste0(" + ", a[k]), replacement = "", fixed = TRUE)
-  }
-  
-  a <- lapply(a, function(stri) {
-    e <- parse(text = stri)[[1]]
-    as.list(e)[-1]
-  })
-  
-  a <- unlist(a, recursive = FALSE)
-  
-  a <- sapply(a, function(e) {
-    
-    e <- substitute(adjust(e), list(e = e))
-    deparse(e)
-    
-  })
-  
-  f <- paste0(f, " + ", paste0(a, collapse = " + "))
-  f <- eval(parse(text = f)[[1]])
-  environment(f) <- env
-  f
-}
-
-
-
-
-
-model_frame_robust <- function(formula, data, enc) {
-  
-  stopifnot(inherits(formula, "formula"))
-  stopifnot(is.environment(enc))
-  
-  pe <- function(x, ...) {
-    eval(parse(text = x), ...)
-  }
-  
-  fo <- formula
-  
-  te <- terms(fo)
-  tl <- attr(te, "term.labels")
-  
-  av <- all.vars(fo)
-  av <- intersect(av, names(data))
-  
-  ## non-interactions as they are
-  l <- lapply(tl, function(stri) {
-    if (stri %in% names(data)) return(data[[stri]])
-    e <- try(pe(stri, envir = data, enclos = enc), silent = TRUE)
-    if (inherits(e, "try-error")) e <- stri
-    e
-  })
-  
-  fa <- attr(te, "factors")
-  
-  whInter <- colnames(fa)[colSums(fa) > 1]
-  
-  whInter <- which(sapply(l, function(elem) {
-    is.character(elem) && length(elem) == 1L && elem %in% whInter
-  }))
-  whVar <- setdiff(seq_along(l), whInter)
-  
-  if (sum(whInter)) {
-    
-    fa <- fa[, whInter, drop = FALSE]
-    interList <- lapply(colnames(fa), function(stri) {
-      
-      e <- parse(text = stri)[[1]]
-      e <- as.list(e)[-1]
-      e <- sapply(e, deparse)
-      
-    })
-    
-    names(interList) <- colnames(fa)
-    
-    interVars <- unique(unlist(interList))
-    
-    
-    interData <- lapply(interVars, function(stri) {
-      if (stri %in% names(data)) return(data[[stri]])
-      eval(parse(text = stri), envir = data, enclos = enc)
-    })
-    
-    interData <- setDT(lapply(interData, as.factor))
-    names(interData) <- interVars
-    
-    on <- copy(names(interData))
-    tn <- paste0("V", seq_along(interData))
-    names(tn) <- on
-    names(interData) <- tn
-    
-    til <- lapply(interList, function(stri) {
-      tn[stri]
-    })
-    names(til) <- sapply(til, paste0, collapse = ":")
-    
-    
-    
-    l[whInter] <- lapply(seq_along(til), function(i) {
-      
-      tmpExpr <- til[[i]]
-      realExpr <- l[whInter][[i]]
-      e <- try(pe(tmpExpr, envir = interData), silent = TRUE)
-      
-      if (inherits(e, "try-error")) {
-        e <- pe(realExpr, envir = data, enclos = enc)
-      }
-      e
-    })
-    
-  }
-  
-  names(l) <- tl
-  l <- as.data.table(l)
-  l
-}
-
-
-
-
-
-dt_robust_by <- function(e, by.var.nms) {
-  stopifnot(
-    length(e) == 1,
-    is.character(e),
-    grepl(x = e, pattern = "by\\s{0,}=\\s{0,}%%BY_VAR_NMS%%"),
-    length(by.var.nms) == 0 || is.character(by.var.nms)
-  )
-  
-  le <- paste0("list(", paste0("`", by.var.nms, "`", collapse = ", "), ")")
-  if (!length(by.var.nms)) le <- "NULL"
-  
-  e <- gsub(x = e, pattern = "%%BY_VAR_NMS%%", fixed = TRUE, replacement = le)
-  eval(parse(text = e), envir = parent.frame(1L))
-  NULL
-}
-
-
-
-
-
-
-
-
-
+
+
+is_expression <- function(e) {
+  
+  ifelse(any(c("call", "name") %in% class(e)), TRUE, FALSE)
+  
+}
+
+is_list_expression <- function(e) {
+  
+  ifelse(is_expression(e) && deparse(as.list(e)[[1]]) == "list", TRUE, FALSE)
+  
+}
+
+is_dollar_expression <- function(e) {
+  
+  ifelse(is_expression(e) && deparse(as.list(e)[[1]]) == "$", TRUE, FALSE)
+  
+}
+
+
+is_variable <- function(x) {
+  
+  varModes <- c("numeric", "complex", "logical", "character", "raw")
+  
+  ifelse(mode(x) %in% varModes, TRUE, FALSE)
+  
+}
+
+
+evalArg <- function(arg, env, enc, ...) {
+  UseMethod("evalArg")
+}
+
+
+evalArg.default <- function(arg, env, enc) {
+  if (is.list(arg)) {
+    arg <- as.list(arg)
+  } else if (is_variable(arg)) {
+    l <- list(arg)
+  }
+  return(arg)
+}
+
+evalArg.name <- function(arg, env, enc) {
+  d <- deparse(arg)
+  out <- try(get(d, envir = env, inherits = FALSE), silent = TRUE)
+  if (inherits(out, "try-error")) {
+    out <- try(get(d, envir = enc, inherits = FALSE), silent = TRUE)
+  }
+  if (inherits(out, "try-error")) {
+    stop("Could not find object ", d, ".")
+  }
+  out <- list(out)
+  names(out) <- d
+  out
+}
+
+evalArg.call <- function(arg, env, enc) {
+  
+  out <- eval(arg, envir = env, enclos = enc)
+  
+  if (is.list(out)) {
+    out <- as.list(out)
+  } else if (is_variable(out)) {
+    out <- list(out)
+    names(out) <- paste0(deparse(arg), collapse = "")
+  }
+  out
+}
+
+evalArg.character <- function(arg, env, enc) {
+  ## NOTE: enc unused
+  se <- substitute(env)
+  out <- lapply(arg, function(stri) {
+    try({
+      get(stri, envir = as.environment(env), inherits = FALSE)
+    }, silent = TRUE)
+  })
+  
+  notFound <- arg[sapply(out, inherits, "try-error")]
+  
+  if (length(notFound)) {
+    if (length(notFound) > 5) notFound <- notFound[1:5]
+    stop("Could not find object(s): ", 
+         paste0(notFound, collapse = ", "), ".")
+  }
+  names(out) <- arg
+  out
+}
+
+evalArg.formula <- function(arg, env, enc) {
+  
+  rhsl <- as.list(RHS2DT(arg, data = env, enclos = enc))
+  
+  rhsl
+  
+}
+
+
+method_classes <- function(f) {
+  
+  stopifnot(is.character(f))
+  e <- utils::methods(f)
+  e <- unlist(lapply(e, as.character))
+  e <- unlist(lapply(e, sub, pattern = paste0(f, "."), replacement = ""))
+  setdiff(e, "default")
+  
+}
+
+
+do_evalPopArg <- function(arg, env, enc) {
+  eam <- method_classes("evalArg")
+  
+  ne <- list(env = env, enc = enc)
+  de <- list(env = enc, enc = baseenv())
+  
+  r <- arg
+  tick <- 1L
+  while (any(class(r) %in% eam)) {
+    
+    envs <-  if (is_dollar_expression(r)) de else ne
+    r <- evalArg(arg = r, env = envs$env, enc = envs$enc)
+    
+    tick <- tick + 1L
+    if (tick == 100L) stop("No result after 100 evaluations")
+  }
+  
+  r
+}
+
+
+
+argType <- function(arg) {
+  
+  tl <- list("NULL" = "NULL", character = "character",
+             list = "call", formula = "formula", 
+             expression = c("call", "name"))
+  
+  tl <- sapply(tl, function(ch) {
+    t <- tryCatch(inherits(arg, ch), 
+                  error = function(e) e,
+                  warning = function(w) w)
+    isTRUE(t)
+  })
+  if (!any(tl)) tl[names(tl) == "expression"] <- TRUE
+  if (tl["list"]) tl["list"] <- substr(deparse(arg), 1, 5) == "list("
+  names(tl)[tl & !duplicated(tl)]
+  
+}
+
+
+
+
+
+evalPopArg2 <- function(data, arg, enclos, DT = TRUE, 
+                        types = c("NULL","character", "list", "expression")) {
+  
+  allowed_types <- c("NULL", "character", "list", "expression", "formula")
+  types <- match.arg(types, allowed_types, 
+                     several.ok = TRUE)
+  if (!argType(arg) %in% types) {
+    stop("Supplied argument not allowed type. Current type: ",
+         argType(arg), ". Allowed types: ",
+         paste0(types, collapse = ", "))
+  }
+  
+  l <- do_evalPopArg(arg = arg, env = data, enc = enclos)
+  l
+}
+
+
+
+
+evalPopArg <- function(
+  data, 
+  arg, 
+  n = 1L, 
+  DT = TRUE, 
+  enclos = NULL, 
+  recursive = TRUE, 
+  types = c("NULL","character", "list", "expression"), 
+  naming = c("DT", "model")
+) {
+  ## arg: an unevaluated AND substitute()'d argument within a function, which may be
+  ## * an expression
+  ## * a list of expressions
+  ## * a character vector of variable names (in a given data set)
+  ## n: steps upstream as in parent.frame(n); 0L refers to calling environment
+  ## of evalPopArg, 1L to calling environment of e.g. sir which uses evalPopArg, etc.
+  ## hence n = 1L should be almost always the right way to go.
+  ## ALTERNATIVELY supply an environment by hand via enclos.
+  ## enclos will override n.
+  ## recursive: if TRUE, evals arg as many times as it is of type language.
+  ## output:
+  ## * vector as a result of an expression
+  ## * list as a result of a list
+  ## * character vector of names
+  ## OR with DT = TRUE, a data.table based on aforementioned results.
+  ## intention: output to be used in by argument of data.table.
+  ## a data.table output is directly usable in by.
+  ## if column names cannot be easily found, BV1, BV2, ... are imputed
+  ## for missing names (unrobustly: such names may already exist, resulting in duplicates)
+  
+  ## naming: DT style uses first element of all.names() where 
+  ## a name has to be created; model style keeps the whole deparsed
+  ## expression. Only applied when DT = TRUE
+  naming <- match.arg(naming[1L], c("DT", "model"))
+  
+  ## types: allowed popArg types of arguments.
+  types <- match.arg(types, c("NULL","character", "list", "expression", "formula"), several.ok = TRUE)
+  
+  if (!is.null(enclos) && !is.environment(enclos)) {
+    stop("enclos must be NULL or an environment")
+  }
+  if (!is.environment(enclos)) enclos <- parent.frame(n + 1L)
+  
+  ## used data may change if expression uses dollar operator, hence
+  ## arg should not be evaluated within data but only its surroundings.
+  use_data <- data
+  use_enc <- enclos
+  dataNames <- names(data)
+  
+  if (uses_dollar(arg, data.names = dataNames)) {
+    use_data <- enclos
+    use_enc <- baseenv()
+  }
+  e <- eval(arg, envir = use_data, enclos = use_enc)
+  if (is.language(e) && !inherits(e, "formula")) {
+    if (!recursive) stop("arg is of type language after evaluating, and recursive = FALSE")
+    
+    tick <- 1L
+    while (is.language(e) && !inherits(e, "formula") && tick < 100L) {
+      arg <- e
+      use_data <- data
+      use_enc <- enclos
+      if (uses_dollar(arg, data.names = dataNames)) {
+        use_data <- enclos
+        use_enc <- baseenv()
+      }
+      e <- eval(arg, envir = use_data, enclos = use_enc)
+      tick <- tick + 1L
+    }
+    if (tick == 100L) stop("arg was of type language even after 100 evaluations. Something went wrong here...")
+    
+    
+    
+  } 
+  argType <- "NULL"
+  if (is.list(e)) argType <- "list" else 
+    if (is.character(e)) argType <- "character" else 
+      if (mode(e) == "numeric" || is.vector(e) || is.factor(e)) argType <- "expression" else 
+        if (inherits(e, "formula")) argType <- "formula"
+  
+  if (!argType %in% types) {
+    stop(
+      "popArg type of evaluated arg not one of the allowed types ",
+      "(set via argument types). Detected type: '", deparse(argType), 
+      "'. Allowed types: ", deparse(types))
+  }
+  if (argType == "NULL") return(NULL)
+  
+  av <- all.vars(arg)
+  if (argType == "character") av <- e
+  
+  ## byNames: names of columns resulting from aggre argument, by which
+  ## pyrs and such are aggregated. same functionality
+  ## as in results seen in e.g.
+  ## dt[, .N, by = list(factor(x), y, z = w)] ## factor, y, z
+  ## note: first object in ags with list or expression aggre is "list"
+  byNames <- NULL
+  arg_substr5 <- substr(paste0(deparse(arg), collapse = " "), 1, 5)
+  if (is.character(e)) {
+    byNames <- e
+  } else if (argType == "list" && arg_substr5 == "list(") {
+    byNames <- sapply(arg[-1], function(x) all.names(x)[1]) 
+  } else if (argType == "expression") {
+    byNames <- all.names(arg)[1]
+  }
+  
+  badNames <- c("$", ":")
+  
+  byNames[byNames %in% badNames] <- paste0("BV", 1:length(byNames))[byNames %in% badNames]
+  
+  if (argType == "formula") {
+    arg <- e
+    use_data <- data
+    use_enc <- enclos
+    e <- RHS2DT(formula = e, data = use_data, enclos = use_enc)
+    if (ncol(e) == 0L || nrow(e) == 0L) e <- data.table() ## e.g. y ~ 1
+    
+  } else if (is.character(e)) {
+    all_names_present(data, e)
+    if (DT) {
+      ## note: e contains variable names in character strings,
+      ## ergo fully named list & DT created
+      l <- lapply(e, function(x) data[[x]])
+      setattr(l, "names", e)
+      setDT(l)
+      e <- l; rm(l)
+    }
+  } else if (is.list(e)) {
+    ## note: fully unnamed list has NULL names()
+    ## partially named list has some "" names
+    ne <- names(e)
+    
+    if (DT && any(sapply(e, is.null))) stop("at least one object in list arg is NULL; cannot form data.table with such list")
+    
+    if (is.null(ne)) ne <- rep("", length(e))
+    
+    
+    wh_bad <- which(ne == "")
+    if (length(wh_bad) > 0) {
+      if (is.null(byNames)) {
+        byNames <- paste0("BV", 1:length(e))
+      }
+      
+      ne[wh_bad] <- byNames[wh_bad]
+      setattr(e, "names", ne)
+    }
+    
+    if (DT) {
+      ## NOTE: used to be setDT, but length of different elements
+      ## in list may differ, which as.data.table handles correctly
+      e <- as.data.table(e)
+    }
+  } else if (mode(e) == "numeric" || is.vector(e) || is.factor(e)) {
+    ## is e.g. a numeric vector or a factor
+    if (DT) {
+      e <- data.table(V1 = e)
+      setnames(e, 1, byNames)
+    }
+  }
+  
+  ## NOTE: e may be of type language at this point if arg was double-quoted
+  ## and recursive = FALSE
+  
+  if (DT) {
+    setDT(e)
+    setattr(e, "all.vars", av)
+    setattr(e, "quoted.arg", arg)
+    setattr(e, "arg.type", argType)
+    if (naming == "model" && ncol(e) > 0L) setnames(e, 1:ncol(e), popArg2ModelNames(arg, type = argType))
+  }
+  e
+}
+
+
+popArgType <- function(arg, data = NULL, n = 1L, enclos = NULL, recursive = TRUE) {
+  ## input: a substitute()'d expression / argument
+  ## NOTE: recursive useful when arg might be quoted twice and want the eventual
+  ## result; need to supply data for it though
+  ## output: type of thingie that was substitute()'d
+  ##  * list (of expressions)
+  ##  * character string vector
+  ##  * an expression (includes symbol)
+  av <- all.vars(arg, unique = TRUE) ## all variables
+  av <- setdiff(av, c("$", "T", "F"))
+  an <- all.names(arg, unique = TRUE) ## all variables and functions
+  af <- setdiff(an, av) ## all functions used
+  
+  a <- deparse(arg)
+  a <- paste0(a, collapse = "") ## lists may somehow produce length > 1 here
+  if (substr(a, 1, 5) == "list(") return("list")
+  if (a == "NULL") return("NULL")
+  ## detection of character arguments is not easy and should not be considered
+  ## fool proof since user may pass e.g. a vector of character strings as a 
+  ## symbol, which can only really be interpreted as an expression
+  if (sum(grep('\\"', a)) && length(setdiff(af, "c")) == 0) return("character")
+  
+  if (is.data.frame(data)) {
+    if (is.symbol(arg) && a %in% names(data)) return("expression")
+    if (length(av) == 1L && av %in% names(data)) return("expression")
+    e <- eval(arg, envir = data[1:min(nrow(data), 20L), ], 
+              enclos = if (is.environment(enclos)) enclos else parent.frame(n + 1L))
+    if (inherits(e, "formula")) return("formula")
+    if (is.null(e)) return("NULL")
+    if (is.list(e)) return("list")
+    if (is.character(e) && all(e %in% names(data))) return("character")
+    if (is.vector(e) || is.factor(e)) return("expression")
+    
+    if (recursive && is.language(e)) return(popArgType(e, data = data, n = n + 1L, enclos = enclos))
+  }
+  
+  "expression"
+  
+}
+popArg2ModelNames <- function(arg, type) {
+  ## INTENTION: given a quoted/substituted expression,
+  ## digs out the expression(s) creating a/multiple column(s)
+  ## and returns the deparsed expression(s) to be used as names
+  ## of columns the same way that models such as lm() display
+  ## the names of expressions used within formula
+  
+  ## some exceptions
+  if (is.data.frame(arg)) return(names(arg))
+  if (is.character(arg)) return(arg)
+  
+  type <- match.arg(type[1L], c("NULL", "character", "list", "expression", "formula"))
+  
+  lang <- NULL
+  lang <- try(is.language(arg) || inherits(arg, "formula"), silent = TRUE)
+  
+  
+  if (inherits(lang, "try-error") || !lang) stop("arg must be a quoted or substituted expression or a formula. Error message: ", lang, ". type of arg: ", typeof(arg), ". Class: ", class(arg), ". Mode: ", mode(arg), ".")
+  
+  d <- oneWhitespace(paste0(deparse(arg)))
+  
+  if (type == "expression") return(d) else 
+    if (type == "NULL") return(NULL) else 
+      if (type == "character") return(eval(arg)) else 
+        if (type == "list") {
+          d <- substr(d, 6, nchar(d)-1L) ## removes "list(" and ")"
+          d <- strsplit(d, ", ")
+          return(unlist(d))
+        } else if (type == "formula") {
+          arg <- eval(arg)
+          d <- names(RHS2list(arg))
+          if (length(d) == 0L) return(NULL) ## e.g. y ~ 1
+          return(d)
+        }
+  stop("could not determine deparsed-expression-names")
+}
+
+
+
+
+
+uses_dollar <- function(q, data.names) {
+  ## INTENTION: determine whether q is an expressions that is evaluated
+  ## outside a data.frame, i.e. one that uses the dollar operator.
+  ## e.g. TF$V1 should not be evaluated in a data.frame even if it has
+  ## the variables TF and V1 since it wont work and was not intended.
+  if (!is.language(q) || inherits(q, "formula")) {
+    return(FALSE)
+  }
+  
+  d <- deparse(q)
+  ## sometimes d is of length > 1 for some reason...
+  d <- paste0(d, collapse = "")
+  d <- oneWhitespace(d)
+  
+  if (substr(d, 1, 4) == "list") {
+    ## lists are not allowed to work in this manner for now.
+    return(FALSE)
+  }
+  
+  if (!grepl(x = d, pattern = "\\$")) {
+    ## does not use dollar operator.
+    return(FALSE)
+  }
+  
+  ## detect if word$word is used in d
+  t <- regexec(pattern = "\\w+\\$\\w+", text = d)
+  if (t != -1) {
+    ## ok, used word$word
+    ## is there are variable with that name in data.names?
+    m <- unlist(regmatches(d, t))
+    if (m %in% data.names) {
+      return(FALSE)
+    }
+    ## if not, it should be evaluated outside the data.
+    return(TRUE)
+  } 
+  
+  return(FALSE)
+}
+
+
+
+
+`%.:%` <- function(x, y) {
+  ## INTENTION: hacking formula calls using `:`
+  ## which is apparently normally evaluated in C... (in model.matrix.default)
+  ## USAGE: e.g. c(1,2) %.:% c(3,4) = c(3, 8)
+  ## (instead of getting warning)
+  if (length(x) > 1L && length(y) > 1L && is.numeric(x) && is.numeric(y)) {
+    return(x*y)
+  } else if (length(x) == 1L && length(y) == 1L && is.numeric(x) && is.numeric(y)) {
+    return(x:y)
+  }
+  as.factor(x):as.factor(y)
+  
+}
+
+
+
+RHS2list <- function(formula, handle.adjust=TRUE) {
+  ## INTENTION: turns the right-hand side of a formula
+  ## into a list of substituted expressions;
+  ## each element in list is an expressions separated
+  ## by a '+' in the formula. needs to be eval()'d,
+  ## preferably using the appropriate data set.
+  if (!inherits(formula, "formula")) stop("not a formula")
+  
+  ## no response
+  formula <- formula[c(1, length(formula))]
+  
+  te <- terms(formula)
+  tl <- attr(te, "term.labels")
+  
+  ## handle adjusting variables (e.g. adjust(V1, V2) -> c("V1", "V2"))
+  adj <- tl[substr(tl, 1, 7) == "adjust("]
+  if (length(adj) == 0L) adj <- NULL
+  if (handle.adjust && !is.null(adj)) {
+    tl <- setdiff(tl, adj)
+    adj <- unlist(lapply(adj, function(stri) {
+      e <- parse(text = stri)[[1]]
+      e <- as.list(e)[-1]
+      unlist(lapply(e, deparse))
+    }))
+    
+    tl <- c(tl, adj)
+  }
+  
+  ## to avoid e.g. c(1,2):c(3,4) NOT evaluating as c(3, 8)
+  l <- lapply(tl, function(x) gsub(pattern = ":", x = x, replacement = "%.:%"))
+  l <- lapply(l, function(x) parse(text = x)[[1L]])
+  
+  names(l) <- tl
+  
+  setattr(l, "adjust", adj)
+  
+  l
+}
+
+uses_colon <- function(e) {
+  if (is.character(e)) e <- parse(text = e)[[1]]
+  stopifnot(is.call(e) || is.name(e))
+  stopifnot(!inherits(e, "formula"))
+  
+  l <- as.list(e)
+  if (deparse(l[[1]]) %in% c(":", "%.:%") && length(l) == 3) {
+    return(TRUE)
+  }
+  FALSE
+}
+
+replace_colon <- function(e) {
+  stopifnot(is.call(e) || is.name(e))
+  stopifnot(!inherits(e, "formula"))
+  
+  if (!uses_colon(e)) return(e)
+  
+  l <- as.list(e)
+  l[[1]] <- parse(text = "popEpi:::`%.:%`")[[1]]
+  as.call(l)
+}
+
+RHS2DT <- function(formula, data = data.frame(), enclos = parent.frame(1L)) {
+  l <- RHS2list(formula)
+  if (length(l) == 0L) return(data.table())
+  adj <- attr(l, "adjust")
+  
+  ## foolproofing in case data contains column named e.g. "a:b" and that is 
+  ## intended to be used
+  dana <- names(data)
+  dana <- gsub(x=dana, pattern=" %.:% ", replacement = ":")
+  dana <- gsub(x=dana, pattern="%.:%", replacement = ":")
+  
+  
+  ld <- lapply(l, deparse)
+  ld <- lapply(ld, function(ch) {
+    ch <- gsub(x=ch, pattern=" %.:% ", replacement = ":")
+    ch <- gsub(x=ch, pattern="%.:%", replacement = ":")
+    int <- if (ch %in% dana) which(dana %in% ch) else ch
+    if (is.integer(int)) data[[names(data)[int]]] else NULL
+  })
+  ld[which(sapply(ld, is.null))] <- NULL
+  l[names(ld)] <- ld
+  
+  ## foolproofs use of function %.:% by explicit referral, i.e. :::
+  ## (this avoids scoping problem)
+  l <- lapply(l, function(elem) {
+    if (!is.call(elem)) return(elem)
+    if (!uses_colon(elem)) return(elem)
+    
+    replace_colon(elem)
+    
+  })
+  
+  l <- lapply(l, function(elem) {
+    eval(expr = elem, envir = data, enclos = enclos)
+  })
+  
+  l <- as.data.table(l)
+  setattr(l, "adjust", adj)
+  l
+}
+
+
+
+
+
+Surv2DT <- function(Surv) {
+  sa <- attributes(Surv)
+  type <- sa$type
+  
+  dt <- as.data.table(lapply(sa$dimnames[[2]], function(col_nm) {
+    Surv[, col_nm]
+  }))
+  setnames(dt, names(dt), sa$dimnames[[2]])
+  
+  statNA <- sum(is.na(dt$status))
+  if (statNA) {
+    stop("Some status indicators (", statNA  ," values in total) were NA. ",
+         "Usual suspects: original status variable ",
+         "has NA values, or you have numeric status variable with more than ",
+         "two levels and you did not assign e.g. type = 'mstate' (e.g. ",
+         "Surv(time = c(1,1,1), event = c(0,1,2), type = 'mstate') works).")
+    
+  }
+  
+  
+  setattr(dt, "type", type)
+  
+  label_sources <- c("time2", "event")
+  lapply(label_sources, function(lbl_src) {
+    if (identical(sa$inputAttributes[[lbl_src]]$class, "factor")) {
+      set(
+        dt, j = "status", 
+        value = factor(dt$status, labels = sa$inputAttributes[[lbl_src]]$levels)
+      )
+    }
+    NULL
+  })
+  
+  dt[]
+}
+
+
+
+
+
+evalPopFormula <- function(formula, data = data.frame(), enclos = parent.frame(2L), subset = NULL, Surv.response = TRUE) {
+  
+  ## INTENTION: given a formula object, returns a DT where each column
+  ## is an evaluated expression from formula (separated by  + )
+  
+  fe <- environment(formula)
+  
+  either <- FALSE
+  if (is.character(Surv.response)) {
+    Surv.response <- match.arg(Surv.response, "either")
+    Surv.response <- TRUE
+    either <- TRUE
+  } else if (!is.logical(Surv.response)) {
+    stop("Surv.response must be either logical or 'either'")
+  }
+  
+  ## subset if needed ----------------------------------------------------------
+  if (!is.null(subset) && !is.logical(subset)) {
+    stop("subset must be NULL or a logical vector and not an expression at ",
+         "this point. If you see this, complain to the package maintainer.")
+  }
+  if (!is.null(subset)) {
+    keepVars <- c(all.vars(formula), "lex.Xst")
+    data <- subsetDTorDF(data, subset = subset, select = keepVars)
+  }
+  
+  
+  ## formula -------------------------------------------------------------------
+  if (!inherits(formula, "formula")) {
+    stop("formula is not of class 'formula'; supply it as e.g. y ~ x")
+  }
+  if (length(formula) < 3L) {
+    stop("formula appears to be one-sided, which is not supported; ",
+         "supply it as e.g. y ~ x")
+  }
+  
+  ## response
+  y <- eval(formula[[2L]], envir = data, enclos = enclos)
+  if (inherits(y, "Surv") && !either && !Surv.response) {
+    stop("Response is a result of using Surv(), which is not allowed in ",
+         "this context.")
+  }
+  
+  if (!inherits(y, "Surv") && !either && Surv.response) {
+    stop("The response of the formula must be a Surv object; ",
+         "see ?Surv (in package survival).")
+  }
+  
+  if (inherits(y, "Surv")) {
+    y <- Surv2DT(y)
+    setcolsnull(y, keep = c("time", "start", "status"), colorder = TRUE)
+    if (!any(c("time", "start") %in% names(y))) {
+      stop("You must supply function Surv a value to the 'time' ",
+           "argument. See ?Surv")
+    }
+    setnames(y, names(y), c("time", "status")[1:ncol(y)])
+  } else {
+    y <- data.table(y)
+    setnames(y, 1, deparse(formula[[2L]]))
+    if (either && inherits(data, "Lexis")) {
+      ## we assume the unmodified lex.Xst to be a useful status variable.
+      if (!"lex.Xst" %in% names(data)) {
+        stop("Supplied a formula without using Surv(), and data was a Lexis ",
+             "object, so assumed you intended to use 'lex.Xst' in data as the ",
+             "status variable in this context, but that column was missing ",
+             "from data.")
+      }
+      setnames(y, 1, "time")
+      y[, "status"] <- data$lex.Xst
+      
+    }
+  }
+  
+  
+  ## RHS
+  l <- RHS2DT(formula, data = data, enclos = enclos)
+  adj <- attr(l, "adjust")
+  
+  ## combine
+  l <- if (length(l) > 0L) cbind(y, l) else y
+  
+  setattr(l, "adjust.names", adj)
+  setattr(l, "print.names", setdiff(names(l), c(adj, names(y))))
+  setattr(l, "Surv.names", names(y))
+  setattr(l, "formula", formula)
+  
+  l
+}
+
+
+evalRecursive <- function(arg, env, enc, max.n = 100L) {
+  ## INTENTION: digs out actual evaluatable value and expression
+  
+  if (missing(env)) env <- environment()
+  if (missing(enc)) enc <- parent.frame(1L)
+  
+  if (is.data.frame(env)) {
+    na <- names(env)
+    env <- env[1:(min(10L, nrow(env))), ]
+    
+    env <- data.frame(env)
+    setattr(env, "names", na)
+    
+  }
+  argSub <- arg
+  
+  tick <- 1L
+  while (!inherits(arg, "try-error") && is.language(arg) && 
+         !inherits(arg, "formula") && tick < max.n) {
+    
+    argSub <- arg
+    arg <- try(eval(argSub, envir = env, enclos = enc), silent = TRUE)
+    
+    tick <- tick + 1L
+  }
+  
+  if (tick == max.n) {
+    stop("evaluated expression ", max.n, 
+         " times and still could not find underlying expression")
+  }
+  if (!is.language(argSub)) argSub <- substitute(arg)
+  list(arg = arg, argSub = argSub, all.vars = all.vars(argSub))
+}
+
+
+usePopFormula <- function(form = NULL, adjust = NULL, data = data.frame(), 
+                          enclos, Surv.response = TRUE) {
+  ## INTENTION: evaluates form and combines with adjust appropriately
+  ## returns a list of the elements dug out from the formula and adjust 
+  ## arguments.
+  # formSub <- substitute(form)
+  al <- evalRecursive(arg = form, env = data, enc = enclos)
+  
+  if (!inherits(al$arg, "formula")) stop("'form' is not a formula object")
+  
+  dt <- evalPopFormula(formula = al$arg, data = data, enclos = enclos, 
+                       Surv.response = Surv.response)
+  adNames <- attr(dt, "adjust.names")
+  prNames <- attr(dt, "print.names")
+  suNames <- attr(dt, "Surv.names")
+  
+  adjust <- evalPopArg(data, adjust, DT = TRUE, recursive = TRUE, 
+                       enclos = new.env(), naming = "model",
+                       types = c("NULL", "character", "list", "expression"))
+  
+  if (is.data.frame(adjust) && (nrow(adjust) == 0L || ncol(adjust) == 0L)) {
+    stop("adjust evaluated to an empty data.frame")
+  }
+  if (!is.null(adjust) && ncol(adjust) > 0L && length(adNames) > 0L) {
+    stop("Cannot both use argument 'adjust' AND use an adjust() term within ",
+         "the formula argument. Please only use one.")
+  }
+  if (is.null(adjust) && length(adNames) > 0L) {
+    adjust <- dt[, .SD, .SDcols = c(adNames)]
+  }
+  
+  
+  print <- NULL
+  if (length(prNames > 0L)) print <- dt[, .SD, .SDcols = eval(prNames)]
+  
+  list(y = dt[, .SD, .SDcols = c(suNames)], 
+       print = print, 
+       adjust = adjust, formula = al$arg)
+}
+
+
+
+
+
+#' @title Adjust Estimates by Categorical Variables
+#' @description This function is only intended to be used within a formula
+#' when supplied to e.g. \code{\link{survtab_ag}} and should not be
+#' used elsewhere. 
+#' @param ... variables to adjust by, e.g. \code{adjust(factor(v1), v2, v3)}
+#' @return Returns a list of promises of the variables supplied which can be
+#' evaluated.
+#' @examples 
+#' 
+#' y ~ x + adjust(z)
+#' @export
+adjust <- function(...) {
+  
+  call <- sys.call(1L)
+  call <- as.list(call)[1L]
+
+  if (deparse(call) %in% c("adjust", "list(adjust)")) {
+    stop("Function adjust() only intended to be used within the formulas of ",
+         "certain functions of package popEpi. See e.g. ?survtab_ag for usage.")
+  }
+  
+  mc <- as.list(match.call())[-1L]
+  if (is.list(mc) && length(mc) == 1) mc <- mc[[1]]
+  mc
+}
+
+
+
+
+
+parse_adjust_formula <- function(f) {
+  
+  env <- environment(f)
+  t <- attr(terms(f), "term.labels")
+  
+  l <- sapply(t, grepl, pattern = "adjust(", fixed = TRUE)
+  
+  a <- t[l]
+  
+  if (!any(l)) return(f)
+  
+  f <- deparse(f)
+  f <- paste0(f, collapse = "")
+  f <- oneWhitespace(f)
+  
+  for (k in seq_along(a)) {
+    f <- sub(x = f, pattern = paste0(" + ", a[k]), replacement = "", fixed = TRUE)
+  }
+  
+  a <- lapply(a, function(stri) {
+    e <- parse(text = stri)[[1]]
+    as.list(e)[-1]
+  })
+  
+  a <- unlist(a, recursive = FALSE)
+  
+  a <- sapply(a, function(e) {
+    
+    e <- substitute(adjust(e), list(e = e))
+    deparse(e)
+    
+  })
+  
+  f <- paste0(f, " + ", paste0(a, collapse = " + "))
+  f <- eval(parse(text = f)[[1]])
+  environment(f) <- env
+  f
+}
+
+
+
+
+
+model_frame_robust <- function(formula, data, enc) {
+  
+  stopifnot(inherits(formula, "formula"))
+  stopifnot(is.environment(enc))
+  
+  pe <- function(x, ...) {
+    eval(parse(text = x), ...)
+  }
+  
+  fo <- formula
+  
+  te <- terms(fo)
+  tl <- attr(te, "term.labels")
+  
+  av <- all.vars(fo)
+  av <- intersect(av, names(data))
+  
+  ## non-interactions as they are
+  l <- lapply(tl, function(stri) {
+    if (stri %in% names(data)) return(data[[stri]])
+    e <- try(pe(stri, envir = data, enclos = enc), silent = TRUE)
+    if (inherits(e, "try-error")) e <- stri
+    e
+  })
+  
+  fa <- attr(te, "factors")
+  
+  whInter <- colnames(fa)[colSums(fa) > 1]
+  
+  whInter <- which(sapply(l, function(elem) {
+    is.character(elem) && length(elem) == 1L && elem %in% whInter
+  }))
+  whVar <- setdiff(seq_along(l), whInter)
+  
+  if (sum(whInter)) {
+    
+    fa <- fa[, whInter, drop = FALSE]
+    interList <- lapply(colnames(fa), function(stri) {
+      
+      e <- parse(text = stri)[[1]]
+      e <- as.list(e)[-1]
+      e <- sapply(e, deparse)
+      
+    })
+    
+    names(interList) <- colnames(fa)
+    
+    interVars <- unique(unlist(interList))
+    
+    
+    interData <- lapply(interVars, function(stri) {
+      if (stri %in% names(data)) return(data[[stri]])
+      eval(parse(text = stri), envir = data, enclos = enc)
+    })
+    
+    interData <- setDT(lapply(interData, as.factor))
+    names(interData) <- interVars
+    
+    on <- copy(names(interData))
+    tn <- paste0("V", seq_along(interData))
+    names(tn) <- on
+    names(interData) <- tn
+    
+    til <- lapply(interList, function(stri) {
+      tn[stri]
+    })
+    names(til) <- sapply(til, paste0, collapse = ":")
+    
+    
+    
+    l[whInter] <- lapply(seq_along(til), function(i) {
+      
+      tmpExpr <- til[[i]]
+      realExpr <- l[whInter][[i]]
+      e <- try(pe(tmpExpr, envir = interData), silent = TRUE)
+      
+      if (inherits(e, "try-error")) {
+        e <- pe(realExpr, envir = data, enclos = enc)
+      }
+      e
+    })
+    
+  }
+  
+  names(l) <- tl
+  l <- as.data.table(l)
+  l
+}
+
+
+
+
+
+dt_robust_by <- function(e, by.var.nms) {
+  stopifnot(
+    length(e) == 1,
+    is.character(e),
+    grepl(x = e, pattern = "by\\s{0,}=\\s{0,}%%BY_VAR_NMS%%"),
+    length(by.var.nms) == 0 || is.character(by.var.nms)
+  )
+  
+  le <- paste0("list(", paste0("`", by.var.nms, "`", collapse = ", "), ")")
+  if (!length(by.var.nms)) le <- "NULL"
+  
+  e <- gsub(x = e, pattern = "%%BY_VAR_NMS%%", fixed = TRUE, replacement = le)
+  eval(parse(text = e), envir = parent.frame(1L))
+  NULL
+}
+
+
+
+
+
+
+
+
+
diff --git a/R/flexyargs.R b/R/flexyargs.R
index d3fe6b8..3ef315a 100644
--- a/R/flexyargs.R
+++ b/R/flexyargs.R
@@ -1,166 +1,166 @@
-
-
-
-#' @title Flexible Variable Usage in \pkg{popEpi} Functions
-#' @author Joonas Miettinen
-#' @name flexible_argument
-#' @description Certain arguments in \pkg{popEpi} can be passed in multiple 
-#' ways. This document shows the usage and a pitfall in the
-#' usage of such flexible arguments.
-#' 
-#' @details 
-#' 
-#' Flexible arguments in \pkg{popEpi} are used to pass variables existing
-#' in your data or in the environment where the function is used 
-#' (for everyday users this is the global environment - in simple terms,
-#' where your data is / your work space). The flexible arguments
-#' are modelled after the \code{by} argument in \code{data.tables} - 
-#' see \code{?data.table}. There are many ways to supply the same information
-#' to certain functions in \pkg{popEpi}, but the possible ways listed below
-#' may be limited in some of them to only allow for using only a part of them.
-#' 
-#' @section Everyday usage:
-#' 
-#' Most commonly you may pass
-#' variable names as character strings, e.g.
-#' 
-#' \code{FUN(arg = c("V1", "V2"), data = x)}
-#' 
-#' which may be stored in advance:
-#' 
-#' \code{vars <- c("V1", "V2")}
-#' 
-#' \code{FUN(arg = vars, data = x)}
-#' 
-#' where \code{x} contains those variables. You may also supply variable
-#' names as symbols:
-#' 
-#' \code{FUN(arg = V1, data = x)}
-#' 
-#' Or as a list of symbols (similarly to as in \code{\link{aggregate}}):
-#' 
-#' \code{FUN(arg = list(V1, V2), data = x)}
-#' 
-#' Or as a list of expressions:
-#' 
-#' \code{FUN(arg = list(V1 + 1, factor(V2)), data = x)}
-#' 
-#' A formula without a left-hand-side specified is sometimes allowed as well:
-#' 
-#' \code{FUN(arg = ~ I(V1 + 1) + factor(V2), data = x)}
-#' 
-#' Using a symbol or a list of symbols/expressions typically
-#' causes the function to look for the variable(s)
-#' first in the supplied data (if any) and then where the function was called.
-#' For everyday users this means you might define e.g.
-#' 
-#' \code{V3 <- factor(letters)}
-#' 
-#' and do e.g.
-#' 
-#' \code{FUN(arg = list(V1 + 1, factor(V2), V3), data = x)}
-#' 
-#' provided \code{V1} and \code{V2} exist in \code{x} or in the function calling
-#' environment.
-#' 
-#' @section A pitfall:
-#' 
-#' There is one way to use flexible arguments incorrectly: By supplying
-#' the name of a variable which exists both in the supplied data
-#' and the calling environment, and intending the latter to be used. E.g.
-#' 
-#' \code{vars <- c("V2")}
-#' 
-#' \code{FUN(arg = V3, data = x)}
-#' 
-#' where \code{x} has a column named \code{vars}. This causes the function to
-#' use \code{x$vars} and NOT \code{x$V2}.
-#' 
-#' @section Advanced:
-#' 
-#' Function programmers are advised to pass character strings
-#' whenever possible. To fool-proof against conflicts as described in the
-#' section above, refer to the calling environment explicitly when
-#' passing the variable containing the character strings:
-#' 
-#' \code{TF <- environment() ## current env to refer to}
-#' 
-#' \code{vars <- c("V1", "V2")}
-#' 
-#' \code{FUN(arg = TF$vars, data = x)}
-#' 
-#' Even if \code{x} has columns named \code{vars} and \code{TF}, 
-#' using \code{TF$vars} does not use those columns but only evaluates
-#' \code{TF$vars}
-#' in the calling environment. This is made possible by the fact
-#' that data is always passed as a \code{data.frame}, within which evaluation
-#' of expressions using the dollar operator is not possible. Therefore
-#' it is safe to assume the data should not be used. However, lists of 
-#' expressions will not be checked for dollar use and will fail in conflict
-#' situations:
-#' 
-#' \code{TF <- environment() ## current env to refer to}
-#' 
-#' \code{vars <- letters[1:5]}
-#' 
-#' \code{x <- data.frame(vars = 1:5, TF = 5:1, V1 = 10:6)}
-#' 
-#' \code{FUN(arg = list(TF$vars, V1), data = x)}
-#' 
-#' On the other hand you may typically also pass quoted (\code{\link{quote}})
-#' or substituted \code{\link{substitute}} expressions etc., where
-#' the \code{env$object} trick will work as well:
-#' 
-#' \code{q <- quote(list(vars, V1))}
-#' 
-#' \code{FUN(arg = TF$q, data = x)}
-#' 
-#' This works even with
-#' 
-#' \code{a <- 1:5}
-#' 
-#' \code{V1 <- quote(TF$a)}
-#' 
-#' \code{FUN(arg = TF$V1, data = x)}
-#' 
-#' So no conflicts should occur.
-#' @family popEpi argument evaluation docs
-#' @examples 
-#' 
-#' data(sire)
-#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
-#' ## note: sire is a simulated cohort integrated into popEpi.
-#' BL <- list(fot=seq(0, 5, by = 1/12))
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-#'              status = status %in% 1:2,
-#'              breaks = BL)
-#'               
-#' x <- aggre(x, by = fot)
-#'
-#' ## silly example of referring to pyrs data by fixed character string;
-#' ## its possible that the real name wont be fixed in a real-life application.
-#' pyrs <- "actual_pyrs"  
-#' TF <- environment()
-#' x$actual_pyrs <- as.numeric(x$pyrs)
-#' x$pyrs <- 1
-#'              
-#' ## this works (uses actual_pyrs eventually)
-#' st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
-#'                  pyrs = TF$pyrs, d = from0to1, 
-#'                  surv.method = "hazard")
-#' ## this would be wrong (sees expression 'pyrs' and uses that column,
-#' ## which is not what is intended here)
-#' st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
-#'                  pyrs = pyrs, d = from0to1,
-#'                  surv.method = "hazard")
-
-NULL
-
-
-
-
-
-
-
-
-
+
+
+
+#' @title Flexible Variable Usage in \pkg{popEpi} Functions
+#' @author Joonas Miettinen
+#' @name flexible_argument
+#' @description Certain arguments in \pkg{popEpi} can be passed in multiple 
+#' ways. This document shows the usage and a pitfall in the
+#' usage of such flexible arguments.
+#' 
+#' @details 
+#' 
+#' Flexible arguments in \pkg{popEpi} are used to pass variables existing
+#' in your data or in the environment where the function is used 
+#' (for everyday users this is the global environment - in simple terms,
+#' where your data is / your work space). The flexible arguments
+#' are modelled after the \code{by} argument in \code{data.tables} - 
+#' see \code{?data.table}. There are many ways to supply the same information
+#' to certain functions in \pkg{popEpi}, but the possible ways listed below
+#' may be limited in some of them to only allow for using only a part of them.
+#' 
+#' @section Everyday usage:
+#' 
+#' Most commonly you may pass
+#' variable names as character strings, e.g.
+#' 
+#' \code{FUN(arg = c("V1", "V2"), data = x)}
+#' 
+#' which may be stored in advance:
+#' 
+#' \code{vars <- c("V1", "V2")}
+#' 
+#' \code{FUN(arg = vars, data = x)}
+#' 
+#' where \code{x} contains those variables. You may also supply variable
+#' names as symbols:
+#' 
+#' \code{FUN(arg = V1, data = x)}
+#' 
+#' Or as a list of symbols (similarly to as in \code{\link{aggregate}}):
+#' 
+#' \code{FUN(arg = list(V1, V2), data = x)}
+#' 
+#' Or as a list of expressions:
+#' 
+#' \code{FUN(arg = list(V1 + 1, factor(V2)), data = x)}
+#' 
+#' A formula without a left-hand-side specified is sometimes allowed as well:
+#' 
+#' \code{FUN(arg = ~ I(V1 + 1) + factor(V2), data = x)}
+#' 
+#' Using a symbol or a list of symbols/expressions typically
+#' causes the function to look for the variable(s)
+#' first in the supplied data (if any) and then where the function was called.
+#' For everyday users this means you might define e.g.
+#' 
+#' \code{V3 <- factor(letters)}
+#' 
+#' and do e.g.
+#' 
+#' \code{FUN(arg = list(V1 + 1, factor(V2), V3), data = x)}
+#' 
+#' provided \code{V1} and \code{V2} exist in \code{x} or in the function calling
+#' environment.
+#' 
+#' @section A pitfall:
+#' 
+#' There is one way to use flexible arguments incorrectly: By supplying
+#' the name of a variable which exists both in the supplied data
+#' and the calling environment, and intending the latter to be used. E.g.
+#' 
+#' \code{vars <- c("V2")}
+#' 
+#' \code{FUN(arg = V3, data = x)}
+#' 
+#' where \code{x} has a column named \code{vars}. This causes the function to
+#' use \code{x$vars} and NOT \code{x$V2}.
+#' 
+#' @section Advanced:
+#' 
+#' Function programmers are advised to pass character strings
+#' whenever possible. To fool-proof against conflicts as described in the
+#' section above, refer to the calling environment explicitly when
+#' passing the variable containing the character strings:
+#' 
+#' \code{TF <- environment() ## current env to refer to}
+#' 
+#' \code{vars <- c("V1", "V2")}
+#' 
+#' \code{FUN(arg = TF$vars, data = x)}
+#' 
+#' Even if \code{x} has columns named \code{vars} and \code{TF}, 
+#' using \code{TF$vars} does not use those columns but only evaluates
+#' \code{TF$vars}
+#' in the calling environment. This is made possible by the fact
+#' that data is always passed as a \code{data.frame}, within which evaluation
+#' of expressions using the dollar operator is not possible. Therefore
+#' it is safe to assume the data should not be used. However, lists of 
+#' expressions will not be checked for dollar use and will fail in conflict
+#' situations:
+#' 
+#' \code{TF <- environment() ## current env to refer to}
+#' 
+#' \code{vars <- letters[1:5]}
+#' 
+#' \code{x <- data.frame(vars = 1:5, TF = 5:1, V1 = 10:6)}
+#' 
+#' \code{FUN(arg = list(TF$vars, V1), data = x)}
+#' 
+#' On the other hand you may typically also pass quoted (\code{\link{quote}})
+#' or substituted \code{\link{substitute}} expressions etc., where
+#' the \code{env$object} trick will work as well:
+#' 
+#' \code{q <- quote(list(vars, V1))}
+#' 
+#' \code{FUN(arg = TF$q, data = x)}
+#' 
+#' This works even with
+#' 
+#' \code{a <- 1:5}
+#' 
+#' \code{V1 <- quote(TF$a)}
+#' 
+#' \code{FUN(arg = TF$V1, data = x)}
+#' 
+#' So no conflicts should occur.
+#' @family popEpi argument evaluation docs
+#' @examples 
+#' 
+#' data(sire)
+#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
+#' ## note: sire is a simulated cohort integrated into popEpi.
+#' BL <- list(fot=seq(0, 5, by = 1/12))
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+#'              status = status %in% 1:2,
+#'              breaks = BL)
+#'               
+#' x <- aggre(x, by = fot)
+#'
+#' ## silly example of referring to pyrs data by fixed character string;
+#' ## its possible that the real name wont be fixed in a real-life application.
+#' pyrs <- "actual_pyrs"  
+#' TF <- environment()
+#' x$actual_pyrs <- as.numeric(x$pyrs)
+#' x$pyrs <- 1
+#'              
+#' ## this works (uses actual_pyrs eventually)
+#' st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
+#'                  pyrs = TF$pyrs, d = from0to1, 
+#'                  surv.method = "hazard")
+#' ## this would be wrong (sees expression 'pyrs' and uses that column,
+#' ## which is not what is intended here)
+#' st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
+#'                  pyrs = pyrs, d = from0to1,
+#'                  surv.method = "hazard")
+
+NULL
+
+
+
+
+
+
+
+
+
diff --git a/R/fractional_years.R b/R/fractional_years.R
index b5b0ce9..3aeb9ff 100644
--- a/R/fractional_years.R
+++ b/R/fractional_years.R
@@ -1,155 +1,155 @@
-
-
-#' @title Convert date objects to fractional years
-#' @author Joonas Miettinen
-#' @description Using Date objects, calculates given 
-#' dates as fractional years.
-#' @param x a \code{Date} object, or anything that \code{link{as.Date}}
-#' accepts
-#' @param year.length character string, either \code{'actual'} or 
-#' \code{'approx'}; can be abbreviated; see Details
-#' @param ... additional arguments passed on to \code{\link{as.Date}};
-#' typically \code{format} when \code{x} is a character string variable,
-#' and \code{origin} when \code{x} is numeric
-#' @import data.table
-#' @export
-#' @details
-#' 
-#' \code{x} should preferably be a \code{date}, \code{Date} or \code{IDate} 
-#' object, although it can also be a character string variable 
-#' which is coerced internally to \code{Date} format 
-#' using \code{\link{as.Date.character}}.
-#' 
-#' When \code{ year.length = 'actual' }, fractional years are calculated as 
-#' \code{ year + (day_in_year-1)/365 } for non-leap-years
-#' and as \code{ year + (day_in_year-1)/366 } for leap years. 
-#' If \code{ year.length = 'approx' }, fractional years are always
-#' calculated as in \code{ year + (day_in_year-1)/365.242199 }. 
-#' 
-#' There is a slight difference, then, between the two methods
-#' when calculating durations between fractional years. For
-#' meticulous accuracy one might instead want to calculate durations using
-#' dates (days) and convert the results to fractional years.
-#'  
-#' Note that dates are effectively converted to fractional years at 
-#' \code{ 00:00:01 } o'clock:
-#' 
-#' 
-#' \code{ get.yrs("2000-01-01") = 2000 }, and
-#' \code{ get.yrs("2000-01-02") = 2000 + 1/365.242199 }. 
-#' 
-#' 
-#' @seealso
-#' \code{\link[Epi]{cal.yr}}, \code{\link{as.Date.yrs}}, \code{\link{as.Date}}
-#' 
-#' @return
-#' A numeric vector of fractional years.
-#' 
-#' @examples
-#' 
-#' data("sire")
-#' sire$dg_yrs <- get.yrs(sire$dg_date)
-#' summary(sire$dg_yrs)
-#' 
-#' ## see: ?as.Date.yrs
-#' dg_date2 <- as.Date(sire$dg_yrs)
-#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-#' 
-#' ## Epi's cal.yr versus get.yrs
-#' d <- as.Date("2000-01-01")
-#' Epi::cal.yr(d) ## 1999.999
-#' get.yrs(d) ## 2000
-#' 
-#' ## "..." passed on to as.Date, so character / numeric also accepted as input
-#' ## (and whatever else as.Date accepts)
-#' get.yrs("2000-06-01")
-#' get.yrs("20000601", format = "%Y%m%d")
-#' get.yrs("1/6/00", format = "%d/%m/%y")
-#' 
-#' get.yrs(100, origin = "1970-01-01")
-#' 
-#' 
-get.yrs <- function(x, year.length = "approx", ...) {
-  as.yrs(x, year.length = year.length, ...)
-}
-
-
-as.yrs <- function(x, year.length, ...) {
-  UseMethod("as.yrs")
-}
-
-as.yrs.Date <- function(x, year.length = "approx", ...) {
-  year.length <- match.arg(year.length, c("actual", "approx"))
-  
-  yl <- 365.242199
-  y <- year(x)
-  if (year.length == "actual") {
-    yl <- ifelse(is_leap_year(y), 366L, 365L)
-  }
-  d <- yday(x)
-  
-  yrs <- y + (d - 1L)/yl
-  setattr(yrs, "year.length", year.length)
-  setattr(yrs, "class", c("yrs", "numeric"))
-  yrs
-}
-
-as.yrs.default <- function(x, year.length = "approx", ...) {
-  
-  x <- as.Date(x, ...)
-  as.yrs(x, year.length = year.length)
-  
-}
-
-
-
-#' @title Coerce Fractional Year Values to Date Values
-#' @author Joonas Miettinen
-#' @param x an \code{yrs} object created by \code{get.yrs}
-#' @param ... unused, included for compatibility with other \code{as.Date}
-#' methods
-#' @description Coerces an \code{yrs} object to a \code{Date} object.
-#' Some loss of information comes if \code{year.length = "approx"} 
-#' was set when using \code{\link{get.yrs}}, so the transformation back
-#' to \code{Date} will not be perfect there. With \code{year.length = "actual"}
-#' the original values are perfectly retrieved.
-#' @examples 
-#' data("sire", package = "popEpi")
-#' 
-#' ## approximate year lengths: here 20 % have an extra day added
-#' sire$dg_yrs <- get.yrs(sire$dg_date)
-#' summary(sire$dg_yrs)
-#' dg_date2 <- as.Date(sire$dg_yrs)
-#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-#' 
-#' ## using actual year lengths
-#' sire$dg_yrs <- get.yrs(sire$dg_date, year.length = "actual")
-#' summary(sire$dg_yrs)
-#' dg_date2 <- as.Date(sire$dg_yrs)
-#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-#' @seealso \code{\link{get.yrs}}
-#' @return
-#' A vector of `Date` values based on the input fractional years.
-#' @export
-as.Date.yrs <- function(x, ...) {
-  
-  yl <- attr(x, "year.length")
-  if (is.null(yl)) {
-    warning("x did not contain meta information about year length used ",
-            "when forming the yrs object. Assuming 'approx'.")
-    yl <- "approx"
-  }
-  
-  y <- as.integer(x)
-  
-  mu <- 365.242199
-  if (yl == "actual") {
-    mu <- ifelse(is_leap_year(y), rep(365L, length(x)), rep(364L, length(x)))
-  }
-  x <- x + 1L/mu
-  yd <- as.integer((x-y)*mu)
-  d <- as.Date(paste0(y, "-01-01")) + yd
-  d
-}
-
-
+
+
+#' @title Convert date objects to fractional years
+#' @author Joonas Miettinen
+#' @description Using Date objects, calculates given 
+#' dates as fractional years.
+#' @param x a \code{Date} object, or anything that \code{link{as.Date}}
+#' accepts
+#' @param year.length character string, either \code{'actual'} or 
+#' \code{'approx'}; can be abbreviated; see Details
+#' @param ... additional arguments passed on to \code{\link{as.Date}};
+#' typically \code{format} when \code{x} is a character string variable,
+#' and \code{origin} when \code{x} is numeric
+#' @import data.table
+#' @export
+#' @details
+#' 
+#' \code{x} should preferably be a \code{date}, \code{Date} or \code{IDate} 
+#' object, although it can also be a character string variable 
+#' which is coerced internally to \code{Date} format 
+#' using \code{\link{as.Date.character}}.
+#' 
+#' When \code{ year.length = 'actual' }, fractional years are calculated as 
+#' \code{ year + (day_in_year-1)/365 } for non-leap-years
+#' and as \code{ year + (day_in_year-1)/366 } for leap years. 
+#' If \code{ year.length = 'approx' }, fractional years are always
+#' calculated as in \code{ year + (day_in_year-1)/365.242199 }. 
+#' 
+#' There is a slight difference, then, between the two methods
+#' when calculating durations between fractional years. For
+#' meticulous accuracy one might instead want to calculate durations using
+#' dates (days) and convert the results to fractional years.
+#'  
+#' Note that dates are effectively converted to fractional years at 
+#' \code{ 00:00:01 } o'clock:
+#' 
+#' 
+#' \code{ get.yrs("2000-01-01") = 2000 }, and
+#' \code{ get.yrs("2000-01-02") = 2000 + 1/365.242199 }. 
+#' 
+#' 
+#' @seealso
+#' \code{\link[Epi]{cal.yr}}, \code{\link{as.Date.yrs}}, \code{\link{as.Date}}
+#' 
+#' @return
+#' A numeric vector of fractional years.
+#' 
+#' @examples
+#' 
+#' data("sire")
+#' sire$dg_yrs <- get.yrs(sire$dg_date)
+#' summary(sire$dg_yrs)
+#' 
+#' ## see: ?as.Date.yrs
+#' dg_date2 <- as.Date(sire$dg_yrs)
+#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+#' 
+#' ## Epi's cal.yr versus get.yrs
+#' d <- as.Date("2000-01-01")
+#' Epi::cal.yr(d) ## 1999.999
+#' get.yrs(d) ## 2000
+#' 
+#' ## "..." passed on to as.Date, so character / numeric also accepted as input
+#' ## (and whatever else as.Date accepts)
+#' get.yrs("2000-06-01")
+#' get.yrs("20000601", format = "%Y%m%d")
+#' get.yrs("1/6/00", format = "%d/%m/%y")
+#' 
+#' get.yrs(100, origin = "1970-01-01")
+#' 
+#' 
+get.yrs <- function(x, year.length = "approx", ...) {
+  as.yrs(x, year.length = year.length, ...)
+}
+
+
+as.yrs <- function(x, year.length, ...) {
+  UseMethod("as.yrs")
+}
+
+as.yrs.Date <- function(x, year.length = "approx", ...) {
+  year.length <- match.arg(year.length, c("actual", "approx"))
+  
+  yl <- 365.242199
+  y <- year(x)
+  if (year.length == "actual") {
+    yl <- ifelse(is_leap_year(y), 366L, 365L)
+  }
+  d <- yday(x)
+  
+  yrs <- y + (d - 1L)/yl
+  setattr(yrs, "year.length", year.length)
+  setattr(yrs, "class", c("yrs", "numeric"))
+  yrs
+}
+
+as.yrs.default <- function(x, year.length = "approx", ...) {
+  
+  x <- as.Date(x, ...)
+  as.yrs(x, year.length = year.length)
+  
+}
+
+
+
+#' @title Coerce Fractional Year Values to Date Values
+#' @author Joonas Miettinen
+#' @param x an \code{yrs} object created by \code{get.yrs}
+#' @param ... unused, included for compatibility with other \code{as.Date}
+#' methods
+#' @description Coerces an \code{yrs} object to a \code{Date} object.
+#' Some loss of information comes if \code{year.length = "approx"} 
+#' was set when using \code{\link{get.yrs}}, so the transformation back
+#' to \code{Date} will not be perfect there. With \code{year.length = "actual"}
+#' the original values are perfectly retrieved.
+#' @examples 
+#' data("sire", package = "popEpi")
+#' 
+#' ## approximate year lengths: here 20 % have an extra day added
+#' sire$dg_yrs <- get.yrs(sire$dg_date)
+#' summary(sire$dg_yrs)
+#' dg_date2 <- as.Date(sire$dg_yrs)
+#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+#' 
+#' ## using actual year lengths
+#' sire$dg_yrs <- get.yrs(sire$dg_date, year.length = "actual")
+#' summary(sire$dg_yrs)
+#' dg_date2 <- as.Date(sire$dg_yrs)
+#' summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+#' @seealso \code{\link{get.yrs}}
+#' @return
+#' A vector of `Date` values based on the input fractional years.
+#' @export
+as.Date.yrs <- function(x, ...) {
+  
+  yl <- attr(x, "year.length")
+  if (is.null(yl)) {
+    warning("x did not contain meta information about year length used ",
+            "when forming the yrs object. Assuming 'approx'.")
+    yl <- "approx"
+  }
+  
+  y <- as.integer(x)
+  
+  mu <- 365.242199
+  if (yl == "actual") {
+    mu <- ifelse(is_leap_year(y), rep(365L, length(x)), rep(364L, length(x)))
+  }
+  x <- x + 1L/mu
+  yd <- as.integer((x-y)*mu)
+  d <- as.Date(paste0(y, "-01-01")) + yd
+  d
+}
+
+
diff --git a/R/incidence_rates.R b/R/incidence_rates.R
index 7eaa433..a6fb24d 100644
--- a/R/incidence_rates.R
+++ b/R/incidence_rates.R
@@ -1,347 +1,347 @@
-#' @title Direct-Standardised Incidence/Mortality Rates
-#' @author Matti Rantanen, Joonas Miettinen
-#'
-#' @description \code{rate} calculates adjusted rates using
-#' preloaded weights data or user specified weights.
-#'
-#' @param data aggregated data (see e.g. \code{\link{lexpand}}, 
-#' \code{\link{aggre}} if you have subject-level data)
-#' @param pyrs person-years variable name in data.
-#' \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{pyrs = pyrs}.
-#' @param obs observations variable name in data.
-#' \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{obs = obs}.
-#' @param adjust variable for adjusting the rates.
-#' \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{adjust = agegroup}.
-#' @param print variable name to stratify the rates.
-#' \link[=flexible_argument]{Flexible input}, typically e.g.
-#' \code{print = sex} or \code{print = list(sex, area)}.
-#' @param weights typically a list of weights or a \code{character} string
-#' specifying an age group standardization scheme; see
-#' the \link[=direct_standardization]{dedicated help page} 
-#' and examples.
-#' 
-#' @param subset a logical expression to subset data.
-#' 
-#' @details Input data needs to be in aggregated format with observations 
-#' and person-years. For individual data use \code{\link{lexpand}}, or
-#' \code{\link{ltable}} and merge person-years manually.
-#' 
-#' The confidence intervals are based on the normal approximation of the logarithm of the rate.
-#' The variance of the log rate that is used to derive the confidence intervals 
-#' is derived using the delta method. 
-#' 
-#' @return Returns a \code{data.table} with observations, person-years, rates and
-#' adjusted rates, if available. Results are stratified by \code{print}.
-#' Adjusted rates are identified with suffix \code{.adj} and  
-#' \code{.lo} and \code{.hi} are for confidence intervals lower and upper 
-#' 95\% bounds, respectively.
-#' The prefix \code{SE.} stands for standard error.
-#' 
-#' @seealso \code{\link{lexpand}}, \code{\link{ltable}}
-#' 
-#' @examples 
-#' ## Prepare data with lexpand and then reformat agegroup.
-#' data(sibr)
-#' x <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,  
-#'              breaks = list(per = c(1990,2000,2010,2020), age = c(0:17*5,Inf)),
-#'              aggre = list(agegroup = age, year.cat = per),
-#'              status =  status != 0)
-#'
-#' x$agegroup <- cut(x$agegroup,  c(0:17*5,Inf), right = FALSE)
-#'
-#' ## calculate rates for selected periods with Nordic 2000 weights:
-#' r1 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-#'             adjust = agegroup, weights = 'nordic')
-#' r1
-#'
-#' ## use total person-years by stratum as weights (some have zero)
-#' w <- ltable(x, by.vars = "agegroup", expr = sum(pyrs))
-#' w[is.na(w$V1),]$V1 <- 0
-#' 
-#' r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-#'             adjust = agegroup,
-#'             weights = w$V1)
-#' r2
-#' 
-#' ## use data.frame of weights:
-#' names(w) <- c("agegroup", "weights")
-#' r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-#'             adjust = agegroup,
-#'             weights = w)
-#' r2
-#' 
-#' ## internal weights (same result as above)
-#' r3 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-#'             adjust = agegroup,
-#'             weights = "internal")
-#' r3
-#'
-#' @import data.table
-#' @export
-#' @family main functions
-#' @family rate functions
-
-rate <- function( data,
-                  obs = NULL,
-                  pyrs = NULL,
-                  print = NULL,
-                  adjust = NULL, 
-                  weights = NULL,
-                  subset = NULL
-) {
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  
-  ## subsetting -----------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = data, substiset = subset, enclos = PF)
-  data <- data[subset,]
-  setDT(data)
-  
-  # evalPopArg
-  obs <- substitute(obs)
-  inc.obs <- evalPopArg(data = data, arg = obs, enclos = PF)
-  if (!length(inc.obs)) {
-    stop("No observations given.")
-  }
-  obsNames <- copy(names(inc.obs))
-  tmpObsNames <- makeTempVarName(data = data, pre = "obs")
-  setnames(inc.obs, obsNames, tmpObsNames)
-  
-  pyrs <- substitute(pyrs)
-  inc.pyr <- evalPopArg(data = data, arg = pyrs, enclos = PF)
-  if (!length(inc.pyr)) {
-    stop("No pyrs given.")
-  }
-  pyrNames <- copy(names(inc.pyr))
-  tmpPyrNames <- makeTempVarName(data = data, pre = "pyr")
-  setnames(inc.pyr, pyrNames, tmpPyrNames)
-  
-  print <- substitute(print)
-  inc.pri <- evalPopArg(data = data, arg = print, enclos = PF)
-  prNames <- tmpPrNames <- NULL
-  if (length(inc.pri)) {
-    prNames <- copy(names(inc.pri))
-    tmpPrNames <- makeTempVarName(data = data, 
-                                  pre = paste0("print", seq_along(prNames)))
-    setnames(inc.pri, prNames, tmpPrNames)
-  }
-  
-  adjust <- substitute(adjust)
-  inc.adj <- evalPopArg(data = data, arg = adjust, enclos = PF)
-  adNames <- tmpAdNames <- NULL
-  if (length(inc.adj)) {
-    adNames <- copy(names(inc.adj))
-    tmpAdNames <- makeTempVarName(data = data, 
-                                  pre = paste0("adjust", seq_along(adNames)))
-    setnames(inc.adj, adNames, tmpAdNames)
-  }
-  
-  ## collect data --------------------------------------------------------------
-  data <- cbind(inc.obs, inc.pyr)
-  if (!is.null(prNames)) data <- cbind(data, inc.pri) 
-  if (!is.null(adNames)) data <- cbind(data, inc.adj)
-  
-  
-  ## handle weights ------------------------------------------------------------
-  weights <- substitute(weights)
-  weights <- eval(weights, envir = PF)
-  weights <- copy(weights)
-  if (length(inc.adj)) {
-    ## rename adjust variables in inc.adj back to original names
-    ## for more human-readable errors in checkWeights if any occur
-    setnames(inc.adj, tmpAdNames, adNames)
-  }
-  
-  checkWeights(weights, inc.adj)
-  if (is.list(weights) && !is.data.frame(weights)) {
-    ## ensure weights list / DF names match to temp adjust var names
-    weights <- weights[adNames]
-    names(weights) <- tmpAdNames
-  } else if (is.data.frame(weights)) {
-    setnames(weights, adNames, tmpAdNames)
-  }
-  
-  ## form table with weights ---------------------------------------------------
-  NA.msg <- "Data contains %%NA_COUNT%% NA values."
-  data <- makeWeightsDT(data, 
-                        values = list(tmpObsNames, tmpPyrNames),
-                        print = tmpPrNames, 
-                        adjust = tmpAdNames, 
-                        weights = weights, 
-                        internal.weights.values = tmpPyrNames,
-                        NA.text = NA.msg)
-  
-  ## estimate standardized rates -----------------------------------------------
-  data <- rate_est(data = data,
-                   obs = tmpObsNames,
-                   pyrs = tmpPyrNames,
-                   print = tmpPrNames,
-                   weights = "weights")
-  
-  ## final touch ---------------------------------------------------------------
-  setDT(data)
-  setattr(data, "class", c("rate", "data.table", "data.frame"))
-  setattr(data, name = 'rate.meta', value = list(obs = obsNames, 
-                                                 pyrs = pyrNames, 
-                                                 weights = weights,
-                                                 adjust = adNames,
-                                                 print = prNames,
-                                                 call = match.call(),
-                                                 NAs = NA))
-  setnames(data, c(tmpObsNames, tmpPyrNames, tmpPrNames),
-           c(obsNames, pyrNames, prNames))
-  
-  # data.frame output option  
-  if (!return_DT()) {
-    setDFpe(data)
-  }
-  
-  return(data[])
-}
-
-#' @export
-getCall.rate <- function (x, ...) {
-  attributes(x)$rate.meta$call
-}
-
-stdr.weights <- function(wp = 'world00_1') {
-  
-  ## This one returns the standard population
-  ## output: data.table with colnames: agegroup, reference
-  ## standard populations are from datasets: stdpop18 and stdpop101
-  allow.pop <- c("world_1966_18of5", 
-                 "europe_1976_18of5", 
-                 "nordic_2000_18of5", 
-                 "world_2000_18of5", 
-                 "world_2000_20of5", 
-                 "world_2000_101of1")
-  wp <- match.arg(wp, allow.pop)
-  
-  if (length(wp) > 1) {
-    stop('Standard population name is not a scalar (length != 1).')
-    
-  } else if (wp %in% allow.pop[1:3]) {
-    
-    # get standard pop
-    sr <- data.table(popEpi::stdpop18)
-    setnames(sr, 1:4, c("agegroup",allow.pop[1:3]))
-    sr[, agegroup := 1:18]
-    sr[, setdiff(allow.pop[1:3], wp) := NULL]
-    
-    setnames(sr, wp, 'reference')
-    
-  } else if (wp %in% allow.pop[4:6]) {
-    
-    sr <- data.table(popEpi::stdpop101)
-    if (wp == "world_2000_18of5") {
-      sr[,agegroup := cut(agegroup, breaks=c(0:17*5,Inf), right=FALSE, labels=FALSE)]
-      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
-    }
-    if (wp == 'world_2000_20of5') {
-      sr[,agegroup := cut(agegroup, breaks=c(0:19*5,Inf), right=FALSE, labels=FALSE)]
-      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
-    }
-    else {
-      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
-    }
-    setnames(sr, "world_std", "reference")
-  }
-  else {
-    stop("Invalid standard population name.")
-  }
-  sr[]
-}
-globalVariables(c('stdpop18','stdpop101','agegroup','world_std'))
-
-
-rate_est <- function(data = data, 
-                     obs = 'obs', 
-                     pyrs = 'pyrs', 
-                     print = NULL, 
-                     weights = NULL
-) {
-  ## This one estimates the rates and calculates CI's and SE's.
-  
-  badVars <- paste0("Internal error: missing following variable names in ",
-                    "working data: %%VARS%%. Complain to the pkg maintainer ",
-                    "if you see this.")
-  all_names_present(data, c(obs, pyrs, print), msg = badVars)
-  
-  data <- data.table(data)
-  if ( is.null(weights) |  !weights %in% colnames(data)) {
-    weights <- NULL
-  }
-  
-  if (all(!is.null(weights), !is.null(obs), !is.null(pyrs))) {
-    # rate.adj
-    
-    f2 <- function(list) list[[1]]/list[[2]]*list[[3]]
-    funx <- function(n,d,w,fun)  eval(parse(text=fun))
-    
-
-    # variance rate.adj for each strata A
-    fun1 <- '(._d_/._n_^2) * ._w_^2'
-    fun2 <- '._d_ / ._n_ * ._w_'
-    
-    make_fun <- function(n = NA, d = NA, w = NA, fun) {
-      fun <- gsub(pattern = "._n_", replacement = n, x = fun)
-      fun <- gsub(pattern = "._d_", replacement = d, x = fun)
-      fun <- gsub(pattern = "._w_", replacement = w, x = fun)
-      parse(text = fun)
-    }
-    eval.me1 <- make_fun(d = obs, n = pyrs, w=weights, fun = fun1)
-    eval.me2 <- make_fun(d = obs, n = pyrs, w=weights, fun = fun2)
-    data[, var.temp := eval(eval.me1)]
-    data[, lam.temp := eval(eval.me2)]
-    # add std weighted rates and variances
-    #data[, ':='(var.temp = funx(d=get(obs), n=get(pyrs), w=get(weights), fun = fun1),
-    #            lam.temp = funx(d=get(obs), n=get(pyrs), w=get(weights), fun = fun2)) ]
-    data[, rate.adj := f2(.SD), .SDcols= c(obs, pyrs, weights)]
-
-    # aggregate data
-    ie <- paste0('list(', obs, '=sum(',obs,',na.rm=TRUE), ', pyrs, '=sum(',pyrs,',na.rm=TRUE),',
-                 'rate.adj=sum(rate.adj,na.rm=TRUE),' ,'lam.temp=sum(lam.temp,na.rm=TRUE), var.temp=sum(var.temp,na.rm=TRUE))') 
-    l <- parse(text = ie)
-
-    data <- data[, eval(l), by=print]
-    # rate.adj: S.E.
-    data[, SE.log.rate.adj := sqrt((1/lam.temp)^2 * var.temp) ] # tämä on log-rate
-    data[, SE.rate.adj := sqrt(var.temp)]
-    # rate.adj: CI
-    data[, ':='(rate.adj.lo = exp( log(rate.adj) - SE.log.rate.adj*1.96 ),
-                rate.adj.hi = exp( log(rate.adj) + SE.log.rate.adj*1.96 )) ]
-    data[,c('lam.temp','var.temp','SE.log.rate.adj') := NULL]
-  }
-  
-  else {
-    ie <- paste0('list(', obs, '=sum(',obs,'), ', pyrs, '=sum(',pyrs,'))') 
-    l <- parse(text = ie)
-    data <- data[, eval(l), by=print]
-  }
-  # rate
-  ia <- paste0('rate := ',obs,'/', pyrs)
-  k <- parse(text = ia)
-  data[, eval(k), by = print]
-  
-  # var(rate)
-  var_r <- paste0('SE.rate := sqrt(',obs,'/(',pyrs,'*',pyrs,'))')
-  k <- parse(text = var_r)
-  data[, eval(k), by = print]
-  
-  # var(log(rate)) and CI
-  eval.me3 <- paste('exp(sqrt(1/',obs,'))')
-  eval.me3 <- parse(text = eval.me3)
-  data[, SE.log.rate := eval(eval.me3)]
-  data[, ':='(rate.lo = exp(log(rate)-log(SE.log.rate)*1.96),
-              rate.hi = exp(log(rate)+log(SE.log.rate)*1.96)) ]
-  data[, SE.log.rate := NULL]
-  return(data[])
-}
-
-globalVariables(c('var.temp','lam.temp','rate.adj','SE.rate.adj','SE.rate','SE.log.rate','SE.log.rate.adj'))
-
+#' @title Direct-Standardised Incidence/Mortality Rates
+#' @author Matti Rantanen, Joonas Miettinen
+#'
+#' @description \code{rate} calculates adjusted rates using
+#' preloaded weights data or user specified weights.
+#'
+#' @param data aggregated data (see e.g. \code{\link{lexpand}}, 
+#' \code{\link{aggre}} if you have subject-level data)
+#' @param pyrs person-years variable name in data.
+#' \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{pyrs = pyrs}.
+#' @param obs observations variable name in data.
+#' \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{obs = obs}.
+#' @param adjust variable for adjusting the rates.
+#' \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{adjust = agegroup}.
+#' @param print variable name to stratify the rates.
+#' \link[=flexible_argument]{Flexible input}, typically e.g.
+#' \code{print = sex} or \code{print = list(sex, area)}.
+#' @param weights typically a list of weights or a \code{character} string
+#' specifying an age group standardization scheme; see
+#' the \link[=direct_standardization]{dedicated help page} 
+#' and examples.
+#' 
+#' @param subset a logical expression to subset data.
+#' 
+#' @details Input data needs to be in aggregated format with observations 
+#' and person-years. For individual data use \code{\link{lexpand}}, or
+#' \code{\link{ltable}} and merge person-years manually.
+#' 
+#' The confidence intervals are based on the normal approximation of the logarithm of the rate.
+#' The variance of the log rate that is used to derive the confidence intervals 
+#' is derived using the delta method. 
+#' 
+#' @return Returns a \code{data.table} with observations, person-years, rates and
+#' adjusted rates, if available. Results are stratified by \code{print}.
+#' Adjusted rates are identified with suffix \code{.adj} and  
+#' \code{.lo} and \code{.hi} are for confidence intervals lower and upper 
+#' 95\% bounds, respectively.
+#' The prefix \code{SE.} stands for standard error.
+#' 
+#' @seealso \code{\link{lexpand}}, \code{\link{ltable}}
+#' 
+#' @examples 
+#' ## Prepare data with lexpand and then reformat agegroup.
+#' data(sibr)
+#' x <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,  
+#'              breaks = list(per = c(1990,2000,2010,2020), age = c(0:17*5,Inf)),
+#'              aggre = list(agegroup = age, year.cat = per),
+#'              status =  status != 0)
+#'
+#' x$agegroup <- cut(x$agegroup,  c(0:17*5,Inf), right = FALSE)
+#'
+#' ## calculate rates for selected periods with Nordic 2000 weights:
+#' r1 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+#'             adjust = agegroup, weights = 'nordic')
+#' r1
+#'
+#' ## use total person-years by stratum as weights (some have zero)
+#' w <- ltable(x, by.vars = "agegroup", expr = sum(pyrs))
+#' w[is.na(w$V1),]$V1 <- 0
+#' 
+#' r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+#'             adjust = agegroup,
+#'             weights = w$V1)
+#' r2
+#' 
+#' ## use data.frame of weights:
+#' names(w) <- c("agegroup", "weights")
+#' r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+#'             adjust = agegroup,
+#'             weights = w)
+#' r2
+#' 
+#' ## internal weights (same result as above)
+#' r3 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+#'             adjust = agegroup,
+#'             weights = "internal")
+#' r3
+#'
+#' @import data.table
+#' @export
+#' @family main functions
+#' @family rate functions
+
+rate <- function( data,
+                  obs = NULL,
+                  pyrs = NULL,
+                  print = NULL,
+                  adjust = NULL, 
+                  weights = NULL,
+                  subset = NULL
+) {
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  
+  ## subsetting -----------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = data, substiset = subset, enclos = PF)
+  data <- data[subset,]
+  setDT(data)
+  
+  # evalPopArg
+  obs <- substitute(obs)
+  inc.obs <- evalPopArg(data = data, arg = obs, enclos = PF)
+  if (!length(inc.obs)) {
+    stop("No observations given.")
+  }
+  obsNames <- copy(names(inc.obs))
+  tmpObsNames <- makeTempVarName(data = data, pre = "obs")
+  setnames(inc.obs, obsNames, tmpObsNames)
+  
+  pyrs <- substitute(pyrs)
+  inc.pyr <- evalPopArg(data = data, arg = pyrs, enclos = PF)
+  if (!length(inc.pyr)) {
+    stop("No pyrs given.")
+  }
+  pyrNames <- copy(names(inc.pyr))
+  tmpPyrNames <- makeTempVarName(data = data, pre = "pyr")
+  setnames(inc.pyr, pyrNames, tmpPyrNames)
+  
+  print <- substitute(print)
+  inc.pri <- evalPopArg(data = data, arg = print, enclos = PF)
+  prNames <- tmpPrNames <- NULL
+  if (length(inc.pri)) {
+    prNames <- copy(names(inc.pri))
+    tmpPrNames <- makeTempVarName(data = data, 
+                                  pre = paste0("print", seq_along(prNames)))
+    setnames(inc.pri, prNames, tmpPrNames)
+  }
+  
+  adjust <- substitute(adjust)
+  inc.adj <- evalPopArg(data = data, arg = adjust, enclos = PF)
+  adNames <- tmpAdNames <- NULL
+  if (length(inc.adj)) {
+    adNames <- copy(names(inc.adj))
+    tmpAdNames <- makeTempVarName(data = data, 
+                                  pre = paste0("adjust", seq_along(adNames)))
+    setnames(inc.adj, adNames, tmpAdNames)
+  }
+  
+  ## collect data --------------------------------------------------------------
+  data <- cbind(inc.obs, inc.pyr)
+  if (!is.null(prNames)) data <- cbind(data, inc.pri) 
+  if (!is.null(adNames)) data <- cbind(data, inc.adj)
+  
+  
+  ## handle weights ------------------------------------------------------------
+  weights <- substitute(weights)
+  weights <- eval(weights, envir = PF)
+  weights <- copy(weights)
+  if (length(inc.adj)) {
+    ## rename adjust variables in inc.adj back to original names
+    ## for more human-readable errors in checkWeights if any occur
+    setnames(inc.adj, tmpAdNames, adNames)
+  }
+  
+  checkWeights(weights, inc.adj)
+  if (is.list(weights) && !is.data.frame(weights)) {
+    ## ensure weights list / DF names match to temp adjust var names
+    weights <- weights[adNames]
+    names(weights) <- tmpAdNames
+  } else if (is.data.frame(weights)) {
+    setnames(weights, adNames, tmpAdNames)
+  }
+  
+  ## form table with weights ---------------------------------------------------
+  NA.msg <- "Data contains %%NA_COUNT%% NA values."
+  data <- makeWeightsDT(data, 
+                        values = list(tmpObsNames, tmpPyrNames),
+                        print = tmpPrNames, 
+                        adjust = tmpAdNames, 
+                        weights = weights, 
+                        internal.weights.values = tmpPyrNames,
+                        NA.text = NA.msg)
+  
+  ## estimate standardized rates -----------------------------------------------
+  data <- rate_est(data = data,
+                   obs = tmpObsNames,
+                   pyrs = tmpPyrNames,
+                   print = tmpPrNames,
+                   weights = "weights")
+  
+  ## final touch ---------------------------------------------------------------
+  setDT(data)
+  setattr(data, "class", c("rate", "data.table", "data.frame"))
+  setattr(data, name = 'rate.meta', value = list(obs = obsNames, 
+                                                 pyrs = pyrNames, 
+                                                 weights = weights,
+                                                 adjust = adNames,
+                                                 print = prNames,
+                                                 call = match.call(),
+                                                 NAs = NA))
+  setnames(data, c(tmpObsNames, tmpPyrNames, tmpPrNames),
+           c(obsNames, pyrNames, prNames))
+  
+  # data.frame output option  
+  if (!return_DT()) {
+    setDFpe(data)
+  }
+  
+  return(data[])
+}
+
+#' @export
+getCall.rate <- function (x, ...) {
+  attributes(x)$rate.meta$call
+}
+
+stdr.weights <- function(wp = 'world00_1') {
+  
+  ## This one returns the standard population
+  ## output: data.table with colnames: agegroup, reference
+  ## standard populations are from datasets: stdpop18 and stdpop101
+  allow.pop <- c("world_1966_18of5", 
+                 "europe_1976_18of5", 
+                 "nordic_2000_18of5", 
+                 "world_2000_18of5", 
+                 "world_2000_20of5", 
+                 "world_2000_101of1")
+  wp <- match.arg(wp, allow.pop)
+  
+  if (length(wp) > 1) {
+    stop('Standard population name is not a scalar (length != 1).')
+    
+  } else if (wp %in% allow.pop[1:3]) {
+    
+    # get standard pop
+    sr <- data.table(popEpi::stdpop18)
+    setnames(sr, 1:4, c("agegroup",allow.pop[1:3]))
+    sr[, agegroup := 1:18]
+    sr[, setdiff(allow.pop[1:3], wp) := NULL]
+    
+    setnames(sr, wp, 'reference')
+    
+  } else if (wp %in% allow.pop[4:6]) {
+    
+    sr <- data.table(popEpi::stdpop101)
+    if (wp == "world_2000_18of5") {
+      sr[,agegroup := cut(agegroup, breaks=c(0:17*5,Inf), right=FALSE, labels=FALSE)]
+      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
+    }
+    if (wp == 'world_2000_20of5') {
+      sr[,agegroup := cut(agegroup, breaks=c(0:19*5,Inf), right=FALSE, labels=FALSE)]
+      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
+    }
+    else {
+      sr <- sr[,list(world_std = sum(world_std)), by="agegroup"]
+    }
+    setnames(sr, "world_std", "reference")
+  }
+  else {
+    stop("Invalid standard population name.")
+  }
+  sr[]
+}
+globalVariables(c('stdpop18','stdpop101','agegroup','world_std'))
+
+
+rate_est <- function(data = data, 
+                     obs = 'obs', 
+                     pyrs = 'pyrs', 
+                     print = NULL, 
+                     weights = NULL
+) {
+  ## This one estimates the rates and calculates CI's and SE's.
+  
+  badVars <- paste0("Internal error: missing following variable names in ",
+                    "working data: %%VARS%%. Complain to the pkg maintainer ",
+                    "if you see this.")
+  all_names_present(data, c(obs, pyrs, print), msg = badVars)
+  
+  data <- data.table(data)
+  if ( is.null(weights) |  !weights %in% colnames(data)) {
+    weights <- NULL
+  }
+  
+  if (all(!is.null(weights), !is.null(obs), !is.null(pyrs))) {
+    # rate.adj
+    
+    f2 <- function(list) list[[1]]/list[[2]]*list[[3]]
+    funx <- function(n,d,w,fun)  eval(parse(text=fun))
+    
+
+    # variance rate.adj for each strata A
+    fun1 <- '(._d_/._n_^2) * ._w_^2'
+    fun2 <- '._d_ / ._n_ * ._w_'
+    
+    make_fun <- function(n = NA, d = NA, w = NA, fun) {
+      fun <- gsub(pattern = "._n_", replacement = n, x = fun)
+      fun <- gsub(pattern = "._d_", replacement = d, x = fun)
+      fun <- gsub(pattern = "._w_", replacement = w, x = fun)
+      parse(text = fun)
+    }
+    eval.me1 <- make_fun(d = obs, n = pyrs, w=weights, fun = fun1)
+    eval.me2 <- make_fun(d = obs, n = pyrs, w=weights, fun = fun2)
+    data[, var.temp := eval(eval.me1)]
+    data[, lam.temp := eval(eval.me2)]
+    # add std weighted rates and variances
+    #data[, ':='(var.temp = funx(d=get(obs), n=get(pyrs), w=get(weights), fun = fun1),
+    #            lam.temp = funx(d=get(obs), n=get(pyrs), w=get(weights), fun = fun2)) ]
+    data[, rate.adj := f2(.SD), .SDcols= c(obs, pyrs, weights)]
+
+    # aggregate data
+    ie <- paste0('list(', obs, '=sum(',obs,',na.rm=TRUE), ', pyrs, '=sum(',pyrs,',na.rm=TRUE),',
+                 'rate.adj=sum(rate.adj,na.rm=TRUE),' ,'lam.temp=sum(lam.temp,na.rm=TRUE), var.temp=sum(var.temp,na.rm=TRUE))') 
+    l <- parse(text = ie)
+
+    data <- data[, eval(l), by=print]
+    # rate.adj: S.E.
+    data[, SE.log.rate.adj := sqrt((1/lam.temp)^2 * var.temp) ] # tämä on log-rate
+    data[, SE.rate.adj := sqrt(var.temp)]
+    # rate.adj: CI
+    data[, ':='(rate.adj.lo = exp( log(rate.adj) - SE.log.rate.adj*1.96 ),
+                rate.adj.hi = exp( log(rate.adj) + SE.log.rate.adj*1.96 )) ]
+    data[,c('lam.temp','var.temp','SE.log.rate.adj') := NULL]
+  }
+  
+  else {
+    ie <- paste0('list(', obs, '=sum(',obs,'), ', pyrs, '=sum(',pyrs,'))') 
+    l <- parse(text = ie)
+    data <- data[, eval(l), by=print]
+  }
+  # rate
+  ia <- paste0('rate := ',obs,'/', pyrs)
+  k <- parse(text = ia)
+  data[, eval(k), by = print]
+  
+  # var(rate)
+  var_r <- paste0('SE.rate := sqrt(',obs,'/(',pyrs,'*',pyrs,'))')
+  k <- parse(text = var_r)
+  data[, eval(k), by = print]
+  
+  # var(log(rate)) and CI
+  eval.me3 <- paste('exp(sqrt(1/',obs,'))')
+  eval.me3 <- parse(text = eval.me3)
+  data[, SE.log.rate := eval(eval.me3)]
+  data[, ':='(rate.lo = exp(log(rate)-log(SE.log.rate)*1.96),
+              rate.hi = exp(log(rate)+log(SE.log.rate)*1.96)) ]
+  data[, SE.log.rate := NULL]
+  return(data[])
+}
+
+globalVariables(c('var.temp','lam.temp','rate.adj','SE.rate.adj','SE.rate','SE.log.rate','SE.log.rate.adj'))
+
diff --git a/R/incidence_rates_utils.R b/R/incidence_rates_utils.R
index 2b47ca8..56e01b1 100644
--- a/R/incidence_rates_utils.R
+++ b/R/incidence_rates_utils.R
@@ -1,135 +1,135 @@
-
-
-#' @title Confidence intervals for the rate ratios
-#' @author Matti Rantanen
-#' @description Calculate rate ratio with confidence intervals for rate objects or observations and person-years.
-#' 
-#' @details Calculate rate ratio of two age standardized rate objects (see \code{\link{rate}}). 
-#' Multiple rates for each objects is supported if there are an equal number of rates. 
-#' Another option is to set \code{x} and \code{y} as a vector of two.
-#' \enumerate{
-#'   \item rate and its standard error, and  set \code{SE.method = TRUE}.
-#'   \item observations and person-year, and  set \code{SE.method = FALSE}.
-#' }
-#' See examples.
-#' 
-#' 
-#' @param x a rate-object, vector of two; rate and standard error or observed and person-years.
-#' @param y a rate-object, vector of two; rate and standard error or observed and person-years.
-#' @param crude set TRUE to use crude rates; default is FALSE.
-#' @param SE.method default TRUE; if \code{x} and \code{y} are vectors of observed and 
-#' person-years, this must be changed to FALSE.
-#' 
-#' @examples 
-#' \donttest{
-#' # two rate ratios; silly example with female rectal / breast cancer 
-#' ## mortality rates
-#' data("sire", package = "popEpi")
-#' data("sibr", package = "popEpi")
-#' 
-#' BL <- list(per = 2000:2005)
-#' 
-#' re <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#'               status = status == 1, breaks = BL, aggre = list(per))
-#' br <- lexpand(sibr, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#'               status = status == 1, breaks = BL, aggre = list(per))
-#' 
-#' r_re <- rate(re, obs = "from0to1", pyrs = "pyrs")
-#' r_br <- rate(br, obs = "from0to1", pyrs = "pyrs")
-#' 
-#' rate_ratio(r_re, r_br, SE.method = TRUE)
-#' }
-#' 
-#' # manually set rates (0.003 and 0.005) and SEs (0.001 and 0.002)
-#' # so that x = y = c('rate', 'SE')
-#' rate_ratio(x= c(0.003, 0.001), y= c(0.005, 0.002), SE.method = TRUE) 
-#' 
-#' # observed numbers (10 and 20) and person-years (30000 and 40000):
-#' rate_ratio(x = c(10, 30000), y = c(20, 40000), SE.method = FALSE)
-#' 
-#' @seealso \code{\link{rate}}
-#' 
-#' @family rate functions
-#' 
-#' @return A vector length of three: rate_ratio, and lower and upper confidence intervals.
-#' 
-#' @export rate_ratio
-#' 
-#' @import data.table
-#' @import stats
-rate_ratio <- function(x, y, crude = FALSE, SE.method = TRUE) {
-  if( inherits(x, 'rate') | inherits(y, 'rate') ) {
-    if(!crude & (!'rate.adj' %in% names(x) | !'rate.adj' %in% names(y))) {
-      crude <- TRUE
-      message('Crude rates used')
-    }
-  }
-  
-  x <- prep.rate.input(x, crude = crude, SE = SE.method)
-  y <- prep.rate.input(y, crude = crude, SE = SE.method)
-  
-  if(SE.method) {
-    ratio <- x[[1]]/y[[1]]
-    
-    # delta method for variance
-    v0 <-  (1/x[[1]])^2*x[[2]]^2 + (1/y[[1]])^2*y[[2]]^2
-    
-    
-    lo <- ratio - v0*1.96 #exp(log(ratio)-log(v0)*1.96)
-    hi <- ratio + v0*1.96 #exp(log(ratio)+log(v0)*1.96)
-    out <- round(data.frame(rate_ratio = ratio, lower = lo, upper = hi), 3)
-  }
-  else {
-    # x and y vector of two:, pyrs
-    pt <- list()
-    out <- data.frame()
-    j <- 1
-    for(j in 1:length(x[[1]])) {
-      pt[[j]] <- poisson.test(x = c(x[[1]][j], y[[1]][j]), T = c(x[[2]][j],y[[2]][j]))
-      out <- rbind(out, round(data.frame(rate_ratio = pt[[j]]$estimate, 
-                                         lower = pt[[j]]$conf.int[1], 
-                                         upper = pt[[j]]$conf.int[2]),3) )
-    }
-  }
-  if(any(out<0)) {
-    warning('Negative estimate or confidence intervals. Tip: set SE.method to FALSE when using observations and person-years.')
-  }
-  return(out)
-}
-
-
-
-prep.rate.input <- function(z, crude, SE) {
-  # this one modulates input to rate_ratio function
-  if(is.vector(z) && length(z) == 2) {
-    # z is obs and pyrs OR rate and SE
-    return(list(z[1], z[2]))
-  }
-  else if(inherits(z,'rate')){
-    if(!SE) { # obs and pyrs
-      att <- attributes(z)
-      setDT(z)
-      a <- z[, get(att$rate.meta$obs)]
-      b <- z[, get(att$rate.meta$pyrs)]
-    }
-    else {
-      if(crude) {
-        a <- z[,rate]
-        b <- z[,SE.rate]
-      } 
-      else {
-        # z is a rate object
-        a <- z[,rate.adj]
-        b <- z[,SE.rate.adj]
-      }
-    }
-  }
-  else{
-    stop('Input is not correct: its neighter a vector of two nor a rate object')
-  }
-  return(list(a,b))
-}
-
-  
-  
-
+
+
+#' @title Confidence intervals for the rate ratios
+#' @author Matti Rantanen
+#' @description Calculate rate ratio with confidence intervals for rate objects or observations and person-years.
+#' 
+#' @details Calculate rate ratio of two age standardized rate objects (see \code{\link{rate}}). 
+#' Multiple rates for each objects is supported if there are an equal number of rates. 
+#' Another option is to set \code{x} and \code{y} as a vector of two.
+#' \enumerate{
+#'   \item rate and its standard error, and  set \code{SE.method = TRUE}.
+#'   \item observations and person-year, and  set \code{SE.method = FALSE}.
+#' }
+#' See examples.
+#' 
+#' 
+#' @param x a rate-object, vector of two; rate and standard error or observed and person-years.
+#' @param y a rate-object, vector of two; rate and standard error or observed and person-years.
+#' @param crude set TRUE to use crude rates; default is FALSE.
+#' @param SE.method default TRUE; if \code{x} and \code{y} are vectors of observed and 
+#' person-years, this must be changed to FALSE.
+#' 
+#' @examples 
+#' \donttest{
+#' # two rate ratios; silly example with female rectal / breast cancer 
+#' ## mortality rates
+#' data("sire", package = "popEpi")
+#' data("sibr", package = "popEpi")
+#' 
+#' BL <- list(per = 2000:2005)
+#' 
+#' re <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#'               status = status == 1, breaks = BL, aggre = list(per))
+#' br <- lexpand(sibr, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#'               status = status == 1, breaks = BL, aggre = list(per))
+#' 
+#' r_re <- rate(re, obs = "from0to1", pyrs = "pyrs")
+#' r_br <- rate(br, obs = "from0to1", pyrs = "pyrs")
+#' 
+#' rate_ratio(r_re, r_br, SE.method = TRUE)
+#' }
+#' 
+#' # manually set rates (0.003 and 0.005) and SEs (0.001 and 0.002)
+#' # so that x = y = c('rate', 'SE')
+#' rate_ratio(x= c(0.003, 0.001), y= c(0.005, 0.002), SE.method = TRUE) 
+#' 
+#' # observed numbers (10 and 20) and person-years (30000 and 40000):
+#' rate_ratio(x = c(10, 30000), y = c(20, 40000), SE.method = FALSE)
+#' 
+#' @seealso \code{\link{rate}}
+#' 
+#' @family rate functions
+#' 
+#' @return A vector length of three: rate_ratio, and lower and upper confidence intervals.
+#' 
+#' @export rate_ratio
+#' 
+#' @import data.table
+#' @import stats
+rate_ratio <- function(x, y, crude = FALSE, SE.method = TRUE) {
+  if( inherits(x, 'rate') | inherits(y, 'rate') ) {
+    if(!crude & (!'rate.adj' %in% names(x) | !'rate.adj' %in% names(y))) {
+      crude <- TRUE
+      message('Crude rates used')
+    }
+  }
+  
+  x <- prep.rate.input(x, crude = crude, SE = SE.method)
+  y <- prep.rate.input(y, crude = crude, SE = SE.method)
+  
+  if(SE.method) {
+    ratio <- x[[1]]/y[[1]]
+    
+    # delta method for variance
+    v0 <-  (1/x[[1]])^2*x[[2]]^2 + (1/y[[1]])^2*y[[2]]^2
+    
+    
+    lo <- ratio - v0*1.96 #exp(log(ratio)-log(v0)*1.96)
+    hi <- ratio + v0*1.96 #exp(log(ratio)+log(v0)*1.96)
+    out <- round(data.frame(rate_ratio = ratio, lower = lo, upper = hi), 3)
+  }
+  else {
+    # x and y vector of two:, pyrs
+    pt <- list()
+    out <- data.frame()
+    j <- 1
+    for(j in 1:length(x[[1]])) {
+      pt[[j]] <- poisson.test(x = c(x[[1]][j], y[[1]][j]), T = c(x[[2]][j],y[[2]][j]))
+      out <- rbind(out, round(data.frame(rate_ratio = pt[[j]]$estimate, 
+                                         lower = pt[[j]]$conf.int[1], 
+                                         upper = pt[[j]]$conf.int[2]),3) )
+    }
+  }
+  if(any(out<0)) {
+    warning('Negative estimate or confidence intervals. Tip: set SE.method to FALSE when using observations and person-years.')
+  }
+  return(out)
+}
+
+
+
+prep.rate.input <- function(z, crude, SE) {
+  # this one modulates input to rate_ratio function
+  if(is.vector(z) && length(z) == 2) {
+    # z is obs and pyrs OR rate and SE
+    return(list(z[1], z[2]))
+  }
+  else if(inherits(z,'rate')){
+    if(!SE) { # obs and pyrs
+      att <- attributes(z)
+      setDT(z)
+      a <- z[, get(att$rate.meta$obs)]
+      b <- z[, get(att$rate.meta$pyrs)]
+    }
+    else {
+      if(crude) {
+        a <- z[,rate]
+        b <- z[,SE.rate]
+      } 
+      else {
+        # z is a rate object
+        a <- z[,rate.adj]
+        b <- z[,SE.rate.adj]
+      }
+    }
+  }
+  else{
+    stop('Input is not correct: its neighter a vector of two nor a rate object')
+  }
+  return(list(a,b))
+}
+
+  
+  
+
diff --git a/R/lexpand.R b/R/lexpand.R
index 355e41d..6df7be7 100644
--- a/R/lexpand.R
+++ b/R/lexpand.R
@@ -1,941 +1,941 @@
-
-#' @title Split case-level observations
-#' @author Joonas Miettinen
-#' @description Given subject-level data, data is split 
-#' by calendar time (\code{per}), \code{age}, and follow-up
-#' time (\code{fot}, from 0 to the end of follow-up) 
-#' into subject-time-interval rows according to 
-#' given \code{breaks} and additionally processed if requested.
-#' @param data dataset of e.g. cancer cases as rows
-#' @param birth birth time in date format 
-#' or fractional years; string, symbol or expression
-#' @param entry entry time in date format 
-#' or fractional years; string, symbol or expression
-#' @param exit exit from follow-up time in date 
-#' format or fractional years; string, symbol or expression
-#' @param event advanced: time of possible event differing from \code{exit};
-#' typically only used in certain SIR/SMR calculations - see Details; 
-#' string, symbol or expression
-#' @param status variable indicating type of event at \code{exit} or \code{event}; 
-#' e.g. \code{status = status != 0}; expression or quoted variable name
-#' @param entry.status input in the same way as \code{status}; 
-#' status at \code{entry}; see Details
-#' @param id optional; an id variable; e.g. \code{id = my_id};  
-#' string, symbol or expression
-#' @param overlapping advanced, logical; if \code{FALSE} AND if \code{data} contains
-#' multiple rows per subject, 
-#' ensures that the timelines of \code{id}-specific rows do not overlap;
-#' this ensures e.g. that person-years are only computed once per subject 
-#' in a multi-state paradigm
-#' @param aggre e.g. \code{aggre = list(sex, fot)}; 
-#' a list of unquoted variables and/or expressions thereof,
-#' which are interpreted as factors; data events and person-years will
-#' be aggregated by the unique combinations of these; see Details
-#' @param aggre.type one of \code{c("unique","cartesian")};
-#' can be abbreviated; see Details
-#' @param breaks a named list of vectors of time breaks; 
-#' e.g. \code{breaks = list(fot=0:5, age=c(0,45,65,Inf))}; see Details
-#' @param drop logical; if \code{TRUE}, drops all resulting rows 
-#' after splitting that reside outside
-#' the time window as defined by the given breaks (all time scales)
-#' @param pophaz a dataset of population hazards to merge
-#'  with split data; see Details
-#' @param pp logical; if \code{TRUE}, computes Pohar-Perme weights using
-#' \code{pophaz}; adds variable with reserved name \code{pp}; 
-#' see Details for computing method
-#' @param subset a logical vector or any logical condition; data is subsetted
-#' before splitting accordingly
-#' @param merge logical; if \code{TRUE}, retains all 
-#' original variables from the data
-#' @param verbose logical; if \code{TRUE}, the function is chatty and 
-#' returns some messages along the way
-#' @param ... e.g. \code{fot = 0:5}; instead of specifying a \code{breaks} list, 
-#' correctly named breaks vectors can be given 
-#' for \code{fot}, \code{age}, and \code{per}; these override any breaks in the
-#' \code{breaks} list; see Examples
-#' 
-#' 
-#' 
-#' @details 
-#' \strong{Basics}
-#' 
-#' \code{\link{lexpand}} splits a given data set (with e.g. cancer diagnoses 
-#' as rows) to subintervals of time over 
-#' calendar time, age, and follow-up time with given time breaks 
-#' using \code{\link{splitMulti}}.
-#' 
-#' The dataset must contain appropriate 
-#' \code{Date} / \code{IDate} / \code{date} format or
-#' other numeric variables that can be used
-#' as the time variables.
-#' 
-#' You may take a look at a simulated cohort 
-#' \code{\link{sire}} as an example of the
-#' minimum required information for processing data with \code{lexpand}.
-#' 
-#' Many arguments can be supplied as a character string naming the appropriate
-#' variable (e.g. \code{"sex"}), as a symbol (e.g. \code{sex}) or as an expression
-#' (e.g. \code{factor(sex, 0:1, c("m", "f"))}) for flexibility.
-#' 
-#' \strong{Breaks}
-#' 
-#' You should define all breaks as left inclusive and right exclusive 
-#' time points (e.g.\code{[a,b)} )
-#' for 1-3 time dimensions so that the last member of a breaks vector
-#' is a meaningful "final upper limit",
-#'  e.g. \code{per = c(2002,2007,2012)} 
-#' to create a last subinterval of the form \code{[2007,2012)}. 
-#' 
-#' All breaks are explicit, i.e. if \code{drop = TRUE},
-#' any data beyond the outermost breaks points are dropped. 
-#' If one wants to have unspecified upper / lower limits on one time scale,
-#' use \code{Inf}: e.g. \code{breaks = list(fot = 0:5, age = c(0,45,Inf))}.
-#' Breaks for \code{per} can also be given in 
-#' \code{Date}/\code{IDate}/\code{date} format, whereupon
-#' they are converted to fractional years before used in splitting.
-#' 
-#' The \code{age} time scale can additionally 
-#' be automatically split into common age grouping schemes
-#' by naming the scheme with an appropriate character string:
-#' 
-#' \itemize{
-#'   \item \code{"18of5"}: age groups 0-4, 5-9, 10-14, ..., 75-79, 80-84, 85+
-#'   \item \code{"20of5"}: age groups 0-4, 5-9, 10-14, ..., 85-89, 90-94, 95+
-#'   \item \code{"101of1"}: age groups 0, 1, 2, ..., 98, 99, 100+
-#' }
-#' 
-#' \strong{Time variables}
-#' 
-#' If any of the given time variables
-#' (\code{birth}, \code{entry}, \code{exit}, \code{event})
-#' is in any kind of date format, they are first coerced to 
-#' fractional years before splitting
-#' using \code{\link{get.yrs}} (with \code{year.length = "actual"}).
-#' 
-#' Sometimes in e.g. SIR/SMR calculation one may want the event time to differ
-#' from the time of exit from follow-up, if the subject is still considered
-#' to be at risk of the event. If \code{event} is specified, the transition to
-#'  \code{status} is moved to \code{event} from \code{exit} 
-#'  using \code{\link[Epi]{cutLexis}}. See Examples.
-#'  
-#' \strong{The status variable}
-#' 
-#' The statuses in the expanded output (\code{lex.Cst} and \code{lex.Xst})
-#' are determined by using either only \code{status} or both \code{status}
-#' and \code{entry.status}. If \code{entry.status = NULL}, the status at entry
-#' is guessed according to the type of variable supplied via \code{status}:
-#' For numeric variables it will be zero, for factors the first level
-#' (\code{levels(status)[1]}) and otherwise the first unique value in alphabetical
-#' order (\code{sort(unique(status))[1]}). 
-#' 
-#' Using numeric or factor status
-#' variables is strongly recommended. Logical expressions are also allowed
-#' (e.g. \code{status = my_status != 0L}) and are converted to integer internally.
-#' 
-#' \strong{Merging population hazard information}
-#' 
-#' To enable computing relative/net survivals with \code{\link{survtab}}
-#' and \code{\link{relpois}}, \code{lexpand} merges an appropriate
-#' population hazard data (\code{pophaz}) to the expanded data 
-#' before dropping rows outside the specified
-#' time window (if \code{drop = TRUE}). \code{pophaz} must, for this reason, 
-#' contain at a minimum the variables named
-#' \code{agegroup}, \code{year}, and \code{haz}. \code{pophaz} may contain additional variables to specify
-#' different population hazard levels in different strata; e.g. \code{popmort} includes \code{sex}.
-#' All the strata-defining variables must be present in the supplied \code{data}. \code{lexpand} will
-#' automatically detect variables with common names in the two datasets and merge using them.
-#' 
-#' Currently \code{year} must be an integer variable specifying the appropriate year. \code{agegroup}
-#' must currently also specify one-year age groups, e.g. \code{popmort} specifies 101 age groups
-#' of length 1 year. In both
-#' \code{year} and \code{agegroup} variables the values are interpreted as the lower bounds of intervals
-#' (and passed on to a \code{cut} call). The mandatory variable \code{haz}
-#' must specify the appropriate average rate at the person-year level;
-#' e.g. \code{haz = -log(survProb)} where \code{survProb} is a one-year conditional
-#' survival probability will be the correct hazard specification. 
-#' 
-#' The corresponding \code{pophaz} population hazard value is merged by using the mid points
-#' of the records after splitting as reference values. E.g. if \code{age=89.9} at the start
-#' of a 1-year interval, then the reference age value is \code{90.4} for merging. 
-#' This way we get a "typical" population hazard level for each record.
-#' 
-#' \strong{Computing Pohar-Perme weights}
-#' 
-#' If \code{pp = TRUE}, Pohar-Perme weights 
-#' (the inverse of cumulative population survival) are computed. This will
-#' create the new \code{pp} variable in the expanded data. \code{pp} is a
-#' reserved name and \code{lexpand} throws exception if a variable with that name
-#' exists in \code{data}.
-#' 
-#' When a survival interval contains one or several rows per subject
-#' (e.g. due to splitting by the \code{per} scale),
-#' \code{pp} is cumulated from the beginning of the first record in a survival
-#' interval for each subject to the mid-point of the remaining time within that
-#' survival interval, and  that value is given for every other record 
-#' that a given person has within the same survival interval. 
-#' 
-#' E.g. with 5 rows of duration \code{1/5} within a survival interval 
-#' \code{[0,1)]}, \code{pp} is determined for all records by a cumulative 
-#' population survival from \code{0} to \code{0.5}. The existing accuracy is used,
-#' so that the weight is cumulated first up to the end of the second row
-#' and then over the remaining distance to the mid-point (first to 0.4, then to
-#' 0.5). This ensures that more accurately merged population hazards are fully
-#' used.
-#' 
-#' \strong{Event not at end of follow-up & overlapping time lines}
-#' 
-#' \code{event} may be used if the event indicated by \code{status} should
-#' occur at a time differing from \code{exit}. If \code{event} is defined,
-#' \code{cutLexis} is used on the data set after coercing it to the \code{Lexis}
-#' format and before splitting. Note that some values of \code{event} are allowed
-#' to be \code{NA} as with \code{cutLexis} to accommodate observations
-#' without an event occurring.
-#' 
-#' Additionally, setting \code{overlapping = FALSE} ensures that (irrespective
-#' of using \code{event}) the each subject defined by \code{id} only has one
-#' continuous time line instead of possibly overlapping time lines if
-#' there are multiple rows in \code{data} by \code{id}.
-#' 
-#' 
-#' \strong{Aggregating}
-#' 
-#' Certain analyses such as SIR/SMR calculations require tables of events and
-#' person-years by the unique combinations (interactions) of several variables. 
-#' For this, \code{aggre} can be specified as a list of such variables 
-#' (preferably \code{factor} variables but not mandatory)
-#'  and any arbitrary functions of the 
-#' variables at one's disposal. E.g. 
-#' 
-#' \code{aggre = list(sex, agegr = cut(dg_age, 0:100))}
-#' 
-#' would tabulate events and person-years by sex and an ad-hoc age group
-#' variable. Every ad-hoc-created variable should be named.
-#' 
-#' \code{fot}, \code{per}, and \code{age} are special reserved variables which,
-#' when present in the \code{aggre} list, are output as categories of the
-#' corresponding time scale variables by using 
-#' e.g. 
-#' 
-#' \code{cut(fot, breaks$fot, right=FALSE)}. 
-#' 
-#' This only works if
-#' the corresponding breaks are defined in \code{breaks} or via "\code{...}".
-#' E.g. 
-#' 
-#' \code{aggre = list(sex, fot.int = fot)} with 
-#' 
-#' \code{breaks = list(fot=0:5)}.
-#' 
-#' The output variable \code{fot.int} in the above example will have
-#' the lower limits of the appropriate intervals as values.
-#' 
-#' \code{aggre} as a named list will output numbers of events and person-years
-#' with the given new names as categorizing variable names, e.g. 
-#' \code{aggre = list(follow_up = fot, gender = sex, agegroup = age)}.
-#' 
-#' The output table has person-years (\code{pyrs}) and event counts
-#' (e.g. \code{from0to1}) as columns. Event counts are the numbers of transitions
-#' (\code{lex.Cst != lex.Xst}) or the \code{lex.Xst} value at a subject's 
-#' last record (subject possibly defined by \code{id}).
-#' 
-#' If \code{aggre.type = "unique"} (alias \code{"non-empty"}), 
-#' the above results are computed for existing
-#' combinations of expressions given in \code{aggre}, but also for non-existing
-#' combinations if \code{aggre.type = "cartesian"} (alias \code{"full"}). E.g. if a
-#' factor variable has levels \code{"a", "b", "c"} but the data is limited
-#' to only have levels \code{"a", "b"} present 
-#' (more than zero rows have these level values), the former setting only
-#' computes results for \code{"a", "b"}, and the latter also for \code{"c"}
-#' and any combination with other variables or expression given in \code{aggre}.
-#' In essence, \code{"cartesian"} forces also combinations of variables used
-#' in \code{aggre} that have no match in data to be shown in the result.
-#' 
-#' If \code{aggre} is not \code{NULL} and \code{pophaz} has been supplied,
-#' \code{lexpand} also aggregates the expected counts of events, which
-#' appears in the output data by the reserved name \code{d.exp}. Additionally,
-#' having \code{pp = TRUE} causes \code{lexpand} to also compute various
-#' Pohar-Perme weighted figures necessary for computing Pohar-Perme net survivals
-#' with \code{\link{survtab_ag}}. This can be slow, so consider what is really
-#' needed. The Pohar-Perme weighted figures have the suffix \code{.pp}. 
-#' 
-#' @return
-#' If \code{aggre = NULL}, returns 
-#' a \code{data.table} or \code{data.frame} 
-#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-#' object expanded to accommodate split observations with time scales as
-#' fractional years and \code{pophaz} merged in if given. Population
-#' hazard levels in new variable \code{pop.haz}, and Pohar-Perme
-#' weights as new variable \code{pp} if requested.
-#' 
-#' If \code{aggre} is defined, returns a long-format 
-#' \code{data.table}/\code{data.frame} with the variable \code{pyrs} (person-years),
-#' and variables for the counts of transitions in state or state at end of 
-#' follow-up formatted \code{fromXtoY}, where \code{X} and \code{Y} are 
-#' the states transitioned from and to, respectively. The data may also have
-#' the columns \code{d.exp} for expected numbers of cases and various
-#' Pohar-Perme weighted figures as identified by the suffix \code{.pp}; see 
-#' Details.
-#' 
-#' 
-#' @examples
-#' \donttest{
-#' ## prepare data for e.g. 5-year cohort survival calculation
-#' x <- lexpand(sire, breaks=list(fot=seq(0, 5, by = 1/12)), 
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'              status =  status != 0, pophaz=popmort)
-#' 
-#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
-#' BL <- list(fot = seq(0, 5, by = 1/12), per = c("2008-01-01", "2013-01-01"))
-#' x <- lexpand(sire, breaks = BL, 
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'              pophaz=popmort, status =  status != 0)
-#' 
-#' ## aggregating
-#' BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-#' ag <- lexpand(sire, breaks = BL, status = status != 0, 
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'               aggre=list(sex, period = per, surv.int = fot))
-#' 
-#' ## aggregating even more
-#' ag <- lexpand(sire, breaks = BL, status = status != 0, 
-#'               birth = bi_date, entry = dg_date, exit = ex_date,
-#'               aggre=list(sex, period = per, surv.int = fot),
-#'               pophaz = popmort, pp = TRUE)
-#' 
-#' ## using "..."
-#' x <- lexpand(sire, fot=0:5, status =  status != 0,
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'              pophaz=popmort) 
-#' 
-#' x <- lexpand(sire, fot=0:5, status =  status != 0, 
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'              aggre=list(sex, surv.int = fot))
-#'              
-#' ## using the "event" argument: it just places the transition to given "status"
-#' ## at the "event" time instead of at the end, if possible using cutLexis
-#' x <- lexpand(sire, status = status, event = dg_date,
-#'              birth = bi_date, entry = dg_date, exit = ex_date,) 
-#' 
-#' ## aggregating with custom "event" time
-#' ## (the transition to status is moved to the "event" time)
-#' x <- lexpand(sire, status = status, event = dg_date, 
-#'              birth = bi_date, entry = dg_date, exit = ex_date,
-#'              per = 1970:2014, age = c(0:100,Inf),
-#'              aggre = list(sex, year = per, agegroup = age)) 
-#' 
-#' }
-#' 
-#' @import data.table
-#' @import Epi
-#' @family splitting functions
-#' @family aggregation functions
-#' @seealso
-#' \code{\link[Epi]{Lexis}}, \code{\link{popmort}}
-#' @export
-lexpand <- function(data, 
-                    birth=NULL, entry=NULL, exit=NULL, event=NULL,
-                    status = status != 0,
-                    entry.status = NULL,
-                    breaks = list(fot=c(0,Inf)),
-                    id = NULL,
-                    overlapping = TRUE,
-                    aggre = NULL,
-                    aggre.type = c("unique", "cartesian"),
-                    drop=TRUE,
-                    pophaz = NULL, pp = TRUE, 
-                    subset = NULL,
-                    merge=TRUE, verbose = FALSE,
-                    ...) {
-  start_time <- proc.time()
-  
-  TF <- environment()
-  PF <- parent.frame(1L)
-  
-  ## data checks
-  if ( missing(data) || nrow(data) == 0) stop("no data found")
-  
-  if (!is.data.frame(data)) stop("data must be a data.frame or data.table")
-  
-  ## to instate global variables to appease R CMD CHECK 
-  .EACHI <- lex.status <- lexpand.id <- lex.exit <- lex.birth <- 
-    lex.entry <- lex.event <- temp.id <- cd <- fot <- age <- per <- 
-    lex.id <- lex.multi <- pop.haz <- lex.Cst <- lex.Xst <- lex.dur <- NULL
-  
-  
-  ## test conflicting variable names -------------------------------------------
-  added_vars <- c("fot", "per", "age", "lex.id", "lex.dur", "lex.Xst", "lex.Cst")
-  if (!is.null(pophaz)) added_vars <- if (pp) c(added_vars, "pp", "pop.haz") else c(added_vars, "pop.haz")
-  conflicted_vars <- intersect(added_vars, names(data))
-  
-  if (merge && length(conflicted_vars) > 0) {
-    conflicted_vars <- paste0("'", conflicted_vars, "'", collapse = ", ")
-    warning("'data' already had variable(s) named ", conflicted_vars, " which lexpand will create, and you have merge = TRUE; this may result in unexpected problems. Rename the variable(s)?")
-  }
-  rm(added_vars, conflicted_vars)
-  
-  ## test aggre type -----------------------------------------------------------
-  aggre.type <- match.arg(aggre.type[1L], c("cartesian", "non-empty", "unique", "cross-product", "full"))
-  if (aggre.type == "cross-product") {
-    aggre.type <- "cartesian"
-    warning("aggre.type value 'cross-product' deprecated and renamed to 'cartesian'; please use that in the future")
-  }
-  
-  ## subsetting-----------------------------------------------------------------
-  ## no copy taken of data!
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset)
-  
-  ## prepping time variables ---------------------------------------------------
-  l <- substitute(list(birth, entry, exit, event, status, entry.status, id))
-  rm(birth, entry, exit, event, status, entry.status, id)
-  
-  lc <- unlist(lapply(l, deparse))
-  lc <- lc[-1] ## first always "list"
-  
-  wh <- which(lc != "NULL")
-  lex_vars <- c("lex.birth","lex.entry","lex.exit","lex.event", "lex.status", "lex.entry.status", "lexpand.id")[wh]
-  if (any(!c("lex.birth", "lex.entry", "lex.exit", "lex.status") %in% lex_vars)) stop("birth, entry, exit and status are mandatory")
-  
-  l <- eval(l, envir = data[subset, ], enclos = PF)
-  l[-wh] <- NULL
-  
-  
-  ## vars can be given as character strings of variable names
-  isChar  <- sapply(l, is.character, simplify = TRUE)
-  if (any(isChar)) {
-    isShort <- sapply(l, function(x) {length(x) == 1L}, simplify = TRUE)
-    whOneChar <- which(isShort & isChar)    
-    
-    whBadChar <- NULL
-    if (length(whOneChar) > 0) {
-      testBadChar <- unlist(l[whOneChar])
-      whBadChar <- whOneChar[!testBadChar %in% names(data)]
-    }
-    
-    if (length(whBadChar) > 0) {
-      
-      badChar <- l[whBadChar]
-      badChar <- paste0(badChar[1:min(length(badChar), 5L)], collapse = ", ")
-      stop("Variables given as a character of length one are interpreted as variable names in data, 
-           but some given characters were not found in data; 
-           check names or input as factor/Date; 
-           first five bad names: ", badChar)
-    }
-    
-    l[whOneChar] <- lapply(l[whOneChar], function(x) {data[subset, ][[x]]})
-  }
-  
-  
-  l <- as.data.table(l)
-  setnames(l, names(l), lex_vars)
-  
-  
-  rm(lex_vars)
-  if (!all(c("lex.birth","lex.entry","lex.exit","lex.status") %in% names(l))) {
-    stop("birth, entry, exit and status are mandatory, but at least one was misspecified/NULL")
-  }
-  
-  if (is.logical(l$lex.status)) l[, lex.status := as.integer(lex.status)]
-  if (is.null(l$lexpand.id)) l[, lexpand.id  := 1:.N]
-  
-  ## checks for merging style --------------------------------------------------
-  if (!is.null(pophaz)) {
-    all_names_present(pophaz, c("agegroup","year","haz"))
-    othMergeVars <- setdiff(names(pophaz), c("agegroup","year","haz"))
-    badOthMergeVars <- setdiff(othMergeVars, names(data))
-    if (length(badOthMergeVars) > 0) {
-      badOthMergeVars <- paste0("'", badOthMergeVars, "'", collapse = ", ")
-      stop("Following variables exist in pophaz but do not exist in data: ", badOthMergeVars, ". Make sure data and pophaz contain variables with the same names that you intend to merge by.")
-    }
-  }
-  
-  if (is.null(pophaz)) {
-    comp_pp <- FALSE
-  } else {
-    comp_pp <- TRUE
-  }
-  
-  ## internally we have "delta" and "actual" methods, 
-  ## the latter being experimental and not visible in the documentation.
-  ## "delta" is always used if pp = TRUE.
-  ## it is possible to choose pp = "actual" as well, though, if you know
-  ## about it.
-  comp_pp <- FALSE
-  if (is.logical(pp) && pp) pp <- "delta"
-  
-  if (!is.null(pp) && is.character(pp)) {
-    pp <- match.arg(pp, c("delta", "actual"))
-    comp_pp <- TRUE
-  } 
-  if (comp_pp && "pp" %in% names(data)) stop("variable named 'pp' in data; this is a reserved name for pohar-perme weights, please rename / remove the variable in data")
-  
-  ## ensure given breaks make any sense ----------------------------------------
-  
-  bl <- list(...)
-  lna <- names(bl)
-  bad_lna <- setdiff(lna, c("fot","per","age"))
-  if (length(bad_lna) > 0) {
-    bad_lna <- paste0("'", bad_lna, "'", collapse = ", ")
-    stop("only arguments named 'fot', 'per' or 'age' currently allowed to be passed via '...'; did you mistype an argument? bad args: ", bad_lna)
-  }
-  lna <- intersect(names(bl), c("fot","per","age"))
-  if (length(lna) > 0) {
-    bl <- bl[lna]
-    if (!is.null(breaks)) breaks[lna] <- NULL
-    breaks <- c(breaks, bl)
-  }
-  rm(bl, lna)
-  
-  brna <- names(breaks)
-  if (length(brna) != length(breaks)) {
-    stop("all elements in breaks list must be named, e.g. list(fot = 0:5, age=c(0,45,65,Inf))")
-  }
-  
-  brna <- intersect(brna, c("fot","per","age"))
-  if (length(brna) == 0) {
-    breaks$fot <- c(0,Inf)
-  }
-  
-  if ("age" %in% brna && is.character(breaks$age)) {
-    schemeNames <- c("18of5", "20of5", "101of1")
-    if (!breaks$age %in% schemeNames) stop("You supplied '", breaks$age, "' as breaks for the age scale, but allowed character strings are: ", paste0("'", schemeNames, "'", collapse = ","))
-    brSchemes <- list(c(seq(0, 85, 5)), c(seq(0, 95, 5), Inf), c(0:100, Inf))
-    names(brSchemes) <- paste0("age_", schemeNames)
-    breaks$age <- brSchemes[paste0("age_",breaks$age)]
-  } 
-  
-  
-  if (any(sapply(breaks, length) == 1L)) {
-    stop("any given non-null vector of breaks must have more than one break!")
-  }
-  
-  # convert to fractional years ------------------------------------------------
-  
-  char2date <- function(obj) {
-    if (is.character(obj) || inherits(obj, "date")) {
-      return(as.IDate(obj))
-    } else {
-      return(obj)
-    }
-  }
-  
-  date2yrs <- function(obj) {    
-    if (is.Date(obj) || inherits(obj, "date")) {
-      get.yrs(obj, year.length = "actual")
-    } else {
-      obj
-    }
-  }
-  
-  breaks <- lapply(breaks, char2date)
-  breaks <- lapply(breaks, date2yrs)
-  
-  time_vars <- intersect(names(l), c("lex.birth", "lex.entry", "lex.exit","lex.event"))
-  l[, (time_vars) := lapply(.SD, date2yrs) , .SDcols = time_vars]
-  
-  if (verbose) cat("given birth, entry, exit, status etc. variables after coercion to numeric \n")
-  if (verbose) print(l)
-  
-  # check data consistency for overlapping = FALSE -----------------------------
-  ## not allowed: for any one unique subject to be true for 
-  ## multiple rows (if overlapping = TRUE):
-  ## * same event values
-  ## * same entry values
-  if (!overlapping) {
-    if ("lex.event" %in% names(l)) {
-      if (all(is.na(l$lex.event))) stop("ALL 'event' values are NA; if this is as intended, please use event = NULL instead")
-      
-      if (any(duplicated(l, by = c("lexpand.id", "lex.event")))) {
-        stop("subject(s) defined by lex.id had several rows where 'event' time had the same value, which is not supported with overlapping = FALSE; perhaps separate them by one day?")
-      } 
-      if (any(l[!is.na(lex.event), lex.entry == lex.event])) {
-        stop("some rows have simultaneous 'entry' and 'event', which is not supported with overlapping = FALSE; perhaps separate them by one day?")
-      }
-    } else if (any(duplicated(l, by = c("lexpand.id", "lex.exit")))) {
-      stop("subject(s) defined by lex.id had several rows where 'exit' time had the same value, which is not supported without 'event' defined; use 'event' or perhaps separate them by one day?")
-    }
-    
-    
-  }
-  
-  
-  # dropping unuseful records --------------------------------------------------
-  test_times <- function(condition, msg, old_subset=l_subset, DT=l) {
-    
-    condition <- substitute(condition)
-    condition <- eval(condition, envir = DT, enclos = parent.frame(1L))
-    
-    new_subset <- old_subset & !(condition & !is.na(condition))
-    old_n <- sum(old_subset)
-    new_n <- sum(new_subset)
-    
-    if (new_n == 0L) {
-      stop("dropping rows where ", msg, " resulted in zero rows left. likely problem: misdefined time variables")
-    }
-    
-    if (new_n < old_n) {
-      message(paste0("dropped ", old_n-new_n, " rows where ", msg))
-    }
-    return(new_subset)
-  }
-  
-  l_subset <- rep(TRUE, nrow(l))
-  
-  l_subset <- test_times(is.na(lex.birth), "birth values are missing")
-  l_subset <- test_times(is.na(lex.entry), "entry values are missing")
-  l_subset <- test_times(is.na(lex.exit), "exit values are missing")
-  
-  if (!is.null(breaks$per)) {
-    l_subset <- test_times(lex.exit < min(breaks$per), "subjects left follow-up before earliest per breaks value")
-  }
-  if (!is.null(breaks$age)) {
-    l_subset <- test_times(lex.exit - lex.birth < min(breaks$age), "subjects left follow-up before lowest age breaks value")
-  }
-  if (!is.null(breaks$fot)) {
-    l_subset <- test_times(lex.exit - lex.entry < min(breaks$fot), "subjects left follow-up before lowest fot breaks value")
-  }
-  l_subset <- test_times(lex.birth >= lex.exit, "birth >= exit")
-  l_subset <- test_times(lex.entry == lex.exit, "entry == exit")
-  l_subset <- test_times(lex.entry >  lex.exit, "entry >  exit")
-  l_subset <- test_times(lex.birth >  lex.entry, "birth > entry")
-  if (!is.null(l$lex.event)) {
-    l_subset <- test_times(lex.event > lex.exit, "event > exit")
-    l_subset <- test_times(lex.event < lex.entry, "event < entry")
-  }
-  l <- l[l_subset]
-  
-  if (verbose) cat("Time taken by checks, prepping and test: ", timetaken(start_time), "\n")
-  
-  # Lexis coercion -------------------------------------------------------------
-  
-  ## status definitions
-  setnames(l, "lex.status", "lex.Xst")
-  if ("lex.entry.status" %in% names(l)) {
-    setnames(l, "lex.entry.status", "lex.Cst")
-  } else {
-    if (is.factor(l$lex.Xst)) {
-      l[, lex.Cst := factor(levels(lex.Xst)[1L], levels=levels(lex.Xst))]
-    } else if (is.double(l$lex.Xst)) {
-      l[, lex.Cst := 0]
-    } else if (is.integer(l$lex.Xst)) {
-      l[, lex.Cst := 0L]
-    } else {
-      l[, lex.Cst := sort(unique(lex.Xst))[1L]]
-    }
-   
-  }
-  
-  # ensure common labels for factors etc.
-  harmonizeStatuses(x = l, C = "lex.Cst", X = "lex.Xst")
-  
-  ## time scales and duration
-  l[, lex.dur := lex.exit - lex.entry]
-  l[, fot := 0]
-  setnames(l, "lex.entry", "per")
-  l[, age := per-lex.birth]
-  setnames(l, "lexpand.id", "lex.id")
-  
-  ## for merging data with l later
-  if (merge) {
-    idt <- data.table(temp.id = 1:nrow(l))
-    l[, temp.id := 1:.N]
-  }
-  
-  ## crop time scale values to obey breaks limits and drop if necessary
-  ## NOTE: goes wrong if need to compute pp weights!
-  #   if (drop && !pp) {
-  #     intelliCrop(x = l, breaks = breaks, allScales = c("fot", "per", "age"), cropStatuses = TRUE)
-  #     l <- intelliDrop(x = l, breaks = breaks, dropNegDur = TRUE)
-  #   }
-  
-  
-  setcolsnull(l, colorder=TRUE, soft=TRUE, 
-              keep = c("lex.id","fot","per","age",
-                       "lex.dur", "lex.Cst", "lex.Xst", "lex.event", "temp.id"))
-  setattr(l, "class", c("Lexis", "data.table", "data.frame"))
-  setattr(l, "time.scales", c("fot","per","age"))
-  setattr(l, "time.since", c("","",""))
-  
-  if (verbose) cat("data just after Lexis coercion: \n")
-  if (verbose) print(l)  
-  
-  # event not at exit time -----------------------------------------------------
-  
-  if ("lex.event" %in% names(l)) {
-    
-    if (!overlapping) {
-      
-      ## using lex.event time, ensure coherence of lex.Cst & lex.Xst
-      ## before cutLexis()
-      tmpFE <- makeTempVarName(l, pre = "fot_end_")
-      l[, (tmpFE) := fot + lex.dur]
-      setkeyv(l, c("lex.id", "lex.event", tmpFE))
-      tmpLX <- makeTempVarName(l, pre = "lag_lex.Xst_")
-      l[, (tmpLX) := shift(lex.Xst, n = 1, type = "lag"), by = lex.id]
-      l[!is.na(get(tmpLX)), lex.Cst := get(tmpLX)]
-      l[, c(tmpFE, tmpLX) := NULL]
-      rm(tmpFE, tmpLX)
-      
-    }
-    
-    if (verbose) cutt <- proc.time()
-    setDF(l)
-    setattr(l, "class", c("Lexis", "data.frame"))
-    l <- Epi::cutLexis(l, cut = l$lex.event, timescale = "per", new.state = l$lex.Xst, precursor.states = unique(l$lex.Cst))
-    setDT(l)
-    setattr(l, "class", c("Lexis", "data.table", "data.frame"))
-    if (verbose) cat("Time taken by cutLexis when defining event time points: ", timetaken(cutt), "\n")
-    
-    if (verbose) cat("Data just after using cutLexis: \n")
-    if (verbose) print(l[])
-    
-  }
-  
-  
-  # overlapping timelines? -----------------------------------------------------
-  
-  if (!overlapping && any(duplicated(l$lex.id))) {
-    tmpFE <- makeTempVarName(l, pre = "fot_end_")
-    l[, (tmpFE) := fot + lex.dur]
-    ## don't keep duplicated rows:
-    ## same end points imply fully overlapping time lines
-    ## e.g. 
-    ## --->
-    ## ->
-    ##  -->
-    ## results in 
-    ## ->
-    ## --->
-    ## we only keep the longest time line with a unique end point.
-
-    # setkeyv(l,  c("lex.id", tmpFE, "fot"))
-    tmpLE <- intersect(names(l), "lex.event")
-    LEval <- if (length(tmpLE) == 0) NULL else -1
-
-    setorderv(l, c("lex.id", tmpFE, tmpLE, "fot"), c(1,1,LEval,1))
-    l <- unique(l, by = c("lex.id", tmpFE))
-    
-    ## end points are kept but starting points are "rolled"
-    ## from first to last row by lex.id to ensure non-overlappingness; e.g.
-    ## ->
-    ## --->
-    ## results in 
-    ## ->
-    ##   ->
-    # setkeyv(l, c("lex.id", tmpFE))
-    # setorderv(l, c("lex.id", tmpLE, tmpFE), c(1, LEval, 1))
-    setkeyv(l, c("lex.id", tmpLE, tmpFE))
-    
-    if (verbose) cat("data just before fixing overlapping time lines \n")
-    if (verbose) print(l)
-    l[, lex.dur := get(tmpFE) - c(min(fot), get(tmpFE)[-.N]), by = lex.id]
-    l[, fot := get(tmpFE) - lex.dur]
-    cumDur <- l[,  list(age = min(age), per = min(per), cd = c(0, cumsum(lex.dur)[-.N])), by = lex.id]
-    cumDur[, age := age+cd]
-    cumDur[, per := per+cd]
-    l[, age := cumDur$age]
-    l[, per := cumDur$per]
-    l[, (tmpFE) := NULL]; rm(cumDur)
-    
-    
-    ## if event used, first row up to event, second row from first event to etc...
-  }
-  
-  setcolsnull(l, "lex.event", soft = TRUE) ## note: lex.event needed in overlapping procedures
-  
-  if (verbose) cat("time and status variables before splitting: \n")
-  if (verbose) print(l)
-  if ("id" %in% ls()) rm("id")
-  
-  
-  # splitting ------------------------------------------------------------------
-  
-  ## determine whether to drop data only after splitting and merging
-  drop_after <- FALSE
-  if (drop == TRUE && comp_pp) {
-    drop <- FALSE
-    drop_after <- TRUE
-  }
-  
-  forceLexisDT(l, breaks = list(fot = NULL, per = NULL, age = NULL),
-               allScales = c("fot", "per", "age"))
-  if (verbose) splittime <- proc.time()
-  l <- splitMulti(l,  breaks = breaks, 
-                  drop = drop, verbose=FALSE, merge = TRUE)
-  setDT(l)
-  setkey(l, lex.id, fot)
-  l[, lex.multi := 1:.N, by = lex.id]
-  if (verbose) cat("Time taken by splitting:", timetaken(splittime), "\n")
-  
-  # merging other variables from data ------------------------------------------
-  
-  if (merge) {
-    setkey(l, temp.id)
-    
-    temp <- data.table(idt, data[subset & !is.na(subset), ][l_subset, ])
-    setkey(temp, temp.id)
-    
-    l <- temp[l]
-    
-    rm(temp, idt)
-    setcolsnull(l, "temp.id")
-    
-    lex_vars <- c("lex.id","lex.multi","fot","per","age", "lex.dur", "lex.Cst", "lex.Xst")
-    setcolorder(l, c(lex_vars, setdiff(names(l), lex_vars)))
-  }
-  rm(data, subset, l_subset)
-  
-  ## aggregating checks --------------------------------------------------------
-  ## NOTE: aggre evaled here using small data subset to check that all needed
-  ## variables are found, etc.
-  aggSub <- substitute(aggre)
-  agTest <- evalPopArg(arg = aggSub, data = l[1:min(10L, .N), ], 
-                       enclos = PF, recursive = TRUE, DT = TRUE)
-  agTy <- attr(agTest, "arg.type")
-  if (is.null(agTy)) agTy <- "NULL"
-  aggSub <- attr(agTest, "quoted.arg")
-  agVars <- attr(agTest, "all.vars")
-  rm(aggre)
-  
-  # merging pophaz and pp-weighting --------------------------------------------
-  if (!is.null(pophaz)) {
-    
-    pophaztime <- proc.time()
-    
-    if (any(c("haz", "pop.haz") %in% names(l))) stop("case data had variable(s) named 'haz' / 'pop.haz', which are reserved for lexpand's internal use. rename/remove them please.")
-    # merge surv.int information -----------------------------------------------
-    NULL_FOT <- FALSE
-    if (is.null(breaks$fot)) {
-      breaks$fot <- l[, c(0, max(fot+lex.dur))] 
-      NULL_FOT <- TRUE
-    }
-    
-    breaks$fot <- sort(unique(breaks$fot))
-    # handle pophaz data -------------------------------------------------------
-    
-    if (!"haz" %in% names(pophaz)) stop("no 'haz' variable in pophaz; please rename you hazard variable to 'haz'")
-    yBy <- xBy <- setdiff(names(pophaz), c("haz"))
-    if (c("year") %in% yBy) xBy[yBy == "year"] <- "per"
-    if (c("agegroup") %in% yBy) xBy[yBy == "agegroup"] <- "age"
-    yByOth <- setdiff(yBy, c("year", "agegroup"))
-    
-    if (any(!yByOth %in% names(l))) 
-      stop("Following variable names not common between pophaz and data: ", paste0("'", yByOth[!yByOth %in% names(l)], "'", collapse = ", "))
-    
-    l <- cutLowMerge(x = l, y = pophaz, by.x = xBy, by.y = yBy, all.x = TRUE,
-                     all.y = FALSE, mid.scales = c("per", "age"), old.nums = TRUE)
-    setnames(l, "haz", "pop.haz")
-    
-    ## check if l's merging time variables were within pophaz's limits ---------
-    nNA <- l[is.na(pop.haz), .N]
-    if (nNA > 0) message("WARNING: after merging pophaz, ", nNA, " rows in split data have NA hazard values!")
-    
-    names(yBy) <- xBy
-    names(xBy) <- yBy
-    for (k in intersect(c("per", "age"), xBy)) {
-      yVar <- yBy[k]
-      kLo <- min(pophaz[[yVar]])
-      kHi <- max(pophaz[[yVar]])
-      mid <- l[, get(k) + lex.dur]
-      nLo <- sum(mid < kLo - .Machine$double.eps^0.5)
-      nHi <- sum(mid > kHi - .Machine$double.eps^0.5)
-      if (nLo > 0) message("WARNING: ", nLo, " rows in split data have NA values due to their mid-points residing below the minimum value of '", yVar, "' in pophaz!")
-      if (nHi > 0) message("NOTE: ", nHi, " rows in split data had values of '", k, "' higher than max of pophaz's '", yVar, "'; the hazard values at '", yVar, "' == ", kHi, " were used for these")
-    }
-    rm(mid)
-    for (k in yByOth) {
-      levsNotOth <- setdiff(unique(l[[k]]), unique(pophaz[[k]]))
-      if (length(levsNotOth) > 0) message("WARNING: following levels (first five) of variable '", k, "' not in pophaz but exist in split data: ", paste0("'",levsNotOth[1:5],"'", collapse = ", "))
-    }
-    
-    
-    # pohar-perme weighting ----------------------------------------------------
-    if (comp_pp) {
-      setkeyv(l, c("lex.id", "fot"))
-      comp_pp_weights(l, surv.scale = "fot", breaks = breaks$fot, haz = "pop.haz", 
-                      style = "delta", verbose = verbose)
-    }
-    merge_msg <- "Time taken by merging pophaz"
-    if (comp_pp) merge_msg <- paste0(merge_msg, " and computing pp")
-    merge_msg <- paste0(merge_msg, ": ")
-    if (verbose) cat(paste0(merge_msg, timetaken(pophaztime), "\n"))
-    
-    
-  }
-  
-  # dropping after merging -----------------------------------------------------
-  if (drop_after) {
-    l <- intelliDrop(x = l, breaks = breaks)
-  }
-  
-  if (verbose) cat("Number of rows after splitting: ", nrow(l),"\n")
-  
-  
-  # aggregating if appropriate -------------------------------------------------
-  if (agTy != "NULL") {
-    
-    setcolsnull(l, keep = c("lex.id","lex.dur", "fot", "per", "age", "lex.Cst", "lex.Xst", agVars, "pop.haz", "pp"))
-    
-    sumVars <- NULL
-    if ("pop.haz" %in% names(l)) {
-      if ("d.exp" %in% names(l)) stop("data had variable named 'd.exp' by which to aggregate, which would be overwritten due to aggregating expected numbers of cases (you have supplied pophaz AND are aggregating); please rename / remove it first.")
-      l[, c("d.exp") := pop.haz*lex.dur ]
-      sumVars <- c(sumVars, "d.exp")
-    }
-    if ("pop.haz" %in% names(l) && comp_pp && "pp" %in% names(l)) {
-      forceLexisDT(l, breaks = breaks, allScales = c("fot", "per", "age"))
-      ppFigs <- comp_pp_weighted_figures(lex = l, haz = "pop.haz", pp = "pp", event.ind = NULL)
-      bad_pp_vars <- intersect(names(ppFigs), names(l))
-      if (length(bad_pp_vars) > 0L) {
-        bad_pp_vars <- paste0("'",bad_pp_vars, "'", collapse = ", ")
-        stop("Data had variable(s) named ", bad_pp_vars, ", by which to aggregate, which would be overwritten due to aggregating expected numbers of cases (you have supplied pophaz AND are aggregating); please rename / remove them first")
-      }
-      l[, names(ppFigs) := ppFigs]
-      sumVars <- c(sumVars, names(ppFigs))
-      rm(ppFigs)
-      
-    }
-    
-    if (verbose) cat("Starting aggregation of split data... \n")
-    setDT(l)
-    forceLexisDT(l, allScales = c("fot", "per", "age"), breaks = breaks)
-    l <- try(aggre(lex = l, by = aggSub, type = aggre.type, verbose = verbose, sum.values = sumVars))
-    if (inherits(l, "try-error")) stop("Something went wrong when calling aggre() within lexpand(). Usual suspect: bad 'by' argument. Error message from aggre(): 
-                                       ", paste0(l[[1]]))
-    if (verbose) cat("Aggregation done. \n")
-    
-    if (!return_DT() && is.data.table(l)) setDFpe(l)
-    
-  } else {
-    
-    
-    # last touch-up --------------------------------------------------------------
-    ## sometimes problems with releasing memory
-    gc()
-    
-    breaks <- lapply(c("fot","per","age"), function(ts_nm) {
-      breaks[[ts_nm]]
-    })
-    names(breaks) <- c("fot","per","age")
-    
-    ## handle attributes
-    setkeyv(l, c("lex.id", "lex.multi"))
-    set(l, j = "lex.multi", value = NULL)
-    setattr(l, "time.scales", c("fot","per","age"))
-    setattr(l, "time.since", c("","",""))
-    setattr(l, "breaks", breaks)
-    setattr(l, "class", c("Lexis","data.table","data.frame"))
-    if (!return_DT() && is.data.table(l)) setDFpe(l)
-    
-    
-    
-  }
-  
-  if (verbose) cat("Time taken by lexpand(): ", timetaken(start_time), "\n")
-  
-  return(l[])
-}
-
-
-globalVariables(c('.EACHI', "dg_date", "ex_date", "bi_date"))
-
+
+#' @title Split case-level observations
+#' @author Joonas Miettinen
+#' @description Given subject-level data, data is split 
+#' by calendar time (\code{per}), \code{age}, and follow-up
+#' time (\code{fot}, from 0 to the end of follow-up) 
+#' into subject-time-interval rows according to 
+#' given \code{breaks} and additionally processed if requested.
+#' @param data dataset of e.g. cancer cases as rows
+#' @param birth birth time in date format 
+#' or fractional years; string, symbol or expression
+#' @param entry entry time in date format 
+#' or fractional years; string, symbol or expression
+#' @param exit exit from follow-up time in date 
+#' format or fractional years; string, symbol or expression
+#' @param event advanced: time of possible event differing from \code{exit};
+#' typically only used in certain SIR/SMR calculations - see Details; 
+#' string, symbol or expression
+#' @param status variable indicating type of event at \code{exit} or \code{event}; 
+#' e.g. \code{status = status != 0}; expression or quoted variable name
+#' @param entry.status input in the same way as \code{status}; 
+#' status at \code{entry}; see Details
+#' @param id optional; an id variable; e.g. \code{id = my_id};  
+#' string, symbol or expression
+#' @param overlapping advanced, logical; if \code{FALSE} AND if \code{data} contains
+#' multiple rows per subject, 
+#' ensures that the timelines of \code{id}-specific rows do not overlap;
+#' this ensures e.g. that person-years are only computed once per subject 
+#' in a multi-state paradigm
+#' @param aggre e.g. \code{aggre = list(sex, fot)}; 
+#' a list of unquoted variables and/or expressions thereof,
+#' which are interpreted as factors; data events and person-years will
+#' be aggregated by the unique combinations of these; see Details
+#' @param aggre.type one of \code{c("unique","cartesian")};
+#' can be abbreviated; see Details
+#' @param breaks a named list of vectors of time breaks; 
+#' e.g. \code{breaks = list(fot=0:5, age=c(0,45,65,Inf))}; see Details
+#' @param drop logical; if \code{TRUE}, drops all resulting rows 
+#' after splitting that reside outside
+#' the time window as defined by the given breaks (all time scales)
+#' @param pophaz a dataset of population hazards to merge
+#'  with split data; see Details
+#' @param pp logical; if \code{TRUE}, computes Pohar-Perme weights using
+#' \code{pophaz}; adds variable with reserved name \code{pp}; 
+#' see Details for computing method
+#' @param subset a logical vector or any logical condition; data is subsetted
+#' before splitting accordingly
+#' @param merge logical; if \code{TRUE}, retains all 
+#' original variables from the data
+#' @param verbose logical; if \code{TRUE}, the function is chatty and 
+#' returns some messages along the way
+#' @param ... e.g. \code{fot = 0:5}; instead of specifying a \code{breaks} list, 
+#' correctly named breaks vectors can be given 
+#' for \code{fot}, \code{age}, and \code{per}; these override any breaks in the
+#' \code{breaks} list; see Examples
+#' 
+#' 
+#' 
+#' @details 
+#' \strong{Basics}
+#' 
+#' \code{\link{lexpand}} splits a given data set (with e.g. cancer diagnoses 
+#' as rows) to subintervals of time over 
+#' calendar time, age, and follow-up time with given time breaks 
+#' using \code{\link{splitMulti}}.
+#' 
+#' The dataset must contain appropriate 
+#' \code{Date} / \code{IDate} / \code{date} format or
+#' other numeric variables that can be used
+#' as the time variables.
+#' 
+#' You may take a look at a simulated cohort 
+#' \code{\link{sire}} as an example of the
+#' minimum required information for processing data with \code{lexpand}.
+#' 
+#' Many arguments can be supplied as a character string naming the appropriate
+#' variable (e.g. \code{"sex"}), as a symbol (e.g. \code{sex}) or as an expression
+#' (e.g. \code{factor(sex, 0:1, c("m", "f"))}) for flexibility.
+#' 
+#' \strong{Breaks}
+#' 
+#' You should define all breaks as left inclusive and right exclusive 
+#' time points (e.g.\code{[a,b)} )
+#' for 1-3 time dimensions so that the last member of a breaks vector
+#' is a meaningful "final upper limit",
+#'  e.g. \code{per = c(2002,2007,2012)} 
+#' to create a last subinterval of the form \code{[2007,2012)}. 
+#' 
+#' All breaks are explicit, i.e. if \code{drop = TRUE},
+#' any data beyond the outermost breaks points are dropped. 
+#' If one wants to have unspecified upper / lower limits on one time scale,
+#' use \code{Inf}: e.g. \code{breaks = list(fot = 0:5, age = c(0,45,Inf))}.
+#' Breaks for \code{per} can also be given in 
+#' \code{Date}/\code{IDate}/\code{date} format, whereupon
+#' they are converted to fractional years before used in splitting.
+#' 
+#' The \code{age} time scale can additionally 
+#' be automatically split into common age grouping schemes
+#' by naming the scheme with an appropriate character string:
+#' 
+#' \itemize{
+#'   \item \code{"18of5"}: age groups 0-4, 5-9, 10-14, ..., 75-79, 80-84, 85+
+#'   \item \code{"20of5"}: age groups 0-4, 5-9, 10-14, ..., 85-89, 90-94, 95+
+#'   \item \code{"101of1"}: age groups 0, 1, 2, ..., 98, 99, 100+
+#' }
+#' 
+#' \strong{Time variables}
+#' 
+#' If any of the given time variables
+#' (\code{birth}, \code{entry}, \code{exit}, \code{event})
+#' is in any kind of date format, they are first coerced to 
+#' fractional years before splitting
+#' using \code{\link{get.yrs}} (with \code{year.length = "actual"}).
+#' 
+#' Sometimes in e.g. SIR/SMR calculation one may want the event time to differ
+#' from the time of exit from follow-up, if the subject is still considered
+#' to be at risk of the event. If \code{event} is specified, the transition to
+#'  \code{status} is moved to \code{event} from \code{exit} 
+#'  using \code{\link[Epi]{cutLexis}}. See Examples.
+#'  
+#' \strong{The status variable}
+#' 
+#' The statuses in the expanded output (\code{lex.Cst} and \code{lex.Xst})
+#' are determined by using either only \code{status} or both \code{status}
+#' and \code{entry.status}. If \code{entry.status = NULL}, the status at entry
+#' is guessed according to the type of variable supplied via \code{status}:
+#' For numeric variables it will be zero, for factors the first level
+#' (\code{levels(status)[1]}) and otherwise the first unique value in alphabetical
+#' order (\code{sort(unique(status))[1]}). 
+#' 
+#' Using numeric or factor status
+#' variables is strongly recommended. Logical expressions are also allowed
+#' (e.g. \code{status = my_status != 0L}) and are converted to integer internally.
+#' 
+#' \strong{Merging population hazard information}
+#' 
+#' To enable computing relative/net survivals with \code{\link{survtab}}
+#' and \code{\link{relpois}}, \code{lexpand} merges an appropriate
+#' population hazard data (\code{pophaz}) to the expanded data 
+#' before dropping rows outside the specified
+#' time window (if \code{drop = TRUE}). \code{pophaz} must, for this reason, 
+#' contain at a minimum the variables named
+#' \code{agegroup}, \code{year}, and \code{haz}. \code{pophaz} may contain additional variables to specify
+#' different population hazard levels in different strata; e.g. \code{popmort} includes \code{sex}.
+#' All the strata-defining variables must be present in the supplied \code{data}. \code{lexpand} will
+#' automatically detect variables with common names in the two datasets and merge using them.
+#' 
+#' Currently \code{year} must be an integer variable specifying the appropriate year. \code{agegroup}
+#' must currently also specify one-year age groups, e.g. \code{popmort} specifies 101 age groups
+#' of length 1 year. In both
+#' \code{year} and \code{agegroup} variables the values are interpreted as the lower bounds of intervals
+#' (and passed on to a \code{cut} call). The mandatory variable \code{haz}
+#' must specify the appropriate average rate at the person-year level;
+#' e.g. \code{haz = -log(survProb)} where \code{survProb} is a one-year conditional
+#' survival probability will be the correct hazard specification. 
+#' 
+#' The corresponding \code{pophaz} population hazard value is merged by using the mid points
+#' of the records after splitting as reference values. E.g. if \code{age=89.9} at the start
+#' of a 1-year interval, then the reference age value is \code{90.4} for merging. 
+#' This way we get a "typical" population hazard level for each record.
+#' 
+#' \strong{Computing Pohar-Perme weights}
+#' 
+#' If \code{pp = TRUE}, Pohar-Perme weights 
+#' (the inverse of cumulative population survival) are computed. This will
+#' create the new \code{pp} variable in the expanded data. \code{pp} is a
+#' reserved name and \code{lexpand} throws exception if a variable with that name
+#' exists in \code{data}.
+#' 
+#' When a survival interval contains one or several rows per subject
+#' (e.g. due to splitting by the \code{per} scale),
+#' \code{pp} is cumulated from the beginning of the first record in a survival
+#' interval for each subject to the mid-point of the remaining time within that
+#' survival interval, and  that value is given for every other record 
+#' that a given person has within the same survival interval. 
+#' 
+#' E.g. with 5 rows of duration \code{1/5} within a survival interval 
+#' \code{[0,1)]}, \code{pp} is determined for all records by a cumulative 
+#' population survival from \code{0} to \code{0.5}. The existing accuracy is used,
+#' so that the weight is cumulated first up to the end of the second row
+#' and then over the remaining distance to the mid-point (first to 0.4, then to
+#' 0.5). This ensures that more accurately merged population hazards are fully
+#' used.
+#' 
+#' \strong{Event not at end of follow-up & overlapping time lines}
+#' 
+#' \code{event} may be used if the event indicated by \code{status} should
+#' occur at a time differing from \code{exit}. If \code{event} is defined,
+#' \code{cutLexis} is used on the data set after coercing it to the \code{Lexis}
+#' format and before splitting. Note that some values of \code{event} are allowed
+#' to be \code{NA} as with \code{cutLexis} to accommodate observations
+#' without an event occurring.
+#' 
+#' Additionally, setting \code{overlapping = FALSE} ensures that (irrespective
+#' of using \code{event}) the each subject defined by \code{id} only has one
+#' continuous time line instead of possibly overlapping time lines if
+#' there are multiple rows in \code{data} by \code{id}.
+#' 
+#' 
+#' \strong{Aggregating}
+#' 
+#' Certain analyses such as SIR/SMR calculations require tables of events and
+#' person-years by the unique combinations (interactions) of several variables. 
+#' For this, \code{aggre} can be specified as a list of such variables 
+#' (preferably \code{factor} variables but not mandatory)
+#'  and any arbitrary functions of the 
+#' variables at one's disposal. E.g. 
+#' 
+#' \code{aggre = list(sex, agegr = cut(dg_age, 0:100))}
+#' 
+#' would tabulate events and person-years by sex and an ad-hoc age group
+#' variable. Every ad-hoc-created variable should be named.
+#' 
+#' \code{fot}, \code{per}, and \code{age} are special reserved variables which,
+#' when present in the \code{aggre} list, are output as categories of the
+#' corresponding time scale variables by using 
+#' e.g. 
+#' 
+#' \code{cut(fot, breaks$fot, right=FALSE)}. 
+#' 
+#' This only works if
+#' the corresponding breaks are defined in \code{breaks} or via "\code{...}".
+#' E.g. 
+#' 
+#' \code{aggre = list(sex, fot.int = fot)} with 
+#' 
+#' \code{breaks = list(fot=0:5)}.
+#' 
+#' The output variable \code{fot.int} in the above example will have
+#' the lower limits of the appropriate intervals as values.
+#' 
+#' \code{aggre} as a named list will output numbers of events and person-years
+#' with the given new names as categorizing variable names, e.g. 
+#' \code{aggre = list(follow_up = fot, gender = sex, agegroup = age)}.
+#' 
+#' The output table has person-years (\code{pyrs}) and event counts
+#' (e.g. \code{from0to1}) as columns. Event counts are the numbers of transitions
+#' (\code{lex.Cst != lex.Xst}) or the \code{lex.Xst} value at a subject's 
+#' last record (subject possibly defined by \code{id}).
+#' 
+#' If \code{aggre.type = "unique"} (alias \code{"non-empty"}), 
+#' the above results are computed for existing
+#' combinations of expressions given in \code{aggre}, but also for non-existing
+#' combinations if \code{aggre.type = "cartesian"} (alias \code{"full"}). E.g. if a
+#' factor variable has levels \code{"a", "b", "c"} but the data is limited
+#' to only have levels \code{"a", "b"} present 
+#' (more than zero rows have these level values), the former setting only
+#' computes results for \code{"a", "b"}, and the latter also for \code{"c"}
+#' and any combination with other variables or expression given in \code{aggre}.
+#' In essence, \code{"cartesian"} forces also combinations of variables used
+#' in \code{aggre} that have no match in data to be shown in the result.
+#' 
+#' If \code{aggre} is not \code{NULL} and \code{pophaz} has been supplied,
+#' \code{lexpand} also aggregates the expected counts of events, which
+#' appears in the output data by the reserved name \code{d.exp}. Additionally,
+#' having \code{pp = TRUE} causes \code{lexpand} to also compute various
+#' Pohar-Perme weighted figures necessary for computing Pohar-Perme net survivals
+#' with \code{\link{survtab_ag}}. This can be slow, so consider what is really
+#' needed. The Pohar-Perme weighted figures have the suffix \code{.pp}. 
+#' 
+#' @return
+#' If \code{aggre = NULL}, returns 
+#' a \code{data.table} or \code{data.frame} 
+#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+#' object expanded to accommodate split observations with time scales as
+#' fractional years and \code{pophaz} merged in if given. Population
+#' hazard levels in new variable \code{pop.haz}, and Pohar-Perme
+#' weights as new variable \code{pp} if requested.
+#' 
+#' If \code{aggre} is defined, returns a long-format 
+#' \code{data.table}/\code{data.frame} with the variable \code{pyrs} (person-years),
+#' and variables for the counts of transitions in state or state at end of 
+#' follow-up formatted \code{fromXtoY}, where \code{X} and \code{Y} are 
+#' the states transitioned from and to, respectively. The data may also have
+#' the columns \code{d.exp} for expected numbers of cases and various
+#' Pohar-Perme weighted figures as identified by the suffix \code{.pp}; see 
+#' Details.
+#' 
+#' 
+#' @examples
+#' \donttest{
+#' ## prepare data for e.g. 5-year cohort survival calculation
+#' x <- lexpand(sire, breaks=list(fot=seq(0, 5, by = 1/12)), 
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'              status =  status != 0, pophaz=popmort)
+#' 
+#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
+#' BL <- list(fot = seq(0, 5, by = 1/12), per = c("2008-01-01", "2013-01-01"))
+#' x <- lexpand(sire, breaks = BL, 
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'              pophaz=popmort, status =  status != 0)
+#' 
+#' ## aggregating
+#' BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+#' ag <- lexpand(sire, breaks = BL, status = status != 0, 
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'               aggre=list(sex, period = per, surv.int = fot))
+#' 
+#' ## aggregating even more
+#' ag <- lexpand(sire, breaks = BL, status = status != 0, 
+#'               birth = bi_date, entry = dg_date, exit = ex_date,
+#'               aggre=list(sex, period = per, surv.int = fot),
+#'               pophaz = popmort, pp = TRUE)
+#' 
+#' ## using "..."
+#' x <- lexpand(sire, fot=0:5, status =  status != 0,
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'              pophaz=popmort) 
+#' 
+#' x <- lexpand(sire, fot=0:5, status =  status != 0, 
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'              aggre=list(sex, surv.int = fot))
+#'              
+#' ## using the "event" argument: it just places the transition to given "status"
+#' ## at the "event" time instead of at the end, if possible using cutLexis
+#' x <- lexpand(sire, status = status, event = dg_date,
+#'              birth = bi_date, entry = dg_date, exit = ex_date,) 
+#' 
+#' ## aggregating with custom "event" time
+#' ## (the transition to status is moved to the "event" time)
+#' x <- lexpand(sire, status = status, event = dg_date, 
+#'              birth = bi_date, entry = dg_date, exit = ex_date,
+#'              per = 1970:2014, age = c(0:100,Inf),
+#'              aggre = list(sex, year = per, agegroup = age)) 
+#' 
+#' }
+#' 
+#' @import data.table
+#' @import Epi
+#' @family splitting functions
+#' @family aggregation functions
+#' @seealso
+#' \code{\link[Epi]{Lexis}}, \code{\link{popmort}}
+#' @export
+lexpand <- function(data, 
+                    birth=NULL, entry=NULL, exit=NULL, event=NULL,
+                    status = status != 0,
+                    entry.status = NULL,
+                    breaks = list(fot=c(0,Inf)),
+                    id = NULL,
+                    overlapping = TRUE,
+                    aggre = NULL,
+                    aggre.type = c("unique", "cartesian"),
+                    drop=TRUE,
+                    pophaz = NULL, pp = TRUE, 
+                    subset = NULL,
+                    merge=TRUE, verbose = FALSE,
+                    ...) {
+  start_time <- proc.time()
+  
+  TF <- environment()
+  PF <- parent.frame(1L)
+  
+  ## data checks
+  if ( missing(data) || nrow(data) == 0) stop("no data found")
+  
+  if (!is.data.frame(data)) stop("data must be a data.frame or data.table")
+  
+  ## to instate global variables to appease R CMD CHECK 
+  .EACHI <- lex.status <- lexpand.id <- lex.exit <- lex.birth <- 
+    lex.entry <- lex.event <- temp.id <- cd <- fot <- age <- per <- 
+    lex.id <- lex.multi <- pop.haz <- lex.Cst <- lex.Xst <- lex.dur <- NULL
+  
+  
+  ## test conflicting variable names -------------------------------------------
+  added_vars <- c("fot", "per", "age", "lex.id", "lex.dur", "lex.Xst", "lex.Cst")
+  if (!is.null(pophaz)) added_vars <- if (pp) c(added_vars, "pp", "pop.haz") else c(added_vars, "pop.haz")
+  conflicted_vars <- intersect(added_vars, names(data))
+  
+  if (merge && length(conflicted_vars) > 0) {
+    conflicted_vars <- paste0("'", conflicted_vars, "'", collapse = ", ")
+    warning("'data' already had variable(s) named ", conflicted_vars, " which lexpand will create, and you have merge = TRUE; this may result in unexpected problems. Rename the variable(s)?")
+  }
+  rm(added_vars, conflicted_vars)
+  
+  ## test aggre type -----------------------------------------------------------
+  aggre.type <- match.arg(aggre.type[1L], c("cartesian", "non-empty", "unique", "cross-product", "full"))
+  if (aggre.type == "cross-product") {
+    aggre.type <- "cartesian"
+    warning("aggre.type value 'cross-product' deprecated and renamed to 'cartesian'; please use that in the future")
+  }
+  
+  ## subsetting-----------------------------------------------------------------
+  ## no copy taken of data!
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset)
+  
+  ## prepping time variables ---------------------------------------------------
+  l <- substitute(list(birth, entry, exit, event, status, entry.status, id))
+  rm(birth, entry, exit, event, status, entry.status, id)
+  
+  lc <- unlist(lapply(l, deparse))
+  lc <- lc[-1] ## first always "list"
+  
+  wh <- which(lc != "NULL")
+  lex_vars <- c("lex.birth","lex.entry","lex.exit","lex.event", "lex.status", "lex.entry.status", "lexpand.id")[wh]
+  if (any(!c("lex.birth", "lex.entry", "lex.exit", "lex.status") %in% lex_vars)) stop("birth, entry, exit and status are mandatory")
+  
+  l <- eval(l, envir = data[subset, ], enclos = PF)
+  l[-wh] <- NULL
+  
+  
+  ## vars can be given as character strings of variable names
+  isChar  <- sapply(l, is.character, simplify = TRUE)
+  if (any(isChar)) {
+    isShort <- sapply(l, function(x) {length(x) == 1L}, simplify = TRUE)
+    whOneChar <- which(isShort & isChar)    
+    
+    whBadChar <- NULL
+    if (length(whOneChar) > 0) {
+      testBadChar <- unlist(l[whOneChar])
+      whBadChar <- whOneChar[!testBadChar %in% names(data)]
+    }
+    
+    if (length(whBadChar) > 0) {
+      
+      badChar <- l[whBadChar]
+      badChar <- paste0(badChar[1:min(length(badChar), 5L)], collapse = ", ")
+      stop("Variables given as a character of length one are interpreted as variable names in data, 
+           but some given characters were not found in data; 
+           check names or input as factor/Date; 
+           first five bad names: ", badChar)
+    }
+    
+    l[whOneChar] <- lapply(l[whOneChar], function(x) {data[subset, ][[x]]})
+  }
+  
+  
+  l <- as.data.table(l)
+  setnames(l, names(l), lex_vars)
+  
+  
+  rm(lex_vars)
+  if (!all(c("lex.birth","lex.entry","lex.exit","lex.status") %in% names(l))) {
+    stop("birth, entry, exit and status are mandatory, but at least one was misspecified/NULL")
+  }
+  
+  if (is.logical(l$lex.status)) l[, lex.status := as.integer(lex.status)]
+  if (is.null(l$lexpand.id)) l[, lexpand.id  := 1:.N]
+  
+  ## checks for merging style --------------------------------------------------
+  if (!is.null(pophaz)) {
+    all_names_present(pophaz, c("agegroup","year","haz"))
+    othMergeVars <- setdiff(names(pophaz), c("agegroup","year","haz"))
+    badOthMergeVars <- setdiff(othMergeVars, names(data))
+    if (length(badOthMergeVars) > 0) {
+      badOthMergeVars <- paste0("'", badOthMergeVars, "'", collapse = ", ")
+      stop("Following variables exist in pophaz but do not exist in data: ", badOthMergeVars, ". Make sure data and pophaz contain variables with the same names that you intend to merge by.")
+    }
+  }
+  
+  if (is.null(pophaz)) {
+    comp_pp <- FALSE
+  } else {
+    comp_pp <- TRUE
+  }
+  
+  ## internally we have "delta" and "actual" methods, 
+  ## the latter being experimental and not visible in the documentation.
+  ## "delta" is always used if pp = TRUE.
+  ## it is possible to choose pp = "actual" as well, though, if you know
+  ## about it.
+  comp_pp <- FALSE
+  if (is.logical(pp) && pp) pp <- "delta"
+  
+  if (!is.null(pp) && is.character(pp)) {
+    pp <- match.arg(pp, c("delta", "actual"))
+    comp_pp <- TRUE
+  } 
+  if (comp_pp && "pp" %in% names(data)) stop("variable named 'pp' in data; this is a reserved name for pohar-perme weights, please rename / remove the variable in data")
+  
+  ## ensure given breaks make any sense ----------------------------------------
+  
+  bl <- list(...)
+  lna <- names(bl)
+  bad_lna <- setdiff(lna, c("fot","per","age"))
+  if (length(bad_lna) > 0) {
+    bad_lna <- paste0("'", bad_lna, "'", collapse = ", ")
+    stop("only arguments named 'fot', 'per' or 'age' currently allowed to be passed via '...'; did you mistype an argument? bad args: ", bad_lna)
+  }
+  lna <- intersect(names(bl), c("fot","per","age"))
+  if (length(lna) > 0) {
+    bl <- bl[lna]
+    if (!is.null(breaks)) breaks[lna] <- NULL
+    breaks <- c(breaks, bl)
+  }
+  rm(bl, lna)
+  
+  brna <- names(breaks)
+  if (length(brna) != length(breaks)) {
+    stop("all elements in breaks list must be named, e.g. list(fot = 0:5, age=c(0,45,65,Inf))")
+  }
+  
+  brna <- intersect(brna, c("fot","per","age"))
+  if (length(brna) == 0) {
+    breaks$fot <- c(0,Inf)
+  }
+  
+  if ("age" %in% brna && is.character(breaks$age)) {
+    schemeNames <- c("18of5", "20of5", "101of1")
+    if (!breaks$age %in% schemeNames) stop("You supplied '", breaks$age, "' as breaks for the age scale, but allowed character strings are: ", paste0("'", schemeNames, "'", collapse = ","))
+    brSchemes <- list(c(seq(0, 85, 5)), c(seq(0, 95, 5), Inf), c(0:100, Inf))
+    names(brSchemes) <- paste0("age_", schemeNames)
+    breaks$age <- brSchemes[paste0("age_",breaks$age)]
+  } 
+  
+  
+  if (any(sapply(breaks, length) == 1L)) {
+    stop("any given non-null vector of breaks must have more than one break!")
+  }
+  
+  # convert to fractional years ------------------------------------------------
+  
+  char2date <- function(obj) {
+    if (is.character(obj) || inherits(obj, "date")) {
+      return(as.IDate(obj))
+    } else {
+      return(obj)
+    }
+  }
+  
+  date2yrs <- function(obj) {    
+    if (is.Date(obj) || inherits(obj, "date")) {
+      get.yrs(obj, year.length = "actual")
+    } else {
+      obj
+    }
+  }
+  
+  breaks <- lapply(breaks, char2date)
+  breaks <- lapply(breaks, date2yrs)
+  
+  time_vars <- intersect(names(l), c("lex.birth", "lex.entry", "lex.exit","lex.event"))
+  l[, (time_vars) := lapply(.SD, date2yrs) , .SDcols = time_vars]
+  
+  if (verbose) cat("given birth, entry, exit, status etc. variables after coercion to numeric \n")
+  if (verbose) print(l)
+  
+  # check data consistency for overlapping = FALSE -----------------------------
+  ## not allowed: for any one unique subject to be true for 
+  ## multiple rows (if overlapping = TRUE):
+  ## * same event values
+  ## * same entry values
+  if (!overlapping) {
+    if ("lex.event" %in% names(l)) {
+      if (all(is.na(l$lex.event))) stop("ALL 'event' values are NA; if this is as intended, please use event = NULL instead")
+      
+      if (any(duplicated(l, by = c("lexpand.id", "lex.event")))) {
+        stop("subject(s) defined by lex.id had several rows where 'event' time had the same value, which is not supported with overlapping = FALSE; perhaps separate them by one day?")
+      } 
+      if (any(l[!is.na(lex.event), lex.entry == lex.event])) {
+        stop("some rows have simultaneous 'entry' and 'event', which is not supported with overlapping = FALSE; perhaps separate them by one day?")
+      }
+    } else if (any(duplicated(l, by = c("lexpand.id", "lex.exit")))) {
+      stop("subject(s) defined by lex.id had several rows where 'exit' time had the same value, which is not supported without 'event' defined; use 'event' or perhaps separate them by one day?")
+    }
+    
+    
+  }
+  
+  
+  # dropping unuseful records --------------------------------------------------
+  test_times <- function(condition, msg, old_subset=l_subset, DT=l) {
+    
+    condition <- substitute(condition)
+    condition <- eval(condition, envir = DT, enclos = parent.frame(1L))
+    
+    new_subset <- old_subset & !(condition & !is.na(condition))
+    old_n <- sum(old_subset)
+    new_n <- sum(new_subset)
+    
+    if (new_n == 0L) {
+      stop("dropping rows where ", msg, " resulted in zero rows left. likely problem: misdefined time variables")
+    }
+    
+    if (new_n < old_n) {
+      message(paste0("dropped ", old_n-new_n, " rows where ", msg))
+    }
+    return(new_subset)
+  }
+  
+  l_subset <- rep(TRUE, nrow(l))
+  
+  l_subset <- test_times(is.na(lex.birth), "birth values are missing")
+  l_subset <- test_times(is.na(lex.entry), "entry values are missing")
+  l_subset <- test_times(is.na(lex.exit), "exit values are missing")
+  
+  if (!is.null(breaks$per)) {
+    l_subset <- test_times(lex.exit < min(breaks$per), "subjects left follow-up before earliest per breaks value")
+  }
+  if (!is.null(breaks$age)) {
+    l_subset <- test_times(lex.exit - lex.birth < min(breaks$age), "subjects left follow-up before lowest age breaks value")
+  }
+  if (!is.null(breaks$fot)) {
+    l_subset <- test_times(lex.exit - lex.entry < min(breaks$fot), "subjects left follow-up before lowest fot breaks value")
+  }
+  l_subset <- test_times(lex.birth >= lex.exit, "birth >= exit")
+  l_subset <- test_times(lex.entry == lex.exit, "entry == exit")
+  l_subset <- test_times(lex.entry >  lex.exit, "entry >  exit")
+  l_subset <- test_times(lex.birth >  lex.entry, "birth > entry")
+  if (!is.null(l$lex.event)) {
+    l_subset <- test_times(lex.event > lex.exit, "event > exit")
+    l_subset <- test_times(lex.event < lex.entry, "event < entry")
+  }
+  l <- l[l_subset]
+  
+  if (verbose) cat("Time taken by checks, prepping and test: ", timetaken(start_time), "\n")
+  
+  # Lexis coercion -------------------------------------------------------------
+  
+  ## status definitions
+  setnames(l, "lex.status", "lex.Xst")
+  if ("lex.entry.status" %in% names(l)) {
+    setnames(l, "lex.entry.status", "lex.Cst")
+  } else {
+    if (is.factor(l$lex.Xst)) {
+      l[, lex.Cst := factor(levels(lex.Xst)[1L], levels=levels(lex.Xst))]
+    } else if (is.double(l$lex.Xst)) {
+      l[, lex.Cst := 0]
+    } else if (is.integer(l$lex.Xst)) {
+      l[, lex.Cst := 0L]
+    } else {
+      l[, lex.Cst := sort(unique(lex.Xst))[1L]]
+    }
+   
+  }
+  
+  # ensure common labels for factors etc.
+  harmonizeStatuses(x = l, C = "lex.Cst", X = "lex.Xst")
+  
+  ## time scales and duration
+  l[, lex.dur := lex.exit - lex.entry]
+  l[, fot := 0]
+  setnames(l, "lex.entry", "per")
+  l[, age := per-lex.birth]
+  setnames(l, "lexpand.id", "lex.id")
+  
+  ## for merging data with l later
+  if (merge) {
+    idt <- data.table(temp.id = 1:nrow(l))
+    l[, temp.id := 1:.N]
+  }
+  
+  ## crop time scale values to obey breaks limits and drop if necessary
+  ## NOTE: goes wrong if need to compute pp weights!
+  #   if (drop && !pp) {
+  #     intelliCrop(x = l, breaks = breaks, allScales = c("fot", "per", "age"), cropStatuses = TRUE)
+  #     l <- intelliDrop(x = l, breaks = breaks, dropNegDur = TRUE)
+  #   }
+  
+  
+  setcolsnull(l, colorder=TRUE, soft=TRUE, 
+              keep = c("lex.id","fot","per","age",
+                       "lex.dur", "lex.Cst", "lex.Xst", "lex.event", "temp.id"))
+  setattr(l, "class", c("Lexis", "data.table", "data.frame"))
+  setattr(l, "time.scales", c("fot","per","age"))
+  setattr(l, "time.since", c("","",""))
+  
+  if (verbose) cat("data just after Lexis coercion: \n")
+  if (verbose) print(l)  
+  
+  # event not at exit time -----------------------------------------------------
+  
+  if ("lex.event" %in% names(l)) {
+    
+    if (!overlapping) {
+      
+      ## using lex.event time, ensure coherence of lex.Cst & lex.Xst
+      ## before cutLexis()
+      tmpFE <- makeTempVarName(l, pre = "fot_end_")
+      l[, (tmpFE) := fot + lex.dur]
+      setkeyv(l, c("lex.id", "lex.event", tmpFE))
+      tmpLX <- makeTempVarName(l, pre = "lag_lex.Xst_")
+      l[, (tmpLX) := shift(lex.Xst, n = 1, type = "lag"), by = lex.id]
+      l[!is.na(get(tmpLX)), lex.Cst := get(tmpLX)]
+      l[, c(tmpFE, tmpLX) := NULL]
+      rm(tmpFE, tmpLX)
+      
+    }
+    
+    if (verbose) cutt <- proc.time()
+    setDF(l)
+    setattr(l, "class", c("Lexis", "data.frame"))
+    l <- Epi::cutLexis(l, cut = l$lex.event, timescale = "per", new.state = l$lex.Xst, precursor.states = unique(l$lex.Cst))
+    setDT(l)
+    setattr(l, "class", c("Lexis", "data.table", "data.frame"))
+    if (verbose) cat("Time taken by cutLexis when defining event time points: ", timetaken(cutt), "\n")
+    
+    if (verbose) cat("Data just after using cutLexis: \n")
+    if (verbose) print(l[])
+    
+  }
+  
+  
+  # overlapping timelines? -----------------------------------------------------
+  
+  if (!overlapping && any(duplicated(l$lex.id))) {
+    tmpFE <- makeTempVarName(l, pre = "fot_end_")
+    l[, (tmpFE) := fot + lex.dur]
+    ## don't keep duplicated rows:
+    ## same end points imply fully overlapping time lines
+    ## e.g. 
+    ## --->
+    ## ->
+    ##  -->
+    ## results in 
+    ## ->
+    ## --->
+    ## we only keep the longest time line with a unique end point.
+
+    # setkeyv(l,  c("lex.id", tmpFE, "fot"))
+    tmpLE <- intersect(names(l), "lex.event")
+    LEval <- if (length(tmpLE) == 0) NULL else -1
+
+    setorderv(l, c("lex.id", tmpFE, tmpLE, "fot"), c(1,1,LEval,1))
+    l <- unique(l, by = c("lex.id", tmpFE))
+    
+    ## end points are kept but starting points are "rolled"
+    ## from first to last row by lex.id to ensure non-overlappingness; e.g.
+    ## ->
+    ## --->
+    ## results in 
+    ## ->
+    ##   ->
+    # setkeyv(l, c("lex.id", tmpFE))
+    # setorderv(l, c("lex.id", tmpLE, tmpFE), c(1, LEval, 1))
+    setkeyv(l, c("lex.id", tmpLE, tmpFE))
+    
+    if (verbose) cat("data just before fixing overlapping time lines \n")
+    if (verbose) print(l)
+    l[, lex.dur := get(tmpFE) - c(min(fot), get(tmpFE)[-.N]), by = lex.id]
+    l[, fot := get(tmpFE) - lex.dur]
+    cumDur <- l[,  list(age = min(age), per = min(per), cd = c(0, cumsum(lex.dur)[-.N])), by = lex.id]
+    cumDur[, age := age+cd]
+    cumDur[, per := per+cd]
+    l[, age := cumDur$age]
+    l[, per := cumDur$per]
+    l[, (tmpFE) := NULL]; rm(cumDur)
+    
+    
+    ## if event used, first row up to event, second row from first event to etc...
+  }
+  
+  setcolsnull(l, "lex.event", soft = TRUE) ## note: lex.event needed in overlapping procedures
+  
+  if (verbose) cat("time and status variables before splitting: \n")
+  if (verbose) print(l)
+  if ("id" %in% ls()) rm("id")
+  
+  
+  # splitting ------------------------------------------------------------------
+  
+  ## determine whether to drop data only after splitting and merging
+  drop_after <- FALSE
+  if (drop == TRUE && comp_pp) {
+    drop <- FALSE
+    drop_after <- TRUE
+  }
+  
+  forceLexisDT(l, breaks = list(fot = NULL, per = NULL, age = NULL),
+               allScales = c("fot", "per", "age"))
+  if (verbose) splittime <- proc.time()
+  l <- splitMulti(l,  breaks = breaks, 
+                  drop = drop, verbose=FALSE, merge = TRUE)
+  setDT(l)
+  setkey(l, lex.id, fot)
+  l[, lex.multi := 1:.N, by = lex.id]
+  if (verbose) cat("Time taken by splitting:", timetaken(splittime), "\n")
+  
+  # merging other variables from data ------------------------------------------
+  
+  if (merge) {
+    setkey(l, temp.id)
+    
+    temp <- data.table(idt, data[subset & !is.na(subset), ][l_subset, ])
+    setkey(temp, temp.id)
+    
+    l <- temp[l]
+    
+    rm(temp, idt)
+    setcolsnull(l, "temp.id")
+    
+    lex_vars <- c("lex.id","lex.multi","fot","per","age", "lex.dur", "lex.Cst", "lex.Xst")
+    setcolorder(l, c(lex_vars, setdiff(names(l), lex_vars)))
+  }
+  rm(data, subset, l_subset)
+  
+  ## aggregating checks --------------------------------------------------------
+  ## NOTE: aggre evaled here using small data subset to check that all needed
+  ## variables are found, etc.
+  aggSub <- substitute(aggre)
+  agTest <- evalPopArg(arg = aggSub, data = l[1:min(10L, .N), ], 
+                       enclos = PF, recursive = TRUE, DT = TRUE)
+  agTy <- attr(agTest, "arg.type")
+  if (is.null(agTy)) agTy <- "NULL"
+  aggSub <- attr(agTest, "quoted.arg")
+  agVars <- attr(agTest, "all.vars")
+  rm(aggre)
+  
+  # merging pophaz and pp-weighting --------------------------------------------
+  if (!is.null(pophaz)) {
+    
+    pophaztime <- proc.time()
+    
+    if (any(c("haz", "pop.haz") %in% names(l))) stop("case data had variable(s) named 'haz' / 'pop.haz', which are reserved for lexpand's internal use. rename/remove them please.")
+    # merge surv.int information -----------------------------------------------
+    NULL_FOT <- FALSE
+    if (is.null(breaks$fot)) {
+      breaks$fot <- l[, c(0, max(fot+lex.dur))] 
+      NULL_FOT <- TRUE
+    }
+    
+    breaks$fot <- sort(unique(breaks$fot))
+    # handle pophaz data -------------------------------------------------------
+    
+    if (!"haz" %in% names(pophaz)) stop("no 'haz' variable in pophaz; please rename you hazard variable to 'haz'")
+    yBy <- xBy <- setdiff(names(pophaz), c("haz"))
+    if (c("year") %in% yBy) xBy[yBy == "year"] <- "per"
+    if (c("agegroup") %in% yBy) xBy[yBy == "agegroup"] <- "age"
+    yByOth <- setdiff(yBy, c("year", "agegroup"))
+    
+    if (any(!yByOth %in% names(l))) 
+      stop("Following variable names not common between pophaz and data: ", paste0("'", yByOth[!yByOth %in% names(l)], "'", collapse = ", "))
+    
+    l <- cutLowMerge(x = l, y = pophaz, by.x = xBy, by.y = yBy, all.x = TRUE,
+                     all.y = FALSE, mid.scales = c("per", "age"), old.nums = TRUE)
+    setnames(l, "haz", "pop.haz")
+    
+    ## check if l's merging time variables were within pophaz's limits ---------
+    nNA <- l[is.na(pop.haz), .N]
+    if (nNA > 0) message("WARNING: after merging pophaz, ", nNA, " rows in split data have NA hazard values!")
+    
+    names(yBy) <- xBy
+    names(xBy) <- yBy
+    for (k in intersect(c("per", "age"), xBy)) {
+      yVar <- yBy[k]
+      kLo <- min(pophaz[[yVar]])
+      kHi <- max(pophaz[[yVar]])
+      mid <- l[, get(k) + lex.dur]
+      nLo <- sum(mid < kLo - .Machine$double.eps^0.5)
+      nHi <- sum(mid > kHi - .Machine$double.eps^0.5)
+      if (nLo > 0) message("WARNING: ", nLo, " rows in split data have NA values due to their mid-points residing below the minimum value of '", yVar, "' in pophaz!")
+      if (nHi > 0) message("NOTE: ", nHi, " rows in split data had values of '", k, "' higher than max of pophaz's '", yVar, "'; the hazard values at '", yVar, "' == ", kHi, " were used for these")
+    }
+    rm(mid)
+    for (k in yByOth) {
+      levsNotOth <- setdiff(unique(l[[k]]), unique(pophaz[[k]]))
+      if (length(levsNotOth) > 0) message("WARNING: following levels (first five) of variable '", k, "' not in pophaz but exist in split data: ", paste0("'",levsNotOth[1:5],"'", collapse = ", "))
+    }
+    
+    
+    # pohar-perme weighting ----------------------------------------------------
+    if (comp_pp) {
+      setkeyv(l, c("lex.id", "fot"))
+      comp_pp_weights(l, surv.scale = "fot", breaks = breaks$fot, haz = "pop.haz", 
+                      style = "delta", verbose = verbose)
+    }
+    merge_msg <- "Time taken by merging pophaz"
+    if (comp_pp) merge_msg <- paste0(merge_msg, " and computing pp")
+    merge_msg <- paste0(merge_msg, ": ")
+    if (verbose) cat(paste0(merge_msg, timetaken(pophaztime), "\n"))
+    
+    
+  }
+  
+  # dropping after merging -----------------------------------------------------
+  if (drop_after) {
+    l <- intelliDrop(x = l, breaks = breaks)
+  }
+  
+  if (verbose) cat("Number of rows after splitting: ", nrow(l),"\n")
+  
+  
+  # aggregating if appropriate -------------------------------------------------
+  if (agTy != "NULL") {
+    
+    setcolsnull(l, keep = c("lex.id","lex.dur", "fot", "per", "age", "lex.Cst", "lex.Xst", agVars, "pop.haz", "pp"))
+    
+    sumVars <- NULL
+    if ("pop.haz" %in% names(l)) {
+      if ("d.exp" %in% names(l)) stop("data had variable named 'd.exp' by which to aggregate, which would be overwritten due to aggregating expected numbers of cases (you have supplied pophaz AND are aggregating); please rename / remove it first.")
+      l[, c("d.exp") := pop.haz*lex.dur ]
+      sumVars <- c(sumVars, "d.exp")
+    }
+    if ("pop.haz" %in% names(l) && comp_pp && "pp" %in% names(l)) {
+      forceLexisDT(l, breaks = breaks, allScales = c("fot", "per", "age"))
+      ppFigs <- comp_pp_weighted_figures(lex = l, haz = "pop.haz", pp = "pp", event.ind = NULL)
+      bad_pp_vars <- intersect(names(ppFigs), names(l))
+      if (length(bad_pp_vars) > 0L) {
+        bad_pp_vars <- paste0("'",bad_pp_vars, "'", collapse = ", ")
+        stop("Data had variable(s) named ", bad_pp_vars, ", by which to aggregate, which would be overwritten due to aggregating expected numbers of cases (you have supplied pophaz AND are aggregating); please rename / remove them first")
+      }
+      l[, names(ppFigs) := ppFigs]
+      sumVars <- c(sumVars, names(ppFigs))
+      rm(ppFigs)
+      
+    }
+    
+    if (verbose) cat("Starting aggregation of split data... \n")
+    setDT(l)
+    forceLexisDT(l, allScales = c("fot", "per", "age"), breaks = breaks)
+    l <- try(aggre(lex = l, by = aggSub, type = aggre.type, verbose = verbose, sum.values = sumVars))
+    if (inherits(l, "try-error")) stop("Something went wrong when calling aggre() within lexpand(). Usual suspect: bad 'by' argument. Error message from aggre(): 
+                                       ", paste0(l[[1]]))
+    if (verbose) cat("Aggregation done. \n")
+    
+    if (!return_DT() && is.data.table(l)) setDFpe(l)
+    
+  } else {
+    
+    
+    # last touch-up --------------------------------------------------------------
+    ## sometimes problems with releasing memory
+    gc()
+    
+    breaks <- lapply(c("fot","per","age"), function(ts_nm) {
+      breaks[[ts_nm]]
+    })
+    names(breaks) <- c("fot","per","age")
+    
+    ## handle attributes
+    setkeyv(l, c("lex.id", "lex.multi"))
+    set(l, j = "lex.multi", value = NULL)
+    setattr(l, "time.scales", c("fot","per","age"))
+    setattr(l, "time.since", c("","",""))
+    setattr(l, "breaks", breaks)
+    setattr(l, "class", c("Lexis","data.table","data.frame"))
+    if (!return_DT() && is.data.table(l)) setDFpe(l)
+    
+    
+    
+  }
+  
+  if (verbose) cat("Time taken by lexpand(): ", timetaken(start_time), "\n")
+  
+  return(l[])
+}
+
+
+globalVariables(c('.EACHI', "dg_date", "ex_date", "bi_date"))
+
diff --git a/R/long_df_and_array.R b/R/long_df_and_array.R
index 9dc3d74..470742f 100644
--- a/R/long_df_and_array.R
+++ b/R/long_df_and_array.R
@@ -1,297 +1,297 @@
-
-
-
-
-
-#' @title `array`s, `data.frame`s and `ratetable`s
-#' @description
-#' Utilities to transform objects between `array`, `data.frame`, and
-#' [survival::ratetable].
-#' @param x `[data.frame, data.table, array, ratetable]` (mandatory, no default)
-#' 
-#' - `long_df_to_array`: a `data.frame`
-#' - `long_df_to_ratetable`: a `data.frame`
-#' - `long_dt_to_array`: a `data.table`
-#' - `long_dt_to_ratetable`: a `data.table`
-#' - `array_to_long_df`: an `array`
-#' - `array_to_long_dt`: an `array`
-#' - `array_to_ratetable`: an `array`
-#' - `ratetable_to_array`: a [survival::ratetable]
-#' - `ratetable_to_long_df`: a [survival::ratetable]
-#' - `ratetable_to_long_dt`: a [survival::ratetable]
-#' @name array_df_ratetable_utils
-#' @examples
-#' 
-#' long_dt <- popEpi::popmort
-#' arr <- long_df_to_array(long_dt, c("agegroup", "year", "sex"), "haz") 
-#' rt <- array_to_ratetable(arr, dim.types = c(2L, 4L, 1L))
-#' 
-#' arr2 <- ratetable_to_array(rt)
-#' long_df2 <- array_to_long_df(arr2)
-#' 
-#' identical(sort(long_dt[["haz"]]), sort(long_df2[["value"]]))
-#' @return
-#' 
-#' - `long_df_to_array`: an `array`
-#' - `long_df_to_ratetable`: a [survival::ratetable]
-#' - `long_dt_to_array`: an `array`
-#' - `long_dt_to_ratetable`: a [survival::ratetable]
-#' - `array_to_long_df`: an `data.frame`
-#' - `array_to_long_dt`: an `data.table`
-#' - `array_to_ratetable`: a [survival::ratetable]
-#' - `ratetable_to_array`: an `array`
-#' - `ratetable_to_long_df`: a `data.frame`
-#' - `ratetable_to_long_dt`: a `data.table`
-
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @param stratum.col.nms `[character]` (mandatory, no default)
-#' 
-#' a vector of column names in `x` by which values are stratified
-#' 
-#' @param value.col.nm `[character]` (mandatory, no default)
-#' 
-#' name of column in `x` containing values (these will be contents of the
-#' array)
-#' @details
-#' - `long_df_to_array`: converts a long-format `data.frame` to an `array`
-#'   with one or more dimensions
-long_df_to_array <- function(x, stratum.col.nms, value.col.nm) {
-  stopifnot(
-    is.data.frame(x),
-    
-    is.character(stratum.col.nms),
-    length(stratum.col.nms) >= 1,
-    stratum.col.nms %in% names(x),
-    !duplicated(stratum.col.nms),
-    
-    length(value.col.nm) == 1,
-    value.col.nm %in% names(x)
-  )
-  
-  dn <- lapply(stratum.col.nms, function(col_nm) {
-    sort(unique(x[[col_nm]]))
-  })
-  names(dn) <- stratum.col.nms
-  
-  d <- vapply(dn, length, integer(1L))
-  n_dims <- length(d)
-  
-  arr <- array(x[[value.col.nm]][0L], d)
-  
-  wh <- do.call(cbind, lapply(stratum.col.nms, function(col_nm) {
-    match(x[[col_nm]], dn[[col_nm]])
-  }))
-  arr[wh] <- x[[value.col.nm]]
-  
-  dimnames(arr) <- dn
-  arr
-}
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `long_df_to_ratetable`: calls `long_df_to_array` and then 
-#'   `array_to_ratetable`
-long_df_to_ratetable <- function(
-  x, 
-  stratum.col.nms, 
-  value.col.nm, 
-  dim.types, 
-  cut.points = NULL
-) {
-  arr <- long_df_to_array(x = x, stratum.col.nms = stratum.col.nms, 
-                          value.col.nm = value.col.nm)
-  array_to_ratetable(x = arr, dim.types = dim.types, cut.points = cut.points)
-}
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `long_dt_to_array`: simply asserts that `x` is a `data.table` and 
-#' calls `long_df_to_array`
-#' @importFrom data.table is.data.table
-long_dt_to_array <- function(x, stratum.col.nms, value.col.nm) {
-  stopifnot(data.table::is.data.table(x))
-  long_df_to_array(x, stratum.col.nms, value.col.nm)
-}
-
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `long_dt_to_ratetable`: calls `long_dt_to_array` and then 
-#'   `array_to_ratetable`
-long_dt_to_ratetable <- function(
-  x, 
-  stratum.col.nms, 
-  value.col.nm, 
-  dim.types, 
-  cut.points = NULL
-) {
-  arr <- long_dt_to_array(x = x, stratum.col.nms = stratum.col.nms, 
-                          value.col.nm = value.col.nm)
-  array_to_ratetable(x = arr, dim.types = dim.types, cut.points = cut.points)
-}
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `array_to_long_df`: converts an array with one or more dimensions into
-#'   a long-format `data.frame`; any [dimnames] are used to name and fill the 
-#'   stratifying columns; for dimensions without a name, `".dX"` is used
-#'   for stratifying column number `X`; for each `k`, if there are no contents
-#'   in `dimnames(x)[[k]]`, the elements of `seq(dim(x)[k])` are used to fill 
-#'   the corresponding stratifying column; the value column always has the name
-#'   `"value"`
-array_to_long_df <- function(x) {
-  stopifnot(
-    is.array(x)
-  )
-  
-  d <- dim(x)
-  
-  dn <- dimnames(x)
-  if (is.null(dn)) {
-    dn <- vector("list", length(d))
-  }
-  dn[] <- lapply(seq_along(dn), function(k) {
-    nm_vec <- dn[[k]]
-    if (is.null(nm_vec)) {
-      nm_vec <- seq(d[k])
-    }
-    nm_vec
-  })
-  if (length(setdiff(names(dn), "")) == 0L) {
-    names(dn) <- rep("", length(dn))
-  }
-  names(dn) <- vapply(seq_along(dn), function(k) {
-    nm <- names(dn)[k]
-    if (nm == "") {
-      nm <- paste0(".d", k)
-    }
-    nm
-  }, character(1))
-  
-  df <- as.data.frame(which(array(TRUE, d), arr.ind = TRUE))
-  names(df) <- names(dn)
-  df[, ] <- lapply(seq_along(d), function(k) {
-    dn[[k]][df[[k]]]
-  })
-  
-  df[["value"]] <- as.vector(x)
-  df
-}
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `array_to_long_dt`: calls `array_to_long_df` and converts result to a 
-#'   `data.table` for convenience
-#' @importFrom data.table setDT
-array_to_long_dt <- function(x) {
-  df <- array_to_long_df(x)
-  data.table::setDT(df)
-  df[]
-}
-
-
-
-
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `array_to_ratetable`: converts an array to a [survival::ratetable]
-#' @param dim.types `[integer]` (mandatory, no default)
-#' 
-#' see `type` under **Details** in [survival::ratetable] 
-#' @param cut.points `[NULL, list]` (optional, default `NULL`)
-#' 
-#' see `cutpoints` under **Details** in [survival::ratetable] 
-#' 
-#' - `NULL`: automatically set using `dimnames(x)` and `dim.types`
-#' - `list`: one element for each dimensions of `x`
-#' @importFrom data.table copy setattr
-array_to_ratetable <- function(x, dim.types, cut.points = NULL) {
-  stopifnot(
-    is.array(x),
-    length(dim.types) == length(dim(x)),
-    dim.types %in% 1:4,
-    
-    is.null(cut.points) || inherits(cut.points, "list")
-  )
-  if (is.null(cut.points)) {
-    cut.points <- lapply(seq_along(dim.types), function(k) {
-      if (dim.types[k] == 1L) {
-        return(NULL)
-      }
-      cp <- as.numeric(dimnames(x)[[k]])
-      if (all(cp %% 1L == 0L)) {
-        cp <- as.integer(cp)
-      }
-      cp
-    })
-  }
-  
-  x <- data.table::copy(x)
-  data.table::setattr(x, "type", as.integer(dim.types))
-  data.table::setattr(x, "cutpoints", cut.points)
-  data.table::setattr(x, "class", "ratetable")
-  x
-}
-
-
-
-
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `ratetable_to_array`: converts a [survival::ratetable] to an array
-#' @importFrom data.table copy setattr
-ratetable_to_array <- function(x) {
-  stopifnot(
-    inherits(x, "ratetable")
-  )
-  array(x, dim = dim(x), dimnames = dimnames(x))
-}
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `ratetable_to_long_df`: calls `ratetable_to_array` and then
-#'   `array_to_long_df`
-#' @importFrom data.table copy setattr
-ratetable_to_long_df <- function(x) {
-  array_to_long_df(ratetable_to_array(x))
-}
-
-#' @rdname array_df_ratetable_utils
-#' @export
-#' @details
-#' - `ratetable_to_long_dt`: calls `ratetable_to_array` and then
-#'   `array_to_long_dt`
-#' @importFrom data.table copy setattr
-ratetable_to_long_dt <- function(x) {
-  array_to_long_dt(ratetable_to_array(x))
-}
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+#' @title `array`s, `data.frame`s and `ratetable`s
+#' @description
+#' Utilities to transform objects between `array`, `data.frame`, and
+#' [survival::ratetable].
+#' @param x `[data.frame, data.table, array, ratetable]` (mandatory, no default)
+#' 
+#' - `long_df_to_array`: a `data.frame`
+#' - `long_df_to_ratetable`: a `data.frame`
+#' - `long_dt_to_array`: a `data.table`
+#' - `long_dt_to_ratetable`: a `data.table`
+#' - `array_to_long_df`: an `array`
+#' - `array_to_long_dt`: an `array`
+#' - `array_to_ratetable`: an `array`
+#' - `ratetable_to_array`: a [survival::ratetable]
+#' - `ratetable_to_long_df`: a [survival::ratetable]
+#' - `ratetable_to_long_dt`: a [survival::ratetable]
+#' @name array_df_ratetable_utils
+#' @examples
+#' 
+#' long_dt <- popEpi::popmort
+#' arr <- long_df_to_array(long_dt, c("agegroup", "year", "sex"), "haz") 
+#' rt <- array_to_ratetable(arr, dim.types = c(2L, 4L, 1L))
+#' 
+#' arr2 <- ratetable_to_array(rt)
+#' long_df2 <- array_to_long_df(arr2)
+#' 
+#' identical(sort(long_dt[["haz"]]), sort(long_df2[["value"]]))
+#' @return
+#' 
+#' - `long_df_to_array`: an `array`
+#' - `long_df_to_ratetable`: a [survival::ratetable]
+#' - `long_dt_to_array`: an `array`
+#' - `long_dt_to_ratetable`: a [survival::ratetable]
+#' - `array_to_long_df`: an `data.frame`
+#' - `array_to_long_dt`: an `data.table`
+#' - `array_to_ratetable`: a [survival::ratetable]
+#' - `ratetable_to_array`: an `array`
+#' - `ratetable_to_long_df`: a `data.frame`
+#' - `ratetable_to_long_dt`: a `data.table`
+
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @param stratum.col.nms `[character]` (mandatory, no default)
+#' 
+#' a vector of column names in `x` by which values are stratified
+#' 
+#' @param value.col.nm `[character]` (mandatory, no default)
+#' 
+#' name of column in `x` containing values (these will be contents of the
+#' array)
+#' @details
+#' - `long_df_to_array`: converts a long-format `data.frame` to an `array`
+#'   with one or more dimensions
+long_df_to_array <- function(x, stratum.col.nms, value.col.nm) {
+  stopifnot(
+    is.data.frame(x),
+    
+    is.character(stratum.col.nms),
+    length(stratum.col.nms) >= 1,
+    stratum.col.nms %in% names(x),
+    !duplicated(stratum.col.nms),
+    
+    length(value.col.nm) == 1,
+    value.col.nm %in% names(x)
+  )
+  
+  dn <- lapply(stratum.col.nms, function(col_nm) {
+    sort(unique(x[[col_nm]]))
+  })
+  names(dn) <- stratum.col.nms
+  
+  d <- vapply(dn, length, integer(1L))
+  n_dims <- length(d)
+  
+  arr <- array(x[[value.col.nm]][0L], d)
+  
+  wh <- do.call(cbind, lapply(stratum.col.nms, function(col_nm) {
+    match(x[[col_nm]], dn[[col_nm]])
+  }))
+  arr[wh] <- x[[value.col.nm]]
+  
+  dimnames(arr) <- dn
+  arr
+}
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `long_df_to_ratetable`: calls `long_df_to_array` and then 
+#'   `array_to_ratetable`
+long_df_to_ratetable <- function(
+  x, 
+  stratum.col.nms, 
+  value.col.nm, 
+  dim.types, 
+  cut.points = NULL
+) {
+  arr <- long_df_to_array(x = x, stratum.col.nms = stratum.col.nms, 
+                          value.col.nm = value.col.nm)
+  array_to_ratetable(x = arr, dim.types = dim.types, cut.points = cut.points)
+}
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `long_dt_to_array`: simply asserts that `x` is a `data.table` and 
+#' calls `long_df_to_array`
+#' @importFrom data.table is.data.table
+long_dt_to_array <- function(x, stratum.col.nms, value.col.nm) {
+  stopifnot(data.table::is.data.table(x))
+  long_df_to_array(x, stratum.col.nms, value.col.nm)
+}
+
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `long_dt_to_ratetable`: calls `long_dt_to_array` and then 
+#'   `array_to_ratetable`
+long_dt_to_ratetable <- function(
+  x, 
+  stratum.col.nms, 
+  value.col.nm, 
+  dim.types, 
+  cut.points = NULL
+) {
+  arr <- long_dt_to_array(x = x, stratum.col.nms = stratum.col.nms, 
+                          value.col.nm = value.col.nm)
+  array_to_ratetable(x = arr, dim.types = dim.types, cut.points = cut.points)
+}
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `array_to_long_df`: converts an array with one or more dimensions into
+#'   a long-format `data.frame`; any [dimnames] are used to name and fill the 
+#'   stratifying columns; for dimensions without a name, `".dX"` is used
+#'   for stratifying column number `X`; for each `k`, if there are no contents
+#'   in `dimnames(x)[[k]]`, the elements of `seq(dim(x)[k])` are used to fill 
+#'   the corresponding stratifying column; the value column always has the name
+#'   `"value"`
+array_to_long_df <- function(x) {
+  stopifnot(
+    is.array(x)
+  )
+  
+  d <- dim(x)
+  
+  dn <- dimnames(x)
+  if (is.null(dn)) {
+    dn <- vector("list", length(d))
+  }
+  dn[] <- lapply(seq_along(dn), function(k) {
+    nm_vec <- dn[[k]]
+    if (is.null(nm_vec)) {
+      nm_vec <- seq(d[k])
+    }
+    nm_vec
+  })
+  if (length(setdiff(names(dn), "")) == 0L) {
+    names(dn) <- rep("", length(dn))
+  }
+  names(dn) <- vapply(seq_along(dn), function(k) {
+    nm <- names(dn)[k]
+    if (nm == "") {
+      nm <- paste0(".d", k)
+    }
+    nm
+  }, character(1))
+  
+  df <- as.data.frame(which(array(TRUE, d), arr.ind = TRUE))
+  names(df) <- names(dn)
+  df[, ] <- lapply(seq_along(d), function(k) {
+    dn[[k]][df[[k]]]
+  })
+  
+  df[["value"]] <- as.vector(x)
+  df
+}
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `array_to_long_dt`: calls `array_to_long_df` and converts result to a 
+#'   `data.table` for convenience
+#' @importFrom data.table setDT
+array_to_long_dt <- function(x) {
+  df <- array_to_long_df(x)
+  data.table::setDT(df)
+  df[]
+}
+
+
+
+
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `array_to_ratetable`: converts an array to a [survival::ratetable]
+#' @param dim.types `[integer]` (mandatory, no default)
+#' 
+#' see `type` under **Details** in [survival::ratetable] 
+#' @param cut.points `[NULL, list]` (optional, default `NULL`)
+#' 
+#' see `cutpoints` under **Details** in [survival::ratetable] 
+#' 
+#' - `NULL`: automatically set using `dimnames(x)` and `dim.types`
+#' - `list`: one element for each dimensions of `x`
+#' @importFrom data.table copy setattr
+array_to_ratetable <- function(x, dim.types, cut.points = NULL) {
+  stopifnot(
+    is.array(x),
+    length(dim.types) == length(dim(x)),
+    dim.types %in% 1:4,
+    
+    is.null(cut.points) || inherits(cut.points, "list")
+  )
+  if (is.null(cut.points)) {
+    cut.points <- lapply(seq_along(dim.types), function(k) {
+      if (dim.types[k] == 1L) {
+        return(NULL)
+      }
+      cp <- as.numeric(dimnames(x)[[k]])
+      if (all(cp %% 1L == 0L)) {
+        cp <- as.integer(cp)
+      }
+      cp
+    })
+  }
+  
+  x <- data.table::copy(x)
+  data.table::setattr(x, "type", as.integer(dim.types))
+  data.table::setattr(x, "cutpoints", cut.points)
+  data.table::setattr(x, "class", "ratetable")
+  x
+}
+
+
+
+
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `ratetable_to_array`: converts a [survival::ratetable] to an array
+#' @importFrom data.table copy setattr
+ratetable_to_array <- function(x) {
+  stopifnot(
+    inherits(x, "ratetable")
+  )
+  array(x, dim = dim(x), dimnames = dimnames(x))
+}
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `ratetable_to_long_df`: calls `ratetable_to_array` and then
+#'   `array_to_long_df`
+#' @importFrom data.table copy setattr
+ratetable_to_long_df <- function(x) {
+  array_to_long_df(ratetable_to_array(x))
+}
+
+#' @rdname array_df_ratetable_utils
+#' @export
+#' @details
+#' - `ratetable_to_long_dt`: calls `ratetable_to_array` and then
+#'   `array_to_long_dt`
+#' @importFrom data.table copy setattr
+ratetable_to_long_dt <- function(x) {
+  array_to_long_dt(ratetable_to_array(x))
+}
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/R/ltable.R b/R/ltable.R
index 7929288..8e31742 100644
--- a/R/ltable.R
+++ b/R/ltable.R
@@ -1,255 +1,255 @@
-#' @title Tabulate Counts and Other Functions by Multiple Variables into a 
-#' Long-Format Table
-#' @author Joonas Miettinen, Matti Rantanen
-#' @description \code{ltable} makes use of \code{data.table} 
-#' capabilities to tabulate frequencies or 
-#' arbitrary functions of given variables into a long format 
-#' \code{data.table}/\code{data.frame}. \code{expr.by.cj} is the 
-#' equivalent for more advanced users.
-#' @param data a \code{data.table}/\code{data.frame}
-#' @param by.vars names of variables that are used for categorization, 
-#' as a character vector, e.g. \code{c('sex','agegroup')}
-#' @param expr object or a list of objects where each object is a function 
-#' of a variable (see: details)
-#' @param subset a logical condition; data is limited accordingly before
-#' evaluating \code{expr} - but the result of \code{expr} is also
-#' returned as \code{NA} for levels not existing in the subset. See Examples.
-#' @param use.levels logical; if \code{TRUE}, uses factor levels of given 
-#' variables if present;  if you want e.g. counts for levels
-#' that actually have zero observations but are levels in a factor variable, 
-#' use this
-#' @param na.rm logical; if \code{TRUE}, drops rows in table that have 
-#' \code{NA} as values in any of \code{by.vars} columns
-#' @param robust logical; if \code{TRUE}, runs the output data's 
-#' \code{by.vars} columns through \code{robust_values} before outputting
-#' @param .SDcols advanced; a character vector of column names 
-#' passed to inside the data.table's brackets 
-#' \code{DT[, , ...]}; see \code{\link{data.table}}; if \code{NULL},
-#' uses all appropriate columns. See Examples for usage.
-#' @param enclos advanced; an environment; the enclosing
-#' environment of the data.
-#' @param ... advanced; other arguments passed to inside the 
-#' data.table's brackets \code{DT[, , ...]}; see \code{\link{data.table}}
-#' 
-#' @import data.table 
-#' 
-#' @details 
-#' 
-#' Returns \code{expr} for each unique combination of given \code{by.vars}.
-#' 
-#' By default makes use of any and all \code{\link{levels}} present for 
-#' each variable in  \code{by.vars}. This is useful,
-#' because even if a subset of the data does not contain observations 
-#' for e.g. a specific age group, those age groups are 
-#' nevertheless presented in the resulting table; e.g. with the default 
-#' \code{expr = list(obs = .N)} all age group levels
-#' are represented by a row and can have  \code{obs = 0}.
-#' 
-#' The function differs from the
-#' vanilla \code{\link{table}} by giving a long format table of values
-#' regardless of the number of \code{by.vars} given.
-#' Make use of e.g. \code{\link{cast_simple}} if data needs to be 
-#' presented in a wide format (e.g. a two-way table).
-#' 
-#' The rows of the long-format table are effectively Cartesian products 
-#' of the levels of each variable in  \code{by.vars},
-#' e.g. with  \code{by.vars = c("sex", "area")} all levels of  
-#' \code{area} are repeated for both levels of  \code{sex}
-#' in the table.
-#' 
-#' The \code{expr} allows the user to apply any function(s) on all 
-#' levels defined by  \code{by.vars}. Here are some examples:
-#' \itemize{
-#'   \item .N or list(.N) is a function used inside a \code{data.table} to 
-#'   calculate counts in each group
-#'   \item list(obs = .N), same as above but user assigned variable name
-#'   \item list(sum(obs), sum(pyrs), mean(dg_age)), multiple objects in a list
-#'   \item list(obs = sum(obs), pyrs = sum(pyrs)), same as above with user 
-#'   defined variable names
-#' }
-#' 
-#' If  \code{use.levels = FALSE}, no \code{levels} information will
-#'  be used. This means that if e.g. the  \code{agegroup}
-#' variable is a factor and has 18 levels defined, but only 15 levels
-#'  are present in the data, no rows for the missing
-#' levels will be shown in the table.
-#' 
-#' \code{na.rm} simply drops any rows from the resulting table where 
-#' any of the  \code{by.vars} values was \code{NA}. 
-#' 
-#' @seealso
-#' \code{\link{table}}, \code{\link{cast_simple}}, \code{\link{melt}}
-#' 
-#' @return
-#' A `data.table` of statistics (e.g. counts) stratified by the columns defined
-#' in `by.vars`.
-#' 
-#' @export ltable
-#' 
-#' @examples
-#' data("sire", package = "popEpi")
-#' sr <- sire
-#' sr$agegroup <- cut(sr$dg_age, breaks=c(0,45,60,75,85,Inf))
-#' ## counts by default
-#' ltable(sr, "agegroup")
-#' 
-#' ## any expression can be given
-#' ltable(sr, "agegroup", list(mage = mean(dg_age)))
-#' ltable(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
-#' 
-#' ## also returns levels where there are zero rows (expressions as NA)
-#' ltable(sr, "agegroup", list(obs = .N, 
-#'                             minage = min(dg_age), 
-#'                             maxage = max(dg_age)), 
-#'        subset = dg_age < 85)
-#'        
-#' #### expr.by.cj
-#' expr.by.cj(sr, "agegroup")
-#' 
-#' ## any arbitrary expression can be given
-#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age)))
-#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
-#' 
-#' ## only uses levels of by.vars present in data
-#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)), 
-#'            subset = dg_age < 70)
-#'            
-#' ## .SDcols trick
-#' expr.by.cj(sr, "agegroup", lapply(.SD, mean), 
-#'            subset = dg_age < 70, .SDcols = c("dg_age", "status"))
-
-ltable <- function(data, 
-                   by.vars = NULL, 
-                   expr = list(obs = .N), 
-                   subset = NULL, 
-                   use.levels = TRUE, 
-                   na.rm = FALSE,
-                   robust = TRUE) {
-  
-  PF <- parent.frame()
-  TF <- environment()
-  
-  e <- substitute(expr)
-  
-  ## eval subset ---------------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset, enclos = PF)
-  
-  ## create table --------------------------------------------------------------
-  res <- expr.by.cj(data = data,
-                    by.vars = by.vars,
-                    expr = e,
-                    subset = subset,
-                    use.levels = use.levels,
-                    na.rm = na.rm,
-                    robust = robust)
-  
-  
-  ## final touch ---------------------------------------------------------------
-  
-  if (!return_DT()) {
-    setDFpe(res)
-  }
-  res
-  
-}
-
-
-
-
-#' @describeIn ltable Somewhat more streamlined \code{ltable} with 
-#' defaults for speed. Explicit determination of enclosing environment
-#' of data.
-#' @export expr.by.cj
-
-expr.by.cj <- function(data, 
-                       by.vars = NULL, 
-                       expr = list(obs = .N), 
-                       subset = NULL, 
-                       use.levels = FALSE, 
-                       na.rm = FALSE,
-                       robust = FALSE,
-                       .SDcols = NULL,
-                       enclos = parent.frame(1L),
-                       ...) {
-  
-  PF <- enclos
-  TF <- environment()
-  
-  
-  ## checks --------------------------------------------------------------------
-  if (!is.data.frame(data)) {
-    stop("Argument 'data' must be data.frame (data.table is fine too)")
-  }
-  
-  stopifnot(is.environment(enclos))
-  stopifnot(is.logical(na.rm))
-  stopifnot(is.logical(use.levels))
-  
-  stopifnot(is.character(by.vars) || is.null(by.vars))
-  all_names_present(data, c(by.vars))
-  
-  stopifnot(is.character(.SDcols) || is.null(.SDcols))
-  all_names_present(data, .SDcols)
-  
-  tab <- data.table(data[1:min(10, nrow(data)),])
-  e <- substitute(expr)
-  e <- tab[, evalRecursive(e, env = .SD, enc = PF)$argSub]
-  
-  ## eval subset ---------------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset, enclos = PF)
-  
-  ## retrieve data to use without taking copy ----------------------------------
-  
-  tabVars <- unique(c(by.vars, all.vars(e), .SDcols))
-  tabVars <- intersect(names(data), tabVars)
-  
-  tab <- mget(tabVars, envir = as.environment(data))
-  setDT(tab)
-  
-  tmpDum <- makeTempVarName(data, pre = "dummy_")
-  if (!length(by.vars)) {
-    if (!length(tab)) {
-      ## no by.vars nor variables in expr
-      tab <- data.table(rep(1L, nrow(data)))
-      setnames(tab, "V1", tmpDum)
-    } else {
-      tab[, c(tmpDum) := 1L]
-    }
-    by.vars <- tmpDum
-  }
-  
-  ## create joining table ------------------------------------------------------
-  lev_fun <- function(x) {
-    if (use.levels && is.factor(x)) {
-      factor(levels(x), levels = levels(x))
-    } else {
-      sort(unique(x), na.last = TRUE)
-    }
-  }
-  cj <- lapply(as.list(tab)[by.vars], lev_fun)
-  cj <- do.call(CJ, c(cj, unique = FALSE, sorted = FALSE))
-  if (na.rm) cj <- na.omit(cj)
-  
-  ## eval expression -----------------------------------------------------------
-  tabe <- "tab[subset][cj, eval(e), 
-                       on = by.vars, 
-                       by = .EACHI, ..."
-  tabe <- if (is.null(.SDcols)) tabe else paste0(tabe, ", .SDcols = .SDcols")
-  tabe <- paste0(tabe ,"]")
-  res <- eval(parse(text = tabe))
-  
-  setcolsnull(res, delete = tmpDum, soft = TRUE)
-  by.vars <- setdiff(by.vars, tmpDum)
-  
-  ## final touch ---------------------------------------------------------------
-  if (length(res)) setcolorder(res, c(by.vars, setdiff(names(res), by.vars)))
-  if (length(by.vars)) setkeyv(res, by.vars)
-  if (!return_DT()) {
-    setDFpe(res)
-  }
-  res
-  
-}
-
+#' @title Tabulate Counts and Other Functions by Multiple Variables into a 
+#' Long-Format Table
+#' @author Joonas Miettinen, Matti Rantanen
+#' @description \code{ltable} makes use of \code{data.table} 
+#' capabilities to tabulate frequencies or 
+#' arbitrary functions of given variables into a long format 
+#' \code{data.table}/\code{data.frame}. \code{expr.by.cj} is the 
+#' equivalent for more advanced users.
+#' @param data a \code{data.table}/\code{data.frame}
+#' @param by.vars names of variables that are used for categorization, 
+#' as a character vector, e.g. \code{c('sex','agegroup')}
+#' @param expr object or a list of objects where each object is a function 
+#' of a variable (see: details)
+#' @param subset a logical condition; data is limited accordingly before
+#' evaluating \code{expr} - but the result of \code{expr} is also
+#' returned as \code{NA} for levels not existing in the subset. See Examples.
+#' @param use.levels logical; if \code{TRUE}, uses factor levels of given 
+#' variables if present;  if you want e.g. counts for levels
+#' that actually have zero observations but are levels in a factor variable, 
+#' use this
+#' @param na.rm logical; if \code{TRUE}, drops rows in table that have 
+#' \code{NA} as values in any of \code{by.vars} columns
+#' @param robust logical; if \code{TRUE}, runs the output data's 
+#' \code{by.vars} columns through \code{robust_values} before outputting
+#' @param .SDcols advanced; a character vector of column names 
+#' passed to inside the data.table's brackets 
+#' \code{DT[, , ...]}; see \code{\link{data.table}}; if \code{NULL},
+#' uses all appropriate columns. See Examples for usage.
+#' @param enclos advanced; an environment; the enclosing
+#' environment of the data.
+#' @param ... advanced; other arguments passed to inside the 
+#' data.table's brackets \code{DT[, , ...]}; see \code{\link{data.table}}
+#' 
+#' @import data.table 
+#' 
+#' @details 
+#' 
+#' Returns \code{expr} for each unique combination of given \code{by.vars}.
+#' 
+#' By default makes use of any and all \code{\link{levels}} present for 
+#' each variable in  \code{by.vars}. This is useful,
+#' because even if a subset of the data does not contain observations 
+#' for e.g. a specific age group, those age groups are 
+#' nevertheless presented in the resulting table; e.g. with the default 
+#' \code{expr = list(obs = .N)} all age group levels
+#' are represented by a row and can have  \code{obs = 0}.
+#' 
+#' The function differs from the
+#' vanilla \code{\link{table}} by giving a long format table of values
+#' regardless of the number of \code{by.vars} given.
+#' Make use of e.g. \code{\link{cast_simple}} if data needs to be 
+#' presented in a wide format (e.g. a two-way table).
+#' 
+#' The rows of the long-format table are effectively Cartesian products 
+#' of the levels of each variable in  \code{by.vars},
+#' e.g. with  \code{by.vars = c("sex", "area")} all levels of  
+#' \code{area} are repeated for both levels of  \code{sex}
+#' in the table.
+#' 
+#' The \code{expr} allows the user to apply any function(s) on all 
+#' levels defined by  \code{by.vars}. Here are some examples:
+#' \itemize{
+#'   \item .N or list(.N) is a function used inside a \code{data.table} to 
+#'   calculate counts in each group
+#'   \item list(obs = .N), same as above but user assigned variable name
+#'   \item list(sum(obs), sum(pyrs), mean(dg_age)), multiple objects in a list
+#'   \item list(obs = sum(obs), pyrs = sum(pyrs)), same as above with user 
+#'   defined variable names
+#' }
+#' 
+#' If  \code{use.levels = FALSE}, no \code{levels} information will
+#'  be used. This means that if e.g. the  \code{agegroup}
+#' variable is a factor and has 18 levels defined, but only 15 levels
+#'  are present in the data, no rows for the missing
+#' levels will be shown in the table.
+#' 
+#' \code{na.rm} simply drops any rows from the resulting table where 
+#' any of the  \code{by.vars} values was \code{NA}. 
+#' 
+#' @seealso
+#' \code{\link{table}}, \code{\link{cast_simple}}, \code{\link{melt}}
+#' 
+#' @return
+#' A `data.table` of statistics (e.g. counts) stratified by the columns defined
+#' in `by.vars`.
+#' 
+#' @export ltable
+#' 
+#' @examples
+#' data("sire", package = "popEpi")
+#' sr <- sire
+#' sr$agegroup <- cut(sr$dg_age, breaks=c(0,45,60,75,85,Inf))
+#' ## counts by default
+#' ltable(sr, "agegroup")
+#' 
+#' ## any expression can be given
+#' ltable(sr, "agegroup", list(mage = mean(dg_age)))
+#' ltable(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
+#' 
+#' ## also returns levels where there are zero rows (expressions as NA)
+#' ltable(sr, "agegroup", list(obs = .N, 
+#'                             minage = min(dg_age), 
+#'                             maxage = max(dg_age)), 
+#'        subset = dg_age < 85)
+#'        
+#' #### expr.by.cj
+#' expr.by.cj(sr, "agegroup")
+#' 
+#' ## any arbitrary expression can be given
+#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age)))
+#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
+#' 
+#' ## only uses levels of by.vars present in data
+#' expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)), 
+#'            subset = dg_age < 70)
+#'            
+#' ## .SDcols trick
+#' expr.by.cj(sr, "agegroup", lapply(.SD, mean), 
+#'            subset = dg_age < 70, .SDcols = c("dg_age", "status"))
+
+ltable <- function(data, 
+                   by.vars = NULL, 
+                   expr = list(obs = .N), 
+                   subset = NULL, 
+                   use.levels = TRUE, 
+                   na.rm = FALSE,
+                   robust = TRUE) {
+  
+  PF <- parent.frame()
+  TF <- environment()
+  
+  e <- substitute(expr)
+  
+  ## eval subset ---------------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset, enclos = PF)
+  
+  ## create table --------------------------------------------------------------
+  res <- expr.by.cj(data = data,
+                    by.vars = by.vars,
+                    expr = e,
+                    subset = subset,
+                    use.levels = use.levels,
+                    na.rm = na.rm,
+                    robust = robust)
+  
+  
+  ## final touch ---------------------------------------------------------------
+  
+  if (!return_DT()) {
+    setDFpe(res)
+  }
+  res
+  
+}
+
+
+
+
+#' @describeIn ltable Somewhat more streamlined \code{ltable} with 
+#' defaults for speed. Explicit determination of enclosing environment
+#' of data.
+#' @export expr.by.cj
+
+expr.by.cj <- function(data, 
+                       by.vars = NULL, 
+                       expr = list(obs = .N), 
+                       subset = NULL, 
+                       use.levels = FALSE, 
+                       na.rm = FALSE,
+                       robust = FALSE,
+                       .SDcols = NULL,
+                       enclos = parent.frame(1L),
+                       ...) {
+  
+  PF <- enclos
+  TF <- environment()
+  
+  
+  ## checks --------------------------------------------------------------------
+  if (!is.data.frame(data)) {
+    stop("Argument 'data' must be data.frame (data.table is fine too)")
+  }
+  
+  stopifnot(is.environment(enclos))
+  stopifnot(is.logical(na.rm))
+  stopifnot(is.logical(use.levels))
+  
+  stopifnot(is.character(by.vars) || is.null(by.vars))
+  all_names_present(data, c(by.vars))
+  
+  stopifnot(is.character(.SDcols) || is.null(.SDcols))
+  all_names_present(data, .SDcols)
+  
+  tab <- data.table(data[1:min(10, nrow(data)),])
+  e <- substitute(expr)
+  e <- tab[, evalRecursive(e, env = .SD, enc = PF)$argSub]
+  
+  ## eval subset ---------------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset, enclos = PF)
+  
+  ## retrieve data to use without taking copy ----------------------------------
+  
+  tabVars <- unique(c(by.vars, all.vars(e), .SDcols))
+  tabVars <- intersect(names(data), tabVars)
+  
+  tab <- mget(tabVars, envir = as.environment(data))
+  setDT(tab)
+  
+  tmpDum <- makeTempVarName(data, pre = "dummy_")
+  if (!length(by.vars)) {
+    if (!length(tab)) {
+      ## no by.vars nor variables in expr
+      tab <- data.table(rep(1L, nrow(data)))
+      setnames(tab, "V1", tmpDum)
+    } else {
+      tab[, c(tmpDum) := 1L]
+    }
+    by.vars <- tmpDum
+  }
+  
+  ## create joining table ------------------------------------------------------
+  lev_fun <- function(x) {
+    if (use.levels && is.factor(x)) {
+      factor(levels(x), levels = levels(x))
+    } else {
+      sort(unique(x), na.last = TRUE)
+    }
+  }
+  cj <- lapply(as.list(tab)[by.vars], lev_fun)
+  cj <- do.call(CJ, c(cj, unique = FALSE, sorted = FALSE))
+  if (na.rm) cj <- na.omit(cj)
+  
+  ## eval expression -----------------------------------------------------------
+  tabe <- "tab[subset][cj, eval(e), 
+                       on = by.vars, 
+                       by = .EACHI, ..."
+  tabe <- if (is.null(.SDcols)) tabe else paste0(tabe, ", .SDcols = .SDcols")
+  tabe <- paste0(tabe ,"]")
+  res <- eval(parse(text = tabe))
+  
+  setcolsnull(res, delete = tmpDum, soft = TRUE)
+  by.vars <- setdiff(by.vars, tmpDum)
+  
+  ## final touch ---------------------------------------------------------------
+  if (length(res)) setcolorder(res, c(by.vars, setdiff(names(res), by.vars)))
+  if (length(by.vars)) setkeyv(res, by.vars)
+  if (!return_DT()) {
+    setDFpe(res)
+  }
+  res
+  
+}
+
diff --git a/R/mean_survival.R b/R/mean_survival.R
index aebfcb1..4138aa5 100644
--- a/R/mean_survival.R
+++ b/R/mean_survival.R
@@ -1,658 +1,658 @@
-
-
-
-#' @title Compute Mean Survival Times Using Extrapolation
-#' @description Computes mean survival times based on survival estimation up to
-#' a point in follow-up time (e.g. 10 years), 
-#' after which survival is extrapolated
-#' using an appropriate hazard data file (\code{pophaz}) to yield the "full"
-#' survival curve. The area under the full survival curve is the mean survival.
-#' @author Joonas Miettinen
-#' @param formula a \code{formula}, e.g. \code{FUT ~ V1} or 
-#' \code{Surv(FUT, lex.Xst) ~ V1}.
-#' Supplied in the same way as to \code{\link{survtab}}, see that help
-#' for more info.
-#' @param data a \code{Lexis} data set; see \code{\link[Epi]{Lexis}}.
-#' @param adjust variables to adjust estimates by, e.g. \code{adjust = "agegr"}.
-#' \link[=flexible_argument]{Flexible input}.
-#' @param weights weights to use to adjust mean survival times. See the
-#' \link[=direct_standardization]{dedicated help page} for more details on 
-#' weighting. \code{survmean}
-#' computes curves separately by all variables to adjust by, computes mean
-#' survival times, and computes weighted means of the mean survival times.
-#' See Examples.
-#' @param breaks a list of breaks defining the time window to compute 
-#' observed survival in, and the intervals used in estimation. E.g.
-#' \code{list(FUT = 0:10)} when \code{FUT} is the follow-up time scale in your
-#' data.
-#' @param pophaz a data set of population hazards passed to 
-#' \code{\link{survtab}} (see the 
-#' \link[=pophaz]{dedicated help page} and the help page of
-#' \code{survtab} for more information). Defines the 
-#' population hazard in the time window where observed survival is estimated.
-#' @param e1.breaks \code{NULL} or a list of breaks defining the time 
-#' window to compute 
-#' \strong{expected} survival in, and the intervals used in estimation. E.g.
-#' \code{list(FUT = 0:100)} when \code{FUT} is the follow-up time scale in your
-#' data to extrapolate up to 100 years from where the observed survival
-#' curve ends. \strong{NOTE:} the breaks on the survival time scale
-#' MUST include the breaks supplied to argument \code{breaks}; see Examples.
-#' If \code{NULL}, uses decent defaults (maximum follow-up time of 50 years).
-#' @param e1.pophaz Same as \code{pophaz}, except this defines the 
-#' population hazard in the time window where \strong{expected} 
-#' survival is estimated. By default uses the same data as 
-#' argument \code{pophaz}.
-#' @param r either a numeric multiplier such as \code{0.995}, \code{"auto"}, or
-#' \code{"autoX"} where \code{X} is an integer;
-#' used to determine the relative survival ratio (RSR) persisting after where 
-#' the estimated observed survival curve ends. See Details.
-#' @param surv.method passed to \code{survtab}; see that help for more info.
-#' @param subset a logical condition; e.g. \code{subset = sex == 1}; 
-#' subsets the data before computations
-#' @param verbose \code{logical}; if \code{TRUE}, the function is returns
-#' some messages and results along the run, which may be useful in debugging
-#' @details
-#' \strong{Basics}
-#' 
-#' \code{survmean} computes mean survival times. For median survival times
-#' (i.e. where 50 % of subjects have died or met some other event)
-#' use \code{\link{survtab}}.
-#' 
-#' The mean survival time is simply the area under the survival curve.
-#' However, since full follow-up rarely happens, the observed survival curves
-#' are extrapolated using expected survival: E.g. one might compute observed
-#' survival till up to 10 years and extrapolate beyond that 
-#' (till e.g. 50 years) to yield an educated guess on the full observed survival
-#' curve. 
-#' 
-#' The area is computed by trapezoidal integration of the area under the curve.
-#' This function also computes the "full" expected survival curve from
-#' T = 0 till e.g. T = 50 depending on supplied arguments. The
-#' expected mean survival time is the area under the 
-#' mean expected survival curve.
-#' This function returns the mean expected survival time to be compared with 
-#' the mean survival time and for computing years of potential life lost (YPLL).
-#' 
-#' Results can be formed by strata and adjusted for e.g. age by using
-#' the \code{formula} argument as in \code{survtab}. See also Examples.
-#' 
-#' \strong{Extrapolation tweaks}
-#' 
-#' Argument \code{r} controls the relative survival ratio (RSR) assumed to
-#' persist beyond the time window where observed survival is computed
-#' (defined by argument \code{breaks}; e.g. up to \code{FUT = 10}).
-#' The RSR is simply \code{RSR_i = p_oi / p_ei} for a time interval \code{i}, 
-#' i.e. the observed divided by the expected 
-#' (conditional, not cumulative) probability of surviving from the beginning of
-#' a time interval till its end. The cumulative product of \code{RSR_i}
-#' over time is the (cumulative) relative survival curve. 
-#' 
-#'
-#' If \code{r} is numeric, e.g. \code{r = 0.995}, that RSR level is assumed
-#' to persist beyond the observed survival curve. 
-#' Numeric \code{r} should be \code{> 0} and expressed at the annual level
-#' when using fractional years as the scale of the time variables.
-#' E.g. if RSR is known to be \code{0.95} at the month level, then the
-#' annualized RSR is \code{0.95^12}. This enables correct usage of the RSR
-#' with survival intervals of varying lengths. When using day-level time 
-#' variables (such as \code{Dates}; see \code{as.Date}), numeric \code{r}
-#' should be expressed at the day level, etc.
-#' 
-#' If \code{r = "auto"} or \code{r = "auto1"}, this function computes
-#' RSR estimates internally and automatically uses the \code{RSR_i}
-#' in the last survival interval in each stratum (and adjusting group)
-#' and assumes that to persist beyond the observed survival curve.
-#' Automatic determination of \code{r} is a good starting point,
-#' but in situations where the RSR estimate is uncertain it may produce poor
-#' results. Using \code{"autoX"} such as \code{"auto6"} causes \code{survmean}
-#' to use the mean of the estimated RSRs in the last X survival intervals, 
-#' which may be more stable.
-#' Automatic determination will not use values \code{>1} but set them to 1. 
-#' Visual inspection of the produced curves is always recommended: see
-#' Examples.
-#' 
-#' One may also tweak the accuracy and length of extrapolation and 
-#' expected survival curve computation by using 
-#' \code{e1.breaks}. By default this is whatever was supplied to \code{breaks}
-#' for the survival time scale, to which
-#' 
-#' \code{c(seq(1/12, 1, 1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))}
-#' 
-#' is added after the maximum value, e.g. with \code{breaks = list(FUT = 0:10)}
-#' we have 
-#' 
-#' \code{..., 10+1/12, ..., 11, 11.2, ..., 2, 3, ..., 19, 20, 25, ... 50}
-#' 
-#' as the \code{e1.breaks}. Supplying \code{e1.breaks} manually requires
-#' the breaks over time survival time scale supplied to argument \code{breaks}
-#' to be reiterated in \code{e1.breaks}; see Examples. \strong{NOTE}: the
-#' default extrapolation breaks assume the time scales in the data to be 
-#' expressed as fractional years, meaning this will work extremely poorly
-#' when using e.g. day-level time scales (such as \code{Date} variables). 
-#' Set the extrapolation breaks manually in such cases.
-#' 
-#' @return 
-#' Returns a \code{data.frame} or \code{data.table} (depending on 
-#' \code{getOptions("popEpi.datatable")}; see \code{?popEpi}) containing the
-#' following columns:
-#' \itemize{
-#'   \item{est}{: The estimated mean survival time}
-#'   \item{exp}{: The computed expected survival time}
-#'   \item{obs}{: Counts of subjects in data}
-#'   \item{YPLL}{: Years of Potential Life Lost, computed as 
-#'   (\code{(exp-est)*obs}) - though your time data may be in e.g. days,
-#'   this column will have the same name regardless.}
-#' }
-#' The returned data also has columns named according to the variables
-#' supplied to the right-hand-side of the formula.
-#' 
-#' 
-#' @examples
-#' 
-#' library(Epi)
-#' ## take 500 subjects randomly for demonstration
-#' data(sire)
-#' sire <- sire[sire$dg_date < sire$ex_date, ]
-#' set.seed(1L)
-#' sire <- sire[sample(x = nrow(sire), size = 500),]
-#' 
-#' ## NOTE: recommended to use factor status variable
-#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-#'            exit = list(CAL = get.yrs(ex_date)),
-#'            data = sire,
-#'            exit.status = factor(status, levels = 0:2,
-#'                                 labels = c("alive", "canD", "othD")),
-#'            merge = TRUE)
-#' 
-#' ## phony variable
-#' set.seed(1L)
-#' x$group <- rbinom(nrow(x), 1, 0.5)
-#' ## age group
-#' x$agegr <- cut(x$dg_age, c(0,45,60,Inf), right=FALSE)
-#' 
-#' ## population hazards data  set
-#' pm <- data.frame(popEpi::popmort)
-#' names(pm) <- c("sex", "CAL", "AGE", "haz")
-#' 
-#' ## breaks to define observed survival estimation
-#' BL <- list(FUT = seq(0, 10, 1/12))
-#' 
-#' ## crude mean survival
-#' sm1 <- survmean(Surv(FUT, lex.Xst != "alive") ~ 1,
-#'                 pophaz = pm, data = x, weights = NULL,
-#'                 breaks = BL)
-#'                 
-#' sm1 <- survmean(FUT ~ 1,
-#'                 pophaz = pm, data = x, weights = NULL,
-#'                 breaks = BL)             
-#' \donttest{
-#' ## mean survival by group                 
-#' sm2 <- survmean(FUT ~ group,
-#'                 pophaz = pm, data = x, weights = NULL,
-#'                 breaks = BL)
-#'                 
-#' ## ... and adjusted for age using internal weights (counts of subjects)      
-#' ## note: need also longer extrapolation here so that all curves
-#' ## converge to zero in the end.
-#' eBL <- list(FUT = c(BL$FUT, 11:75))
-#' sm3 <- survmean(FUT ~ group + adjust(agegr),
-#'                 pophaz = pm, data = x, weights = "internal",
-#'                 breaks = BL, e1.breaks = eBL)
-#' }
-
-#' ## visual inspection of how realistic extrapolation is for each stratum;
-#' ## solid lines are observed + extrapolated survivals;
-#' ## dashed lines are expected survivals
-#' plot(sm1)
-#' \donttest{
-#' ## plotting object with both stratification and standardization
-#' ## plots curves for each strata-std.group combination
-#' plot(sm3)
-#' 
-#' ## for finer control of plotting these curves, you may extract
-#' ## from the survmean object using e.g.
-#' attributes(sm3)$survmean.meta$curves
-#' 
-#' 
-#' #### using Dates
-#' 
-#' x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-#'            exit = list(CAL = ex_date),
-#'            data = sire[sire$dg_date < sire$ex_date, ],
-#'            exit.status = factor(status, levels = 0:2, 
-#'                                 labels = c("alive", "canD", "othD")), 
-#'            merge = TRUE)
-#' ## phony group variable
-#' set.seed(1L)
-#' x$group <- rbinom(nrow(x), 1, 0.5)
-#' 
-#'                   
-#' ## NOTE: population hazard should be reported at the same scale
-#' ## as time variables in your Lexis data.
-#' data(popmort, package = "popEpi")
-#' pm <- data.frame(popmort)
-#' names(pm) <- c("sex", "CAL", "AGE", "haz")
-#' ## from year to day level
-#' pm$haz <- pm$haz/365.25 
-#' pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
-#' pm$AGE <- pm$AGE*365.25 
-#' 
-#' BL <- list(FUT = seq(0, 8, 1/12)*365.25)
-#' eBL <- list(FUT = c(BL$FUT, c(8.25,8.5,9:60)*365.25))
-#' smd <- survmean(FUT ~ group, data = x, 
-#'                 pophaz = pm, verbose = TRUE, r = "auto5",
-#'                 breaks = BL, e1.breaks = eBL)     
-#' plot(smd)
-#' }
-#' 
-
-#' 
-#' @export
-#' @family survmean functions
-#' @family main functions
-#' 
-
-survmean <- function(formula, data, adjust = NULL, weights = NULL, 
-                     breaks=NULL, pophaz = NULL, 
-                     e1.breaks = NULL, e1.pophaz = pophaz, r = "auto", 
-                     surv.method = "hazard", subset = NULL, verbose = FALSE) {
-  pt <- proc.time()
-  TF__ <- environment()
-  PF__ <- parent.frame(1L)
-  
-  attr_form <- copy(formula)
-  
-  surv.method <- match.arg(surv.method, c("hazard", "lifetable"))
-  
-  ## appease R CMD CHECK (due to using vars in DT[] only)
-  r.e2 <- last.p.e2 <- surv <- survmean_type <- est <- Tstart <- Tstop <- 
-    lex.id <- surv.int <- delta <- surv.exp <- obs <- NULL
-  
-  checkLexisData(data, check.breaks = FALSE)
-  checkPophaz(data, pophaz, haz.name = "haz")
-  checkPophaz(data, e1.pophaz, haz.name = "haz")
-  pophaz <- setDT(copy(pophaz))
-  e1.pophaz <- setDT(copy(e1.pophaz))
-  
-  if (is.numeric(r) && r < 0L) stop("numeric r must be > 0, e.g. r = 0.95")
-  if (is.character(r)) {
-    if (substr(r, 1, 4) != "auto") {
-      stop("character string r must start with 'auto'; e.g. `auto` and ",
-           "`auto5` are accepted.")
-    }
-    if (r == "auto") r <- "auto1"
-    
-    auto_ints <- regmatches(r, regexec("\\d+", text = r))
-    auto_ints <- as.integer(auto_ints)
-    r <- "auto"
-  }
-  
-  tscales_all <- attr(data, "time.scales")
-  breaks_old <- attr(data, "breaks")
-  
-  
-  
-  ## breaks --------------------------------------------------------------------
-  
-  if (!is.null(breaks_old)) checkBreaksList(data, breaks_old)
-  if (is.null(breaks)) breaks <- breaks_old
-  
-  checkBreaksList(data, breaks)
-  
-  ## hmm - will later on set breaks on the found survival scale
-  if (!is.null(e1.breaks))  checkBreaksList(data, e1.breaks)
-  
-  ## prep & subset data --------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset)
-  
-  x <- setDT(data[subset, ])
-  forceLexisDT(x, breaks = breaks_old, allScales = tscales_all)
-  
-  ## ensure variables to merge pophaz datas by are kept ------------------------
-  ## NOTE: temp var names avoid conflicts down the line
-  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
-  
-  pophaz_vars <- c(names(pophaz), names(e1.pophaz))
-  pophaz_vars <- setdiff(pophaz_vars, c(tscales_all, "haz"))
-  pophaz_vars <- intersect(pophaz_vars, names(x))
-  pophaz_vars_tmp <- makeTempVarName(names = avoid, pre = pophaz_vars)
-  if (!length(pophaz_vars)) {
-    pophaz_vars_tmp <- NULL
-  } else {
-    pophaz_vars_wh <- which(pophaz_vars %in% names(pophaz))
-    if (sum(pophaz_vars_wh)) {
-      setnames(pophaz, old = pophaz_vars[pophaz_vars_wh], 
-               new = pophaz_vars_tmp[pophaz_vars_wh])
-    }
-    pophaz_vars_wh <- which(pophaz_vars %in% names(e1.pophaz))
-    if (sum(pophaz_vars_wh)) {
-      setnames(e1.pophaz, old = pophaz_vars[pophaz_vars_wh], 
-               new = pophaz_vars_tmp[pophaz_vars_wh])
-    }
-    x[, (pophaz_vars_tmp) := copy(.SD), .SDcols = pophaz_vars]
-  }
-  
-  ## determine printing & adjusting vars ---------------------------------------
-  adSub <- substitute(adjust)
-  foList <- usePopFormula(formula, adjust = adSub, data = x, enclos = PF__, 
-                          Surv.response = "either")
-  
-  ## will avoid conflicts using temp names for tabulating variables
-  adjust_vars <- names(foList$adjust)
-  print_vars <- names(foList$print)
-  by_vars <- c(print_vars, adjust_vars)
-  
-  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
-  adjust_vars_tmp <- makeTempVarName(names = avoid, pre = adjust_vars)
-  if (!length(adjust_vars)) adjust_vars_tmp <- NULL
-  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
-  print_vars_tmp <- makeTempVarName(names = avoid, pre = print_vars)
-  if (!length(print_vars)) print_vars_tmp <- NULL
-  by_vars_tmp  <- c(print_vars_tmp, adjust_vars_tmp)
-  
-  
-  lex_vars <- c("lex.id", tscales_all, "lex.dur", "lex.Cst", "lex.Xst")
-  setcolsnull(x, keep = c(lex_vars, pophaz_vars_tmp), soft = FALSE)
-  if (length(adjust_vars) > 0L) x[, (adjust_vars_tmp) := foList$adjust]
-  if (length(print_vars) > 0L) x[, (print_vars_tmp) := foList$print]
-  
-  ## formula for survtab: we estimate survivals by all levels of both
-  ## print and adjust; adjusting here means computing directly adjusted
-  ## estimates of the mean survival time, so mean survival times are
-  ## weighted later on.
-  
-  formula <- paste0(deparse(formula[[2L]]), " ~ ")
-  if (length(c(adjust_vars_tmp, print_vars_tmp)) > 0L) {
-    formula <- paste0(formula, paste0(c(print_vars_tmp, adjust_vars_tmp), 
-                                      collapse = " + "))
-  } else {
-    formula <- paste0(formula, "1")
-  }
-  formula <- as.formula(formula)
-  
-  ## detect survival time scale ------------------------------------------------
-  tscale_surv <- detectSurvivalTimeScale(lex = x, values = foList$y$time)
-  
-  ## check weights & adjust ----------------------------------------------------
-  test_obs <- x[, .(obs=.N),  keyby=eval(TF__$by_vars_tmp)]
-  if (length(by_vars)) setnames(test_obs, by_vars_tmp, by_vars)
-  if (length(weights) && !length(adjust_vars)) {
-    weights <- NULL
-    warning("Replaced weights with NULL due to not supplying variables to ",
-            "adjust by.")
-  }
-  mwDTtest <- makeWeightsDT(test_obs, values = list("obs"), print = print_vars,
-                            adjust = adjust_vars, weights = weights, 
-                            internal.weights.values = "obs")
-  if (length(by_vars)) setnames(test_obs, by_vars, by_vars_tmp)
-  
-  ## figure out extrapolation breaks -------------------------------------------
-  ## now that the survival time scale is known this can actually be done.
-  
-  if (is.null(e1.breaks)) {
-    e1.breaks <- copy(breaks[tscale_surv])
-    addBreaks <- max(e1.breaks[[tscale_surv]]) + 
-      c(seq(0,1,1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))
-    e1.breaks[[tscale_surv]] <- unique(c(e1.breaks[[tscale_surv]], addBreaks))
-    
-    checkBreaksList(x, e1.breaks)
-  }
-  if (!tscale_surv %in% names(e1.breaks)) {
-    stop("The survival time scale must be included in the list of breaks ",
-         "to extrapolate by ('e1.breaks').")
-  }
-  if (!all(breaks[[tscale_surv]] %in% e1.breaks[[tscale_surv]])) {
-    stop("The vector of breaks in 'breaks' for the survival time scale MUST",
-         "be a subset of the breaks for the survival time scale in ",
-         "'e1.breaks'. E.g. the former could be 0:10 and the latter 0:100.")
-  }
-  
-  if (verbose) {
-    cat("Time taken by prepping data:", timetaken(pt), "\n")
-  }
-  
-  
-  ## compute observed survivals ------------------------------------------------
-  ## NOTE: do not adjust here; adjust in original formula means weighting
-  ## the mean survival time results.
-  
-  st <- survtab(formula, data = x, breaks = breaks, 
-                pophaz = pophaz,
-                relsurv.method = "e2",
-                surv.type = "surv.rel", 
-                surv.method = surv.method)
-  
-  st_keep_vars <- c(by_vars_tmp, "Tstop", "r.e2", "surv.obs")
-  all_names_present(
-    st,  st_keep_vars, 
-    msg = paste0("Internal error: expected to have variables ",
-                 "%%VARS%% after computing observed survivals ",
-                 "but didn't. Blame the package maintainer if you ",
-                 "see this.")
-  )
-  setcolsnull(st, keep = st_keep_vars, colorder = TRUE)
-  setDT(st)
-  setkeyv(st, c(by_vars_tmp, "Tstop"))
-  st[, "Tstart" := c(0, Tstop[-.N]), by = eval(by_vars_tmp)]
-  
-  ## decumulate for later cumulation
-  st[, c("r.e2", "surv.obs") := lapply(.SD, function(col) col/c(1, col[-.N])), 
-     by = eval(by_vars_tmp),
-     .SDcols = c("r.e2", "surv.obs")
-     ]
-  
-  
-  if (verbose) {
-    cat("Time taken by estimating relative survival curves:", 
-        timetaken(pt), "\n")
-  }
-  
-  ## compute overall expected survival -----------------------------------------
-  ## 1) take only those individuals that were diagnosed in the time window
-  ##    defined by breaks list in argument 'breaks'
-  pt <- proc.time()
-  setkeyv(x, c("lex.id", tscale_surv))
-  tol <- .Machine$double.eps^0.5
-  xe <- unique(x, by = key(x))[x[[tscale_surv]] < TF__$tol, ] ## pick rows with entry to FU
-  
-  if (length(breaks) > 1L) {
-    ## e.g. a period window was defined and we only use subjects
-    ## entering follow-up in the time window.
-    breaks_drop_tmp <- setdiff(names(breaks), tscale_surv)
-    breaks_drop_tmp <- breaks[breaks_drop_tmp]
-    breaks_drop_tmp <- lapply(breaks_drop_tmp, range)
-    
-    expr <- mapply(function(ch, ra) {
-      paste0("between(", ch, ", ", ra[1], ", ", ra[2] - tol, ", incbounds = TRUE)")
-    }, ch = names(breaks_drop_tmp), ra = breaks_drop_tmp, SIMPLIFY = FALSE)
-    
-    expr <- lapply(expr, function(e) eval(parse(text = e), envir = xe))
-    setDT(expr)
-    expr <- expr[, rowSums(.SD)]  == ncol(expr)
-    xe <- xe[expr, ]
-  }
-  
-  xe <- x[lex.id %in% unique(xe[["lex.id"]])]
-  forceLexisDT(xe, breaks = breaks_old, allScales = tscales_all, key = FALSE)
-  
-  ## 2) compute Ederer I expected survival curves from T = 0 till e.g. T = 100
-  e1 <- comp_e1(xe, breaks = e1.breaks, pophaz = e1.pophaz, immortal = TRUE, 
-                survScale = tscale_surv, by = by_vars_tmp, id = "lex.id")
-  setnames(e1, tscale_surv, "Tstop")
-  e1[, "Tstart" := c(0, Tstop[-.N]), by = eval(by_vars_tmp)]
-  e1[, "surv.int" := cut(Tstart, breaks = e1.breaks[[tscale_surv]], 
-                         right = FALSE, labels = FALSE)]
-  e1[, "delta" := Tstop - Tstart]
-  
-  ## decumulate for later cumulation
-  e1[, "surv.exp" := surv.exp/c(1, surv.exp[-.N]), by = eval(by_vars_tmp)]
-  
-  if (verbose) {
-    cat("Time taken by computing overall expected survival curves:", 
-        timetaken(pt), "\n")
-  }
-  
-  ## compute counts of subjects ------------------------------------------------
-  ## these correspond to the counts of patients for which expected survival
-  ## was computed. If observed survival is e.g. a period estimated curve,
-  ## we only use subjects entering follow-up in the period window.
-  N_subjects <- xe[!duplicated(lex.id)][, 
-                                        list(obs=.N), 
-                                        keyby=eval(by_vars_tmp)
-                                        ]
-  
-  ## combine all estimates into one data set -----------------------------------
-  pt <- proc.time()
-  
-  st[, "surv.int" := cut(Tstart, breaks = e1.breaks[[tscale_surv]], 
-                         right = FALSE, labels = FALSE)]
-  
-  x <- merge(e1, st[, .SD, .SDcols = c(by_vars_tmp, "surv.int", "r.e2", "surv.obs")], 
-             by = c(by_vars_tmp,"surv.int"), all = TRUE)
-  setkeyv(x, c(by_vars_tmp, "surv.int"))
-  
-  ## extrapolation RSR definition ----------------------------------------------
-  if (is.numeric(r)) {
-    ## manually given RSR for extrapolated part of the obs.surv curve
-    ## here it is assumed that r is annualized
-    set(x, j = "last.p.e2", value = r^x[["delta"]]) 
-    
-    
-  } else {
-    ## add last non-NA values as separate column
-    
-    st <- st[, .SD[(.N-TF__$auto_ints+1):.N], by = eval(by_vars_tmp)]
-    
-    st[, "delta" := Tstop - Tstart]
-    st[, "r.e2" := r.e2^(1/delta)] ## "annualized" RSRs
-    
-    ## mean annualized RSR in last N intervas by strata
-    st <- st[, .(last.p.e2 = mean(r.e2)), by = eval(by_vars_tmp)]
-    st[, "last.p.e2" := pmin(1, last.p.e2)]
-    if (verbose) {
-      cat("Using following table of mean RSR estimates",
-          "(scaled to RSRs applicable to a time interval one",
-          "unit of time wide, e.g. one year or one day)",
-          "based on", auto_ints, "interval(s) from the end of the relative",
-          "survival curve by strata: \n")
-      prST <- data.table(st)
-      setnames(prST, c(by_vars_tmp, "last.p.e2"), c(by_vars, "RSR"))
-      print(prST)
-    }
-    
-    if (length(by_vars_tmp)) {
-      x <- merge(x, st, by = by_vars_tmp, all = TRUE)
-    } else {
-      set(x, j = "last.p.e2", value = st$last.p.e2)
-    }
-    x[, "last.p.e2" := last.p.e2^(delta)] ## back to non-annualized RSRs
-    ## enforce RSR in extrapolated part of observed curve to at most 1
-    x[, "last.p.e2" := pmin(last.p.e2, 1)]
-  }
-  
-  x[is.na(r.e2), "r.e2" := last.p.e2]
-  x[, "surv" := r.e2*surv.exp]
-  # setnames(x, "surv.obs", "surv")
-  # x[is.na(surv), "surv" := surv.exp*last.p.e2]
-  
-  ## cumulate again
-  setkeyv(x, c(by_vars_tmp, "surv.int"))
-  x[, c("surv", "surv.exp") := lapply(.SD, cumprod),
-    .SDcols = c("surv", "surv.exp"), by = eval(by_vars_tmp)]
-  
-  x2 <- copy(x)
-  x[, "surv.exp" := NULL]
-  x2[, "surv" := NULL]
-  setnames(x2, "surv.exp", "surv")
-  x <- rbind(x, x2)
-  x[, "survmean_type" := rep(c("est", "exp"), each = nrow(x2))]
-  
-  setcolsnull(
-    x, 
-    keep = c(by_vars_tmp, "survmean_type", 
-             "surv.int", "Tstart", "Tstop", 
-             "delta", "surv", "surv.exp"),
-    colorder = TRUE
-  )
-  
-  ## check curve convergence to zero -------------------------------------------
-  ## a good integration is based on curves that get very close to 
-  ## zero in the end
-  mi <- x[, .(surv = round(min(surv),4)*100), 
-          keyby = eval(c(by_vars_tmp, "survmean_type"))]
-  
-  if (any(mi$surv > 1)) {
-    warning("One or several of the curves used to compute mean survival times ",
-            "or expected mean survival times was > 1 % at the lowest point. ",
-            "Mean survival estimates may be significantly biased. To avoid ",
-            "this, supply breaks to 'e1.breaks' which make the curves longer ",
-            ", e.g. e1.breaks = list(FUT = 0:150) where time scale FUT ",
-            "is the survival time scale (yours may have a different name).")
-  }
-  mi[, "surv" := paste0(formatC(surv, digits = 2, format = "f"), " %")]
-  mi[, "survmean_type" := factor(survmean_type, c("est", "exp"),
-                                 c("Observed", "Expected"))]
-  setnames(mi, c("survmean_type", "surv"), 
-           c("Obs./Exp. curve", "Lowest value"))
-  if (length(by_vars)) setnames(mi, by_vars_tmp, by_vars)
-  if (verbose) {
-    cat("Lowest points in observed / expected survival curves by strata:\n")
-    print(mi)
-  }
-  
-  ## integrating by trapezoid areas --------------------------------------------
-  ## trapezoid area: WIDTH*(HEIGHT1 + HEIGHT2)/2
-  ## so we compute "average interval survivals" for each interval t_i
-  ## and multiply with interval length.
-  
-  setkeyv(x, c(by_vars_tmp, "survmean_type",  "Tstop"))
-  sm <- x[, .(survmean = sum(delta*(surv + c(1, surv[-.N]))/2L)), 
-          keyby = c(by_vars_tmp, "survmean_type")]
-  
-  ## cast ----------------------------------------------------------------------
-  
-  sm <- cast_simple(sm, columns = "survmean_type", 
-                    rows = by_vars_tmp, values = "survmean")
-  
-  ## add numbers of subjects, compute YPLL -------------------------------------
-  setkeyv(sm, by_vars_tmp); setkeyv(N_subjects, by_vars_tmp)
-  sm[, "obs" := N_subjects$obs]
-  sm[, "YPLL" := (exp-est)*obs]
-  
-  
-  ## adjusting -----------------------------------------------------------------
-  
-  sm <- makeWeightsDT(sm, values = list(c("est", "exp", "obs", "YPLL")),
-                      print = print_vars_tmp, adjust = adjust_vars_tmp,
-                      weights = weights, internal.weights.values = "obs")
-  if (length(adjust_vars)) {
-    vv <- c("est", "exp", "obs", "YPLL")
-    sm[, c("est", "exp") := lapply(.SD, function(col) col*sm$weights), 
-       .SDcols = c("est", "exp")]
-    sm <- sm[, lapply(.SD, sum), .SDcols = vv, by = eval(print_vars_tmp)]
-  }
-  
-  if (verbose) {
-    cat("Time taken by final touches:", timetaken(pt), "\n")
-  }
-  
-  ## final touch ---------------------------------------------------------------
-  if (length(print_vars)) setnames(sm, print_vars_tmp, print_vars)
-  
-  at <- list(call = match.call(), 
-             formula = attr_form,
-             print = print_vars, 
-             adjust = adjust_vars, 
-             tprint = print_vars_tmp, 
-             tadjust = adjust_vars_tmp,
-             breaks = breaks, 
-             e1.breaks = e1.breaks, 
-             survScale = tscale_surv,
-             curves = copy(x))
-  setattr(sm, "class", c("survmean","data.table", "data.frame"))
-  setattr(sm, "survmean.meta", at)
-  if (!return_DT()) setDFpe(sm)
-  return(sm[])
-}
-
+
+
+
+#' @title Compute Mean Survival Times Using Extrapolation
+#' @description Computes mean survival times based on survival estimation up to
+#' a point in follow-up time (e.g. 10 years), 
+#' after which survival is extrapolated
+#' using an appropriate hazard data file (\code{pophaz}) to yield the "full"
+#' survival curve. The area under the full survival curve is the mean survival.
+#' @author Joonas Miettinen
+#' @param formula a \code{formula}, e.g. \code{FUT ~ V1} or 
+#' \code{Surv(FUT, lex.Xst) ~ V1}.
+#' Supplied in the same way as to \code{\link{survtab}}, see that help
+#' for more info.
+#' @param data a \code{Lexis} data set; see \code{\link[Epi]{Lexis}}.
+#' @param adjust variables to adjust estimates by, e.g. \code{adjust = "agegr"}.
+#' \link[=flexible_argument]{Flexible input}.
+#' @param weights weights to use to adjust mean survival times. See the
+#' \link[=direct_standardization]{dedicated help page} for more details on 
+#' weighting. \code{survmean}
+#' computes curves separately by all variables to adjust by, computes mean
+#' survival times, and computes weighted means of the mean survival times.
+#' See Examples.
+#' @param breaks a list of breaks defining the time window to compute 
+#' observed survival in, and the intervals used in estimation. E.g.
+#' \code{list(FUT = 0:10)} when \code{FUT} is the follow-up time scale in your
+#' data.
+#' @param pophaz a data set of population hazards passed to 
+#' \code{\link{survtab}} (see the 
+#' \link[=pophaz]{dedicated help page} and the help page of
+#' \code{survtab} for more information). Defines the 
+#' population hazard in the time window where observed survival is estimated.
+#' @param e1.breaks \code{NULL} or a list of breaks defining the time 
+#' window to compute 
+#' \strong{expected} survival in, and the intervals used in estimation. E.g.
+#' \code{list(FUT = 0:100)} when \code{FUT} is the follow-up time scale in your
+#' data to extrapolate up to 100 years from where the observed survival
+#' curve ends. \strong{NOTE:} the breaks on the survival time scale
+#' MUST include the breaks supplied to argument \code{breaks}; see Examples.
+#' If \code{NULL}, uses decent defaults (maximum follow-up time of 50 years).
+#' @param e1.pophaz Same as \code{pophaz}, except this defines the 
+#' population hazard in the time window where \strong{expected} 
+#' survival is estimated. By default uses the same data as 
+#' argument \code{pophaz}.
+#' @param r either a numeric multiplier such as \code{0.995}, \code{"auto"}, or
+#' \code{"autoX"} where \code{X} is an integer;
+#' used to determine the relative survival ratio (RSR) persisting after where 
+#' the estimated observed survival curve ends. See Details.
+#' @param surv.method passed to \code{survtab}; see that help for more info.
+#' @param subset a logical condition; e.g. \code{subset = sex == 1}; 
+#' subsets the data before computations
+#' @param verbose \code{logical}; if \code{TRUE}, the function is returns
+#' some messages and results along the run, which may be useful in debugging
+#' @details
+#' \strong{Basics}
+#' 
+#' \code{survmean} computes mean survival times. For median survival times
+#' (i.e. where 50 % of subjects have died or met some other event)
+#' use \code{\link{survtab}}.
+#' 
+#' The mean survival time is simply the area under the survival curve.
+#' However, since full follow-up rarely happens, the observed survival curves
+#' are extrapolated using expected survival: E.g. one might compute observed
+#' survival till up to 10 years and extrapolate beyond that 
+#' (till e.g. 50 years) to yield an educated guess on the full observed survival
+#' curve. 
+#' 
+#' The area is computed by trapezoidal integration of the area under the curve.
+#' This function also computes the "full" expected survival curve from
+#' T = 0 till e.g. T = 50 depending on supplied arguments. The
+#' expected mean survival time is the area under the 
+#' mean expected survival curve.
+#' This function returns the mean expected survival time to be compared with 
+#' the mean survival time and for computing years of potential life lost (YPLL).
+#' 
+#' Results can be formed by strata and adjusted for e.g. age by using
+#' the \code{formula} argument as in \code{survtab}. See also Examples.
+#' 
+#' \strong{Extrapolation tweaks}
+#' 
+#' Argument \code{r} controls the relative survival ratio (RSR) assumed to
+#' persist beyond the time window where observed survival is computed
+#' (defined by argument \code{breaks}; e.g. up to \code{FUT = 10}).
+#' The RSR is simply \code{RSR_i = p_oi / p_ei} for a time interval \code{i}, 
+#' i.e. the observed divided by the expected 
+#' (conditional, not cumulative) probability of surviving from the beginning of
+#' a time interval till its end. The cumulative product of \code{RSR_i}
+#' over time is the (cumulative) relative survival curve. 
+#' 
+#'
+#' If \code{r} is numeric, e.g. \code{r = 0.995}, that RSR level is assumed
+#' to persist beyond the observed survival curve. 
+#' Numeric \code{r} should be \code{> 0} and expressed at the annual level
+#' when using fractional years as the scale of the time variables.
+#' E.g. if RSR is known to be \code{0.95} at the month level, then the
+#' annualized RSR is \code{0.95^12}. This enables correct usage of the RSR
+#' with survival intervals of varying lengths. When using day-level time 
+#' variables (such as \code{Dates}; see \code{as.Date}), numeric \code{r}
+#' should be expressed at the day level, etc.
+#' 
+#' If \code{r = "auto"} or \code{r = "auto1"}, this function computes
+#' RSR estimates internally and automatically uses the \code{RSR_i}
+#' in the last survival interval in each stratum (and adjusting group)
+#' and assumes that to persist beyond the observed survival curve.
+#' Automatic determination of \code{r} is a good starting point,
+#' but in situations where the RSR estimate is uncertain it may produce poor
+#' results. Using \code{"autoX"} such as \code{"auto6"} causes \code{survmean}
+#' to use the mean of the estimated RSRs in the last X survival intervals, 
+#' which may be more stable.
+#' Automatic determination will not use values \code{>1} but set them to 1. 
+#' Visual inspection of the produced curves is always recommended: see
+#' Examples.
+#' 
+#' One may also tweak the accuracy and length of extrapolation and 
+#' expected survival curve computation by using 
+#' \code{e1.breaks}. By default this is whatever was supplied to \code{breaks}
+#' for the survival time scale, to which
+#' 
+#' \code{c(seq(1/12, 1, 1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))}
+#' 
+#' is added after the maximum value, e.g. with \code{breaks = list(FUT = 0:10)}
+#' we have 
+#' 
+#' \code{..., 10+1/12, ..., 11, 11.2, ..., 2, 3, ..., 19, 20, 25, ... 50}
+#' 
+#' as the \code{e1.breaks}. Supplying \code{e1.breaks} manually requires
+#' the breaks over time survival time scale supplied to argument \code{breaks}
+#' to be reiterated in \code{e1.breaks}; see Examples. \strong{NOTE}: the
+#' default extrapolation breaks assume the time scales in the data to be 
+#' expressed as fractional years, meaning this will work extremely poorly
+#' when using e.g. day-level time scales (such as \code{Date} variables). 
+#' Set the extrapolation breaks manually in such cases.
+#' 
+#' @return 
+#' Returns a \code{data.frame} or \code{data.table} (depending on 
+#' \code{getOptions("popEpi.datatable")}; see \code{?popEpi}) containing the
+#' following columns:
+#' \itemize{
+#'   \item{est}{: The estimated mean survival time}
+#'   \item{exp}{: The computed expected survival time}
+#'   \item{obs}{: Counts of subjects in data}
+#'   \item{YPLL}{: Years of Potential Life Lost, computed as 
+#'   (\code{(exp-est)*obs}) - though your time data may be in e.g. days,
+#'   this column will have the same name regardless.}
+#' }
+#' The returned data also has columns named according to the variables
+#' supplied to the right-hand-side of the formula.
+#' 
+#' 
+#' @examples
+#' 
+#' library(Epi)
+#' ## take 500 subjects randomly for demonstration
+#' data(sire)
+#' sire <- sire[sire$dg_date < sire$ex_date, ]
+#' set.seed(1L)
+#' sire <- sire[sample(x = nrow(sire), size = 500),]
+#' 
+#' ## NOTE: recommended to use factor status variable
+#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+#'            exit = list(CAL = get.yrs(ex_date)),
+#'            data = sire,
+#'            exit.status = factor(status, levels = 0:2,
+#'                                 labels = c("alive", "canD", "othD")),
+#'            merge = TRUE)
+#' 
+#' ## phony variable
+#' set.seed(1L)
+#' x$group <- rbinom(nrow(x), 1, 0.5)
+#' ## age group
+#' x$agegr <- cut(x$dg_age, c(0,45,60,Inf), right=FALSE)
+#' 
+#' ## population hazards data  set
+#' pm <- data.frame(popEpi::popmort)
+#' names(pm) <- c("sex", "CAL", "AGE", "haz")
+#' 
+#' ## breaks to define observed survival estimation
+#' BL <- list(FUT = seq(0, 10, 1/12))
+#' 
+#' ## crude mean survival
+#' sm1 <- survmean(Surv(FUT, lex.Xst != "alive") ~ 1,
+#'                 pophaz = pm, data = x, weights = NULL,
+#'                 breaks = BL)
+#'                 
+#' sm1 <- survmean(FUT ~ 1,
+#'                 pophaz = pm, data = x, weights = NULL,
+#'                 breaks = BL)             
+#' \donttest{
+#' ## mean survival by group                 
+#' sm2 <- survmean(FUT ~ group,
+#'                 pophaz = pm, data = x, weights = NULL,
+#'                 breaks = BL)
+#'                 
+#' ## ... and adjusted for age using internal weights (counts of subjects)      
+#' ## note: need also longer extrapolation here so that all curves
+#' ## converge to zero in the end.
+#' eBL <- list(FUT = c(BL$FUT, 11:75))
+#' sm3 <- survmean(FUT ~ group + adjust(agegr),
+#'                 pophaz = pm, data = x, weights = "internal",
+#'                 breaks = BL, e1.breaks = eBL)
+#' }
+
+#' ## visual inspection of how realistic extrapolation is for each stratum;
+#' ## solid lines are observed + extrapolated survivals;
+#' ## dashed lines are expected survivals
+#' plot(sm1)
+#' \donttest{
+#' ## plotting object with both stratification and standardization
+#' ## plots curves for each strata-std.group combination
+#' plot(sm3)
+#' 
+#' ## for finer control of plotting these curves, you may extract
+#' ## from the survmean object using e.g.
+#' attributes(sm3)$survmean.meta$curves
+#' 
+#' 
+#' #### using Dates
+#' 
+#' x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+#'            exit = list(CAL = ex_date),
+#'            data = sire[sire$dg_date < sire$ex_date, ],
+#'            exit.status = factor(status, levels = 0:2, 
+#'                                 labels = c("alive", "canD", "othD")), 
+#'            merge = TRUE)
+#' ## phony group variable
+#' set.seed(1L)
+#' x$group <- rbinom(nrow(x), 1, 0.5)
+#' 
+#'                   
+#' ## NOTE: population hazard should be reported at the same scale
+#' ## as time variables in your Lexis data.
+#' data(popmort, package = "popEpi")
+#' pm <- data.frame(popmort)
+#' names(pm) <- c("sex", "CAL", "AGE", "haz")
+#' ## from year to day level
+#' pm$haz <- pm$haz/365.25 
+#' pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
+#' pm$AGE <- pm$AGE*365.25 
+#' 
+#' BL <- list(FUT = seq(0, 8, 1/12)*365.25)
+#' eBL <- list(FUT = c(BL$FUT, c(8.25,8.5,9:60)*365.25))
+#' smd <- survmean(FUT ~ group, data = x, 
+#'                 pophaz = pm, verbose = TRUE, r = "auto5",
+#'                 breaks = BL, e1.breaks = eBL)     
+#' plot(smd)
+#' }
+#' 
+
+#' 
+#' @export
+#' @family survmean functions
+#' @family main functions
+#' 
+
+survmean <- function(formula, data, adjust = NULL, weights = NULL, 
+                     breaks=NULL, pophaz = NULL, 
+                     e1.breaks = NULL, e1.pophaz = pophaz, r = "auto", 
+                     surv.method = "hazard", subset = NULL, verbose = FALSE) {
+  pt <- proc.time()
+  TF__ <- environment()
+  PF__ <- parent.frame(1L)
+  
+  attr_form <- copy(formula)
+  
+  surv.method <- match.arg(surv.method, c("hazard", "lifetable"))
+  
+  ## appease R CMD CHECK (due to using vars in DT[] only)
+  r.e2 <- last.p.e2 <- surv <- survmean_type <- est <- Tstart <- Tstop <- 
+    lex.id <- surv.int <- delta <- surv.exp <- obs <- NULL
+  
+  checkLexisData(data, check.breaks = FALSE)
+  checkPophaz(data, pophaz, haz.name = "haz")
+  checkPophaz(data, e1.pophaz, haz.name = "haz")
+  pophaz <- setDT(copy(pophaz))
+  e1.pophaz <- setDT(copy(e1.pophaz))
+  
+  if (is.numeric(r) && r < 0L) stop("numeric r must be > 0, e.g. r = 0.95")
+  if (is.character(r)) {
+    if (substr(r, 1, 4) != "auto") {
+      stop("character string r must start with 'auto'; e.g. `auto` and ",
+           "`auto5` are accepted.")
+    }
+    if (r == "auto") r <- "auto1"
+    
+    auto_ints <- regmatches(r, regexec("\\d+", text = r))
+    auto_ints <- as.integer(auto_ints)
+    r <- "auto"
+  }
+  
+  tscales_all <- attr(data, "time.scales")
+  breaks_old <- attr(data, "breaks")
+  
+  
+  
+  ## breaks --------------------------------------------------------------------
+  
+  if (!is.null(breaks_old)) checkBreaksList(data, breaks_old)
+  if (is.null(breaks)) breaks <- breaks_old
+  
+  checkBreaksList(data, breaks)
+  
+  ## hmm - will later on set breaks on the found survival scale
+  if (!is.null(e1.breaks))  checkBreaksList(data, e1.breaks)
+  
+  ## prep & subset data --------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset)
+  
+  x <- setDT(data[subset, ])
+  forceLexisDT(x, breaks = breaks_old, allScales = tscales_all)
+  
+  ## ensure variables to merge pophaz datas by are kept ------------------------
+  ## NOTE: temp var names avoid conflicts down the line
+  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
+  
+  pophaz_vars <- c(names(pophaz), names(e1.pophaz))
+  pophaz_vars <- setdiff(pophaz_vars, c(tscales_all, "haz"))
+  pophaz_vars <- intersect(pophaz_vars, names(x))
+  pophaz_vars_tmp <- makeTempVarName(names = avoid, pre = pophaz_vars)
+  if (!length(pophaz_vars)) {
+    pophaz_vars_tmp <- NULL
+  } else {
+    pophaz_vars_wh <- which(pophaz_vars %in% names(pophaz))
+    if (sum(pophaz_vars_wh)) {
+      setnames(pophaz, old = pophaz_vars[pophaz_vars_wh], 
+               new = pophaz_vars_tmp[pophaz_vars_wh])
+    }
+    pophaz_vars_wh <- which(pophaz_vars %in% names(e1.pophaz))
+    if (sum(pophaz_vars_wh)) {
+      setnames(e1.pophaz, old = pophaz_vars[pophaz_vars_wh], 
+               new = pophaz_vars_tmp[pophaz_vars_wh])
+    }
+    x[, (pophaz_vars_tmp) := copy(.SD), .SDcols = pophaz_vars]
+  }
+  
+  ## determine printing & adjusting vars ---------------------------------------
+  adSub <- substitute(adjust)
+  foList <- usePopFormula(formula, adjust = adSub, data = x, enclos = PF__, 
+                          Surv.response = "either")
+  
+  ## will avoid conflicts using temp names for tabulating variables
+  adjust_vars <- names(foList$adjust)
+  print_vars <- names(foList$print)
+  by_vars <- c(print_vars, adjust_vars)
+  
+  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
+  adjust_vars_tmp <- makeTempVarName(names = avoid, pre = adjust_vars)
+  if (!length(adjust_vars)) adjust_vars_tmp <- NULL
+  avoid <- unique(c(names(data), names(x), names(pophaz), names(e1.pophaz)))
+  print_vars_tmp <- makeTempVarName(names = avoid, pre = print_vars)
+  if (!length(print_vars)) print_vars_tmp <- NULL
+  by_vars_tmp  <- c(print_vars_tmp, adjust_vars_tmp)
+  
+  
+  lex_vars <- c("lex.id", tscales_all, "lex.dur", "lex.Cst", "lex.Xst")
+  setcolsnull(x, keep = c(lex_vars, pophaz_vars_tmp), soft = FALSE)
+  if (length(adjust_vars) > 0L) x[, (adjust_vars_tmp) := foList$adjust]
+  if (length(print_vars) > 0L) x[, (print_vars_tmp) := foList$print]
+  
+  ## formula for survtab: we estimate survivals by all levels of both
+  ## print and adjust; adjusting here means computing directly adjusted
+  ## estimates of the mean survival time, so mean survival times are
+  ## weighted later on.
+  
+  formula <- paste0(deparse(formula[[2L]]), " ~ ")
+  if (length(c(adjust_vars_tmp, print_vars_tmp)) > 0L) {
+    formula <- paste0(formula, paste0(c(print_vars_tmp, adjust_vars_tmp), 
+                                      collapse = " + "))
+  } else {
+    formula <- paste0(formula, "1")
+  }
+  formula <- as.formula(formula)
+  
+  ## detect survival time scale ------------------------------------------------
+  tscale_surv <- detectSurvivalTimeScale(lex = x, values = foList$y$time)
+  
+  ## check weights & adjust ----------------------------------------------------
+  test_obs <- x[, .(obs=.N),  keyby=eval(TF__$by_vars_tmp)]
+  if (length(by_vars)) setnames(test_obs, by_vars_tmp, by_vars)
+  if (length(weights) && !length(adjust_vars)) {
+    weights <- NULL
+    warning("Replaced weights with NULL due to not supplying variables to ",
+            "adjust by.")
+  }
+  mwDTtest <- makeWeightsDT(test_obs, values = list("obs"), print = print_vars,
+                            adjust = adjust_vars, weights = weights, 
+                            internal.weights.values = "obs")
+  if (length(by_vars)) setnames(test_obs, by_vars, by_vars_tmp)
+  
+  ## figure out extrapolation breaks -------------------------------------------
+  ## now that the survival time scale is known this can actually be done.
+  
+  if (is.null(e1.breaks)) {
+    e1.breaks <- copy(breaks[tscale_surv])
+    addBreaks <- max(e1.breaks[[tscale_surv]]) + 
+      c(seq(0,1,1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))
+    e1.breaks[[tscale_surv]] <- unique(c(e1.breaks[[tscale_surv]], addBreaks))
+    
+    checkBreaksList(x, e1.breaks)
+  }
+  if (!tscale_surv %in% names(e1.breaks)) {
+    stop("The survival time scale must be included in the list of breaks ",
+         "to extrapolate by ('e1.breaks').")
+  }
+  if (!all(breaks[[tscale_surv]] %in% e1.breaks[[tscale_surv]])) {
+    stop("The vector of breaks in 'breaks' for the survival time scale MUST",
+         "be a subset of the breaks for the survival time scale in ",
+         "'e1.breaks'. E.g. the former could be 0:10 and the latter 0:100.")
+  }
+  
+  if (verbose) {
+    cat("Time taken by prepping data:", timetaken(pt), "\n")
+  }
+  
+  
+  ## compute observed survivals ------------------------------------------------
+  ## NOTE: do not adjust here; adjust in original formula means weighting
+  ## the mean survival time results.
+  
+  st <- survtab(formula, data = x, breaks = breaks, 
+                pophaz = pophaz,
+                relsurv.method = "e2",
+                surv.type = "surv.rel", 
+                surv.method = surv.method)
+  
+  st_keep_vars <- c(by_vars_tmp, "Tstop", "r.e2", "surv.obs")
+  all_names_present(
+    st,  st_keep_vars, 
+    msg = paste0("Internal error: expected to have variables ",
+                 "%%VARS%% after computing observed survivals ",
+                 "but didn't. Blame the package maintainer if you ",
+                 "see this.")
+  )
+  setcolsnull(st, keep = st_keep_vars, colorder = TRUE)
+  setDT(st)
+  setkeyv(st, c(by_vars_tmp, "Tstop"))
+  st[, "Tstart" := c(0, Tstop[-.N]), by = eval(by_vars_tmp)]
+  
+  ## decumulate for later cumulation
+  st[, c("r.e2", "surv.obs") := lapply(.SD, function(col) col/c(1, col[-.N])), 
+     by = eval(by_vars_tmp),
+     .SDcols = c("r.e2", "surv.obs")
+     ]
+  
+  
+  if (verbose) {
+    cat("Time taken by estimating relative survival curves:", 
+        timetaken(pt), "\n")
+  }
+  
+  ## compute overall expected survival -----------------------------------------
+  ## 1) take only those individuals that were diagnosed in the time window
+  ##    defined by breaks list in argument 'breaks'
+  pt <- proc.time()
+  setkeyv(x, c("lex.id", tscale_surv))
+  tol <- .Machine$double.eps^0.5
+  xe <- unique(x, by = key(x))[x[[tscale_surv]] < TF__$tol, ] ## pick rows with entry to FU
+  
+  if (length(breaks) > 1L) {
+    ## e.g. a period window was defined and we only use subjects
+    ## entering follow-up in the time window.
+    breaks_drop_tmp <- setdiff(names(breaks), tscale_surv)
+    breaks_drop_tmp <- breaks[breaks_drop_tmp]
+    breaks_drop_tmp <- lapply(breaks_drop_tmp, range)
+    
+    expr <- mapply(function(ch, ra) {
+      paste0("between(", ch, ", ", ra[1], ", ", ra[2] - tol, ", incbounds = TRUE)")
+    }, ch = names(breaks_drop_tmp), ra = breaks_drop_tmp, SIMPLIFY = FALSE)
+    
+    expr <- lapply(expr, function(e) eval(parse(text = e), envir = xe))
+    setDT(expr)
+    expr <- expr[, rowSums(.SD)]  == ncol(expr)
+    xe <- xe[expr, ]
+  }
+  
+  xe <- x[lex.id %in% unique(xe[["lex.id"]])]
+  forceLexisDT(xe, breaks = breaks_old, allScales = tscales_all, key = FALSE)
+  
+  ## 2) compute Ederer I expected survival curves from T = 0 till e.g. T = 100
+  e1 <- comp_e1(xe, breaks = e1.breaks, pophaz = e1.pophaz, immortal = TRUE, 
+                survScale = tscale_surv, by = by_vars_tmp, id = "lex.id")
+  setnames(e1, tscale_surv, "Tstop")
+  e1[, "Tstart" := c(0, Tstop[-.N]), by = eval(by_vars_tmp)]
+  e1[, "surv.int" := cut(Tstart, breaks = e1.breaks[[tscale_surv]], 
+                         right = FALSE, labels = FALSE)]
+  e1[, "delta" := Tstop - Tstart]
+  
+  ## decumulate for later cumulation
+  e1[, "surv.exp" := surv.exp/c(1, surv.exp[-.N]), by = eval(by_vars_tmp)]
+  
+  if (verbose) {
+    cat("Time taken by computing overall expected survival curves:", 
+        timetaken(pt), "\n")
+  }
+  
+  ## compute counts of subjects ------------------------------------------------
+  ## these correspond to the counts of patients for which expected survival
+  ## was computed. If observed survival is e.g. a period estimated curve,
+  ## we only use subjects entering follow-up in the period window.
+  N_subjects <- xe[!duplicated(lex.id)][, 
+                                        list(obs=.N), 
+                                        keyby=eval(by_vars_tmp)
+                                        ]
+  
+  ## combine all estimates into one data set -----------------------------------
+  pt <- proc.time()
+  
+  st[, "surv.int" := cut(Tstart, breaks = e1.breaks[[tscale_surv]], 
+                         right = FALSE, labels = FALSE)]
+  
+  x <- merge(e1, st[, .SD, .SDcols = c(by_vars_tmp, "surv.int", "r.e2", "surv.obs")], 
+             by = c(by_vars_tmp,"surv.int"), all = TRUE)
+  setkeyv(x, c(by_vars_tmp, "surv.int"))
+  
+  ## extrapolation RSR definition ----------------------------------------------
+  if (is.numeric(r)) {
+    ## manually given RSR for extrapolated part of the obs.surv curve
+    ## here it is assumed that r is annualized
+    set(x, j = "last.p.e2", value = r^x[["delta"]]) 
+    
+    
+  } else {
+    ## add last non-NA values as separate column
+    
+    st <- st[, .SD[(.N-TF__$auto_ints+1):.N], by = eval(by_vars_tmp)]
+    
+    st[, "delta" := Tstop - Tstart]
+    st[, "r.e2" := r.e2^(1/delta)] ## "annualized" RSRs
+    
+    ## mean annualized RSR in last N intervas by strata
+    st <- st[, .(last.p.e2 = mean(r.e2)), by = eval(by_vars_tmp)]
+    st[, "last.p.e2" := pmin(1, last.p.e2)]
+    if (verbose) {
+      cat("Using following table of mean RSR estimates",
+          "(scaled to RSRs applicable to a time interval one",
+          "unit of time wide, e.g. one year or one day)",
+          "based on", auto_ints, "interval(s) from the end of the relative",
+          "survival curve by strata: \n")
+      prST <- data.table(st)
+      setnames(prST, c(by_vars_tmp, "last.p.e2"), c(by_vars, "RSR"))
+      print(prST)
+    }
+    
+    if (length(by_vars_tmp)) {
+      x <- merge(x, st, by = by_vars_tmp, all = TRUE)
+    } else {
+      set(x, j = "last.p.e2", value = st$last.p.e2)
+    }
+    x[, "last.p.e2" := last.p.e2^(delta)] ## back to non-annualized RSRs
+    ## enforce RSR in extrapolated part of observed curve to at most 1
+    x[, "last.p.e2" := pmin(last.p.e2, 1)]
+  }
+  
+  x[is.na(r.e2), "r.e2" := last.p.e2]
+  x[, "surv" := r.e2*surv.exp]
+  # setnames(x, "surv.obs", "surv")
+  # x[is.na(surv), "surv" := surv.exp*last.p.e2]
+  
+  ## cumulate again
+  setkeyv(x, c(by_vars_tmp, "surv.int"))
+  x[, c("surv", "surv.exp") := lapply(.SD, cumprod),
+    .SDcols = c("surv", "surv.exp"), by = eval(by_vars_tmp)]
+  
+  x2 <- copy(x)
+  x[, "surv.exp" := NULL]
+  x2[, "surv" := NULL]
+  setnames(x2, "surv.exp", "surv")
+  x <- rbind(x, x2)
+  x[, "survmean_type" := rep(c("est", "exp"), each = nrow(x2))]
+  
+  setcolsnull(
+    x, 
+    keep = c(by_vars_tmp, "survmean_type", 
+             "surv.int", "Tstart", "Tstop", 
+             "delta", "surv", "surv.exp"),
+    colorder = TRUE
+  )
+  
+  ## check curve convergence to zero -------------------------------------------
+  ## a good integration is based on curves that get very close to 
+  ## zero in the end
+  mi <- x[, .(surv = round(min(surv),4)*100), 
+          keyby = eval(c(by_vars_tmp, "survmean_type"))]
+  
+  if (any(mi$surv > 1)) {
+    warning("One or several of the curves used to compute mean survival times ",
+            "or expected mean survival times was > 1 % at the lowest point. ",
+            "Mean survival estimates may be significantly biased. To avoid ",
+            "this, supply breaks to 'e1.breaks' which make the curves longer ",
+            ", e.g. e1.breaks = list(FUT = 0:150) where time scale FUT ",
+            "is the survival time scale (yours may have a different name).")
+  }
+  mi[, "surv" := paste0(formatC(surv, digits = 2, format = "f"), " %")]
+  mi[, "survmean_type" := factor(survmean_type, c("est", "exp"),
+                                 c("Observed", "Expected"))]
+  setnames(mi, c("survmean_type", "surv"), 
+           c("Obs./Exp. curve", "Lowest value"))
+  if (length(by_vars)) setnames(mi, by_vars_tmp, by_vars)
+  if (verbose) {
+    cat("Lowest points in observed / expected survival curves by strata:\n")
+    print(mi)
+  }
+  
+  ## integrating by trapezoid areas --------------------------------------------
+  ## trapezoid area: WIDTH*(HEIGHT1 + HEIGHT2)/2
+  ## so we compute "average interval survivals" for each interval t_i
+  ## and multiply with interval length.
+  
+  setkeyv(x, c(by_vars_tmp, "survmean_type",  "Tstop"))
+  sm <- x[, .(survmean = sum(delta*(surv + c(1, surv[-.N]))/2L)), 
+          keyby = c(by_vars_tmp, "survmean_type")]
+  
+  ## cast ----------------------------------------------------------------------
+  
+  sm <- cast_simple(sm, columns = "survmean_type", 
+                    rows = by_vars_tmp, values = "survmean")
+  
+  ## add numbers of subjects, compute YPLL -------------------------------------
+  setkeyv(sm, by_vars_tmp); setkeyv(N_subjects, by_vars_tmp)
+  sm[, "obs" := N_subjects$obs]
+  sm[, "YPLL" := (exp-est)*obs]
+  
+  
+  ## adjusting -----------------------------------------------------------------
+  
+  sm <- makeWeightsDT(sm, values = list(c("est", "exp", "obs", "YPLL")),
+                      print = print_vars_tmp, adjust = adjust_vars_tmp,
+                      weights = weights, internal.weights.values = "obs")
+  if (length(adjust_vars)) {
+    vv <- c("est", "exp", "obs", "YPLL")
+    sm[, c("est", "exp") := lapply(.SD, function(col) col*sm$weights), 
+       .SDcols = c("est", "exp")]
+    sm <- sm[, lapply(.SD, sum), .SDcols = vv, by = eval(print_vars_tmp)]
+  }
+  
+  if (verbose) {
+    cat("Time taken by final touches:", timetaken(pt), "\n")
+  }
+  
+  ## final touch ---------------------------------------------------------------
+  if (length(print_vars)) setnames(sm, print_vars_tmp, print_vars)
+  
+  at <- list(call = match.call(), 
+             formula = attr_form,
+             print = print_vars, 
+             adjust = adjust_vars, 
+             tprint = print_vars_tmp, 
+             tadjust = adjust_vars_tmp,
+             breaks = breaks, 
+             e1.breaks = e1.breaks, 
+             survScale = tscale_surv,
+             curves = copy(x))
+  setattr(sm, "class", c("survmean","data.table", "data.frame"))
+  setattr(sm, "survmean.meta", at)
+  if (!return_DT()) setDFpe(sm)
+  return(sm[])
+}
+
diff --git a/R/popEpi_package.r b/R/popEpi_package.r
index 545154f..366771a 100644
--- a/R/popEpi_package.r
+++ b/R/popEpi_package.r
@@ -1,35 +1,35 @@
-#' popEpi
-#'
-#' @name popEpi
-#' @docType package
-#' @title popEpi: Functions for large-scale epidemiological analysis
-#' @description 
-#' \pkg{popEpi} is built for the needs of registry-based (large-scale)
-#' epidemiological analysis. This is in most part enabled by the 
-#' efficient \pkg{data.table} package for handling and aggregating large data sets. 
-#' 
-#' \pkg{popEpi} currently supplies some utility functions such as \code{\link{splitMulti}}
-#' and \code{\link{get.yrs}} for preparing large data sets for epidemiological analysis.
-#' Included are also a a few functions that can be used in 
-#' epidemiological analysis such as \code{\link{sir}} for estimating
-#' standardized incidence/mortality ratios (SIRs/SMRs) and \code{\link{survtab}} for 
-#' estimating observed and relative/net survival as well as cumulative incidence
-#' functions (CIFs). In particular, \code{survtab} implements the Ederer II 
-#' (Ederer and Heise (1959)) and
-#' Pohar Perme estimators (Pohar Perme, Stare, and Esteve (2012) 
-#' \doi{10.1111/j.1541-0420.2011.01640.x}) and allows for easy 
-#' age-standardisation.
-#' 
-#' Since there are many benefits to using \code{data.tables}, \pkg{popEpi} returns
-#' outputs by default in the \code{data.table} format where appropriate. 
-#' Since \code{data.table}
-#' objects are usually modified by reference, this may have surprising side 
-#' effects for users uninitiated in using \code{data.table}. To ensure
-#' that appropriate outputs are in the \code{data.frame} format, set
-#' \code{options("popEpi.datatable" = FALSE)}. However, \code{data.table}
-#' usage is recommended due to better performance and testing coverage. 
-#' \code{data.table} is used
-#' by most functions internally in both cases.
-#' 
-NULL
-
+#' popEpi
+#'
+#' @name popEpi
+#' @docType package
+#' @title popEpi: Functions for large-scale epidemiological analysis
+#' @description 
+#' \pkg{popEpi} is built for the needs of registry-based (large-scale)
+#' epidemiological analysis. This is in most part enabled by the 
+#' efficient \pkg{data.table} package for handling and aggregating large data sets. 
+#' 
+#' \pkg{popEpi} currently supplies some utility functions such as \code{\link{splitMulti}}
+#' and \code{\link{get.yrs}} for preparing large data sets for epidemiological analysis.
+#' Included are also a a few functions that can be used in 
+#' epidemiological analysis such as \code{\link{sir}} for estimating
+#' standardized incidence/mortality ratios (SIRs/SMRs) and \code{\link{survtab}} for 
+#' estimating observed and relative/net survival as well as cumulative incidence
+#' functions (CIFs). In particular, \code{survtab} implements the Ederer II 
+#' (Ederer and Heise (1959)) and
+#' Pohar Perme estimators (Pohar Perme, Stare, and Esteve (2012) 
+#' \doi{10.1111/j.1541-0420.2011.01640.x}) and allows for easy 
+#' age-standardisation.
+#' 
+#' Since there are many benefits to using \code{data.tables}, \pkg{popEpi} returns
+#' outputs by default in the \code{data.table} format where appropriate. 
+#' Since \code{data.table}
+#' objects are usually modified by reference, this may have surprising side 
+#' effects for users uninitiated in using \code{data.table}. To ensure
+#' that appropriate outputs are in the \code{data.frame} format, set
+#' \code{options("popEpi.datatable" = FALSE)}. However, \code{data.table}
+#' usage is recommended due to better performance and testing coverage. 
+#' \code{data.table} is used
+#' by most functions internally in both cases.
+#' 
+NULL
+
diff --git a/R/pophaz.R b/R/pophaz.R
index b5a163d..485939c 100644
--- a/R/pophaz.R
+++ b/R/pophaz.R
@@ -1,70 +1,70 @@
-
-
-
-#' @title Expected / Population Hazard Data Sets Usage in \pkg{popEpi}
-#' @author Joonas Miettinen
-#' @name pophaz
-#' @description 
-#' 
-#' Several functions in \pkg{popEpi} make use of population or expected
-#' hazards in computing the intended estimates (e.g. \code{\link{survtab}}).
-#' This document explains using such data sets in this package.
-#' 
-#' @details 
-#' 
-#' Population hazard data sets (pophaz for short) in \pkg{popEpi} should
-#' be \code{data.frame}s in the "long" format where one of the columns must be
-#' named \code{haz} (for hazard), and other columns define the values or 
-#' levels in variables relating to subjects in your data. For example,
-#' \code{\link{popmort}} contains Finnish population mortality hazards
-#' by sex, calendar year, and 1-year age group.
-#' 
-#' \tabular{rrrr}{
-#' \code{sex} \tab \code{year} \tab \code{agegroup} \tab \code{haz} \cr
-#' 0 \tab 1951 \tab 0 \tab 0.036363176\cr  
-#' 0 \tab 1951 \tab 1 \tab 0.003616547\cr
-#' 0 \tab 1951 \tab 2 \tab 0.002172384\cr
-#' 0 \tab 1951 \tab 3 \tab 0.001581249\cr
-#' 0 \tab 1951 \tab 4 \tab 0.001180690\cr
-#' 0 \tab 1951 \tab 5 \tab 0.001070595
-#' }
-#' 
-#' The names of the columns should match to the names of the variables
-#' that you have in your subject-level data. Time variables in your pophaz
-#' may also correspond to \code{Lexis} time scales; see 
-#' \code{\link{survtab}}.
-#' 
-#' Any time variables (as they usually have) should be coded consistently:
-#' When using fractional years in your data, the time variables in your pophaz
-#' must also be coded in fractional years. When using e.g. \code{Date}s in your
-#' data, ensure that the pophaz time variables are coded at the level of days
-#' (or \code{Date}s for calendar time). 
-#' 
-#' The \code{haz} variable in your pophaz should also be coded consistently
-#' with the used time variables. E.g. \code{haz} values in life-tables
-#' reported as deaths per person-year should be multiplied by 365.25 when
-#' using day-level time variables. Typically you'll have calendar time and age
-#' expressed in years, which means \code{haz} should be expressed as the number
-#' of deaths per person-year.
-#' 
-#' If you have your population hazards in a \code{ratetable} object
-#' usable by functions in \pkg{survival} and \pkg{relsurv}, you may
-#' transform them to long-format \code{data.frame}s using
-#' \code{\link{ratetable_to_long_dt}}. Ensure, however, that the
-#' created \code{haz} column is coded at the right level (events per
-#' days or years typically).
-#' 
-#' National statistical institutions, the WHO, and e.g. the Human
-#' Life-Table Database supply life-table data.
-#' 
-
-NULL
-
-
-
-
-
-
-
-
-
+
+
+
+#' @title Expected / Population Hazard Data Sets Usage in \pkg{popEpi}
+#' @author Joonas Miettinen
+#' @name pophaz
+#' @description 
+#' 
+#' Several functions in \pkg{popEpi} make use of population or expected
+#' hazards in computing the intended estimates (e.g. \code{\link{survtab}}).
+#' This document explains using such data sets in this package.
+#' 
+#' @details 
+#' 
+#' Population hazard data sets (pophaz for short) in \pkg{popEpi} should
+#' be \code{data.frame}s in the "long" format where one of the columns must be
+#' named \code{haz} (for hazard), and other columns define the values or 
+#' levels in variables relating to subjects in your data. For example,
+#' \code{\link{popmort}} contains Finnish population mortality hazards
+#' by sex, calendar year, and 1-year age group.
+#' 
+#' \tabular{rrrr}{
+#' \code{sex} \tab \code{year} \tab \code{agegroup} \tab \code{haz} \cr
+#' 0 \tab 1951 \tab 0 \tab 0.036363176\cr  
+#' 0 \tab 1951 \tab 1 \tab 0.003616547\cr
+#' 0 \tab 1951 \tab 2 \tab 0.002172384\cr
+#' 0 \tab 1951 \tab 3 \tab 0.001581249\cr
+#' 0 \tab 1951 \tab 4 \tab 0.001180690\cr
+#' 0 \tab 1951 \tab 5 \tab 0.001070595
+#' }
+#' 
+#' The names of the columns should match to the names of the variables
+#' that you have in your subject-level data. Time variables in your pophaz
+#' may also correspond to \code{Lexis} time scales; see 
+#' \code{\link{survtab}}.
+#' 
+#' Any time variables (as they usually have) should be coded consistently:
+#' When using fractional years in your data, the time variables in your pophaz
+#' must also be coded in fractional years. When using e.g. \code{Date}s in your
+#' data, ensure that the pophaz time variables are coded at the level of days
+#' (or \code{Date}s for calendar time). 
+#' 
+#' The \code{haz} variable in your pophaz should also be coded consistently
+#' with the used time variables. E.g. \code{haz} values in life-tables
+#' reported as deaths per person-year should be multiplied by 365.25 when
+#' using day-level time variables. Typically you'll have calendar time and age
+#' expressed in years, which means \code{haz} should be expressed as the number
+#' of deaths per person-year.
+#' 
+#' If you have your population hazards in a \code{ratetable} object
+#' usable by functions in \pkg{survival} and \pkg{relsurv}, you may
+#' transform them to long-format \code{data.frame}s using
+#' \code{\link{ratetable_to_long_dt}}. Ensure, however, that the
+#' created \code{haz} column is coded at the right level (events per
+#' days or years typically).
+#' 
+#' National statistical institutions, the WHO, and e.g. the Human
+#' Life-Table Database supply life-table data.
+#' 
+
+NULL
+
+
+
+
+
+
+
+
+
diff --git a/R/prevalence.R b/R/prevalence.R
index 7e73b89..40a60c3 100644
--- a/R/prevalence.R
+++ b/R/prevalence.R
@@ -1,142 +1,142 @@
-
-
-
-
-prevtab <- function(
-  formula, 
-  data, 
-  meanpop = NULL, 
-  breaks = NULL, 
-  adjust = NULL, 
-  weights = NULL, 
-  subset = NULL, 
-  verbose = FALSE
-) {
-  
-  PF <- parent.frame(1L)
-  TF <- environment()
-  
-  checkLexisData(data)
-  checkPophaz(data, meanpop, haz.name = "meanpop")
-  meanpop <- data.table(meanpop)
-  
-  allScales <- attr(data, "time.scales")
-  oldBreaks <- attr(data, "breaks")
-  
-  lexis_vars <- c(allScales, "lex.dur", "lex.id", "lex.Cst", "lex.Xst")
-  lexis_vars <- intersect(names(data), lexis_vars)
-  
-  meanpop_vars <- setdiff(names(meanpop), "meanpop")
-  
-  print_vars <- print_vars_tmp <- NULL
-  adjust_vars <- adjust_vars_tmp <- NULL
-  
-  all_vars <- unique(intersect(names(data), c(lexis_vars, print_vars, adjust_vars, meanpop_vars)))
-  
-  ## appease R CMD CHECK -------------------------------------------------------
-  at.risk <- NULL
-  
-  
-  ## subsetting ----------------------------------------------------------------
-  sb <- substitute(subset)
-  subset <- evalLogicalSubset(data, substiset = sb, enclos = PF)
-  
-  ## data with only time scales and by variables -------------------------------
-  
-  data = data[subset, ]
-  by_data <- usePopFormula(formula, Surv.response = FALSE, 
-                           data = data, enclos = PF)
-  x <- setDT(mget(lexis_vars, as.environment(data)))
-  if (!is.null(by_data[["print"]])) {
-    print_vars <- names(by_data[["print"]])
-    print_vars_tmp <- makeTempVarName(names = all_vars, pre = print_vars)
-    set(x, j = print_vars_tmp, value = by_data[["print"]])
-  }
-  if (!is.null(by_data[["adjust"]])) {
-    adjust_vars <- names(by_data[["adjust"]])
-    adjust_vars_tmp <- makeTempVarName(names = all_vars, pre = adjust_vars)
-    set(x, j = adjust_vars_tmp, value = by_data[["adjust"]])
-  }
-  forceLexisDT(x, breaks = oldBreaks, allScales = allScales)
-  rm("by_data")
-  x[, c("lex.Cst", "lex.Xst") := 0L]
-  
-  by_vars <- c(print_vars, adjust_vars)
-  by_vars_tmp <- c(print_vars_tmp, adjust_vars_tmp)
-  
-  meanpop_vars_tmp <- makeTempVarName(
-    names = c(names(x), names(data), names(meanpop)), pre = meanpop_vars
-  )
-  meanpop_vars_tmp <- unlist(lapply(seq_along(meanpop_vars), function(i) {
-    if (!meanpop_vars[i] %in% by_vars) {
-      return(meanpop_vars_tmp[i])
-    }
-    wh <- which(by_vars == meanpop_vars[i])
-    by_vars_tmp[wh]
-  }))
-  
-  lapply(seq_along(meanpop_vars), function(i) {
-    set(x, j = meanpop_vars_tmp[i], value = data[[meanpop_vars[i]]])
-  })
-  
-  ## Splitting to ensure breaks exist; also takes copy -------------------------
-  x <- splitMulti(x, breaks = breaks, drop = TRUE, merge = TRUE)
-  forceLexisDT(x, breaks = attr(x, "breaks"), allScales = allScales)
-  
-  newBreaks <- copy(attr(x, "breaks"))
-  
-  
-  ## detect prevalence time scale ----------------------------------------------
-  prevScale <- detectSurvivalTimeScale(data, eval(formula[[2]], envir = data))
-  
-  
-  ## limit to prevalence time points -------------------------------------------
-  ## since we want prevalence at certain points of time along the prevalence
-  ## time scale, prevScale values not at the breaks are not considered at all.
-  j <- list(newBreaks[[prevScale]])
-  names(j) <- prevScale
-  x <- x[j, on = prevScale, nomatch = 0L]
-  
-  ## aggregate -----------------------------------------------------------------
-  aggre_vars <-  unique(c(print_vars_tmp, adjust_vars_tmp, meanpop_vars_tmp))
-  print(aggre_vars)
-  ag <- aggre(x, by = aggre_vars)
-  ag <- setDT(ag)
-  
-  setkeyv(
-    ag, c(aggre_vars, setdiff(aggre_vars, intersect(lexis_vars, print_vars_tmp)))
-  )
-  ag[, "n" := cumsum(at.risk), by = eval(aggre_vars)]
-  ag <- setDT(mget(c(aggre_vars, meanpop_vars_tmp, "n"), as.environment(ag)))
-  
-  ## compute prevalence rates if appropriate -----------------------------------
-  print(ag)
-  ag <- merge(ag, meanpop, by = meanpop_vars_tmp)
-  
-  if (length(c(print_vars_tmp, adjust_vars_tmp))) {
-    setnames(ag, c(print_vars_tmp, adjust_vars_tmp), 
-             c(print_vars, adjust_vars_tmp))
-  }
-  ag <- rate(data = ag, obs = "n", pyrs = "meanpop", print = print_vars,
-             adjust = adjust_vars, weights = weights)
-  
-  return(ag)
-}
-
-
-
-
-prevtab_ag <- function(
-  formula, 
-  data, 
-  meanpop = NULL, 
-  adjust = NULL, 
-  weights = NULL, 
-  subset = NULL, 
-  verbose = FALSE
-) {
-  ## prevtab(per ~ sex + fot)
-  
-  
-}
-
+
+
+
+
+prevtab <- function(
+  formula, 
+  data, 
+  meanpop = NULL, 
+  breaks = NULL, 
+  adjust = NULL, 
+  weights = NULL, 
+  subset = NULL, 
+  verbose = FALSE
+) {
+  
+  PF <- parent.frame(1L)
+  TF <- environment()
+  
+  checkLexisData(data)
+  checkPophaz(data, meanpop, haz.name = "meanpop")
+  meanpop <- data.table(meanpop)
+  
+  allScales <- attr(data, "time.scales")
+  oldBreaks <- attr(data, "breaks")
+  
+  lexis_vars <- c(allScales, "lex.dur", "lex.id", "lex.Cst", "lex.Xst")
+  lexis_vars <- intersect(names(data), lexis_vars)
+  
+  meanpop_vars <- setdiff(names(meanpop), "meanpop")
+  
+  print_vars <- print_vars_tmp <- NULL
+  adjust_vars <- adjust_vars_tmp <- NULL
+  
+  all_vars <- unique(intersect(names(data), c(lexis_vars, print_vars, adjust_vars, meanpop_vars)))
+  
+  ## appease R CMD CHECK -------------------------------------------------------
+  at.risk <- NULL
+  
+  
+  ## subsetting ----------------------------------------------------------------
+  sb <- substitute(subset)
+  subset <- evalLogicalSubset(data, substiset = sb, enclos = PF)
+  
+  ## data with only time scales and by variables -------------------------------
+  
+  data = data[subset, ]
+  by_data <- usePopFormula(formula, Surv.response = FALSE, 
+                           data = data, enclos = PF)
+  x <- setDT(mget(lexis_vars, as.environment(data)))
+  if (!is.null(by_data[["print"]])) {
+    print_vars <- names(by_data[["print"]])
+    print_vars_tmp <- makeTempVarName(names = all_vars, pre = print_vars)
+    set(x, j = print_vars_tmp, value = by_data[["print"]])
+  }
+  if (!is.null(by_data[["adjust"]])) {
+    adjust_vars <- names(by_data[["adjust"]])
+    adjust_vars_tmp <- makeTempVarName(names = all_vars, pre = adjust_vars)
+    set(x, j = adjust_vars_tmp, value = by_data[["adjust"]])
+  }
+  forceLexisDT(x, breaks = oldBreaks, allScales = allScales)
+  rm("by_data")
+  x[, c("lex.Cst", "lex.Xst") := 0L]
+  
+  by_vars <- c(print_vars, adjust_vars)
+  by_vars_tmp <- c(print_vars_tmp, adjust_vars_tmp)
+  
+  meanpop_vars_tmp <- makeTempVarName(
+    names = c(names(x), names(data), names(meanpop)), pre = meanpop_vars
+  )
+  meanpop_vars_tmp <- unlist(lapply(seq_along(meanpop_vars), function(i) {
+    if (!meanpop_vars[i] %in% by_vars) {
+      return(meanpop_vars_tmp[i])
+    }
+    wh <- which(by_vars == meanpop_vars[i])
+    by_vars_tmp[wh]
+  }))
+  
+  lapply(seq_along(meanpop_vars), function(i) {
+    set(x, j = meanpop_vars_tmp[i], value = data[[meanpop_vars[i]]])
+  })
+  
+  ## Splitting to ensure breaks exist; also takes copy -------------------------
+  x <- splitMulti(x, breaks = breaks, drop = TRUE, merge = TRUE)
+  forceLexisDT(x, breaks = attr(x, "breaks"), allScales = allScales)
+  
+  newBreaks <- copy(attr(x, "breaks"))
+  
+  
+  ## detect prevalence time scale ----------------------------------------------
+  prevScale <- detectSurvivalTimeScale(data, eval(formula[[2]], envir = data))
+  
+  
+  ## limit to prevalence time points -------------------------------------------
+  ## since we want prevalence at certain points of time along the prevalence
+  ## time scale, prevScale values not at the breaks are not considered at all.
+  j <- list(newBreaks[[prevScale]])
+  names(j) <- prevScale
+  x <- x[j, on = prevScale, nomatch = 0L]
+  
+  ## aggregate -----------------------------------------------------------------
+  aggre_vars <-  unique(c(print_vars_tmp, adjust_vars_tmp, meanpop_vars_tmp))
+  print(aggre_vars)
+  ag <- aggre(x, by = aggre_vars)
+  ag <- setDT(ag)
+  
+  setkeyv(
+    ag, c(aggre_vars, setdiff(aggre_vars, intersect(lexis_vars, print_vars_tmp)))
+  )
+  ag[, "n" := cumsum(at.risk), by = eval(aggre_vars)]
+  ag <- setDT(mget(c(aggre_vars, meanpop_vars_tmp, "n"), as.environment(ag)))
+  
+  ## compute prevalence rates if appropriate -----------------------------------
+  print(ag)
+  ag <- merge(ag, meanpop, by = meanpop_vars_tmp)
+  
+  if (length(c(print_vars_tmp, adjust_vars_tmp))) {
+    setnames(ag, c(print_vars_tmp, adjust_vars_tmp), 
+             c(print_vars, adjust_vars_tmp))
+  }
+  ag <- rate(data = ag, obs = "n", pyrs = "meanpop", print = print_vars,
+             adjust = adjust_vars, weights = weights)
+  
+  return(ag)
+}
+
+
+
+
+prevtab_ag <- function(
+  formula, 
+  data, 
+  meanpop = NULL, 
+  adjust = NULL, 
+  weights = NULL, 
+  subset = NULL, 
+  verbose = FALSE
+) {
+  ## prevtab(per ~ sex + fot)
+  
+  
+}
+
diff --git a/R/relative_poisson.R b/R/relative_poisson.R
index 9614dd4..39bf7c8 100644
--- a/R/relative_poisson.R
+++ b/R/relative_poisson.R
@@ -1,616 +1,616 @@
-#' @title Excess hazard Poisson model
-#' @author Joonas Miettinen, Karri Seppa
-#' @description Estimate a Poisson piecewise constant excess
-#' hazards model
-#' @param data a dataset split with e.g. \code{\link{lexpand}};
-#' must have expected hazard merged within
-#' @param formula a formula which is passed on to \code{glm}; see Details
-#' @param fot.breaks optional; a numeric vector of [a,b) breaks to specify
-#' survival intervals over the follow-up time; if \code{NULL}, the 
-#' existing breaks along the mandatory \code{fot} time scale in \code{data}
-#' are used (e.g. the breaks for \code{fot} supplied to \code{lexpand})
-#' @param subset a logical vector or condition; e.g. \code{subset = sex == 1};
-#' limits the data before estimation
-#' @param check logical; if \code{TRUE}, tabulates excess cases by all
-#' factor variables in the formula to check for negative / \code{NA} 
-#' excess cases before fitting the GLM
-#' @param ... any argument passed on to \code{glm}
-#' @import stats
-#' @details
-#' 
-#' \strong{Basics}
-#' 
-#' \code{relpois} employs a custom link function of the Poisson variety
-#' to estimate piecewise constant parametric excess hazards. The pieces
-#' are determined by \code{fot.breaks}. A \code{log(person-years)} offset
-#' is passed automatically to the \code{glm} call.
-#' 
-#' \strong{Formula usage}
-#' 
-#' The formula can be used like any ordinary \code{glm} formula. The user must
-#' define the outcome in some manner, which is usually \code{lex.Xst} after splitting
-#' with e.g. \code{lexpand}. The exception is the possibility of including 
-#' the baseline excess hazard terms by including the 
-#' reserved term \code{FOT} in the formula.
-#' 
-#' For example, \code{lex.Xst != 0 ~ FOT + agegr} estimates a model with constant
-#' excess hazards at the follow-up intervals as specified by 
-#' the pertinent breaks used in splitting \code{data},
-#' as well as for the different age groups.
-#' \code{FOT} is created ad hoc if it is used in the formula.
-#' If you leave out \code{FOT}, the hazard is effectively
-#' assumed to be constant across the whole follow-up time. 
-#' 
-#' You can also simply use your own follow-up time interval variable that
-#' you have created before calling \code{relpois}. However, when using 
-#' \code{FOT}, \code{relpois} automatically checks for e.g. 
-#' negative excess cases in follow-up intervals,
-#' allowing for quickly finding splitting breaks
-#' where model estimation is possible. It also drops any data outside the
-#' follow-up time window.
-#' 
-#' \strong{Splitting and merging population hazard}
-#' 
-#' The easiest way to both split and to include population hazard information is 
-#' by using \code{\link{lexpand}}. You may also fairly easily do it by hand
-#' by splitting first and then merging in your population hazard information.
-#' 
-#' 
-#' \strong{Data requirements}
-#' 
-#' The population hazard information must be available for each record and named
-#' \code{pop.haz}. The follow-up time variable must be named \code{"fot"} e.g.
-#' as a result of using \code{lexpand}. The \code{lex.dur} variable must also
-#' be present, containing person-year information. 
-#' 
-#' 
-#' @return
-#' A \code{glm} object created using a custom Poisson family construct. Some
-#' \code{glm} methods are applicable.
-#' 
-#' @seealso
-#' \code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
-#' @family main functions
-#' @family relpois functions
-#' @export relpois
-#' 
-#' @examples
-#' ## use the simulated rectal cancer cohort
-#' data("sire", package = "popEpi")
-#' sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
-#' 
-#' ## usable straight away after splitting
-#' fb <- c(0,3/12,6/12,1,2,3,4,5)
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date,
-#'              exit = ex_date, status=status,
-#'              breaks = list(fot=fb), pophaz=popmort)
-#' rpm <- relpois(x, formula = lex.Xst %in% 1:2 ~ FOT + agegr)
-#'  
-#' ## some methods for glm work. e.g. test for interaction
-#' \donttest{
-#' rpm2 <- relpois(x, formula = lex.Xst %in% 1:2 ~ FOT*agegr)
-#' anova(rpm, rpm2, test="LRT")
-#' AIC(rpm, rpm2)
-#' ## update() won't work currently
-#' }
-#' @references
-#' Paul W Dickman, Andy Sloggett, Michael Hills, and Timo Hakulinen.
-#' Regression models for relative survival. 
-#' Stat Med. 2004 Jan 15;23(1):51-64.
-#' \doi{10.1002/sim.1597}
-
-
-relpois <- function(data, 
-                    formula, 
-                    fot.breaks = NULL, subset = NULL, check=TRUE, ...) {
-  ## R CMD CHECK appeasement
-  lex.dur <- NULL
-  
-  ## prep arguments ------------------------------------------------------------
-  excess_cases <- fot <- pop.haz <-  NULL ## appease R CMD CHECK
-  
-  ## somehow the class of the data is being altered by this function
-  oldClass <- class(data)
-  
-  if (missing(formula) || !inherits(formula, "formula")) stop("formula not defined")
-  
-  form_vars <- all.vars(formula)
-  dataname <- as.name(deparse(substitute(data)))
-  
-  if (!inherits(data, "Lexis")) {
-    stop("data is not a Lexis object; data must be a result of splitting or using Lexis")
-  }
-  
-  if ("FOT" %in% names(data)) {
-    stop("FOT is a reserved name but you have a variable with that name in data; rename/delete it first")
-  }
-  
-  # wasDF <- FALSE
-  if (!is.data.table(data)) {
-    data <- copy(data)
-    setDT(data)
-    message("Took a copy of your data because it was a data.frame and not a data.table. This may take up a lot of memory.")
-    message("It is recommended to convert your data to data.table before using this function using as.data.table or setDT")
-  }
-  
-  req_vars <- unique(c("lex.id", "fot", "lex.dur", "pop.haz", setdiff(form_vars, "FOT")))
-  all_names_present(data, req_vars)
-  
-  surv.breaks <- attr(data, "breaks")$fot
-  if (is.null(surv.breaks)) {
-    stop("did not find any breaks information in data attributes named 'fot';
-            probable reason: split data was edited after splitting - ",
-         "don't do that")
-  } else {
-    if (!is.null(fot.breaks)) {
-      if (any(!fot.breaks %in% surv.breaks)) {
-        stop("fot.breaks must be a subset of the breaks for 'fot' used in splitting;
-           type attr(data, 'breaks')$fot to see the breaks you used in splitting")
-      } else {
-        surv.breaks <- fot.breaks
-      }
-    }
-    
-  }
-  
-  ## prep & subset data --------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset)
-  
-  if (any(is.na(data[subset, ]$pop.haz))) {
-    stop("some pop.haz are NA")
-  }
-  
-  on.exit({
-    setcolsnull(data, c("FOT", tmpdexp), soft = TRUE)
-  }, add = TRUE)
-  
-  if ("FOT" %in% form_vars)  {
-    data[, "FOT" := cut(fot, breaks = surv.breaks, right = FALSE)]
-    # set(data, j = "FOT", value = cut(data$fot, breaks = surv.breaks, right=FALSE))
-    subset <- subset & !is.na(data$FOT)
-  }
-  tmpdexp <- makeTempVarName(data, pre = "TEMP_d.exp_")
-  data[, c(tmpdexp) := pop.haz*lex.dur]
-  # set(data, j = tmpdexp, value = data$pop.haz * data$lex.dur)
-  
-  
-  if (check) {
-    ## test for negative excess cases in factor variable combinations ------------
-    
-    ## determine factor variables for cross-tabulating
-    
-    fac_vars <- colnames(attr(terms.formula(formula), "factors"))
-    
-    fac_list <- paste0(fac_vars, collapse=", ")
-    fac_list <- paste0("list(", fac_list, ")")
-    fac_list <- parse(text=fac_list)
-    
-    wh_fac <- as.data.table(data)[subset, unlist(lapply(eval(fac_list), is.factor))]
-    fac_vars <- fac_vars[wh_fac]
-    
-    if (length(fac_vars) == 0) fac_vars <- NULL
-    fac_list <- paste0(fac_vars, collapse=", ")
-    fac_list <- paste0("list(", fac_list, ")")
-    fac_list <- parse(text=fac_list)
-    
-    
-    ## test negativity of excess cases
-    LHS <- as.character(formula)
-    LHS <- LHS[2]
-    LHS <- parse(text = LHS)
-    
-    excas <- as.data.table(data)[subset, list(excess_cases = sum(eval(LHS)-get(tmpdexp))), keyby=eval(fac_list)]
-    setnames(excas, 1:ncol(excas), c(fac_vars, "excess_cases"))
-    
-    
-    if (any(is.na(excas$excess_cases))) {
-      stop("some excess cases were NA; is pop.haz available for all records?")
-    }
-    excas <- excas[excess_cases <= 0]
-    if (any(excas$excess_cases<=0)) {
-      print(excas)
-      warning("negative excess cases found in some combinations of factor variables;
-           see printed table and try e.g. wider FOT intervals")
-    }
-  }
-  
-  
-  ## custom poisson family -----------------------------------------------------
-  RPL <- copy(poisson())
-  RPL$link <- "glm relative survival model with Poisson error"
-  RPL$linkfun <- function(mu, d.exp = data[[tmpdexp]][subset]) log(mu - d.exp)
-  RPL$linkinv <- function(eta, d.exp = data[[tmpdexp]][subset]) d.exp + exp(eta)
-  
-  
-  RPL$initialize <- substitute( {
-    if (any(y < 0)) stop(paste("Negative values not allowed for", 
-                               "the Poisson family"))
-    n <- rep.int(1, nobs)
-    mustart <- pmax(y, d.exp) + 0.1
-  }, list(d.exp = data[[tmpdexp]][subset]) )
-  
-  
-  
-  ## glm call ------------------------------------------------------------------
-  ## update() won't work
-  ## anova() works
-  
-  
-  ml <- glm(formula = formula, data=data[subset,], offset=log(lex.dur),
-            #             subset = subset, ## Error in xj[i] : invalid subscript type 'closure' 
-            family = RPL, ...)
-  
-  ## final touches -------------------------------------------------------------
-  ml$d.exp <- data[subset, ][[tmpdexp]]
-  ml$FOT <- data[subset, ]$FOT
-  ml$fot.breaks <- surv.breaks
-  ml$call$data <- dataname
-  ml$call$formula <- formula
-  setattr(ml, "class", c("relpois", "glm", "lm"))
-  
-  setattr(data, "class", oldClass) ## see beginning of function
-  
-  
-  ml
-}
-
-
-
-#' @title Excess hazard Poisson model
-#' @author Joonas Miettinen, Karri Seppa
-#' @description Estimate a Poisson Piecewise Constant Excess
-#' Hazards Model
-#' @param formula a formula with the counts of events as the response.
-#' Passed on to \code{glm}. May contain usage of the \code{offset()} function
-#' instead of supplying the offset for the Poisson model via the argument
-#' \code{offset}.
-#' @param data an \code{aggre} object (an aggregated data set; 
-#' see \code{\link{as.aggre}} and \code{\link{aggre}})
-#' @param d.exp the counts of expected cases. Mandatory.
-#' E.g. \code{d.exp = EXC_CASES}, where \code{EXC_CASES} is a column in data.
-#' @param offset the offset for the Poisson model, supplied as e.g.
-#' \code{offset = log(PTIME)}, where \code{PTIME} is a subject-time
-#' variable in data. Not mandatory, but almost always should be supplied.
-#' @param breaks optional; a numeric vector of [a,b) breaks to specify
-#' survival intervals over the follow-up time; if \code{NULL}, the 
-#' existing breaks along the mandatory time scale mentioned in \code{formula}
-#' are used
-#' @param subset a logical vector or condition; e.g. \code{subset = sex == 1};
-#' limits the data before estimation
-#' @param piecewise \code{logical}; if \code{TRUE}, and if any time scale
-#' from data is used (mentioned) in the formula, the time scale is 
-#' transformed into a factor variable indicating intervals on the time scale.
-#' Otherwise the time scale left as it is, usually a numeric variable.
-#' E.g. if \code{formula = counts ~ TS1*VAR1}, \code{TS1} is transformed
-#' into a factor before fitting model.
-#' @param check \code{logical}; if \code{TRUE}, performs check on the 
-#' negativity excess cases by factor-like covariates in formula - 
-#' negative excess cases will very likely lead to non-converging model
-#' @param ... any other argument passed on to \code{\link[stats]{glm}} such as 
-#' \code{control} or \code{weights}
-#' @import stats
-#' 
-#' @return
-#' A \code{relpois} object created using a custom Poisson family construct.
-#' 
-#' @seealso
-#' \code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
-#' @family main functions
-#' @family relpois functions
-#' @examples
-#' ## use the simulated rectal cancer cohort
-#' data(sire, package = "popEpi")
-#' sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
-#' 
-#' ## create aggregated example data
-#' fb <- c(0,3/12,6/12,1,2,3,4,5)
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date,
-#'              exit = ex_date, status=status %in% 1:2,
-#'              breaks = list(fot=fb), 
-#'              pophaz=popmort, pp = FALSE,
-#'              aggre = list(agegr, fot))
-#'              
-#' ## fit model using aggregated data
-#' rpm <- relpois_ag(formula = from0to1 ~ fot + agegr,  data = x,
-#'                   d.exp = d.exp, offset = log(pyrs))
-#' summary(rpm)
-#'  
-#' ## the usual functions for handling glm models work
-#' rpm2 <- update(rpm, . ~ fot*agegr)
-#' anova(rpm, rpm2, test="LRT")
-#' AIC(rpm, rpm2)
-#' 
-#' ## other features such as residuals or predicting are not guaranteed
-#' ## to work as intended.
-#' @export
-
-relpois_ag <- function(formula, data, d.exp, offset = NULL, breaks = NULL, subset = NULL, piecewise = TRUE, check = TRUE, ...) {
-  
-  TF <- environment()
-  PF <- parent.frame(1L)
-  original_formula <- formula
-  
-  if (!inherits(data, "aggre")) {
-    stop("data is not an aggre object. Please aggregate your data first using ",
-         "e.g. lexpand(). If your data is pre-aggregated, use as.aggre() to ",
-         "mark it as such.")
-  }
-  
-  formula <- evalRecursive(formula, env = TF, enc = PF)$arg
-  if (missing(formula) || !inherits(formula, "formula")) stop("formula not defined")
-  
-  
-  
-  
-  ## detect survival time scale ------------------------------------------------
-  oldBreaks <- copy(attr(data, "breaks"))
-  allScales <- names(oldBreaks)
-  if (is.null(oldBreaks)) {
-    stop("data does not have breaks information. Is it a result of using ",
-         "aggre() or as.aggre()?")
-  }
-  survScale <- intersect(all.vars(formula), allScales)
-  if (length(survScale) > 1L) {
-    stop("Found several used time scales in formula, which is not supported ",
-         "(found ", paste0("'", survScale, "'", collapse = ", "), ")")
-  }
-  
-  ## check supplied breaks -----------------------------------------------------
-  if (is.numeric(breaks)) {
-    breaks <- list(breaks)
-    names(breaks) <- survScale
-  }
-  
-  if (!is.null(breaks)) {
-    if (!all_breaks_in(breaks, oldBreaks)) {
-      stop("Supplied breaks must be subset of the breaks used in splitting/",
-           "aggregating data. See the latter using e.g. ",
-           "attributes(x)$aggre.meta$breaks where x is your aggregated data.")
-    }
-  }
-  
-  ## pre-find args -------------------------------------------------------------
-  desub <- substitute(d.exp)
-  sub_d.exp <- evalRecursive(desub, env = data[1L, ], enc = PF)$argSub
-  offsub <- substitute(offset)
-  sub_offset <- evalRecursive(offsub, env = data[1L, ], enc = PF)$argSub
-  
-  ## prep & subset data --------------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset)
-  
-  av <- c(all.vars(formula), all.vars(sub_d.exp), all.vars(sub_offset))
-  av <- intersect(names(data), av)
-  
-  x <- subsetDTorDF(data, subset = subset, select = av)
-  
-  setDT(x)
-  setattr(x, "class", c("aggre", "data.table", "data.frame"))
-  
-  ## handle breaks -------------------------------------------------------------
-  if (!is.null(breaks)) {
-    if (!piecewise) {
-      stop("Supplied breaks but piecewise = FALSE. Please select piecewise = ",
-           "TRUE if you want piecewise estimates defined by the breaks.")
-    }
-    
-  } 
-  
-  cutBreaks <- breaks
-  othScales <- setdiff(names(oldBreaks), names(cutBreaks))
-  cutBreaks[othScales] <- oldBreaks[othScales]
-  cutBreaks[sapply(cutBreaks, length) < 2L] <- NULL
-  
-  if (piecewise && length(cutBreaks)) {
-    
-    for (sc in names(cutBreaks)) {
-      set(x, j = sc, value = cut(x[[sc]], breaks = cutBreaks[[sc]], 
-                                 right = FALSE, labels = FALSE))
-      
-      pieces <- round(cutBreaks[[sc]], 2L)
-      pieces <- paste0("[", pieces[-length(pieces)], ", ", pieces[-1L], ")")
-      set(x, j = sc, value = pieces[x[[sc]]])
-    }
-    
-  }
-  
-  ## eval value args -----------------------------------------------------------
-  d.exp <- evalPopArg(x, sub_d.exp, enclos = PF, 
-                      DT = TRUE, recursive = TRUE)
-  if (is.null(d.exp)) stop("argument d.exp was not supplied")
-  d.exp <- rowSums(d.exp)
-  if (length(d.exp) == nrow(data)) d.exp <- d.exp[subset]
-  
-  offset <- evalPopArg(x, sub_offset, enclos = PF, 
-                       DT = TRUE, recursive = TRUE)
-  if (!is.null(offset)) offset <- rowSums(offset)
-  if (length(offset) == nrow(data)) offset <- offset[subset]
-  
-  ## check excess cases --------------------------------------------------------
-  d <- eval(formula[[2]], envir = x, enclos = PF)
-  check_excess_cases(d = d, d.exp = d.exp, data = x, 
-                     formula = formula, enclos = PF)
-  
-  ## custom poisson family -----------------------------------------------------
-  RPL <- copy(poisson())
-  RPL$link <- "glm relative survival model with Poisson error"
-  RPL$linkfun <- function(mu, d.exp = TF$d.exp) {
-    log(mu - d.exp)
-  }
-  RPL$linkinv <- function(eta, d.exp = TF$d.exp) {
-    d.exp + exp(eta)
-  }
-  
-  RPL$initialize <- substitute( {
-    if (any(y < 0)) stop(paste("Negative values not allowed for", 
-                               "the Poisson family"))
-    n <- rep.int(1, nobs)
-    mustart <- pmax(y, d.exp) + 0.1
-  }, list(d.exp = TF$d.exp) )
-  
-  ## glm call ------------------------------------------------------------------
-  
-  ## NOTE: parent.frame(3L) to find this (this function's) environment
-  ml <- glm(formula = formula, data=x, offset = parent.frame(3L)$offset, family = RPL, ...)
-  
-  ## final touches -------------------------------------------------------------
-  
-  ml$call <- match.call()
-  setattr(ml, "class", c("relpois", "glm", "lm"))
-  
-  ml
-}
-
-
-
-
-
-
-check_excess_cases <- function(d, d.exp, formula, data, enclos = parent.frame(1)) {
-  # @title Check Excess Counts for a Relative Poisson Model
-  # @description Checks that the excess counts by strata all exceed 0.
-  # @param d a vector of observed counts of cases
-  # @param d.exp a vector of expected counts of cases
-  # @param a formula, the right side of which is inspected for factor-like 
-  # stratifying variables (factors and character variables)
-  # @param data a data set to eval formula in its context
-  # @param enclos passed on to RHS2DT() to evaluate formula to columns;
-  # enclosing environment of data
-  PF <- parent.frame(1)
-  tF <- environment()
-  
-  d.exc <- NULL
-  
-  by <- RHS2DT(formula, data = data, enclos = enclos)
-  if (!length(by)) by <- list()
-  facVars <- names(by)[sapply(by, function(col) is.factor(col) || is.character(col))]
-  
-  d <- substitute(d)
-  d <- eval(d, envir = data, enclos = PF)
-  d.exp <- substitute(d.exp)
-  d.exp <- eval(d.exp, envir = data, enclos = PF)
-  
-  if (length(facVars)) {
-    by <- setDT(mget(facVars, as.environment(by)))
-  } else {
-    by <- list()
-  }
-  
-  dt <- data.table(d = d, d.exp = d.exp)
-  dt[, d.exc := d - d.exp]
-  
-  for (k in seq_along(names(by))) {
-    bycol <- names(by)[k]
-    
-    tab <- dt[, lapply(.SD, sum), keyby = .(by[[bycol]])][d.exc <= 0L, ]
-    setnames(tab, 1, bycol)
-    if (nrow(tab)) {
-      on.exit(print(tab))
-      stop("There are negative excess cases in the data calculated separately ",
-           "by the factor-like variables ", 
-           paste0("'", facVars, "'", collapse = ", "), ". The model is not ",
-           "estimable with negative excess cases in strata. ",
-           "Infracting levels:")
-    }
-    
-  }
-
-  if (!length(by)) {
-    tab <- dt[, lapply(.SD, sum)]
-    if (tab$d.exc <= 0L) {
-      stop("The marginal sum of excess cases is negative; the model cannot ",
-           "be fitted. ")
-    }
-  }
- 
-  
- 
-  
-  invisible(NULL)
-}
-
-
-
-
-
-
-relpois_lex <- function(formula, 
-                        data, 
-                        pophaz = NULL, 
-                        breaks = NULL, 
-                        subset = NULL, 
-                        check = TRUE, 
-                        ...) {
-  PF <- parent.frame(1)
-  TF <- environment()
-  
-  form <- agVars <- NULL
-  
-  
-  ## checks --------------------------------------------------------------------
-  
-  checkLexisData(data)
-  checkPophaz(lex = data, ph = pophaz)
-  if (!is.null(breaks)) checkBreaksList(breaks)
-  
-  oldBreaks <- copy(attr(data, "breaks"))
-  allScales <- copy(attr(data, "time.scales"))
-  
-  
-  ## detect which time scale used ----------------------------------------------
-  
-  survScale <- intersect(all.vars(formula), allScales)
-  if (length(survScale) > 1L) {
-    stop("Found several used time scales in formula, which is not supported ",
-         "(found ", paste0("'", survScale, "'", collapse = ", "), ")")
-  }
-  ## subset --------------------------------------------------------------------
-  
-  sb <- substitute(subset)
-  subset <- evalLogicalSubset(data, sb, enclos = PF)
-  x <- data[subset, ]
-  
-  ## essentially same steps as in survtab() here, maybe make that
-  ## into a function / generalize lexpand.
-  
-  ## splitting -----------------------------------------------------------------
-  if (is.numeric(breaks) && length(survScale)) {
-    breaks <- list(breaks)
-    names(breaks) <- survScale
-  }
-  if (!is.null(breaks)) x <- splitMulti(x, breaks = breaks, drop = TRUE)
-  newBreaks <- copy(attr(x, "breaks"))
-  
-  ## merge in pophaz -----------------------------------------------------------
-  haz <- makeTempVarName(x, pre = "haz_")
-  ph <- data.table(pophaz)
-  phVars <- setdiff(names(ph), "haz")
-  setnames(ph, "haz", haz)
-  x <- cutLowMerge(x, pophaz, by = phVars, all.x = TRUE, all.y = FALSE, 
-                   old.nums = TRUE, mid.scales = intersect(allScales, phVars))
-  
-  # expected cases
-  d.exp <- makeTempVarName(x, pre = "d.exp_")
-  set(x, j = d.exp, value = x$lex.dur * x[[haz]])
-  
-  ## aggregating ---------------------------------------------------------------
-  ag <- model.frame(formula[-2], data = x) ## without response
-  setDT(ag)
-  set(ag, j = d.exp, value = x[[d.exp]])
-  d <- makeTempVarName(x, pre = "d_")
-  set(ag, j = d, value = eval(form))
-  ag <- aggre(x, by = agVars, sum.values = d.exp)
-  rm(x)
-  
-  ag_form <- formula
-  ag_form[[2]] <- quote(from0to1)
-  
-  rp <- relpois_ag(ag_form, data = data, breaks = NULL)
-  
-  rp$call <- match.call()
-  rp$formula <- formula
-  
-  rp
-}
-
+#' @title Excess hazard Poisson model
+#' @author Joonas Miettinen, Karri Seppa
+#' @description Estimate a Poisson piecewise constant excess
+#' hazards model
+#' @param data a dataset split with e.g. \code{\link{lexpand}};
+#' must have expected hazard merged within
+#' @param formula a formula which is passed on to \code{glm}; see Details
+#' @param fot.breaks optional; a numeric vector of [a,b) breaks to specify
+#' survival intervals over the follow-up time; if \code{NULL}, the 
+#' existing breaks along the mandatory \code{fot} time scale in \code{data}
+#' are used (e.g. the breaks for \code{fot} supplied to \code{lexpand})
+#' @param subset a logical vector or condition; e.g. \code{subset = sex == 1};
+#' limits the data before estimation
+#' @param check logical; if \code{TRUE}, tabulates excess cases by all
+#' factor variables in the formula to check for negative / \code{NA} 
+#' excess cases before fitting the GLM
+#' @param ... any argument passed on to \code{glm}
+#' @import stats
+#' @details
+#' 
+#' \strong{Basics}
+#' 
+#' \code{relpois} employs a custom link function of the Poisson variety
+#' to estimate piecewise constant parametric excess hazards. The pieces
+#' are determined by \code{fot.breaks}. A \code{log(person-years)} offset
+#' is passed automatically to the \code{glm} call.
+#' 
+#' \strong{Formula usage}
+#' 
+#' The formula can be used like any ordinary \code{glm} formula. The user must
+#' define the outcome in some manner, which is usually \code{lex.Xst} after splitting
+#' with e.g. \code{lexpand}. The exception is the possibility of including 
+#' the baseline excess hazard terms by including the 
+#' reserved term \code{FOT} in the formula.
+#' 
+#' For example, \code{lex.Xst != 0 ~ FOT + agegr} estimates a model with constant
+#' excess hazards at the follow-up intervals as specified by 
+#' the pertinent breaks used in splitting \code{data},
+#' as well as for the different age groups.
+#' \code{FOT} is created ad hoc if it is used in the formula.
+#' If you leave out \code{FOT}, the hazard is effectively
+#' assumed to be constant across the whole follow-up time. 
+#' 
+#' You can also simply use your own follow-up time interval variable that
+#' you have created before calling \code{relpois}. However, when using 
+#' \code{FOT}, \code{relpois} automatically checks for e.g. 
+#' negative excess cases in follow-up intervals,
+#' allowing for quickly finding splitting breaks
+#' where model estimation is possible. It also drops any data outside the
+#' follow-up time window.
+#' 
+#' \strong{Splitting and merging population hazard}
+#' 
+#' The easiest way to both split and to include population hazard information is 
+#' by using \code{\link{lexpand}}. You may also fairly easily do it by hand
+#' by splitting first and then merging in your population hazard information.
+#' 
+#' 
+#' \strong{Data requirements}
+#' 
+#' The population hazard information must be available for each record and named
+#' \code{pop.haz}. The follow-up time variable must be named \code{"fot"} e.g.
+#' as a result of using \code{lexpand}. The \code{lex.dur} variable must also
+#' be present, containing person-year information. 
+#' 
+#' 
+#' @return
+#' A \code{glm} object created using a custom Poisson family construct. Some
+#' \code{glm} methods are applicable.
+#' 
+#' @seealso
+#' \code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
+#' @family main functions
+#' @family relpois functions
+#' @export relpois
+#' 
+#' @examples
+#' ## use the simulated rectal cancer cohort
+#' data("sire", package = "popEpi")
+#' sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
+#' 
+#' ## usable straight away after splitting
+#' fb <- c(0,3/12,6/12,1,2,3,4,5)
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date,
+#'              exit = ex_date, status=status,
+#'              breaks = list(fot=fb), pophaz=popmort)
+#' rpm <- relpois(x, formula = lex.Xst %in% 1:2 ~ FOT + agegr)
+#'  
+#' ## some methods for glm work. e.g. test for interaction
+#' \donttest{
+#' rpm2 <- relpois(x, formula = lex.Xst %in% 1:2 ~ FOT*agegr)
+#' anova(rpm, rpm2, test="LRT")
+#' AIC(rpm, rpm2)
+#' ## update() won't work currently
+#' }
+#' @references
+#' Paul W Dickman, Andy Sloggett, Michael Hills, and Timo Hakulinen.
+#' Regression models for relative survival. 
+#' Stat Med. 2004 Jan 15;23(1):51-64.
+#' \doi{10.1002/sim.1597}
+
+
+relpois <- function(data, 
+                    formula, 
+                    fot.breaks = NULL, subset = NULL, check=TRUE, ...) {
+  ## R CMD CHECK appeasement
+  lex.dur <- NULL
+  
+  ## prep arguments ------------------------------------------------------------
+  excess_cases <- fot <- pop.haz <-  NULL ## appease R CMD CHECK
+  
+  ## somehow the class of the data is being altered by this function
+  oldClass <- class(data)
+  
+  if (missing(formula) || !inherits(formula, "formula")) stop("formula not defined")
+  
+  form_vars <- all.vars(formula)
+  dataname <- as.name(deparse(substitute(data)))
+  
+  if (!inherits(data, "Lexis")) {
+    stop("data is not a Lexis object; data must be a result of splitting or using Lexis")
+  }
+  
+  if ("FOT" %in% names(data)) {
+    stop("FOT is a reserved name but you have a variable with that name in data; rename/delete it first")
+  }
+  
+  # wasDF <- FALSE
+  if (!is.data.table(data)) {
+    data <- copy(data)
+    setDT(data)
+    message("Took a copy of your data because it was a data.frame and not a data.table. This may take up a lot of memory.")
+    message("It is recommended to convert your data to data.table before using this function using as.data.table or setDT")
+  }
+  
+  req_vars <- unique(c("lex.id", "fot", "lex.dur", "pop.haz", setdiff(form_vars, "FOT")))
+  all_names_present(data, req_vars)
+  
+  surv.breaks <- attr(data, "breaks")$fot
+  if (is.null(surv.breaks)) {
+    stop("did not find any breaks information in data attributes named 'fot';
+            probable reason: split data was edited after splitting - ",
+         "don't do that")
+  } else {
+    if (!is.null(fot.breaks)) {
+      if (any(!fot.breaks %in% surv.breaks)) {
+        stop("fot.breaks must be a subset of the breaks for 'fot' used in splitting;
+           type attr(data, 'breaks')$fot to see the breaks you used in splitting")
+      } else {
+        surv.breaks <- fot.breaks
+      }
+    }
+    
+  }
+  
+  ## prep & subset data --------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset)
+  
+  if (any(is.na(data[subset, ]$pop.haz))) {
+    stop("some pop.haz are NA")
+  }
+  
+  on.exit({
+    setcolsnull(data, c("FOT", tmpdexp), soft = TRUE)
+  }, add = TRUE)
+  
+  if ("FOT" %in% form_vars)  {
+    data[, "FOT" := cut(fot, breaks = surv.breaks, right = FALSE)]
+    # set(data, j = "FOT", value = cut(data$fot, breaks = surv.breaks, right=FALSE))
+    subset <- subset & !is.na(data$FOT)
+  }
+  tmpdexp <- makeTempVarName(data, pre = "TEMP_d.exp_")
+  data[, c(tmpdexp) := pop.haz*lex.dur]
+  # set(data, j = tmpdexp, value = data$pop.haz * data$lex.dur)
+  
+  
+  if (check) {
+    ## test for negative excess cases in factor variable combinations ------------
+    
+    ## determine factor variables for cross-tabulating
+    
+    fac_vars <- colnames(attr(terms.formula(formula), "factors"))
+    
+    fac_list <- paste0(fac_vars, collapse=", ")
+    fac_list <- paste0("list(", fac_list, ")")
+    fac_list <- parse(text=fac_list)
+    
+    wh_fac <- as.data.table(data)[subset, unlist(lapply(eval(fac_list), is.factor))]
+    fac_vars <- fac_vars[wh_fac]
+    
+    if (length(fac_vars) == 0) fac_vars <- NULL
+    fac_list <- paste0(fac_vars, collapse=", ")
+    fac_list <- paste0("list(", fac_list, ")")
+    fac_list <- parse(text=fac_list)
+    
+    
+    ## test negativity of excess cases
+    LHS <- as.character(formula)
+    LHS <- LHS[2]
+    LHS <- parse(text = LHS)
+    
+    excas <- as.data.table(data)[subset, list(excess_cases = sum(eval(LHS)-get(tmpdexp))), keyby=eval(fac_list)]
+    setnames(excas, 1:ncol(excas), c(fac_vars, "excess_cases"))
+    
+    
+    if (any(is.na(excas$excess_cases))) {
+      stop("some excess cases were NA; is pop.haz available for all records?")
+    }
+    excas <- excas[excess_cases <= 0]
+    if (any(excas$excess_cases<=0)) {
+      print(excas)
+      warning("negative excess cases found in some combinations of factor variables;
+           see printed table and try e.g. wider FOT intervals")
+    }
+  }
+  
+  
+  ## custom poisson family -----------------------------------------------------
+  RPL <- copy(poisson())
+  RPL$link <- "glm relative survival model with Poisson error"
+  RPL$linkfun <- function(mu, d.exp = data[[tmpdexp]][subset]) log(mu - d.exp)
+  RPL$linkinv <- function(eta, d.exp = data[[tmpdexp]][subset]) d.exp + exp(eta)
+  
+  
+  RPL$initialize <- substitute( {
+    if (any(y < 0)) stop(paste("Negative values not allowed for", 
+                               "the Poisson family"))
+    n <- rep.int(1, nobs)
+    mustart <- pmax(y, d.exp) + 0.1
+  }, list(d.exp = data[[tmpdexp]][subset]) )
+  
+  
+  
+  ## glm call ------------------------------------------------------------------
+  ## update() won't work
+  ## anova() works
+  
+  
+  ml <- glm(formula = formula, data=data[subset,], offset=log(lex.dur),
+            #             subset = subset, ## Error in xj[i] : invalid subscript type 'closure' 
+            family = RPL, ...)
+  
+  ## final touches -------------------------------------------------------------
+  ml$d.exp <- data[subset, ][[tmpdexp]]
+  ml$FOT <- data[subset, ]$FOT
+  ml$fot.breaks <- surv.breaks
+  ml$call$data <- dataname
+  ml$call$formula <- formula
+  setattr(ml, "class", c("relpois", "glm", "lm"))
+  
+  setattr(data, "class", oldClass) ## see beginning of function
+  
+  
+  ml
+}
+
+
+
+#' @title Excess hazard Poisson model
+#' @author Joonas Miettinen, Karri Seppa
+#' @description Estimate a Poisson Piecewise Constant Excess
+#' Hazards Model
+#' @param formula a formula with the counts of events as the response.
+#' Passed on to \code{glm}. May contain usage of the \code{offset()} function
+#' instead of supplying the offset for the Poisson model via the argument
+#' \code{offset}.
+#' @param data an \code{aggre} object (an aggregated data set; 
+#' see \code{\link{as.aggre}} and \code{\link{aggre}})
+#' @param d.exp the counts of expected cases. Mandatory.
+#' E.g. \code{d.exp = EXC_CASES}, where \code{EXC_CASES} is a column in data.
+#' @param offset the offset for the Poisson model, supplied as e.g.
+#' \code{offset = log(PTIME)}, where \code{PTIME} is a subject-time
+#' variable in data. Not mandatory, but almost always should be supplied.
+#' @param breaks optional; a numeric vector of [a,b) breaks to specify
+#' survival intervals over the follow-up time; if \code{NULL}, the 
+#' existing breaks along the mandatory time scale mentioned in \code{formula}
+#' are used
+#' @param subset a logical vector or condition; e.g. \code{subset = sex == 1};
+#' limits the data before estimation
+#' @param piecewise \code{logical}; if \code{TRUE}, and if any time scale
+#' from data is used (mentioned) in the formula, the time scale is 
+#' transformed into a factor variable indicating intervals on the time scale.
+#' Otherwise the time scale left as it is, usually a numeric variable.
+#' E.g. if \code{formula = counts ~ TS1*VAR1}, \code{TS1} is transformed
+#' into a factor before fitting model.
+#' @param check \code{logical}; if \code{TRUE}, performs check on the 
+#' negativity excess cases by factor-like covariates in formula - 
+#' negative excess cases will very likely lead to non-converging model
+#' @param ... any other argument passed on to \code{\link[stats]{glm}} such as 
+#' \code{control} or \code{weights}
+#' @import stats
+#' 
+#' @return
+#' A \code{relpois} object created using a custom Poisson family construct.
+#' 
+#' @seealso
+#' \code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
+#' @family main functions
+#' @family relpois functions
+#' @examples
+#' ## use the simulated rectal cancer cohort
+#' data(sire, package = "popEpi")
+#' sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
+#' 
+#' ## create aggregated example data
+#' fb <- c(0,3/12,6/12,1,2,3,4,5)
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date,
+#'              exit = ex_date, status=status %in% 1:2,
+#'              breaks = list(fot=fb), 
+#'              pophaz=popmort, pp = FALSE,
+#'              aggre = list(agegr, fot))
+#'              
+#' ## fit model using aggregated data
+#' rpm <- relpois_ag(formula = from0to1 ~ fot + agegr,  data = x,
+#'                   d.exp = d.exp, offset = log(pyrs))
+#' summary(rpm)
+#'  
+#' ## the usual functions for handling glm models work
+#' rpm2 <- update(rpm, . ~ fot*agegr)
+#' anova(rpm, rpm2, test="LRT")
+#' AIC(rpm, rpm2)
+#' 
+#' ## other features such as residuals or predicting are not guaranteed
+#' ## to work as intended.
+#' @export
+
+relpois_ag <- function(formula, data, d.exp, offset = NULL, breaks = NULL, subset = NULL, piecewise = TRUE, check = TRUE, ...) {
+  
+  TF <- environment()
+  PF <- parent.frame(1L)
+  original_formula <- formula
+  
+  if (!inherits(data, "aggre")) {
+    stop("data is not an aggre object. Please aggregate your data first using ",
+         "e.g. lexpand(). If your data is pre-aggregated, use as.aggre() to ",
+         "mark it as such.")
+  }
+  
+  formula <- evalRecursive(formula, env = TF, enc = PF)$arg
+  if (missing(formula) || !inherits(formula, "formula")) stop("formula not defined")
+  
+  
+  
+  
+  ## detect survival time scale ------------------------------------------------
+  oldBreaks <- copy(attr(data, "breaks"))
+  allScales <- names(oldBreaks)
+  if (is.null(oldBreaks)) {
+    stop("data does not have breaks information. Is it a result of using ",
+         "aggre() or as.aggre()?")
+  }
+  survScale <- intersect(all.vars(formula), allScales)
+  if (length(survScale) > 1L) {
+    stop("Found several used time scales in formula, which is not supported ",
+         "(found ", paste0("'", survScale, "'", collapse = ", "), ")")
+  }
+  
+  ## check supplied breaks -----------------------------------------------------
+  if (is.numeric(breaks)) {
+    breaks <- list(breaks)
+    names(breaks) <- survScale
+  }
+  
+  if (!is.null(breaks)) {
+    if (!all_breaks_in(breaks, oldBreaks)) {
+      stop("Supplied breaks must be subset of the breaks used in splitting/",
+           "aggregating data. See the latter using e.g. ",
+           "attributes(x)$aggre.meta$breaks where x is your aggregated data.")
+    }
+  }
+  
+  ## pre-find args -------------------------------------------------------------
+  desub <- substitute(d.exp)
+  sub_d.exp <- evalRecursive(desub, env = data[1L, ], enc = PF)$argSub
+  offsub <- substitute(offset)
+  sub_offset <- evalRecursive(offsub, env = data[1L, ], enc = PF)$argSub
+  
+  ## prep & subset data --------------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset)
+  
+  av <- c(all.vars(formula), all.vars(sub_d.exp), all.vars(sub_offset))
+  av <- intersect(names(data), av)
+  
+  x <- subsetDTorDF(data, subset = subset, select = av)
+  
+  setDT(x)
+  setattr(x, "class", c("aggre", "data.table", "data.frame"))
+  
+  ## handle breaks -------------------------------------------------------------
+  if (!is.null(breaks)) {
+    if (!piecewise) {
+      stop("Supplied breaks but piecewise = FALSE. Please select piecewise = ",
+           "TRUE if you want piecewise estimates defined by the breaks.")
+    }
+    
+  } 
+  
+  cutBreaks <- breaks
+  othScales <- setdiff(names(oldBreaks), names(cutBreaks))
+  cutBreaks[othScales] <- oldBreaks[othScales]
+  cutBreaks[sapply(cutBreaks, length) < 2L] <- NULL
+  
+  if (piecewise && length(cutBreaks)) {
+    
+    for (sc in names(cutBreaks)) {
+      set(x, j = sc, value = cut(x[[sc]], breaks = cutBreaks[[sc]], 
+                                 right = FALSE, labels = FALSE))
+      
+      pieces <- round(cutBreaks[[sc]], 2L)
+      pieces <- paste0("[", pieces[-length(pieces)], ", ", pieces[-1L], ")")
+      set(x, j = sc, value = pieces[x[[sc]]])
+    }
+    
+  }
+  
+  ## eval value args -----------------------------------------------------------
+  d.exp <- evalPopArg(x, sub_d.exp, enclos = PF, 
+                      DT = TRUE, recursive = TRUE)
+  if (is.null(d.exp)) stop("argument d.exp was not supplied")
+  d.exp <- rowSums(d.exp)
+  if (length(d.exp) == nrow(data)) d.exp <- d.exp[subset]
+  
+  offset <- evalPopArg(x, sub_offset, enclos = PF, 
+                       DT = TRUE, recursive = TRUE)
+  if (!is.null(offset)) offset <- rowSums(offset)
+  if (length(offset) == nrow(data)) offset <- offset[subset]
+  
+  ## check excess cases --------------------------------------------------------
+  d <- eval(formula[[2]], envir = x, enclos = PF)
+  check_excess_cases(d = d, d.exp = d.exp, data = x, 
+                     formula = formula, enclos = PF)
+  
+  ## custom poisson family -----------------------------------------------------
+  RPL <- copy(poisson())
+  RPL$link <- "glm relative survival model with Poisson error"
+  RPL$linkfun <- function(mu, d.exp = TF$d.exp) {
+    log(mu - d.exp)
+  }
+  RPL$linkinv <- function(eta, d.exp = TF$d.exp) {
+    d.exp + exp(eta)
+  }
+  
+  RPL$initialize <- substitute( {
+    if (any(y < 0)) stop(paste("Negative values not allowed for", 
+                               "the Poisson family"))
+    n <- rep.int(1, nobs)
+    mustart <- pmax(y, d.exp) + 0.1
+  }, list(d.exp = TF$d.exp) )
+  
+  ## glm call ------------------------------------------------------------------
+  
+  ## NOTE: parent.frame(3L) to find this (this function's) environment
+  ml <- glm(formula = formula, data=x, offset = parent.frame(3L)$offset, family = RPL, ...)
+  
+  ## final touches -------------------------------------------------------------
+  
+  ml$call <- match.call()
+  setattr(ml, "class", c("relpois", "glm", "lm"))
+  
+  ml
+}
+
+
+
+
+
+
+check_excess_cases <- function(d, d.exp, formula, data, enclos = parent.frame(1)) {
+  # @title Check Excess Counts for a Relative Poisson Model
+  # @description Checks that the excess counts by strata all exceed 0.
+  # @param d a vector of observed counts of cases
+  # @param d.exp a vector of expected counts of cases
+  # @param a formula, the right side of which is inspected for factor-like 
+  # stratifying variables (factors and character variables)
+  # @param data a data set to eval formula in its context
+  # @param enclos passed on to RHS2DT() to evaluate formula to columns;
+  # enclosing environment of data
+  PF <- parent.frame(1)
+  tF <- environment()
+  
+  d.exc <- NULL
+  
+  by <- RHS2DT(formula, data = data, enclos = enclos)
+  if (!length(by)) by <- list()
+  facVars <- names(by)[sapply(by, function(col) is.factor(col) || is.character(col))]
+  
+  d <- substitute(d)
+  d <- eval(d, envir = data, enclos = PF)
+  d.exp <- substitute(d.exp)
+  d.exp <- eval(d.exp, envir = data, enclos = PF)
+  
+  if (length(facVars)) {
+    by <- setDT(mget(facVars, as.environment(by)))
+  } else {
+    by <- list()
+  }
+  
+  dt <- data.table(d = d, d.exp = d.exp)
+  dt[, d.exc := d - d.exp]
+  
+  for (k in seq_along(names(by))) {
+    bycol <- names(by)[k]
+    
+    tab <- dt[, lapply(.SD, sum), keyby = .(by[[bycol]])][d.exc <= 0L, ]
+    setnames(tab, 1, bycol)
+    if (nrow(tab)) {
+      on.exit(print(tab))
+      stop("There are negative excess cases in the data calculated separately ",
+           "by the factor-like variables ", 
+           paste0("'", facVars, "'", collapse = ", "), ". The model is not ",
+           "estimable with negative excess cases in strata. ",
+           "Infracting levels:")
+    }
+    
+  }
+
+  if (!length(by)) {
+    tab <- dt[, lapply(.SD, sum)]
+    if (tab$d.exc <= 0L) {
+      stop("The marginal sum of excess cases is negative; the model cannot ",
+           "be fitted. ")
+    }
+  }
+ 
+  
+ 
+  
+  invisible(NULL)
+}
+
+
+
+
+
+
+relpois_lex <- function(formula, 
+                        data, 
+                        pophaz = NULL, 
+                        breaks = NULL, 
+                        subset = NULL, 
+                        check = TRUE, 
+                        ...) {
+  PF <- parent.frame(1)
+  TF <- environment()
+  
+  form <- agVars <- NULL
+  
+  
+  ## checks --------------------------------------------------------------------
+  
+  checkLexisData(data)
+  checkPophaz(lex = data, ph = pophaz)
+  if (!is.null(breaks)) checkBreaksList(breaks)
+  
+  oldBreaks <- copy(attr(data, "breaks"))
+  allScales <- copy(attr(data, "time.scales"))
+  
+  
+  ## detect which time scale used ----------------------------------------------
+  
+  survScale <- intersect(all.vars(formula), allScales)
+  if (length(survScale) > 1L) {
+    stop("Found several used time scales in formula, which is not supported ",
+         "(found ", paste0("'", survScale, "'", collapse = ", "), ")")
+  }
+  ## subset --------------------------------------------------------------------
+  
+  sb <- substitute(subset)
+  subset <- evalLogicalSubset(data, sb, enclos = PF)
+  x <- data[subset, ]
+  
+  ## essentially same steps as in survtab() here, maybe make that
+  ## into a function / generalize lexpand.
+  
+  ## splitting -----------------------------------------------------------------
+  if (is.numeric(breaks) && length(survScale)) {
+    breaks <- list(breaks)
+    names(breaks) <- survScale
+  }
+  if (!is.null(breaks)) x <- splitMulti(x, breaks = breaks, drop = TRUE)
+  newBreaks <- copy(attr(x, "breaks"))
+  
+  ## merge in pophaz -----------------------------------------------------------
+  haz <- makeTempVarName(x, pre = "haz_")
+  ph <- data.table(pophaz)
+  phVars <- setdiff(names(ph), "haz")
+  setnames(ph, "haz", haz)
+  x <- cutLowMerge(x, pophaz, by = phVars, all.x = TRUE, all.y = FALSE, 
+                   old.nums = TRUE, mid.scales = intersect(allScales, phVars))
+  
+  # expected cases
+  d.exp <- makeTempVarName(x, pre = "d.exp_")
+  set(x, j = d.exp, value = x$lex.dur * x[[haz]])
+  
+  ## aggregating ---------------------------------------------------------------
+  ag <- model.frame(formula[-2], data = x) ## without response
+  setDT(ag)
+  set(ag, j = d.exp, value = x[[d.exp]])
+  d <- makeTempVarName(x, pre = "d_")
+  set(ag, j = d, value = eval(form))
+  ag <- aggre(x, by = agVars, sum.values = d.exp)
+  rm(x)
+  
+  ag_form <- formula
+  ag_form[[2]] <- quote(from0to1)
+  
+  rp <- relpois_ag(ag_form, data = data, breaks = NULL)
+  
+  rp$call <- match.call()
+  rp$formula <- formula
+  
+  rp
+}
+
diff --git a/R/relative_poisson_net_survival.R b/R/relative_poisson_net_survival.R
index 81b50ae..21adde1 100644
--- a/R/relative_poisson_net_survival.R
+++ b/R/relative_poisson_net_survival.R
@@ -1,190 +1,190 @@
-#' @title Marginal piecewise parametric relative survival curve
-#' @author Joonas Miettinen
-#' @description Fit a marginal relative survival curve based on a \code{relpois} fit
-#' @param object a \code{relpois} object
-#' @details
-#' Estimates a marginal curve, i.e. the average of all
-#' possible individual curves. 
-#' 
-#' Only supported when the reserved \code{FOT} variable was used in \code{relpois}.
-#' Computes a curve for each unique combination of covariates (e.g. 4 sets) 
-#' and returns a weighted average curve based on the counts
-#' of subjects for each combination (e.g. 1000, 125, 50, 25 respectively). 
-#' Fairly fast when only categorical variables have been used, otherwise
-#' go get a cup of coffee.
-#' 
-#' If delayed entry is present in data due to period analysis limiting,
-#' the marginal curve is constructed only for those whose follow-up started
-#' in the respective period.
-#' @return
-#' A `data.table` of relative survival curves.
-#' @export 
-#' @family relpois functions
-#' 
-#' @import data.table
-#' @import Epi
-#' @import stats
-#' 
-#' @examples
-#' \donttest{
-#' ## use the simulated rectal cancer cohort
-#' data("sire", package = "popEpi")
-#' ab <- c(0,45,55,65,70,Inf)
-#' sire$agegr <- cut(sire$dg_age, breaks = ab, right = FALSE)
-#'
-#' BL <- list(fot= seq(0,10,1/12))
-#' pm <- data.frame(popEpi::popmort)
-#' x <- lexpand(sire, breaks=BL, pophaz=pm, 
-#'              birth = bi_date, 
-#'              entry = dg_date, exit = ex_date, 
-#'              status  = status %in% 1:2)
-#' 
-#' rpm <- relpois(x, formula = lex.Xst %in% 1:2 ~ -1+ FOT + agegr, 
-#'                fot.breaks=c(0,0.25,0.5,1:8,10))
-#' pmc <- rpcurve(rpm)
-#'
-#' ## compare with non-parametric estimates
-#' names(pm) <- c("sex", "per", "age", "haz")
-#' x$agegr <- cut(x$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
-#' st <- survtab(fot ~ adjust(agegr), data = x, weights = "internal",
-#'               pophaz = pm)
-#'
-#'
-#' plot(st, y = "r.e2.as")
-#' lines(y = pmc$est, x = pmc$Tstop, col="red")
-#' }
-#' 
-#' 
-#' 
-
-rpcurve <- function(object) {
-  
-  ## appease R CMD CHECK
-  Tstart <- FOT <- uni_id <- uni_n <- uni_w <- 
-    lo <- hi <- lex.Xst <- est <- fot <- pop.haz <- delta <- 
-    Tstop <- Tstar <- lex.id <- fot <- lex.multi <- pyrs <- NULL
-  
-  ## sanity checks -------------------------------------------------------------
-  if (!inherits(object, "relpois")) {
-    stop("object does not have class 'relpois'; see ?relpois")
-  }
-  
-  if (!"FOT" %in% all.vars(object$formula)) {
-    stop("No FOT variable in model formula; see Details in ?rpcurve")
-  }
-  
-  
-  ## collate surv.ints, breaks, deltas -----------------------------------------
-  fot_levels <- as.factor(sort(as.character(unique(object$model$FOT))))
-  fb <- sort(object$fot.breaks)
-  fb <- data.table(Tstart = fb[-length(fb)], Tstop = fb[-1])
-  fb[, FOT := fot_levels]
-  fb[, delta := Tstop-Tstart]
-  n_ints <- nrow(fb)
-  
-  ## model data / model matrix construction ------------------------------------
-  modmat <- data.table(object$data)
-  if (!"lex.multi" %in% names(modmat)) {
-    setkey(modmat, lex.id, fot)
-    modmat[, lex.multi := 1:.N, by = lex.id]
-  }
-  setkey(modmat, lex.id, lex.multi)
-  modmat <- unique(modmat, by = "lex.id")
-  modmat <- modmat[fot == 0] ## with period data, only non-delayed entries used
-  modmat <- modmat[rep(1:.N, each = n_ints)]
-  IDs <- modmat$lex.id
-  n_matrows <- length(IDs)
-  
-  setcolsnull(modmat, keep = c(all.vars(object$formula)))
-  setcolsnull(modmat, "FOT")
-  modmat <- cbind(fb[, list(FOT=FOT, lex.dur = delta)], modmat)
-  modmat[, "lex.Xst_factor" := factor(levels(as.factor(lex.Xst))[1])]
-  modmat[, "lex.Xst" := NULL]
-  data.table::setnames(modmat, "lex.Xst_factor", "lex.Xst")
-  modmat[, "order" := 1:.N]
-  
-  ## unique sets of covariates only
-  umodmat <- unique(modmat, by = setdiff(names(modmat), c("lex.dur","lex.Xst","order")))
-  umodmat[, uni_id := rep(1:(nrow(umodmat)/n_ints), each=n_ints)]
-    
-  setkeyv(umodmat, setdiff(names(modmat), c("lex.dur","lex.Xst","order","uni_id")))
-  setkeyv(modmat, setdiff(names(modmat), c("lex.dur","lex.Xst","order","uni_id")))
-  
-  uni_n <- umodmat[modmat, list(uni_n = .N / n_ints), by = "uni_id"][["uni_n"]]
-  uni_n <- rep(uni_n, times = nrow(umodmat) / length(uni_n))
-  data.table::set(umodmat, j = "uni_n", value = uni_n)
-  
-  setkeyv(umodmat, c("uni_id", "order"))
-  mean_weights <- umodmat$uni_n
-  IDs <- umodmat$uni_id
-  
-  setcolsnull(umodmat, delete=c("order","uni_id","uni_n"))
-  
-  modmat <- stats::model.matrix(object, data=umodmat)
-  mmattrs <- attributes(modmat)
-  mmattrs$dimnames <- mmattrs$dim <- NULL
-
-  
-  ## (unique covariate) subject-specific curve fits ----------------------------
-  l <- split(data.table(modmat), IDs)
-  l <- lapply(l, as.matrix)
-  attrsetter <- function(obj) {
-    mostattributes(obj) <- c(attributes(obj), mmattrs)
-    obj
-  }
-  l <- lapply(l, attrsetter)
-
-  epicumgetter <- function(x, ...) {
-    Epi::ci.cum(ctr.mat = x, ..., alpha = 1-0.95, Exp = TRUE, ci.Exp = TRUE)
-  }
-  
-  tab <- lapply(l, epicumgetter, obj=object, intl = fb$delta); rm(l)
-
-  ## collate & compute relative survivals --------------------------------------
-  tab <- lapply(tab, as.data.table)
-  tab <- rbindlist(tab)
-  setnames(tab, names(tab), c("est", "lo", "hi", "SE"))
-  fot_values <- rep(fot_levels, times = nrow(tab) / length(fot_levels))
-  data.table::set(tab, j = "FOT", value = fot_values)
-  tab[, "uni_id" := IDs]
-  tab[, "uni_w"  := mean_weights]
-  Haz2RS <- function(x) {
-    sum(exp(-x)*tab$uni_w)/n_matrows
-  }
-  tab <- tab[, lapply(list(est=est,lo=lo,hi=hi), `-`)]
-  tab <- tab[, lapply(list(est=est,lo=lo,hi=hi), exp)]
-  tab[, `:=`(est=est*mean_weights,lo=lo*mean_weights,hi=hi*mean_weights)]
-  data.table::set(tab, j = "FOT", value = fot_values)
-  tab <- tab[, lapply(list(est=est, lo=lo, hi=hi), sum), by = "FOT"]
-  tab <- tab[
-    j = lapply(list(est=est, lo=lo, hi=hi), function(x) {
-      x / (n_matrows / n_ints)
-    }), 
-    by = "FOT"
-    ]
-
-  setkeyv(tab, "FOT")
-  setkeyv(fb, "FOT")
-  tab <- fb[tab, on = "FOT"]
-  
-  ## disabled CI computation in 0.2.2 due to lack of testing & certainty of 
-  ## correctness
-  setcolsnull(tab, c("lo", "hi"))
-  
-  setattr(tab, "class", c("data.table", "data.frame"))
-  if (!return_DT()) setDFpe(tab)
-  tab[]
-}
-
-#' @title Relative Poisson family object
-#' @author Karri Seppa
-#' @description A family object for GLM fitting of relative Poisson models
-#' @format 
-#' A list very similar to that created by \code{poisson()}.
-#' @export
-#' @family relpois functions
-RPL <- data.table::copy(stats::poisson())
-RPL$link <- "glm relative survival model with Poisson error"
-RPL$linkfun <- function(mu, d.exp) log(mu - d.exp)
-RPL$linkinv <- function(eta, d.exp) d.exp + exp(eta)
-
+#' @title Marginal piecewise parametric relative survival curve
+#' @author Joonas Miettinen
+#' @description Fit a marginal relative survival curve based on a \code{relpois} fit
+#' @param object a \code{relpois} object
+#' @details
+#' Estimates a marginal curve, i.e. the average of all
+#' possible individual curves. 
+#' 
+#' Only supported when the reserved \code{FOT} variable was used in \code{relpois}.
+#' Computes a curve for each unique combination of covariates (e.g. 4 sets) 
+#' and returns a weighted average curve based on the counts
+#' of subjects for each combination (e.g. 1000, 125, 50, 25 respectively). 
+#' Fairly fast when only categorical variables have been used, otherwise
+#' go get a cup of coffee.
+#' 
+#' If delayed entry is present in data due to period analysis limiting,
+#' the marginal curve is constructed only for those whose follow-up started
+#' in the respective period.
+#' @return
+#' A `data.table` of relative survival curves.
+#' @export 
+#' @family relpois functions
+#' 
+#' @import data.table
+#' @import Epi
+#' @import stats
+#' 
+#' @examples
+#' \donttest{
+#' ## use the simulated rectal cancer cohort
+#' data("sire", package = "popEpi")
+#' ab <- c(0,45,55,65,70,Inf)
+#' sire$agegr <- cut(sire$dg_age, breaks = ab, right = FALSE)
+#'
+#' BL <- list(fot= seq(0,10,1/12))
+#' pm <- data.frame(popEpi::popmort)
+#' x <- lexpand(sire, breaks=BL, pophaz=pm, 
+#'              birth = bi_date, 
+#'              entry = dg_date, exit = ex_date, 
+#'              status  = status %in% 1:2)
+#' 
+#' rpm <- relpois(x, formula = lex.Xst %in% 1:2 ~ -1+ FOT + agegr, 
+#'                fot.breaks=c(0,0.25,0.5,1:8,10))
+#' pmc <- rpcurve(rpm)
+#'
+#' ## compare with non-parametric estimates
+#' names(pm) <- c("sex", "per", "age", "haz")
+#' x$agegr <- cut(x$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
+#' st <- survtab(fot ~ adjust(agegr), data = x, weights = "internal",
+#'               pophaz = pm)
+#'
+#'
+#' plot(st, y = "r.e2.as")
+#' lines(y = pmc$est, x = pmc$Tstop, col="red")
+#' }
+#' 
+#' 
+#' 
+
+rpcurve <- function(object) {
+  
+  ## appease R CMD CHECK
+  Tstart <- FOT <- uni_id <- uni_n <- uni_w <- 
+    lo <- hi <- lex.Xst <- est <- fot <- pop.haz <- delta <- 
+    Tstop <- Tstar <- lex.id <- fot <- lex.multi <- pyrs <- NULL
+  
+  ## sanity checks -------------------------------------------------------------
+  if (!inherits(object, "relpois")) {
+    stop("object does not have class 'relpois'; see ?relpois")
+  }
+  
+  if (!"FOT" %in% all.vars(object$formula)) {
+    stop("No FOT variable in model formula; see Details in ?rpcurve")
+  }
+  
+  
+  ## collate surv.ints, breaks, deltas -----------------------------------------
+  fot_levels <- as.factor(sort(as.character(unique(object$model$FOT))))
+  fb <- sort(object$fot.breaks)
+  fb <- data.table(Tstart = fb[-length(fb)], Tstop = fb[-1])
+  fb[, FOT := fot_levels]
+  fb[, delta := Tstop-Tstart]
+  n_ints <- nrow(fb)
+  
+  ## model data / model matrix construction ------------------------------------
+  modmat <- data.table(object$data)
+  if (!"lex.multi" %in% names(modmat)) {
+    setkey(modmat, lex.id, fot)
+    modmat[, lex.multi := 1:.N, by = lex.id]
+  }
+  setkey(modmat, lex.id, lex.multi)
+  modmat <- unique(modmat, by = "lex.id")
+  modmat <- modmat[fot == 0] ## with period data, only non-delayed entries used
+  modmat <- modmat[rep(1:.N, each = n_ints)]
+  IDs <- modmat$lex.id
+  n_matrows <- length(IDs)
+  
+  setcolsnull(modmat, keep = c(all.vars(object$formula)))
+  setcolsnull(modmat, "FOT")
+  modmat <- cbind(fb[, list(FOT=FOT, lex.dur = delta)], modmat)
+  modmat[, "lex.Xst_factor" := factor(levels(as.factor(lex.Xst))[1])]
+  modmat[, "lex.Xst" := NULL]
+  data.table::setnames(modmat, "lex.Xst_factor", "lex.Xst")
+  modmat[, "order" := 1:.N]
+  
+  ## unique sets of covariates only
+  umodmat <- unique(modmat, by = setdiff(names(modmat), c("lex.dur","lex.Xst","order")))
+  umodmat[, uni_id := rep(1:(nrow(umodmat)/n_ints), each=n_ints)]
+    
+  setkeyv(umodmat, setdiff(names(modmat), c("lex.dur","lex.Xst","order","uni_id")))
+  setkeyv(modmat, setdiff(names(modmat), c("lex.dur","lex.Xst","order","uni_id")))
+  
+  uni_n <- umodmat[modmat, list(uni_n = .N / n_ints), by = "uni_id"][["uni_n"]]
+  uni_n <- rep(uni_n, times = nrow(umodmat) / length(uni_n))
+  data.table::set(umodmat, j = "uni_n", value = uni_n)
+  
+  setkeyv(umodmat, c("uni_id", "order"))
+  mean_weights <- umodmat$uni_n
+  IDs <- umodmat$uni_id
+  
+  setcolsnull(umodmat, delete=c("order","uni_id","uni_n"))
+  
+  modmat <- stats::model.matrix(object, data=umodmat)
+  mmattrs <- attributes(modmat)
+  mmattrs$dimnames <- mmattrs$dim <- NULL
+
+  
+  ## (unique covariate) subject-specific curve fits ----------------------------
+  l <- split(data.table(modmat), IDs)
+  l <- lapply(l, as.matrix)
+  attrsetter <- function(obj) {
+    mostattributes(obj) <- c(attributes(obj), mmattrs)
+    obj
+  }
+  l <- lapply(l, attrsetter)
+
+  epicumgetter <- function(x, ...) {
+    Epi::ci.cum(ctr.mat = x, ..., alpha = 1-0.95, Exp = TRUE, ci.Exp = TRUE)
+  }
+  
+  tab <- lapply(l, epicumgetter, obj=object, intl = fb$delta); rm(l)
+
+  ## collate & compute relative survivals --------------------------------------
+  tab <- lapply(tab, as.data.table)
+  tab <- rbindlist(tab)
+  setnames(tab, names(tab), c("est", "lo", "hi", "SE"))
+  fot_values <- rep(fot_levels, times = nrow(tab) / length(fot_levels))
+  data.table::set(tab, j = "FOT", value = fot_values)
+  tab[, "uni_id" := IDs]
+  tab[, "uni_w"  := mean_weights]
+  Haz2RS <- function(x) {
+    sum(exp(-x)*tab$uni_w)/n_matrows
+  }
+  tab <- tab[, lapply(list(est=est,lo=lo,hi=hi), `-`)]
+  tab <- tab[, lapply(list(est=est,lo=lo,hi=hi), exp)]
+  tab[, `:=`(est=est*mean_weights,lo=lo*mean_weights,hi=hi*mean_weights)]
+  data.table::set(tab, j = "FOT", value = fot_values)
+  tab <- tab[, lapply(list(est=est, lo=lo, hi=hi), sum), by = "FOT"]
+  tab <- tab[
+    j = lapply(list(est=est, lo=lo, hi=hi), function(x) {
+      x / (n_matrows / n_ints)
+    }), 
+    by = "FOT"
+    ]
+
+  setkeyv(tab, "FOT")
+  setkeyv(fb, "FOT")
+  tab <- fb[tab, on = "FOT"]
+  
+  ## disabled CI computation in 0.2.2 due to lack of testing & certainty of 
+  ## correctness
+  setcolsnull(tab, c("lo", "hi"))
+  
+  setattr(tab, "class", c("data.table", "data.frame"))
+  if (!return_DT()) setDFpe(tab)
+  tab[]
+}
+
+#' @title Relative Poisson family object
+#' @author Karri Seppa
+#' @description A family object for GLM fitting of relative Poisson models
+#' @format 
+#' A list very similar to that created by \code{poisson()}.
+#' @export
+#' @family relpois functions
+RPL <- data.table::copy(stats::poisson())
+RPL$link <- "glm relative survival model with Poisson error"
+RPL$linkfun <- function(mu, d.exp) log(mu - d.exp)
+RPL$linkinv <- function(eta, d.exp) d.exp + exp(eta)
+
diff --git a/R/sir.R b/R/sir.R
index 081a8b8..2d2a0e0 100644
--- a/R/sir.R
+++ b/R/sir.R
@@ -1,1452 +1,1452 @@
-#' @title Calculate SIR or SMR
-#' @author Matti Rantanen, Joonas Miettinen
-#' @description Poisson modelled standardised incidence or mortality ratios (SIRs / SMRs) i.e. 
-#' indirect method for calculating standardised rates. SIR is a ratio of observed and expected cases.
-#' Expected cases are derived by multiplying the strata-specific population rate with the
-#' corresponding person-years of the cohort.
-#' 
-#' @details \code{sir} is a comprehensive tool for modelling SIRs/SMRs with flexible 
-#' options to adjust and print SIRs, test homogeneity and utilize 
-#' multi-state data. The cohort data and the variable names for observation 
-#' counts and person-years are required.
-#' The reference data is optional, since the cohort data 
-#' can be stratified (\code{print}) and compared to total.
-#' 
-#' 
-#' \strong{Adjust and print}
-#' 
-#' A SIR can be adjusted or standardised using the covariates found in both \code{coh.data} and \code{ref.data}.
-#' Variable to adjust are given in \code{adjust}.
-#' Variable names needs to match in both \code{coh.data} and \code{ref.data}. 
-#' Typical variables to adjust by are gender, age group and calendar period.
-#' 
-#' \code{print} is used to stratify the SIR output. In other words, the variables 
-#' assigned to \code{print} are the covariates of the Poisson model.
-#' Variable levels are treated as categorical.
-#' Variables can be assigned in both \code{print} and \code{adjust}. 
-#' This means the output it adjusted and printed by these variables.
-#' 
-#' \code{print} can also be a list of expressions. This enables changing variable 
-#' names or transforming variables with functions such as \code{cut} and \code{round}.
-#' For example, the existing variables \code{agegroup} and \code{year} could be
-#' transformed to new levels using \code{cut} by
-#' 
-#' \code{print = list( age.category = cut(agegroup, breaks = c(0,50,75,100)), 
-#' year.cat = cut(year, seq(1950,2010,20)))}
-#' 
-#' 
-#' \strong{ref.rate or ref.obs & ref.pyrs}
-#' 
-#' The population rate variable can be given to the \code{ref.rate} parameter. 
-#' That is, when using e.g. the \code{popmort} or a comparable data file, one may
-#' supply \code{ref.rate} instead of \code{ref.obs} and \code{ref.pyrs}, which
-#' will be ignored if \code{ref.rate} is supplied. 
-#' 
-#' 
-#' Note that if all the stratifying variables in 
-#' \code{ref.data} are not listed in \code{adjust}, 
-#' or when the categories are otherwise combined,
-#' the (unweighted) mean of rates is used for computing expected cases.
-#' This might incur a small bias in comparison to when exact numbers of observations
-#' and person-years are available. 
-#' 
-#' 
-#' 
-#' \strong{mstate}
-#' 
-#' E.g. using \code{lexpand} it's possible to compute counts for several outcomes
-#' so that the population at risk is same for each 
-#' outcome such as a certain kind of cancer. 
-#' The transition counts are in wide data format, 
-#' and the relevant columns can be supplied to \code{sir}
-#' in a vector via the \code{coh.obs} argument. 
-#' The name of the corresponding new column in \code{ref.data} is given in
-#' \code{mstate}. It's recommended to include the \code{mstate} variable in \code{adjust},
-#' so the corresponding information should also be available in \code{ref.data}.
-#' More examples in sir-vignette.
-#' 
-#' This approach is analogous to where SIRs are calculated separately their 
-#' own function calls.
-#' 
-#' 
-#' \strong{Other parameters}
-#' 
-#' \code{univariate} confidence intervals are calculated using exact 
-#' Poisson intervals (\code{poisson.ci}). The options \code{profile} and \code{wald} are
-#' is based on a Poisson regression model: profile-likelihood confidence intervals 
-#' or Wald's normal-approximation. P-value is Poisson model based \code{conf.type}
-#' or calculated using the method described by Breslow and Day. Function automatically
-#' switches to another \code{conf.type} if calculation is not possible with a message.
-#' Usually model fit fails if there is print stratum with zero expected values.
-#' 
-#' 
-#' The LRT p-value tests the levels of \code{print}. The test can be either 
-#' \code{"homogeneity"}, a likelihood ratio test where the model variables defined in
-#' \code{print} (factor) is compared to the constant model.
-#' Option \code{"trend"} tests if the linear trend of the continuous variable in
-#' \code{print} is significant (using model comparison).
-#' 
-#' 
-#' \strong{EAR: Excess Absolute Risk}
-#' 
-#' Excess Absolute Risk is a simple way to quantify the absolute difference between cohort risk and 
-#' population risk.
-#' Make sure that the person-years are calculated accordingly before using EAR. (when using mstate)
-#' 
-#' Formula for EAR:
-#' \deqn{EAR = \frac{observed - expected}{person years} \times 1000.}{EAR = (obs - exp)/pyrs * 1000.}
-#' 
-#' \strong{Data format}
-#' 
-#' The data should be given in tabulated format. That is the number of observations 
-#' and person-years are represented for each stratum.
-#' Note that also individual data is allowed as long as each observations, 
-#' person-years, and print and adjust variables are presented in columns.
-#' The extra variables and levels are reduced automatically before estimating SIRs. 
-#' Example of data format:
-#' 
-#' \tabular{rrrrr}{
-#'   sex \tab age \tab period \tab obs \tab pyrs \cr
-#'   0 \tab 1 \tab 2010 \tab 0 \tab 390 \cr
-#'   0 \tab 2 \tab 2010 \tab 5 \tab 385 \cr
-#'   1 \tab 1 \tab 2010 \tab 3 \tab 308 \cr
-#'   1 \tab 2 \tab 2010 \tab 12 \tab 315
-#' }
-#' 
-#' 
-#' @param coh.data aggregated cohort data, see e.g. \code{\link{lexpand}}
-#' @param coh.pyrs variable name for person years in cohort data; 
-#' quoted (as a string \code{'myvar'}) or unquoted (AKA as a name; \code{myvar})
-#' @param coh.obs variable name for observed cases; quoted or unquoted. A vector when using \code{mstata}.
-#' @param ref.data population data. Can be left NULL if \code{coh.data} 
-#' is stratified in \code{print}. See \code{\link{pophaz}} for details.
-#' @param ref.rate population rate variable (cases/person-years). Overwrites 
-#' arguments \code{ref.pyrs} and \code{ref.obs}. Quoted or unquoted
-#' @param ref.pyrs variable name for person-years in population data; quoted or unquoted
-#' @param ref.obs variable name for observed cases; quoted or unquoted
-#' @param subset logical condition to select data from \code{coh.data} before any computations
-#' @param adjust variable names for adjusting without stratifying output; quoted vector or unquoted list
-#' @param print variable names to stratify results; quoted vector or unquoted named list with functions
-#' @param mstate set column names for cause specific observations; quoted or unquoted. Relevant only
-#' when \code{coh.obs} length is two or more. See details.
-#' @param test.type Test for equal SIRs. Test available are 'homogeneity' and 'trend'.
-#' @param conf.type Confidence interval type: 'profile'(=default), 'wald' or 'univariate'.
-#' @param conf.level Level of type-I error in confidence intervals, default 0.05 is 95\% CI.
-#' @param EAR logical; TRUE calculates Excess Absolute Risks for univariate SIRs.
-#' (see details)
-
-#' 
-#' @examples 
-#' data(popmort)
-#' data(sire)
-#' c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-#'               breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-#'               aggre = list(fot, agegroup = age, year = per, sex) )
-#' ## SMR due other causes: status = 2
-#' se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs', 
-#'            ref.data = popmort, ref.rate = 'haz', 
-#'            adjust = c('agegroup', 'year', 'sex'), print = 'fot')
-#' se
-#' ## for examples see: vignette('sir')
-#' 
-#' 
-#' @seealso \code{\link{lexpand}}
-#' \href{../doc/sir.html}{A SIR calculation vignette}
-#' @family sir functions
-#' @family main functions
-#' 
-#' @return A sir-object that is a \code{data.table} with meta information in the attributes.
-#' 
-#' @export
-#' 
-#' @import data.table
-#' @import stats
-
-
-
-
-sir <- function( coh.data, 
-                 coh.obs,
-                 coh.pyrs,
-                 ref.data = NULL,
-                 ref.obs = NULL,
-                 ref.pyrs = NULL, ref.rate = NULL,
-                 subset = NULL,
-                 print = NULL,
-                 adjust = NULL,
-                 mstate = NULL,
-                 test.type = 'homogeneity',
-                 conf.type = 'profile',
-                 conf.level = 0.95,
-                 EAR = FALSE){
-
-  coh.data <- data.table(coh.data)
-  
-  ## subsetting---------------------------------------------------------------
-  ## no copy taken of data!
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = coh.data, substiset = subset)
-  coh.data <- coh.data[subset,]
-  
-  
-  # print list --------------------------------------------------------------
-  
-  # env1 <- environment() # set environment where to assign new print
-  # coh.data <- data_list(data = coh.data, arg.list = substitute(print), env = env1)
-  
-  mstate <- as.character(substitute(mstate))
-  if(length(mstate) == 0) {
-    mstate <- NULL
-  }
-  if(!is.null(mstate)) {
-    coh.data[,(mstate) := 0L] 
-  }
-  
-  # evalPopArg
-  coh.obs <- substitute(coh.obs)
-  c.obs <- evalPopArg(data = coh.data, arg = coh.obs)
-  coh.obs <- names(c.obs)
-  
-  coh.pyrs <- substitute(coh.pyrs)
-  c.pyr <- evalPopArg(data = coh.data, arg = coh.pyrs)
-  coh.pyrs <- names(c.pyr)
-  
-  print <- substitute(print)
-  c.pri <- evalPopArg(data = coh.data, arg = print)
-  print <- names(c.pri)
-
-  adjust <- substitute(adjust)
-  c.adj <- evalPopArg(data = coh.data, arg = adjust)
-  adjust <- names(c.adj)
-  
-  # collect data
-  coh.data <- cbind(c.obs, c.pyr)
-  if(!is.null(print))  coh.data <- cbind(coh.data, c.pri) 
-  if(!is.null(adjust)) coh.data <- cbind(coh.data, c.adj)
-  
-  if( !is.null(ref.data) ){
-    ref.obs <- as.character(substitute(ref.obs))
-    ref.pyrs <- as.character(substitute(ref.pyrs))
-    ref.rate <- as.character(substitute(ref.rate))
-    
-    if (length(ref.obs) == 0) ref.obs <- NULL
-    if (length(ref.pyrs) == 0) ref.pyrs <- NULL
-    if (length(ref.rate) == 0) ref.rate <- NULL
-  }
-
-
-  # print(coh.data)
-
-  st <- sir_table( coh.data = coh.data, 
-                   coh.obs = coh.obs,
-                   coh.pyrs = coh.pyrs,
-                   ref.data = ref.data,
-                   ref.obs = ref.obs,
-                   ref.pyrs = ref.pyrs, 
-                   ref.rate = ref.rate,
-                   print = print,
-                   adjust = adjust,
-                   mstate = mstate)
-
-  results <- sir_est( table = st,
-                      print = print,
-                      adjust = adjust,
-                      conf.type = conf.type, 
-                      test.type = test.type,
-                      conf.level = conf.level,
-                      EAR = EAR)
-  
-  ## final touch ---------------------------------------------------------------
-  
-
-  #setDT(data)
-  if (!return_DT()) {
-    for (i in 1:3) {
-      if (!is.null(results[[i]])) {
-        setDFpe(results[[i]])
-      }
-    }  
-  }
-  
-  data <- copy(results[[2]])
-  setattr(data, name = 'sir.meta', value = list(adjust = adjust,
-                                                print = print,
-                                                call = match.call(),
-                                                lrt.test= results$'lrt.test',
-                                                conf.type = results$'conf.type',
-                                                conf.level = conf.level,
-                                                lrt.test.type = results$'test.type',
-                                                pooled.sir = results[[1]]))
-  setattr(data, "class", c("sir", "data.table", "data.frame"))
-  return(data)
-}
-
-
-#' @title Estimate splines for SIR or SMR
-#' @author Matti Rantanen, Joonas Miettinen
-#' 
-#' @description Splines for standardised incidence or mortality ratio. A useful 
-#' tool to e.g. check whether a constant SIR can be assumed for all calendar periods,
-#' age groups or follow-up intervals. Splines can be fitted for these time dimensions
-#' separately or in the same model.
-#' 
-#' @param coh.data cohort data with observations and at risk time variables
-#' @param coh.pyrs variable name for person-years in cohort data
-#' @param coh.obs variable name for observed cases
-#' @param ref.data aggregated population data
-#' @param ref.rate population rate observed/expected. This overwrites the parameters
-#' \code{ref.pyrs} and \code{ref.obs}.
-#' @param ref.pyrs variable name for person-years in population data
-#' @param ref.obs variable name for observed cases
-#' @param subset logical condition to subset \code{coh.data} before any computations
-#' @param adjust variable names for adjusting the expected cases
-#' @param print variable names for which to estimate SIRs/SMRs and 
-#' associated splines separately 
-#' @param mstate set column names for cause specific observations. Relevant only
-#' when coh.obs length is two or more. See help for \code{sir}.
-#' @param spline variable name(s) for the splines
-#' @param knots number knots (vector),  pre-defined knots (list of vectors) or for optimal number of knots left NULL
-#' @param dependent.splines logical; if TRUE, all splines are fitted in same model.
-#' @param reference.points fixed reference values for rate ratios. If left \code{NULL}
-#' the smallest value is the reference point (where SIR = 1). 
-#' Ignored if \code{dependent.splines = FALSE}
-#' 
-#' 
-#' @details 
-#' 
-#' See \code{\link{sir}} for help on SIR/SMR estimation in general; usage of splines
-#' is discussed below.
-#' 
-#' \strong{The spline variables}
-#' 
-#' The model can include one, two or three splines variables.
-#' Variables can be included in the same model selecting \code{dependent.splines = TRUE}
-#' and SIR ratios are calculated (first one is the SIR, others SIR ratios). 
-#' Reference points vector can be set via \code{reference.points}
-#' where first element of the vector is the reference point for first ratio.
-#' 
-#' Variable(s) to fit splines are given as a vector in argument \code{spline}.
-#' Order will affect the results.
-#' 
-#' 
-#' \strong{dependent.splines} 
-#' 
-#' By default dependent.splines is FALSE and all splines are fitted in separate models. 
-#' If TRUE, the first variable in \code{spline} is a function of a SIR and other(s) are ratios.
-#' 
-#' \strong{knots}
-#' 
-#' There are three options to set knots to splines:
-#' 
-#' Set the number of knots for each spline variable with a \strong{vector}. 
-#' The knots are automatically placed to the quantiles of observed cases in cohort data. 
-#' The first and last knots are always the maximum and minimum values, so knot
-#' value needs to be at least two.
-#' 
-#' Predefined knot places can be set with a \strong{list} of vectors.
-#' The vector for each spline in the list specifies the knot places. The lowest 
-#' and the largest values are the boundary knots and these should be checked beforehand.
-#' 
-#' If \code{knots} is left \strong{NULL}, the model searches the optimal number 
-#' of knots by model AIC by fitting models iteratively from 2 to 15 knots and 
-#' the one with smallest AIC is selected.
-#' If \code{dependent.splines = TRUE}, the number of knots is searched by fitting each spline
-#' variable separately.
-#' 
-#' 
-#' \strong{print}
-#' 
-#' Splines can be stratified by the levels of variable given in \code{print}. If 
-#' \code{print} is a vector, only the first variable is accounted for. The knots 
-#' are placed globally for all levels of \code{print}. This also ensures that the likelihood 
-#' ratio test is valid.
-#' Splines are also fitted independently for each level of \code{print}.
-#' This allows for searching interactions, e.g. by fitting spline for period 
-#' (\code{splines='period'}) for each age group (\code{print = 'agegroup'}).
-#' 
-#' 
-#' \strong{p-values}
-#' 
-#' The output p-value is a test of whether the splines are equal (homogenous)
-#' at different levels of \code{print}. 
-#' The test is based on the likelihood ratio test, where the full model 
-#' includes \code{print} and is 
-#' compared to a null model without it.
-#' When \code{(dependent.splines = TRUE)} the p-value returned is a global p-value.
-#' Otherwise the p-value is spline-specific.
-#' 
-#' 
-#' @return A list of data.frames and vectors.
-#' Three spline estimates are named as \code{spline.est.A/B/C} and the corresponding values
-#' in \code{spline.seq.A/B/C} for manual plotting
-#' 
-#' 
-#' @seealso \code{\link{splitMulti}} 
-#' \href{../doc/sir.html}{A SIR calculation vignette}
-#' @family sir functions
-#' @family main functions
-#' 
-#' @export sirspline
-#' @import data.table 
-#' @import splines
-#' @import stats
-#' 
-#' @examples \donttest{
-#' ## for examples see: vignette('sir')
-#' }
-
-sirspline <- function( coh.data, 
-                       coh.obs,
-                       coh.pyrs,
-                       ref.data = NULL,
-                       ref.obs = NULL,
-                       ref.pyrs = NULL, 
-                       ref.rate = NULL,
-                       subset = NULL,
-                       print = NULL,
-                       adjust = NULL,
-                       mstate = NULL,
-                       spline,
-                       knots = NULL,
-                       reference.points = NULL,
-                       dependent.splines = TRUE){
-  
-  coh.data <- data.table(coh.data)
-  
-  ## subsetting-----------------------------------------------------------------
-  ## no copy taken of data!
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = coh.data, substiset = subset)
-  coh.data <- coh.data[subset,]
-  
-  # print list --------------------------------------------------------------
-
-  env1 <- environment()
-  coh.data <- data_list(data = coh.data, arg.list = substitute(print), env = env1)
-  
-  mstate <- as.character(substitute(mstate))
-  if(length(mstate) == 0) {
-    mstate <- NULL
-  }
-  if(!is.null(mstate)) {
-    coh.data[,(mstate) := 0L] 
-  }
-  
-  # evalPopArg
-  
-  spline <- substitute(spline)
-  c.spl <- evalPopArg(data = coh.data, arg = spline)
-  spline <- names(c.spl)
-  
-  coh.obs <- substitute(coh.obs)
-  c.obs <- evalPopArg(data = coh.data, arg = coh.obs)
-  coh.obs <- names(c.obs)
-  
-  coh.pyrs <- substitute(coh.pyrs)
-  c.pyr <- evalPopArg(data = coh.data, arg = coh.pyrs)
-  coh.pyrs <- names(c.pyr)
-  
-  print <- substitute(print)
-  c.pri <- evalPopArg(data = coh.data, arg = print)
-  print <- names(c.pri)
-  
-  adjust <- substitute(adjust)
-  c.adj <- evalPopArg(data = coh.data, arg = adjust)
-  adjust <- names(c.adj)
-
-  # collect data
-  coh.data <- cbind(c.obs, c.pyr, c.spl)
-  if(!is.null(print))  {
-    coh.data <- cbind(coh.data, c.pri[, print[!print %in% spline], with=FALSE])
-  }
-  if(!is.null(adjust)) {
-    coh.data <- cbind(coh.data, c.adj[, adjust[!adjust %in% spline], with=FALSE])
-  }
-  
-  if( !is.null(ref.data) ){
-    ref.obs <- as.character(substitute(ref.obs))
-    ref.pyrs <- as.character(substitute(ref.pyrs))
-    ref.rate <- as.character(substitute(ref.rate))
-    
-    if (length(ref.obs) == 0) ref.obs <- NULL
-    if (length(ref.pyrs) == 0) ref.pyrs <- NULL
-    if (length(ref.rate) == 0) ref.rate <- NULL
-  }
-  
-  st <- sir_table( coh.data = coh.data, 
-                   coh.obs = coh.obs,
-                   coh.pyrs = coh.pyrs,
-                   ref.data = ref.data,
-                   ref.obs = ref.obs,
-                   ref.pyrs = ref.pyrs, ref.rate = ref.rate,
-                   print = print,
-                   adjust = adjust,
-                   mstate = mstate,
-                   spline = spline)
-
-  results <- sir_spline( table = st, 
-                         print = print,
-                         adjust = adjust,
-                         spline = spline,
-                         knots = knots,
-                         reference.points = reference.points,
-                         dependent.splines = dependent.splines)
-  
-  setclass(results, c('sirspline', 'pe', class(results)))
-  return(results)  
-}
-
-
-
-
-
-
-# Input: two data.table:s
-# output: one data.table including rates
-#' @import stats
-#' @import data.table
-sir_table <- function( coh.data, 
-                       coh.obs,
-                       coh.pyrs,
-                       ref.data = NULL,
-                       ref.obs = NULL,
-                       ref.pyrs = NULL, 
-                       ref.rate = NULL,
-                       print = NULL,
-                       adjust = NULL,
-                       spline = NULL,
-                       mstate = NULL) {
-
-
-  # initial checks -------------------------------------------------
-  
-  if(is.null(ref.data)) {
-    if(is.null(print)){
-      stop('Both ref.data and print cannot be NULL.')
-    }
-    ref.data <- data.table(coh.data)
-    ref.obs <- coh.obs
-    ref.pyrs <- coh.pyrs
-  }
-  
-  coh.data <- data.table(coh.data)
-  ref.data <- data.table(ref.data)
-
-  vl <- unique( c(coh.pyrs, coh.obs, adjust, print) )
-  if( !is.null(mstate) )  {
-    vl <- vl[which( vl != mstate )]
-  }
-  all_names_present(coh.data, vl )
-  
-  if ( !is.null(ref.pyrs) & !is.null(ref.obs) ) {
-    all_names_present(ref.data, c(ref.pyrs, ref.obs, adjust))
-  }
-  
-  # Melt lexpand data -------------------------------------------------------
-  
-  if( length(coh.obs) > 1 ) {
-    if( is.null(mstate) ){
-      stop('coh.obs length is > 1. Set variable name for mstate.')
-    }
-    if( !mstate %in% names(ref.data) ){
-      warning('mstate variable name does not match names in ref.data.')
-    }
-
-    aggre <- unique(c(adjust, print, spline, coh.pyrs))
-    aggre <- aggre[which(aggre != mstate)]
-
-    coh.data <- melt( data = coh.data, id.vars = aggre, measure.vars = coh.obs, 
-                      value.name = 'coh.observations', 
-                      variable.name = mstate, variable.factor = FALSE)
-    coh.obs <- 'coh.observations'
-  
-    # parse Y name form string 'formXtoY'
-    q <- quote(
-      robust_values(substr(get(mstate), 
-                           start = regexpr( pattern = 'to', text = get(mstate) ) + 2, 
-                           stop  = nchar(x = get(mstate) )))
-      )
-    coh.data[,(mstate) := eval(q) ]
-    
-    if( !(mstate %in% adjust)) {
-      warning('Consider including mstate variable also in adjust. See help(sir) for details.')
-    }
-  }
-  
-  # prepare data steps, reduce dimensions -----------------------------------
-  
-  setnames(coh.data, c(coh.obs, coh.pyrs), c('coh.observations','coh.personyears'))  
-  
-  
-  coh.data <- expr.by.cj(data = coh.data,
-                         by.vars = unique( sort(c(adjust, print, spline)) ), 
-                         expr = list(coh.observations = sum(coh.observations), 
-                                     coh.personyears  = sum(coh.personyears)))
-  #coh.data <- na2zero(coh.data)
-  #coh.data <- na.omit(coh.data) 
-
-  coh.data[is.na(coh.observations), coh.observations := 0]
-  coh.data[is.na(coh.personyears), coh.personyears := 0]
-  coh.data <- na.omit(coh.data)
-  
-  # rates
-  if( !is.null(ref.rate) ){
-    setnames(ref.data, ref.rate, 'ref.rate')
-    ref.data <- expr.by.cj(data = ref.data, by.vars = c(adjust), 
-                           expr = list(ref.rate = mean(ref.rate)))
-  } else {
-    setnames(ref.data, c(ref.obs, ref.pyrs), c('ref.obs','ref.pyrs'))
-    ref.data <- expr.by.cj(data = ref.data, by.vars = c(adjust), 
-                           expr = list(ref.obs = sum(ref.obs),
-                                       ref.pyrs= sum(ref.pyrs)))
-    ref.data[, ref.rate := ref.obs / ref.pyrs ]
-  }
-  
-  # Merge
-  sir.table <- merge(coh.data, ref.data, by=c(adjust), all.x=TRUE)
-  sir.table[, expected := ref.rate * coh.personyears]
-  sir.table <- na2zero(sir.table)
-  
-  if ( !is.null(print) | !is.null(spline)){
-    sir.table <- sir.table[ ,list(observed = sum(coh.observations), 
-                                  expected = sum(expected),
-                                  pyrs = sum(coh.personyears)), 
-                           by = c(unique(c(print, spline)))]
-    setkeyv(sir.table, c(print, spline))
-  }
-  else {
-    sir.table <- sir.table[ ,list(observed = sum(coh.observations), 
-                                  expected = sum(expected),
-                                  pyrs = sum(coh.personyears))]
-  }
-  return(sir.table)
-}
-
-
-
-
-# Input: sir.table
-# Output: list of data.tables and values
-sir_est <- function( table,
-                     print = NULL,
-                     adjust = NULL,
-                     EAR = FALSE,
-                     test.type = 'homogeneity',
-                     conf.level = 0.95,
-                     conf.type = 'profile') {
-  pyrs <- NULL ## APPEASE R CMD CHECK
-  setDT(table)
-
-  if(!is.numeric(conf.level) | conf.level > 1) {
-    stop('Confidence level must be a numeric value between 0-1')
-  }
-  # function to SIR p-value
-  chi.p <- function(o, e) {
-    pchisq( ( (abs(o - e) - 0.5)^2)/e, df=1, lower.tail=FALSE)
-  }
-
-  # total sir
-  combined <- data.table(table)[,list(observed = sum(observed), 
-                             expected = sum(expected),
-                             pyrs = sum(pyrs))]
-  combined[ ,':='(sir = observed/expected,
-                  sir.lo = poisson.ci(observed, expected, conf.level=conf.level)[,4],
-                  sir.hi = poisson.ci(observed, expected, conf.level=conf.level)[,5],
-                  p_value  = chi.p(observed, expected))]
-  
-  # Poisson regression ------------------------------------------------------
-  
-  # write model formula
-  fa <- a <- NULL
-  sir.formula <- paste('observed ~ 1') 
-  if(!is.null(print)){
-    fa <- rev(print) #  fa <- print
-    
-    # drop variables with only one value
-    u <- c(t(table[, lapply(.SD, uniqueN), .SDcols = fa]))
-    if (length(u[u==1]) > 0){
-      message('Variable "', paste(fa[which(u==1)], collapse = '","'),'" (has only one level) removed from model.')
-      fa <- fa[-which(u==1)]
-    }
-    if(length(fa)>0){
-      # model formula
-      a <- paste0('as.factor(',paste( fa, collapse = '):as.factor('),')')
-      sir.formula <- paste('observed ~ 0 +', a)
-    }
-  }
-  # fit model if possible -----------------------------------------------------
-  
-  fit <- tryCatch(do.call("glm", list(formula = terms(as.formula(sir.formula), keep.order = FALSE), 
-                                      offset = log(table[,expected]), 
-                                      data = table, family = poisson(log))), 
-                  error=function(f) NULL )
-  
-  if(!is.null(fit)) eg <- expand.grid(fit$xlevels) # for further testing
-  
-  
-  # LRT test (homogeneity or trend) --------------------------------------------
-  
-  test.type <- match.arg(test.type, c('homogeneity','trend'))
-  
-  lrt_sig <- NULL  
-  if( sir.formula != 'observed ~ 1' & !is.null(fit) ) {
-    if (test.type == 'homogeneity') covariates <- a
-    if (test.type == 'trend') covariates <- paste(print, collapse=' + ')
-     
-    fit_full <- tryCatch(
-      do.call("glm", list(formula = terms(as.formula( paste0('observed ~ 1 + ', a) )), 
-                          offset = log(table[,expected]), 
-                          data = table, family=poisson(log))), 
-      error=function(f) NULL )
-    
-    fit_null <- tryCatch(
-      do.call("glm", list(formula = terms(as.formula('observed ~ 1') ), 
-                          offset = log(table[,expected]), 
-                          data = table, family=poisson(log))), 
-      error=function(f) NULL )
-    
-    if (!is.null(fit_full)){
-      lrt <- anova(fit_full, fit_null, test = 'Chisq')
-      lrt_sig <- lrt[['Pr(>Chi)']][2]    
-    }
-  }
-  
-  # confidence intervals ----------------------------------------------------
-  
-  conf.type <- match.arg(conf.type, c('wald','profile','univariate'))
-  ci.info <- NULL
-  ci <- NULL
-  
-  if (is.null(fit) & conf.type %in% c('wald','profile')) {
-    conf.type <- 'univariate'
-    ci.info <- 'Model fitting failed. Univariate confidence intervals selected.'
-    if(any(table$expected == 0)) {
-      ci.info <- paste(ci.info, '(zero values in expected)')
-    }
-  }
-  
-  if (conf.type == 'profile') {
-    
-    confint_glm <- function(object, parm, level = 0.95, trace = FALSE, ...) {
-      pnames <- names(coef(object))
-      if (missing(parm)) {
-        parm <- seq_along(pnames)
-      }
-      else if (is.character(parm)) {
-        parm <- match(parm, pnames, nomatch = 0L)
-      }
-      object <- profile(object, which = parm, alpha = (1 - level)/4,  trace = trace)
-      confint(object, parm = parm, level = level, trace = trace, ...)
-    }
-
-    ci <- suppressMessages( suppressWarnings( 
-      tryCatch(exp(confint_glm(fit, level=conf.level)), error=function(e) NULL )
-    ))
-    if(!is.null(ci)) {
-      ci <- as.data.table(ci)
-      if (is.null(print) | length(fa)==0) ci <- data.table(t(ci)) # transpose if only one row
-    } else {
-      conf.type <- 'wald'
-      ci.info <- 'Could not solve profile-likelihood. Wald confidence intervals selected.'
-    }
-  }
-  
-  if (conf.type == 'wald') { 
-    ci <- data.table( exp(confint.default(fit)) )
-  }
-  
-  if(conf.type == 'univariate') {
-    ci <- data.table(poisson.ci(table$observed, table$expected, conf.level = conf.level))[,.(lower, upper)]
-    pv <- chi.p(table$observed, table$expected)
-  } else {
-    pv <- as.vector(summary(fit)$coef[, "Pr(>|z|)"]) 
-  }
-  if(!is.null(ci.info)) message(ci.info)
-
-  # collect results -----------------------------------------------------
-  
-  setnames(ci, 1:2, c('sir.lo','sir.hi'))
-  
-  table[, ':=' ( sir = observed/expected,
-              sir.lo = ci[, sir.lo],
-              sir.hi = ci[, sir.hi],
-              p_value = round(pv,5))]
-  
-
-  # Round results -----------------------------------------------------------
-  
-  cols1 <- c('sir','sir.lo','sir.hi','expected','pyrs')
-  
-  table[,(cols1) := lapply(.SD, round, digits=4), .SDcols=cols1]
-  combined[,(cols1) := lapply(.SD, round, digits=4), .SDcols=cols1]
-
-  
-  # tests -----------------------------------
-  
-  if (table[!is.na(sir) & (sir < sir.lo | sir > sir.hi), .N] > 0) {
-    warning('There is something wrong with confidence intervals')
-  }
-  if (table[!is.na(sir.lo) & !is.na(sir.hi)][sir.lo > sir.hi, .N] > 0) {
-    warning('CIs might be incorrect')
-  }
-
-  if(!is.null(fit) & length(fa)>0) {
-    # pseudo test if the modelled confidence intervals are merged correctly:
-    t1 <- copy(table)[,lapply(.SD, factor),.SDcols = fa]
-    if(any(t1 != data.table(eg))) {
-      message('CIs levels might not match. Contact the package maintainer and use univariate CIs.')
-    }
-  }
-  
-  # EAR -----------------------------------------------------------------
-  if (EAR) {
-    table[,EAR := round((observed - expected)/pyrs * 1000, 3)]
-  }
-  
-  
-  results <- list(total = combined, 
-                  table = table, 
-                  adjusted = adjust,
-                  lrt.test = lrt_sig,
-                  test.type = test.type,
-                  conf.type = conf.type,
-                  ci.info = ci.info)
-  return(results)
-}
-
-
-#' @export
-getCall.sir <- function (x, ...) {
-  attributes(x)$sir.meta$call
-}
-
-
-# Input: sir.table
-# Output: estimates and sequences for plotting splines
-#' @import splines
-#' @import data.table
-#' @import stats
-sir_spline <- function(  table,
-                         print = NULL,
-                         adjust = NULL,
-                         spline,
-                         knots = NULL,
-                         reference.points = NULL,
-                         dependent.splines = TRUE){
-  knts <- 
-    spline.seq.A <- 
-    spline.seq.B <- 
-    spline.seq.C <- 
-    spline.est.A <- 
-    spline.est.B <- 
-    spline.est.C <- NULL
-  
-  if (!is.null(knots) & length(knots) != length(spline) ) {
-    stop('Arguments spline and knots has to be same length.')
-  }
-  
-  
-  # Spline functions -------------------------------------------------------
-  
-  # function to get spline seq
-  spline.seq <- function(data, spline.var=NULL) {
-    # palauttaa jotaina
-    if(is.na(spline.var)) {
-      return(NULL)
-    }
-    spline.seq <- seq( min( data[,get(spline.var)] ), 
-                       max( data[,get(spline.var)] ), length.out = 100)
-    return(spline.seq)
-  }
-  
-  # function to search optimal number of knots by AIC
-  spline.knots <- function(data, knots = NULL, spline.vars = NULL){    
-    # search optimal number of knots
-    if( is.null(knots) ) {
-      knts <- list()
-      for (jj in 1:length(spline.vars)) {
-        # reduce data to fit model
-        data0 <- data[,list(observed=sum(observed), expected = sum(expected)), by = eval(spline.vars[jj])] 
-        data0 <- data0[expected > 0]
-        spline.fit <- glm(observed ~ 1, offset=log(expected), family=poisson(log), data = data0)
-        aic0 <- summary(spline.fit)[['aic']]
-        limit <- 20
-        ii <- 2
-        while(  ii < limit ){
-          tmp.knots <- ii
-          knts[jj] <- list( data0[ ,quantile( rep(get(spline.vars[jj]),observed), probs = seq(0,100,length.out = tmp.knots)/100)] )
-          spline.fit <- glm(observed ~ Ns(get(spline.vars[jj]), knots = knts[[jj]]), offset=log(expected), family=poisson(log), data=data0)
-          aic0 <- c(aic0, summary(spline.fit)[['aic']])
-          ii <- ii + 1
-        }
-        tmp.knots <- which(aic0 == min(aic0))[1]
-        if(tmp.knots == 1) {
-          message(paste0('Null model better than spline in ', jj))
-          tmp.knots <- 2
-        }
-        knts[jj] <- list(data0[ ,quantile( rep(get(spline.vars[jj]),observed), probs = seq(0,100,length.out = tmp.knots)/100)])
-        rm(tmp.knots)
-      }
-      knots <- unlist(lapply(knts, length))
-    }
-    else {
-      # knot predefined
-      if( is.list(knots) ){
-        knts <- knots
-        knots <- unlist(lapply(knots, length))
-      } 
-      # knot number predefined
-      else {
-        if( any(knots < 2) ) { 
-          message('Min knots number set to 2.') 
-          knots[knots < 2] <- 2
-        }
-        knts <- list()
-        for(i in 1:length(knots)) {
-          knts[i] <- list( data[ ,quantile( rep(get(spline.vars[i]), observed), probs = seq(0,100,length.out = knots[i])/100)])
-        }
-      }
-    }
-    names(knts) <- spline.vars
-    return(knts)
-  }
-  
-  # function to estimate 2-3 dim splines in same model
-  spline.estimates.dep <- function(sir.spline = sir.spline,
-                                   spline.seq.A = spline.seq.A,
-                                   spline.seq.B = spline.seq.B,
-                                   spline.seq.C = spline.seq.C,
-                                   reference.points = reference.points,
-                                   knts = knts
-  ){
-    
-    if( all(!is.null(reference.points), (length(reference.points) + 1) != length(spline)) ){
-      stop('Parameter reference.points length should be length of spline - 1.')
-    }
-    
-    
-    form <- 'Ns(get(spline[[1]]), kn=knts[[1]])'
-    nsA <- Ns( spline.seq.A, knots = knts[[1]])
-    if ( length(spline) >= 2) {
-      form <- paste0(form, ' + Ns(get(spline[[2]]), kn=knts[[2]])')
-      nsB <- Ns( spline.seq.B, knots = knts[[2]])
-    }
-    if ( length(spline) == 3) {
-      form <- paste0(form, ' + Ns(get(spline[[3]]), kn=knts[[3]])')
-      nsC <- Ns( spline.seq.C, knots = knts[[3]])
-    }
-    
-    form <- paste0('observed ~ ', form)
-    spline.fit <- do.call("glm", list(formula = as.formula(form),
-                                      offset = log(sir.spline[expected > 0,expected]),
-                                      family = poisson,
-                                      data = sir.spline[expected>0]))
-    if( any( ci.exp(spline.fit)[,1] == 1) ){
-      message("NA's in spline estimates.")
-    }
-    
-    aic <- summary(spline.fit)[['aic']]
-    
-    rf.C <- rf.B <- NA
-    # set assigned reference points or get minimum values
-    if( !is.null(reference.points) ) {
-      rf.B <- reference.points[1]
-      rf.C <- reference.points[2]
-    } 
-    else {
-      rf.B <- min( sir.spline[,get(spline[2])] )
-      if(!is.na(spline[3])) {
-        rf.C <- min( sir.spline[,get(spline[3])] )
-      }
-    }
-    
-    if( !is.na(rf.B) )  {
-      B <- Ns( rep(rf.B, 100), knots = knts[[2]])
-      if( findInterval(rf.B, range(sir.spline[,get(spline[2])])) != 1 ) {
-        message("WARNING: reference point 2 doesn't fall into spline variable interval")
-      }
-    }
-    
-    if( !is.na(rf.C) ){
-      C <- Ns( rep(rf.C, 100), knots = knts[[3]])
-      if( findInterval(rf.C, range(sir.spline[,get(spline[3])])) != 1) {
-        message("WARNING: reference point 3 doesn't fall into spline variable interval")
-      }
-    } 
-    
-    # make subset of model parameters
-    if( !is.null(knts[2]) ) {
-      sub.B <- which( grepl('spline[[2]]', names(spline.fit$coefficients),fixed = TRUE) )
-    }
-    if( !is.null(knts[3]) ) {
-      sub.C <- which( grepl('spline[[3]]', names(spline.fit$coefficients),fixed = TRUE) )
-    }
-    if ( length(spline) == 2) {
-      spline.est.A <- ci.exp(spline.fit, ctr.mat = cbind(1, nsA, nsB))
-      spline.est.B <- ci.exp(spline.fit, subset = sub.B, ctr.mat = nsB - B)
-      spline.est.C <- NULL      
-    }
-    if ( length(spline) == 3) {
-      spline.est.A <- ci.exp(spline.fit, ctr.mat = cbind(1, nsA, nsB, nsC))
-      spline.est.B <- ci.exp(spline.fit, subset= sub.B, ctr.mat = nsB - B)
-      spline.est.C <- ci.exp(spline.fit, subset= sub.C, ctr.mat = nsC - C)
-    }
-    list(a = spline.est.A, 
-         b = spline.est.B, 
-         c = spline.est.C)
-  }
-  
-  # function to estimate independet splines
-  spline.estimates.uni <- function(data, spline.var, spline.seq, knots, knum) {  
-    if(is.na(spline.var)) return(NULL)
-    knots <- knots[[knum]]
-    data <- data[,list(observed=sum(observed), expected = sum(expected)), by = eval(spline.var)][expected > 0]
-    spline.uni <- glm(observed ~ Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data)
-    nsx <- Ns( spline.seq, knots = knots)
-    spline.est <- ci.exp(spline.uni, ctr.mat = cbind(1, nsx))
-    spline.est
-  }
-  
-  
-  
-  # Poisson regression Splines -------------------------------------------------
-  
-  sir.spline <- data.table(table)
-  
-  # convert spline variables to numeric
-  temp.fun <- function(x){
-    as.numeric(as.character(x))
-  }
-  sir.spline[, (spline) := lapply(.SD, temp.fun), .SDcols = spline]
-  
-  
-  
-  # set knots
-  knts <- spline.knots(data=sir.spline, knots = knots, spline.vars = spline)  
-  
-  # set sequences
-  spline.seq.A <- spline.seq(data=sir.spline, spline.var=spline[1])
-  spline.seq.B <- spline.seq(data=sir.spline, spline.var=spline[2])
-  spline.seq.C <- spline.seq(data=sir.spline, spline.var=spline[3])
-  
-  if( length(spline) == 1 ) {
-    dependent.splines <- FALSE
-  }
-  
-  # convert print to factor
-  print <- print[1]
-  
-  # loop for each level of print:
-  if( !is.null(print) ) {
-    prnt.levels <- sir.spline[,unique( get(print) )]
-    sir.spline[,(print) := factor(get(print))]
-  }
-  else {
-    print <- 'temp'
-    sir.spline[,temp := 1]
-    prnt.levels <- 1
-  }
-  
-  spline.est.A <- NULL
-  spline.est.B <- NULL
-  spline.est.C <- NULL
-  
-  for(i in prnt.levels){
-    if( dependent.splines ) {
-      out <- spline.estimates.dep(sir.spline = sir.spline[get(print) == i],
-                                  spline.seq.A = spline.seq.A,
-                                  spline.seq.B = spline.seq.B,
-                                  spline.seq.C = spline.seq.C,
-                                  reference.points = reference.points,
-                                  knts = knts)
-      est.A <- out[['a']]
-      est.B <- out[['b']]
-      est.C <- out[['c']]
-    }
-    else{
-      est.A <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[1], spline.seq = spline.seq.A, knots = knts, knum = 1)
-      est.B <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[2], spline.seq = spline.seq.B, knots = knts, knum = 2)  
-      est.C <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[3], spline.seq = spline.seq.C, knots = knts, knum = 3)
-    }
-    
-    add_i <- function(est.x, i){
-      if(is.null(est.x)) {
-        return(NULL)
-      }
-      cbind(i, data.frame(est.x))
-    }
-    
-    
-    est.A <- add_i(est.A, i)
-    est.B <- add_i(est.B, i)
-    est.C <- add_i(est.C, i)
-    
-    spline.est.A <- rbind(spline.est.A, est.A)
-    spline.est.B <- rbind(spline.est.B, est.B)
-    spline.est.C <- rbind(spline.est.C, est.C)
-  }
-  
-  # get p-value and anova-table
-  anovas <- NULL
-  p <- NULL
-  if(dependent.splines) {
-    form.a <- 'Ns(get(spline[[1]]), kn=knts[[1]]) + Ns(get(spline[[2]]), kn=knts[[2]])'
-    form.b <- 'get(print):Ns(get(spline[[1]]), kn=knts[[1]]) + get(print):Ns(get(spline[[2]]), kn=knts[[2]])'
-    if ( length(spline) == 3) {
-      form.a <- paste0(form.a, ' + Ns(get(spline[[3]]), kn=knts[[3]])')
-      form.b <- paste0(form.b, ' + get(print):Ns(get(spline[[3]]), kn=knts[[3]])')
-    }
-    
-    fit.fun <- function( form.string ){
-      do.call("glm", list(formula = as.formula( form.string ),
-                          offset = log(sir.spline[expected > 0,expected]),
-                          family = poisson,
-                          data = sir.spline[expected>0]))
-    }
-    
-    fit.1 <- fit.fun( paste0('observed ~ ', form.a) )
-    fit.2 <- fit.fun( paste0('observed ~ ', 'get(print)+', form.a))
-    fit.3 <- fit.fun( paste0('observed ~ ', form.b))
-    fit.4 <- fit.fun( paste0('observed ~ ', 'get(print)+', form.b) )
-    
-    global.p<- anova(fit.4, fit.1, test='LRT')
-    level.p <- anova(fit.2, fit.1, test='LRT')
-    #shape.p <- anova(fit.4, fit.3, test='LRT')
-    
-    anovas <- list(global.p = global.p, level.p = level.p)
-    p <- rbind(global.p[['Pr(>Chi)']][2], level.p[['Pr(>Chi)']][2]) # , shape.p, 
-  }
-  else {    
-    lrt.uni <- function(data=sir.spline, spline.var=spline[1], print=print, knots=knts, knum = 1) {
-      if (is.na(spline.var)) return (NULL)
-      data <- data.table(data)
-      knots <- knots[[knum]]
-      fit0 <- glm(observed ~ get(print)+Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
-      fit1 <- glm(observed ~ Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
-      fit2 <- glm(observed ~ get(print)*Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
-      anova(fit2,fit1,fit0, test='Chisq') # [['Pr(>Chi)']][2]
-    }
-    
-    var1.p <- lrt.uni(spline.var = spline[1], print=print, knots=knts, knum = 1)
-    var2.p <- lrt.uni(spline.var = spline[2], print=print, knots=knts, knum = 2)
-    var3.p <- lrt.uni(spline.var = spline[3], print=print, knots=knts, knum = 3)
-    
-    p <- list(spline.a = var1.p[['Pr(>Chi)']][2], 
-              spline.b = var2.p[['Pr(>Chi)']][2], 
-              spline.c = var3.p[['Pr(>Chi)']][2])
-    anovas <- list(spline.a = var1.p, spline.b = var2.p, spline.c = var3.p)
-  }
-  
-  output <- list( spline.est.A = spline.est.A,
-                  spline.est.B = spline.est.B,
-                  spline.est.C = spline.est.C,
-                  spline.seq.A = spline.seq.A,
-                  spline.seq.B = spline.seq.B,
-                  spline.seq.C = spline.seq.C,
-                  adjust = adjust,
-                  print = print,
-                  spline = spline,
-                  anovas = anovas,
-                  knots = knts,
-                  spline.dependent = dependent.splines,
-                  p.values = p)
-  output
-}
-
-# input data and argument list. replaces print in upper environment with name a vector.
-data_list <- function( data, arg.list, env ) {
-  if(missing(env)){
-    arg.list <- substitute(arg.list)
-    env <- parent.frame()
-  }
-  d <- data.table(data)
-  
-  l <- eval(arg.list, envir = d, enclos = parent.frame()) 
-  
-  if( is.list( l ) ) {
-    n <- intersect(names(l), names(d))
-    if(length(n)>0){
-      d[,(n) := NULL]
-    }
-    #     if(is.null(names(l))) {
-    #       v <- 1:length(l)
-    #       setnames(l, v, paste0('V', v))
-    #     }
-    l <- as.data.table(l)
-    l <- data.table(l)
-    assign('print', colnames(l), envir = env) # set names to parent environment
-    if( ncol(d) > 0) {
-      l <- data.table(d, l)
-    }
-    return(l)
-  } else { 
-    return(data) 
-  }
-}
-
-#' @export
-coef.sir <- function(object, ...) {
-  factors <- attr(object, 'sir.meta')$print
-  
-  q <- paste("paste(",paste(factors,collapse=","),", sep = ':')")
-  q <- parse(text=q)
-  n <- object[,eval(q)]
-  
-  res <- object$sir
-  attr(res, 'names') <- n
-  
-  res
-}
-
-
-
-
-#' @export
-confint.sir <- function(object, parm, level = 0.95, conf.type = 'profile', 
-                        test.type = 'homogeneity', ...) {
-
-  meta <- attr(object, 'sir.meta')
-  object <- copy(object)
-  object <- sir_est(table = object,
-                    print = meta$print,
-                    adjust = NULL,
-                    conf.type = conf.type, 
-                    test.type = test.type,
-                    conf.level = level,
-                    EAR = FALSE)
-  object <- object$table
-  q <- paste("paste(",paste(meta$print,collapse=","),", sep = ':')")
-  q <- parse(text=q)
-  n <- object[,eval(q)]
-
-  res <- cbind(object$sir.lo, object$sir.hi)
-  
-  rownames(res) <- n
-  colnames(res) <- paste( c( (1-level)/2*100, (1 - (1-level)/2)*100), '%')
-
-  res
-}
-
-
-#' @title Calculate SMR
-#' @author Matti Rantanen
-#' @description Calculate Standardized Mortality Ratios (SMRs) using 
-#' a single data set that includes
-#' observed and expected cases and additionally person-years.
-#' 
-#' @details These functions are intended to calculate SMRs from a single data set 
-#' that includes both observed and expected number of cases. For example utilizing the
-#' argument \code{pop.haz} of the \code{\link{lexpand}}.
-#' 
-#' \code{sir_lex} automatically exports the transition \code{fromXtoY} using the first
-#' state in \code{lex.Str} as \code{0} and all other as \code{1}. No missing values
-#' is allowed in observed, pop.haz or person-years.
-#' 
-#' @param x Data set e.g. \code{aggre} or \code{Lexis} object 
-#' (see: \code{\link{lexpand}})
-#' @param obs Variable name of the observed cases in the data set
-#' @param exp Variable name or expression for expected cases
-#' @param pyrs Variable name for person-years (optional)
-#' @param print Variables or expression to stratify the results
-#' @param test.type Test for equal SIRs. Test available are 'homogeneity' and 'trend'
-#' @param conf.level Level of type-I error in confidence intervals, default 0.05 is 95\% CI
-#' @param conf.type select confidence interval type: (default=) `profile`, `wald`, `univariate`
-#' @param subset a logical vector for subsetting data
-#' 
-#' @seealso \code{\link{lexpand}}
-#' \href{../doc/sir.html}{A SIR calculation vignette}
-#' @family sir functions
-#' 
-#' @return A sir object
-#' 
-#' @examples 
-#' 
-#' \donttest{
-#' BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-#' 
-#' ## Aggregated data
-#' x1 <- lexpand(sire, breaks = BL, status = status != 0, 
-#'               birth = bi_date, entry = dg_date, exit = ex_date,
-#'               pophaz=popmort,
-#'               aggre=list(sex, period = per, surv.int = fot))
-#' sir_ag(x1, print = 'period')
-#'
-#'
-#' # no aggreate or breaks
-#' x2 <- lexpand(sire, status = status != 0, 
-#'               birth = bi_date, entry = dg_date, exit = ex_date,
-#'               pophaz=popmort)
-#' sir_lex(x2, breaks = BL, print = 'per')
-#' }
-#' 
-#' @import data.table
-#' @import stats
-#' @export
-sir_exp <- function(x, obs, exp, pyrs=NULL, print = NULL, 
-                    conf.type = 'profile', test.type = 'homogeneity',
-                    conf.level = 0.95, subset = NULL) {
-  
-  # subsetting
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data = x, substiset = subset)
-  x <- x[subset,]
-  
-  # evalPopArg
-  obs <- substitute(obs)
-  c.obs <- evalPopArg(data = x, arg = obs)
-  obs <- names(c.obs)
-  
-  
-  print <- substitute(print)
-  c.pri <- evalPopArg(data = x, arg = print)
-  print <- names(c.pri)
-  
-  exp <- substitute(exp)
-  c.exp <- evalPopArg(data = x, arg = exp)
-  exp <- names(c.exp)
-  
-  pyrs <- substitute(pyrs)
-  c.pyr <- evalPopArg(data = x, arg = pyrs)
-  if(is.null(c.pyr)) c.pyr <- data.table(pyrs=0)
-  pyrs <- names(c.pyr)
-  
-  # collect data
-  x <- cbind(c.obs, c.pyr, c.exp)
-  if(any(is.na(x))) stop('Missing values in expected cases.')
-  if(!is.null(print))  x<- cbind(x, c.pri) 
-  
-  express <- paste0('list(observed = sum(', obs, '), expected = sum(',exp,'), pyrs = sum(', pyrs,'))')
-  # aggregate
-  es <- parse(text = express)
-  y <- x[, eval(es), keyby = print] # keyby is must
-  
-  results <- sir_est( table = y,
-                      print = print,
-                      adjust = NULL,
-                      conf.type = conf.type, 
-                      test.type = test.type,
-                      conf.level = conf.level,
-                      EAR = FALSE)
-  
-  #setDT(data)
-  if (!return_DT()) {
-    for (i in 1:2) {
-      if (!is.null(results[[i]])) {
-        setDFpe(results[[i]])
-      }
-    }  
-  }
-  
-  data <- copy(results[[2]])
-  setattr(data, name = 'sir.meta', value = list(adjust = NULL,
-                                                print = print,
-                                                call = match.call(),
-                                                lrt.test= results$'lrt.test',
-                                                conf.type = results$'conf.type',
-                                                conf.level = conf.level,
-                                                lrt.test.type = results$'test.type',
-                                                pooled.sir = results[[1]]))
-  setattr(data, "class", c("sir", "data.table", "data.frame"))
-  return(data)
-}
-
-
-
-#' Calculate SMRs from a split Lexis object  
-#' 
-#' @description \code{sir_lex} solves SMR from a \code{\link{Lexis}} object 
-#' calculated with \code{lexpand}.
-#' 
-#' @param breaks a named list to split age group (age), period (per) or follow-up (fot). 
-#' @param ... pass arguments to \code{sir_exp}
-#' 
-#' 
-#' @describeIn sir_exp
-#' 
-#' @export
-
-sir_lex <- function(x, print = NULL, breaks = NULL, ... ) {
-  
-  ## R CMD CHECK appeasement
-  lex.dur <- NULL
-  
-  if(!inherits(x, 'Lexis')) {
-    stop('x has to be a Lexis object (see lexpand or Lexis)')
-  }
-  if(!"pop.haz" %in% names(x)) {
-    stop("Variable pop.haz not found in the data.")
-  }
-
-
-  # reformat date breaks
-  if(!is.null(breaks)) {
-    breaks <- lapply(breaks, function(x) {
-      if(is.character(x)) c(cal.yr(as.Date(x)))
-      else x
-    })
-  }
-  
-  print <- substitute(print)
-  # copy to retain the attributes
-  x <- copy(x)
-  
-  # guess the first value
-  first_value <- lapply(c("lex.Cst", "lex.Xst"), function(var) {
-    if (is.factor(x[[var]])) levels(x[[var]]) else sort(unique(x[[var]]))
-  })
-  first_value <- unique(unlist(first_value))[1]
-  
-  col <- x$lex.Xst
-  set(x, j = "lex.Cst", value = 0L)
-  set(x, j = "lex.Xst", value = ifelse(col == first_value, 0L, 1L))
-  
-  if(!is.null(breaks)) {
-    x <- splitMulti(x, breaks = breaks)
-  }
-  
-  a <- copy(attr(x, "time.scales"))
-  a <- a[!vapply(get_breaks(x), is.null, logical(1))]
-  x[, d.exp := pop.haz*lex.dur]
-  
-  TF <- environment()
-  
-  if(any(is.na(x[,d.exp]))) stop('Missing values in either pop.haz or lex.dur.')
-  x <- aggre(x, by = TF$a, sum.values = 'd.exp')
-  if(!'from0to1' %in% names(x)) {
-    stop('Could not find any transitions between states in lexis')
-  }
-  x <- sir_exp(x = x, obs = 'from0to1', print = print, exp = 'd.exp', pyrs = 'pyrs', ...)
-  # override the match.call from sir_exp 
-  attr(x, 'sir.meta')$call <- match.call()
-  return(x)
-}
-
-
-#' SMR method for an \code{aggre} object.
-#' 
-#' @description \code{sir_ag} solves SMR from a \code{\link{aggre}} object 
-#' calculated using \code{\link{lexpand}}.
-#' 
-#' @describeIn sir_exp
-#' 
-#' @export
-
-sir_ag <- function(x, obs = 'from0to1', print = attr(x, 'aggre.meta')$by, exp = 'd.exp', pyrs = 'pyrs', ... ) {
-  
-  if(!inherits(x, 'aggre')) {
-    stop('x should be an aggre object (see lexpand or sir_lex)')
-  }
-  obs <- substitute(obs)
-  print <- substitute(print)
-  
-  x <- copy(x)
-  x <- sir_exp(x = x, obs = obs, print = print, exp = 'd.exp', pyrs = 'pyrs', ...) # original
-  attr(x, 'sir.meta')$call <- match.call() # override the call from sir_exp
-  x
-}
-
-
-
-globalVariables(c('observed','expected','p_adj','p_value','temp','coh.observations','coh.personyears',
-                  'd.exp', 'lower', 'pop.haz', 'sir.hi','sir.lo','upper'))
-
+#' @title Calculate SIR or SMR
+#' @author Matti Rantanen, Joonas Miettinen
+#' @description Poisson modelled standardised incidence or mortality ratios (SIRs / SMRs) i.e. 
+#' indirect method for calculating standardised rates. SIR is a ratio of observed and expected cases.
+#' Expected cases are derived by multiplying the strata-specific population rate with the
+#' corresponding person-years of the cohort.
+#' 
+#' @details \code{sir} is a comprehensive tool for modelling SIRs/SMRs with flexible 
+#' options to adjust and print SIRs, test homogeneity and utilize 
+#' multi-state data. The cohort data and the variable names for observation 
+#' counts and person-years are required.
+#' The reference data is optional, since the cohort data 
+#' can be stratified (\code{print}) and compared to total.
+#' 
+#' 
+#' \strong{Adjust and print}
+#' 
+#' A SIR can be adjusted or standardised using the covariates found in both \code{coh.data} and \code{ref.data}.
+#' Variable to adjust are given in \code{adjust}.
+#' Variable names needs to match in both \code{coh.data} and \code{ref.data}. 
+#' Typical variables to adjust by are gender, age group and calendar period.
+#' 
+#' \code{print} is used to stratify the SIR output. In other words, the variables 
+#' assigned to \code{print} are the covariates of the Poisson model.
+#' Variable levels are treated as categorical.
+#' Variables can be assigned in both \code{print} and \code{adjust}. 
+#' This means the output it adjusted and printed by these variables.
+#' 
+#' \code{print} can also be a list of expressions. This enables changing variable 
+#' names or transforming variables with functions such as \code{cut} and \code{round}.
+#' For example, the existing variables \code{agegroup} and \code{year} could be
+#' transformed to new levels using \code{cut} by
+#' 
+#' \code{print = list( age.category = cut(agegroup, breaks = c(0,50,75,100)), 
+#' year.cat = cut(year, seq(1950,2010,20)))}
+#' 
+#' 
+#' \strong{ref.rate or ref.obs & ref.pyrs}
+#' 
+#' The population rate variable can be given to the \code{ref.rate} parameter. 
+#' That is, when using e.g. the \code{popmort} or a comparable data file, one may
+#' supply \code{ref.rate} instead of \code{ref.obs} and \code{ref.pyrs}, which
+#' will be ignored if \code{ref.rate} is supplied. 
+#' 
+#' 
+#' Note that if all the stratifying variables in 
+#' \code{ref.data} are not listed in \code{adjust}, 
+#' or when the categories are otherwise combined,
+#' the (unweighted) mean of rates is used for computing expected cases.
+#' This might incur a small bias in comparison to when exact numbers of observations
+#' and person-years are available. 
+#' 
+#' 
+#' 
+#' \strong{mstate}
+#' 
+#' E.g. using \code{lexpand} it's possible to compute counts for several outcomes
+#' so that the population at risk is same for each 
+#' outcome such as a certain kind of cancer. 
+#' The transition counts are in wide data format, 
+#' and the relevant columns can be supplied to \code{sir}
+#' in a vector via the \code{coh.obs} argument. 
+#' The name of the corresponding new column in \code{ref.data} is given in
+#' \code{mstate}. It's recommended to include the \code{mstate} variable in \code{adjust},
+#' so the corresponding information should also be available in \code{ref.data}.
+#' More examples in sir-vignette.
+#' 
+#' This approach is analogous to where SIRs are calculated separately their 
+#' own function calls.
+#' 
+#' 
+#' \strong{Other parameters}
+#' 
+#' \code{univariate} confidence intervals are calculated using exact 
+#' Poisson intervals (\code{poisson.ci}). The options \code{profile} and \code{wald} are
+#' is based on a Poisson regression model: profile-likelihood confidence intervals 
+#' or Wald's normal-approximation. P-value is Poisson model based \code{conf.type}
+#' or calculated using the method described by Breslow and Day. Function automatically
+#' switches to another \code{conf.type} if calculation is not possible with a message.
+#' Usually model fit fails if there is print stratum with zero expected values.
+#' 
+#' 
+#' The LRT p-value tests the levels of \code{print}. The test can be either 
+#' \code{"homogeneity"}, a likelihood ratio test where the model variables defined in
+#' \code{print} (factor) is compared to the constant model.
+#' Option \code{"trend"} tests if the linear trend of the continuous variable in
+#' \code{print} is significant (using model comparison).
+#' 
+#' 
+#' \strong{EAR: Excess Absolute Risk}
+#' 
+#' Excess Absolute Risk is a simple way to quantify the absolute difference between cohort risk and 
+#' population risk.
+#' Make sure that the person-years are calculated accordingly before using EAR. (when using mstate)
+#' 
+#' Formula for EAR:
+#' \deqn{EAR = \frac{observed - expected}{person years} \times 1000.}{EAR = (obs - exp)/pyrs * 1000.}
+#' 
+#' \strong{Data format}
+#' 
+#' The data should be given in tabulated format. That is the number of observations 
+#' and person-years are represented for each stratum.
+#' Note that also individual data is allowed as long as each observations, 
+#' person-years, and print and adjust variables are presented in columns.
+#' The extra variables and levels are reduced automatically before estimating SIRs. 
+#' Example of data format:
+#' 
+#' \tabular{rrrrr}{
+#'   sex \tab age \tab period \tab obs \tab pyrs \cr
+#'   0 \tab 1 \tab 2010 \tab 0 \tab 390 \cr
+#'   0 \tab 2 \tab 2010 \tab 5 \tab 385 \cr
+#'   1 \tab 1 \tab 2010 \tab 3 \tab 308 \cr
+#'   1 \tab 2 \tab 2010 \tab 12 \tab 315
+#' }
+#' 
+#' 
+#' @param coh.data aggregated cohort data, see e.g. \code{\link{lexpand}}
+#' @param coh.pyrs variable name for person years in cohort data; 
+#' quoted (as a string \code{'myvar'}) or unquoted (AKA as a name; \code{myvar})
+#' @param coh.obs variable name for observed cases; quoted or unquoted. A vector when using \code{mstata}.
+#' @param ref.data population data. Can be left NULL if \code{coh.data} 
+#' is stratified in \code{print}. See \code{\link{pophaz}} for details.
+#' @param ref.rate population rate variable (cases/person-years). Overwrites 
+#' arguments \code{ref.pyrs} and \code{ref.obs}. Quoted or unquoted
+#' @param ref.pyrs variable name for person-years in population data; quoted or unquoted
+#' @param ref.obs variable name for observed cases; quoted or unquoted
+#' @param subset logical condition to select data from \code{coh.data} before any computations
+#' @param adjust variable names for adjusting without stratifying output; quoted vector or unquoted list
+#' @param print variable names to stratify results; quoted vector or unquoted named list with functions
+#' @param mstate set column names for cause specific observations; quoted or unquoted. Relevant only
+#' when \code{coh.obs} length is two or more. See details.
+#' @param test.type Test for equal SIRs. Test available are 'homogeneity' and 'trend'.
+#' @param conf.type Confidence interval type: 'profile'(=default), 'wald' or 'univariate'.
+#' @param conf.level Level of type-I error in confidence intervals, default 0.05 is 95\% CI.
+#' @param EAR logical; TRUE calculates Excess Absolute Risks for univariate SIRs.
+#' (see details)
+
+#' 
+#' @examples 
+#' data(popmort)
+#' data(sire)
+#' c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+#'               breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+#'               aggre = list(fot, agegroup = age, year = per, sex) )
+#' ## SMR due other causes: status = 2
+#' se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs', 
+#'            ref.data = popmort, ref.rate = 'haz', 
+#'            adjust = c('agegroup', 'year', 'sex'), print = 'fot')
+#' se
+#' ## for examples see: vignette('sir')
+#' 
+#' 
+#' @seealso \code{\link{lexpand}}
+#' \href{../doc/sir.html}{A SIR calculation vignette}
+#' @family sir functions
+#' @family main functions
+#' 
+#' @return A sir-object that is a \code{data.table} with meta information in the attributes.
+#' 
+#' @export
+#' 
+#' @import data.table
+#' @import stats
+
+
+
+
+sir <- function( coh.data, 
+                 coh.obs,
+                 coh.pyrs,
+                 ref.data = NULL,
+                 ref.obs = NULL,
+                 ref.pyrs = NULL, ref.rate = NULL,
+                 subset = NULL,
+                 print = NULL,
+                 adjust = NULL,
+                 mstate = NULL,
+                 test.type = 'homogeneity',
+                 conf.type = 'profile',
+                 conf.level = 0.95,
+                 EAR = FALSE){
+
+  coh.data <- data.table(coh.data)
+  
+  ## subsetting---------------------------------------------------------------
+  ## no copy taken of data!
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = coh.data, substiset = subset)
+  coh.data <- coh.data[subset,]
+  
+  
+  # print list --------------------------------------------------------------
+  
+  # env1 <- environment() # set environment where to assign new print
+  # coh.data <- data_list(data = coh.data, arg.list = substitute(print), env = env1)
+  
+  mstate <- as.character(substitute(mstate))
+  if(length(mstate) == 0) {
+    mstate <- NULL
+  }
+  if(!is.null(mstate)) {
+    coh.data[,(mstate) := 0L] 
+  }
+  
+  # evalPopArg
+  coh.obs <- substitute(coh.obs)
+  c.obs <- evalPopArg(data = coh.data, arg = coh.obs)
+  coh.obs <- names(c.obs)
+  
+  coh.pyrs <- substitute(coh.pyrs)
+  c.pyr <- evalPopArg(data = coh.data, arg = coh.pyrs)
+  coh.pyrs <- names(c.pyr)
+  
+  print <- substitute(print)
+  c.pri <- evalPopArg(data = coh.data, arg = print)
+  print <- names(c.pri)
+
+  adjust <- substitute(adjust)
+  c.adj <- evalPopArg(data = coh.data, arg = adjust)
+  adjust <- names(c.adj)
+  
+  # collect data
+  coh.data <- cbind(c.obs, c.pyr)
+  if(!is.null(print))  coh.data <- cbind(coh.data, c.pri) 
+  if(!is.null(adjust)) coh.data <- cbind(coh.data, c.adj)
+  
+  if( !is.null(ref.data) ){
+    ref.obs <- as.character(substitute(ref.obs))
+    ref.pyrs <- as.character(substitute(ref.pyrs))
+    ref.rate <- as.character(substitute(ref.rate))
+    
+    if (length(ref.obs) == 0) ref.obs <- NULL
+    if (length(ref.pyrs) == 0) ref.pyrs <- NULL
+    if (length(ref.rate) == 0) ref.rate <- NULL
+  }
+
+
+  # print(coh.data)
+
+  st <- sir_table( coh.data = coh.data, 
+                   coh.obs = coh.obs,
+                   coh.pyrs = coh.pyrs,
+                   ref.data = ref.data,
+                   ref.obs = ref.obs,
+                   ref.pyrs = ref.pyrs, 
+                   ref.rate = ref.rate,
+                   print = print,
+                   adjust = adjust,
+                   mstate = mstate)
+
+  results <- sir_est( table = st,
+                      print = print,
+                      adjust = adjust,
+                      conf.type = conf.type, 
+                      test.type = test.type,
+                      conf.level = conf.level,
+                      EAR = EAR)
+  
+  ## final touch ---------------------------------------------------------------
+  
+
+  #setDT(data)
+  if (!return_DT()) {
+    for (i in 1:3) {
+      if (!is.null(results[[i]])) {
+        setDFpe(results[[i]])
+      }
+    }  
+  }
+  
+  data <- copy(results[[2]])
+  setattr(data, name = 'sir.meta', value = list(adjust = adjust,
+                                                print = print,
+                                                call = match.call(),
+                                                lrt.test= results$'lrt.test',
+                                                conf.type = results$'conf.type',
+                                                conf.level = conf.level,
+                                                lrt.test.type = results$'test.type',
+                                                pooled.sir = results[[1]]))
+  setattr(data, "class", c("sir", "data.table", "data.frame"))
+  return(data)
+}
+
+
+#' @title Estimate splines for SIR or SMR
+#' @author Matti Rantanen, Joonas Miettinen
+#' 
+#' @description Splines for standardised incidence or mortality ratio. A useful 
+#' tool to e.g. check whether a constant SIR can be assumed for all calendar periods,
+#' age groups or follow-up intervals. Splines can be fitted for these time dimensions
+#' separately or in the same model.
+#' 
+#' @param coh.data cohort data with observations and at risk time variables
+#' @param coh.pyrs variable name for person-years in cohort data
+#' @param coh.obs variable name for observed cases
+#' @param ref.data aggregated population data
+#' @param ref.rate population rate observed/expected. This overwrites the parameters
+#' \code{ref.pyrs} and \code{ref.obs}.
+#' @param ref.pyrs variable name for person-years in population data
+#' @param ref.obs variable name for observed cases
+#' @param subset logical condition to subset \code{coh.data} before any computations
+#' @param adjust variable names for adjusting the expected cases
+#' @param print variable names for which to estimate SIRs/SMRs and 
+#' associated splines separately 
+#' @param mstate set column names for cause specific observations. Relevant only
+#' when coh.obs length is two or more. See help for \code{sir}.
+#' @param spline variable name(s) for the splines
+#' @param knots number knots (vector),  pre-defined knots (list of vectors) or for optimal number of knots left NULL
+#' @param dependent.splines logical; if TRUE, all splines are fitted in same model.
+#' @param reference.points fixed reference values for rate ratios. If left \code{NULL}
+#' the smallest value is the reference point (where SIR = 1). 
+#' Ignored if \code{dependent.splines = FALSE}
+#' 
+#' 
+#' @details 
+#' 
+#' See \code{\link{sir}} for help on SIR/SMR estimation in general; usage of splines
+#' is discussed below.
+#' 
+#' \strong{The spline variables}
+#' 
+#' The model can include one, two or three splines variables.
+#' Variables can be included in the same model selecting \code{dependent.splines = TRUE}
+#' and SIR ratios are calculated (first one is the SIR, others SIR ratios). 
+#' Reference points vector can be set via \code{reference.points}
+#' where first element of the vector is the reference point for first ratio.
+#' 
+#' Variable(s) to fit splines are given as a vector in argument \code{spline}.
+#' Order will affect the results.
+#' 
+#' 
+#' \strong{dependent.splines} 
+#' 
+#' By default dependent.splines is FALSE and all splines are fitted in separate models. 
+#' If TRUE, the first variable in \code{spline} is a function of a SIR and other(s) are ratios.
+#' 
+#' \strong{knots}
+#' 
+#' There are three options to set knots to splines:
+#' 
+#' Set the number of knots for each spline variable with a \strong{vector}. 
+#' The knots are automatically placed to the quantiles of observed cases in cohort data. 
+#' The first and last knots are always the maximum and minimum values, so knot
+#' value needs to be at least two.
+#' 
+#' Predefined knot places can be set with a \strong{list} of vectors.
+#' The vector for each spline in the list specifies the knot places. The lowest 
+#' and the largest values are the boundary knots and these should be checked beforehand.
+#' 
+#' If \code{knots} is left \strong{NULL}, the model searches the optimal number 
+#' of knots by model AIC by fitting models iteratively from 2 to 15 knots and 
+#' the one with smallest AIC is selected.
+#' If \code{dependent.splines = TRUE}, the number of knots is searched by fitting each spline
+#' variable separately.
+#' 
+#' 
+#' \strong{print}
+#' 
+#' Splines can be stratified by the levels of variable given in \code{print}. If 
+#' \code{print} is a vector, only the first variable is accounted for. The knots 
+#' are placed globally for all levels of \code{print}. This also ensures that the likelihood 
+#' ratio test is valid.
+#' Splines are also fitted independently for each level of \code{print}.
+#' This allows for searching interactions, e.g. by fitting spline for period 
+#' (\code{splines='period'}) for each age group (\code{print = 'agegroup'}).
+#' 
+#' 
+#' \strong{p-values}
+#' 
+#' The output p-value is a test of whether the splines are equal (homogenous)
+#' at different levels of \code{print}. 
+#' The test is based on the likelihood ratio test, where the full model 
+#' includes \code{print} and is 
+#' compared to a null model without it.
+#' When \code{(dependent.splines = TRUE)} the p-value returned is a global p-value.
+#' Otherwise the p-value is spline-specific.
+#' 
+#' 
+#' @return A list of data.frames and vectors.
+#' Three spline estimates are named as \code{spline.est.A/B/C} and the corresponding values
+#' in \code{spline.seq.A/B/C} for manual plotting
+#' 
+#' 
+#' @seealso \code{\link{splitMulti}} 
+#' \href{../doc/sir.html}{A SIR calculation vignette}
+#' @family sir functions
+#' @family main functions
+#' 
+#' @export sirspline
+#' @import data.table 
+#' @import splines
+#' @import stats
+#' 
+#' @examples \donttest{
+#' ## for examples see: vignette('sir')
+#' }
+
+sirspline <- function( coh.data, 
+                       coh.obs,
+                       coh.pyrs,
+                       ref.data = NULL,
+                       ref.obs = NULL,
+                       ref.pyrs = NULL, 
+                       ref.rate = NULL,
+                       subset = NULL,
+                       print = NULL,
+                       adjust = NULL,
+                       mstate = NULL,
+                       spline,
+                       knots = NULL,
+                       reference.points = NULL,
+                       dependent.splines = TRUE){
+  
+  coh.data <- data.table(coh.data)
+  
+  ## subsetting-----------------------------------------------------------------
+  ## no copy taken of data!
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = coh.data, substiset = subset)
+  coh.data <- coh.data[subset,]
+  
+  # print list --------------------------------------------------------------
+
+  env1 <- environment()
+  coh.data <- data_list(data = coh.data, arg.list = substitute(print), env = env1)
+  
+  mstate <- as.character(substitute(mstate))
+  if(length(mstate) == 0) {
+    mstate <- NULL
+  }
+  if(!is.null(mstate)) {
+    coh.data[,(mstate) := 0L] 
+  }
+  
+  # evalPopArg
+  
+  spline <- substitute(spline)
+  c.spl <- evalPopArg(data = coh.data, arg = spline)
+  spline <- names(c.spl)
+  
+  coh.obs <- substitute(coh.obs)
+  c.obs <- evalPopArg(data = coh.data, arg = coh.obs)
+  coh.obs <- names(c.obs)
+  
+  coh.pyrs <- substitute(coh.pyrs)
+  c.pyr <- evalPopArg(data = coh.data, arg = coh.pyrs)
+  coh.pyrs <- names(c.pyr)
+  
+  print <- substitute(print)
+  c.pri <- evalPopArg(data = coh.data, arg = print)
+  print <- names(c.pri)
+  
+  adjust <- substitute(adjust)
+  c.adj <- evalPopArg(data = coh.data, arg = adjust)
+  adjust <- names(c.adj)
+
+  # collect data
+  coh.data <- cbind(c.obs, c.pyr, c.spl)
+  if(!is.null(print))  {
+    coh.data <- cbind(coh.data, c.pri[, print[!print %in% spline], with=FALSE])
+  }
+  if(!is.null(adjust)) {
+    coh.data <- cbind(coh.data, c.adj[, adjust[!adjust %in% spline], with=FALSE])
+  }
+  
+  if( !is.null(ref.data) ){
+    ref.obs <- as.character(substitute(ref.obs))
+    ref.pyrs <- as.character(substitute(ref.pyrs))
+    ref.rate <- as.character(substitute(ref.rate))
+    
+    if (length(ref.obs) == 0) ref.obs <- NULL
+    if (length(ref.pyrs) == 0) ref.pyrs <- NULL
+    if (length(ref.rate) == 0) ref.rate <- NULL
+  }
+  
+  st <- sir_table( coh.data = coh.data, 
+                   coh.obs = coh.obs,
+                   coh.pyrs = coh.pyrs,
+                   ref.data = ref.data,
+                   ref.obs = ref.obs,
+                   ref.pyrs = ref.pyrs, ref.rate = ref.rate,
+                   print = print,
+                   adjust = adjust,
+                   mstate = mstate,
+                   spline = spline)
+
+  results <- sir_spline( table = st, 
+                         print = print,
+                         adjust = adjust,
+                         spline = spline,
+                         knots = knots,
+                         reference.points = reference.points,
+                         dependent.splines = dependent.splines)
+  
+  setclass(results, c('sirspline', 'pe', class(results)))
+  return(results)  
+}
+
+
+
+
+
+
+# Input: two data.table:s
+# output: one data.table including rates
+#' @import stats
+#' @import data.table
+sir_table <- function( coh.data, 
+                       coh.obs,
+                       coh.pyrs,
+                       ref.data = NULL,
+                       ref.obs = NULL,
+                       ref.pyrs = NULL, 
+                       ref.rate = NULL,
+                       print = NULL,
+                       adjust = NULL,
+                       spline = NULL,
+                       mstate = NULL) {
+
+
+  # initial checks -------------------------------------------------
+  
+  if(is.null(ref.data)) {
+    if(is.null(print)){
+      stop('Both ref.data and print cannot be NULL.')
+    }
+    ref.data <- data.table(coh.data)
+    ref.obs <- coh.obs
+    ref.pyrs <- coh.pyrs
+  }
+  
+  coh.data <- data.table(coh.data)
+  ref.data <- data.table(ref.data)
+
+  vl <- unique( c(coh.pyrs, coh.obs, adjust, print) )
+  if( !is.null(mstate) )  {
+    vl <- vl[which( vl != mstate )]
+  }
+  all_names_present(coh.data, vl )
+  
+  if ( !is.null(ref.pyrs) & !is.null(ref.obs) ) {
+    all_names_present(ref.data, c(ref.pyrs, ref.obs, adjust))
+  }
+  
+  # Melt lexpand data -------------------------------------------------------
+  
+  if( length(coh.obs) > 1 ) {
+    if( is.null(mstate) ){
+      stop('coh.obs length is > 1. Set variable name for mstate.')
+    }
+    if( !mstate %in% names(ref.data) ){
+      warning('mstate variable name does not match names in ref.data.')
+    }
+
+    aggre <- unique(c(adjust, print, spline, coh.pyrs))
+    aggre <- aggre[which(aggre != mstate)]
+
+    coh.data <- melt( data = coh.data, id.vars = aggre, measure.vars = coh.obs, 
+                      value.name = 'coh.observations', 
+                      variable.name = mstate, variable.factor = FALSE)
+    coh.obs <- 'coh.observations'
+  
+    # parse Y name form string 'formXtoY'
+    q <- quote(
+      robust_values(substr(get(mstate), 
+                           start = regexpr( pattern = 'to', text = get(mstate) ) + 2, 
+                           stop  = nchar(x = get(mstate) )))
+      )
+    coh.data[,(mstate) := eval(q) ]
+    
+    if( !(mstate %in% adjust)) {
+      warning('Consider including mstate variable also in adjust. See help(sir) for details.')
+    }
+  }
+  
+  # prepare data steps, reduce dimensions -----------------------------------
+  
+  setnames(coh.data, c(coh.obs, coh.pyrs), c('coh.observations','coh.personyears'))  
+  
+  
+  coh.data <- expr.by.cj(data = coh.data,
+                         by.vars = unique( sort(c(adjust, print, spline)) ), 
+                         expr = list(coh.observations = sum(coh.observations), 
+                                     coh.personyears  = sum(coh.personyears)))
+  #coh.data <- na2zero(coh.data)
+  #coh.data <- na.omit(coh.data) 
+
+  coh.data[is.na(coh.observations), coh.observations := 0]
+  coh.data[is.na(coh.personyears), coh.personyears := 0]
+  coh.data <- na.omit(coh.data)
+  
+  # rates
+  if( !is.null(ref.rate) ){
+    setnames(ref.data, ref.rate, 'ref.rate')
+    ref.data <- expr.by.cj(data = ref.data, by.vars = c(adjust), 
+                           expr = list(ref.rate = mean(ref.rate)))
+  } else {
+    setnames(ref.data, c(ref.obs, ref.pyrs), c('ref.obs','ref.pyrs'))
+    ref.data <- expr.by.cj(data = ref.data, by.vars = c(adjust), 
+                           expr = list(ref.obs = sum(ref.obs),
+                                       ref.pyrs= sum(ref.pyrs)))
+    ref.data[, ref.rate := ref.obs / ref.pyrs ]
+  }
+  
+  # Merge
+  sir.table <- merge(coh.data, ref.data, by=c(adjust), all.x=TRUE)
+  sir.table[, expected := ref.rate * coh.personyears]
+  sir.table <- na2zero(sir.table)
+  
+  if ( !is.null(print) | !is.null(spline)){
+    sir.table <- sir.table[ ,list(observed = sum(coh.observations), 
+                                  expected = sum(expected),
+                                  pyrs = sum(coh.personyears)), 
+                           by = c(unique(c(print, spline)))]
+    setkeyv(sir.table, c(print, spline))
+  }
+  else {
+    sir.table <- sir.table[ ,list(observed = sum(coh.observations), 
+                                  expected = sum(expected),
+                                  pyrs = sum(coh.personyears))]
+  }
+  return(sir.table)
+}
+
+
+
+
+# Input: sir.table
+# Output: list of data.tables and values
+sir_est <- function( table,
+                     print = NULL,
+                     adjust = NULL,
+                     EAR = FALSE,
+                     test.type = 'homogeneity',
+                     conf.level = 0.95,
+                     conf.type = 'profile') {
+  pyrs <- NULL ## APPEASE R CMD CHECK
+  setDT(table)
+
+  if(!is.numeric(conf.level) | conf.level > 1) {
+    stop('Confidence level must be a numeric value between 0-1')
+  }
+  # function to SIR p-value
+  chi.p <- function(o, e) {
+    pchisq( ( (abs(o - e) - 0.5)^2)/e, df=1, lower.tail=FALSE)
+  }
+
+  # total sir
+  combined <- data.table(table)[,list(observed = sum(observed), 
+                             expected = sum(expected),
+                             pyrs = sum(pyrs))]
+  combined[ ,':='(sir = observed/expected,
+                  sir.lo = poisson.ci(observed, expected, conf.level=conf.level)[,4],
+                  sir.hi = poisson.ci(observed, expected, conf.level=conf.level)[,5],
+                  p_value  = chi.p(observed, expected))]
+  
+  # Poisson regression ------------------------------------------------------
+  
+  # write model formula
+  fa <- a <- NULL
+  sir.formula <- paste('observed ~ 1') 
+  if(!is.null(print)){
+    fa <- rev(print) #  fa <- print
+    
+    # drop variables with only one value
+    u <- c(t(table[, lapply(.SD, uniqueN), .SDcols = fa]))
+    if (length(u[u==1]) > 0){
+      message('Variable "', paste(fa[which(u==1)], collapse = '","'),'" (has only one level) removed from model.')
+      fa <- fa[-which(u==1)]
+    }
+    if(length(fa)>0){
+      # model formula
+      a <- paste0('as.factor(',paste( fa, collapse = '):as.factor('),')')
+      sir.formula <- paste('observed ~ 0 +', a)
+    }
+  }
+  # fit model if possible -----------------------------------------------------
+  
+  fit <- tryCatch(do.call("glm", list(formula = terms(as.formula(sir.formula), keep.order = FALSE), 
+                                      offset = log(table[,expected]), 
+                                      data = table, family = poisson(log))), 
+                  error=function(f) NULL )
+  
+  if(!is.null(fit)) eg <- expand.grid(fit$xlevels) # for further testing
+  
+  
+  # LRT test (homogeneity or trend) --------------------------------------------
+  
+  test.type <- match.arg(test.type, c('homogeneity','trend'))
+  
+  lrt_sig <- NULL  
+  if( sir.formula != 'observed ~ 1' & !is.null(fit) ) {
+    if (test.type == 'homogeneity') covariates <- a
+    if (test.type == 'trend') covariates <- paste(print, collapse=' + ')
+     
+    fit_full <- tryCatch(
+      do.call("glm", list(formula = terms(as.formula( paste0('observed ~ 1 + ', a) )), 
+                          offset = log(table[,expected]), 
+                          data = table, family=poisson(log))), 
+      error=function(f) NULL )
+    
+    fit_null <- tryCatch(
+      do.call("glm", list(formula = terms(as.formula('observed ~ 1') ), 
+                          offset = log(table[,expected]), 
+                          data = table, family=poisson(log))), 
+      error=function(f) NULL )
+    
+    if (!is.null(fit_full)){
+      lrt <- anova(fit_full, fit_null, test = 'Chisq')
+      lrt_sig <- lrt[['Pr(>Chi)']][2]    
+    }
+  }
+  
+  # confidence intervals ----------------------------------------------------
+  
+  conf.type <- match.arg(conf.type, c('wald','profile','univariate'))
+  ci.info <- NULL
+  ci <- NULL
+  
+  if (is.null(fit) & conf.type %in% c('wald','profile')) {
+    conf.type <- 'univariate'
+    ci.info <- 'Model fitting failed. Univariate confidence intervals selected.'
+    if(any(table$expected == 0)) {
+      ci.info <- paste(ci.info, '(zero values in expected)')
+    }
+  }
+  
+  if (conf.type == 'profile') {
+    
+    confint_glm <- function(object, parm, level = 0.95, trace = FALSE, ...) {
+      pnames <- names(coef(object))
+      if (missing(parm)) {
+        parm <- seq_along(pnames)
+      }
+      else if (is.character(parm)) {
+        parm <- match(parm, pnames, nomatch = 0L)
+      }
+      object <- profile(object, which = parm, alpha = (1 - level)/4,  trace = trace)
+      confint(object, parm = parm, level = level, trace = trace, ...)
+    }
+
+    ci <- suppressMessages( suppressWarnings( 
+      tryCatch(exp(confint_glm(fit, level=conf.level)), error=function(e) NULL )
+    ))
+    if(!is.null(ci)) {
+      ci <- as.data.table(ci)
+      if (is.null(print) | length(fa)==0) ci <- data.table(t(ci)) # transpose if only one row
+    } else {
+      conf.type <- 'wald'
+      ci.info <- 'Could not solve profile-likelihood. Wald confidence intervals selected.'
+    }
+  }
+  
+  if (conf.type == 'wald') { 
+    ci <- data.table( exp(confint.default(fit)) )
+  }
+  
+  if(conf.type == 'univariate') {
+    ci <- data.table(poisson.ci(table$observed, table$expected, conf.level = conf.level))[,.(lower, upper)]
+    pv <- chi.p(table$observed, table$expected)
+  } else {
+    pv <- as.vector(summary(fit)$coef[, "Pr(>|z|)"]) 
+  }
+  if(!is.null(ci.info)) message(ci.info)
+
+  # collect results -----------------------------------------------------
+  
+  setnames(ci, 1:2, c('sir.lo','sir.hi'))
+  
+  table[, ':=' ( sir = observed/expected,
+              sir.lo = ci[, sir.lo],
+              sir.hi = ci[, sir.hi],
+              p_value = round(pv,5))]
+  
+
+  # Round results -----------------------------------------------------------
+  
+  cols1 <- c('sir','sir.lo','sir.hi','expected','pyrs')
+  
+  table[,(cols1) := lapply(.SD, round, digits=4), .SDcols=cols1]
+  combined[,(cols1) := lapply(.SD, round, digits=4), .SDcols=cols1]
+
+  
+  # tests -----------------------------------
+  
+  if (table[!is.na(sir) & (sir < sir.lo | sir > sir.hi), .N] > 0) {
+    warning('There is something wrong with confidence intervals')
+  }
+  if (table[!is.na(sir.lo) & !is.na(sir.hi)][sir.lo > sir.hi, .N] > 0) {
+    warning('CIs might be incorrect')
+  }
+
+  if(!is.null(fit) & length(fa)>0) {
+    # pseudo test if the modelled confidence intervals are merged correctly:
+    t1 <- copy(table)[,lapply(.SD, factor),.SDcols = fa]
+    if(any(t1 != data.table(eg))) {
+      message('CIs levels might not match. Contact the package maintainer and use univariate CIs.')
+    }
+  }
+  
+  # EAR -----------------------------------------------------------------
+  if (EAR) {
+    table[,EAR := round((observed - expected)/pyrs * 1000, 3)]
+  }
+  
+  
+  results <- list(total = combined, 
+                  table = table, 
+                  adjusted = adjust,
+                  lrt.test = lrt_sig,
+                  test.type = test.type,
+                  conf.type = conf.type,
+                  ci.info = ci.info)
+  return(results)
+}
+
+
+#' @export
+getCall.sir <- function (x, ...) {
+  attributes(x)$sir.meta$call
+}
+
+
+# Input: sir.table
+# Output: estimates and sequences for plotting splines
+#' @import splines
+#' @import data.table
+#' @import stats
+sir_spline <- function(  table,
+                         print = NULL,
+                         adjust = NULL,
+                         spline,
+                         knots = NULL,
+                         reference.points = NULL,
+                         dependent.splines = TRUE){
+  knts <- 
+    spline.seq.A <- 
+    spline.seq.B <- 
+    spline.seq.C <- 
+    spline.est.A <- 
+    spline.est.B <- 
+    spline.est.C <- NULL
+  
+  if (!is.null(knots) & length(knots) != length(spline) ) {
+    stop('Arguments spline and knots has to be same length.')
+  }
+  
+  
+  # Spline functions -------------------------------------------------------
+  
+  # function to get spline seq
+  spline.seq <- function(data, spline.var=NULL) {
+    # palauttaa jotaina
+    if(is.na(spline.var)) {
+      return(NULL)
+    }
+    spline.seq <- seq( min( data[,get(spline.var)] ), 
+                       max( data[,get(spline.var)] ), length.out = 100)
+    return(spline.seq)
+  }
+  
+  # function to search optimal number of knots by AIC
+  spline.knots <- function(data, knots = NULL, spline.vars = NULL){    
+    # search optimal number of knots
+    if( is.null(knots) ) {
+      knts <- list()
+      for (jj in 1:length(spline.vars)) {
+        # reduce data to fit model
+        data0 <- data[,list(observed=sum(observed), expected = sum(expected)), by = eval(spline.vars[jj])] 
+        data0 <- data0[expected > 0]
+        spline.fit <- glm(observed ~ 1, offset=log(expected), family=poisson(log), data = data0)
+        aic0 <- summary(spline.fit)[['aic']]
+        limit <- 20
+        ii <- 2
+        while(  ii < limit ){
+          tmp.knots <- ii
+          knts[jj] <- list( data0[ ,quantile( rep(get(spline.vars[jj]),observed), probs = seq(0,100,length.out = tmp.knots)/100)] )
+          spline.fit <- glm(observed ~ Ns(get(spline.vars[jj]), knots = knts[[jj]]), offset=log(expected), family=poisson(log), data=data0)
+          aic0 <- c(aic0, summary(spline.fit)[['aic']])
+          ii <- ii + 1
+        }
+        tmp.knots <- which(aic0 == min(aic0))[1]
+        if(tmp.knots == 1) {
+          message(paste0('Null model better than spline in ', jj))
+          tmp.knots <- 2
+        }
+        knts[jj] <- list(data0[ ,quantile( rep(get(spline.vars[jj]),observed), probs = seq(0,100,length.out = tmp.knots)/100)])
+        rm(tmp.knots)
+      }
+      knots <- unlist(lapply(knts, length))
+    }
+    else {
+      # knot predefined
+      if( is.list(knots) ){
+        knts <- knots
+        knots <- unlist(lapply(knots, length))
+      } 
+      # knot number predefined
+      else {
+        if( any(knots < 2) ) { 
+          message('Min knots number set to 2.') 
+          knots[knots < 2] <- 2
+        }
+        knts <- list()
+        for(i in 1:length(knots)) {
+          knts[i] <- list( data[ ,quantile( rep(get(spline.vars[i]), observed), probs = seq(0,100,length.out = knots[i])/100)])
+        }
+      }
+    }
+    names(knts) <- spline.vars
+    return(knts)
+  }
+  
+  # function to estimate 2-3 dim splines in same model
+  spline.estimates.dep <- function(sir.spline = sir.spline,
+                                   spline.seq.A = spline.seq.A,
+                                   spline.seq.B = spline.seq.B,
+                                   spline.seq.C = spline.seq.C,
+                                   reference.points = reference.points,
+                                   knts = knts
+  ){
+    
+    if( all(!is.null(reference.points), (length(reference.points) + 1) != length(spline)) ){
+      stop('Parameter reference.points length should be length of spline - 1.')
+    }
+    
+    
+    form <- 'Ns(get(spline[[1]]), kn=knts[[1]])'
+    nsA <- Ns( spline.seq.A, knots = knts[[1]])
+    if ( length(spline) >= 2) {
+      form <- paste0(form, ' + Ns(get(spline[[2]]), kn=knts[[2]])')
+      nsB <- Ns( spline.seq.B, knots = knts[[2]])
+    }
+    if ( length(spline) == 3) {
+      form <- paste0(form, ' + Ns(get(spline[[3]]), kn=knts[[3]])')
+      nsC <- Ns( spline.seq.C, knots = knts[[3]])
+    }
+    
+    form <- paste0('observed ~ ', form)
+    spline.fit <- do.call("glm", list(formula = as.formula(form),
+                                      offset = log(sir.spline[expected > 0,expected]),
+                                      family = poisson,
+                                      data = sir.spline[expected>0]))
+    if( any( ci.exp(spline.fit)[,1] == 1) ){
+      message("NA's in spline estimates.")
+    }
+    
+    aic <- summary(spline.fit)[['aic']]
+    
+    rf.C <- rf.B <- NA
+    # set assigned reference points or get minimum values
+    if( !is.null(reference.points) ) {
+      rf.B <- reference.points[1]
+      rf.C <- reference.points[2]
+    } 
+    else {
+      rf.B <- min( sir.spline[,get(spline[2])] )
+      if(!is.na(spline[3])) {
+        rf.C <- min( sir.spline[,get(spline[3])] )
+      }
+    }
+    
+    if( !is.na(rf.B) )  {
+      B <- Ns( rep(rf.B, 100), knots = knts[[2]])
+      if( findInterval(rf.B, range(sir.spline[,get(spline[2])])) != 1 ) {
+        message("WARNING: reference point 2 doesn't fall into spline variable interval")
+      }
+    }
+    
+    if( !is.na(rf.C) ){
+      C <- Ns( rep(rf.C, 100), knots = knts[[3]])
+      if( findInterval(rf.C, range(sir.spline[,get(spline[3])])) != 1) {
+        message("WARNING: reference point 3 doesn't fall into spline variable interval")
+      }
+    } 
+    
+    # make subset of model parameters
+    if( !is.null(knts[2]) ) {
+      sub.B <- which( grepl('spline[[2]]', names(spline.fit$coefficients),fixed = TRUE) )
+    }
+    if( !is.null(knts[3]) ) {
+      sub.C <- which( grepl('spline[[3]]', names(spline.fit$coefficients),fixed = TRUE) )
+    }
+    if ( length(spline) == 2) {
+      spline.est.A <- ci.exp(spline.fit, ctr.mat = cbind(1, nsA, nsB))
+      spline.est.B <- ci.exp(spline.fit, subset = sub.B, ctr.mat = nsB - B)
+      spline.est.C <- NULL      
+    }
+    if ( length(spline) == 3) {
+      spline.est.A <- ci.exp(spline.fit, ctr.mat = cbind(1, nsA, nsB, nsC))
+      spline.est.B <- ci.exp(spline.fit, subset= sub.B, ctr.mat = nsB - B)
+      spline.est.C <- ci.exp(spline.fit, subset= sub.C, ctr.mat = nsC - C)
+    }
+    list(a = spline.est.A, 
+         b = spline.est.B, 
+         c = spline.est.C)
+  }
+  
+  # function to estimate independet splines
+  spline.estimates.uni <- function(data, spline.var, spline.seq, knots, knum) {  
+    if(is.na(spline.var)) return(NULL)
+    knots <- knots[[knum]]
+    data <- data[,list(observed=sum(observed), expected = sum(expected)), by = eval(spline.var)][expected > 0]
+    spline.uni <- glm(observed ~ Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data)
+    nsx <- Ns( spline.seq, knots = knots)
+    spline.est <- ci.exp(spline.uni, ctr.mat = cbind(1, nsx))
+    spline.est
+  }
+  
+  
+  
+  # Poisson regression Splines -------------------------------------------------
+  
+  sir.spline <- data.table(table)
+  
+  # convert spline variables to numeric
+  temp.fun <- function(x){
+    as.numeric(as.character(x))
+  }
+  sir.spline[, (spline) := lapply(.SD, temp.fun), .SDcols = spline]
+  
+  
+  
+  # set knots
+  knts <- spline.knots(data=sir.spline, knots = knots, spline.vars = spline)  
+  
+  # set sequences
+  spline.seq.A <- spline.seq(data=sir.spline, spline.var=spline[1])
+  spline.seq.B <- spline.seq(data=sir.spline, spline.var=spline[2])
+  spline.seq.C <- spline.seq(data=sir.spline, spline.var=spline[3])
+  
+  if( length(spline) == 1 ) {
+    dependent.splines <- FALSE
+  }
+  
+  # convert print to factor
+  print <- print[1]
+  
+  # loop for each level of print:
+  if( !is.null(print) ) {
+    prnt.levels <- sir.spline[,unique( get(print) )]
+    sir.spline[,(print) := factor(get(print))]
+  }
+  else {
+    print <- 'temp'
+    sir.spline[,temp := 1]
+    prnt.levels <- 1
+  }
+  
+  spline.est.A <- NULL
+  spline.est.B <- NULL
+  spline.est.C <- NULL
+  
+  for(i in prnt.levels){
+    if( dependent.splines ) {
+      out <- spline.estimates.dep(sir.spline = sir.spline[get(print) == i],
+                                  spline.seq.A = spline.seq.A,
+                                  spline.seq.B = spline.seq.B,
+                                  spline.seq.C = spline.seq.C,
+                                  reference.points = reference.points,
+                                  knts = knts)
+      est.A <- out[['a']]
+      est.B <- out[['b']]
+      est.C <- out[['c']]
+    }
+    else{
+      est.A <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[1], spline.seq = spline.seq.A, knots = knts, knum = 1)
+      est.B <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[2], spline.seq = spline.seq.B, knots = knts, knum = 2)  
+      est.C <- spline.estimates.uni(data = sir.spline[get(print) == i], spline.var = spline[3], spline.seq = spline.seq.C, knots = knts, knum = 3)
+    }
+    
+    add_i <- function(est.x, i){
+      if(is.null(est.x)) {
+        return(NULL)
+      }
+      cbind(i, data.frame(est.x))
+    }
+    
+    
+    est.A <- add_i(est.A, i)
+    est.B <- add_i(est.B, i)
+    est.C <- add_i(est.C, i)
+    
+    spline.est.A <- rbind(spline.est.A, est.A)
+    spline.est.B <- rbind(spline.est.B, est.B)
+    spline.est.C <- rbind(spline.est.C, est.C)
+  }
+  
+  # get p-value and anova-table
+  anovas <- NULL
+  p <- NULL
+  if(dependent.splines) {
+    form.a <- 'Ns(get(spline[[1]]), kn=knts[[1]]) + Ns(get(spline[[2]]), kn=knts[[2]])'
+    form.b <- 'get(print):Ns(get(spline[[1]]), kn=knts[[1]]) + get(print):Ns(get(spline[[2]]), kn=knts[[2]])'
+    if ( length(spline) == 3) {
+      form.a <- paste0(form.a, ' + Ns(get(spline[[3]]), kn=knts[[3]])')
+      form.b <- paste0(form.b, ' + get(print):Ns(get(spline[[3]]), kn=knts[[3]])')
+    }
+    
+    fit.fun <- function( form.string ){
+      do.call("glm", list(formula = as.formula( form.string ),
+                          offset = log(sir.spline[expected > 0,expected]),
+                          family = poisson,
+                          data = sir.spline[expected>0]))
+    }
+    
+    fit.1 <- fit.fun( paste0('observed ~ ', form.a) )
+    fit.2 <- fit.fun( paste0('observed ~ ', 'get(print)+', form.a))
+    fit.3 <- fit.fun( paste0('observed ~ ', form.b))
+    fit.4 <- fit.fun( paste0('observed ~ ', 'get(print)+', form.b) )
+    
+    global.p<- anova(fit.4, fit.1, test='LRT')
+    level.p <- anova(fit.2, fit.1, test='LRT')
+    #shape.p <- anova(fit.4, fit.3, test='LRT')
+    
+    anovas <- list(global.p = global.p, level.p = level.p)
+    p <- rbind(global.p[['Pr(>Chi)']][2], level.p[['Pr(>Chi)']][2]) # , shape.p, 
+  }
+  else {    
+    lrt.uni <- function(data=sir.spline, spline.var=spline[1], print=print, knots=knts, knum = 1) {
+      if (is.na(spline.var)) return (NULL)
+      data <- data.table(data)
+      knots <- knots[[knum]]
+      fit0 <- glm(observed ~ get(print)+Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
+      fit1 <- glm(observed ~ Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
+      fit2 <- glm(observed ~ get(print)*Ns(get(spline.var), knots = knots), offset=log(expected), family=poisson(log), data = data[expected>0])
+      anova(fit2,fit1,fit0, test='Chisq') # [['Pr(>Chi)']][2]
+    }
+    
+    var1.p <- lrt.uni(spline.var = spline[1], print=print, knots=knts, knum = 1)
+    var2.p <- lrt.uni(spline.var = spline[2], print=print, knots=knts, knum = 2)
+    var3.p <- lrt.uni(spline.var = spline[3], print=print, knots=knts, knum = 3)
+    
+    p <- list(spline.a = var1.p[['Pr(>Chi)']][2], 
+              spline.b = var2.p[['Pr(>Chi)']][2], 
+              spline.c = var3.p[['Pr(>Chi)']][2])
+    anovas <- list(spline.a = var1.p, spline.b = var2.p, spline.c = var3.p)
+  }
+  
+  output <- list( spline.est.A = spline.est.A,
+                  spline.est.B = spline.est.B,
+                  spline.est.C = spline.est.C,
+                  spline.seq.A = spline.seq.A,
+                  spline.seq.B = spline.seq.B,
+                  spline.seq.C = spline.seq.C,
+                  adjust = adjust,
+                  print = print,
+                  spline = spline,
+                  anovas = anovas,
+                  knots = knts,
+                  spline.dependent = dependent.splines,
+                  p.values = p)
+  output
+}
+
+# input data and argument list. replaces print in upper environment with name a vector.
+data_list <- function( data, arg.list, env ) {
+  if(missing(env)){
+    arg.list <- substitute(arg.list)
+    env <- parent.frame()
+  }
+  d <- data.table(data)
+  
+  l <- eval(arg.list, envir = d, enclos = parent.frame()) 
+  
+  if( is.list( l ) ) {
+    n <- intersect(names(l), names(d))
+    if(length(n)>0){
+      d[,(n) := NULL]
+    }
+    #     if(is.null(names(l))) {
+    #       v <- 1:length(l)
+    #       setnames(l, v, paste0('V', v))
+    #     }
+    l <- as.data.table(l)
+    l <- data.table(l)
+    assign('print', colnames(l), envir = env) # set names to parent environment
+    if( ncol(d) > 0) {
+      l <- data.table(d, l)
+    }
+    return(l)
+  } else { 
+    return(data) 
+  }
+}
+
+#' @export
+coef.sir <- function(object, ...) {
+  factors <- attr(object, 'sir.meta')$print
+  
+  q <- paste("paste(",paste(factors,collapse=","),", sep = ':')")
+  q <- parse(text=q)
+  n <- object[,eval(q)]
+  
+  res <- object$sir
+  attr(res, 'names') <- n
+  
+  res
+}
+
+
+
+
+#' @export
+confint.sir <- function(object, parm, level = 0.95, conf.type = 'profile', 
+                        test.type = 'homogeneity', ...) {
+
+  meta <- attr(object, 'sir.meta')
+  object <- copy(object)
+  object <- sir_est(table = object,
+                    print = meta$print,
+                    adjust = NULL,
+                    conf.type = conf.type, 
+                    test.type = test.type,
+                    conf.level = level,
+                    EAR = FALSE)
+  object <- object$table
+  q <- paste("paste(",paste(meta$print,collapse=","),", sep = ':')")
+  q <- parse(text=q)
+  n <- object[,eval(q)]
+
+  res <- cbind(object$sir.lo, object$sir.hi)
+  
+  rownames(res) <- n
+  colnames(res) <- paste( c( (1-level)/2*100, (1 - (1-level)/2)*100), '%')
+
+  res
+}
+
+
+#' @title Calculate SMR
+#' @author Matti Rantanen
+#' @description Calculate Standardized Mortality Ratios (SMRs) using 
+#' a single data set that includes
+#' observed and expected cases and additionally person-years.
+#' 
+#' @details These functions are intended to calculate SMRs from a single data set 
+#' that includes both observed and expected number of cases. For example utilizing the
+#' argument \code{pop.haz} of the \code{\link{lexpand}}.
+#' 
+#' \code{sir_lex} automatically exports the transition \code{fromXtoY} using the first
+#' state in \code{lex.Str} as \code{0} and all other as \code{1}. No missing values
+#' is allowed in observed, pop.haz or person-years.
+#' 
+#' @param x Data set e.g. \code{aggre} or \code{Lexis} object 
+#' (see: \code{\link{lexpand}})
+#' @param obs Variable name of the observed cases in the data set
+#' @param exp Variable name or expression for expected cases
+#' @param pyrs Variable name for person-years (optional)
+#' @param print Variables or expression to stratify the results
+#' @param test.type Test for equal SIRs. Test available are 'homogeneity' and 'trend'
+#' @param conf.level Level of type-I error in confidence intervals, default 0.05 is 95\% CI
+#' @param conf.type select confidence interval type: (default=) `profile`, `wald`, `univariate`
+#' @param subset a logical vector for subsetting data
+#' 
+#' @seealso \code{\link{lexpand}}
+#' \href{../doc/sir.html}{A SIR calculation vignette}
+#' @family sir functions
+#' 
+#' @return A sir object
+#' 
+#' @examples 
+#' 
+#' \donttest{
+#' BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+#' 
+#' ## Aggregated data
+#' x1 <- lexpand(sire, breaks = BL, status = status != 0, 
+#'               birth = bi_date, entry = dg_date, exit = ex_date,
+#'               pophaz=popmort,
+#'               aggre=list(sex, period = per, surv.int = fot))
+#' sir_ag(x1, print = 'period')
+#'
+#'
+#' # no aggreate or breaks
+#' x2 <- lexpand(sire, status = status != 0, 
+#'               birth = bi_date, entry = dg_date, exit = ex_date,
+#'               pophaz=popmort)
+#' sir_lex(x2, breaks = BL, print = 'per')
+#' }
+#' 
+#' @import data.table
+#' @import stats
+#' @export
+sir_exp <- function(x, obs, exp, pyrs=NULL, print = NULL, 
+                    conf.type = 'profile', test.type = 'homogeneity',
+                    conf.level = 0.95, subset = NULL) {
+  
+  # subsetting
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data = x, substiset = subset)
+  x <- x[subset,]
+  
+  # evalPopArg
+  obs <- substitute(obs)
+  c.obs <- evalPopArg(data = x, arg = obs)
+  obs <- names(c.obs)
+  
+  
+  print <- substitute(print)
+  c.pri <- evalPopArg(data = x, arg = print)
+  print <- names(c.pri)
+  
+  exp <- substitute(exp)
+  c.exp <- evalPopArg(data = x, arg = exp)
+  exp <- names(c.exp)
+  
+  pyrs <- substitute(pyrs)
+  c.pyr <- evalPopArg(data = x, arg = pyrs)
+  if(is.null(c.pyr)) c.pyr <- data.table(pyrs=0)
+  pyrs <- names(c.pyr)
+  
+  # collect data
+  x <- cbind(c.obs, c.pyr, c.exp)
+  if(any(is.na(x))) stop('Missing values in expected cases.')
+  if(!is.null(print))  x<- cbind(x, c.pri) 
+  
+  express <- paste0('list(observed = sum(', obs, '), expected = sum(',exp,'), pyrs = sum(', pyrs,'))')
+  # aggregate
+  es <- parse(text = express)
+  y <- x[, eval(es), keyby = print] # keyby is must
+  
+  results <- sir_est( table = y,
+                      print = print,
+                      adjust = NULL,
+                      conf.type = conf.type, 
+                      test.type = test.type,
+                      conf.level = conf.level,
+                      EAR = FALSE)
+  
+  #setDT(data)
+  if (!return_DT()) {
+    for (i in 1:2) {
+      if (!is.null(results[[i]])) {
+        setDFpe(results[[i]])
+      }
+    }  
+  }
+  
+  data <- copy(results[[2]])
+  setattr(data, name = 'sir.meta', value = list(adjust = NULL,
+                                                print = print,
+                                                call = match.call(),
+                                                lrt.test= results$'lrt.test',
+                                                conf.type = results$'conf.type',
+                                                conf.level = conf.level,
+                                                lrt.test.type = results$'test.type',
+                                                pooled.sir = results[[1]]))
+  setattr(data, "class", c("sir", "data.table", "data.frame"))
+  return(data)
+}
+
+
+
+#' Calculate SMRs from a split Lexis object  
+#' 
+#' @description \code{sir_lex} solves SMR from a \code{\link{Lexis}} object 
+#' calculated with \code{lexpand}.
+#' 
+#' @param breaks a named list to split age group (age), period (per) or follow-up (fot). 
+#' @param ... pass arguments to \code{sir_exp}
+#' 
+#' 
+#' @describeIn sir_exp
+#' 
+#' @export
+
+sir_lex <- function(x, print = NULL, breaks = NULL, ... ) {
+  
+  ## R CMD CHECK appeasement
+  lex.dur <- NULL
+  
+  if(!inherits(x, 'Lexis')) {
+    stop('x has to be a Lexis object (see lexpand or Lexis)')
+  }
+  if(!"pop.haz" %in% names(x)) {
+    stop("Variable pop.haz not found in the data.")
+  }
+
+
+  # reformat date breaks
+  if(!is.null(breaks)) {
+    breaks <- lapply(breaks, function(x) {
+      if(is.character(x)) c(cal.yr(as.Date(x)))
+      else x
+    })
+  }
+  
+  print <- substitute(print)
+  # copy to retain the attributes
+  x <- copy(x)
+  
+  # guess the first value
+  first_value <- lapply(c("lex.Cst", "lex.Xst"), function(var) {
+    if (is.factor(x[[var]])) levels(x[[var]]) else sort(unique(x[[var]]))
+  })
+  first_value <- unique(unlist(first_value))[1]
+  
+  col <- x$lex.Xst
+  set(x, j = "lex.Cst", value = 0L)
+  set(x, j = "lex.Xst", value = ifelse(col == first_value, 0L, 1L))
+  
+  if(!is.null(breaks)) {
+    x <- splitMulti(x, breaks = breaks)
+  }
+  
+  a <- copy(attr(x, "time.scales"))
+  a <- a[!vapply(get_breaks(x), is.null, logical(1))]
+  x[, d.exp := pop.haz*lex.dur]
+  
+  TF <- environment()
+  
+  if(any(is.na(x[,d.exp]))) stop('Missing values in either pop.haz or lex.dur.')
+  x <- aggre(x, by = TF$a, sum.values = 'd.exp')
+  if(!'from0to1' %in% names(x)) {
+    stop('Could not find any transitions between states in lexis')
+  }
+  x <- sir_exp(x = x, obs = 'from0to1', print = print, exp = 'd.exp', pyrs = 'pyrs', ...)
+  # override the match.call from sir_exp 
+  attr(x, 'sir.meta')$call <- match.call()
+  return(x)
+}
+
+
+#' SMR method for an \code{aggre} object.
+#' 
+#' @description \code{sir_ag} solves SMR from a \code{\link{aggre}} object 
+#' calculated using \code{\link{lexpand}}.
+#' 
+#' @describeIn sir_exp
+#' 
+#' @export
+
+sir_ag <- function(x, obs = 'from0to1', print = attr(x, 'aggre.meta')$by, exp = 'd.exp', pyrs = 'pyrs', ... ) {
+  
+  if(!inherits(x, 'aggre')) {
+    stop('x should be an aggre object (see lexpand or sir_lex)')
+  }
+  obs <- substitute(obs)
+  print <- substitute(print)
+  
+  x <- copy(x)
+  x <- sir_exp(x = x, obs = obs, print = print, exp = 'd.exp', pyrs = 'pyrs', ...) # original
+  attr(x, 'sir.meta')$call <- match.call() # override the call from sir_exp
+  x
+}
+
+
+
+globalVariables(c('observed','expected','p_adj','p_value','temp','coh.observations','coh.personyears',
+                  'd.exp', 'lower', 'pop.haz', 'sir.hi','sir.lo','upper'))
+
diff --git a/R/sir_utils.R b/R/sir_utils.R
index ef45065..18ee92a 100644
--- a/R/sir_utils.R
+++ b/R/sir_utils.R
@@ -1,115 +1,115 @@
-#' @title Confidence intervals for the ratio of two SIRs/SMRs
-#' @author Matti Rantanen
-#' @description Calculate ratio of two SIRs/SMRs and the confidence intervals of the ratio.
-#' 
-#' @details Function works with pooled sir-objects i.e. the \code{print} argument in \code{sir} is ignored.
-#' Also \code{x} and \code{y} can be a vector of two where first index is the
-#' observed cases and second is expected cases (see examples).
-#' Note that the ratio of two SIR's is only applicable when the age distributions are similar
-#' in both populations.
-#'
-#' \strong{Formula}
-#' 
-#' The observed number of first sir \code{O1} is considered as a Binomial variable with sample 
-#' size of \code{O1+O2}. The confidence intervals for Binomial proportion \code{A} 
-#' is solved using \code{exact} or \code{asymptotic} 
-#' method. Now the CI for ratio \code{O1/O2} is \code{B = A/(1 - A)}. And further the CI for SIR/SMR 
-#' is B*E2/E1. (Ederer and Mantel)
-#' 
-#' @param x a sir-object or a vector of two; observed and expected cases.
-#' @param y a sir-object or a vector of two; observed and expected cases.
-#' @param conf.level the type-I error in confidence intervals, default 0.95 for 95\% CI.
-#' @param type How the binomial confidence intervals are calculated (default:) \code{exact} or \code{asymptotic}.
-#' @param alternative The null-hypothesis test: (default:) \code{two.sided}, \code{less}, \code{greater}
-#' @param digits number of digits in the output
-#' 
-#' @note
-#' Parameter \code{alternative} is always \code{two.sided} when parameter 
-#' \code{type} is set to \code{asymptotic}.
-#' 
-#' @examples 
-#' ## Ratio for sir-object and the same values given manually:
-#' 
-#' 
-#' ## create example dataset
-#' dt1 <- data.frame(obs = rep(c(5,7), 10),
-#'                   pyrs = rep(c(250,300,350,400), 5),
-#'                   var = 1:20)
-#' Ref <- data.frame(obs = rep(c(50,70,80,100), 5),
-#'                  pyrs = rep(c(2500,3000,3500,4000), 5),
-#'                  var = 1:20)
-#' ## sir using the function
-#' s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
-#'           ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
-#'           adjust = var)
-#'
-#' ## Ratio is simply 1:
-#' sir_ratio(s1, c(120, 150))
-#' 
-#' @seealso \code{\link{sir}}
-#' \href{../doc/sir.html}{A SIR calculation vignette}
-#' 
-#' @references 
-#' Statistics with Confidence: Confidence Intervals and Statistical Guidelines, 
-#' Douglas Altman, 2000. ISBN: 978-0-727-91375-3
-#' 
-#' @family sir functions
-#' 
-#' @return A vector length of three: sir_ratio, and lower and upper confidence intervals.
-#' 
-#' @export sir_ratio
-#' 
-#' @import data.table
-#' @import stats
-
-
-sir_ratio <- function(x, y, digits = 3, alternative = 'two.sided', 
-                      conf.level = 0.95, type = 'exact') {
-  # prepare input values: x
-  # Tests are located in test_sir script.
-  if(inherits(x = x, what = 'sir')){
-    O1 <- sum(x$observed)
-    E1 <- sum(x$expected)
-  }
-  else if(is.vector(x) && length(x) == 2) {
-    O1 <- x[1]
-    E1 <- x[2]
-  }
-  else{
-    stop('Input x is not correct: x is neighter a vector of 2 nor sir-object')
-  }
-  # prepare y:
-  if(inherits(y,'sir')){
-    O2 <- sum(y$observed)
-    E2 <- sum(y$expected)
-  }
-  else if(is.vector(y) && length(y) == 2) {
-    O2 <- y[1]
-    E2 <- y[2]
-  }
-  else{
-    stop('Input y is not correct: y is neighter a vector of 2 nor sir-object')
-  }
-  
-  type <- match.arg(type, c('asymptotic', 'exact'), several.ok = FALSE)
-  alternative <- match.arg(alternative, c('two.sided','less', 'greater'), several.ok = FALSE)
-  # conf.level
-  
-  p <- O1/(O1+O2)
-  if(type == 'asymptotic') {
-    alpha <- (1 - conf.level)/2
-    Ex <- p + c(-qnorm(1-alpha),qnorm(1-alpha)) * sqrt((1/(O1+O2))*p*(1-p))
-    if( alternative != 'two.sided') {
-      message('Test changed to two.sided when asymptotic.')
-      alternative <- 'two.sided'
-    }
-  }
-  if(type == 'exact') {
-    Ex <- binom.test(c(O1,O2), p = 0.5, alternative = alternative, conf.level = conf.level)$conf.int
-  }
-  B = Ex/(1-Ex)
-  
-  res <- round(c(sir_ratio = (O1/E1)/(O2/E2), lower=(B*(E2/E1))[1], upper = (B*(E2/E1))[2]), digits = digits)
-  return(res)
-}
-
+#' @title Confidence intervals for the ratio of two SIRs/SMRs
+#' @author Matti Rantanen
+#' @description Calculate ratio of two SIRs/SMRs and the confidence intervals of the ratio.
+#' 
+#' @details Function works with pooled sir-objects i.e. the \code{print} argument in \code{sir} is ignored.
+#' Also \code{x} and \code{y} can be a vector of two where first index is the
+#' observed cases and second is expected cases (see examples).
+#' Note that the ratio of two SIR's is only applicable when the age distributions are similar
+#' in both populations.
+#'
+#' \strong{Formula}
+#' 
+#' The observed number of first sir \code{O1} is considered as a Binomial variable with sample 
+#' size of \code{O1+O2}. The confidence intervals for Binomial proportion \code{A} 
+#' is solved using \code{exact} or \code{asymptotic} 
+#' method. Now the CI for ratio \code{O1/O2} is \code{B = A/(1 - A)}. And further the CI for SIR/SMR 
+#' is B*E2/E1. (Ederer and Mantel)
+#' 
+#' @param x a sir-object or a vector of two; observed and expected cases.
+#' @param y a sir-object or a vector of two; observed and expected cases.
+#' @param conf.level the type-I error in confidence intervals, default 0.95 for 95\% CI.
+#' @param type How the binomial confidence intervals are calculated (default:) \code{exact} or \code{asymptotic}.
+#' @param alternative The null-hypothesis test: (default:) \code{two.sided}, \code{less}, \code{greater}
+#' @param digits number of digits in the output
+#' 
+#' @note
+#' Parameter \code{alternative} is always \code{two.sided} when parameter 
+#' \code{type} is set to \code{asymptotic}.
+#' 
+#' @examples 
+#' ## Ratio for sir-object and the same values given manually:
+#' 
+#' 
+#' ## create example dataset
+#' dt1 <- data.frame(obs = rep(c(5,7), 10),
+#'                   pyrs = rep(c(250,300,350,400), 5),
+#'                   var = 1:20)
+#' Ref <- data.frame(obs = rep(c(50,70,80,100), 5),
+#'                  pyrs = rep(c(2500,3000,3500,4000), 5),
+#'                  var = 1:20)
+#' ## sir using the function
+#' s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
+#'           ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
+#'           adjust = var)
+#'
+#' ## Ratio is simply 1:
+#' sir_ratio(s1, c(120, 150))
+#' 
+#' @seealso \code{\link{sir}}
+#' \href{../doc/sir.html}{A SIR calculation vignette}
+#' 
+#' @references 
+#' Statistics with Confidence: Confidence Intervals and Statistical Guidelines, 
+#' Douglas Altman, 2000. ISBN: 978-0-727-91375-3
+#' 
+#' @family sir functions
+#' 
+#' @return A vector length of three: sir_ratio, and lower and upper confidence intervals.
+#' 
+#' @export sir_ratio
+#' 
+#' @import data.table
+#' @import stats
+
+
+sir_ratio <- function(x, y, digits = 3, alternative = 'two.sided', 
+                      conf.level = 0.95, type = 'exact') {
+  # prepare input values: x
+  # Tests are located in test_sir script.
+  if(inherits(x = x, what = 'sir')){
+    O1 <- sum(x$observed)
+    E1 <- sum(x$expected)
+  }
+  else if(is.vector(x) && length(x) == 2) {
+    O1 <- x[1]
+    E1 <- x[2]
+  }
+  else{
+    stop('Input x is not correct: x is neighter a vector of 2 nor sir-object')
+  }
+  # prepare y:
+  if(inherits(y,'sir')){
+    O2 <- sum(y$observed)
+    E2 <- sum(y$expected)
+  }
+  else if(is.vector(y) && length(y) == 2) {
+    O2 <- y[1]
+    E2 <- y[2]
+  }
+  else{
+    stop('Input y is not correct: y is neighter a vector of 2 nor sir-object')
+  }
+  
+  type <- match.arg(type, c('asymptotic', 'exact'), several.ok = FALSE)
+  alternative <- match.arg(alternative, c('two.sided','less', 'greater'), several.ok = FALSE)
+  # conf.level
+  
+  p <- O1/(O1+O2)
+  if(type == 'asymptotic') {
+    alpha <- (1 - conf.level)/2
+    Ex <- p + c(-qnorm(1-alpha),qnorm(1-alpha)) * sqrt((1/(O1+O2))*p*(1-p))
+    if( alternative != 'two.sided') {
+      message('Test changed to two.sided when asymptotic.')
+      alternative <- 'two.sided'
+    }
+  }
+  if(type == 'exact') {
+    Ex <- binom.test(c(O1,O2), p = 0.5, alternative = alternative, conf.level = conf.level)$conf.int
+  }
+  B = Ex/(1-Ex)
+  
+  res <- round(c(sir_ratio = (O1/E1)/(O2/E2), lower=(B*(E2/E1))[1], upper = (B*(E2/E1))[2]), digits = digits)
+  return(res)
+}
+
diff --git a/R/splitLexisDT.R b/R/splitLexisDT.R
index 08cef4e..b75cb79 100644
--- a/R/splitLexisDT.R
+++ b/R/splitLexisDT.R
@@ -1,301 +1,301 @@
-#' @title Split case-level observations
-#' @author Joonas Miettinen
-#' @description Split a \code{Lexis} object along one time scale
-#' (as \code{\link[Epi]{splitLexis}}) with speed
-#' @param lex a Lexis object, split or not
-#' @param breaks a vector of \code{[a,b)} breaks to split \code{data} by
-#' @param timeScale a character string; name of the time scale to split by
-#' @param merge logical; if \code{TRUE}, retains all variables 
-#' from the original data - i.e. original variables are
-#' repeated for all the rows by original subject
-#' @param drop logical; if \code{TRUE}, drops all resulting rows 
-#' after expansion that reside outside the time window
-#' defined by the given breaks
-#' 
-#' 
-#' @details 
-#' 
-#' \code{splitLexisDT} is in essence a \pkg{data.table} version of
-#' \code{splitLexis} or \code{survSplit} for splitting along a single
-#' time scale. It requires a Lexis object as input, which may have already
-#' been split along some time scale.
-#' 
-#' Unlike \code{splitLexis}, \code{splitLexisDT} drops observed time outside
-#' the roof and floor of \code{breaks} by default - with \code{drop = FALSE}
-#' the functions have identical behaviour.
-#' 
-#' The \code{Lexis} time scale variables can be of any arbitrary 
-#' format, e.g. \code{Date},
-#' fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
-#' or other. However, using \code{date} variables (from package \pkg{date})
-#' are not recommended, as \code{date} variables are always stored as integers,
-#' whereas \code{Date} variables (see \code{?as.Date}) are typically stored
-#' in double ("numeric") format. This allows for breaking days into fractions
-#' as well, when using e.g. hypothetical years of 365.25 days.
-#' 
-#' @return
-#' A \code{data.table} or \code{data.frame} 
-#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-#' object expanded to accommodate split observations.
-#' 
-#' @export
-#' @family splitting functions
-#' @examples
-#' library(Epi)
-#' data("sire", package = "popEpi")
-#' x <- Lexis(data=sire[1000:1100, ], 
-#'            entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
-#'            exit=list(per=get.yrs(ex_date)), exit.status=status)
-#' BL <- list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
-#' 
-#' x2 <- splitMulti(x, breaks = BL, drop = FALSE)
-#' 
-#' x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
-#' x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
-#' 
-#' x4 <- splitLexis(x,  breaks = BL$fot, time.scale = "fot")
-#' x4 <- splitLexis(x4, breaks = BL$per, time.scale = "per")
-#' ## all produce identical results
-#' 
-#' ## using Date variables
-#' x <- Lexis(data=sire[1000:1100, ], 
-#'            entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
-#'            exit=list(per=ex_date), exit.status=status)
-#' BL <- list(fot = 0:5*365.25, per = as.Date(c("2008-01-01", "2013-01-01")))
-#' 
-#' x2 <- splitMulti(x, breaks = BL, drop = FALSE)
-#' 
-#' x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
-#' x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
-#' 
-#' ## splitLexis may not work when using Dates
-splitLexisDT <- function(lex, breaks, timeScale, merge = TRUE, drop = TRUE) {
-  
-  do_split <- TRUE
-  
-  tol <- .Machine$double.eps^0.5
-  checkLexisData(lex, check.breaks = FALSE)
-  
-  attr_list <- copy(attributes(lex)[c("time.scales", "breaks", "time.since")])
-  allScales <- attr_list[["time.scales"]]
-  allBreaks <- attr_list[["breaks"]]
-  
-  if (!timeScale %in% allScales) {
-    stop("timeScale '", timeScale,"' not among following existing time scales: ", 
-         paste0("'", allScales, "'", collapse = ", "))
-  }
-  
-  ## lexVars: if !merge, will drop all but these (NOTE: checkLexisData
-  ## check for existence of these)
-  lexVars <- c("lex.id", "lex.multi", allScales, "lex.dur", "lex.Cst", "lex.Xst")
-  lexVars <- intersect(lexVars, names(lex))
-  othVars <- setdiff(names(lex), lexVars)
-  
-  ## basic checks on breaks
-  if (drop && length(breaks) == 1L) {
-    stop("Length of breaks vector is one, but argument 'drop' is TRUE. ",
-         "Cannot do dropping with only one break. Either supply at least ",
-         "two breaks or set drop = FALSE.")
-  }
-  if (length(breaks) == 0L) {
-    stop("No breaks supplied (length of breaks is zero).")
-  }
-  
-  ## remove any existing breaks already split by;
-  ## NOTE: setdiff would break Date format breaks!
-  orig_breaks <- copy(breaks)
-  if (length(allBreaks[[timeScale]])) {
-    ## because any test like (x %in% NULL) results in FALSE.
-    breaks <- breaks[!breaks %in% allBreaks[[timeScale]]]
-  }
-  
-  breaks <- matchBreakTypes(lex, breaks, timeScale, modify.lex = FALSE) 
-  
-  if (length(breaks) == 0L || (length(orig_breaks) == 2L && drop)) {
-    ## former means no additional splitting to do. (we still crop & drop
-    ## if argument drop = TRUE)
-    ## latter means we only need to crop & drop.
-    do_split <- FALSE
-    breaks <- orig_breaks
-  }
-  
-  breaks <- sort(breaks)
-  if (!drop) breaks <- protectFromDrop(breaks)
-  
-  BL <- list(breaks)
-  setattr(BL, "names", timeScale)
-  checkBreaksList(x = lex, breaks = BL)
-  
-  
-  ## use subset lex if dropping for efficiency
-  orig_lex <- lex
-  if (drop) {
-    keepVars <- if (merge) NULL else lexVars ## NULL: all vars
-    lex <- subsetDTorDF(lex, select = keepVars)
-    rm(keepVars)
-    lex <- data.table(lex)
-    
-    setattr(lex, "class", c("Lexis", "data.table", "data.frame"))
-    
-    lex <- intelliCrop(lex, breaks = BL, allScales = allScales, 
-                       cropStatuses = TRUE, tol = tol)
-    lex <- intelliDrop(lex, breaks = BL, dropNegDur = TRUE, 
-                       check = FALSE, tol = tol)
-  }
-  
-  if (!do_split) {
-    
-    l <- if (!drop) copy(lex) else lex
-    
-  } else {
-    
-    ## currently cannot handle NA values in split time scale; will add them in
-    ## the end
-    ts_is_na <- is.na(lex[[timeScale]])
-    ts_any_na <- any(ts_is_na)
-    if (ts_any_na) {
-      warning("NA values in the time scale you are splitting along ('", 
-              timeScale,"'). Results may deviate from that produced by ",
-              "splitLexis from package Epi. For safety you may want to split ",
-              "using only the data with no NA values and combine the the split",
-              " data with the NA-valued data using rbind.")
-      lex_na <- lex[ts_is_na, ]
-      lex <- lex[!ts_is_na, ]
-    }
-    
-    
-    ## will use this due to step below (and laziness)
-    ts_values <- lex[[timeScale]]
-    ## Date objects are based on doubles and therefore keep the most information
-    if (inherits(ts_values, c("IDate", "date", "dates"))) ts_values <- as.Date(ts_values)
-    
-    N_expand <- length(breaks)
-    N_subjects <- nrow(lex)
-    
-    ## use tmp id to ensure correct status rolling -----------------------------
-    id_dt <- data.table(
-      tmp_id_values = 1:nrow(lex),
-      orig_id_values = lex[["lex.id"]]
-    )
-    on.exit(set(lex, j = "lex.id", value = id_dt[["orig_id_values"]]))
-    set(lex, j = "lex.id", value = id_dt[["tmp_id_values"]])
-    
-    ## quick data expansion ------------------------------------------------------
-    
-    l <- vector(mode = "list", length = N_expand)
-    l[[1]] <- data.table(lex)
-    
-    if (!merge) setcolsnull(l[[1]], keep = lexVars, soft = FALSE)
-    
-    tmpID <- makeTempVarName(data = l[[1]], pre = "TEMP_SPLITTING_ID")
-    tmpIE <- makeTempVarName(data = l[[1]], pre = "TEMP_SPLIT_INT_END")
-    
-    set(l[[1]], j = tmpID, value = 1:nrow(l[[1]]))
-    if (N_expand > 1L) {
-      for (k in 2:(N_expand)) {
-        l[[k]] <- l[[1]]
-      }
-    }
-    
-    l <- rbindlist(l)
-    
-    ## time scale value determination --------------------------------------------
-    set(l, j = tmpIE,  value = rep(breaks, each = N_subjects))
-    set(l, j = tmpIE,  value = pmin(l[[tmpIE]], l[[timeScale]] + l$lex.dur) )
-    set(l, j = timeScale, value = c(
-      ts_values, 
-      pmax(ts_values, rep(breaks[-length(breaks)], each = N_subjects))
-    ))
-    
-    set(l, j = "lex.dur", value = l[[tmpIE]] - l[[timeScale]] )
-    
-    ## other time scale values ---------------------------------------------------
-    otherScales <- setdiff(allScales, timeScale)
-    if (length(otherScales) > 0) {
-      ## change in timeScale
-      ts_delta <- l[[timeScale]] - ts_values
-      for (k in otherScales) {
-        set(l, j = k, value = lex[[k]] + ts_delta)
-      }
-    }
-    
-    ## dropping ----------------------------------------------------------------
-    ## drops very very small intervals as well as dur <= 0
-    has_zero_dur <- l[["lex.dur"]] < tol
-    if (any(has_zero_dur)) {
-      l <- l[!has_zero_dur]
-    }
-    
-    
-    
-    ## roll states -------------------------------------------------------------
-    # this avoids duplicate deaths, etc., where appropriate.
-    setkeyv(l, c("lex.id", timeScale))
-    lex_id <- mget_cols(c("lex.Cst", "lex.Xst", "lex.id"), data = lex)
-    setattr(lex_id, "time.scales", allScales)
-    roll_lexis_status_inplace(
-      unsplit.data = lex_id, split.data = l, id.var = "lex.id"
-    )
-    rm("lex_id")
-    
-    
-    set(l, j = c(tmpIE, tmpID), value = NULL)
-    
-    ## revert to original IDs --------------------------------------------------
-    set(l, j = "lex.id", value = {
-      id_dt[
-        i = list(tmp_id_values = l$lex.id), 
-        j = .SD, 
-        on = "tmp_id_values", 
-        .SDcols = "orig_id_values"
-        ]
-    })
-    
-    if (ts_any_na) {
-      l <- rbind(l, lex_na)
-      setkeyv(l, c("lex.id", timeScale))
-    }
-    
-  }
-  
-  ## harmonize statuses --------------------------------------------------------
-  harmonizeStatuses(x = l, C = "lex.Cst", X = "lex.Xst")
-  
-  
-  ## ensure time scales and lex.dur have same (ish) class as before ------------
-  for (k in c(allScales, "lex.dur")) {
-    
-    if (inherits(orig_lex[[k]], "difftime") && !inherits(l[[k]], "difftime")){
-      setattr(l[[k]], "class", "difftime")
-      setattr(l[[k]], "units", attr(orig_lex[[k]], "units"))
-    } else if (is.numeric(orig_lex[[k]]) && inherits(l[[k]], "difftime")) {
-      set(l, j = k, value = as.numeric(l[[k]]))
-    }
-    
-  }
-  
-  ## harmonize time scales -----------------------------------------------------
-  ## numeric time scales are forced to the lowest common denominator:
-  ## difftime -> integer -> double (though difftime is not numeric class)
-  harmonizeNumericTimeScales(l, times = c(allScales, "lex.dur"))
-  
-  
-  ## final touch & attributes --------------------------------------------------
-  setcolorder(l, neworder = intersect(c(lexVars, othVars), names(l))) #merge=T/F
-  if (!drop) breaks <- unprotectFromDrop(breaks)
-  allBreaks[[timeScale]] <- sort(unique(c(allBreaks[[timeScale]], breaks)))
-  
-  allBreaks <- lapply(allScales, function(scale_nm) {
-    allBreaks[[scale_nm]] ## intentionally NULL if not there
-  }) 
-  names(allBreaks) <- allScales
-  
-  setattr(l, "breaks", allBreaks)
-  setattr(l, "time.scales", allScales)
-  setattr(l, "time.since", attr_list[["time.since"]])
-  setattr(l, "class", c("Lexis","data.table","data.frame"))
-  if (!return_DT()) setDFpe(l)
-  
-  l[]
-}
-
+#' @title Split case-level observations
+#' @author Joonas Miettinen
+#' @description Split a \code{Lexis} object along one time scale
+#' (as \code{\link[Epi]{splitLexis}}) with speed
+#' @param lex a Lexis object, split or not
+#' @param breaks a vector of \code{[a,b)} breaks to split \code{data} by
+#' @param timeScale a character string; name of the time scale to split by
+#' @param merge logical; if \code{TRUE}, retains all variables 
+#' from the original data - i.e. original variables are
+#' repeated for all the rows by original subject
+#' @param drop logical; if \code{TRUE}, drops all resulting rows 
+#' after expansion that reside outside the time window
+#' defined by the given breaks
+#' 
+#' 
+#' @details 
+#' 
+#' \code{splitLexisDT} is in essence a \pkg{data.table} version of
+#' \code{splitLexis} or \code{survSplit} for splitting along a single
+#' time scale. It requires a Lexis object as input, which may have already
+#' been split along some time scale.
+#' 
+#' Unlike \code{splitLexis}, \code{splitLexisDT} drops observed time outside
+#' the roof and floor of \code{breaks} by default - with \code{drop = FALSE}
+#' the functions have identical behaviour.
+#' 
+#' The \code{Lexis} time scale variables can be of any arbitrary 
+#' format, e.g. \code{Date},
+#' fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
+#' or other. However, using \code{date} variables (from package \pkg{date})
+#' are not recommended, as \code{date} variables are always stored as integers,
+#' whereas \code{Date} variables (see \code{?as.Date}) are typically stored
+#' in double ("numeric") format. This allows for breaking days into fractions
+#' as well, when using e.g. hypothetical years of 365.25 days.
+#' 
+#' @return
+#' A \code{data.table} or \code{data.frame} 
+#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+#' object expanded to accommodate split observations.
+#' 
+#' @export
+#' @family splitting functions
+#' @examples
+#' library(Epi)
+#' data("sire", package = "popEpi")
+#' x <- Lexis(data=sire[1000:1100, ], 
+#'            entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
+#'            exit=list(per=get.yrs(ex_date)), exit.status=status)
+#' BL <- list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
+#' 
+#' x2 <- splitMulti(x, breaks = BL, drop = FALSE)
+#' 
+#' x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
+#' x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
+#' 
+#' x4 <- splitLexis(x,  breaks = BL$fot, time.scale = "fot")
+#' x4 <- splitLexis(x4, breaks = BL$per, time.scale = "per")
+#' ## all produce identical results
+#' 
+#' ## using Date variables
+#' x <- Lexis(data=sire[1000:1100, ], 
+#'            entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
+#'            exit=list(per=ex_date), exit.status=status)
+#' BL <- list(fot = 0:5*365.25, per = as.Date(c("2008-01-01", "2013-01-01")))
+#' 
+#' x2 <- splitMulti(x, breaks = BL, drop = FALSE)
+#' 
+#' x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
+#' x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
+#' 
+#' ## splitLexis may not work when using Dates
+splitLexisDT <- function(lex, breaks, timeScale, merge = TRUE, drop = TRUE) {
+  
+  do_split <- TRUE
+  
+  tol <- .Machine$double.eps^0.5
+  checkLexisData(lex, check.breaks = FALSE)
+  
+  attr_list <- copy(attributes(lex)[c("time.scales", "breaks", "time.since")])
+  allScales <- attr_list[["time.scales"]]
+  allBreaks <- attr_list[["breaks"]]
+  
+  if (!timeScale %in% allScales) {
+    stop("timeScale '", timeScale,"' not among following existing time scales: ", 
+         paste0("'", allScales, "'", collapse = ", "))
+  }
+  
+  ## lexVars: if !merge, will drop all but these (NOTE: checkLexisData
+  ## check for existence of these)
+  lexVars <- c("lex.id", "lex.multi", allScales, "lex.dur", "lex.Cst", "lex.Xst")
+  lexVars <- intersect(lexVars, names(lex))
+  othVars <- setdiff(names(lex), lexVars)
+  
+  ## basic checks on breaks
+  if (drop && length(breaks) == 1L) {
+    stop("Length of breaks vector is one, but argument 'drop' is TRUE. ",
+         "Cannot do dropping with only one break. Either supply at least ",
+         "two breaks or set drop = FALSE.")
+  }
+  if (length(breaks) == 0L) {
+    stop("No breaks supplied (length of breaks is zero).")
+  }
+  
+  ## remove any existing breaks already split by;
+  ## NOTE: setdiff would break Date format breaks!
+  orig_breaks <- copy(breaks)
+  if (length(allBreaks[[timeScale]])) {
+    ## because any test like (x %in% NULL) results in FALSE.
+    breaks <- breaks[!breaks %in% allBreaks[[timeScale]]]
+  }
+  
+  breaks <- matchBreakTypes(lex, breaks, timeScale, modify.lex = FALSE) 
+  
+  if (length(breaks) == 0L || (length(orig_breaks) == 2L && drop)) {
+    ## former means no additional splitting to do. (we still crop & drop
+    ## if argument drop = TRUE)
+    ## latter means we only need to crop & drop.
+    do_split <- FALSE
+    breaks <- orig_breaks
+  }
+  
+  breaks <- sort(breaks)
+  if (!drop) breaks <- protectFromDrop(breaks)
+  
+  BL <- list(breaks)
+  setattr(BL, "names", timeScale)
+  checkBreaksList(x = lex, breaks = BL)
+  
+  
+  ## use subset lex if dropping for efficiency
+  orig_lex <- lex
+  if (drop) {
+    keepVars <- if (merge) NULL else lexVars ## NULL: all vars
+    lex <- subsetDTorDF(lex, select = keepVars)
+    rm(keepVars)
+    lex <- data.table(lex)
+    
+    setattr(lex, "class", c("Lexis", "data.table", "data.frame"))
+    
+    lex <- intelliCrop(lex, breaks = BL, allScales = allScales, 
+                       cropStatuses = TRUE, tol = tol)
+    lex <- intelliDrop(lex, breaks = BL, dropNegDur = TRUE, 
+                       check = FALSE, tol = tol)
+  }
+  
+  if (!do_split) {
+    
+    l <- if (!drop) copy(lex) else lex
+    
+  } else {
+    
+    ## currently cannot handle NA values in split time scale; will add them in
+    ## the end
+    ts_is_na <- is.na(lex[[timeScale]])
+    ts_any_na <- any(ts_is_na)
+    if (ts_any_na) {
+      warning("NA values in the time scale you are splitting along ('", 
+              timeScale,"'). Results may deviate from that produced by ",
+              "splitLexis from package Epi. For safety you may want to split ",
+              "using only the data with no NA values and combine the the split",
+              " data with the NA-valued data using rbind.")
+      lex_na <- lex[ts_is_na, ]
+      lex <- lex[!ts_is_na, ]
+    }
+    
+    
+    ## will use this due to step below (and laziness)
+    ts_values <- lex[[timeScale]]
+    ## Date objects are based on doubles and therefore keep the most information
+    if (inherits(ts_values, c("IDate", "date", "dates"))) ts_values <- as.Date(ts_values)
+    
+    N_expand <- length(breaks)
+    N_subjects <- nrow(lex)
+    
+    ## use tmp id to ensure correct status rolling -----------------------------
+    id_dt <- data.table(
+      tmp_id_values = 1:nrow(lex),
+      orig_id_values = lex[["lex.id"]]
+    )
+    on.exit(set(lex, j = "lex.id", value = id_dt[["orig_id_values"]]))
+    set(lex, j = "lex.id", value = id_dt[["tmp_id_values"]])
+    
+    ## quick data expansion ------------------------------------------------------
+    
+    l <- vector(mode = "list", length = N_expand)
+    l[[1]] <- data.table(lex)
+    
+    if (!merge) setcolsnull(l[[1]], keep = lexVars, soft = FALSE)
+    
+    tmpID <- makeTempVarName(data = l[[1]], pre = "TEMP_SPLITTING_ID")
+    tmpIE <- makeTempVarName(data = l[[1]], pre = "TEMP_SPLIT_INT_END")
+    
+    set(l[[1]], j = tmpID, value = 1:nrow(l[[1]]))
+    if (N_expand > 1L) {
+      for (k in 2:(N_expand)) {
+        l[[k]] <- l[[1]]
+      }
+    }
+    
+    l <- rbindlist(l)
+    
+    ## time scale value determination --------------------------------------------
+    set(l, j = tmpIE,  value = rep(breaks, each = N_subjects))
+    set(l, j = tmpIE,  value = pmin(l[[tmpIE]], l[[timeScale]] + l$lex.dur) )
+    set(l, j = timeScale, value = c(
+      ts_values, 
+      pmax(ts_values, rep(breaks[-length(breaks)], each = N_subjects))
+    ))
+    
+    set(l, j = "lex.dur", value = l[[tmpIE]] - l[[timeScale]] )
+    
+    ## other time scale values ---------------------------------------------------
+    otherScales <- setdiff(allScales, timeScale)
+    if (length(otherScales) > 0) {
+      ## change in timeScale
+      ts_delta <- l[[timeScale]] - ts_values
+      for (k in otherScales) {
+        set(l, j = k, value = lex[[k]] + ts_delta)
+      }
+    }
+    
+    ## dropping ----------------------------------------------------------------
+    ## drops very very small intervals as well as dur <= 0
+    has_zero_dur <- l[["lex.dur"]] < tol
+    if (any(has_zero_dur)) {
+      l <- l[!has_zero_dur]
+    }
+    
+    
+    
+    ## roll states -------------------------------------------------------------
+    # this avoids duplicate deaths, etc., where appropriate.
+    setkeyv(l, c("lex.id", timeScale))
+    lex_id <- mget_cols(c("lex.Cst", "lex.Xst", "lex.id"), data = lex)
+    setattr(lex_id, "time.scales", allScales)
+    roll_lexis_status_inplace(
+      unsplit.data = lex_id, split.data = l, id.var = "lex.id"
+    )
+    rm("lex_id")
+    
+    
+    set(l, j = c(tmpIE, tmpID), value = NULL)
+    
+    ## revert to original IDs --------------------------------------------------
+    set(l, j = "lex.id", value = {
+      id_dt[
+        i = list(tmp_id_values = l$lex.id), 
+        j = .SD, 
+        on = "tmp_id_values", 
+        .SDcols = "orig_id_values"
+        ]
+    })
+    
+    if (ts_any_na) {
+      l <- rbind(l, lex_na)
+      setkeyv(l, c("lex.id", timeScale))
+    }
+    
+  }
+  
+  ## harmonize statuses --------------------------------------------------------
+  harmonizeStatuses(x = l, C = "lex.Cst", X = "lex.Xst")
+  
+  
+  ## ensure time scales and lex.dur have same (ish) class as before ------------
+  for (k in c(allScales, "lex.dur")) {
+    
+    if (inherits(orig_lex[[k]], "difftime") && !inherits(l[[k]], "difftime")){
+      setattr(l[[k]], "class", "difftime")
+      setattr(l[[k]], "units", attr(orig_lex[[k]], "units"))
+    } else if (is.numeric(orig_lex[[k]]) && inherits(l[[k]], "difftime")) {
+      set(l, j = k, value = as.numeric(l[[k]]))
+    }
+    
+  }
+  
+  ## harmonize time scales -----------------------------------------------------
+  ## numeric time scales are forced to the lowest common denominator:
+  ## difftime -> integer -> double (though difftime is not numeric class)
+  harmonizeNumericTimeScales(l, times = c(allScales, "lex.dur"))
+  
+  
+  ## final touch & attributes --------------------------------------------------
+  setcolorder(l, neworder = intersect(c(lexVars, othVars), names(l))) #merge=T/F
+  if (!drop) breaks <- unprotectFromDrop(breaks)
+  allBreaks[[timeScale]] <- sort(unique(c(allBreaks[[timeScale]], breaks)))
+  
+  allBreaks <- lapply(allScales, function(scale_nm) {
+    allBreaks[[scale_nm]] ## intentionally NULL if not there
+  }) 
+  names(allBreaks) <- allScales
+  
+  setattr(l, "breaks", allBreaks)
+  setattr(l, "time.scales", allScales)
+  setattr(l, "time.since", attr_list[["time.since"]])
+  setattr(l, "class", c("Lexis","data.table","data.frame"))
+  if (!return_DT()) setDFpe(l)
+  
+  l[]
+}
+
diff --git a/R/splitMulti.R b/R/splitMulti.R
index 703a048..9b6f4bd 100644
--- a/R/splitMulti.R
+++ b/R/splitMulti.R
@@ -1,280 +1,280 @@
-#' @title Split case-level observations
-#' @author Joonas Miettinen
-#' @description Split a \code{Lexis} object along multiple time scales
-#' with speed and ease
-#' @param data a Lexis object with event cases as rows
-#' @param breaks a list of named numeric vectors of breaks; see Details and Examples
-#' @param ... alternate way of supplying breaks as named vectors;
-#' e.g. \code{fot = 0:5} instead of \code{breaks = list(fot = 0:5)};
-#' if \code{breaks} is not \code{NULL}, \code{breaks} is used and any breaks
-#' passed through \code{...} are NOT used; note also that due to partial 
-#' matching of argument names in R, 
-#' if you supply e.g. \code{dat = my_breaks} and you 
-#' do not pass argument \code{data} explicitly (\code{data = my_data}), then R
-#' interprets this as \code{data = my_breaks} --- so choose the names of your
-#' time scales wisely
-#' @param drop logical; if \code{TRUE}, drops all resulting rows 
-#' after expansion that reside outside the time window
-#' defined by the given breaks
-#' @param merge logical; if \code{TRUE}, retains all variables 
-#' from the original data - i.e. original variables are
-#' repeated for all the rows by original subject
-#' @param verbose logical; if \code{TRUE}, the function is chatty 
-#' and returns some messages along the way
-#' 
-#' 
-#' @details 
-#' 
-#' \code{splitMulti} is in essence a \pkg{data.table} version of
-#'  \code{splitLexis} or \code{survSplit} for splitting along multiple
-#'  time scales.
-#' It requires a Lexis object as input.
-#' 
-#' The \code{breaks} must be a list of named vectors of the appropriate type. 
-#' The breaks are fully explicit and
-#' left-inclusive and right exclusive, e.g. \code{fot=c(0,5)} 
-#' forces the data to only include time between
-#' \code{[0,5)} for each original row (unless \code{drop = FALSE}). 
-#' Use \code{Inf} or \code{-Inf} for open-ended intervals,
-#'  e.g. \code{per=c(1990,1995,Inf)} creates the intervals 
-#'  \code{[1990,1995), [1995, Inf)}.
-#'  
-#' Instead of specifying \code{breaks}, one may make use of the \code{...}
-#' argument to pass breaks: e.g. 
-#' 
-#' \code{splitMulti(x, breaks = list(fot = 0:5))} 
-#' 
-#' is equivalent to
-#' 
-#' \code{splitMulti(x, fot = 0:5)}.
-#' 
-#' Multiple breaks can be supplied in the same manner. However, if both
-#' \code{breaks} and \code{...} are used, only the breaks in \code{breaks}
-#' are utilized within the function. 
-#' 
-#' The \code{Lexis} time scale variables can be of any arbitrary 
-#' format, e.g. \code{Date},
-#' fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
-#' or other. However, using \code{date} variables (from package \pkg{date})
-#' are not recommended, as \code{date} variables are always stored as integers,
-#' whereas \code{Date} variables (see \code{?as.Date}) are typically stored
-#' in double ("numeric") format. This allows for breaking days into fractions
-#' as well, when using e.g. hypothetical years of 365.25 days.
-#'  
-#' @return
-#' A \code{data.table} or \code{data.frame} 
-#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-#' object expanded to accommodate split observations.
-#' 
-#' @examples
-#' #### let's prepare data for computing period method survivals
-#' #### in case there are problems with dates, we first 
-#' #### convert to fractional years.
-#' \donttest{
-#' library("Epi")
-#' library("data.table")
-#' data("sire", package = "popEpi")
-#' x <- Lexis(data=sire[dg_date < ex_date, ], 
-#'            entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
-#'            exit=list(per=get.yrs(ex_date)), exit.status=status)
-#' x2 <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013)))
-#' # equivalently:
-#' x2 <- splitMulti(x, fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
-#' 
-#' ## using dates; note: breaks must be expressed as dates or days!
-#' x <- Lexis(data=sire[dg_date < ex_date, ], 
-#'            entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
-#'            exit=list(per=ex_date), exit.status=status)
-#' BL <- list(fot = seq(0, 5, by = 3/12)*365.242199,
-#'            per = as.Date(paste0(c(1980:2014),"-01-01")),
-#'            age = c(0,45,85,Inf)*365.242199)
-#' x2 <- splitMulti(x, breaks = BL, verbose=TRUE)
-#' 
-#' 
-#' ## multistate example (healty - sick - dead)
-#' sire2 <- data.frame(sire)
-#' sire2 <- sire2[sire2$dg_date < sire2$ex_date, ]
-#' 
-#' set.seed(1L) 
-#' not_sick <- sample.int(nrow(sire2), 6000L, replace = FALSE)
-#' sire2$dg_date[not_sick] <- NA
-#' sire2$status[!is.na(sire2$dg_date) & sire2$status == 0] <- -1
-#' 
-#' sire2$status[sire2$status==2] <- 1
-#' sire2$status <- factor(sire2$status, levels = c(0, -1, 1), 
-#'                        labels = c("healthy", "sick", "dead"))
-#'  
-#' xm <- Lexis(data = sire2, 
-#'             entry = list(fot=0, per=get.yrs(bi_date), age=0), 
-#'             exit = list(per=get.yrs(ex_date)), exit.status=status)
-#' xm2 <- cutLexis(xm, cut = get.yrs(xm$dg_date), 
-#'                 timescale = "per", 
-#'                 new.state = "sick")
-#' xm2[xm2$lex.id == 6L, ]
-#' 
-#' xm2 <- splitMulti(xm2, breaks = list(fot = seq(0,150,25)))
-#' xm2[xm2$lex.id == 6L, ]
-#' }
-#' 
-#' @import data.table 
-#' @import Epi
-#' 
-#' @export
-#' @family splitting functions
-#' @seealso
-#' \code{\link[Epi]{splitLexis}}, \code{\link[Epi]{Lexis}},  
-#' \code{\link[survival]{survSplit}}
-#' 
-splitMulti <- function(data,
-                       breaks = NULL,
-                       ...,
-                       drop=TRUE,
-                       merge=TRUE,
-                       verbose=FALSE) {
-  
-  lex.id <- lex.dur <- NULL ## APPEASE R CMD CHECK
-  
-  ## basic checks --------------------------------------------------------------
-  if (verbose) {stime <- proc.time()}
-  
-  breaks <- splitMultiPreCheck(data = data, breaks = breaks, ...)
-  
-  ## collect necessary data ----------------------------------------------------
-  attr_list <- copy(attributes(data)[c("time.scales", "breaks", "time.since")])
-  allScales <- attr_list$time.scales
-  splitScales <- names(breaks)
-  
-  keep_nms <- if (merge) names(data) else {
-    intersect(
-      names(data), 
-      c("lex.id", "lex.Cst", "lex.Xst", allScales)
-    )
-  }
-  # this is not a copy!
-  dt <- mget_cols(keep_nms, data = data)
-  forceLexisDT(dt, breaks = attr(data, "breaks"), allScales = allScales,
-               key = FALSE)
-  
-  ## check if even need to do splitting ----------------------------------------
-  
-  oldBreaks <- copy(attr(data, "breaks"))
-  tryCatch(checkBreaksList(data, oldBreaks), error = function(e) {
-    stop("Error in splitMulti: \n",
-         "Old breaks existing in Lexis data did not pass testing. Error ",
-         "message from test: \n", e, call. = FALSE)
-  })
-  
-  ## only do split if all breaks are NOT in the breaks that the data
-  ## has already been split by.
-  do_split <- TRUE
-  do_split <- !all_breaks_in(breaks, oldBreaks, x = data)
-  
-  if (!do_split) {
-    l <- setDT(copy(dt))
-    setkeyv(l, c("lex.id", allScales[1]))
-  } else {
-    
-    ## temp IDS ----------------------------------------------------------------
-    # used to ensure correct splitting and lex status rolling
-    
-    id_dt <- data.table(
-      orig_id_values = dt$lex.id, 
-      temp_id_values = 1:nrow(dt), 
-      key = "temp_id_values"
-    )
-    
-    on.exit(set(dt, j = "lex.id", value = id_dt[["orig_id_values"]]))
-    set(dt, j = "lex.id", value = id_dt[["temp_id_values"]])
-    
-    l <- vector(mode = "list", length = length(splitScales))
-    setattr(l, "names", splitScales)
-    for (v in splitScales) {
-      l[[v]] <- splitLexisDT(dt, breaks = breaks[[v]], 
-                             merge = merge, drop = FALSE, timeScale = v)
-      breaks[[v]] <- attr(l[[v]], "breaks")[[v]]
-    }
-    l <- rbindlist(l)
-    
-    s1 <- allScales[1]
-    setkeyv(l, c("lex.id", s1))
-    
-    if (length(splitScales) > 1L) {
-      ## roll time scale values, re-compute interval lengths (lex.dur) ---------
-      
-      tmp_ie <- makeTempVarName(names = names(l), pre = "TEMP_INT_END_")
-      l[, (tmp_ie) := shift(.SD, n = 1, type = "lead"), 
-        .SDcols = s1, by = "lex.id"]
-      is_last_row <- is.na(l[[tmp_ie]])
-      
-      l[is_last_row, (tmp_ie) := lex.dur + .SD, .SDcols = s1]
-      
-      set(l, j = "lex.dur", value = l[[tmp_ie]] - l[[s1]])
-      set(l, j = tmp_ie, value = NULL)
-    }
-    
-    has_zero_dur <- l[["lex.dur"]] < .Machine$double.eps^0.5
-    if (any(has_zero_dur)) {
-      l <- l[!has_zero_dur, ]
-    }
-    
-    ## ensure statuses are as expected -----------------------------------------
-    
-    
-    setkeyv(l, c("lex.id", s1))
-    roll_lexis_status_inplace(
-      unsplit.data = dt, split.data = l, id.var = "lex.id"
-    )
-    
-    ## dt$lex.id from temporary values to original values ----------------------
-    # merge in correct IDs also to split data
-    on.exit()
-    set(dt, j = "lex.id", value = id_dt$lex.id)
-    
-    
-    tmpID <- makeTempVarName(names = names(l), pre = "TEMP_SPLITMULTI_ID_")
-    setnames(l, old = "lex.id", new = tmpID)
-    set(l, j = "lex.id", value = {id_dt[
-      i = .(l[[tmpID]]), 
-      j = .SD, 
-      on = "temp_id_values",
-      .SDcols = "orig_id_values"
-      ]})
-    set(l, j = tmpID, value = NULL)
-    rm("id_dt")
-    
-  }
-  
-  if (drop) l <- intelliDrop(l, breaks = breaks, dropNegDur = FALSE)
-  
-  if (nrow(l) == 0) {
-    warning("no data left after dropping; check breaks?")
-  }
-  
-  order <- c("lex.id", "lex.multi", allScales, "lex.dur", "lex.Cst", "lex.Xst")
-  order <- c(order, setdiff(names(l), order))
-  order <- intersect(order, names(l))
-  setcolorder(l, order)
-  
-  if (verbose) cat("time taken by splitting process: ", timetaken(stime), "\n")
-  
-  
-  breaks <- lapply(allScales, function(scale_nm) {
-    ## allowed to NULL also
-    br <- c(breaks[[scale_nm]], oldBreaks[[scale_nm]])
-    if (is.null(br)) return(br)
-    sort(unique(br))
-  }) 
-  names(breaks) <- allScales
-  
-  setattr(l, "time.scales", allScales)
-  setattr(l, "time.since", attr_list[["time.since"]])
-  setattr(l, "breaks", breaks)
-  setattr(l, "class", c("Lexis","data.table","data.frame"))
-  if (!return_DT()) setDFpe(l)
-  
-  l[]
-  
-}
-
-globalVariables(".")
-
+#' @title Split case-level observations
+#' @author Joonas Miettinen
+#' @description Split a \code{Lexis} object along multiple time scales
+#' with speed and ease
+#' @param data a Lexis object with event cases as rows
+#' @param breaks a list of named numeric vectors of breaks; see Details and Examples
+#' @param ... alternate way of supplying breaks as named vectors;
+#' e.g. \code{fot = 0:5} instead of \code{breaks = list(fot = 0:5)};
+#' if \code{breaks} is not \code{NULL}, \code{breaks} is used and any breaks
+#' passed through \code{...} are NOT used; note also that due to partial 
+#' matching of argument names in R, 
+#' if you supply e.g. \code{dat = my_breaks} and you 
+#' do not pass argument \code{data} explicitly (\code{data = my_data}), then R
+#' interprets this as \code{data = my_breaks} --- so choose the names of your
+#' time scales wisely
+#' @param drop logical; if \code{TRUE}, drops all resulting rows 
+#' after expansion that reside outside the time window
+#' defined by the given breaks
+#' @param merge logical; if \code{TRUE}, retains all variables 
+#' from the original data - i.e. original variables are
+#' repeated for all the rows by original subject
+#' @param verbose logical; if \code{TRUE}, the function is chatty 
+#' and returns some messages along the way
+#' 
+#' 
+#' @details 
+#' 
+#' \code{splitMulti} is in essence a \pkg{data.table} version of
+#'  \code{splitLexis} or \code{survSplit} for splitting along multiple
+#'  time scales.
+#' It requires a Lexis object as input.
+#' 
+#' The \code{breaks} must be a list of named vectors of the appropriate type. 
+#' The breaks are fully explicit and
+#' left-inclusive and right exclusive, e.g. \code{fot=c(0,5)} 
+#' forces the data to only include time between
+#' \code{[0,5)} for each original row (unless \code{drop = FALSE}). 
+#' Use \code{Inf} or \code{-Inf} for open-ended intervals,
+#'  e.g. \code{per=c(1990,1995,Inf)} creates the intervals 
+#'  \code{[1990,1995), [1995, Inf)}.
+#'  
+#' Instead of specifying \code{breaks}, one may make use of the \code{...}
+#' argument to pass breaks: e.g. 
+#' 
+#' \code{splitMulti(x, breaks = list(fot = 0:5))} 
+#' 
+#' is equivalent to
+#' 
+#' \code{splitMulti(x, fot = 0:5)}.
+#' 
+#' Multiple breaks can be supplied in the same manner. However, if both
+#' \code{breaks} and \code{...} are used, only the breaks in \code{breaks}
+#' are utilized within the function. 
+#' 
+#' The \code{Lexis} time scale variables can be of any arbitrary 
+#' format, e.g. \code{Date},
+#' fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
+#' or other. However, using \code{date} variables (from package \pkg{date})
+#' are not recommended, as \code{date} variables are always stored as integers,
+#' whereas \code{Date} variables (see \code{?as.Date}) are typically stored
+#' in double ("numeric") format. This allows for breaking days into fractions
+#' as well, when using e.g. hypothetical years of 365.25 days.
+#'  
+#' @return
+#' A \code{data.table} or \code{data.frame} 
+#' (depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+#' object expanded to accommodate split observations.
+#' 
+#' @examples
+#' #### let's prepare data for computing period method survivals
+#' #### in case there are problems with dates, we first 
+#' #### convert to fractional years.
+#' \donttest{
+#' library("Epi")
+#' library("data.table")
+#' data("sire", package = "popEpi")
+#' x <- Lexis(data=sire[dg_date < ex_date, ], 
+#'            entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
+#'            exit=list(per=get.yrs(ex_date)), exit.status=status)
+#' x2 <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013)))
+#' # equivalently:
+#' x2 <- splitMulti(x, fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
+#' 
+#' ## using dates; note: breaks must be expressed as dates or days!
+#' x <- Lexis(data=sire[dg_date < ex_date, ], 
+#'            entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
+#'            exit=list(per=ex_date), exit.status=status)
+#' BL <- list(fot = seq(0, 5, by = 3/12)*365.242199,
+#'            per = as.Date(paste0(c(1980:2014),"-01-01")),
+#'            age = c(0,45,85,Inf)*365.242199)
+#' x2 <- splitMulti(x, breaks = BL, verbose=TRUE)
+#' 
+#' 
+#' ## multistate example (healty - sick - dead)
+#' sire2 <- data.frame(sire)
+#' sire2 <- sire2[sire2$dg_date < sire2$ex_date, ]
+#' 
+#' set.seed(1L) 
+#' not_sick <- sample.int(nrow(sire2), 6000L, replace = FALSE)
+#' sire2$dg_date[not_sick] <- NA
+#' sire2$status[!is.na(sire2$dg_date) & sire2$status == 0] <- -1
+#' 
+#' sire2$status[sire2$status==2] <- 1
+#' sire2$status <- factor(sire2$status, levels = c(0, -1, 1), 
+#'                        labels = c("healthy", "sick", "dead"))
+#'  
+#' xm <- Lexis(data = sire2, 
+#'             entry = list(fot=0, per=get.yrs(bi_date), age=0), 
+#'             exit = list(per=get.yrs(ex_date)), exit.status=status)
+#' xm2 <- cutLexis(xm, cut = get.yrs(xm$dg_date), 
+#'                 timescale = "per", 
+#'                 new.state = "sick")
+#' xm2[xm2$lex.id == 6L, ]
+#' 
+#' xm2 <- splitMulti(xm2, breaks = list(fot = seq(0,150,25)))
+#' xm2[xm2$lex.id == 6L, ]
+#' }
+#' 
+#' @import data.table 
+#' @import Epi
+#' 
+#' @export
+#' @family splitting functions
+#' @seealso
+#' \code{\link[Epi]{splitLexis}}, \code{\link[Epi]{Lexis}},  
+#' \code{\link[survival]{survSplit}}
+#' 
+splitMulti <- function(data,
+                       breaks = NULL,
+                       ...,
+                       drop=TRUE,
+                       merge=TRUE,
+                       verbose=FALSE) {
+  
+  lex.id <- lex.dur <- NULL ## APPEASE R CMD CHECK
+  
+  ## basic checks --------------------------------------------------------------
+  if (verbose) {stime <- proc.time()}
+  
+  breaks <- splitMultiPreCheck(data = data, breaks = breaks, ...)
+  
+  ## collect necessary data ----------------------------------------------------
+  attr_list <- copy(attributes(data)[c("time.scales", "breaks", "time.since")])
+  allScales <- attr_list$time.scales
+  splitScales <- names(breaks)
+  
+  keep_nms <- if (merge) names(data) else {
+    intersect(
+      names(data), 
+      c("lex.id", "lex.Cst", "lex.Xst", allScales)
+    )
+  }
+  # this is not a copy!
+  dt <- mget_cols(keep_nms, data = data)
+  forceLexisDT(dt, breaks = attr(data, "breaks"), allScales = allScales,
+               key = FALSE)
+  
+  ## check if even need to do splitting ----------------------------------------
+  
+  oldBreaks <- copy(attr(data, "breaks"))
+  tryCatch(checkBreaksList(data, oldBreaks), error = function(e) {
+    stop("Error in splitMulti: \n",
+         "Old breaks existing in Lexis data did not pass testing. Error ",
+         "message from test: \n", e, call. = FALSE)
+  })
+  
+  ## only do split if all breaks are NOT in the breaks that the data
+  ## has already been split by.
+  do_split <- TRUE
+  do_split <- !all_breaks_in(breaks, oldBreaks, x = data)
+  
+  if (!do_split) {
+    l <- setDT(copy(dt))
+    setkeyv(l, c("lex.id", allScales[1]))
+  } else {
+    
+    ## temp IDS ----------------------------------------------------------------
+    # used to ensure correct splitting and lex status rolling
+    
+    id_dt <- data.table(
+      orig_id_values = dt$lex.id, 
+      temp_id_values = 1:nrow(dt), 
+      key = "temp_id_values"
+    )
+    
+    on.exit(set(dt, j = "lex.id", value = id_dt[["orig_id_values"]]))
+    set(dt, j = "lex.id", value = id_dt[["temp_id_values"]])
+    
+    l <- vector(mode = "list", length = length(splitScales))
+    setattr(l, "names", splitScales)
+    for (v in splitScales) {
+      l[[v]] <- splitLexisDT(dt, breaks = breaks[[v]], 
+                             merge = merge, drop = FALSE, timeScale = v)
+      breaks[[v]] <- attr(l[[v]], "breaks")[[v]]
+    }
+    l <- rbindlist(l)
+    
+    s1 <- allScales[1]
+    setkeyv(l, c("lex.id", s1))
+    
+    if (length(splitScales) > 1L) {
+      ## roll time scale values, re-compute interval lengths (lex.dur) ---------
+      
+      tmp_ie <- makeTempVarName(names = names(l), pre = "TEMP_INT_END_")
+      l[, (tmp_ie) := shift(.SD, n = 1, type = "lead"), 
+        .SDcols = s1, by = "lex.id"]
+      is_last_row <- is.na(l[[tmp_ie]])
+      
+      l[is_last_row, (tmp_ie) := lex.dur + .SD, .SDcols = s1]
+      
+      set(l, j = "lex.dur", value = l[[tmp_ie]] - l[[s1]])
+      set(l, j = tmp_ie, value = NULL)
+    }
+    
+    has_zero_dur <- l[["lex.dur"]] < .Machine$double.eps^0.5
+    if (any(has_zero_dur)) {
+      l <- l[!has_zero_dur, ]
+    }
+    
+    ## ensure statuses are as expected -----------------------------------------
+    
+    
+    setkeyv(l, c("lex.id", s1))
+    roll_lexis_status_inplace(
+      unsplit.data = dt, split.data = l, id.var = "lex.id"
+    )
+    
+    ## dt$lex.id from temporary values to original values ----------------------
+    # merge in correct IDs also to split data
+    on.exit()
+    set(dt, j = "lex.id", value = id_dt$lex.id)
+    
+    
+    tmpID <- makeTempVarName(names = names(l), pre = "TEMP_SPLITMULTI_ID_")
+    setnames(l, old = "lex.id", new = tmpID)
+    set(l, j = "lex.id", value = {id_dt[
+      i = .(l[[tmpID]]), 
+      j = .SD, 
+      on = "temp_id_values",
+      .SDcols = "orig_id_values"
+      ]})
+    set(l, j = tmpID, value = NULL)
+    rm("id_dt")
+    
+  }
+  
+  if (drop) l <- intelliDrop(l, breaks = breaks, dropNegDur = FALSE)
+  
+  if (nrow(l) == 0) {
+    warning("no data left after dropping; check breaks?")
+  }
+  
+  order <- c("lex.id", "lex.multi", allScales, "lex.dur", "lex.Cst", "lex.Xst")
+  order <- c(order, setdiff(names(l), order))
+  order <- intersect(order, names(l))
+  setcolorder(l, order)
+  
+  if (verbose) cat("time taken by splitting process: ", timetaken(stime), "\n")
+  
+  
+  breaks <- lapply(allScales, function(scale_nm) {
+    ## allowed to NULL also
+    br <- c(breaks[[scale_nm]], oldBreaks[[scale_nm]])
+    if (is.null(br)) return(br)
+    sort(unique(br))
+  }) 
+  names(breaks) <- allScales
+  
+  setattr(l, "time.scales", allScales)
+  setattr(l, "time.since", attr_list[["time.since"]])
+  setattr(l, "breaks", breaks)
+  setattr(l, "class", c("Lexis","data.table","data.frame"))
+  if (!return_DT()) setDFpe(l)
+  
+  l[]
+  
+}
+
+globalVariables(".")
+
diff --git a/R/splitting_utility_functions.R b/R/splitting_utility_functions.R
index ee0977d..447a510 100644
--- a/R/splitting_utility_functions.R
+++ b/R/splitting_utility_functions.R
@@ -1,1442 +1,1442 @@
-all_breaks_in <- function(bl1, bl2, x = NULL) {
-  ## INTENTION: return TRUE/FALSE depending on whether bl1 is a subset of bl2;
-  ## this means that each element in bl1 exists in bl2, and that those elements
-  ## are each subsets of the corresponding elements in bl2.
-  ## this is handy to check whether the some Lexis data has already
-  ## been split using the breaks in bl1.
-  ## NOTE: use checkBreakList() on each list separately before this.
-  
-  if (!is.list(bl1) || !is.list(bl2)) {
-    stop("Arguments bl1 and bl2 must be lists of breaks as supplied to e.g. ",
-         "splitMulti.")
-  }
-  
-  if (inherits(x, "Lexis")) {
-    checkLexisData(x)
-    checkBreaksList(x, bl1)
-    checkBreaksList(x, bl2)
-  }
-  
-  ce <- intersect(names(bl1), names(bl2))
-  if (length(ce) != length(bl1)) return(FALSE)
-  
-  test <- mapply(function(l1, l2) {
-    all(l1 %in% l2)
-  }, l1 = bl1, l2 = bl2[ce], SIMPLIFY = FALSE)
-  
-  all(unlist(test))
-}
-
-
-checkBreaksList <- function(x, breaks = list(fot = 0:5)) {
-  if (is.null(breaks)) stop("breaks is NULL")
-  if (!is.list(breaks)) stop("breaks needs to be a list")
-  if (!is.data.frame(x)) stop("x needs to be a data.frame")
-  timeScales <- names(breaks)
-  if (length(breaks) == 0L) stop("length of breaks list is zero")
-  if (length(timeScales) != length(breaks)) stop("breaks needs to be a fully named list")
-  
-  bad_scales <- setdiff(timeScales, names(x))
-  if (length(bad_scales) > 0) {
-    stop("at least one breaks list name wasn't a variable in data; bad names: ", 
-         paste0("'", bad_scales, "'", collapse = ", "))
-  }
-  lens <- lapply(breaks, function(el) if (is.null(el)) -1 else length(el))
-  badLens <- names(lens[unlist(lens) == 0L])
-  if (length(badLens)) {
-    badLens <- paste0("'", badLens, "'", collapse = ", ")
-    stop("Elements in breaks list for the following time scales were of ",
-         "length zero but not NULL: ", badLens, ". Breaks list may only ",
-         "contain elements of length > 0 or elements that are NULL.")
-  }
-  invisible(NULL)
-}
-
-checkPophaz <- function(lex, ph, haz.name = "haz") {
-  ## INTENTION: checks a Lexis data set against the pophaz data set for
-  ## consistency (e.g. existing variables to merge by)
-  
-  if (!is.data.frame(ph)) {
-    stop("Data set containing population/expected hazards must be a data.frame",
-         " (or a data.table, which is also a data.frame).")
-  }
-  
-  if (!haz.name %in% names(ph)) {
-    stop("Data set containing population/expected hazards does not contain a ",
-         "column named 'haz'. Make sure the name is exactly that (",
-         "case sensitive).")
-  }
-  
-  if (haz.name %in% names(lex)) {
-    stop("Lexis data set already contains a column named 'haz', which is a ",
-         "reserved name for the population hazard variable to be merged. ",
-         "Please rename/delete 'haz' from/in your Lexis data first.")
-  }
-  
-  if (!is.data.frame(ph)) {
-    stop("Data set of expected/population hazards must be a data.frame.")
-  }
-  
-  bn <- setdiff(names(ph), haz.name)
-  
-  if (length(bn) == 0L) {
-    stop("No variables in expected/population hazards data set to use in merge ",
-         "with Lexis data. Ensure that the pop. haz. data set containts some ",
-         "variables to merge by (e.g. sex, calendar year, and age group)")
-  }
-  if (!all(bn %in% names(lex))) {
-    badbn <- paste0("'", setdiff(bn, names(lex)), "'", collapse = ", ")
-    stop("Lexis data set did not have following variable(s) that were in ",
-         "the expected/population hazards data set: ", badbn,". ",
-         "Ensure you have supplied the right data and that the names of the ",
-         "intended variables match.")
-  }
-  
-  mergeVars <- setdiff(names(ph), haz.name)
-  dup <- any(duplicated(as.data.table(ph), by = mergeVars))
-  if (dup) {
-    stop("Supplied data set of population/expected hzards has duplicated rows ",
-         "by the variables ", paste0("'",mergeVars, "'", collapse = ", "),
-         " which prevents correct usage of the data set. Please ensure no rows",
-         " area duplicated in the data set before proceeding. Tip: use e.g. ",
-         "duplicated(PH, by = c('V1', 'V2')) to check for duplicatedness in ",
-         "your data set (here named PH) by the variables V1 and V2."
-    )
-  }
-  
-  invisible()
-}
-
-
-
-
-
-intelliCrop <- function(
-  x, 
-  breaks = list(fot = 0:5), 
-  allScales = NULL, 
-  cropStatuses = FALSE, 
-  tol = .Machine$double.eps^0.5
-) {
-  
-  ## appease R CMD CHECK
-  lex.dur <- lex.Xst <- lex.Cst <- NULL
-  
-  checkBreaksList(x = x, breaks = breaks)
-  breaks[unlist(lapply(breaks, length)) == 0L] <- NULL
-  if (!is.data.table(x)) stop("x needs to be a data.table")
-  
-  cropScales <- names(breaks)
-  
-  all_names_present(x, c("lex.dur", allScales))
-  
-  if (cropStatuses) {
-    origEnd <- x$lex.dur + x[[allScales[1L]]]
-  }
-  
-  
-  deltas <- mapply(function(b, y) pmax(min(b), y) - y, SIMPLIFY = FALSE,
-                   b = breaks, y = mget_cols(cropScales, x))
-  ## below: baseline (zero value without assigning zero of bad class)
-  deltas <- c(deltas, list(x[[cropScales[1]]][1L] - x[[cropScales[1]]][1L]))
-  deltas <- do.call(pmax, deltas)
-  
-  set(x, j = allScales, value = mget_cols(allScales, x) + deltas)
-  set(x, j = "lex.dur", value = x[["lex.dur"]] - deltas)
-  
-  durs <- mapply(function(b, y) max(b) - y, SIMPLIFY = FALSE,
-                 b = breaks, y = mget_cols(cropScales, x))
-  durs$lex.dur <- x$lex.dur
-  durs <- do.call(pmin, durs)
-  ## now have max durs by row, i.e. up to roof of breaks at most,
-  ## or to ((original lex.dur) - (deltas)) if that is smaller.
-  ## (being cropped or exiting before roof of breaks)
-  
-  set(x, j = "lex.dur", value = durs)
-  
-  if (cropStatuses) {
-    harmonizeStatuses(x, C = "lex.Cst", X = "lex.Xst")
-    wh_was_cropped <- which(x[["lex.dur"]] + x[[allScales[1L]]] + tol < origEnd)
-    set(x, i = wh_was_cropped, j = "lex.Xst", 
-        value = x[["lex.Cst"]][wh_was_cropped])
-  }
-  
-  invisible(x)
-}
-
-
-
-
-
-harmonizeStatuses <- function(x, C = "lex.Cst", X = "lex.Xst") {
-  
-  clC <- class(x[[C]])
-  clX <- class(x[[X]])
-  tyC <- typeof(x[[C]])
-  tyX <- typeof(x[[X]])
-  cl <- c(clC, clX)
-  
-  if (tyC != tyX && clC != clX) {
-    if (is.numeric(x[[C]]) && is.numeric(x[[X]])) {
-      harmonizeNumeric(x = x, v1="lex.Cst", v2="lex.Xst")
-      
-    } else if (is.factor(x[[C]]) || is.factor(x[[X]])) {
-      if (!is.factor(x[[C]])) set(x, j = C, value = as.factor(x[[C]]))
-      if (!is.factor(x[[X]])) set(x, j = X, value = as.factor(x[[X]]))
-      
-    }
-  }
-  
-  if (any(cl == "factor")) {
-    harmonizeFactors(x = x,  v1="lex.Cst", v2="lex.Xst")
-  }
-  
-}
-
-harmonizeNumericTimeScales <- function(x, times = NULL) {
-  ## INTENTION: given a Lexis data set with some time scales, ensure
-  ## that the classes of the time scales comply to the lowest denominator,
-  ## e.g. "double" and "integer" -> both "double"
-  
-  if (is.null(times)) {
-    times <- c(attr(x, "time.scales"), "lex.dur")
-  }
-  
-  msg <- paste0("Expected working data to have time scales %%VARS%%, but it ",
-                "didn't. This is an internal error: If you see this, complain ",
-                "to the package maintainer.")
-  all_names_present(x, times, msg = msg)
-  xt <- lapply(times, function(ch) x[[ch]])
-  names(xt) <- times
-  
-  harmoClasses <- c("numeric", "integer", "difftime")
-  cl <- lapply(xt, class)
-  wh <- unlist(lapply(cl, function(ch) {
-    any(ch %in% harmoClasses)
-  }))
-  ha <- times[wh]
-  hacl <- unique(unlist(cl[wh]))
-  
-  if (length(ha) > 1L) {
-    ## more than one class present and need to use common lowest denom
-    newMode <- as.double
-    
-    if (all(ha %in% c("integer", "difftime"))) {
-      ## all numeric times are integers or difftimes
-      newMode <- as.integer
-    }
-    for (var in ha) {
-      ## modify in place
-      set(x, j = var, value = newMode(x[[var]]))
-    }
-    
-    
-  }
-  invisible(NULL)
-}
-
-
-
-
-
-harmonizeNumeric <- function(x, v1="lex.Cst", v2="lex.Xst") {
-  ## assumes v1, v2 are numeric variable names in x  
-  
-  if (!is.numeric(x[[v1]]) || !is.numeric(x[[v2]])) {
-    print(class(x[[v1]]))
-    print(class(x[[v2]]))
-    stop("v1 and/or v2 is/are not of class numeric")
-  }
-  
-  if (!is.integer(x[[v1]])) set(x, j = v1, value = try2int(x[[v1]]))
-  if (!is.integer(x[[v2]])) set(x, j = v2, value = try2int(x[[v2]]))
-  
-  if (typeof(x[[v1]]) != typeof(x[[v2]])) {
-    
-    if (is.double(x[[v1]]))  set(x, j = v1, value = as.double(x[[v1]]))
-    if (is.double(x[[v2]]))  set(x, j = v2, value = as.double(x[[v2]]))
-    
-  }  
-  
-}
-
-
-
-
-
-harmonizeFactors <- function(x, v1="lex.Cst", v2="lex.Xst") {
-  ## assumes v1, v2 are factor names in x
-  
-  if (!is.factor(x[[v1]]) || !is.factor(x[[v2]])) {
-    stop("v1 and/or v2 is/are not of class factor")
-  }
-  
-  glab1 <- union(levels(x[[v1]]), levels(x[[v2]]))
-  glab2 <- union(levels(x[[v2]]), levels(x[[v1]]))
-  
-  
-  
-  setattr(x[[v1]], "levels", glab1)
-  setattr(x[[v2]], "levels", glab2)
-  
-}
-
-
-
-
-
-intelliDrop <- function(x, breaks = list(fot = 0:5), dropNegDur = TRUE, check = FALSE, tol = .Machine$double.eps^0.5, subset = NULL)  {
-  
-  if (!is.data.table(x)) {
-    stop("x needs to be a data.table; if you see this message, complain ",
-         "to the package maintainer")
-  }
-  checkBreaksList(x = x, breaks = breaks)
-  breaks[unlist(lapply(breaks, length)) < 2] <- NULL
-  timeScales <- names(breaks)
-  
-  if (check) {
-    checkLexisData(x)
-  }
-  
-  ra <- lapply(breaks, range)
-  ra <- lapply(ra, diff)
-  ts <- names(sort(unlist(ra))) ## shortest first
-  mi <- lapply(breaks, min)
-  ma <- lapply(breaks, max)
-  
-  substi <- substitute(subset)
-  subset <- evalLogicalSubset(x, substiset = substi)
-  
-  if (dropNegDur) subset[subset] <- subset[subset] & x$lex.dur[subset] > 0L
-  
-  ## figure out latest exit and first entry; don't need to test for dropping
-  ## if e.g. all left follow-up before the max in breaks
-  max_end <- lapply(ts, function(ch) min(x[[ch]] + x$lex.dur))
-  min_start <- lapply(ts, function(ch) max(x[[ch]]))
-  names(max_end) <- names(min_start) <- ts
-  
-  for (k in ts) {
-    mik <- mi[[k]]
-    mak <- ma[[k]]
-    
-    if (max_end[[k]] < mak + tol) {
-      tmpSD <- x[subset, .SD, .SDcols = c(k, "lex.dur")]
-      tmpSD <- setDT(lapply(tmpSD, as.numeric))
-      subset[subset] <- rowSums(tmpSD)  <=  mak + tol
-    }
-    if (min_start[[k]] + tol > mik) {
-      subset[subset] <- x[subset,][[k]] > mik - tol
-    }
-    
-    if (all(!subset)) {
-      stop("Dropped all remaining rows from data when subsetting by the  ",
-           "Lexis time scale '", k, "'. Range of values in data: ", 
-           paste0(round(range(x[[k]]),4), collapse = "-"), ". Min/Max breaks ",
-           "(used to subset data): ", mik, "/", mak, ".")
-    }
-    
-  }
-  
-  
-  x[subset, ]
-}
-
-
-matchBreakTypes <- function(lex, breaks, timeScale, modify.lex = FALSE) {
-  if (is.character(breaks)) {
-    breaks <- as.IDate(breaks)
-  }
-  clb <- class(breaks)
-  clb <- clb[length(clb)]
-  cts <- class(lex[[timeScale]])
-  cts <- cts[length(cts)]
-  
-  if (clb != cts) {
-    if (is.Date(breaks) && !is.Date(lex[[timeScale]])) {
-      breaks <- try2int(as.double(breaks))
-    } else if (is.integer(breaks) && is.double(lex[[timeScale]])) {
-      breaks <- as.double(breaks)
-    } else if (is.double(breaks) && is.integer(lex[[timeScale]])) {
-      breaks <- try2int(breaks)
-    }
-    
-  }
-  
-  if (modify.lex && clb != cts) {
-    if (!is.Date(breaks) && is.Date(lex[[timeScale]])) {
-      
-      if (clb == "double") {
-        set(lex, j = timeScale, value = as.double(lex[[timeScale]]))
-      } else {
-        set(lex, j = timeScale, value = as.integer(lex[[timeScale]]))
-      }
-      
-    } else if (is.double(breaks) && is.integer(lex[[timeScale]])) {
-      set(lex, j = timeScale, value = as.double(lex[[timeScale]]))
-    }
-    
-  }
-  breaks
-}
-
-protectFromDrop <- function(breaks, lower = FALSE) {
-  old_breaks <- copy(breaks)
-  if (length(breaks) == 0L) {
-    stop("Length of breaks to 'protect' from dropping is zero.")
-  }
-  if (is.Date(breaks)) {
-    breaks <- c(breaks, max(breaks) + 1e4L)
-    if (lower) breaks <- c(min(breaks) - 1e4L, breaks)
-    
-  } else if (is.integer(breaks))  {
-    breaks <- c(breaks, 1e6L)
-    if (lower) breaks <- c(-1e6L, breaks)
-    
-  } else if (is.double(breaks)) {
-    breaks <- c(breaks, Inf)
-    if (lower) breaks <- c(-Inf, breaks)
-    
-  } else {
-    stop("breaks were not Date, integer or double")
-  }
-  setattr(breaks, "unprotected", old_breaks)
-  breaks
-}
-
-unprotectFromDrop <- function(breaks) {
-  up <- attr(breaks, "unprotected")
-  if (is.null(up) || length(up) == 0L) {
-    stop("Could not 'unprotect' breaks from dropping as the required ",
-         "attribute was not found. If you see this it is most likely ",
-         "an internal error and you should complain to the pkg maintainer.")
-  }
-  up
-}
-
-
-
-
-setLexisDT <- function(data, entry, exit, entry.status, exit.status, id = NULL, select = NULL) {
-  
-  ## appease R CMD CHECK
-  lex.Cst <- lex.Xst <- NULL
-  
-  if (!is.data.table(data)) stop("not a data.table")
-  if (inherits(data, "Lexis")) stop("already a Lexis object")
-  
-  if (!is.null(select) && !is.character(select)) stop("select was not a character vector of names")
-  
-  entry <- substitute(entry)
-  exit <- substitute(exit)
-  entry <- eval(entry, envir = data, enclos = parent.frame())
-  exit <- eval(exit, envir = data, enclos = parent.frame())
-  enNames <- names(entry)
-  exNames <- names(exit)
-  
-  timeScales <- union(enNames, exNames)
-  if (any(timeScales %in% names(data))) stop("at least one named time scales already present in data; original names mandatory")
-  enNeeded <- setdiff(timeScales, enNames)
-  enPresent <- setdiff(timeScales, enNeeded)
-  durVar <- intersect(enNames, exNames)
-  if (length(durVar) > 1) stop("you have more than 1 time scales in both entry and exit; only one mandatory")
-  
-  enVars <- paste0(enNames, "_en")
-  exVars <- paste0(exNames, "_ex")
-  setattr(entry, "names", enVars)
-  setattr(exit, "names", exVars)
-  
-  l <- as.data.table(c(entry, exit))
-  rm(entry, exit)
-  
-  ## duration
-  exV <- paste0(durVar, "_ex")
-  enV <- paste0(durVar, "_en")
-  set(l, j = "lex.dur", value = l[[exV]] - l[[enV]])
-  rm(exV, enV)
-  
-  ## time scale starting points
-  if (length(enNeeded) > 0) {
-    for (ts in enNeeded) {
-      exV <- paste0(ts, "_ex")
-      set(l, j = ts, value = l[[exV]] - l$lex.dur)
-    }
-  }
-  setnames(l, paste0(enPresent, "_en"), enPresent)
-  
-  # no longer need time scale end points
-  for (k in exVars) {
-    set(l, j = k, value = NULL)
-  }
-  
-  ## status definition
-  data[, lex.Cst := entry.status]
-  data[, lex.Xst := exit.status]
-  
-  harmonizeStatuses(data, C = "lex.Cst", X = "lex.Xst")
-  
-  ## all time scales etc. into data
-  data[, names(l) := l]
-  
-  
-  id <- substitute(id)
-  id <- eval(id, envir = data, enclos = parent.frame())
-  if (!is.null(id)) set(data, j = "lex.id", value = id)
-  rm(id)
-  
-  if (!is.null(select)) {
-    
-    delVars <- setdiff(names(data), c(names(l), select))
-    if (length(delVars) > 0) {
-      l[, (delVars) := NULL]
-    }
-  }
-  
-  rm(l)
-  lexVars <- c("lex.id", timeScales, "lex.dur", "lex.Cst", "lex.Xst")
-  setcolorder(data, c(lexVars, setdiff(names(data), lexVars)))
-  
-  setattr(data, "time.scales", timeScales)
-  setattr(data, "time.since", rep("", times = length(timeScales)))
-  setattr(data, "class", c("Lexis", "data.table", "data.frame"))
-  
-  
-}
-
-checkLexisData <- function(lex, check.breaks = FALSE) {
-  ## INTENTION: checks Lexis attributes
-  ## OUTPUT: nothing
-  
-  if (missing(lex)) {
-    stop("Data object not supplied")
-  }
-  if (!inherits(lex, "Lexis")) {
-    stop("Data not a Lexis object; it has class(es) ", deparse(class(lex)))
-  }
-  if (!is.data.frame(lex)) {
-    stop("Data is not a data.frame / data.table; it has class(es) ",
-         deparse(class(lex)))
-  }
-  if (nrow(lex) == 0) {
-    stop("Data has zero rows")
-  }
-  allScales <- attr(lex, "time.scales")
-  if (length(allScales) == 0) {
-    stop("no time scales appear to be defined; is data a Lexis object?")
-  }
-  
-  badScales <- setdiff(allScales, names(lex))
-  if (length(badScales) > 0) {
-    badScales <- paste0("'", badScales, "'", collapse = ", ")
-    stop("Following time scales found in data's attributes but not present ",
-         "in data: ", badScales)
-  }
-  
-  lexVars <- c("lex.dur", "lex.id", "lex.Cst", "lex.Xst")
-  blv <- setdiff(lexVars, names(lex))
-  if (length(blv) > 0) {
-    blv <- paste0("'", blv, "'", collapse = ", ")
-    stop("Following Lexis variables not found in data: ", blv)
-  }
-  
-  if (check.breaks) {
-    BL <- attr(lex, "breaks")
-    if (is.null(BL)) stop("No breaks list in data attributes")
-    checkBreaksList(lex, breaks = BL)
-  }
-  
-  invisible()
-}
-
-
-splitMultiPreCheck <- function(data = NULL, breaks = NULL, ...) {
-  
-  ## INTENTION: checks for discrepancies between data and breaks, etc.
-  ## OUTPUT: cleaned-up list of breaks
-  checkLexisData(data)
-  allScales <- attr(data, "time.scales")
-  
-  if (!is.null(breaks) && !is.list(breaks)) {
-    stop("breaks must be a list; see examples in ?splitMulti")
-  }
-  if (is.null(breaks)) {
-    breaks <- list(...)
-    breaks <- breaks[intersect(names(breaks), allScales)]
-  }
-  
-  if (length(breaks) == 0) stop("no breaks defined!")
-  
-  splitScales <- names(breaks)
-  ## NULL breaks imply not used
-  for (k in splitScales) {
-    if (length(breaks[[k]]) == 0) {
-      breaks[k] <- NULL
-    }
-  } 
-  
-  checkBreaksList(x = data, breaks = breaks)
-  
-  splitScales <- names(breaks)
-  
-  if (!all(splitScales %in% allScales)) {
-    stop("breaks must be a list with at least one named vector corresponding to used time scales \n
-         e.g. breaks = list(fot = 0:5)")
-  }
-  
-  if (!all(splitScales %in% names(data))) {
-    stop("At least one vector name in breaks list is not a variable name in the data")
-  }
-  breaks
-}
-
-forceLexisDT <- function(x, breaks = NULL, allScales = NULL, key = TRUE) {
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  setattr(x, "breaks", breaks)
-  setattr(x, "time.scales", allScales)
-  # alloc.col(x)
-  if (key) setkeyv(x, c("lex.id", names(breaks)[1L]))
-  invisible(x)
-}
-
-
-doCutLexisDT <- function(lex, cut = dg_date, timeScale = "per", by = "lex.id", n = 1L) {
-  
-  checkLexisData(lex, check.breaks = FALSE)
-  
-  x <- unique(lex, by = by)
-  cut <- evalq(cut, envir = lex, enclos = parent.frame(n = n + 1L))
-  delta <- cut - x[[timeScale]]
-  
-  allScales <- attr(lex, "time.scales")
-  
-  setDT(x)
-  for (v in allScales) {
-    set(x, j = v, value = x[[v]] + delta)
-  }
-  
-  set(x, j = "lex.dur", value = 0)
-  
-  tmp <- list()
-  tmp$isCut <- makeTempVarName(lex, pre = "isCut_")
-  
-  set(x, j = tmp$isCut, value = 1L)
-  on.exit(setcolsnull(x, unlist(tmp)))
-  
-  x <- rbindlist(list(lex, x), use.names = TRUE, fill = TRUE)
-  x[1:nrow(lex), (tmp$isCut) := 0L]
-  
-  ## NOTE: new cut row being the first or last row
-  ## implies it resides outside old observations
-  ## OR it is equal to lowest/highest value
-  setkeyv(x, c(by, allScales))
-  setkeyv(x, by)
-  x <- x[!((duplicated(x, by = key(x)) | duplicated(x, by = key(x), fromLast = TRUE)) & x[[tmp$isCut]] == 0L)]
-  stop("not ready")
-}
-
-# data <- data.table(birth = 2000:2000, entry=2002:2003, 
-#                    exit=2011:2012, event=c(2010,2011), 
-#                    status=1:0)
-# 
-# lex <- lexpand(data = data, birth = birth, entry = entry, 
-#                exit = exit, event = event, 
-#                id = 1L, entry.status = 99L,
-#                status = status, overlapping = TRUE)
-
-lexpile <- function(lex, by = "lex.id", subset = NULL) {
-  ## PURPOSE: given several rows per id in a Lexis object,
-  ## collate data into form where
-  ## - no subject has any overlapping time lines
-  ## - lex.Cst and lex.Xst are logical, i.e. 0 -> 1, 1 -> 1, 1 -> 2
-  ## this should be made to work with both split and unsplit Lexis data.
-  
-  data <- NULL # R CMD CHECK appeasement
-  
-  checkLexisData(lex, check.breaks = FALSE)
-  
-  allScales <- attr(lex, "time.scales")
-  sc <- allScales[1L]
-  
-  all_names_present(lex, by)
-  
-  if (is.character(lex$lex.Cst) || is.character(lex$lex.Xst)) {
-    stop("This function requires lex.Cst and lex.Xst to be integer, double (i.e. numeric) or factor variables to determine the order of possible statuses!")
-  }
-  
-  ## need to take copy eventually ----------------------------------------------
-  attrs <- attributes(lex)
-  subset <- evalLogicalSubset(data, substitute(subset))
-  x <- lex[subset,]
-  forceLexisDT(x, breaks = attrs$breaks, allScales = attrs$time.scales)
-  alloc.col(x)
-  
-  
-  ## ensure status harmony -----------------------------------------------------
-  harmonizeStatuses(x = x, X = "lex.Xst", C = "lex.Cst")
-  exStat <- if (is.factor(x$lex.Xst)) levels(x$lex.Xst) else sort(unique(x$lex.Xst))
-  enStat <- if (is.factor(x$lex.Cst)) levels(x$lex.Cst) else sort(unique(x$lex.Cst))
-  allStat <- c(setdiff(enStat, exStat), exStat) ## enStat & exStat equal if factors used
-  
-  ## avoiding side effects -----------------------------------------------------
-  oldKey <- key(lex)
-  tmp <- list()
-  tmp$order<- makeTempVarName(x, pre = "order_")
-  
-  on.exit({
-    if (length(oldKey) > 0) setkeyv(x, oldKey) else 
-      setorderv(x, tmp$order)
-  }, add = TRUE)
-  
-  on.exit({
-    setcolsnull(x, unlist(tmp$order), soft = TRUE)
-  }, add = TRUE)
-  
-  x[, c(tmp$order) := 1:.N]
-  
-  ## check for need for lexpiling ----------------------------------------------
-  setkeyv(x, by)
-  if (sum(duplicated(x, by = key(x))) == 0L) return(lex)
-  
-  
-  ## figure out what statuses are used -----------------------------------------
-  
-  tmp$ev <- makeTempVarName(x, pre = "event_")
-  x[, c(tmp$ev) := detectEvents(x, breaks = attrs$breaks, by = by)]
-  
-  tmp$scEnds <- paste0(allScales, "_end")
-  tmp$scEnds <- makeTempVarName(lex, pre = tmp$scEnds)
-  x[, c(tmp$scEnds) := lapply(.SD, function(x) x + lex$lex.dur), .SDcols = allScales]
-  
-  ## NOTE: rows for a given subject ending in simultaneously with at least
-  ## one being a transition will not be allowed.
-  setkeyv(x, c(by, tmp$scEnds[1L]))
-  
-  whDup <- duplicated(x, fromLast = FALSE, by = key(x)) | duplicated(x, fromLast = TRUE, by = key(x))
-  dupTest <- x[whDup, 1L %in% unique(.SD), .SDcols = tmp$ev]
-  rm(whDup)
-  
-  if (dupTest) stop("At least one subject had at least two simultaneous events.")
-  ## NOTE: if interval ends AND status are the very same for M rows,
-  ## then the M rows are necessarily nested with one or more covering
-  ## the whole time line. Only need to keep the one.
-  setkeyv(x, c(tmp$scEnds, "lex.Cst", "lex.Xst"))
-  setorderv(x, c(allScales,tmp$scEnds, "lex.Cst", "lex.Xst"))
-  
-  x <- unique(x, by = key(x))
-  stop("unfinished")
-  
-}
-
-contractLexis <- function(x, breaks, drop = TRUE) {
-  stop("This doesnt do anything yet")
-  ## INTENTION: given a Lexis object and breaks,
-  ## ensures data is split by the breaks and contracts the split rows
-  ## so that the data is split at the level of the supplied breaks.
-  ## e.g. with x split by fot = seq(0, 5, 1/12) and with supplying
-  ## breaks = list(fot = 0:5), rows within 0-1 are collated into one row etc.
-  
-  ## PROBLEM: a subject may have e.g. rows spaning 0.5 - 1 which are requested
-  ## to be contracted to one row spanning 0-1.
-  
-  
-}
-
-
-
-
-#' @title Prepare Exposure Data for Aggregation
-#' @description \code{prepExpo} uses a \code{Lexis} object of periods of exposure
-#' to fill gaps between the periods and overall entry and exit times without
-#' accumulating exposure time in periods of no exposure, and splits the
-#' result if requested.
-#' @param lex a \code{\link[Epi]{Lexis}} object with ONLY periods of exposure
-#' as rows; one or multiple rows per subject allowed
-#' @param freezeScales a character vector naming \code{Lexis} time scales of exposure
-#' which should be frozen in periods where no exposure occurs (in the gap
-#' time periods) 
-#' @param cutScale the \code{Lexis} time scale along which the subject-specific
-#' ultimate entry and exit times are specified
-#' @param entry an expression; the time of entry to follow-up which may be earlier, at, or after
-#' the first time of exposure in \code{freezeScales}; evaluated separately
-#' for each unique combination of \code{by}, so e.g. with 
-#' \code{entry = min(Var1)} and \code{by = "lex.id"} it 
-#' sets the \code{lex.id}-specific minima of \code{Var1} to be the original times
-#' of entry for each \code{lex.id}
-#' @param exit the same as \code{entry} but for the ultimate exit time per unique
-#' combination of \code{by}
-#' @param by a character vector indicating variable names in \code{lex},
-#' the unique combinations of which identify separate subjects for which
-#' to fill gaps in the records from \code{entry} to \code{exit};
-#' for novices of \code{{\link{data.table}}}, this is passed to a 
-#' \code{data.table}'s \code{by} argument.
-#' @param breaks a named list of breaks; 
-#' e.g. \code{list(work = 0:20,per = 1995:2015)}; passed on to 
-#' \code{\link{splitMulti}} so see that function's help for more details
-#' @param freezeDummy a character string; specifies the name for a dummy variable
-#' that this function will create and add to output which 
-#' identifies rows where the \code{freezeScales} are frozen and where not
-#' (\code{0} implies not frozen, \code{1} implies frozen);
-#' if \code{NULL}, no dummy is created
-#' @param subset a logical condition to subset data by before computations;
-#' e.g. \code{subset = sex == "male"}
-#' @param verbose logical; if \code{TRUE}, the function is chatty and returns
-#' some messages and timings during its run.
-#' @param ... additional arguments passed on to \code{\link{splitMulti}}
-#' @details 
-#' 
-#' \code{prepExpo} is a convenience function for the purpose of eventually aggregating 
-#' person-time and events in categories of not only normally progressing 
-#' \code{Lexis} time scales but also some time scales which should not
-#' progress sometimes. For example a person may work at a production facility
-#' only intermittently, meaning exposure time (to work-related substances 
-#' for example) should not progress outside of periods of work. This allows for
-#' e.g. a correct aggregation of person-time and events by categories of cumulative
-#' time of exposure.
-#' 
-#' Given a \code{Lexis} object containing rows (time lines)
-#' where a subject is exposed to something (and NO periods without exposure),
-#' fills any gaps between exposure periods for each unique combination of \code{by}
-#' and the subject-specific "ultimate" \code{entry} and \code{exit} times,
-#' "freezes" the cumulative exposure times in periods of no exposure,
-#' and splits data using \code{breaks} passed to \code{\link{splitMulti}}
-#' if requested. Results in a (split) \code{Lexis} object where \code{freezeScales}
-#' do not progress in time periods where no exposure was recorded in \code{lex}.
-#' 
-#' This function assumes that \code{entry} and \code{exit} arguments are the
-#' same for each row within a unique combination of variables named in \code{by}.
-#' E.g. with \code{by = "lex.id"} only each \code{lex.id} has a unique value
-#' for \code{entry} and \code{exit} at most.
-#' 
-#' The supplied \code{breaks} split the data using \code{splitMulti}, with
-#' the exception that breaks supplied concerning any frozen time scales
-#' ONLY split the rows where the time scales are not frozen. E.g.
-#' with \code{freezeScales = "work"}, 
-#' \code{breaks = list(work = 0:10, cal = 1995:2010)} splits all rows over
-#' \code{"cal"} but only non-frozen rows over \code{"work"}.
-#' 
-#' Only supports frozen time scales that advance and freeze contemporaneously:
-#' e.g. it would not currently be possible to take into account the cumulative
-#' time working at a facility and the cumulative time doing a single task
-#' at the facility, if the two are not exactly the same. On the other hand
-#' one might use the same time scale for different exposure types, supply them
-#' as separate rows, and identify the different exposures using a dummy variable.
-#' @return 
-#' 
-#' Returns a \code{Lexis} object that has been split if \code{breaks} is specified.
-#' The resulting time is also a \code{data.table} if 
-#' \code{options("popEpi.datatable") == TRUE} (see: \code{?popEpi})
-#' 
-#' @import data.table
-#' @export
-prepExpo <- function(lex, freezeScales = "work", cutScale = "per", entry = min(get(cutScale)),
-                     exit = max(get(cutScale)), by = "lex.id", breaks = NULL, freezeDummy = NULL, subset = NULL,
-                     verbose = FALSE, ...) {
-  ## R CMD CHECK appeasement
-  lex.dur <- NULL
-  
-  if (verbose) allTime <- proc.time()
-  
-  ## check breaks & data -------------------------------------------------------
-  breaks <- evalq(breaks)
-  dumBreaks <- structure(list(c(-Inf, Inf)), names = cutScale, internal_prepExpo_dummy = TRUE)
-  if (is.null(breaks)) breaks <- dumBreaks
-  breaks <- splitMultiPreCheck(data = lex, breaks = breaks)
-  if (!is.null(attr(breaks, "internal_prepExpo_dummy"))) breaks <- NULL
-  checkLexisData(lex)
-  oldBreaks <- attr(lex, "breaks")
-  if (!is.null(breaks)) checkBreaksList(lex, breaks)
-  checkBreaksList(lex, oldBreaks)
-  
-  
-  ## data ----------------------------------------------------------------------
-  
-  subset <- evalLogicalSubset(data = lex, substitute(subset))
-  x <- if (!all(subset)) evalq(lex)[subset, ] else copy(evalq(lex))
-  
-  setDT(x)
-  
-  allScales <- attr(lex, "time.scales")
-  linkScales <- setdiff(allScales, freezeScales)
-  othScales <- setdiff(linkScales, cutScale)
-  
-  setkeyv(x, c(by, cutScale))
-  
-  l <- list() ## will hold temp var names; this avoids collisions with names of vars in x
-  l$cutScale <- cutScale
-  l$freezeScales <- freezeScales
-  l$liquidScales <- setdiff(allScales, freezeScales)
-  l$by <- by
-  rm(cutScale, freezeScales, by)
-  
-  ## args ----------------------------------------------------------------------
-  if (verbose) argTime <- proc.time()
-  tol <- .Machine$double.eps^0.75
-  
-  if (is.character(freezeDummy) && freezeDummy %in% names(lex)) stop("Variable named in freezeDummy already exists in data; freezeDummy is inteded for creating a new dummy for identifying the rows where freezeScales are frozen. Please supply an original variable name to freezeDummy")
-  
-  if (!is.character(l$by)) stop("by must be given as a vector of character strings naming columns in lex")
-  all_names_present(lex, l$by)
-  
-  enSub <- substitute(entry)
-  exSub <- substitute(exit)
-  
-  PF <- parent.frame(1L)
-  l$en <- makeTempVarName(x, pre = "entry_")
-  l$ex <- makeTempVarName(x, pre = "exit_")
-  x[, c(l$ex, l$en) := list(eval(exSub, envir = .SD, enclos = PF), 
-                            eval(enSub, envir = .SD, enclos = PF)), by = c(l$by)]
-  
-  ## tests disabled for now...
-  #   testTime <- proc.time()
-  #   
-  #   test <- x[, .N, by = list(r = get(l$cutScale) + lex.dur > get(l$ex) - tol)]
-  #   if(test[r == TRUE, .N] > 0) stop("exit must currently be higher than or equal to the maximum of cutScale (on subject basis defined using by); you may use breaks instead to limit the data")
-  #   
-  #   test <- x[, .N, by = list(r = get(l$cutScale) + tol < get(l$en))]
-  #   if(test[r == TRUE, .N] > 0) stop("entry must currently be lower than or equal to the minimum of cutScale (on subject basis defined using by); you may use breaks instead to limit the data")
-  #   if (verbose) cat("Finished checking entry and exit. Time taken: ", timetaken(argTime), "\n")
-  if (verbose) cat("Finished evaluating entry and exit and checking args. Time taken: ", timetaken(argTime), "\n")
-  
-  ## create rows to fill gaps --------------------------------------------------
-  if (verbose) fillTime <- proc.time()
-  x2 <- copy(x)
-  x2[, (l$freezeScales) := NA]
-  x2 <- rbind(x2, unique(x2, by = c(l$by), fromLast = TRUE))
-  
-  l$delta <- makeTempVarName(x2, pre = "delta_")
-  x2[, (l$delta) := c(get(l$en)[1], get(l$cutScale)[-c(1,.N)], max(get(l$cutScale)+lex.dur)) - get(l$cutScale), by = c(l$by)]
-  x2[, c(linkScales) := lapply(mget(linkScales), function(x) x + get(l$delta)), by = c(l$by)]
-  
-  setcolsnull(x2, l$delta)
-  
-  l$order <- makeTempVarName(x, pre = "order_")
-  x[, (l$order) := (1:.N)*2, by = c(l$by)]
-  x2[, (l$order) := (1:.N)*2-1, by = c(l$by)]
-  
-  x <- rbindlist(list(x, x2))
-  rm(x2)
-  setkeyv(x, c(l$by, l$cutScale, l$order))
-  setkeyv(x, c(l$by))
-  set(x, j = l$order, value = as.integer(x[[l$order]]))
-  x[, (l$order) := 1:.N, by = c(l$by)]
-  
-  if (verbose) cat("Finished expanding data to accommodate filling gaps. Time taken: ", timetaken(fillTime), "\n")
-  ## handle time scale values --------------------------------------------------
-  if (verbose) valueTime <- proc.time()
-  
-  l$CSE <- makeTempVarName(x, pre = paste0(l$cutScale, "_end_"))
-  l$LCS <- makeTempVarName(x, pre = paste0("lead1_",l$cutScale, "_"))
-  x[, (l$CSE) := lex.dur + get(l$cutScale)]
-  x[, (l$LCS)  := shift(get(l$cutScale), n = 1L, type = c("lead"), fill = NA), by = c(l$by)]
-  
-  
-  x[!duplicated(x, fromLast = TRUE, by = key(x)), c(l$LCS, l$CSE) := get(l$ex)]
-  x[, (l$CSE) := pmin(get(l$LCS), get(l$CSE))]
-  x[, (l$cutScale) := sort(c(get(l$en)[1L],shift(get(l$CSE), n = 1L, type = "lag", fill = NA)[-1])), by = c(l$by)]
-  x[, lex.dur := get(l$CSE) - get(l$cutScale)]
-  
-  ## bring up other than frozen and cut scales to bear -------------------------
-  x[, (othScales) := lapply(mget(othScales), function(x) {min(x) + c(0, cumsum(lex.dur)[-.N])}), by = c(l$by)]
-  
-  
-  ## frozen scales should make sense cumulatively ------------------------------
-  ## indicates frozenness: 0 = not frozen, 1 = frozen
-  l$frz <- makeTempVarName(x, pre = "frozen_")
-  x[, (l$frz) := 0L]
-  frozens <- x[,is.na(get(l$freezeScales[1]))]
-  x[frozens, (l$frz) := 1L]
-  
-  
-  ## alternate method: just use lex.durs and only cumulate in non-frozen rows
-  x[, (l$freezeScales) := lapply(mget(l$freezeScales), function(x) {
-    x <- max(0, min(x-lex.dur, na.rm=TRUE))
-    x <- x + c(0, as.double(cumsum(as.integer(!get(l$frz))*lex.dur))[-.N])
-  }), by = c(l$by)]
-  
-  x <- x[lex.dur > .Machine$double.eps^0.5, ]
-  
-  if (verbose) cat("Finished computing correct values for time scales. Time taken: ", timetaken(valueTime), "\n")
-  
-  ## splitting separately ------------------------------------------------------
-  if (!is.null(breaks)) {
-    if (verbose) splitTime <- proc.time()
-    x_frozen <- x[get(l$frz) == 1L,]
-    x <- x[get(l$frz) == 0L]
-    forceLexisDT(x, allScales = allScales, breaks = oldBreaks)
-    
-    ## NOTE: since we only split by the frozen time scales by pretending
-    ## they are NOT Lexis time scales (temporarily), and since one should 
-    ## pass the appropriate breaks info of pre-existing breaks to splitMulti,
-    ## choose only breaks for non-frozen time scales to include in x_frozen's
-    ## attributes here.
-    ## (e.g. when work history is no longer accumulating)
-    frzBreaks <- breaks[l$liquidScales]
-    oldFrzBreaks <- oldBreaks[l$liquidScales]
-    emptyFrzBreaks <- vector("list", length = length(l$liquidScales))
-    names(emptyFrzBreaks) <- l$liquidScales
-    if (length(oldFrzBreaks)) {
-      emptyFrzBreaks[names(oldFrzBreaks)] <- oldFrzBreaks
-    }
-    forceLexisDT(x_frozen, allScales = l$liquidScales, breaks = emptyFrzBreaks)
-    
-    if (length(frzBreaks) > 0) {
-      ## do (also) split for all time scales where also the frozen
-      ## time scales are split. This is allowed for times where the
-      ## frozen time scales have not been frozen
-      ## (e.g. work history is accumulating)
-      x_frozen <- splitMulti(x_frozen, breaks = frzBreaks, ...)
-    }
-    
-    ## do (also) split where also split
-    x <- splitMulti(x, breaks = breaks, ...)
-    breaks <- attr(x, "breaks") ## new breaks appended by splitMulti
-    
-    setDT(x)
-    setDT(x_frozen)
-    x <- rbindlist(list(x, x_frozen), use.names = TRUE); rm(x_frozen)
-    forceLexisDT(x, breaks = breaks, allScales = allScales)
-    if (verbose) cat("Finished splitting data. Time taken: ", timetaken(splitTime), "\n")
-  }
-  
-  ## final touch ---------------------------------------------------------------
-  
-  setDT(x)
-  if (is.character(freezeDummy)) setnames(x, l$frz, freezeDummy)
-  setkeyv(x, c(l$by, l$order))
-  delCols <- setdiff(names(l), c("by", "cutScale", "freezeScales", 
-                                 "liquidScales",
-                                 "linkScales", "allScales", "othScales"))
-  delCols <- unlist(l[delCols])
-  setcolsnull(x, keep = names(lex), colorder = TRUE)
-  
-  setattr(x, "time.scales", allScales)
-  setattr(x, "breaks", breaks)
-  setattr(x, "time.since", rep("", length(allScales)))
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  if (!return_DT()) setDFpe(x)
-  
-  if (verbose) cat("Finished prepExpo run. Time taken: ", timetaken(allTime), "\n")
-  
-  x[]
-}
-
-
-
-doComparisonWithEpi <- function(lexDT, lexDTdrop, lexDF, breaks) {
-  BL <- NULL
-  if (!is.list(breaks)) stop("breaks needs to be a list")
-  requireNamespace("Epi")
-  requireNamespace("testthat")
-  
-  allScales <- attr(lexDF, "time.scales")
-  sc1 <- allScales[1]
-  setDT(lexDT)
-  setDT(lexDTdrop)
-  setDT(lexDF)
-  setkeyv(lexDT, c("lex.id", sc1))
-  setkeyv(lexDTdrop, c("lex.id", sc1))
-  setkeyv(lexDF, c("lex.id", sc1))
-  
-  testthat::expect_equal(attr(lexDT, "time.scales"), attr(lexDF, "time.scales"))
-  testthat::expect_equal(attr(lexDT, "time.since"), attr(lexDF, "time.since"))
-  
-  testthat::expect_equal(attr(lexDTdrop, "time.scales"), attr(lexDF, "time.scales"))
-  testthat::expect_equal(attr(lexDTdrop, "time.since"), attr(lexDF, "time.since"))
-  
-  doTestBarrage(dt1 = lexDT, dt2 = lexDF, allScales = allScales)
-  rm(lexDT)
-  
-  lexDF <- intelliDrop(x = lexDF, breaks = breaks)
-  
-  doTestBarrage(dt1 = lexDTdrop, dt2 = lexDF, allScales = allScales)
-  
-}
-
-doTestBarrage <- function(dt1, dt2, allScales, testTimes = TRUE, testStatuses = TRUE) {
-  requireNamespace("Epi")
-  requireNamespace("testthat")
-  
-  lex.id <- lex.dur <- NULL ## APPEASE R CMD CHECK
-  
-  testthat::expect_equal(sum(dt1$lex.dur), 
-                         sum(dt2$lex.dur), 
-                         check.attributes = FALSE)
-  testthat::expect_equal(dt1[, sum(lex.dur), keyby = lex.id]$V1, 
-                         dt2[, sum(lex.dur), keyby = lex.id]$V1, 
-                         check.attributes = FALSE)
-  
-  all_names_present(dt1, allScales)
-  all_names_present(dt2, allScales)
-  
-  if (testTimes) {
-    for (k in allScales) {
-      testthat::expect_equal(dt1[[k]], dt2[[k]], 
-                             check.attributes = TRUE)
-    }
-  }
-  
-  if (testStatuses) {
-    testthat::expect_equal(dt1$lex.Cst, dt2$lex.Cst, check.attributes = FALSE)
-    testthat::expect_equal(dt1$lex.Xst, dt2$lex.Xst, check.attributes = FALSE)
-    
-    testthat::expect_equal(levels(dt1$lex.Cst), levels(dt2$lex.Cst), check.attributes = FALSE)
-    testthat::expect_equal(levels(dt1$lex.Xst), levels(dt2$lex.Xst), check.attributes = FALSE)
-    
-    testthat::expect_true(all(class(dt2$lex.Cst) %in% class(dt1$lex.Cst)))
-    testthat::expect_true(all(class(dt2$lex.Xst) %in% class(dt1$lex.Xst)))
-  }
-  
-  invisible(NULL)
-}
-
-compareSLDTWithEpi <- function(data, breaks, timeScale) {
-  requireNamespace("Epi")
-  requireNamespace("testthat")
-  
-  if (!inherits(data, "Lexis")) stop("data gotta be a Lexis object broseph")
-  
-  lexDT <- splitLexisDT(data, breaks = breaks, timeScale = timeScale, merge = TRUE, drop = FALSE)
-  lexDTdrop <- splitLexisDT(data, breaks = breaks, timeScale = timeScale, merge = TRUE, drop = TRUE)
-  lexDF <- splitLexis(data, breaks = breaks, time.scale = timeScale) ## without dropping
-  ## this treatment done in splitLexisDT (difftime -> integer -> double)
-  harmonizeNumericTimeScales(lexDF, times = c(Epi::timeScales(lexDF), "lex.dur"))
-  
-  BL <- list(breaks)
-  setattr(BL, "names", timeScale)
-  
-  doComparisonWithEpi(lexDT = lexDT, lexDTdrop = lexDTdrop, lexDF = lexDF, breaks = BL)
-  
-  invisible(NULL)
-}
-
-
-
-
-
-splitMultiEpi <- function(data, breaks = list(fot = 0:5), drop) {
-  
-  for (k in names(breaks)) {
-    data <- splitLexis(data, breaks = breaks[[k]], time.scale = k)
-  }
-  
-  forceLexisDT(
-    data, breaks = attr(data, "breaks"), 
-    allScales = attr(data, "time.scales"), 
-    key = FALSE
-  )
-  if (drop) data <- intelliDrop(data, breaks = breaks)
-  data
-}
-
-
-
-
-
-compareSMWithEpi <- function(data, breaks = list(fot=0:5)) {
-  requireNamespace("Epi")
-  requireNamespace("testthat")
-  
-  lexDT <- splitMulti(data, breaks = breaks, merge = TRUE, drop = FALSE)
-  lexDTdrop <- splitMulti(data, breaks = breaks, merge = TRUE, drop = TRUE)
-  lexDF <- splitMultiEpi(data, breaks = breaks, drop = FALSE)
-  
-  doComparisonWithEpi(lexDT=lexDT, lexDTdrop = lexDTdrop, lexDF=lexDF, breaks = breaks)
-  
-  invisible(NULL)
-}
-
-
-
-
-
-summarize_Lexis <- function(x) {
-  
-  lex.Cst <- lex.Xst <- NULL ## appease R CMD CHECK
-  
-  dur <- sum(x$lex.dur)
-  status_vars <- paste0("lex.", c("Cst", "Xst"))
-  time_scales <- copy(attr(x, "time.scales"))
-  dt <- copy(setDT(
-    mget(c(status_vars, "lex.id", time_scales[1]), as.environment(x))
-  ))
-  setkeyv(dt, c("lex.id", time_scales[1]))
-  dt <- unique(dt, by = c("lex.id"), fromLast = TRUE)
-  
-  n <- dt[, .N, keyby = status_vars]
-  rm("dt")
-  n[, "transition" := paste0(lex.Cst, "->", lex.Xst)]
-  n <- cast_simple(data = n, columns = "transition", values = "N")
-  return(cbind(lex.dur = dur, n))
-}
-
-
-
-
-
-roll_lexis_status_inplace <- function(unsplit.data, split.data, id.var) {
-  
-  ## R CMD CHECK appeasement
-  lex.Cst <- lex.Xst <- NULL
-  
-  stopifnot(
-    is.data.table(split.data),
-    length(key(split.data)) > 1,
-    key(split.data)[1] == id.var,
-    key(split.data)[2] %in% attr(unsplit.data, "time.scales"),
-    id.var %in% names(unsplit.data),
-    id.var %in% names(split.data),
-    uniqueN(unsplit.data[[id.var]]) == nrow(unsplit.data)
-  )
-  
-  status_vars <- c("lex.Cst", "lex.Xst")
-  status_ud <- mget_cols(c(id.var, status_vars), unsplit.data)
-  
-  join <- structure(list(split.data[[id.var]]), names = id.var)
-  lex_cst <- status_ud[
-    i = join, 
-    j = lex.Cst, 
-    on = id.var
-    ]
-  storage.mode(lex_cst) <- storage.mode(split.data[["lex.Cst"]])
-  set(split.data, j = status_vars, value = list(lex_cst, lex_cst))
-  
-  wh_last_row <- which(!duplicated(split.data, by = id.var, fromLast = TRUE))
-  join <- structure(list(split.data[[id.var]][wh_last_row]), names = id.var)
-  last_lex_xst <- status_ud[
-    i = join, 
-    j = lex.Xst, 
-    on = id.var
-    ]
-  storage.mode(last_lex_xst) <- storage.mode(split.data[["lex.Xst"]])
-  set(split.data, i = wh_last_row, j = "lex.Xst", value = last_lex_xst)
-  
-  
-  NULL
-}
-
-
-
-
-random_splitting_on <- function(
-  lex,
-  n.max.breaks = 20
-) {
-  stopifnot(
-    inherits(lex, "Lexis")
-  )
-  
-  ts_nms <- attr(lex, "time.scales")
-  brks <- attr(lex, "breaks")
-  timesince <- attr(lex, "time.since")
-  lex_vars <- c(paste0("lex.", c("id", "Cst", "Xst", "dur")), ts_nms)
-  non_lex_vars <- setdiff(names(lex), lex_vars)
-  lex <- mget_cols(lex_vars, lex)
-  setattr(lex, "time.scales", ts_nms)
-  setattr(lex, "breaks", brks)
-  setattr(lex, "time.since", timesince)
-  setattr(lex, "class", c("Lexis", "data.table", "data.frame"))
-  checkLexisData(lex)
-  
-  n_split_ts <- sample(seq_along(ts_nms), 1)
-  split_ts_nms <- sample(ts_nms, size = n_split_ts)
-  
-  do_drop <- sample(list(FALSE, TRUE), size = 1)[[1]]
-  
-  bl <- lapply(split_ts_nms, function(split_ts_nm) {
-    r <- c(min(lex[[split_ts_nm]]), max(lex[[split_ts_nm]] + lex[["lex.dur"]]))
-    d <- diff(r)
-    
-    br_r <- if (do_drop) 2:n.max.breaks else 1:n.max.breaks
-    n_br <- sample(br_r, 1)
-    
-    ## allow breaks outside observed data, but at least one break must be
-    ## not outside range of values in data
-    extrema <- r + c(-1,1)*d*0.05
-    l <- rep(extrema[1], n_br)
-    u <- rep(extrema[2], n_br)
-    u[1] <- l[1] <- mean(r)
-    sort(unique(runif(min = l, max = u, n = n_br)))
-  })
-  names(bl) <- split_ts_nms
-  
-  es <- ps  <- lex
-  for (ts_nm in split_ts_nms) {
-    es <- Epi::splitLexis(es, breaks = bl[[ts_nm]], time.scale = ts_nm)
-    forceLexisDT(es, breaks = attr(es, "breaks"), allScales = ts_nms,
-                 key = FALSE)
-    if (do_drop) {
-      es <- intelliDrop(es, breaks = bl[ts_nm])
-    }
-    ps <- splitLexisDT(ps, breaks = bl[[ts_nm]], timeScale = ts_nm, 
-                       drop = do_drop)
-  }
-  
-  psm <- splitMulti(lex, breaks = bl, drop = do_drop)
-  
-  list(es = es, ps = ps, psm = psm)
-}
-
-
-
-
-
-random_Lexis <- function(
-  n.rows = c(100, 1000, 2000), 
-  n.time.scales = 1:10,
-  n.statuses = 2:10,
-  n.other.vars = 1
-) {
-  
-  row_n <- sample(as.list(n.rows), 1)[[1]]
-  
-  ts_n <- sample(as.list(n.time.scales), 1)[[1]]
-  
-  st_n <- sample(as.list(n.statuses), 1)[[1]]
-  
-  dt <- setDT(lapply(1:ts_n, function(i) {
-    runif(min = 0, max = 1000, n = row_n)
-  }))
-  ts_nms <- paste0("lex_ts_", formatC(seq_len(ncol(dt)), flag = "0", width = 3))
-  setnames(dt, names(dt), ts_nms)
-  
-  dt[, "lex.Cst" := sample(1:st_n, size = .N, replace = TRUE)]
-  dt[, "lex.Xst" := sample(1:st_n, size = .N, replace = TRUE)]
-  dt[, "lex.id" := sample(1:.N, .N, replace = FALSE)]
-  dt[, "lex.dur" := runif(n = .N, min = 0, max = 10)]
-  
-  oth_n <- sample(as.list(n.other.vars), 1)[[1]]
-  lapply(seq_len(oth_n), function(i) {
-    set(
-      dt, j = makeTempVarName(names = names(dt), pre = "nonlexvar_"), 
-      value = sample(1:100, size = nrow(dt), replace = TRUE)
-    )
-  })
-  
-  brks <- lapply(ts_nms, function(nm) NULL)
-  names(brks) <- ts_nms
-  
-  forceLexisDT(dt, breaks = brks, allScales = ts_nms, key = TRUE)
-  checkLexisData(dt, check.breaks = TRUE)
-  dt[]
-}
-
-
-
-
-
-random_splitting_on_random_data <- function(
-  n.datasets = 100, 
-  n.rows = 1000,
-  n.time.scales = 1:10,
-  n.breaks = 10:100,
-  n.statuses = 1:5,
-  n.other.vars = 1
-) {
-  
-  neql <- vector("list", n.datasets)
-  
-  for (i in 1:n.datasets) {
-    
-    
-    set.seed(get_random_seed())
-    
-    drop <- sample(list(TRUE, FALSE), 1)[[1]]
-    drop <- FALSE
-    
-    dt <- random_Lexis(
-      n.rows = n.rows,
-      n.time.scales = n.time.scales,
-      n.statuses = n.statuses,
-      n.other.vars = n.other.vars
-    )
-    
-    ts_names <- copy(attr(dt, "time.scales"))
-    ts_n <- length(ts_names)
-    
-    dt_bl <- lapply(ts_names, function(x) NULL)
-    names(dt_bl) <- ts_names
-    forceLexisDT(dt, breaks = dt_bl, allScales = ts_names)
-    
-    br_n <- unlist(sample(as.list(n.breaks), ts_n))
-    names(br_n) <- ts_names
-    BL <- lapply(ts_names, function(ts_name) {
-      runif(n = br_n[[ts_name]], min = -100, max = 100)
-    })
-    names(BL) <- ts_names
-    
-    BL <- BL[sample(ts_names, ts_n)]
-    
-    split_pop <- splitMulti(dt, breaks = BL, drop = drop)
-    split_epi <- splitMultiEpi(dt, breaks = BL, drop = drop)
-    
-    setkeyv(split_epi, c("lex.id", ts_names[1]))
-    setkeyv(split_pop, c("lex.id", ts_names[1]))
-    
-    summary_epi <- summarize_Lexis(split_epi)
-    summary_pop <- summarize_Lexis(split_pop)
-    
-    eq <- all.equal(summary_pop, summary_epi, check.attributes = FALSE)
-    if (!isTRUE(eq)) {
-      message("split_epi, split_pop not equal in tick ", i, "")
-      neql[[i]] <- mget(c(
-        "drop", "row_n", "ts_n", "ts_names", 
-        "dt", "dt_bl", "br_n", "BL",
-        "split_epi", "split_pop", "eq",
-        "summary_epi", "summary_pop",
-        "used_seed"
-      ))
-    }
-  }
-  
-  neql[vapply(neql, is.null, logical(1))]
-  neql
-}
-
-
-
-
-
-do_split <- function(x, ts, all.ts, breaks, drop = TRUE, merge = TRUE) {
-  
-  ## unfinished v2 splitlexisDT work horse
-  stopifnot(
-    is.integer(x[["lex.id"]])
-  )
-  
-  id_dt <- data.table(
-    "orig" = x[["lex.id"]],
-    "temp" = 1:nrow(x)
-  )
-  set(x, j = "lex.id", value = id_dt[["temp"]])
-  
-  split <- mget_cols(c(ts, "lex.id", "lex.dur"), x)
-  
-  BL <- structure(list(breaks), names = ts)
-  if (drop) {
-    split <- intelliCrop(split, breaks = BL, allScales = all.ts, 
-                         cropStatuses = TRUE)
-    split <- intelliDrop(x = split, breaks = BL, 
-                         check = FALSE, dropNegDur = TRUE)
-  }
-  n_subjects <- nrow(split)
-  ts_values <- split[[ts]]
-  
-  split <- rbindlist(lapply(1:length(breaks), function(i) get("split")))
-  
-  tmp_ie_nm <-  makeTempVarName(names = names(x), pre = "do_split_tmp_ie_")
-  
-  set(split, j = tmp_ie_nm,  value = rep(breaks, each = n_subjects))
-  set(split, j = tmp_ie_nm,  value = {
-    pmin(split[[tmp_ie_nm]], split[[ts]] + split[["lex.dur"]])
-  })
-  set(split, j = ts, value = c(
-    ts_values, 
-    pmax(ts_values, rep(breaks[-length(breaks)], each = n_subjects))
-  ))
-  
-}
-
-
-
-
-
+all_breaks_in <- function(bl1, bl2, x = NULL) {
+  ## INTENTION: return TRUE/FALSE depending on whether bl1 is a subset of bl2;
+  ## this means that each element in bl1 exists in bl2, and that those elements
+  ## are each subsets of the corresponding elements in bl2.
+  ## this is handy to check whether the some Lexis data has already
+  ## been split using the breaks in bl1.
+  ## NOTE: use checkBreakList() on each list separately before this.
+  
+  if (!is.list(bl1) || !is.list(bl2)) {
+    stop("Arguments bl1 and bl2 must be lists of breaks as supplied to e.g. ",
+         "splitMulti.")
+  }
+  
+  if (inherits(x, "Lexis")) {
+    checkLexisData(x)
+    checkBreaksList(x, bl1)
+    checkBreaksList(x, bl2)
+  }
+  
+  ce <- intersect(names(bl1), names(bl2))
+  if (length(ce) != length(bl1)) return(FALSE)
+  
+  test <- mapply(function(l1, l2) {
+    all(l1 %in% l2)
+  }, l1 = bl1, l2 = bl2[ce], SIMPLIFY = FALSE)
+  
+  all(unlist(test))
+}
+
+
+checkBreaksList <- function(x, breaks = list(fot = 0:5)) {
+  if (is.null(breaks)) stop("breaks is NULL")
+  if (!is.list(breaks)) stop("breaks needs to be a list")
+  if (!is.data.frame(x)) stop("x needs to be a data.frame")
+  timeScales <- names(breaks)
+  if (length(breaks) == 0L) stop("length of breaks list is zero")
+  if (length(timeScales) != length(breaks)) stop("breaks needs to be a fully named list")
+  
+  bad_scales <- setdiff(timeScales, names(x))
+  if (length(bad_scales) > 0) {
+    stop("at least one breaks list name wasn't a variable in data; bad names: ", 
+         paste0("'", bad_scales, "'", collapse = ", "))
+  }
+  lens <- lapply(breaks, function(el) if (is.null(el)) -1 else length(el))
+  badLens <- names(lens[unlist(lens) == 0L])
+  if (length(badLens)) {
+    badLens <- paste0("'", badLens, "'", collapse = ", ")
+    stop("Elements in breaks list for the following time scales were of ",
+         "length zero but not NULL: ", badLens, ". Breaks list may only ",
+         "contain elements of length > 0 or elements that are NULL.")
+  }
+  invisible(NULL)
+}
+
+checkPophaz <- function(lex, ph, haz.name = "haz") {
+  ## INTENTION: checks a Lexis data set against the pophaz data set for
+  ## consistency (e.g. existing variables to merge by)
+  
+  if (!is.data.frame(ph)) {
+    stop("Data set containing population/expected hazards must be a data.frame",
+         " (or a data.table, which is also a data.frame).")
+  }
+  
+  if (!haz.name %in% names(ph)) {
+    stop("Data set containing population/expected hazards does not contain a ",
+         "column named 'haz'. Make sure the name is exactly that (",
+         "case sensitive).")
+  }
+  
+  if (haz.name %in% names(lex)) {
+    stop("Lexis data set already contains a column named 'haz', which is a ",
+         "reserved name for the population hazard variable to be merged. ",
+         "Please rename/delete 'haz' from/in your Lexis data first.")
+  }
+  
+  if (!is.data.frame(ph)) {
+    stop("Data set of expected/population hazards must be a data.frame.")
+  }
+  
+  bn <- setdiff(names(ph), haz.name)
+  
+  if (length(bn) == 0L) {
+    stop("No variables in expected/population hazards data set to use in merge ",
+         "with Lexis data. Ensure that the pop. haz. data set containts some ",
+         "variables to merge by (e.g. sex, calendar year, and age group)")
+  }
+  if (!all(bn %in% names(lex))) {
+    badbn <- paste0("'", setdiff(bn, names(lex)), "'", collapse = ", ")
+    stop("Lexis data set did not have following variable(s) that were in ",
+         "the expected/population hazards data set: ", badbn,". ",
+         "Ensure you have supplied the right data and that the names of the ",
+         "intended variables match.")
+  }
+  
+  mergeVars <- setdiff(names(ph), haz.name)
+  dup <- any(duplicated(as.data.table(ph), by = mergeVars))
+  if (dup) {
+    stop("Supplied data set of population/expected hzards has duplicated rows ",
+         "by the variables ", paste0("'",mergeVars, "'", collapse = ", "),
+         " which prevents correct usage of the data set. Please ensure no rows",
+         " area duplicated in the data set before proceeding. Tip: use e.g. ",
+         "duplicated(PH, by = c('V1', 'V2')) to check for duplicatedness in ",
+         "your data set (here named PH) by the variables V1 and V2."
+    )
+  }
+  
+  invisible()
+}
+
+
+
+
+
+intelliCrop <- function(
+  x, 
+  breaks = list(fot = 0:5), 
+  allScales = NULL, 
+  cropStatuses = FALSE, 
+  tol = .Machine$double.eps^0.5
+) {
+  
+  ## appease R CMD CHECK
+  lex.dur <- lex.Xst <- lex.Cst <- NULL
+  
+  checkBreaksList(x = x, breaks = breaks)
+  breaks[unlist(lapply(breaks, length)) == 0L] <- NULL
+  if (!is.data.table(x)) stop("x needs to be a data.table")
+  
+  cropScales <- names(breaks)
+  
+  all_names_present(x, c("lex.dur", allScales))
+  
+  if (cropStatuses) {
+    origEnd <- x$lex.dur + x[[allScales[1L]]]
+  }
+  
+  
+  deltas <- mapply(function(b, y) pmax(min(b), y) - y, SIMPLIFY = FALSE,
+                   b = breaks, y = mget_cols(cropScales, x))
+  ## below: baseline (zero value without assigning zero of bad class)
+  deltas <- c(deltas, list(x[[cropScales[1]]][1L] - x[[cropScales[1]]][1L]))
+  deltas <- do.call(pmax, deltas)
+  
+  set(x, j = allScales, value = mget_cols(allScales, x) + deltas)
+  set(x, j = "lex.dur", value = x[["lex.dur"]] - deltas)
+  
+  durs <- mapply(function(b, y) max(b) - y, SIMPLIFY = FALSE,
+                 b = breaks, y = mget_cols(cropScales, x))
+  durs$lex.dur <- x$lex.dur
+  durs <- do.call(pmin, durs)
+  ## now have max durs by row, i.e. up to roof of breaks at most,
+  ## or to ((original lex.dur) - (deltas)) if that is smaller.
+  ## (being cropped or exiting before roof of breaks)
+  
+  set(x, j = "lex.dur", value = durs)
+  
+  if (cropStatuses) {
+    harmonizeStatuses(x, C = "lex.Cst", X = "lex.Xst")
+    wh_was_cropped <- which(x[["lex.dur"]] + x[[allScales[1L]]] + tol < origEnd)
+    set(x, i = wh_was_cropped, j = "lex.Xst", 
+        value = x[["lex.Cst"]][wh_was_cropped])
+  }
+  
+  invisible(x)
+}
+
+
+
+
+
+harmonizeStatuses <- function(x, C = "lex.Cst", X = "lex.Xst") {
+  
+  clC <- class(x[[C]])
+  clX <- class(x[[X]])
+  tyC <- typeof(x[[C]])
+  tyX <- typeof(x[[X]])
+  cl <- c(clC, clX)
+  
+  if (tyC != tyX && clC != clX) {
+    if (is.numeric(x[[C]]) && is.numeric(x[[X]])) {
+      harmonizeNumeric(x = x, v1="lex.Cst", v2="lex.Xst")
+      
+    } else if (is.factor(x[[C]]) || is.factor(x[[X]])) {
+      if (!is.factor(x[[C]])) set(x, j = C, value = as.factor(x[[C]]))
+      if (!is.factor(x[[X]])) set(x, j = X, value = as.factor(x[[X]]))
+      
+    }
+  }
+  
+  if (any(cl == "factor")) {
+    harmonizeFactors(x = x,  v1="lex.Cst", v2="lex.Xst")
+  }
+  
+}
+
+harmonizeNumericTimeScales <- function(x, times = NULL) {
+  ## INTENTION: given a Lexis data set with some time scales, ensure
+  ## that the classes of the time scales comply to the lowest denominator,
+  ## e.g. "double" and "integer" -> both "double"
+  
+  if (is.null(times)) {
+    times <- c(attr(x, "time.scales"), "lex.dur")
+  }
+  
+  msg <- paste0("Expected working data to have time scales %%VARS%%, but it ",
+                "didn't. This is an internal error: If you see this, complain ",
+                "to the package maintainer.")
+  all_names_present(x, times, msg = msg)
+  xt <- lapply(times, function(ch) x[[ch]])
+  names(xt) <- times
+  
+  harmoClasses <- c("numeric", "integer", "difftime")
+  cl <- lapply(xt, class)
+  wh <- unlist(lapply(cl, function(ch) {
+    any(ch %in% harmoClasses)
+  }))
+  ha <- times[wh]
+  hacl <- unique(unlist(cl[wh]))
+  
+  if (length(ha) > 1L) {
+    ## more than one class present and need to use common lowest denom
+    newMode <- as.double
+    
+    if (all(ha %in% c("integer", "difftime"))) {
+      ## all numeric times are integers or difftimes
+      newMode <- as.integer
+    }
+    for (var in ha) {
+      ## modify in place
+      set(x, j = var, value = newMode(x[[var]]))
+    }
+    
+    
+  }
+  invisible(NULL)
+}
+
+
+
+
+
+harmonizeNumeric <- function(x, v1="lex.Cst", v2="lex.Xst") {
+  ## assumes v1, v2 are numeric variable names in x  
+  
+  if (!is.numeric(x[[v1]]) || !is.numeric(x[[v2]])) {
+    print(class(x[[v1]]))
+    print(class(x[[v2]]))
+    stop("v1 and/or v2 is/are not of class numeric")
+  }
+  
+  if (!is.integer(x[[v1]])) set(x, j = v1, value = try2int(x[[v1]]))
+  if (!is.integer(x[[v2]])) set(x, j = v2, value = try2int(x[[v2]]))
+  
+  if (typeof(x[[v1]]) != typeof(x[[v2]])) {
+    
+    if (is.double(x[[v1]]))  set(x, j = v1, value = as.double(x[[v1]]))
+    if (is.double(x[[v2]]))  set(x, j = v2, value = as.double(x[[v2]]))
+    
+  }  
+  
+}
+
+
+
+
+
+harmonizeFactors <- function(x, v1="lex.Cst", v2="lex.Xst") {
+  ## assumes v1, v2 are factor names in x
+  
+  if (!is.factor(x[[v1]]) || !is.factor(x[[v2]])) {
+    stop("v1 and/or v2 is/are not of class factor")
+  }
+  
+  glab1 <- union(levels(x[[v1]]), levels(x[[v2]]))
+  glab2 <- union(levels(x[[v2]]), levels(x[[v1]]))
+  
+  
+  
+  setattr(x[[v1]], "levels", glab1)
+  setattr(x[[v2]], "levels", glab2)
+  
+}
+
+
+
+
+
+intelliDrop <- function(x, breaks = list(fot = 0:5), dropNegDur = TRUE, check = FALSE, tol = .Machine$double.eps^0.5, subset = NULL)  {
+  
+  if (!is.data.table(x)) {
+    stop("x needs to be a data.table; if you see this message, complain ",
+         "to the package maintainer")
+  }
+  checkBreaksList(x = x, breaks = breaks)
+  breaks[unlist(lapply(breaks, length)) < 2] <- NULL
+  timeScales <- names(breaks)
+  
+  if (check) {
+    checkLexisData(x)
+  }
+  
+  ra <- lapply(breaks, range)
+  ra <- lapply(ra, diff)
+  ts <- names(sort(unlist(ra))) ## shortest first
+  mi <- lapply(breaks, min)
+  ma <- lapply(breaks, max)
+  
+  substi <- substitute(subset)
+  subset <- evalLogicalSubset(x, substiset = substi)
+  
+  if (dropNegDur) subset[subset] <- subset[subset] & x$lex.dur[subset] > 0L
+  
+  ## figure out latest exit and first entry; don't need to test for dropping
+  ## if e.g. all left follow-up before the max in breaks
+  max_end <- lapply(ts, function(ch) min(x[[ch]] + x$lex.dur))
+  min_start <- lapply(ts, function(ch) max(x[[ch]]))
+  names(max_end) <- names(min_start) <- ts
+  
+  for (k in ts) {
+    mik <- mi[[k]]
+    mak <- ma[[k]]
+    
+    if (max_end[[k]] < mak + tol) {
+      tmpSD <- x[subset, .SD, .SDcols = c(k, "lex.dur")]
+      tmpSD <- setDT(lapply(tmpSD, as.numeric))
+      subset[subset] <- rowSums(tmpSD)  <=  mak + tol
+    }
+    if (min_start[[k]] + tol > mik) {
+      subset[subset] <- x[subset,][[k]] > mik - tol
+    }
+    
+    if (all(!subset)) {
+      stop("Dropped all remaining rows from data when subsetting by the  ",
+           "Lexis time scale '", k, "'. Range of values in data: ", 
+           paste0(round(range(x[[k]]),4), collapse = "-"), ". Min/Max breaks ",
+           "(used to subset data): ", mik, "/", mak, ".")
+    }
+    
+  }
+  
+  
+  x[subset, ]
+}
+
+
+matchBreakTypes <- function(lex, breaks, timeScale, modify.lex = FALSE) {
+  if (is.character(breaks)) {
+    breaks <- as.IDate(breaks)
+  }
+  clb <- class(breaks)
+  clb <- clb[length(clb)]
+  cts <- class(lex[[timeScale]])
+  cts <- cts[length(cts)]
+  
+  if (clb != cts) {
+    if (is.Date(breaks) && !is.Date(lex[[timeScale]])) {
+      breaks <- try2int(as.double(breaks))
+    } else if (is.integer(breaks) && is.double(lex[[timeScale]])) {
+      breaks <- as.double(breaks)
+    } else if (is.double(breaks) && is.integer(lex[[timeScale]])) {
+      breaks <- try2int(breaks)
+    }
+    
+  }
+  
+  if (modify.lex && clb != cts) {
+    if (!is.Date(breaks) && is.Date(lex[[timeScale]])) {
+      
+      if (clb == "double") {
+        set(lex, j = timeScale, value = as.double(lex[[timeScale]]))
+      } else {
+        set(lex, j = timeScale, value = as.integer(lex[[timeScale]]))
+      }
+      
+    } else if (is.double(breaks) && is.integer(lex[[timeScale]])) {
+      set(lex, j = timeScale, value = as.double(lex[[timeScale]]))
+    }
+    
+  }
+  breaks
+}
+
+protectFromDrop <- function(breaks, lower = FALSE) {
+  old_breaks <- copy(breaks)
+  if (length(breaks) == 0L) {
+    stop("Length of breaks to 'protect' from dropping is zero.")
+  }
+  if (is.Date(breaks)) {
+    breaks <- c(breaks, max(breaks) + 1e4L)
+    if (lower) breaks <- c(min(breaks) - 1e4L, breaks)
+    
+  } else if (is.integer(breaks))  {
+    breaks <- c(breaks, 1e6L)
+    if (lower) breaks <- c(-1e6L, breaks)
+    
+  } else if (is.double(breaks)) {
+    breaks <- c(breaks, Inf)
+    if (lower) breaks <- c(-Inf, breaks)
+    
+  } else {
+    stop("breaks were not Date, integer or double")
+  }
+  setattr(breaks, "unprotected", old_breaks)
+  breaks
+}
+
+unprotectFromDrop <- function(breaks) {
+  up <- attr(breaks, "unprotected")
+  if (is.null(up) || length(up) == 0L) {
+    stop("Could not 'unprotect' breaks from dropping as the required ",
+         "attribute was not found. If you see this it is most likely ",
+         "an internal error and you should complain to the pkg maintainer.")
+  }
+  up
+}
+
+
+
+
+setLexisDT <- function(data, entry, exit, entry.status, exit.status, id = NULL, select = NULL) {
+  
+  ## appease R CMD CHECK
+  lex.Cst <- lex.Xst <- NULL
+  
+  if (!is.data.table(data)) stop("not a data.table")
+  if (inherits(data, "Lexis")) stop("already a Lexis object")
+  
+  if (!is.null(select) && !is.character(select)) stop("select was not a character vector of names")
+  
+  entry <- substitute(entry)
+  exit <- substitute(exit)
+  entry <- eval(entry, envir = data, enclos = parent.frame())
+  exit <- eval(exit, envir = data, enclos = parent.frame())
+  enNames <- names(entry)
+  exNames <- names(exit)
+  
+  timeScales <- union(enNames, exNames)
+  if (any(timeScales %in% names(data))) stop("at least one named time scales already present in data; original names mandatory")
+  enNeeded <- setdiff(timeScales, enNames)
+  enPresent <- setdiff(timeScales, enNeeded)
+  durVar <- intersect(enNames, exNames)
+  if (length(durVar) > 1) stop("you have more than 1 time scales in both entry and exit; only one mandatory")
+  
+  enVars <- paste0(enNames, "_en")
+  exVars <- paste0(exNames, "_ex")
+  setattr(entry, "names", enVars)
+  setattr(exit, "names", exVars)
+  
+  l <- as.data.table(c(entry, exit))
+  rm(entry, exit)
+  
+  ## duration
+  exV <- paste0(durVar, "_ex")
+  enV <- paste0(durVar, "_en")
+  set(l, j = "lex.dur", value = l[[exV]] - l[[enV]])
+  rm(exV, enV)
+  
+  ## time scale starting points
+  if (length(enNeeded) > 0) {
+    for (ts in enNeeded) {
+      exV <- paste0(ts, "_ex")
+      set(l, j = ts, value = l[[exV]] - l$lex.dur)
+    }
+  }
+  setnames(l, paste0(enPresent, "_en"), enPresent)
+  
+  # no longer need time scale end points
+  for (k in exVars) {
+    set(l, j = k, value = NULL)
+  }
+  
+  ## status definition
+  data[, lex.Cst := entry.status]
+  data[, lex.Xst := exit.status]
+  
+  harmonizeStatuses(data, C = "lex.Cst", X = "lex.Xst")
+  
+  ## all time scales etc. into data
+  data[, names(l) := l]
+  
+  
+  id <- substitute(id)
+  id <- eval(id, envir = data, enclos = parent.frame())
+  if (!is.null(id)) set(data, j = "lex.id", value = id)
+  rm(id)
+  
+  if (!is.null(select)) {
+    
+    delVars <- setdiff(names(data), c(names(l), select))
+    if (length(delVars) > 0) {
+      l[, (delVars) := NULL]
+    }
+  }
+  
+  rm(l)
+  lexVars <- c("lex.id", timeScales, "lex.dur", "lex.Cst", "lex.Xst")
+  setcolorder(data, c(lexVars, setdiff(names(data), lexVars)))
+  
+  setattr(data, "time.scales", timeScales)
+  setattr(data, "time.since", rep("", times = length(timeScales)))
+  setattr(data, "class", c("Lexis", "data.table", "data.frame"))
+  
+  
+}
+
+checkLexisData <- function(lex, check.breaks = FALSE) {
+  ## INTENTION: checks Lexis attributes
+  ## OUTPUT: nothing
+  
+  if (missing(lex)) {
+    stop("Data object not supplied")
+  }
+  if (!inherits(lex, "Lexis")) {
+    stop("Data not a Lexis object; it has class(es) ", deparse(class(lex)))
+  }
+  if (!is.data.frame(lex)) {
+    stop("Data is not a data.frame / data.table; it has class(es) ",
+         deparse(class(lex)))
+  }
+  if (nrow(lex) == 0) {
+    stop("Data has zero rows")
+  }
+  allScales <- attr(lex, "time.scales")
+  if (length(allScales) == 0) {
+    stop("no time scales appear to be defined; is data a Lexis object?")
+  }
+  
+  badScales <- setdiff(allScales, names(lex))
+  if (length(badScales) > 0) {
+    badScales <- paste0("'", badScales, "'", collapse = ", ")
+    stop("Following time scales found in data's attributes but not present ",
+         "in data: ", badScales)
+  }
+  
+  lexVars <- c("lex.dur", "lex.id", "lex.Cst", "lex.Xst")
+  blv <- setdiff(lexVars, names(lex))
+  if (length(blv) > 0) {
+    blv <- paste0("'", blv, "'", collapse = ", ")
+    stop("Following Lexis variables not found in data: ", blv)
+  }
+  
+  if (check.breaks) {
+    BL <- attr(lex, "breaks")
+    if (is.null(BL)) stop("No breaks list in data attributes")
+    checkBreaksList(lex, breaks = BL)
+  }
+  
+  invisible()
+}
+
+
+splitMultiPreCheck <- function(data = NULL, breaks = NULL, ...) {
+  
+  ## INTENTION: checks for discrepancies between data and breaks, etc.
+  ## OUTPUT: cleaned-up list of breaks
+  checkLexisData(data)
+  allScales <- attr(data, "time.scales")
+  
+  if (!is.null(breaks) && !is.list(breaks)) {
+    stop("breaks must be a list; see examples in ?splitMulti")
+  }
+  if (is.null(breaks)) {
+    breaks <- list(...)
+    breaks <- breaks[intersect(names(breaks), allScales)]
+  }
+  
+  if (length(breaks) == 0) stop("no breaks defined!")
+  
+  splitScales <- names(breaks)
+  ## NULL breaks imply not used
+  for (k in splitScales) {
+    if (length(breaks[[k]]) == 0) {
+      breaks[k] <- NULL
+    }
+  } 
+  
+  checkBreaksList(x = data, breaks = breaks)
+  
+  splitScales <- names(breaks)
+  
+  if (!all(splitScales %in% allScales)) {
+    stop("breaks must be a list with at least one named vector corresponding to used time scales \n
+         e.g. breaks = list(fot = 0:5)")
+  }
+  
+  if (!all(splitScales %in% names(data))) {
+    stop("At least one vector name in breaks list is not a variable name in the data")
+  }
+  breaks
+}
+
+forceLexisDT <- function(x, breaks = NULL, allScales = NULL, key = TRUE) {
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  setattr(x, "breaks", breaks)
+  setattr(x, "time.scales", allScales)
+  # alloc.col(x)
+  if (key) setkeyv(x, c("lex.id", names(breaks)[1L]))
+  invisible(x)
+}
+
+
+doCutLexisDT <- function(lex, cut = dg_date, timeScale = "per", by = "lex.id", n = 1L) {
+  
+  checkLexisData(lex, check.breaks = FALSE)
+  
+  x <- unique(lex, by = by)
+  cut <- evalq(cut, envir = lex, enclos = parent.frame(n = n + 1L))
+  delta <- cut - x[[timeScale]]
+  
+  allScales <- attr(lex, "time.scales")
+  
+  setDT(x)
+  for (v in allScales) {
+    set(x, j = v, value = x[[v]] + delta)
+  }
+  
+  set(x, j = "lex.dur", value = 0)
+  
+  tmp <- list()
+  tmp$isCut <- makeTempVarName(lex, pre = "isCut_")
+  
+  set(x, j = tmp$isCut, value = 1L)
+  on.exit(setcolsnull(x, unlist(tmp)))
+  
+  x <- rbindlist(list(lex, x), use.names = TRUE, fill = TRUE)
+  x[1:nrow(lex), (tmp$isCut) := 0L]
+  
+  ## NOTE: new cut row being the first or last row
+  ## implies it resides outside old observations
+  ## OR it is equal to lowest/highest value
+  setkeyv(x, c(by, allScales))
+  setkeyv(x, by)
+  x <- x[!((duplicated(x, by = key(x)) | duplicated(x, by = key(x), fromLast = TRUE)) & x[[tmp$isCut]] == 0L)]
+  stop("not ready")
+}
+
+# data <- data.table(birth = 2000:2000, entry=2002:2003, 
+#                    exit=2011:2012, event=c(2010,2011), 
+#                    status=1:0)
+# 
+# lex <- lexpand(data = data, birth = birth, entry = entry, 
+#                exit = exit, event = event, 
+#                id = 1L, entry.status = 99L,
+#                status = status, overlapping = TRUE)
+
+lexpile <- function(lex, by = "lex.id", subset = NULL) {
+  ## PURPOSE: given several rows per id in a Lexis object,
+  ## collate data into form where
+  ## - no subject has any overlapping time lines
+  ## - lex.Cst and lex.Xst are logical, i.e. 0 -> 1, 1 -> 1, 1 -> 2
+  ## this should be made to work with both split and unsplit Lexis data.
+  
+  data <- NULL # R CMD CHECK appeasement
+  
+  checkLexisData(lex, check.breaks = FALSE)
+  
+  allScales <- attr(lex, "time.scales")
+  sc <- allScales[1L]
+  
+  all_names_present(lex, by)
+  
+  if (is.character(lex$lex.Cst) || is.character(lex$lex.Xst)) {
+    stop("This function requires lex.Cst and lex.Xst to be integer, double (i.e. numeric) or factor variables to determine the order of possible statuses!")
+  }
+  
+  ## need to take copy eventually ----------------------------------------------
+  attrs <- attributes(lex)
+  subset <- evalLogicalSubset(data, substitute(subset))
+  x <- lex[subset,]
+  forceLexisDT(x, breaks = attrs$breaks, allScales = attrs$time.scales)
+  alloc.col(x)
+  
+  
+  ## ensure status harmony -----------------------------------------------------
+  harmonizeStatuses(x = x, X = "lex.Xst", C = "lex.Cst")
+  exStat <- if (is.factor(x$lex.Xst)) levels(x$lex.Xst) else sort(unique(x$lex.Xst))
+  enStat <- if (is.factor(x$lex.Cst)) levels(x$lex.Cst) else sort(unique(x$lex.Cst))
+  allStat <- c(setdiff(enStat, exStat), exStat) ## enStat & exStat equal if factors used
+  
+  ## avoiding side effects -----------------------------------------------------
+  oldKey <- key(lex)
+  tmp <- list()
+  tmp$order<- makeTempVarName(x, pre = "order_")
+  
+  on.exit({
+    if (length(oldKey) > 0) setkeyv(x, oldKey) else 
+      setorderv(x, tmp$order)
+  }, add = TRUE)
+  
+  on.exit({
+    setcolsnull(x, unlist(tmp$order), soft = TRUE)
+  }, add = TRUE)
+  
+  x[, c(tmp$order) := 1:.N]
+  
+  ## check for need for lexpiling ----------------------------------------------
+  setkeyv(x, by)
+  if (sum(duplicated(x, by = key(x))) == 0L) return(lex)
+  
+  
+  ## figure out what statuses are used -----------------------------------------
+  
+  tmp$ev <- makeTempVarName(x, pre = "event_")
+  x[, c(tmp$ev) := detectEvents(x, breaks = attrs$breaks, by = by)]
+  
+  tmp$scEnds <- paste0(allScales, "_end")
+  tmp$scEnds <- makeTempVarName(lex, pre = tmp$scEnds)
+  x[, c(tmp$scEnds) := lapply(.SD, function(x) x + lex$lex.dur), .SDcols = allScales]
+  
+  ## NOTE: rows for a given subject ending in simultaneously with at least
+  ## one being a transition will not be allowed.
+  setkeyv(x, c(by, tmp$scEnds[1L]))
+  
+  whDup <- duplicated(x, fromLast = FALSE, by = key(x)) | duplicated(x, fromLast = TRUE, by = key(x))
+  dupTest <- x[whDup, 1L %in% unique(.SD), .SDcols = tmp$ev]
+  rm(whDup)
+  
+  if (dupTest) stop("At least one subject had at least two simultaneous events.")
+  ## NOTE: if interval ends AND status are the very same for M rows,
+  ## then the M rows are necessarily nested with one or more covering
+  ## the whole time line. Only need to keep the one.
+  setkeyv(x, c(tmp$scEnds, "lex.Cst", "lex.Xst"))
+  setorderv(x, c(allScales,tmp$scEnds, "lex.Cst", "lex.Xst"))
+  
+  x <- unique(x, by = key(x))
+  stop("unfinished")
+  
+}
+
+contractLexis <- function(x, breaks, drop = TRUE) {
+  stop("This doesnt do anything yet")
+  ## INTENTION: given a Lexis object and breaks,
+  ## ensures data is split by the breaks and contracts the split rows
+  ## so that the data is split at the level of the supplied breaks.
+  ## e.g. with x split by fot = seq(0, 5, 1/12) and with supplying
+  ## breaks = list(fot = 0:5), rows within 0-1 are collated into one row etc.
+  
+  ## PROBLEM: a subject may have e.g. rows spaning 0.5 - 1 which are requested
+  ## to be contracted to one row spanning 0-1.
+  
+  
+}
+
+
+
+
+#' @title Prepare Exposure Data for Aggregation
+#' @description \code{prepExpo} uses a \code{Lexis} object of periods of exposure
+#' to fill gaps between the periods and overall entry and exit times without
+#' accumulating exposure time in periods of no exposure, and splits the
+#' result if requested.
+#' @param lex a \code{\link[Epi]{Lexis}} object with ONLY periods of exposure
+#' as rows; one or multiple rows per subject allowed
+#' @param freezeScales a character vector naming \code{Lexis} time scales of exposure
+#' which should be frozen in periods where no exposure occurs (in the gap
+#' time periods) 
+#' @param cutScale the \code{Lexis} time scale along which the subject-specific
+#' ultimate entry and exit times are specified
+#' @param entry an expression; the time of entry to follow-up which may be earlier, at, or after
+#' the first time of exposure in \code{freezeScales}; evaluated separately
+#' for each unique combination of \code{by}, so e.g. with 
+#' \code{entry = min(Var1)} and \code{by = "lex.id"} it 
+#' sets the \code{lex.id}-specific minima of \code{Var1} to be the original times
+#' of entry for each \code{lex.id}
+#' @param exit the same as \code{entry} but for the ultimate exit time per unique
+#' combination of \code{by}
+#' @param by a character vector indicating variable names in \code{lex},
+#' the unique combinations of which identify separate subjects for which
+#' to fill gaps in the records from \code{entry} to \code{exit};
+#' for novices of \code{{\link{data.table}}}, this is passed to a 
+#' \code{data.table}'s \code{by} argument.
+#' @param breaks a named list of breaks; 
+#' e.g. \code{list(work = 0:20,per = 1995:2015)}; passed on to 
+#' \code{\link{splitMulti}} so see that function's help for more details
+#' @param freezeDummy a character string; specifies the name for a dummy variable
+#' that this function will create and add to output which 
+#' identifies rows where the \code{freezeScales} are frozen and where not
+#' (\code{0} implies not frozen, \code{1} implies frozen);
+#' if \code{NULL}, no dummy is created
+#' @param subset a logical condition to subset data by before computations;
+#' e.g. \code{subset = sex == "male"}
+#' @param verbose logical; if \code{TRUE}, the function is chatty and returns
+#' some messages and timings during its run.
+#' @param ... additional arguments passed on to \code{\link{splitMulti}}
+#' @details 
+#' 
+#' \code{prepExpo} is a convenience function for the purpose of eventually aggregating 
+#' person-time and events in categories of not only normally progressing 
+#' \code{Lexis} time scales but also some time scales which should not
+#' progress sometimes. For example a person may work at a production facility
+#' only intermittently, meaning exposure time (to work-related substances 
+#' for example) should not progress outside of periods of work. This allows for
+#' e.g. a correct aggregation of person-time and events by categories of cumulative
+#' time of exposure.
+#' 
+#' Given a \code{Lexis} object containing rows (time lines)
+#' where a subject is exposed to something (and NO periods without exposure),
+#' fills any gaps between exposure periods for each unique combination of \code{by}
+#' and the subject-specific "ultimate" \code{entry} and \code{exit} times,
+#' "freezes" the cumulative exposure times in periods of no exposure,
+#' and splits data using \code{breaks} passed to \code{\link{splitMulti}}
+#' if requested. Results in a (split) \code{Lexis} object where \code{freezeScales}
+#' do not progress in time periods where no exposure was recorded in \code{lex}.
+#' 
+#' This function assumes that \code{entry} and \code{exit} arguments are the
+#' same for each row within a unique combination of variables named in \code{by}.
+#' E.g. with \code{by = "lex.id"} only each \code{lex.id} has a unique value
+#' for \code{entry} and \code{exit} at most.
+#' 
+#' The supplied \code{breaks} split the data using \code{splitMulti}, with
+#' the exception that breaks supplied concerning any frozen time scales
+#' ONLY split the rows where the time scales are not frozen. E.g.
+#' with \code{freezeScales = "work"}, 
+#' \code{breaks = list(work = 0:10, cal = 1995:2010)} splits all rows over
+#' \code{"cal"} but only non-frozen rows over \code{"work"}.
+#' 
+#' Only supports frozen time scales that advance and freeze contemporaneously:
+#' e.g. it would not currently be possible to take into account the cumulative
+#' time working at a facility and the cumulative time doing a single task
+#' at the facility, if the two are not exactly the same. On the other hand
+#' one might use the same time scale for different exposure types, supply them
+#' as separate rows, and identify the different exposures using a dummy variable.
+#' @return 
+#' 
+#' Returns a \code{Lexis} object that has been split if \code{breaks} is specified.
+#' The resulting time is also a \code{data.table} if 
+#' \code{options("popEpi.datatable") == TRUE} (see: \code{?popEpi})
+#' 
+#' @import data.table
+#' @export
+prepExpo <- function(lex, freezeScales = "work", cutScale = "per", entry = min(get(cutScale)),
+                     exit = max(get(cutScale)), by = "lex.id", breaks = NULL, freezeDummy = NULL, subset = NULL,
+                     verbose = FALSE, ...) {
+  ## R CMD CHECK appeasement
+  lex.dur <- NULL
+  
+  if (verbose) allTime <- proc.time()
+  
+  ## check breaks & data -------------------------------------------------------
+  breaks <- evalq(breaks)
+  dumBreaks <- structure(list(c(-Inf, Inf)), names = cutScale, internal_prepExpo_dummy = TRUE)
+  if (is.null(breaks)) breaks <- dumBreaks
+  breaks <- splitMultiPreCheck(data = lex, breaks = breaks)
+  if (!is.null(attr(breaks, "internal_prepExpo_dummy"))) breaks <- NULL
+  checkLexisData(lex)
+  oldBreaks <- attr(lex, "breaks")
+  if (!is.null(breaks)) checkBreaksList(lex, breaks)
+  checkBreaksList(lex, oldBreaks)
+  
+  
+  ## data ----------------------------------------------------------------------
+  
+  subset <- evalLogicalSubset(data = lex, substitute(subset))
+  x <- if (!all(subset)) evalq(lex)[subset, ] else copy(evalq(lex))
+  
+  setDT(x)
+  
+  allScales <- attr(lex, "time.scales")
+  linkScales <- setdiff(allScales, freezeScales)
+  othScales <- setdiff(linkScales, cutScale)
+  
+  setkeyv(x, c(by, cutScale))
+  
+  l <- list() ## will hold temp var names; this avoids collisions with names of vars in x
+  l$cutScale <- cutScale
+  l$freezeScales <- freezeScales
+  l$liquidScales <- setdiff(allScales, freezeScales)
+  l$by <- by
+  rm(cutScale, freezeScales, by)
+  
+  ## args ----------------------------------------------------------------------
+  if (verbose) argTime <- proc.time()
+  tol <- .Machine$double.eps^0.75
+  
+  if (is.character(freezeDummy) && freezeDummy %in% names(lex)) stop("Variable named in freezeDummy already exists in data; freezeDummy is inteded for creating a new dummy for identifying the rows where freezeScales are frozen. Please supply an original variable name to freezeDummy")
+  
+  if (!is.character(l$by)) stop("by must be given as a vector of character strings naming columns in lex")
+  all_names_present(lex, l$by)
+  
+  enSub <- substitute(entry)
+  exSub <- substitute(exit)
+  
+  PF <- parent.frame(1L)
+  l$en <- makeTempVarName(x, pre = "entry_")
+  l$ex <- makeTempVarName(x, pre = "exit_")
+  x[, c(l$ex, l$en) := list(eval(exSub, envir = .SD, enclos = PF), 
+                            eval(enSub, envir = .SD, enclos = PF)), by = c(l$by)]
+  
+  ## tests disabled for now...
+  #   testTime <- proc.time()
+  #   
+  #   test <- x[, .N, by = list(r = get(l$cutScale) + lex.dur > get(l$ex) - tol)]
+  #   if(test[r == TRUE, .N] > 0) stop("exit must currently be higher than or equal to the maximum of cutScale (on subject basis defined using by); you may use breaks instead to limit the data")
+  #   
+  #   test <- x[, .N, by = list(r = get(l$cutScale) + tol < get(l$en))]
+  #   if(test[r == TRUE, .N] > 0) stop("entry must currently be lower than or equal to the minimum of cutScale (on subject basis defined using by); you may use breaks instead to limit the data")
+  #   if (verbose) cat("Finished checking entry and exit. Time taken: ", timetaken(argTime), "\n")
+  if (verbose) cat("Finished evaluating entry and exit and checking args. Time taken: ", timetaken(argTime), "\n")
+  
+  ## create rows to fill gaps --------------------------------------------------
+  if (verbose) fillTime <- proc.time()
+  x2 <- copy(x)
+  x2[, (l$freezeScales) := NA]
+  x2 <- rbind(x2, unique(x2, by = c(l$by), fromLast = TRUE))
+  
+  l$delta <- makeTempVarName(x2, pre = "delta_")
+  x2[, (l$delta) := c(get(l$en)[1], get(l$cutScale)[-c(1,.N)], max(get(l$cutScale)+lex.dur)) - get(l$cutScale), by = c(l$by)]
+  x2[, c(linkScales) := lapply(mget(linkScales), function(x) x + get(l$delta)), by = c(l$by)]
+  
+  setcolsnull(x2, l$delta)
+  
+  l$order <- makeTempVarName(x, pre = "order_")
+  x[, (l$order) := (1:.N)*2, by = c(l$by)]
+  x2[, (l$order) := (1:.N)*2-1, by = c(l$by)]
+  
+  x <- rbindlist(list(x, x2))
+  rm(x2)
+  setkeyv(x, c(l$by, l$cutScale, l$order))
+  setkeyv(x, c(l$by))
+  set(x, j = l$order, value = as.integer(x[[l$order]]))
+  x[, (l$order) := 1:.N, by = c(l$by)]
+  
+  if (verbose) cat("Finished expanding data to accommodate filling gaps. Time taken: ", timetaken(fillTime), "\n")
+  ## handle time scale values --------------------------------------------------
+  if (verbose) valueTime <- proc.time()
+  
+  l$CSE <- makeTempVarName(x, pre = paste0(l$cutScale, "_end_"))
+  l$LCS <- makeTempVarName(x, pre = paste0("lead1_",l$cutScale, "_"))
+  x[, (l$CSE) := lex.dur + get(l$cutScale)]
+  x[, (l$LCS)  := shift(get(l$cutScale), n = 1L, type = c("lead"), fill = NA), by = c(l$by)]
+  
+  
+  x[!duplicated(x, fromLast = TRUE, by = key(x)), c(l$LCS, l$CSE) := get(l$ex)]
+  x[, (l$CSE) := pmin(get(l$LCS), get(l$CSE))]
+  x[, (l$cutScale) := sort(c(get(l$en)[1L],shift(get(l$CSE), n = 1L, type = "lag", fill = NA)[-1])), by = c(l$by)]
+  x[, lex.dur := get(l$CSE) - get(l$cutScale)]
+  
+  ## bring up other than frozen and cut scales to bear -------------------------
+  x[, (othScales) := lapply(mget(othScales), function(x) {min(x) + c(0, cumsum(lex.dur)[-.N])}), by = c(l$by)]
+  
+  
+  ## frozen scales should make sense cumulatively ------------------------------
+  ## indicates frozenness: 0 = not frozen, 1 = frozen
+  l$frz <- makeTempVarName(x, pre = "frozen_")
+  x[, (l$frz) := 0L]
+  frozens <- x[,is.na(get(l$freezeScales[1]))]
+  x[frozens, (l$frz) := 1L]
+  
+  
+  ## alternate method: just use lex.durs and only cumulate in non-frozen rows
+  x[, (l$freezeScales) := lapply(mget(l$freezeScales), function(x) {
+    x <- max(0, min(x-lex.dur, na.rm=TRUE))
+    x <- x + c(0, as.double(cumsum(as.integer(!get(l$frz))*lex.dur))[-.N])
+  }), by = c(l$by)]
+  
+  x <- x[lex.dur > .Machine$double.eps^0.5, ]
+  
+  if (verbose) cat("Finished computing correct values for time scales. Time taken: ", timetaken(valueTime), "\n")
+  
+  ## splitting separately ------------------------------------------------------
+  if (!is.null(breaks)) {
+    if (verbose) splitTime <- proc.time()
+    x_frozen <- x[get(l$frz) == 1L,]
+    x <- x[get(l$frz) == 0L]
+    forceLexisDT(x, allScales = allScales, breaks = oldBreaks)
+    
+    ## NOTE: since we only split by the frozen time scales by pretending
+    ## they are NOT Lexis time scales (temporarily), and since one should 
+    ## pass the appropriate breaks info of pre-existing breaks to splitMulti,
+    ## choose only breaks for non-frozen time scales to include in x_frozen's
+    ## attributes here.
+    ## (e.g. when work history is no longer accumulating)
+    frzBreaks <- breaks[l$liquidScales]
+    oldFrzBreaks <- oldBreaks[l$liquidScales]
+    emptyFrzBreaks <- vector("list", length = length(l$liquidScales))
+    names(emptyFrzBreaks) <- l$liquidScales
+    if (length(oldFrzBreaks)) {
+      emptyFrzBreaks[names(oldFrzBreaks)] <- oldFrzBreaks
+    }
+    forceLexisDT(x_frozen, allScales = l$liquidScales, breaks = emptyFrzBreaks)
+    
+    if (length(frzBreaks) > 0) {
+      ## do (also) split for all time scales where also the frozen
+      ## time scales are split. This is allowed for times where the
+      ## frozen time scales have not been frozen
+      ## (e.g. work history is accumulating)
+      x_frozen <- splitMulti(x_frozen, breaks = frzBreaks, ...)
+    }
+    
+    ## do (also) split where also split
+    x <- splitMulti(x, breaks = breaks, ...)
+    breaks <- attr(x, "breaks") ## new breaks appended by splitMulti
+    
+    setDT(x)
+    setDT(x_frozen)
+    x <- rbindlist(list(x, x_frozen), use.names = TRUE); rm(x_frozen)
+    forceLexisDT(x, breaks = breaks, allScales = allScales)
+    if (verbose) cat("Finished splitting data. Time taken: ", timetaken(splitTime), "\n")
+  }
+  
+  ## final touch ---------------------------------------------------------------
+  
+  setDT(x)
+  if (is.character(freezeDummy)) setnames(x, l$frz, freezeDummy)
+  setkeyv(x, c(l$by, l$order))
+  delCols <- setdiff(names(l), c("by", "cutScale", "freezeScales", 
+                                 "liquidScales",
+                                 "linkScales", "allScales", "othScales"))
+  delCols <- unlist(l[delCols])
+  setcolsnull(x, keep = names(lex), colorder = TRUE)
+  
+  setattr(x, "time.scales", allScales)
+  setattr(x, "breaks", breaks)
+  setattr(x, "time.since", rep("", length(allScales)))
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  if (!return_DT()) setDFpe(x)
+  
+  if (verbose) cat("Finished prepExpo run. Time taken: ", timetaken(allTime), "\n")
+  
+  x[]
+}
+
+
+
+doComparisonWithEpi <- function(lexDT, lexDTdrop, lexDF, breaks) {
+  BL <- NULL
+  if (!is.list(breaks)) stop("breaks needs to be a list")
+  requireNamespace("Epi")
+  requireNamespace("testthat")
+  
+  allScales <- attr(lexDF, "time.scales")
+  sc1 <- allScales[1]
+  setDT(lexDT)
+  setDT(lexDTdrop)
+  setDT(lexDF)
+  setkeyv(lexDT, c("lex.id", sc1))
+  setkeyv(lexDTdrop, c("lex.id", sc1))
+  setkeyv(lexDF, c("lex.id", sc1))
+  
+  testthat::expect_equal(attr(lexDT, "time.scales"), attr(lexDF, "time.scales"))
+  testthat::expect_equal(attr(lexDT, "time.since"), attr(lexDF, "time.since"))
+  
+  testthat::expect_equal(attr(lexDTdrop, "time.scales"), attr(lexDF, "time.scales"))
+  testthat::expect_equal(attr(lexDTdrop, "time.since"), attr(lexDF, "time.since"))
+  
+  doTestBarrage(dt1 = lexDT, dt2 = lexDF, allScales = allScales)
+  rm(lexDT)
+  
+  lexDF <- intelliDrop(x = lexDF, breaks = breaks)
+  
+  doTestBarrage(dt1 = lexDTdrop, dt2 = lexDF, allScales = allScales)
+  
+}
+
+doTestBarrage <- function(dt1, dt2, allScales, testTimes = TRUE, testStatuses = TRUE) {
+  requireNamespace("Epi")
+  requireNamespace("testthat")
+  
+  lex.id <- lex.dur <- NULL ## APPEASE R CMD CHECK
+  
+  testthat::expect_equal(sum(dt1$lex.dur), 
+                         sum(dt2$lex.dur), 
+                         check.attributes = FALSE)
+  testthat::expect_equal(dt1[, sum(lex.dur), keyby = lex.id]$V1, 
+                         dt2[, sum(lex.dur), keyby = lex.id]$V1, 
+                         check.attributes = FALSE)
+  
+  all_names_present(dt1, allScales)
+  all_names_present(dt2, allScales)
+  
+  if (testTimes) {
+    for (k in allScales) {
+      testthat::expect_equal(dt1[[k]], dt2[[k]], 
+                             check.attributes = TRUE)
+    }
+  }
+  
+  if (testStatuses) {
+    testthat::expect_equal(dt1$lex.Cst, dt2$lex.Cst, check.attributes = FALSE)
+    testthat::expect_equal(dt1$lex.Xst, dt2$lex.Xst, check.attributes = FALSE)
+    
+    testthat::expect_equal(levels(dt1$lex.Cst), levels(dt2$lex.Cst), check.attributes = FALSE)
+    testthat::expect_equal(levels(dt1$lex.Xst), levels(dt2$lex.Xst), check.attributes = FALSE)
+    
+    testthat::expect_true(all(class(dt2$lex.Cst) %in% class(dt1$lex.Cst)))
+    testthat::expect_true(all(class(dt2$lex.Xst) %in% class(dt1$lex.Xst)))
+  }
+  
+  invisible(NULL)
+}
+
+compareSLDTWithEpi <- function(data, breaks, timeScale) {
+  requireNamespace("Epi")
+  requireNamespace("testthat")
+  
+  if (!inherits(data, "Lexis")) stop("data gotta be a Lexis object broseph")
+  
+  lexDT <- splitLexisDT(data, breaks = breaks, timeScale = timeScale, merge = TRUE, drop = FALSE)
+  lexDTdrop <- splitLexisDT(data, breaks = breaks, timeScale = timeScale, merge = TRUE, drop = TRUE)
+  lexDF <- splitLexis(data, breaks = breaks, time.scale = timeScale) ## without dropping
+  ## this treatment done in splitLexisDT (difftime -> integer -> double)
+  harmonizeNumericTimeScales(lexDF, times = c(Epi::timeScales(lexDF), "lex.dur"))
+  
+  BL <- list(breaks)
+  setattr(BL, "names", timeScale)
+  
+  doComparisonWithEpi(lexDT = lexDT, lexDTdrop = lexDTdrop, lexDF = lexDF, breaks = BL)
+  
+  invisible(NULL)
+}
+
+
+
+
+
+splitMultiEpi <- function(data, breaks = list(fot = 0:5), drop) {
+  
+  for (k in names(breaks)) {
+    data <- splitLexis(data, breaks = breaks[[k]], time.scale = k)
+  }
+  
+  forceLexisDT(
+    data, breaks = attr(data, "breaks"), 
+    allScales = attr(data, "time.scales"), 
+    key = FALSE
+  )
+  if (drop) data <- intelliDrop(data, breaks = breaks)
+  data
+}
+
+
+
+
+
+compareSMWithEpi <- function(data, breaks = list(fot=0:5)) {
+  requireNamespace("Epi")
+  requireNamespace("testthat")
+  
+  lexDT <- splitMulti(data, breaks = breaks, merge = TRUE, drop = FALSE)
+  lexDTdrop <- splitMulti(data, breaks = breaks, merge = TRUE, drop = TRUE)
+  lexDF <- splitMultiEpi(data, breaks = breaks, drop = FALSE)
+  
+  doComparisonWithEpi(lexDT=lexDT, lexDTdrop = lexDTdrop, lexDF=lexDF, breaks = breaks)
+  
+  invisible(NULL)
+}
+
+
+
+
+
+summarize_Lexis <- function(x) {
+  
+  lex.Cst <- lex.Xst <- NULL ## appease R CMD CHECK
+  
+  dur <- sum(x$lex.dur)
+  status_vars <- paste0("lex.", c("Cst", "Xst"))
+  time_scales <- copy(attr(x, "time.scales"))
+  dt <- copy(setDT(
+    mget(c(status_vars, "lex.id", time_scales[1]), as.environment(x))
+  ))
+  setkeyv(dt, c("lex.id", time_scales[1]))
+  dt <- unique(dt, by = c("lex.id"), fromLast = TRUE)
+  
+  n <- dt[, .N, keyby = status_vars]
+  rm("dt")
+  n[, "transition" := paste0(lex.Cst, "->", lex.Xst)]
+  n <- cast_simple(data = n, columns = "transition", values = "N")
+  return(cbind(lex.dur = dur, n))
+}
+
+
+
+
+
+roll_lexis_status_inplace <- function(unsplit.data, split.data, id.var) {
+  
+  ## R CMD CHECK appeasement
+  lex.Cst <- lex.Xst <- NULL
+  
+  stopifnot(
+    is.data.table(split.data),
+    length(key(split.data)) > 1,
+    key(split.data)[1] == id.var,
+    key(split.data)[2] %in% attr(unsplit.data, "time.scales"),
+    id.var %in% names(unsplit.data),
+    id.var %in% names(split.data),
+    uniqueN(unsplit.data[[id.var]]) == nrow(unsplit.data)
+  )
+  
+  status_vars <- c("lex.Cst", "lex.Xst")
+  status_ud <- mget_cols(c(id.var, status_vars), unsplit.data)
+  
+  join <- structure(list(split.data[[id.var]]), names = id.var)
+  lex_cst <- status_ud[
+    i = join, 
+    j = lex.Cst, 
+    on = id.var
+    ]
+  storage.mode(lex_cst) <- storage.mode(split.data[["lex.Cst"]])
+  set(split.data, j = status_vars, value = list(lex_cst, lex_cst))
+  
+  wh_last_row <- which(!duplicated(split.data, by = id.var, fromLast = TRUE))
+  join <- structure(list(split.data[[id.var]][wh_last_row]), names = id.var)
+  last_lex_xst <- status_ud[
+    i = join, 
+    j = lex.Xst, 
+    on = id.var
+    ]
+  storage.mode(last_lex_xst) <- storage.mode(split.data[["lex.Xst"]])
+  set(split.data, i = wh_last_row, j = "lex.Xst", value = last_lex_xst)
+  
+  
+  NULL
+}
+
+
+
+
+random_splitting_on <- function(
+  lex,
+  n.max.breaks = 20
+) {
+  stopifnot(
+    inherits(lex, "Lexis")
+  )
+  
+  ts_nms <- attr(lex, "time.scales")
+  brks <- attr(lex, "breaks")
+  timesince <- attr(lex, "time.since")
+  lex_vars <- c(paste0("lex.", c("id", "Cst", "Xst", "dur")), ts_nms)
+  non_lex_vars <- setdiff(names(lex), lex_vars)
+  lex <- mget_cols(lex_vars, lex)
+  setattr(lex, "time.scales", ts_nms)
+  setattr(lex, "breaks", brks)
+  setattr(lex, "time.since", timesince)
+  setattr(lex, "class", c("Lexis", "data.table", "data.frame"))
+  checkLexisData(lex)
+  
+  n_split_ts <- sample(seq_along(ts_nms), 1)
+  split_ts_nms <- sample(ts_nms, size = n_split_ts)
+  
+  do_drop <- sample(list(FALSE, TRUE), size = 1)[[1]]
+  
+  bl <- lapply(split_ts_nms, function(split_ts_nm) {
+    r <- c(min(lex[[split_ts_nm]]), max(lex[[split_ts_nm]] + lex[["lex.dur"]]))
+    d <- diff(r)
+    
+    br_r <- if (do_drop) 2:n.max.breaks else 1:n.max.breaks
+    n_br <- sample(br_r, 1)
+    
+    ## allow breaks outside observed data, but at least one break must be
+    ## not outside range of values in data
+    extrema <- r + c(-1,1)*d*0.05
+    l <- rep(extrema[1], n_br)
+    u <- rep(extrema[2], n_br)
+    u[1] <- l[1] <- mean(r)
+    sort(unique(runif(min = l, max = u, n = n_br)))
+  })
+  names(bl) <- split_ts_nms
+  
+  es <- ps  <- lex
+  for (ts_nm in split_ts_nms) {
+    es <- Epi::splitLexis(es, breaks = bl[[ts_nm]], time.scale = ts_nm)
+    forceLexisDT(es, breaks = attr(es, "breaks"), allScales = ts_nms,
+                 key = FALSE)
+    if (do_drop) {
+      es <- intelliDrop(es, breaks = bl[ts_nm])
+    }
+    ps <- splitLexisDT(ps, breaks = bl[[ts_nm]], timeScale = ts_nm, 
+                       drop = do_drop)
+  }
+  
+  psm <- splitMulti(lex, breaks = bl, drop = do_drop)
+  
+  list(es = es, ps = ps, psm = psm)
+}
+
+
+
+
+
+random_Lexis <- function(
+  n.rows = c(100, 1000, 2000), 
+  n.time.scales = 1:10,
+  n.statuses = 2:10,
+  n.other.vars = 1
+) {
+  
+  row_n <- sample(as.list(n.rows), 1)[[1]]
+  
+  ts_n <- sample(as.list(n.time.scales), 1)[[1]]
+  
+  st_n <- sample(as.list(n.statuses), 1)[[1]]
+  
+  dt <- setDT(lapply(1:ts_n, function(i) {
+    runif(min = 0, max = 1000, n = row_n)
+  }))
+  ts_nms <- paste0("lex_ts_", formatC(seq_len(ncol(dt)), flag = "0", width = 3))
+  setnames(dt, names(dt), ts_nms)
+  
+  dt[, "lex.Cst" := sample(1:st_n, size = .N, replace = TRUE)]
+  dt[, "lex.Xst" := sample(1:st_n, size = .N, replace = TRUE)]
+  dt[, "lex.id" := sample(1:.N, .N, replace = FALSE)]
+  dt[, "lex.dur" := runif(n = .N, min = 0, max = 10)]
+  
+  oth_n <- sample(as.list(n.other.vars), 1)[[1]]
+  lapply(seq_len(oth_n), function(i) {
+    set(
+      dt, j = makeTempVarName(names = names(dt), pre = "nonlexvar_"), 
+      value = sample(1:100, size = nrow(dt), replace = TRUE)
+    )
+  })
+  
+  brks <- lapply(ts_nms, function(nm) NULL)
+  names(brks) <- ts_nms
+  
+  forceLexisDT(dt, breaks = brks, allScales = ts_nms, key = TRUE)
+  checkLexisData(dt, check.breaks = TRUE)
+  dt[]
+}
+
+
+
+
+
+random_splitting_on_random_data <- function(
+  n.datasets = 100, 
+  n.rows = 1000,
+  n.time.scales = 1:10,
+  n.breaks = 10:100,
+  n.statuses = 1:5,
+  n.other.vars = 1
+) {
+  
+  neql <- vector("list", n.datasets)
+  
+  for (i in 1:n.datasets) {
+    
+    
+    set.seed(get_random_seed())
+    
+    drop <- sample(list(TRUE, FALSE), 1)[[1]]
+    drop <- FALSE
+    
+    dt <- random_Lexis(
+      n.rows = n.rows,
+      n.time.scales = n.time.scales,
+      n.statuses = n.statuses,
+      n.other.vars = n.other.vars
+    )
+    
+    ts_names <- copy(attr(dt, "time.scales"))
+    ts_n <- length(ts_names)
+    
+    dt_bl <- lapply(ts_names, function(x) NULL)
+    names(dt_bl) <- ts_names
+    forceLexisDT(dt, breaks = dt_bl, allScales = ts_names)
+    
+    br_n <- unlist(sample(as.list(n.breaks), ts_n))
+    names(br_n) <- ts_names
+    BL <- lapply(ts_names, function(ts_name) {
+      runif(n = br_n[[ts_name]], min = -100, max = 100)
+    })
+    names(BL) <- ts_names
+    
+    BL <- BL[sample(ts_names, ts_n)]
+    
+    split_pop <- splitMulti(dt, breaks = BL, drop = drop)
+    split_epi <- splitMultiEpi(dt, breaks = BL, drop = drop)
+    
+    setkeyv(split_epi, c("lex.id", ts_names[1]))
+    setkeyv(split_pop, c("lex.id", ts_names[1]))
+    
+    summary_epi <- summarize_Lexis(split_epi)
+    summary_pop <- summarize_Lexis(split_pop)
+    
+    eq <- all.equal(summary_pop, summary_epi, check.attributes = FALSE)
+    if (!isTRUE(eq)) {
+      message("split_epi, split_pop not equal in tick ", i, "")
+      neql[[i]] <- mget(c(
+        "drop", "row_n", "ts_n", "ts_names", 
+        "dt", "dt_bl", "br_n", "BL",
+        "split_epi", "split_pop", "eq",
+        "summary_epi", "summary_pop",
+        "used_seed"
+      ))
+    }
+  }
+  
+  neql[vapply(neql, is.null, logical(1))]
+  neql
+}
+
+
+
+
+
+do_split <- function(x, ts, all.ts, breaks, drop = TRUE, merge = TRUE) {
+  
+  ## unfinished v2 splitlexisDT work horse
+  stopifnot(
+    is.integer(x[["lex.id"]])
+  )
+  
+  id_dt <- data.table(
+    "orig" = x[["lex.id"]],
+    "temp" = 1:nrow(x)
+  )
+  set(x, j = "lex.id", value = id_dt[["temp"]])
+  
+  split <- mget_cols(c(ts, "lex.id", "lex.dur"), x)
+  
+  BL <- structure(list(breaks), names = ts)
+  if (drop) {
+    split <- intelliCrop(split, breaks = BL, allScales = all.ts, 
+                         cropStatuses = TRUE)
+    split <- intelliDrop(x = split, breaks = BL, 
+                         check = FALSE, dropNegDur = TRUE)
+  }
+  n_subjects <- nrow(split)
+  ts_values <- split[[ts]]
+  
+  split <- rbindlist(lapply(1:length(breaks), function(i) get("split")))
+  
+  tmp_ie_nm <-  makeTempVarName(names = names(x), pre = "do_split_tmp_ie_")
+  
+  set(split, j = tmp_ie_nm,  value = rep(breaks, each = n_subjects))
+  set(split, j = tmp_ie_nm,  value = {
+    pmin(split[[tmp_ie_nm]], split[[ts]] + split[["lex.dur"]])
+  })
+  set(split, j = ts, value = c(
+    ts_values, 
+    pmax(ts_values, rep(breaks[-length(breaks)], each = n_subjects))
+  ))
+  
+}
+
+
+
+
+
diff --git a/R/startup_message.R b/R/startup_message.R
index d3a096a..4211350 100644
--- a/R/startup_message.R
+++ b/R/startup_message.R
@@ -1,43 +1,43 @@
-
-
-
-
-
-.onAttach <- function(...) {
-  
-  if (interactive()) {
-    msg <- paste0("Using popEpi. See ?popEpi for info and ",
-                  "news(package = \"popEpi\") for news.")
-    packageStartupMessage(msg)
-  }
-  
-  using_r_devel <- grepl(pattern = "devel", x = base::R.version$status)
-  if (using_r_devel) {
-    ## memory leak problem in data.table 1.11.2 in R-devel (3.6.0 atm)
-    requireNamespace("data.table")
-    data.table::setDTthreads(threads = 1L)
-  }
-  invisible(NULL)
-}
-
-
-
-
-
-.onLoad <- function(...) {
-  opt <- getOption("popEpi.datatable")
-  if (!is.null(opt) && is.logical(opt) && !isTRUE(opt)) {
-    warning("Option 'popEpi.datatable' was set to TRUE when loading popEpi.", 
-            call. = FALSE)
-  }
-  options("popEpi.datatable" = TRUE)
-  invisible(NULL)
-}
-
-
-
-
-
-
-
-
+
+
+
+
+
+.onAttach <- function(...) {
+  
+  if (interactive()) {
+    msg <- paste0("Using popEpi. See ?popEpi for info and ",
+                  "news(package = \"popEpi\") for news.")
+    packageStartupMessage(msg)
+  }
+  
+  using_r_devel <- grepl(pattern = "devel", x = base::R.version$status)
+  if (using_r_devel) {
+    ## memory leak problem in data.table 1.11.2 in R-devel (3.6.0 atm)
+    requireNamespace("data.table")
+    data.table::setDTthreads(threads = 1L)
+  }
+  invisible(NULL)
+}
+
+
+
+
+
+.onLoad <- function(...) {
+  opt <- getOption("popEpi.datatable")
+  if (!is.null(opt) && is.logical(opt) && !isTRUE(opt)) {
+    warning("Option 'popEpi.datatable' was set to TRUE when loading popEpi.", 
+            call. = FALSE)
+  }
+  options("popEpi.datatable" = TRUE)
+  invisible(NULL)
+}
+
+
+
+
+
+
+
+
diff --git a/R/survival_aggregated.R b/R/survival_aggregated.R
index 604d939..bfc52ba 100644
--- a/R/survival_aggregated.R
+++ b/R/survival_aggregated.R
@@ -1,898 +1,898 @@
-#' @template survival_doc_template
-#' @param formula a \code{formula}; the response 
-#' must be the time scale to compute survival time function estimates
-#' over, e.g. \code{fot ~ sex}. Variables on the right-hand side of the formula
-#' separated by \code{+} are considered stratifying variables, for which 
-#' estimates are computed separately. May contain usage of \code{adjust()} 
-#' --- see Details and Examples.
-#' @param data since popEpi 0.4.0, a \code{data.frame}
-#' containing variables used in \code{formula} and other arguments.
-#' \code{aggre} objects are recommended as they contain information on any
-#' time scales and are therefore safer; for creating \code{aggre} objects see
-#' \code{\link{as.aggre}} when your data is already aggregated and \code{aggre}
-#' for aggregating split \code{Lexis} objects.
-#' 
-#' @param surv.breaks a vector of breaks on the 
-#' survival time scale. Optional if \code{data} is an \code{aggre} object
-#' and mandatory otherwise. Must define each intended interval;
-#' e.g. \code{surv.breaks = 0:5} when data has intervals defined by 
-#' breaks \code{seq(0, 5, 1/12)} will aggregate to wider intervals first.
-#' It is generally recommended (and sufficient; 
-#' see Seppa, Dyban and Hakulinen (2015)) to use monthly
-#' intervals where applicable.
-#' 
-#' @param n variable containing counts of subjects at-risk at the start of a 
-#' time interval; e.g. \code{n = "at.risk"}. 
-#' Required when \code{surv.method = "lifetable"}.
-#' \link[=flexible_argument]{Flexible input}.
-#' 
-#' @param d variable(s) containing counts of subjects experiencing an event. 
-#' With only one type of event, e.g. \code{d = "deaths"}. With multiple types of 
-#' events (for CIF or cause-specific survival estimation), supply e.g.
-#' \code{d = c("canD", "othD")}. If the survival time function to be estimated
-#' does not use multiple types of events, supplying more than one variable
-#' to \code{d} simply causes the variables to be added together. 
-#' Always required. \link[=flexible_argument]{Flexible input}.
-#' 
-#' @param n.cens variable containing counts of subjects censored during a 
-#' survival time interval; E.g. \code{n.cens = "alive"}.
-#' Required when \code{surv.method = "lifetable"}. 
-#' \link[=flexible_argument]{Flexible input}.
-
-#' @param pyrs variable containing total subject-time accumulated within a 
-#' survival time interval; E.g. \code{pyrs = "pyrs"}. 
-#' Required when \code{surv.method = "hazard"}. Flexible input.
-
-#' @param d.exp variable denoting total "expected numbers of events" 
-#' (typically computed \code{pyrs * pop.haz}, where 
-#' \code{pop.haz} is the expected hazard level) 
-#' accumulated within a survival time interval; E.g. \code{pyrs = "pyrs"}.
-#' Required when computing EdererII relative survivals or 
-#' CIFs based on excess counts of events. Flexible input.
-
-#' @param n.pp variable containing total Pohar-Perme weighted counts of
-#' subjects at risk in an interval,
-#' supplied as argument \code{n} is supplied. 
-#' Computed originally on the subject
-#' level as analogous to \code{pp * as.integer(status == "at-risk")}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-#' 
-#' @param d.pp variable(s) containing Pohar-Perme weighted counts of events,
-#' supplied as argument \code{d} is supplied. Computed originally on the subject
-#' level as analogous to \code{pp * as.integer(status == some_event)}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-
-#' @param d.pp.2 variable(s) containing total Pohar-Perme 
-#' "double-weighted" counts of events,
-#' supplied as argument \code{d} is supplied. Computed originally on the subject
-#' level as analogous to \code{pp * pp * as.integer(status == some_event)}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-
-#' @param n.cens.pp variable containing total Pohar-Perme weighted counts 
-#' censorings,
-#' supplied as argument \code{n.cens} is supplied. 
-#' Computed originally on the subject
-#' level as analogous to \code{pp * as.integer(status == "censored")}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-
-#' @param pyrs.pp variable containing total Pohar-Perme weighted subject-times,
-#' supplied as argument \code{pyrs} is supplied. 
-#' Computed originally on the subject
-#' level as analogous to \code{pp * pyrs}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-
-#' @param d.exp.pp variable containing total Pohar-Perme weighted counts 
-#' of excess events,
-#' supplied as argument \code{pyrs} is supplied. 
-#' Computed originally on the subject
-#' level as analogous to \code{pp * d.exp}.
-#' Required when \code{relsurv.method = "pp"}. Flexible input.
-#' 
-#' 
-#' @section Data requirements:
-#' 
-#' \code{survtab_ag} computes estimates of survival time functions using 
-#' pre-aggregated data. For using subject-level data directly, use 
-#' \code{\link{survtab}}. For aggregating data, see \code{\link{lexpand}}
-#' and \code{\link{aggre}}. 
-#' 
-#' By default, and if data is an \code{aggre} object (not mandatory), 
-#' \code{survtab_ag} makes use of the exact same breaks that were used in 
-#' splitting the original data (with e.g. \code{lexpand}), so it is not 
-#' necessary to specify any \code{surv.breaks}. If specified, the 
-#' \code{surv.breaks} must be a subset of the pertinent 
-#' pre-existing breaks. When data is not an \code{aggre} object, breaks
-#' must always be specified. Interval lengths (\code{delta} in output) are 
-#' also calculated based on whichever breaks are used, 
-#' so the upper limit of the breaks should
-#' therefore be meaningful and never e.g. \code{Inf}. 
-#' 
-#' 
-#' @examples
-#' ## see more examples with explanations in vignette("survtab_examples")
-#' 
-#' #### survtab_ag usage
-#' 
-#' data("sire", package = "popEpi")
-#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
-#' ## note: sire is a simulated cohort integrated into popEpi.
-#' BL <- list(fot=seq(0, 5, by = 1/12),
-#'            per = c("2008-01-01", "2013-01-01"))
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-#'              status = status %in% 1:2,
-#'              breaks = BL,
-#'              pophaz = popmort,
-#'              aggre = list(fot))
-#'              
-#' ## calculate relative EdererII period method
-#' ## NOTE: x is an aggre object here, so surv.breaks are deduced
-#' ## automatically
-#' st <- survtab_ag(fot ~ 1, data = x)
-#' 
-#' summary(st, t = 1:5) ## annual estimates
-#' summary(st, q = list(r.e2 = 0.75)) ## 1st interval where r.e2 < 0.75 at end
-#' \donttest{
-#' plot(st)
-#' 
-#' 
-#' ## non-aggre data: first call to survtab_ag would fail
-#' df <- data.frame(x)
-#' # st <- survtab_ag(fot ~ 1, data = x)
-#' st <- survtab_ag(fot ~ 1, data = x, surv.breaks = BL$fot)
-#' 
-#' ## calculate age-standardised 5-year relative survival ratio using 
-#' ## Ederer II method and period approach 
-#' 
-#' sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
-#' BL <- list(fot=seq(0, 5, by = 1/12),
-#'            per = c("2008-01-01", "2013-01-01"))
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-#'              status = status %in% 1:2,
-#'              breaks = BL,
-#'              pophaz = popmort,
-#'              aggre = list(agegr, fot))
-#' 
-#' ## age standardisation using internal weights (age distribution of 
-#' ## patients diagnosed within the period window)
-#' ## (NOTE: what is done here is equivalent to using weights = "internal")
-#' w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0], FUN = sum)
-#' names(w) <- c("agegr", "weights")
-#' 
-#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
-#' plot(st, y = "r.e2.as", col = c("blue"))
-#' 
-#' ## age standardisation using ICSS1 weights
-#' data(ICSS)
-#' cut <- c(0, 45, 55, 65, 75, Inf)
-#' agegr <- cut(ICSS$age, cut, right = FALSE)
-#' w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-#' names(w) <- c("agegr", "weights")
-#'
-#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
-#' lines(st, y = "r.e2.as", col = c("red"))
-#' 
-#' 
-#' ## cause-specific survival
-#' sire$stat <- factor(sire$status, 0:2, c("alive", "canD", "othD"))
-#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-#'              status = stat,
-#'              breaks = BL,
-#'              pophaz = popmort,
-#'              aggre = list(agegr, fot))
-#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w,
-#'                  d = c("fromalivetocanD", "fromalivetoothD"),
-#'                  surv.type = "surv.cause")
-#' plot(st, y = "surv.obs.fromalivetocanD.as")
-#' lines(st, y = "surv.obs.fromalivetoothD.as", col = "red")
-#' 
-#' 
-#' }
-#' @export
-survtab_ag <- function(formula = NULL,
-                       
-                       data, 
-                       
-                       adjust = NULL,
-                       weights = NULL,
-                       
-                       surv.breaks = NULL, 
-                       
-                       n = "at.risk",
-                       d = "from0to1",
-                       n.cens = "from0to0",
-                       pyrs = "pyrs",
-                       d.exp = "d.exp",
-                       
-                       n.pp = NULL,
-                       d.pp = "d.pp",
-                       d.pp.2 = "d.pp.2",
-                       n.cens.pp = "n.cens.pp",
-                       pyrs.pp = "pyrs.pp",
-                       d.exp.pp = "d.exp.pp",
-                       
-                       surv.type="surv.rel", 
-                       surv.method="hazard", 
-                       relsurv.method="e2",  
-                       
-                       subset = NULL,
-                       
-                       conf.level = 0.95, 
-                       conf.type = "log-log",
-                       
-                       verbose=FALSE) {
-  
-  if (verbose) starttime <- proc.time()
-  
-  Tstop <- delta <- Tstart <- surv.int <- n.eff <- n.eff.pp <- surv.obs <- 
-     lag1_surv.obs <- p.obs <- CIF.rel <- NULL ## APPEASE R CMD CHECK
-  
-  TF <- environment()
-  PF <- parent.frame(1L)
-  
-  this_call <- match.call()
-  used_args <- as.list(this_call)[-1L]
-  fl <- formals("survtab_ag")
-  used_args <- c(used_args, fl[!names(fl) %in% names(used_args)])
-  used_args <- used_args[names(fl)]
-  rm(fl)
-  
-  attrs <- copy(attributes(data))
-  
-  # check data -----------------------------------------------------------------
-  if (missing(data) || nrow(data) == 0) stop("data missing or has no rows")
-  
-  # check arguments ------------------------------------------------------------
-  
-  surv.type <- match.arg(surv.type, c("surv.obs","surv.rel","surv.cause", "cif.obs", "cif.rel"))
-  surv.method <- match.arg(surv.method, c("lifetable","hazard"))
-  relsurv.method <- match.arg(relsurv.method, c("e2", "pp", "EdererII", "Pohar-Perme", "pohar-perme", "edererII", "ederer2"))
-  if (relsurv.method %in% c("EdererII", "edererII", "ederer2")) relsurv.method <- "e2"
-  if (relsurv.method %in% c("Pohar-Perme", "pohar-perme")) relsurv.method <- "pp"
-  relsurv.method <- match.arg(relsurv.method, c("e2", "pp"))
-  conf.type <- match.arg(conf.type, c("log","log-log","plain"))
-  
-  
-  ## argument 'formula' pre-check ----------------------------------------------
-  if (!(inherits(formula, "formula") && length(formula) == 3L)) {
-    stop("Argument 'formula' does not appear to be a two-sided formula. ",
-         "Usage: e.g. fot ~ sex")
-  }
-  surv.scale <- deparse(formula[[2]])
-  if (!surv.scale %in% names(data)) {
-    stop("Left-hand-side of formula must be a column in data; e.g. ",
-         "fot ~ sex, where 'fot' is the name of a column in data.")
-  }
-  
-  ## check breaks --------------------------------------------------------------
-  
-  surv.breaks <- select_breaks(data = data, ts = surv.scale, br = surv.breaks)
-  surv.breaks <- sort(unique(surv.breaks))
-  # if (!breaks_in_data(surv.breaks, surv.scale, data)) {
-  #   stop("Used breaks do not all appear to exist in data. Make sure the ",
-  #        "breaks match to the values that your time scale variable has in the ",
-  #        "data.")
-  # }
-  
-  # data prep & subsetting -----------------------------------------------------
-  subset <- substitute(subset)
-  subset <- evalLogicalSubset(data, subset)
-  
-  origData <- data
-  
-  data <- data[subset, ]
-  setDT(data)
-  
-  # handle count etc. variables ------------------------------------------------
-  
-  valVars <- c("d")
-  valVars <- c(valVars, if (surv.method == "hazard") "pyrs" else c("n", "n.cens"))
-  
-  valVars <- c(valVars, if (surv.type == "surv.rel" && relsurv.method == "e2")  "d.exp" else NULL)
-  
-  valVars <- c(valVars, if (surv.type == "cif.rel")  "d.exp" else NULL)
-  
-  ppVars <- c("d.pp", "d.exp.pp", "d.pp.2", 
-              if (surv.method == "hazard") "pyrs.pp" else c("n.cens.pp", "n.pp"))
-  valVars <- c(valVars, if (surv.type == "surv.rel" && relsurv.method == "pp") ppVars else NULL)
-  
-  fo <- formals("survtab_ag")
-  mc <- as.list(match.call())[-1]
-  mc <- c(mc, fo[!names(fo) %in% names(mc)])
-  
-  mc <- mc[valVars]
-  
-  
-  mc <- lapply(mc, function(elem) {
-    evalPopArg(data = data, arg = elem, DT = TRUE, enclos = PF, recursive = TRUE)
-    })
-  
-  
-  ## if given as multiple vars, combine these into one (e.g. n  = c("v1", "v2"))
-  combValVars <- c("n", "n.cens", "d.exp", "d.pp", "d.exp.pp", "d.pp.2",
-                   "pyrs", "pyrs.pp", "n.cens.pp", "n.pp")
-  combValVars <- intersect(names(mc), combValVars)
-  mc[combValVars] <- lapply(combValVars, function(val_var) {
-    tab <- mc[[val_var]]
-    if (!is.data.frame(tab) || length(tab) == 1L) return(tab)
-    tab_na <- names(tab)
-    e <- paste0(tab_na, collapse = " + ")
-    e <- parse(text = e)
-    tab <- data.table(V1 = tab[, eval(e)])
-    setnames(tab, "V1", val_var)
-    tab
-  })
-  
-  ## NOTE: this does not delete but sets the value to NULL.
-  mc[unlist(lapply(mc, function(x) {
-    NROW(x) == 0L || is.null(x) || is.language(x) || inherits(x, "try-error")
-    }))] <- NULL
-  
-  lackVars <- setdiff(valVars, names(mc[!unlist(lapply(mc, is.null))]))
-  if (length(lackVars) > 0) {
-    stop("Following arguments were NULL or could not be evaluated but are ",
-         "required: ", paste0("'", lackVars, "'", collapse = ", "), ". ",
-         "Usual suspects: arguments are NULL or refer to variables that ",
-         "cannot be found in data.")
-  }
-  
-  eventVars <- NULL
-  mc[[1]] <- data.table(mc[[1L]]) ## this avoids an exotic error in set().
-  nl <- lapply(mc, names)
-  for (k in 1:length(mc)) {
-    jay <- argName <- names(mc[k])
-    cn <- names(mc[[k]])
-    
-    if (length(cn) > 1) jay <- paste0(jay, ".", cn) ## e.g. d.1, d.2, ...
-    if (argName %in% c("d")) {
-      eventVars <- jay
-      if (surv.type %in% c("surv.cause") && length(cn) == 1L) {
-        stop("surv.type = 'surv.cause', but only one type of event supplied ",
-             "via argument 'd'. If you want to compute cause-specific ",
-             "survivals, please supply multiple types of events via ",
-             "'d'; otherwise use surv.type = 'surv.obs'") 
-      } else  if (length(cn) > 1 && !argName %in% c("d","d.pp", "d.pp.2", "n.pp")) {
-        stop("'", argName, "' has/evaluates to ", length(cn), 
-             " columns; only 'd', 'd.pp', and 'd'pp.2', 'n.pp' may evaluate ",
-             "to more than one column of the value arguments")
-      }
-      
-    }
-    
-    setnames(mc[[k]], cn, jay)
-    set(mc[[1]], j = jay, value = mc[[k]])
-    nl[[argName]] <- jay
-  }
-  mc <- mc[[1]]
-  
-  if (!is.null(eventVars)) {
-    set(mc, j = "d", value = rowSums(mc[, mget(eventVars)]))
-    valVars <- unique(c(valVars, "d", eventVars))
-  }
-  
-  
-  all_names_present(mc, valVars, 
-                    msg = paste0("Expected internal temp data to have ",
-                                 "variables %%VARS%% at this point, but didn't",
-                                 ". This is most likely a bug and should be ",
-                                 "reported to pkg maintainer."))
-  setcolorder(mc, valVars)
-  
-  ## addition: internal weights use n at beginning of first interval
-  
-  if (is.character(weights)) {
-    checkWeights(weights)
-    if (!"n" %in% valVars) {
-      n <- substitute(n)
-      mc$n <- evalPopArg(n, data = data, enclos = PF)
-      
-      valVars <- unique(c(valVars, "n"))
-      
-      if (is.null(mc$n)) {
-        
-        stop("Requested internal weights to be computed and used to standardize ", 
-             "estimates, but argument 'n' not supplied. This is currently ",
-             "required for computing internal weights (the values of 'n' ", 
-             "in the first interval will be used for this). Please supply 'n' ",
-             "or supply hand-made weights (preferred for your clarity).")
-      }
-    } 
-    
-    data[, c("n") := mc$n]
-    
-  }
-  
-  
-  # making weighted table of aggregated values ---------------------------------
-  ## NOTE: at-risk counts require special treatment when surv.breaks
-  ## are a subset of the available breaks: cannot sum at-risk figures!
-  ## instead should simply pick the value at the start of the
-  ## (now larger) interval. Will accomplish this by setting values not
-  ## at the start of an interval to zero and summing anyway.
-  if (surv.method == "lifetable") {
-    wh_internal <- list(surv.breaks)
-    names(wh_internal) <- surv.scale
-    wh_internal <- data[wh_internal, on = eval(surv.scale), which = TRUE]
-    wh_internal <- setdiff(1:nrow(data), wh_internal)
-    mc[wh_internal, intersect(c("n", "n.pp"), names(mc)) := 0L]
-  }
-  
-  ## NOTE: while ssSub will pass the whole column of e.g. fot values, which will
-  ## not limit the data to e.g. up 5 years of follow-up if original data went 
-  ## further, surv.breaks may be only up to 5 years and will limit the data
-  ## in makeWeightsDT using a CJ-merge-trick appropriately (via custom.levels).
-  bl <- list(surv.breaks)
-  setattr(bl, "names", surv.scale)
-  
-  adjust <- evalPopArg(data, adjust, enclos = PF, naming = "model")
-  
-  iws <- NULL
-  if (is.character(weights) && pmatch(weights, c("internal", "cohort"), 0)) {
-    if (!"n" %in% names(data)) {
-      stop("Need 'n' specified for when using internal weights: Internal ",
-           "weights are computed as the counts of subjects at the start of ",
-           "follow-up.")
-    }
-    iws <- makeTempVarName(data, pre = "internal_weights_")
-    data[, c(iws) := 0.0]
-    data[data[[surv.scale]] == surv.breaks[1], c(iws) := n]
-  }
-  
-  data <- makeWeightsDT(data = data, values = list(mc), enclos = PF,
-                        print = NULL, formula = formula, adjust = adjust, 
-                        by.other = surv.scale, Surv.response = FALSE,
-                        custom.levels = bl, weights = weights,
-                        internal.weights.values = iws,
-                        custom.levels.cut.low = surv.scale)
-  
-  allVars <- attr(data, "makeWeightsDT")
-  allVars[] <- lapply(allVars, function(x) if (length(x) == 0L) NULL else x)
-  prVars <- allVars$prVars
-  adVars <- allVars$adVars
-  # boVars <- allVars$boVars ## this is surv.scale
-  valVars <- allVars$vaVars
-  
-  ## to avoid e.g. 'factor(sex, 1:2)' going bonkers
-  prVars_orig <- prVars
-  if (length(prVars) > 0L) {
-    prVars <- makeTempVarName(names = c(names(data), adVars), 
-                              pre = paste0("print_", 1:length(prVars)))
-  }
-  adVars_orig <- adVars
-  if (length(adVars) > 0L) {
-    adVars <- makeTempVarName(names = c(names(data), prVars), 
-                              pre = paste0("print_", 1:length(adVars)))
-  }
-  if (length(c(prVars, adVars))) setnames(data, c(prVars_orig, adVars_orig), c(prVars, adVars))
-  byVars <- c(prVars, adVars)
-  
-  # formulate some needed variables --------------------------------------------
-  setkeyv(data, c(byVars, surv.scale))
-  data[, "Tstop" := rep(surv.breaks[-1],length.out=.N)]
-  setnames(data, surv.scale, "Tstart")
-  data[, "delta" := Tstop - Tstart]
-  data[, "surv.int" := 1:.N, by = eval(byVars)]
-  setcolorder(data, c(byVars, "surv.int", "Tstart", "Tstop", "delta", valVars, intersect(names(data), "weights")))
-  
-  if (surv.method == "lifetable") {
-    testEvents <- data[, n - shift(n, n = 1, type = "lead", fill = NA), by = eval(byVars)]$V1
-    testEvents <- data$n.cens + data$d - testEvents
-    
-    if (sum(abs(testEvents), na.rm = TRUE)) {
-      on.exit({
-        
-        data[, "n.cens + d - (n-lead1_n)" := testEvents]
-        wh <- testEvents != 0L
-        wh <- wh & !is.na(wh)
-        if (interactive()) {
-          printSD <- c(byVars, "Tstop", "d", "n", "n.cens", 
-                       "n.cens + d - (n-lead1_n)")
-          print(data[wh, .SD, .SDcols = printSD], top = 5, nrow = 10)
-          
-        }
-        
-      }, add = TRUE)
-      
-      stop("Supplied n.cens and d do not sum to total number of events and ",
-           "censorings based on n alone. Note that lifetable analysis ",
-           "is currently not supported for period analysis (or other ",
-           "comparable limitations of data).",
-           if (interactive())" See table below and check your variables.")
-    }
-    rm(testEvents)
-    data[, "n.eff" := n - n.cens/2L]
-  }
-  
-  
-  # compute observed survivals  ------------------------------------------------
-  if (verbose) ostime <- proc.time()
-  
-  if (surv.method=="lifetable") {
-    comp.st.surv.obs.lif(surv.table = data, surv.by.vars = byVars)
-  }
-  if (surv.method=="hazard") {
-    comp.st.surv.obs.haz(surv.table = data, surv.by.vars = byVars)
-  }
-  
-  data <- comp.st.conf.ints(data, al=1-conf.level, surv="surv.obs", transform = conf.type)
-  
-  if (verbose) {
-    message("* popEpi::survtab_ag: computed observed survival estimates; ",
-            data.table::timetaken(ostime))
-  }
-  
-  ## empty surv.int checking ---------------------------------------------------
-  testVar <- if (surv.method == "lifetable") "n" else "pyrs"
-  ## sum over adjusting variables
-  data <- test_empty_surv_ints(data, by = c(prVars, adVars), 
-                               show.by = c(prVars_orig, adVars_orig),
-                               sum.over = adVars,
-                               test.var = testVar)
-  
-  ## sum over nothing
-  if (length(adVars) > 0L) {
-    data <- test_empty_surv_ints(data, by = c(prVars, adVars), 
-                                 show.by = c(prVars_orig, adVars_orig),
-                                 sum.over = NULL, test.var = testVar)
-  }
-  
-  ## if adjusting, crop all estimates by adjusting variables
-  ## to shortest estimate
-  if (length(adVars)) {
-    adLe <- data[, list(min = min(surv.int), max = max(surv.int)), keyby = eval(adVars)]
-    adLe <- c(max(adLe$min), min(adLe$max))
-    data <- data[surv.int %in% `:`(adLe[1L], adLe[2L])]
-  }
-  
-  # create and print table of bad surv.ints ------------------------------------
-  
-  badObsSurv <- data$surv.obs == 0 | is.na(data$surv.obs)
-  if (sum(badObsSurv)) {
-    
-    zerotab <- data[badObsSurv, 
-                    list(first.bad.surv.int = min(as.integer(surv.int)), 
-                         last.bad.surv.int = max(as.integer(surv.int)), 
-                         surv.obs=min(surv.obs)), keyby = eval(byVars)]
-    
-    
-    message("* popEpi::survtab_ag: Some cumulative surv.obs were zero or NA:")
-    if (length(byVars)) setnames(zerotab, c(prVars, adVars), c(prVars_orig, adVars_orig))
-    print(zerotab)
-    if (surv.method == "lifetable" && data[surv.obs == 0, .N] > 0) {
-      message("* popEpi::survtab_ag: NOTE: Zero surv.obs leads to zero ",
-              "relative survivals as well. Adjusting with weights WILL use ",
-              "the zero surv.obs / relative survival values.")
-    }
-    
-  }
-  rm(badObsSurv)
-  
-  # compute cause-specific survivals  ------------------------------------------
-  if (surv.type == "surv.cause") {
-    
-    ## NOTE: these related to adjusting life-table estimates for delayed entry...
-    #       data[, "n.eff" := n - n.cens/2 + n.de/2 + n.de.cens/4] # + d.de/2
-    #       "n.cens_1" := n.cens + (d-d_1)
-    #       "n.de.cens" := n.de.cens + (d.de - d.de_1)
-    
-    if (surv.method == "lifetable") {
-      for (k in eventVars) {
-        k <- gsub(pattern = "d_", replacement = "", x = k)
-        d_k <- paste0("d_", k)
-        # d.de_k <- paste0("d.de_",k)
-        
-        n.eff_k <- paste0("n.eff_",k)
-        
-        ## old: " := n - (n.cens + (d-", d_k,")/2 + n.de/2 + (n.de.cens + d.de - ", d.de_k,")/4 )"
-        # expr <- paste0(n.eff_k, " := n - (n.cens + (d-", d_k,")/2 )")
-        
-        set(data, j = c(n.eff_k), value = data$n.eff + (data$d - data[[d_k]])/2L ) # + d.de/2
-        # data[,  eval(parse(text = expr), envir = .SD)]
-        
-      }
-      
-    }
-    
-    surv_names <- names(data)[grep("surv.obs", names(data))]
-    surv_names <- c("d", if (surv.method == "lifetable") "n.eff" else NULL, surv_names)
-    setnames(data, surv_names, paste0(surv_names, ".orig"))
-    
-    for (k in eventVars) {
-      
-      k <- gsub(pattern = "d.", replacement = "", x = k)
-      setnames(data, paste0("d.",k), "d")
-      
-      if (surv.method=="hazard") {
-        comp.st.surv.obs.haz(surv.table = data, surv.by.vars = byVars)
-      } else {
-        setnames(data, paste0("n.eff_", k), "n.eff")
-        comp.st.surv.obs.lif(surv.table = data, surv.by.vars = byVars)
-      }
-      os.table <- comp.st.conf.ints(data, al=1-conf.level, surv="surv.obs", transform = conf.type)
-      
-      new_surv_names <- setdiff(surv_names, c("d", if (surv.method == "lifetable") "n.eff" else NULL))
-      new_surv_names <- gsub("surv.obs", paste0("surv.obs.", k), new_surv_names)
-      new_surv_names <- c(paste0(c("d.", if (surv.method == "lifetable") "n.eff." else NULL), k), new_surv_names)
-      setnames(data, surv_names, new_surv_names)
-      
-      
-    }
-    setnames(data, paste0(surv_names, ".orig"), surv_names)
-  }
-  
-  # compute cause-specific/excess-case CIFs ------------------------------------
-  if (surv.type %in% c("cif.obs", "cif.rel")) {
-    
-    data[, "lag1_surv.obs" := shift(surv.obs, n = 1L, type = "lag", fill = 1), by = eval(byVars)]
-    data[, "p.obs" := surv.obs/lag1_surv.obs]
-    
-    if (surv.type == "cif.obs") {
-      for (k in eventVars) {
-        
-        k <- gsub("d.", "", x = k)
-        d.k <- paste0("d.", k)
-        
-        d.var <- paste0("d.",k)
-        q.var <- paste0("q.", k)
-        CIF_var <- paste0("CIF_", k)
-        data[, (q.var)   := (1-p.obs)*get(d.var)/d]
-        data[get(d.var) == 0L | d == 0L, (q.var) := 0]
-        data[, (CIF_var) := cumsum(lag1_surv.obs*get(q.var)), by = eval(byVars)]
-      }
-    }
-    
-    if (surv.type == "cif.rel") {
-      ## assuming d.exp in data
-      data[, "CIF.rel" := (1-p.obs)*(d-d.exp)/d]
-      data[d.exp>d, "CIF.rel" := NA]
-      data[, "CIF.rel" := cumsum(lag1_surv.obs*CIF.rel), by = eval(byVars)]
-    }
-    
-    ## SEs currently not known for CIFs; impute 0 to make adjusting work
-    CIF_vars <- names(data)[substr(names(data),1,3) == "CIF"]
-    data[, c(paste0("SE.", CIF_vars)) := 0L]
-    
-    setcolsnull(data, c("lag1_surv.obs", "p.obs", paste0("q.", substr(eventVars, 3, nchar(eventVars)))))
-    
-  }
-  
-  
-  # relative survivals ---------------------------------------------------------
-  if (surv.type == "surv.rel" & relsurv.method == "e2") {
-    
-    # compute r.e2 -------------------------------------------------------------
-    comp.st.rs <- function(rs.table, rs.by.vars = byVars) {
-      
-      p.exp <- delta <- surv.exp <- surv.obs <- n.eff.pp <- 
-        surv.obs <- NULL ## APPEASE R CMD CHECK
-      ## EdererII
-      
-      ##-------------
-      if (surv.method == "hazard") {
-        rs.table[, "p.exp" := exp(-delta*d.exp/pyrs)] 
-        rs.table[, "surv.exp" := cumprod(p.exp), by = eval(rs.by.vars)]
-        comp.st.r.e2.haz(surv.table = rs.table, surv.by.vars = rs.by.vars)
-      } else {
-        rs.table[, "p.exp" := 1 - d.exp/n]
-        rs.table[, "surv.exp" := cumprod(p.exp), by = eval(rs.by.vars)]
-        
-        if (rs.table[, min(surv.obs, na.rm=TRUE) == 0]) {
-          rs.table[surv.obs == 0, "surv.exp" := 1]
-        }
-        
-        comp.st.r.e2.lif(surv.table = rs.table, surv.by.vars = rs.by.vars)
-        
-        if (rs.table[, min(surv.obs, na.rm=TRUE) == 0]) {
-          rs.table[surv.obs == 0, intersect(c("surv.exp","r.e2","SE.r.e2","r.e2.lo","r.e2.hi"), names(rs.table)) := 0]
-        }
-      }
-      
-      ## ------------
-      
-      rs.table <- comp.st.conf.ints(rs.table, al=1-conf.level, surv="r.e2", transform = conf.type)
-      
-      return(rs.table)
-    }
-    
-    data <- comp.st.rs(rs.table = data)
-    
-    
-  }
-  
-  # compute r.pp ---------------------------------------------------------------
-  if (surv.type == "surv.rel" & relsurv.method == "pp") {
-    
-    all_names_present(data, c("d.pp", "d.exp.pp", "d.pp.2"))
-    ## pohar perme: analysis weighted by expected cumulative survival
-    comp.st.pp <- function(pp.table, by.vars = byVars) {
-      ## relative survival
-      if (surv.method == "hazard") {
-        all_names_present(data, c("pyrs.pp"),
-                          msg = paste0("internal error: work data did not have",
-                                       " variable named pyrs.pp. Complain ",
-                                       "to package maintainer if you see this."))
-        comp.st.r.pp.haz(surv.table = pp.table, surv.by.vars = by.vars)
-      } else {
-        data[, "n.eff.pp" := n.pp - 0.5*n.cens.pp]
-        all_names_present(data, c("n.pp", "n.cens.pp", "n.eff.pp"),
-                          msg = paste0("internal error: work data did not have",
-                                       " variable named n.eff.pp. Complain ",
-                                       "to package maintainer if you see this."))
-        comp.st.r.pp.lif(surv.table = pp.table, surv.by.vars = by.vars)
-        
-        if (pp.table[, min(surv.obs, na.rm=TRUE) == 0]) {
-          pp.table[surv.obs == 0, intersect(c("r.pp","SE.r.pp","r.pp.lo","r.pp.hi"), names(pp.table)) := 0]
-        }
-      }
-      
-      pp.table <- comp.st.conf.ints(pp.table, al=1-conf.level, surv="r.pp", transform = conf.type )
-      
-      return(pp.table)
-    }
-    data <- comp.st.pp(pp.table = data)
-  }
-  
-  # compute adjusted estimates -------------------------------------------------
-  if ("weights" %in% names(data)) {
-    w_est_vars <- names(data)[substr(names(data), 1, 8) == "surv.obs"]
-    w_est_vars <- c(w_est_vars, "r.e2", "r.pp")
-    w_est_vars <- c(w_est_vars,  names(data)[substr(names(data),1,3)=="CIF"])
-    w_est_vars <- intersect(w_est_vars, names(data))
-    w_est_vars <- w_est_vars[unlist(lapply(w_est_vars, function(x) {
-      !substr(x, nchar(x)-2L, nchar(x)) %in% c(".lo", ".hi")
-      }))]
-    w_se_vars <- paste0("SE.", w_est_vars)
-    
-    w_est <- data[, lapply(.SD, function(x) sum(x*weights)), 
-                   keyby = c(prVars, "surv.int"), .SDcols = w_est_vars]
-    w_se <- data[, lapply(.SD, function(x) sqrt(sum((x^2)*(weights^2)))),
-                   keyby = c(prVars, "surv.int"), .SDcols = w_se_vars]
-    
-    data <- data[, lapply(mget(valVars), sum), 
-                 keyby = c(prVars, "surv.int", "Tstart", "Tstop", "delta")]
-    set(data, j = w_se_vars, value = mget(w_se_vars, as.environment(w_se)))
-    set(data, j = w_est_vars, value = mget(w_est_vars, as.environment(w_est)))
-    
-    setnames(data, old = c(w_est_vars, w_se_vars), 
-             new = paste0(c(w_est_vars, w_se_vars), ".as"))
-    
-    for (var in paste0(w_est_vars, ".as")) {
-      data <- comp.st.conf.ints(data, al = 1-conf.level, 
-                                surv = var, transform = conf.type)
-    }
-    
-    
-  }
-  
-  # clean-up -------------------------------------------------------------------
-  ## back to original names of print / adjust (used to avoid e.g. 
-  ## 'factor(V1, 1:2)' going bonkers in data.table)
-  if (length(c(prVars))) setnames(data, c(prVars), c(prVars_orig))
-  prVars <- prVars_orig
-  adVars <- adVars_orig
-  
-  ## reorder table, format numeric values, etc.
-  
-  miscVars <- intersect(names(data), c("surv.int", "Tstart", "Tstop", "delta"))
-  
-  survVars <- c("surv.obs.lo","surv.obs","surv.obs.hi","SE.surv.obs",
-                "r.e2.lo","r.e2","r.e2.hi","SE.r.e2",
-                "r.pp.lo","r.pp","r.pp.hi","SE.r.pp",
-                paste0("CIF.rel.", c("lo", "", "hi")), "SE.CIF.rel",
-                "surv.obs.as.lo","surv.obs.as","surv.obs.as.hi","SE.surv.obs.as",
-                "r.e2.as.lo","r.e2.as","r.e2.as.hi","SE.r.e2.as",
-                "r.pp.as.lo","r.pp.as","r.pp.as.hi","SE.r.pp.as",
-                paste0("CIF.rel.as.", c("lo", "", "hi")), "SE.CIF.rel.as"
-  )
-  survVars <- intersect(survVars, names(data))
-  
-  ## which variables are estimates, SEs, CIs, etc.
-  survVars.ca <- setdiff(names(data), c(prVars, valVars, miscVars, survVars))
-  CIF_vars <- survVars.ca[substr(survVars.ca, 1,3)=="CIF" | substr(survVars.ca, 1,6)=="SE.CIF"]
-  survVars <- c(survVars, CIF_vars)
-  
-  surv.obs.vars <- survVars.ca[substr(survVars.ca, 1,8) == "surv.obs" | substr(survVars.ca, 1,11) == "SE.surv.obs"]
-  survVars <- c(survVars, surv.obs.vars)
-  
-  survVars <- unique(intersect(survVars, names(data)))
-  
-  ## remove some unuseful variables
-  setcolsnull(data, c("SE.A", "SE.B"))
-  setcolsnull(data, survVars[substr(survVars, 1, 6) == "SE.CIF"]) ## since they are zero for now
-  survVars <- intersect(survVars, names(data))
-  
-  SEVars <- survVars[substr(survVars, 1, 3) == "SE."]
-  CIVars <- survVars[substr(survVars, nchar(survVars) - 2L, nchar(survVars)) %in% c(".lo", ".hi")]
-  estVars <- setdiff(survVars, c(SEVars, CIVars))
-  
-  order <- unique(c(prVars, miscVars, valVars, survVars))
-  order <- intersect(order, names(data))
-  
-  setcolsnull(data, setdiff(names(data), order))
-  setcolorder(data,order)
-  
-  setkeyv(data, c(prVars, "surv.int"))
-  
-  # attributes -----------------------------------------------------------------
-  setkeyv(data, c(prVars, "surv.int"))
-  setattr(data, "class", c("survtab", "data.table", "data.frame"))
-  if (!return_DT()) setDFpe(data)
-  if (length(prVars) == 0) prVars <- NULL ## might be character(0) 
-  
-  used_args$data <- origData
-  used_args$formula <- formula
-  used_args$weights <- evalRecursive(arg = weights, env = PF)$weights
-  
-  arglist <- list(call = this_call, 
-                  arguments = used_args,
-                  surv.scale = surv.scale,
-                  surv.breaks = surv.breaks,
-                  print.vars = prVars,
-                  adjust.vars = adVars,
-                  value.vars = valVars,
-                  misc.vars = miscVars,
-                  surv.vars = survVars,
-                  est.vars = estVars,
-                  SE.vars = SEVars,
-                  CI.vars = CIVars)
-  varsArgs <- substr(names(arglist), nchar(names(arglist))-4L, nchar(names(arglist))) == ".vars"
-  varsArgs <- names(arglist)[varsArgs]
-  arglist[varsArgs] <- lapply(arglist[varsArgs], function(x) if (length(x) == 0L) NULL else x)
-                  
-  setattr(data, "survtab.meta", arglist)
-  
-  if (verbose) {
-    message("* popEpi::survtab_ag: finished whole process; ",
-            data.table::timetaken(starttime))
-  }
-  data[]
-}
-
-
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
-#               aggre = list(sex, fot), fot = seq(0, 5, 1/12))
-# ag[, d.exp := pmax(0L, from0to1 - 3L)]
-# st <- survtab_ag(ag, surv.type = "surv.obs", surv.method = "hazard")
-# st <- survtab_ag(ag, surv.type = "surv.cause", surv.method = "hazard", d = list(a = from0to1-3, b = 3))
-
-# sire <- copy(sire)
-# sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
-#               aggre = list(sex, agegr = cut(dg_age, c(0,60,70,80, Inf), labels = FALSE), fot), 
-#               fot = seq(0, 5, 1/12))
-# ag <- lexpand(sire, birth = "bi_date", entry = "bi_date", exit = "ex_date",
-#               status = status %in% 1:2,
-#               aggre = list(sex, age), 
-#               age = seq(0, 100, 1))
-# wdt <- data.table(agegr = 1:4, weights = c(0.2, 0.4, 0.3, 0.1))
-# wli <- list(agegr = c(0.2, 0.4, 0.3, 0.1))
-# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, surv.type = "surv.obs", surv.method = "hazard", weights = wli)
-# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, surv.type = "surv.rel", 
-#                  d.pp = "from0to1.pp", d.pp.2 = "from0to1.pp.2", 
-#                  d.exp.pp = "d.exp.pp", pyrs.pp = "ptime.pp",
-#                  surv.method = "hazard", weights = wli,
-#                  relsurv.method = "pp")
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status, pophaz = popmort, pp = TRUE,
-#               aggre = list(sex, agegr = cut(dg_age, c(0,60,70,80, Inf), labels = FALSE), fot), 
-#               fot = seq(0, 5, 1/12))
-# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, 
-#                  d = list(cand = from0to1, othd = from0to2),
-#                  surv.type = "surv.cause", weights = wli)
-# st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs", surv.method = "hazard", adjust = "agegr", weights = wli)
-# st <- survtab_ag(fot ~ adjust(agegr), data = ag, surv.type = "surv.obs", weights = wli)
-# st <- survtab_ag(fot ~ 1, data = ag, adjust = "agegr", surv.type = "surv.obs", weights = wli)
-# st <- survtab_ag(fot ~ 1, data = ag, adjust = "agegr", surv.type = "surv.obs", weights = wli)
-# st <- survtab_ag(fot ~ 1, data = ag, surv.type = "surv.obs")
-
-# wli2 <- wli
-# wli$sex <- c(0.4, 0.6)
-# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.obs", weights = wli)
-# st <- survtab_ag(fot ~ adjust(agegr), data = ag, surv.type = "surv.obs", weights = wli["agegr"])
-# ag[, d.exp := pmax(from0to1 - 1, 0L)]
-# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.rel", weights = wli)
-# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.cause", weights = wli)
-# ag[, othd := pmax(from0to1 - 1L, 0L)]
-# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, d = list(cand = from0to1, othd = pmax(from0to1-1L, 0L)), surv.type = "surv.cause", weights = wli)
-
+#' @template survival_doc_template
+#' @param formula a \code{formula}; the response 
+#' must be the time scale to compute survival time function estimates
+#' over, e.g. \code{fot ~ sex}. Variables on the right-hand side of the formula
+#' separated by \code{+} are considered stratifying variables, for which 
+#' estimates are computed separately. May contain usage of \code{adjust()} 
+#' --- see Details and Examples.
+#' @param data since popEpi 0.4.0, a \code{data.frame}
+#' containing variables used in \code{formula} and other arguments.
+#' \code{aggre} objects are recommended as they contain information on any
+#' time scales and are therefore safer; for creating \code{aggre} objects see
+#' \code{\link{as.aggre}} when your data is already aggregated and \code{aggre}
+#' for aggregating split \code{Lexis} objects.
+#' 
+#' @param surv.breaks a vector of breaks on the 
+#' survival time scale. Optional if \code{data} is an \code{aggre} object
+#' and mandatory otherwise. Must define each intended interval;
+#' e.g. \code{surv.breaks = 0:5} when data has intervals defined by 
+#' breaks \code{seq(0, 5, 1/12)} will aggregate to wider intervals first.
+#' It is generally recommended (and sufficient; 
+#' see Seppa, Dyban and Hakulinen (2015)) to use monthly
+#' intervals where applicable.
+#' 
+#' @param n variable containing counts of subjects at-risk at the start of a 
+#' time interval; e.g. \code{n = "at.risk"}. 
+#' Required when \code{surv.method = "lifetable"}.
+#' \link[=flexible_argument]{Flexible input}.
+#' 
+#' @param d variable(s) containing counts of subjects experiencing an event. 
+#' With only one type of event, e.g. \code{d = "deaths"}. With multiple types of 
+#' events (for CIF or cause-specific survival estimation), supply e.g.
+#' \code{d = c("canD", "othD")}. If the survival time function to be estimated
+#' does not use multiple types of events, supplying more than one variable
+#' to \code{d} simply causes the variables to be added together. 
+#' Always required. \link[=flexible_argument]{Flexible input}.
+#' 
+#' @param n.cens variable containing counts of subjects censored during a 
+#' survival time interval; E.g. \code{n.cens = "alive"}.
+#' Required when \code{surv.method = "lifetable"}. 
+#' \link[=flexible_argument]{Flexible input}.
+
+#' @param pyrs variable containing total subject-time accumulated within a 
+#' survival time interval; E.g. \code{pyrs = "pyrs"}. 
+#' Required when \code{surv.method = "hazard"}. Flexible input.
+
+#' @param d.exp variable denoting total "expected numbers of events" 
+#' (typically computed \code{pyrs * pop.haz}, where 
+#' \code{pop.haz} is the expected hazard level) 
+#' accumulated within a survival time interval; E.g. \code{pyrs = "pyrs"}.
+#' Required when computing EdererII relative survivals or 
+#' CIFs based on excess counts of events. Flexible input.
+
+#' @param n.pp variable containing total Pohar-Perme weighted counts of
+#' subjects at risk in an interval,
+#' supplied as argument \code{n} is supplied. 
+#' Computed originally on the subject
+#' level as analogous to \code{pp * as.integer(status == "at-risk")}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+#' 
+#' @param d.pp variable(s) containing Pohar-Perme weighted counts of events,
+#' supplied as argument \code{d} is supplied. Computed originally on the subject
+#' level as analogous to \code{pp * as.integer(status == some_event)}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+
+#' @param d.pp.2 variable(s) containing total Pohar-Perme 
+#' "double-weighted" counts of events,
+#' supplied as argument \code{d} is supplied. Computed originally on the subject
+#' level as analogous to \code{pp * pp * as.integer(status == some_event)}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+
+#' @param n.cens.pp variable containing total Pohar-Perme weighted counts 
+#' censorings,
+#' supplied as argument \code{n.cens} is supplied. 
+#' Computed originally on the subject
+#' level as analogous to \code{pp * as.integer(status == "censored")}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+
+#' @param pyrs.pp variable containing total Pohar-Perme weighted subject-times,
+#' supplied as argument \code{pyrs} is supplied. 
+#' Computed originally on the subject
+#' level as analogous to \code{pp * pyrs}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+
+#' @param d.exp.pp variable containing total Pohar-Perme weighted counts 
+#' of excess events,
+#' supplied as argument \code{pyrs} is supplied. 
+#' Computed originally on the subject
+#' level as analogous to \code{pp * d.exp}.
+#' Required when \code{relsurv.method = "pp"}. Flexible input.
+#' 
+#' 
+#' @section Data requirements:
+#' 
+#' \code{survtab_ag} computes estimates of survival time functions using 
+#' pre-aggregated data. For using subject-level data directly, use 
+#' \code{\link{survtab}}. For aggregating data, see \code{\link{lexpand}}
+#' and \code{\link{aggre}}. 
+#' 
+#' By default, and if data is an \code{aggre} object (not mandatory), 
+#' \code{survtab_ag} makes use of the exact same breaks that were used in 
+#' splitting the original data (with e.g. \code{lexpand}), so it is not 
+#' necessary to specify any \code{surv.breaks}. If specified, the 
+#' \code{surv.breaks} must be a subset of the pertinent 
+#' pre-existing breaks. When data is not an \code{aggre} object, breaks
+#' must always be specified. Interval lengths (\code{delta} in output) are 
+#' also calculated based on whichever breaks are used, 
+#' so the upper limit of the breaks should
+#' therefore be meaningful and never e.g. \code{Inf}. 
+#' 
+#' 
+#' @examples
+#' ## see more examples with explanations in vignette("survtab_examples")
+#' 
+#' #### survtab_ag usage
+#' 
+#' data("sire", package = "popEpi")
+#' ## prepare data for e.g. 5-year "period analysis" for 2008-2012
+#' ## note: sire is a simulated cohort integrated into popEpi.
+#' BL <- list(fot=seq(0, 5, by = 1/12),
+#'            per = c("2008-01-01", "2013-01-01"))
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+#'              status = status %in% 1:2,
+#'              breaks = BL,
+#'              pophaz = popmort,
+#'              aggre = list(fot))
+#'              
+#' ## calculate relative EdererII period method
+#' ## NOTE: x is an aggre object here, so surv.breaks are deduced
+#' ## automatically
+#' st <- survtab_ag(fot ~ 1, data = x)
+#' 
+#' summary(st, t = 1:5) ## annual estimates
+#' summary(st, q = list(r.e2 = 0.75)) ## 1st interval where r.e2 < 0.75 at end
+#' \donttest{
+#' plot(st)
+#' 
+#' 
+#' ## non-aggre data: first call to survtab_ag would fail
+#' df <- data.frame(x)
+#' # st <- survtab_ag(fot ~ 1, data = x)
+#' st <- survtab_ag(fot ~ 1, data = x, surv.breaks = BL$fot)
+#' 
+#' ## calculate age-standardised 5-year relative survival ratio using 
+#' ## Ederer II method and period approach 
+#' 
+#' sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
+#' BL <- list(fot=seq(0, 5, by = 1/12),
+#'            per = c("2008-01-01", "2013-01-01"))
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+#'              status = status %in% 1:2,
+#'              breaks = BL,
+#'              pophaz = popmort,
+#'              aggre = list(agegr, fot))
+#' 
+#' ## age standardisation using internal weights (age distribution of 
+#' ## patients diagnosed within the period window)
+#' ## (NOTE: what is done here is equivalent to using weights = "internal")
+#' w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0], FUN = sum)
+#' names(w) <- c("agegr", "weights")
+#' 
+#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
+#' plot(st, y = "r.e2.as", col = c("blue"))
+#' 
+#' ## age standardisation using ICSS1 weights
+#' data(ICSS)
+#' cut <- c(0, 45, 55, 65, 75, Inf)
+#' agegr <- cut(ICSS$age, cut, right = FALSE)
+#' w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+#' names(w) <- c("agegr", "weights")
+#'
+#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
+#' lines(st, y = "r.e2.as", col = c("red"))
+#' 
+#' 
+#' ## cause-specific survival
+#' sire$stat <- factor(sire$status, 0:2, c("alive", "canD", "othD"))
+#' x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+#'              status = stat,
+#'              breaks = BL,
+#'              pophaz = popmort,
+#'              aggre = list(agegr, fot))
+#' st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w,
+#'                  d = c("fromalivetocanD", "fromalivetoothD"),
+#'                  surv.type = "surv.cause")
+#' plot(st, y = "surv.obs.fromalivetocanD.as")
+#' lines(st, y = "surv.obs.fromalivetoothD.as", col = "red")
+#' 
+#' 
+#' }
+#' @export
+survtab_ag <- function(formula = NULL,
+                       
+                       data, 
+                       
+                       adjust = NULL,
+                       weights = NULL,
+                       
+                       surv.breaks = NULL, 
+                       
+                       n = "at.risk",
+                       d = "from0to1",
+                       n.cens = "from0to0",
+                       pyrs = "pyrs",
+                       d.exp = "d.exp",
+                       
+                       n.pp = NULL,
+                       d.pp = "d.pp",
+                       d.pp.2 = "d.pp.2",
+                       n.cens.pp = "n.cens.pp",
+                       pyrs.pp = "pyrs.pp",
+                       d.exp.pp = "d.exp.pp",
+                       
+                       surv.type="surv.rel", 
+                       surv.method="hazard", 
+                       relsurv.method="e2",  
+                       
+                       subset = NULL,
+                       
+                       conf.level = 0.95, 
+                       conf.type = "log-log",
+                       
+                       verbose=FALSE) {
+  
+  if (verbose) starttime <- proc.time()
+  
+  Tstop <- delta <- Tstart <- surv.int <- n.eff <- n.eff.pp <- surv.obs <- 
+     lag1_surv.obs <- p.obs <- CIF.rel <- NULL ## APPEASE R CMD CHECK
+  
+  TF <- environment()
+  PF <- parent.frame(1L)
+  
+  this_call <- match.call()
+  used_args <- as.list(this_call)[-1L]
+  fl <- formals("survtab_ag")
+  used_args <- c(used_args, fl[!names(fl) %in% names(used_args)])
+  used_args <- used_args[names(fl)]
+  rm(fl)
+  
+  attrs <- copy(attributes(data))
+  
+  # check data -----------------------------------------------------------------
+  if (missing(data) || nrow(data) == 0) stop("data missing or has no rows")
+  
+  # check arguments ------------------------------------------------------------
+  
+  surv.type <- match.arg(surv.type, c("surv.obs","surv.rel","surv.cause", "cif.obs", "cif.rel"))
+  surv.method <- match.arg(surv.method, c("lifetable","hazard"))
+  relsurv.method <- match.arg(relsurv.method, c("e2", "pp", "EdererII", "Pohar-Perme", "pohar-perme", "edererII", "ederer2"))
+  if (relsurv.method %in% c("EdererII", "edererII", "ederer2")) relsurv.method <- "e2"
+  if (relsurv.method %in% c("Pohar-Perme", "pohar-perme")) relsurv.method <- "pp"
+  relsurv.method <- match.arg(relsurv.method, c("e2", "pp"))
+  conf.type <- match.arg(conf.type, c("log","log-log","plain"))
+  
+  
+  ## argument 'formula' pre-check ----------------------------------------------
+  if (!(inherits(formula, "formula") && length(formula) == 3L)) {
+    stop("Argument 'formula' does not appear to be a two-sided formula. ",
+         "Usage: e.g. fot ~ sex")
+  }
+  surv.scale <- deparse(formula[[2]])
+  if (!surv.scale %in% names(data)) {
+    stop("Left-hand-side of formula must be a column in data; e.g. ",
+         "fot ~ sex, where 'fot' is the name of a column in data.")
+  }
+  
+  ## check breaks --------------------------------------------------------------
+  
+  surv.breaks <- select_breaks(data = data, ts = surv.scale, br = surv.breaks)
+  surv.breaks <- sort(unique(surv.breaks))
+  # if (!breaks_in_data(surv.breaks, surv.scale, data)) {
+  #   stop("Used breaks do not all appear to exist in data. Make sure the ",
+  #        "breaks match to the values that your time scale variable has in the ",
+  #        "data.")
+  # }
+  
+  # data prep & subsetting -----------------------------------------------------
+  subset <- substitute(subset)
+  subset <- evalLogicalSubset(data, subset)
+  
+  origData <- data
+  
+  data <- data[subset, ]
+  setDT(data)
+  
+  # handle count etc. variables ------------------------------------------------
+  
+  valVars <- c("d")
+  valVars <- c(valVars, if (surv.method == "hazard") "pyrs" else c("n", "n.cens"))
+  
+  valVars <- c(valVars, if (surv.type == "surv.rel" && relsurv.method == "e2")  "d.exp" else NULL)
+  
+  valVars <- c(valVars, if (surv.type == "cif.rel")  "d.exp" else NULL)
+  
+  ppVars <- c("d.pp", "d.exp.pp", "d.pp.2", 
+              if (surv.method == "hazard") "pyrs.pp" else c("n.cens.pp", "n.pp"))
+  valVars <- c(valVars, if (surv.type == "surv.rel" && relsurv.method == "pp") ppVars else NULL)
+  
+  fo <- formals("survtab_ag")
+  mc <- as.list(match.call())[-1]
+  mc <- c(mc, fo[!names(fo) %in% names(mc)])
+  
+  mc <- mc[valVars]
+  
+  
+  mc <- lapply(mc, function(elem) {
+    evalPopArg(data = data, arg = elem, DT = TRUE, enclos = PF, recursive = TRUE)
+    })
+  
+  
+  ## if given as multiple vars, combine these into one (e.g. n  = c("v1", "v2"))
+  combValVars <- c("n", "n.cens", "d.exp", "d.pp", "d.exp.pp", "d.pp.2",
+                   "pyrs", "pyrs.pp", "n.cens.pp", "n.pp")
+  combValVars <- intersect(names(mc), combValVars)
+  mc[combValVars] <- lapply(combValVars, function(val_var) {
+    tab <- mc[[val_var]]
+    if (!is.data.frame(tab) || length(tab) == 1L) return(tab)
+    tab_na <- names(tab)
+    e <- paste0(tab_na, collapse = " + ")
+    e <- parse(text = e)
+    tab <- data.table(V1 = tab[, eval(e)])
+    setnames(tab, "V1", val_var)
+    tab
+  })
+  
+  ## NOTE: this does not delete but sets the value to NULL.
+  mc[unlist(lapply(mc, function(x) {
+    NROW(x) == 0L || is.null(x) || is.language(x) || inherits(x, "try-error")
+    }))] <- NULL
+  
+  lackVars <- setdiff(valVars, names(mc[!unlist(lapply(mc, is.null))]))
+  if (length(lackVars) > 0) {
+    stop("Following arguments were NULL or could not be evaluated but are ",
+         "required: ", paste0("'", lackVars, "'", collapse = ", "), ". ",
+         "Usual suspects: arguments are NULL or refer to variables that ",
+         "cannot be found in data.")
+  }
+  
+  eventVars <- NULL
+  mc[[1]] <- data.table(mc[[1L]]) ## this avoids an exotic error in set().
+  nl <- lapply(mc, names)
+  for (k in 1:length(mc)) {
+    jay <- argName <- names(mc[k])
+    cn <- names(mc[[k]])
+    
+    if (length(cn) > 1) jay <- paste0(jay, ".", cn) ## e.g. d.1, d.2, ...
+    if (argName %in% c("d")) {
+      eventVars <- jay
+      if (surv.type %in% c("surv.cause") && length(cn) == 1L) {
+        stop("surv.type = 'surv.cause', but only one type of event supplied ",
+             "via argument 'd'. If you want to compute cause-specific ",
+             "survivals, please supply multiple types of events via ",
+             "'d'; otherwise use surv.type = 'surv.obs'") 
+      } else  if (length(cn) > 1 && !argName %in% c("d","d.pp", "d.pp.2", "n.pp")) {
+        stop("'", argName, "' has/evaluates to ", length(cn), 
+             " columns; only 'd', 'd.pp', and 'd'pp.2', 'n.pp' may evaluate ",
+             "to more than one column of the value arguments")
+      }
+      
+    }
+    
+    setnames(mc[[k]], cn, jay)
+    set(mc[[1]], j = jay, value = mc[[k]])
+    nl[[argName]] <- jay
+  }
+  mc <- mc[[1]]
+  
+  if (!is.null(eventVars)) {
+    set(mc, j = "d", value = rowSums(mc[, mget(eventVars)]))
+    valVars <- unique(c(valVars, "d", eventVars))
+  }
+  
+  
+  all_names_present(mc, valVars, 
+                    msg = paste0("Expected internal temp data to have ",
+                                 "variables %%VARS%% at this point, but didn't",
+                                 ". This is most likely a bug and should be ",
+                                 "reported to pkg maintainer."))
+  setcolorder(mc, valVars)
+  
+  ## addition: internal weights use n at beginning of first interval
+  
+  if (is.character(weights)) {
+    checkWeights(weights)
+    if (!"n" %in% valVars) {
+      n <- substitute(n)
+      mc$n <- evalPopArg(n, data = data, enclos = PF)
+      
+      valVars <- unique(c(valVars, "n"))
+      
+      if (is.null(mc$n)) {
+        
+        stop("Requested internal weights to be computed and used to standardize ", 
+             "estimates, but argument 'n' not supplied. This is currently ",
+             "required for computing internal weights (the values of 'n' ", 
+             "in the first interval will be used for this). Please supply 'n' ",
+             "or supply hand-made weights (preferred for your clarity).")
+      }
+    } 
+    
+    data[, c("n") := mc$n]
+    
+  }
+  
+  
+  # making weighted table of aggregated values ---------------------------------
+  ## NOTE: at-risk counts require special treatment when surv.breaks
+  ## are a subset of the available breaks: cannot sum at-risk figures!
+  ## instead should simply pick the value at the start of the
+  ## (now larger) interval. Will accomplish this by setting values not
+  ## at the start of an interval to zero and summing anyway.
+  if (surv.method == "lifetable") {
+    wh_internal <- list(surv.breaks)
+    names(wh_internal) <- surv.scale
+    wh_internal <- data[wh_internal, on = eval(surv.scale), which = TRUE]
+    wh_internal <- setdiff(1:nrow(data), wh_internal)
+    mc[wh_internal, intersect(c("n", "n.pp"), names(mc)) := 0L]
+  }
+  
+  ## NOTE: while ssSub will pass the whole column of e.g. fot values, which will
+  ## not limit the data to e.g. up 5 years of follow-up if original data went 
+  ## further, surv.breaks may be only up to 5 years and will limit the data
+  ## in makeWeightsDT using a CJ-merge-trick appropriately (via custom.levels).
+  bl <- list(surv.breaks)
+  setattr(bl, "names", surv.scale)
+  
+  adjust <- evalPopArg(data, adjust, enclos = PF, naming = "model")
+  
+  iws <- NULL
+  if (is.character(weights) && pmatch(weights, c("internal", "cohort"), 0)) {
+    if (!"n" %in% names(data)) {
+      stop("Need 'n' specified for when using internal weights: Internal ",
+           "weights are computed as the counts of subjects at the start of ",
+           "follow-up.")
+    }
+    iws <- makeTempVarName(data, pre = "internal_weights_")
+    data[, c(iws) := 0.0]
+    data[data[[surv.scale]] == surv.breaks[1], c(iws) := n]
+  }
+  
+  data <- makeWeightsDT(data = data, values = list(mc), enclos = PF,
+                        print = NULL, formula = formula, adjust = adjust, 
+                        by.other = surv.scale, Surv.response = FALSE,
+                        custom.levels = bl, weights = weights,
+                        internal.weights.values = iws,
+                        custom.levels.cut.low = surv.scale)
+  
+  allVars <- attr(data, "makeWeightsDT")
+  allVars[] <- lapply(allVars, function(x) if (length(x) == 0L) NULL else x)
+  prVars <- allVars$prVars
+  adVars <- allVars$adVars
+  # boVars <- allVars$boVars ## this is surv.scale
+  valVars <- allVars$vaVars
+  
+  ## to avoid e.g. 'factor(sex, 1:2)' going bonkers
+  prVars_orig <- prVars
+  if (length(prVars) > 0L) {
+    prVars <- makeTempVarName(names = c(names(data), adVars), 
+                              pre = paste0("print_", 1:length(prVars)))
+  }
+  adVars_orig <- adVars
+  if (length(adVars) > 0L) {
+    adVars <- makeTempVarName(names = c(names(data), prVars), 
+                              pre = paste0("print_", 1:length(adVars)))
+  }
+  if (length(c(prVars, adVars))) setnames(data, c(prVars_orig, adVars_orig), c(prVars, adVars))
+  byVars <- c(prVars, adVars)
+  
+  # formulate some needed variables --------------------------------------------
+  setkeyv(data, c(byVars, surv.scale))
+  data[, "Tstop" := rep(surv.breaks[-1],length.out=.N)]
+  setnames(data, surv.scale, "Tstart")
+  data[, "delta" := Tstop - Tstart]
+  data[, "surv.int" := 1:.N, by = eval(byVars)]
+  setcolorder(data, c(byVars, "surv.int", "Tstart", "Tstop", "delta", valVars, intersect(names(data), "weights")))
+  
+  if (surv.method == "lifetable") {
+    testEvents <- data[, n - shift(n, n = 1, type = "lead", fill = NA), by = eval(byVars)]$V1
+    testEvents <- data$n.cens + data$d - testEvents
+    
+    if (sum(abs(testEvents), na.rm = TRUE)) {
+      on.exit({
+        
+        data[, "n.cens + d - (n-lead1_n)" := testEvents]
+        wh <- testEvents != 0L
+        wh <- wh & !is.na(wh)
+        if (interactive()) {
+          printSD <- c(byVars, "Tstop", "d", "n", "n.cens", 
+                       "n.cens + d - (n-lead1_n)")
+          print(data[wh, .SD, .SDcols = printSD], top = 5, nrow = 10)
+          
+        }
+        
+      }, add = TRUE)
+      
+      stop("Supplied n.cens and d do not sum to total number of events and ",
+           "censorings based on n alone. Note that lifetable analysis ",
+           "is currently not supported for period analysis (or other ",
+           "comparable limitations of data).",
+           if (interactive())" See table below and check your variables.")
+    }
+    rm(testEvents)
+    data[, "n.eff" := n - n.cens/2L]
+  }
+  
+  
+  # compute observed survivals  ------------------------------------------------
+  if (verbose) ostime <- proc.time()
+  
+  if (surv.method=="lifetable") {
+    comp.st.surv.obs.lif(surv.table = data, surv.by.vars = byVars)
+  }
+  if (surv.method=="hazard") {
+    comp.st.surv.obs.haz(surv.table = data, surv.by.vars = byVars)
+  }
+  
+  data <- comp.st.conf.ints(data, al=1-conf.level, surv="surv.obs", transform = conf.type)
+  
+  if (verbose) {
+    message("* popEpi::survtab_ag: computed observed survival estimates; ",
+            data.table::timetaken(ostime))
+  }
+  
+  ## empty surv.int checking ---------------------------------------------------
+  testVar <- if (surv.method == "lifetable") "n" else "pyrs"
+  ## sum over adjusting variables
+  data <- test_empty_surv_ints(data, by = c(prVars, adVars), 
+                               show.by = c(prVars_orig, adVars_orig),
+                               sum.over = adVars,
+                               test.var = testVar)
+  
+  ## sum over nothing
+  if (length(adVars) > 0L) {
+    data <- test_empty_surv_ints(data, by = c(prVars, adVars), 
+                                 show.by = c(prVars_orig, adVars_orig),
+                                 sum.over = NULL, test.var = testVar)
+  }
+  
+  ## if adjusting, crop all estimates by adjusting variables
+  ## to shortest estimate
+  if (length(adVars)) {
+    adLe <- data[, list(min = min(surv.int), max = max(surv.int)), keyby = eval(adVars)]
+    adLe <- c(max(adLe$min), min(adLe$max))
+    data <- data[surv.int %in% `:`(adLe[1L], adLe[2L])]
+  }
+  
+  # create and print table of bad surv.ints ------------------------------------
+  
+  badObsSurv <- data$surv.obs == 0 | is.na(data$surv.obs)
+  if (sum(badObsSurv)) {
+    
+    zerotab <- data[badObsSurv, 
+                    list(first.bad.surv.int = min(as.integer(surv.int)), 
+                         last.bad.surv.int = max(as.integer(surv.int)), 
+                         surv.obs=min(surv.obs)), keyby = eval(byVars)]
+    
+    
+    message("* popEpi::survtab_ag: Some cumulative surv.obs were zero or NA:")
+    if (length(byVars)) setnames(zerotab, c(prVars, adVars), c(prVars_orig, adVars_orig))
+    print(zerotab)
+    if (surv.method == "lifetable" && data[surv.obs == 0, .N] > 0) {
+      message("* popEpi::survtab_ag: NOTE: Zero surv.obs leads to zero ",
+              "relative survivals as well. Adjusting with weights WILL use ",
+              "the zero surv.obs / relative survival values.")
+    }
+    
+  }
+  rm(badObsSurv)
+  
+  # compute cause-specific survivals  ------------------------------------------
+  if (surv.type == "surv.cause") {
+    
+    ## NOTE: these related to adjusting life-table estimates for delayed entry...
+    #       data[, "n.eff" := n - n.cens/2 + n.de/2 + n.de.cens/4] # + d.de/2
+    #       "n.cens_1" := n.cens + (d-d_1)
+    #       "n.de.cens" := n.de.cens + (d.de - d.de_1)
+    
+    if (surv.method == "lifetable") {
+      for (k in eventVars) {
+        k <- gsub(pattern = "d_", replacement = "", x = k)
+        d_k <- paste0("d_", k)
+        # d.de_k <- paste0("d.de_",k)
+        
+        n.eff_k <- paste0("n.eff_",k)
+        
+        ## old: " := n - (n.cens + (d-", d_k,")/2 + n.de/2 + (n.de.cens + d.de - ", d.de_k,")/4 )"
+        # expr <- paste0(n.eff_k, " := n - (n.cens + (d-", d_k,")/2 )")
+        
+        set(data, j = c(n.eff_k), value = data$n.eff + (data$d - data[[d_k]])/2L ) # + d.de/2
+        # data[,  eval(parse(text = expr), envir = .SD)]
+        
+      }
+      
+    }
+    
+    surv_names <- names(data)[grep("surv.obs", names(data))]
+    surv_names <- c("d", if (surv.method == "lifetable") "n.eff" else NULL, surv_names)
+    setnames(data, surv_names, paste0(surv_names, ".orig"))
+    
+    for (k in eventVars) {
+      
+      k <- gsub(pattern = "d.", replacement = "", x = k)
+      setnames(data, paste0("d.",k), "d")
+      
+      if (surv.method=="hazard") {
+        comp.st.surv.obs.haz(surv.table = data, surv.by.vars = byVars)
+      } else {
+        setnames(data, paste0("n.eff_", k), "n.eff")
+        comp.st.surv.obs.lif(surv.table = data, surv.by.vars = byVars)
+      }
+      os.table <- comp.st.conf.ints(data, al=1-conf.level, surv="surv.obs", transform = conf.type)
+      
+      new_surv_names <- setdiff(surv_names, c("d", if (surv.method == "lifetable") "n.eff" else NULL))
+      new_surv_names <- gsub("surv.obs", paste0("surv.obs.", k), new_surv_names)
+      new_surv_names <- c(paste0(c("d.", if (surv.method == "lifetable") "n.eff." else NULL), k), new_surv_names)
+      setnames(data, surv_names, new_surv_names)
+      
+      
+    }
+    setnames(data, paste0(surv_names, ".orig"), surv_names)
+  }
+  
+  # compute cause-specific/excess-case CIFs ------------------------------------
+  if (surv.type %in% c("cif.obs", "cif.rel")) {
+    
+    data[, "lag1_surv.obs" := shift(surv.obs, n = 1L, type = "lag", fill = 1), by = eval(byVars)]
+    data[, "p.obs" := surv.obs/lag1_surv.obs]
+    
+    if (surv.type == "cif.obs") {
+      for (k in eventVars) {
+        
+        k <- gsub("d.", "", x = k)
+        d.k <- paste0("d.", k)
+        
+        d.var <- paste0("d.",k)
+        q.var <- paste0("q.", k)
+        CIF_var <- paste0("CIF_", k)
+        data[, (q.var)   := (1-p.obs)*get(d.var)/d]
+        data[get(d.var) == 0L | d == 0L, (q.var) := 0]
+        data[, (CIF_var) := cumsum(lag1_surv.obs*get(q.var)), by = eval(byVars)]
+      }
+    }
+    
+    if (surv.type == "cif.rel") {
+      ## assuming d.exp in data
+      data[, "CIF.rel" := (1-p.obs)*(d-d.exp)/d]
+      data[d.exp>d, "CIF.rel" := NA]
+      data[, "CIF.rel" := cumsum(lag1_surv.obs*CIF.rel), by = eval(byVars)]
+    }
+    
+    ## SEs currently not known for CIFs; impute 0 to make adjusting work
+    CIF_vars <- names(data)[substr(names(data),1,3) == "CIF"]
+    data[, c(paste0("SE.", CIF_vars)) := 0L]
+    
+    setcolsnull(data, c("lag1_surv.obs", "p.obs", paste0("q.", substr(eventVars, 3, nchar(eventVars)))))
+    
+  }
+  
+  
+  # relative survivals ---------------------------------------------------------
+  if (surv.type == "surv.rel" & relsurv.method == "e2") {
+    
+    # compute r.e2 -------------------------------------------------------------
+    comp.st.rs <- function(rs.table, rs.by.vars = byVars) {
+      
+      p.exp <- delta <- surv.exp <- surv.obs <- n.eff.pp <- 
+        surv.obs <- NULL ## APPEASE R CMD CHECK
+      ## EdererII
+      
+      ##-------------
+      if (surv.method == "hazard") {
+        rs.table[, "p.exp" := exp(-delta*d.exp/pyrs)] 
+        rs.table[, "surv.exp" := cumprod(p.exp), by = eval(rs.by.vars)]
+        comp.st.r.e2.haz(surv.table = rs.table, surv.by.vars = rs.by.vars)
+      } else {
+        rs.table[, "p.exp" := 1 - d.exp/n]
+        rs.table[, "surv.exp" := cumprod(p.exp), by = eval(rs.by.vars)]
+        
+        if (rs.table[, min(surv.obs, na.rm=TRUE) == 0]) {
+          rs.table[surv.obs == 0, "surv.exp" := 1]
+        }
+        
+        comp.st.r.e2.lif(surv.table = rs.table, surv.by.vars = rs.by.vars)
+        
+        if (rs.table[, min(surv.obs, na.rm=TRUE) == 0]) {
+          rs.table[surv.obs == 0, intersect(c("surv.exp","r.e2","SE.r.e2","r.e2.lo","r.e2.hi"), names(rs.table)) := 0]
+        }
+      }
+      
+      ## ------------
+      
+      rs.table <- comp.st.conf.ints(rs.table, al=1-conf.level, surv="r.e2", transform = conf.type)
+      
+      return(rs.table)
+    }
+    
+    data <- comp.st.rs(rs.table = data)
+    
+    
+  }
+  
+  # compute r.pp ---------------------------------------------------------------
+  if (surv.type == "surv.rel" & relsurv.method == "pp") {
+    
+    all_names_present(data, c("d.pp", "d.exp.pp", "d.pp.2"))
+    ## pohar perme: analysis weighted by expected cumulative survival
+    comp.st.pp <- function(pp.table, by.vars = byVars) {
+      ## relative survival
+      if (surv.method == "hazard") {
+        all_names_present(data, c("pyrs.pp"),
+                          msg = paste0("internal error: work data did not have",
+                                       " variable named pyrs.pp. Complain ",
+                                       "to package maintainer if you see this."))
+        comp.st.r.pp.haz(surv.table = pp.table, surv.by.vars = by.vars)
+      } else {
+        data[, "n.eff.pp" := n.pp - 0.5*n.cens.pp]
+        all_names_present(data, c("n.pp", "n.cens.pp", "n.eff.pp"),
+                          msg = paste0("internal error: work data did not have",
+                                       " variable named n.eff.pp. Complain ",
+                                       "to package maintainer if you see this."))
+        comp.st.r.pp.lif(surv.table = pp.table, surv.by.vars = by.vars)
+        
+        if (pp.table[, min(surv.obs, na.rm=TRUE) == 0]) {
+          pp.table[surv.obs == 0, intersect(c("r.pp","SE.r.pp","r.pp.lo","r.pp.hi"), names(pp.table)) := 0]
+        }
+      }
+      
+      pp.table <- comp.st.conf.ints(pp.table, al=1-conf.level, surv="r.pp", transform = conf.type )
+      
+      return(pp.table)
+    }
+    data <- comp.st.pp(pp.table = data)
+  }
+  
+  # compute adjusted estimates -------------------------------------------------
+  if ("weights" %in% names(data)) {
+    w_est_vars <- names(data)[substr(names(data), 1, 8) == "surv.obs"]
+    w_est_vars <- c(w_est_vars, "r.e2", "r.pp")
+    w_est_vars <- c(w_est_vars,  names(data)[substr(names(data),1,3)=="CIF"])
+    w_est_vars <- intersect(w_est_vars, names(data))
+    w_est_vars <- w_est_vars[unlist(lapply(w_est_vars, function(x) {
+      !substr(x, nchar(x)-2L, nchar(x)) %in% c(".lo", ".hi")
+      }))]
+    w_se_vars <- paste0("SE.", w_est_vars)
+    
+    w_est <- data[, lapply(.SD, function(x) sum(x*weights)), 
+                   keyby = c(prVars, "surv.int"), .SDcols = w_est_vars]
+    w_se <- data[, lapply(.SD, function(x) sqrt(sum((x^2)*(weights^2)))),
+                   keyby = c(prVars, "surv.int"), .SDcols = w_se_vars]
+    
+    data <- data[, lapply(mget(valVars), sum), 
+                 keyby = c(prVars, "surv.int", "Tstart", "Tstop", "delta")]
+    set(data, j = w_se_vars, value = mget(w_se_vars, as.environment(w_se)))
+    set(data, j = w_est_vars, value = mget(w_est_vars, as.environment(w_est)))
+    
+    setnames(data, old = c(w_est_vars, w_se_vars), 
+             new = paste0(c(w_est_vars, w_se_vars), ".as"))
+    
+    for (var in paste0(w_est_vars, ".as")) {
+      data <- comp.st.conf.ints(data, al = 1-conf.level, 
+                                surv = var, transform = conf.type)
+    }
+    
+    
+  }
+  
+  # clean-up -------------------------------------------------------------------
+  ## back to original names of print / adjust (used to avoid e.g. 
+  ## 'factor(V1, 1:2)' going bonkers in data.table)
+  if (length(c(prVars))) setnames(data, c(prVars), c(prVars_orig))
+  prVars <- prVars_orig
+  adVars <- adVars_orig
+  
+  ## reorder table, format numeric values, etc.
+  
+  miscVars <- intersect(names(data), c("surv.int", "Tstart", "Tstop", "delta"))
+  
+  survVars <- c("surv.obs.lo","surv.obs","surv.obs.hi","SE.surv.obs",
+                "r.e2.lo","r.e2","r.e2.hi","SE.r.e2",
+                "r.pp.lo","r.pp","r.pp.hi","SE.r.pp",
+                paste0("CIF.rel.", c("lo", "", "hi")), "SE.CIF.rel",
+                "surv.obs.as.lo","surv.obs.as","surv.obs.as.hi","SE.surv.obs.as",
+                "r.e2.as.lo","r.e2.as","r.e2.as.hi","SE.r.e2.as",
+                "r.pp.as.lo","r.pp.as","r.pp.as.hi","SE.r.pp.as",
+                paste0("CIF.rel.as.", c("lo", "", "hi")), "SE.CIF.rel.as"
+  )
+  survVars <- intersect(survVars, names(data))
+  
+  ## which variables are estimates, SEs, CIs, etc.
+  survVars.ca <- setdiff(names(data), c(prVars, valVars, miscVars, survVars))
+  CIF_vars <- survVars.ca[substr(survVars.ca, 1,3)=="CIF" | substr(survVars.ca, 1,6)=="SE.CIF"]
+  survVars <- c(survVars, CIF_vars)
+  
+  surv.obs.vars <- survVars.ca[substr(survVars.ca, 1,8) == "surv.obs" | substr(survVars.ca, 1,11) == "SE.surv.obs"]
+  survVars <- c(survVars, surv.obs.vars)
+  
+  survVars <- unique(intersect(survVars, names(data)))
+  
+  ## remove some unuseful variables
+  setcolsnull(data, c("SE.A", "SE.B"))
+  setcolsnull(data, survVars[substr(survVars, 1, 6) == "SE.CIF"]) ## since they are zero for now
+  survVars <- intersect(survVars, names(data))
+  
+  SEVars <- survVars[substr(survVars, 1, 3) == "SE."]
+  CIVars <- survVars[substr(survVars, nchar(survVars) - 2L, nchar(survVars)) %in% c(".lo", ".hi")]
+  estVars <- setdiff(survVars, c(SEVars, CIVars))
+  
+  order <- unique(c(prVars, miscVars, valVars, survVars))
+  order <- intersect(order, names(data))
+  
+  setcolsnull(data, setdiff(names(data), order))
+  setcolorder(data,order)
+  
+  setkeyv(data, c(prVars, "surv.int"))
+  
+  # attributes -----------------------------------------------------------------
+  setkeyv(data, c(prVars, "surv.int"))
+  setattr(data, "class", c("survtab", "data.table", "data.frame"))
+  if (!return_DT()) setDFpe(data)
+  if (length(prVars) == 0) prVars <- NULL ## might be character(0) 
+  
+  used_args$data <- origData
+  used_args$formula <- formula
+  used_args$weights <- evalRecursive(arg = weights, env = PF)$weights
+  
+  arglist <- list(call = this_call, 
+                  arguments = used_args,
+                  surv.scale = surv.scale,
+                  surv.breaks = surv.breaks,
+                  print.vars = prVars,
+                  adjust.vars = adVars,
+                  value.vars = valVars,
+                  misc.vars = miscVars,
+                  surv.vars = survVars,
+                  est.vars = estVars,
+                  SE.vars = SEVars,
+                  CI.vars = CIVars)
+  varsArgs <- substr(names(arglist), nchar(names(arglist))-4L, nchar(names(arglist))) == ".vars"
+  varsArgs <- names(arglist)[varsArgs]
+  arglist[varsArgs] <- lapply(arglist[varsArgs], function(x) if (length(x) == 0L) NULL else x)
+                  
+  setattr(data, "survtab.meta", arglist)
+  
+  if (verbose) {
+    message("* popEpi::survtab_ag: finished whole process; ",
+            data.table::timetaken(starttime))
+  }
+  data[]
+}
+
+
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
+#               aggre = list(sex, fot), fot = seq(0, 5, 1/12))
+# ag[, d.exp := pmax(0L, from0to1 - 3L)]
+# st <- survtab_ag(ag, surv.type = "surv.obs", surv.method = "hazard")
+# st <- survtab_ag(ag, surv.type = "surv.cause", surv.method = "hazard", d = list(a = from0to1-3, b = 3))
+
+# sire <- copy(sire)
+# sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
+#               aggre = list(sex, agegr = cut(dg_age, c(0,60,70,80, Inf), labels = FALSE), fot), 
+#               fot = seq(0, 5, 1/12))
+# ag <- lexpand(sire, birth = "bi_date", entry = "bi_date", exit = "ex_date",
+#               status = status %in% 1:2,
+#               aggre = list(sex, age), 
+#               age = seq(0, 100, 1))
+# wdt <- data.table(agegr = 1:4, weights = c(0.2, 0.4, 0.3, 0.1))
+# wli <- list(agegr = c(0.2, 0.4, 0.3, 0.1))
+# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, surv.type = "surv.obs", surv.method = "hazard", weights = wli)
+# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, surv.type = "surv.rel", 
+#                  d.pp = "from0to1.pp", d.pp.2 = "from0to1.pp.2", 
+#                  d.exp.pp = "d.exp.pp", pyrs.pp = "ptime.pp",
+#                  surv.method = "hazard", weights = wli,
+#                  relsurv.method = "pp")
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status, pophaz = popmort, pp = TRUE,
+#               aggre = list(sex, agegr = cut(dg_age, c(0,60,70,80, Inf), labels = FALSE), fot), 
+#               fot = seq(0, 5, 1/12))
+# st <- survtab_ag(fot ~ sex + adjust(agegr), data = ag, 
+#                  d = list(cand = from0to1, othd = from0to2),
+#                  surv.type = "surv.cause", weights = wli)
+# st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs", surv.method = "hazard", adjust = "agegr", weights = wli)
+# st <- survtab_ag(fot ~ adjust(agegr), data = ag, surv.type = "surv.obs", weights = wli)
+# st <- survtab_ag(fot ~ 1, data = ag, adjust = "agegr", surv.type = "surv.obs", weights = wli)
+# st <- survtab_ag(fot ~ 1, data = ag, adjust = "agegr", surv.type = "surv.obs", weights = wli)
+# st <- survtab_ag(fot ~ 1, data = ag, surv.type = "surv.obs")
+
+# wli2 <- wli
+# wli$sex <- c(0.4, 0.6)
+# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.obs", weights = wli)
+# st <- survtab_ag(fot ~ adjust(agegr), data = ag, surv.type = "surv.obs", weights = wli["agegr"])
+# ag[, d.exp := pmax(from0to1 - 1, 0L)]
+# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.rel", weights = wli)
+# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, surv.type = "surv.cause", weights = wli)
+# ag[, othd := pmax(from0to1 - 1L, 0L)]
+# st <- survtab_ag(fot ~ adjust(sex, agegr), data = ag, d = list(cand = from0to1, othd = pmax(from0to1-1L, 0L)), surv.type = "surv.cause", weights = wli)
+
diff --git a/R/survival_lexis.R b/R/survival_lexis.R
index 0269b2d..5be1641 100644
--- a/R/survival_lexis.R
+++ b/R/survival_lexis.R
@@ -1,612 +1,612 @@
-
-
-
-
-
-#' @template survival_doc_template
-#' @param formula a \code{formula}; e.g. \code{fot ~ sex},
-#' where \code{fot} is the time scale over which you wish to estimate a
-#' survival time function; this
-#' assumes that \code{lex.Xst} in your data is the status variable in the
-#' intended format (almost always right). 
-#' To be explicit, use \code{\link[survival]{Surv}}: e.g. 
-#' \code{Surv(fot, lex.Xst) ~ sex}. 
-#' Variables on the right-hand side of the formula
-#' separated by \code{+} are considered stratifying variables, for which 
-#' estimates are computed separately. May contain usage of \code{adjust()} 
-#' --- see Details and Examples.
-#' @param data a \code{Lexis} object with at least the survival time scale
-#' @param breaks a named list of breaks, e.g.
-#' \code{list(FUT = 0:5)}. If data is not split in advance, \code{breaks}
-#' must at the very least contain a vector of breaks to split the survival time 
-#' scale (mentioned in argument \code{formula}). If data has already been split
-#' (using e.g. \code{\link{splitMulti}}) along at least the used survival time
-#' scale, this may be \code{NULL}. It is generally recommended (and sufficient; 
-#' see Seppa, Dyban and Hakulinen (2015)) to use monthly
-#' intervals where applicable.
-#' @param pophaz a \code{data.frame} containing
-#' expected hazards for the event of interest to occur. See the
-#' \link[=pophaz]{dedicated help page}. Required when
-#' \code{surv.type = "surv.rel"} or \code{"cif.rel"}. \code{pophaz} must
-#' contain one column named \code{"haz"}, and any number of other columns
-#' identifying levels of variables to do a merge with split data within
-#' \code{survtab}. Some columns may be time scales, which will
-#' allow for the expected hazard to vary by e.g. calendar time and age.
-#'  
-#' 
-#' 
-#' @examples
-#' \donttest{
-#' data("sire", package = "popEpi")
-#' library(Epi)
-#'
-#' ## NOTE: recommended to use factor status variable
-#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-#'            exit = list(CAL = get.yrs(ex_date)), 
-#'            data = sire[sire$dg_date < sire$ex_date, ],
-#'            exit.status = factor(status, levels = 0:2, 
-#'                                 labels = c("alive", "canD", "othD")), 
-#'            merge = TRUE)
-#' 
-#' ## phony group variable
-#' set.seed(1L)
-#' x$group <- rbinom(nrow(x), 1, 0.5)
-#' 
-#' ## observed survival. explicit supplying of status:
-#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-#'               surv.type = "surv.obs",
-#'               breaks = list(FUT = seq(0, 5, 1/12)))
-#' ## this assumes the status is lex.Xst (right 99.9 % of the time)
-#' st <- survtab(FUT ~ group, data = x, 
-#'               surv.type = "surv.obs",
-#'               breaks = list(FUT = seq(0, 5, 1/12)))
-#'               
-#' ## relative survival (ederer II)
-#' data("popmort", package = "popEpi")
-#' pm <- data.frame(popmort)
-#' names(pm) <- c("sex", "CAL", "AGE", "haz")
-#' st <- survtab(FUT ~ group, data = x, 
-#'               surv.type = "surv.rel",
-#'               pophaz = pm,
-#'               breaks = list(FUT = seq(0, 5, 1/12)))
-#' 
-#' ## ICSS weights usage
-#' data("ICSS", package = "popEpi")
-#' cut <- c(0, 30, 50, 70, Inf)
-#' agegr <- cut(ICSS$age, cut, right = FALSE)
-#' w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-#' x$agegr <- cut(x$dg_age, cut, right = FALSE)
-#' st <- survtab(FUT ~ group + adjust(agegr), data = x, 
-#'               surv.type = "surv.rel",
-#'               pophaz = pm, weights = w$ICSS1,
-#'               breaks = list(FUT = seq(0, 5, 1/12)))
-#' 
-#' #### using dates with survtab
-#' x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-#'            exit = list(CAL = ex_date),
-#'            data = sire[sire$dg_date < sire$ex_date, ],
-#'            exit.status = factor(status, levels = 0:2, 
-#'                                 labels = c("alive", "canD", "othD")), 
-#'            merge = TRUE)
-#' ## phony group variable
-#' set.seed(1L)
-#' x$group <- rbinom(nrow(x), 1, 0.5)
-#' 
-#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-#'               surv.type = "surv.obs",
-#'               breaks = list(FUT = seq(0, 5, 1/12)*365.25))    
-#'                   
-#' ## NOTE: population hazard should be reported at the same scale
-#' ## as time variables in your Lexis data.
-#' data(popmort, package = "popEpi")
-#' pm <- data.frame(popmort)
-#' names(pm) <- c("sex", "CAL", "AGE", "haz")
-#' ## from year to day level
-#' pm$haz <- pm$haz/365.25 
-#' pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
-#' pm$AGE <- pm$AGE*365.25 
-#' 
-#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-#'               surv.type = "surv.rel", relsurv.method = "e2",
-#'               pophaz = pm,
-#'               breaks = list(FUT = seq(0, 5, 1/12)*365.25))  
-#' }
-#' @export
-survtab <- function(formula, data, adjust = NULL, breaks = NULL, 
-                    pophaz = NULL, weights = NULL, surv.type = "surv.rel", 
-                    surv.method = "hazard", relsurv.method = "e2", 
-                    subset = NULL, 
-                    conf.level = 0.95, 
-                    conf.type = "log-log",
-                    verbose = FALSE) {
-  
-  TF <- environment()
-  PF <- parent.frame()
-  this_call <- match.call()
-  startTime <- proc.time()
-  
-  ## appease R CMD CHECK -------------------------------------------------------
-  lex.Cst <- lex.Xst <- lex.dur <- NULL
-  
-  ## checks --------------------------------------------------------------------
-  
-  if (missing(formula)) stop("Formula not defined!")
-  
-  checkLexisData(data)
-  
-  allScales <- attr(data, "time.scales")
-  splitScales <- names(breaks)
-  
-  ## ensure breaks make sense --------------------------------------------------
-  oldBreaks <- attr(data, "breaks")
-  checkBreaksList(data, breaks = oldBreaks)
-  testOldBreaks <- setdiff(oldBreaks, list(NULL))
-  if (is.null(breaks) && !length(testOldBreaks)) {
-    stop("No breaks supplied via argument 'breaks', and data has not been ",
-         "split in advance. Please supply a list of breaks ",
-         "to argument 'breaks'")
-  }
-  if (is.null(breaks)) breaks <- oldBreaks
-  checkBreaksList(data, breaks = breaks)
-  ## match break types to time scale types
-  ## (don't try to match time scales to breaks)
-  splitScales <- names(breaks)
-  for (k in splitScales) {
-    breaks[[k]] <- matchBreakTypes(data, breaks = breaks[[k]], timeScale = k)
-  }
-  
-  comp_pp <- FALSE
-  drop <- TRUE
-  if (surv.type == "surv.rel" && relsurv.method == "pp") comp_pp <- TRUE
-  if (comp_pp) drop <- FALSE
-  
-  
-  ## data & subset -------------------------------------------------------------
-  subset <- evalLogicalSubset(data, substitute(subset))
-  x <- data[subset, ]; rm(subset)
-  setDT(x)
-  forceLexisDT(x, breaks = NULL, allScales = allScales, key = TRUE)
-  
-  ## pre-eval of print & adjust ------------------------------------------------
-  
-  adSub <- substitute(adjust)
-  adTest <- evalRecursive(adSub, env = x, enc = PF)
-  if (!is.null(adTest)) {
-    adSub <- adTest$argSub
-    adVars <- all.vars(adSub)
-  } else {
-    adSub <- substitute(NULL)
-    adVars <- NULL
-  } 
-  
-  formula <- evalRecursive(formula, env = TF, enc = PF)$arg
-  foVars <- all.vars(formula)
-  
-  if (!inherits(formula,"formula")) {
-    stop("Argument 'formula' is not a formula object. Usage: e.g. ",
-         "Surv(fot, lex.Xst %in% 1:2) ~ sex")
-  }
-  if (length(formula) != 3L) {
-    stop("Argument 'formula'must be two-sided. Usage: e.g. ",
-         "Surv(fot, lex.Xst %in% 1:2) ~ sex")
-  }
-  ## eval print & adjust -------------------------------------------------------
-  ## this adjust passed to resulting data's attributes at the end
-  adSub <- substitute(adjust)
-  adjust <- evalPopArg(data = x, arg = adSub, 
-                       enclos = PF, DT = TRUE, 
-                       recursive = TRUE)
-  
-  l <- usePopFormula(form = formula, adjust = adjust, data = x, enclos = PF, 
-                     Surv.response = "either")
-  prVars <- names(l$print)
-  adVars <- names(l$adjust)
-  
-  
-  ## check weights makes sense with respect to adjust --------------------------
-  if (length(adVars) > 0L && !is.null(weights)) {
-    checkWeights(weights, adjust = l$adjust)
-    
-  }
-  
-  ## check pophaz --------------------------------------------------------------
-  
-  if (surv.type %in% c("surv.rel", "cif.rel")) {
-    checkPophaz(x, pophaz, haz.name = "haz")
-  }
-  pophazVars <- setdiff(names(pophaz), "haz")
-  
-  ## only keep necessary variables ---------------------------------------------
-  
-  setcolsnull(x, keep = c("lex.id", "lex.dur", allScales, 
-                          "lex.Cst", "lex.Xst", pophazVars))
-  if (length(prVars)) x[, c(prVars)] <- l$print
-  if (length(adVars)) x[, c(adVars)] <- l$adjust
-  
-  
-  ## simplify event and censoring indicators -----------------------------------
-  cens.values <- event.values <- NULL
-  all.values <- if (is.factor(l$y$status)) levels(l$y$status) else 
-    sort(unique(l$y$status))
-  cens.values <- all.values[1L]
-  event.values <- setdiff(all.values, cens.values)
-  
-  if (is.numeric(l$y$status) && all(unique(l$y$status) %in% 0:1)) {
-    ## this should apply to situations where status coded 0/1
-    ## and both 0/1 present or only 1 present
-    if (all(unique(l$y$status) %in% 0L)) {
-      stop("All status values were zero, i.e. all obs were censored. ",
-           "Check that you passed the correct status variable or --- if this ",
-           "was intended --- code the status variable to 0/1 so that 1 ",
-           "corresponds to the event taking place and 0 not.")
-    }
-    cens.values <- 0L
-    event.values <- 1L
-  }
-  
-  x[, lex.Cst := NULL]
-  x[, lex.Cst := TF$cens.values]
-  x[, lex.Xst := NULL]
-  x[, lex.Xst := l$y$status]
-  harmonizeStatuses(x, C = "lex.Cst", X = "lex.Xst")
-  
-  if (!surv.type %in% c("cif.obs", "surv.cause")) {
-    ## this simplifies computations
-    
-    x[, lex.Cst := NULL]
-    x[, lex.Cst := 0L]
-    setcolorder(x, c(intersect(names(data), names(x)), 
-                     setdiff(names(x), names(data))))
-    
-    x[, lex.Xst := as.integer(lex.Xst %in% TF$event.values)]
-    cens.values <- 0L
-    event.values <- 1L
-    if (x[, sum(lex.Xst)] == 0L) {
-      stop("There are no events in the data. Ensure that the event argument ",
-           "used in Surv() makes sense.")
-    }
-  }
-  
-  ## detect which time scale used ----------------------------------------------
-  
-  survScale <- detectSurvivalTimeScale(lex = x, values = l$y$time)
-  
-  
-  ## crop data to speed up computations ----------------------------------------
-  cropBreaks <- breaks
-  if (surv.type == "surv.rel" && relsurv.method == "pp")  {
-    ## pp-weights have to be computed from entry to follow-up till roof of breaks;
-    ## can only crop along the survival time scale
-    cropBreaks <- breaks[1L]
-    cb <- protectFromDrop(cropBreaks[[1L]], lower = TRUE)
-    cb <- c(min(cb), max(cropBreaks[[1L]]))
-    cropBreaks[[1L]] <- cb
-  }
-  
-  
-  intelliCrop(x = x, breaks = cropBreaks, allScales = allScales, cropStatuses = TRUE)
-  x <- intelliDrop(x, breaks = cropBreaks, dropNegDur = TRUE, check = TRUE)
-  setDT(x)
-  forceLexisDT(x, breaks = oldBreaks, allScales = allScales, key = TRUE)
-  
-  ## splitting -----------------------------------------------------------------
-  
-  splitTime <- proc.time()
-  setDT(x)
-  forceLexisDT(x, breaks = oldBreaks, allScales = allScales, key = TRUE)
-  x <- splitMulti(x, breaks = breaks, drop = FALSE, merge = TRUE)
-  setDT(x)
-  forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
-  if (verbose) {
-    message("* popEpi::survtab: Time taken by splitting Lexis data: ", 
-            timetaken(splitTime), "\n")
-  }
-  
-  ## pophaz merge --------------------------------------------------------------
-  if (!is.null(pophaz)) {
-    hazTime <- proc.time()
-    haz <- NULL ## appease R CMD CHECK
-    x <- cutLowMerge(x, pophaz, by = pophazVars, 
-                     mid.scales = intersect(pophazVars, allScales))
-    setDT(x)
-    forceLexisDT(x, breaks = breaks, allScales =allScales, key = TRUE)
-    if (verbose) {
-      message("* popEpi::survtab: Time taken by merging population hazards ",
-              "with split Lexis data: ", timetaken(hazTime))
-    }
-  }
-  
-  ## pp computation ------------------------------------------------------------
-  ppNames <- d.pp <- d.pp.2 <- d.exp.pp <- ptime.pp <- 
-    at.risk.pp <- n.cens.pp <- NULL
-  
-  if (comp_pp) {
-    ppTime <- proc.time()
-    setkeyv(x, c("lex.id", survScale))
-    comp_pp_weights(x, surv.scale = survScale, 
-                    breaks = breaks[[survScale]], haz = "haz", 
-                    style = "delta", verbose = FALSE)
-    setDT(x)
-    forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
-    
-    if (verbose) {
-      message("* popEpi::survtab: computed Pohar Perme weights; ", 
-              data.table::timetaken(ppTime))
-    }
-    
-    intelliCrop(x = x, breaks = breaks, allScales = allScales, cropStatuses = TRUE)
-    x <- intelliDrop(x, breaks = breaks, dropNegDur = TRUE, check = TRUE)
-    forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
-    
-    ppTime <- proc.time()
-    pp <- comp_pp_weighted_figures(x, haz = "haz", pp = "pp", by = "lex.id")
-    ppNames <- makeTempVarName(x, pre = names(pp))
-    x[, c(TF$ppNames) := TF$pp] ## note: TF$pp avoids conflicts
-    rm(pp)
-    
-    d.pp.2 <- ppNames[substr(ppNames, 1, 13) == "from0to1.pp.2"]
-    d.pp <- ppNames[substr(ppNames, 1, 11) == "from0to1.pp"]
-    d.pp <- setdiff(d.pp, d.pp.2)
-    d.exp.pp <- ppNames[substr(ppNames, 1, 8) == "d.exp.pp"]
-    ptime.pp <- ppNames[substr(ppNames, 1, 8) == "ptime.pp"]
-    n.cens.pp <- ppNames[substr(ppNames, 1, 11) == "from0to0.pp"]
-    n.cens.pp <- n.cens.pp[substr(n.cens.pp, 1,13) != "from0to0.pp.2"]
-    at.risk.pp <-  ppNames[substr(ppNames, 1, 10) == "at.risk.pp"]
-    d.exp.pp <-  ppNames[substr(ppNames, 1, 8) == "d.exp.pp"]
-    
-    if (verbose) {
-      message("* popEpi::survtab: computed Pohar Perme weighted counts and ",
-              "person-times; ", data.table::timetaken(ppTime))
-    }
-  }
-  
-  d.exp <- NULL
-  if (surv.type %in% c("surv.rel", "cif.rel") && "haz" %in% names(x)) {
-    d.exp <- makeTempVarName(x, pre = "d.exp_")
-    x[, c(TF$d.exp) := lex.dur * haz]
-  }
-  
-  ## aggregation ---------------------------------------------------------------
-  aggreTime <- proc.time()
-  
-  ## this includes time scale to compute survivals over
-  aggreVars <- c(prVars, adVars, survScale) 
-  
-  setDT(x)
-  forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
-  
-  if (verbose) {
-    message("* popEpi::survtab: calling popEpi::aggre")
-  }
-  x <- aggre(x, by = aggreVars, verbose = verbose,
-             sum.values = c(d.exp, ppNames))
-  if (verbose) {
-    message("* popEpi::survtab: finished aggre call successfully")
-  }
-  setDT(x)
-  setattr(x, "class", c("aggre", "data.table", "data.frame"))
-  
-  if (verbose) {
-    message("* popEpi::survtab: done aggregating split Lexis data; ",
-            data.table::timetaken(aggreTime))
-  }
-  
-  ## neater column names -------------------------------------------------------
-  ## in case there are zero obs that are censored
-  censCols <- paste0("from", cens.values, "to", cens.values)
-  if (all(!censCols %in% names(x))) {
-    x[, c(censCols) := 0L]
-  }
-  
-  ## e.g. fromAlivetoDead -> Dead; looks better in survtab_ag output
-  evCols <- paste0("from", cens.values, "to", c(cens.values, event.values))
-  whEC <- which(evCols %in% names(x))
-  
-  if (sum(whEC)) {
-    setnames(x, evCols[whEC], 
-             as.character(c(cens.values, event.values)[whEC]))
-  }
-  
-  ## survtab_ag ----------------------------------------------------------------
-  dn <- intersect(event.values, names(x))
-  if (length(dn) == 0L) {
-    stop("Internal error: no event variables in work data. Complain to the ",
-         "package maintainer if you see this - unless there are no events ",
-         "in the data?")
-  }
-  n.cens <- intersect(cens.values, names(x))
-  
-  if (length(prVars) == 0L) {
-    prVars <- "1"
-  } 
-  
-  form <- as.formula(paste0(survScale, " ~ ", paste0(prVars, collapse = " + ")))
-  if (verbose) {
-    message("* popEpi::survtab: calling popEpi::survtab_ag")
-  }
-  st <- survtab_ag(data = x, 
-                   formula = TF$form,
-                   adjust = TF$adVars,
-                   
-                   weights = TF$weights, 
-                   
-                   d = TF$dn, pyrs = "pyrs", n = "at.risk",
-                   d.exp = TF$d.exp, n.cens = TF$n.cens,
-                   
-                   n.pp = TF$at.risk.pp,
-                   d.pp = TF$d.pp, d.exp.pp = TF$d.exp.pp, d.pp.2 = TF$d.pp.2, 
-                   n.cens.pp = TF$n.cens.pp, pyrs.pp = TF$ptime.pp,
-                   
-                   surv.type = surv.type,
-                   surv.method = surv.method,
-                   relsurv.method = relsurv.method,
-                   
-                   conf.type = conf.type,
-                   conf.level = conf.level,
-                   
-                   verbose = verbose)
-  if (verbose) {
-    message("* popEpi::survtab: successfully called popEpi::survtab_ag")
-  }
-  
-  ## attributes ----------------------------------------------------------------
-  attributes(st)$survtab.meta$call <- this_call
-  attributes(st)$survtab.meta$arguments$adjust <- adjust
-  attributes(st)$survtab.meta$arguments$conf.type <- conf.type
-  attributes(st)$survtab.meta$arguments$conf.level <- conf.level
-  
-  attributes(st)$survtab.meta$arguments$surv.type <- surv.type
-  attributes(st)$survtab.meta$arguments$surv.method <- surv.method
-  attributes(st)$survtab.meta$arguments$relsurv.method <- relsurv.method
-  
-  if (verbose) {
-    message("* popEpi::survtab: finished; ", data.table::timetaken(startTime))
-  }
-  return(st[])
-}
-# library(Epi)
-# library(popEpi)
-# dt <- copy(sire)[dg_date < ex_date,]
-# dt[, agegr := cut(dg_age, c(0,50,75,Inf))]
-# dt[, sex := rbinom(n = .N, size = 1, prob = 0.5)]
-# dt <- Lexis(data = dt, entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-#             exit = list(CAL = get.yrs(ex_date)), entry.status = 0L, exit.status = status, merge = TRUE)
-# pm <- copy(popEpi::popmort)
-# setnames(pm, c("agegroup", "year"), c("AGE", "CAL"))
-# st <- survtab(data = dt, formula = Surv(FUT, lex.Xst) ~ 1, #adjust = "agegr", 
-#                   # pophaz = pm,
-#                   surv.type = "surv.obs",
-#                   # weights = list(agegr = c(0.2,0.4,0.4)),
-#                   breaks = list(FUT = seq(0,5,1/12)))
-# st <- survtab(dt, print = NULL, #adjust = "agegr", 
-#                   # pophaz = pm,
-#                   surv.type = "surv.obs",
-#                   # weights = list(agegr = c(0.2,0.4,0.4)),
-#                   breaks = list(AGE = seq(0,100, 1)))
-# st <- survtab(dt, print = NULL, #adjust = "agegr", 
-#                   pophaz = pm,
-#                   surv.type = "surv.rel",
-#                   relsurv.method = "pp",
-#                   # weights = list(agegr = c(0.2,0.4,0.4)),
-#                   breaks = list(FUT = seq(0,5,1/12)))
-# st <- survtab(dt, print = NULL, adjust = c("sex","agegr"), 
-#                   pophaz = pm,
-#                   surv.type = "surv.rel",
-#                   relsurv.method = "pp",
-#                   weights = list(sex = c(0.5, 0.5), agegr = c(0.2,0.4,0.4)),
-#                   breaks = list(FUT = seq(0,5,1/12)))
-
-
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
-#               fot = seq(0, 5, 1/12))
-# pm2 <- copy(popEpi::popmort)
-# setnames(pm2, c("year", "agegroup"), c("per", "age"))
-# st <- survtab(ag, print = NULL, #adjust = c("sex","agegr"), 
-#                   pophaz = pm2,
-#                   surv.type = "surv.rel",
-#                   relsurv.method = "pp",
-#                   #weights = list(sex = c(0.5, 0.5), agegr = c(0.2,0.4,0.4)),
-#                   breaks = list(fot = seq(0,5,1/12)))
-detectEvents <- function(x, breaks, tol = .Machine$double.eps^0.5, by = "lex.id") {
-  ## INTENTION: given a Lexis object, determines which rows
-  ## have an event (a transition or end-point) within the window
-  ## determined by breaks (a list of breaks as supplied to e.g. splitMulti).
-  ## Usable with split and unsplit data, though it is best to do this
-  ## before splitting for efficiency.
-  ## NOTE: by should be a character vector specifying variables that identify
-  ## unique subjects; the idea is that each subject only has one end-point
-  ## and some transitions
-  ## NOTE: breaks should be a list of breaks or NULL; if it is NULL,
-  ## it is NOT checked whether observations were cut short by the breaks used.
-  ## observations cut short are not any kind of events.
-  ## OUTPUT: an integer vector coding events as follows:
-  ## 0: no event within breaks (row cut short by breaks or subject 
-  ##    has multiple rows, of which this is not an event)
-  ## 1: transition within breaks
-  ## 2: original end-point within breaks and no transition occured (i.e. censoring)
-  if (!is.data.table(x)) stop("x must be a data.table; if you see this, send the package maintainer an email")
-  # checkLexisData(x)
-  if (!inherits(x, "Lexis")) stop("data not a Lexis object")
-  if (!is.null(breaks)) {
-    checkBreaksList(x, breaks)
-    breaks[unlist(lapply(breaks, length)) == 0L] <- NULL
-  }
-  
-  ## R CMD CHECK appeasement
-  lex.Cst <- lex.Xst <- NULL
-  
-  tmp <- list()
-  oldKey <- key(x)
-  if (length(oldKey) == 0L) {
-    tmp$order <- makeTempVarName(x, pre = "order_")
-    on.exit(if (tmp$order %in% names(x)) setorderv(x, tmp$order), add = TRUE)
-    on.exit(setcolsnull(x, tmp$order, soft = TRUE), add = TRUE)
-    set(x, j = tmp$order, value = 1:nrow(x))
-  } else on.exit(setkeyv(x, oldKey), add = TRUE)
-    
-  
-  setkeyv(x, c(by, names(breaks)[1L]))
-  setkeyv(x, by)
-  ## rows that actually can be events: transitions and last rows by subject
-  whTr <- x[, lex.Cst != lex.Xst]
-  whLa <- !duplicated(x, fromLast = TRUE, by=key(x))
-  whEv <- whTr | whLa
-  
-  if (!is.null(breaks)) {
-    
-    splitScales <- names(breaks)
-    if (any(!splitScales %in% names(x))) stop("Following time scales missing from data that data was split by: ", paste0("'", setdiff(splitScales, names(x)), "'", collapse = ", "))
-    
-    brmax <- lapply(breaks, max)
-    brmin <- lapply(breaks, min)
-    
-    ## detect rows residing within breaks window
-    for (sc in splitScales) {
-      z <- (x$lex.dur + x[[sc]])[whEv]
-      tol_sc <- if (is.double(z)) tol else 0L
-      
-      ## NOTE: if max of orig values within breaks window, then all may be events
-      if (!(max(z) + tol_sc < brmax[[sc]])) whEv[whEv] <- z < brmax[[sc]] - tol_sc
-      if (!(min(z) - tol_sc > brmin[[sc]])) whEv[whEv] <- z > brmin[[sc]] + tol_sc
-      
-    }
-    ## whEv now indicates rows that may be events AND which reside within breaks window. 
-  }
-  
-  ## censored events are not transitions, but must reside within breaks window.
-  whCe <- whLa & !whTr & whEv
-  
-  ## need to add event indicator to data since it has been reordered,
-  ## reorder back old order, and return the event indicator.
-  tmp$ind <- makeTempVarName(x, pre = "event_indicator_")
-  on.exit(setcolsnull(x, delete = tmp$ind, soft = TRUE), add = TRUE)
-  evInd <- as.integer(whEv)
-  evInd <- ifelse(whCe, 2L, evInd)
-  set(x, j = tmp$ind, value = evInd)
-  
-  if (length(oldKey) == 0L) {
-    setkeyv(x, NULL)
-    setorderv(x, tmp$order)
-    set(x, j = tmp$order, value = NULL)
-  } else setkeyv(x, oldKey)
-  
-  
-  evInd <- x[[tmp$ind]]
-  set(x, j = tmp$ind, value = NULL)
-  on.exit(expr = {}, add = FALSE) ## removes on.exit expressions from earlier
-  
-  
-  if (!identical(oldKey, key(x))) stop("keys do not match at function end; send an email to package maintainer if you see this")
-  
-  evInd
-}
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+#' @template survival_doc_template
+#' @param formula a \code{formula}; e.g. \code{fot ~ sex},
+#' where \code{fot} is the time scale over which you wish to estimate a
+#' survival time function; this
+#' assumes that \code{lex.Xst} in your data is the status variable in the
+#' intended format (almost always right). 
+#' To be explicit, use \code{\link[survival]{Surv}}: e.g. 
+#' \code{Surv(fot, lex.Xst) ~ sex}. 
+#' Variables on the right-hand side of the formula
+#' separated by \code{+} are considered stratifying variables, for which 
+#' estimates are computed separately. May contain usage of \code{adjust()} 
+#' --- see Details and Examples.
+#' @param data a \code{Lexis} object with at least the survival time scale
+#' @param breaks a named list of breaks, e.g.
+#' \code{list(FUT = 0:5)}. If data is not split in advance, \code{breaks}
+#' must at the very least contain a vector of breaks to split the survival time 
+#' scale (mentioned in argument \code{formula}). If data has already been split
+#' (using e.g. \code{\link{splitMulti}}) along at least the used survival time
+#' scale, this may be \code{NULL}. It is generally recommended (and sufficient; 
+#' see Seppa, Dyban and Hakulinen (2015)) to use monthly
+#' intervals where applicable.
+#' @param pophaz a \code{data.frame} containing
+#' expected hazards for the event of interest to occur. See the
+#' \link[=pophaz]{dedicated help page}. Required when
+#' \code{surv.type = "surv.rel"} or \code{"cif.rel"}. \code{pophaz} must
+#' contain one column named \code{"haz"}, and any number of other columns
+#' identifying levels of variables to do a merge with split data within
+#' \code{survtab}. Some columns may be time scales, which will
+#' allow for the expected hazard to vary by e.g. calendar time and age.
+#'  
+#' 
+#' 
+#' @examples
+#' \donttest{
+#' data("sire", package = "popEpi")
+#' library(Epi)
+#'
+#' ## NOTE: recommended to use factor status variable
+#' x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+#'            exit = list(CAL = get.yrs(ex_date)), 
+#'            data = sire[sire$dg_date < sire$ex_date, ],
+#'            exit.status = factor(status, levels = 0:2, 
+#'                                 labels = c("alive", "canD", "othD")), 
+#'            merge = TRUE)
+#' 
+#' ## phony group variable
+#' set.seed(1L)
+#' x$group <- rbinom(nrow(x), 1, 0.5)
+#' 
+#' ## observed survival. explicit supplying of status:
+#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+#'               surv.type = "surv.obs",
+#'               breaks = list(FUT = seq(0, 5, 1/12)))
+#' ## this assumes the status is lex.Xst (right 99.9 % of the time)
+#' st <- survtab(FUT ~ group, data = x, 
+#'               surv.type = "surv.obs",
+#'               breaks = list(FUT = seq(0, 5, 1/12)))
+#'               
+#' ## relative survival (ederer II)
+#' data("popmort", package = "popEpi")
+#' pm <- data.frame(popmort)
+#' names(pm) <- c("sex", "CAL", "AGE", "haz")
+#' st <- survtab(FUT ~ group, data = x, 
+#'               surv.type = "surv.rel",
+#'               pophaz = pm,
+#'               breaks = list(FUT = seq(0, 5, 1/12)))
+#' 
+#' ## ICSS weights usage
+#' data("ICSS", package = "popEpi")
+#' cut <- c(0, 30, 50, 70, Inf)
+#' agegr <- cut(ICSS$age, cut, right = FALSE)
+#' w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+#' x$agegr <- cut(x$dg_age, cut, right = FALSE)
+#' st <- survtab(FUT ~ group + adjust(agegr), data = x, 
+#'               surv.type = "surv.rel",
+#'               pophaz = pm, weights = w$ICSS1,
+#'               breaks = list(FUT = seq(0, 5, 1/12)))
+#' 
+#' #### using dates with survtab
+#' x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+#'            exit = list(CAL = ex_date),
+#'            data = sire[sire$dg_date < sire$ex_date, ],
+#'            exit.status = factor(status, levels = 0:2, 
+#'                                 labels = c("alive", "canD", "othD")), 
+#'            merge = TRUE)
+#' ## phony group variable
+#' set.seed(1L)
+#' x$group <- rbinom(nrow(x), 1, 0.5)
+#' 
+#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+#'               surv.type = "surv.obs",
+#'               breaks = list(FUT = seq(0, 5, 1/12)*365.25))    
+#'                   
+#' ## NOTE: population hazard should be reported at the same scale
+#' ## as time variables in your Lexis data.
+#' data(popmort, package = "popEpi")
+#' pm <- data.frame(popmort)
+#' names(pm) <- c("sex", "CAL", "AGE", "haz")
+#' ## from year to day level
+#' pm$haz <- pm$haz/365.25 
+#' pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
+#' pm$AGE <- pm$AGE*365.25 
+#' 
+#' st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+#'               surv.type = "surv.rel", relsurv.method = "e2",
+#'               pophaz = pm,
+#'               breaks = list(FUT = seq(0, 5, 1/12)*365.25))  
+#' }
+#' @export
+survtab <- function(formula, data, adjust = NULL, breaks = NULL, 
+                    pophaz = NULL, weights = NULL, surv.type = "surv.rel", 
+                    surv.method = "hazard", relsurv.method = "e2", 
+                    subset = NULL, 
+                    conf.level = 0.95, 
+                    conf.type = "log-log",
+                    verbose = FALSE) {
+  
+  TF <- environment()
+  PF <- parent.frame()
+  this_call <- match.call()
+  startTime <- proc.time()
+  
+  ## appease R CMD CHECK -------------------------------------------------------
+  lex.Cst <- lex.Xst <- lex.dur <- NULL
+  
+  ## checks --------------------------------------------------------------------
+  
+  if (missing(formula)) stop("Formula not defined!")
+  
+  checkLexisData(data)
+  
+  allScales <- attr(data, "time.scales")
+  splitScales <- names(breaks)
+  
+  ## ensure breaks make sense --------------------------------------------------
+  oldBreaks <- attr(data, "breaks")
+  checkBreaksList(data, breaks = oldBreaks)
+  testOldBreaks <- setdiff(oldBreaks, list(NULL))
+  if (is.null(breaks) && !length(testOldBreaks)) {
+    stop("No breaks supplied via argument 'breaks', and data has not been ",
+         "split in advance. Please supply a list of breaks ",
+         "to argument 'breaks'")
+  }
+  if (is.null(breaks)) breaks <- oldBreaks
+  checkBreaksList(data, breaks = breaks)
+  ## match break types to time scale types
+  ## (don't try to match time scales to breaks)
+  splitScales <- names(breaks)
+  for (k in splitScales) {
+    breaks[[k]] <- matchBreakTypes(data, breaks = breaks[[k]], timeScale = k)
+  }
+  
+  comp_pp <- FALSE
+  drop <- TRUE
+  if (surv.type == "surv.rel" && relsurv.method == "pp") comp_pp <- TRUE
+  if (comp_pp) drop <- FALSE
+  
+  
+  ## data & subset -------------------------------------------------------------
+  subset <- evalLogicalSubset(data, substitute(subset))
+  x <- data[subset, ]; rm(subset)
+  setDT(x)
+  forceLexisDT(x, breaks = NULL, allScales = allScales, key = TRUE)
+  
+  ## pre-eval of print & adjust ------------------------------------------------
+  
+  adSub <- substitute(adjust)
+  adTest <- evalRecursive(adSub, env = x, enc = PF)
+  if (!is.null(adTest)) {
+    adSub <- adTest$argSub
+    adVars <- all.vars(adSub)
+  } else {
+    adSub <- substitute(NULL)
+    adVars <- NULL
+  } 
+  
+  formula <- evalRecursive(formula, env = TF, enc = PF)$arg
+  foVars <- all.vars(formula)
+  
+  if (!inherits(formula,"formula")) {
+    stop("Argument 'formula' is not a formula object. Usage: e.g. ",
+         "Surv(fot, lex.Xst %in% 1:2) ~ sex")
+  }
+  if (length(formula) != 3L) {
+    stop("Argument 'formula'must be two-sided. Usage: e.g. ",
+         "Surv(fot, lex.Xst %in% 1:2) ~ sex")
+  }
+  ## eval print & adjust -------------------------------------------------------
+  ## this adjust passed to resulting data's attributes at the end
+  adSub <- substitute(adjust)
+  adjust <- evalPopArg(data = x, arg = adSub, 
+                       enclos = PF, DT = TRUE, 
+                       recursive = TRUE)
+  
+  l <- usePopFormula(form = formula, adjust = adjust, data = x, enclos = PF, 
+                     Surv.response = "either")
+  prVars <- names(l$print)
+  adVars <- names(l$adjust)
+  
+  
+  ## check weights makes sense with respect to adjust --------------------------
+  if (length(adVars) > 0L && !is.null(weights)) {
+    checkWeights(weights, adjust = l$adjust)
+    
+  }
+  
+  ## check pophaz --------------------------------------------------------------
+  
+  if (surv.type %in% c("surv.rel", "cif.rel")) {
+    checkPophaz(x, pophaz, haz.name = "haz")
+  }
+  pophazVars <- setdiff(names(pophaz), "haz")
+  
+  ## only keep necessary variables ---------------------------------------------
+  
+  setcolsnull(x, keep = c("lex.id", "lex.dur", allScales, 
+                          "lex.Cst", "lex.Xst", pophazVars))
+  if (length(prVars)) x[, c(prVars)] <- l$print
+  if (length(adVars)) x[, c(adVars)] <- l$adjust
+  
+  
+  ## simplify event and censoring indicators -----------------------------------
+  cens.values <- event.values <- NULL
+  all.values <- if (is.factor(l$y$status)) levels(l$y$status) else 
+    sort(unique(l$y$status))
+  cens.values <- all.values[1L]
+  event.values <- setdiff(all.values, cens.values)
+  
+  if (is.numeric(l$y$status) && all(unique(l$y$status) %in% 0:1)) {
+    ## this should apply to situations where status coded 0/1
+    ## and both 0/1 present or only 1 present
+    if (all(unique(l$y$status) %in% 0L)) {
+      stop("All status values were zero, i.e. all obs were censored. ",
+           "Check that you passed the correct status variable or --- if this ",
+           "was intended --- code the status variable to 0/1 so that 1 ",
+           "corresponds to the event taking place and 0 not.")
+    }
+    cens.values <- 0L
+    event.values <- 1L
+  }
+  
+  x[, lex.Cst := NULL]
+  x[, lex.Cst := TF$cens.values]
+  x[, lex.Xst := NULL]
+  x[, lex.Xst := l$y$status]
+  harmonizeStatuses(x, C = "lex.Cst", X = "lex.Xst")
+  
+  if (!surv.type %in% c("cif.obs", "surv.cause")) {
+    ## this simplifies computations
+    
+    x[, lex.Cst := NULL]
+    x[, lex.Cst := 0L]
+    setcolorder(x, c(intersect(names(data), names(x)), 
+                     setdiff(names(x), names(data))))
+    
+    x[, lex.Xst := as.integer(lex.Xst %in% TF$event.values)]
+    cens.values <- 0L
+    event.values <- 1L
+    if (x[, sum(lex.Xst)] == 0L) {
+      stop("There are no events in the data. Ensure that the event argument ",
+           "used in Surv() makes sense.")
+    }
+  }
+  
+  ## detect which time scale used ----------------------------------------------
+  
+  survScale <- detectSurvivalTimeScale(lex = x, values = l$y$time)
+  
+  
+  ## crop data to speed up computations ----------------------------------------
+  cropBreaks <- breaks
+  if (surv.type == "surv.rel" && relsurv.method == "pp")  {
+    ## pp-weights have to be computed from entry to follow-up till roof of breaks;
+    ## can only crop along the survival time scale
+    cropBreaks <- breaks[1L]
+    cb <- protectFromDrop(cropBreaks[[1L]], lower = TRUE)
+    cb <- c(min(cb), max(cropBreaks[[1L]]))
+    cropBreaks[[1L]] <- cb
+  }
+  
+  
+  intelliCrop(x = x, breaks = cropBreaks, allScales = allScales, cropStatuses = TRUE)
+  x <- intelliDrop(x, breaks = cropBreaks, dropNegDur = TRUE, check = TRUE)
+  setDT(x)
+  forceLexisDT(x, breaks = oldBreaks, allScales = allScales, key = TRUE)
+  
+  ## splitting -----------------------------------------------------------------
+  
+  splitTime <- proc.time()
+  setDT(x)
+  forceLexisDT(x, breaks = oldBreaks, allScales = allScales, key = TRUE)
+  x <- splitMulti(x, breaks = breaks, drop = FALSE, merge = TRUE)
+  setDT(x)
+  forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
+  if (verbose) {
+    message("* popEpi::survtab: Time taken by splitting Lexis data: ", 
+            timetaken(splitTime), "\n")
+  }
+  
+  ## pophaz merge --------------------------------------------------------------
+  if (!is.null(pophaz)) {
+    hazTime <- proc.time()
+    haz <- NULL ## appease R CMD CHECK
+    x <- cutLowMerge(x, pophaz, by = pophazVars, 
+                     mid.scales = intersect(pophazVars, allScales))
+    setDT(x)
+    forceLexisDT(x, breaks = breaks, allScales =allScales, key = TRUE)
+    if (verbose) {
+      message("* popEpi::survtab: Time taken by merging population hazards ",
+              "with split Lexis data: ", timetaken(hazTime))
+    }
+  }
+  
+  ## pp computation ------------------------------------------------------------
+  ppNames <- d.pp <- d.pp.2 <- d.exp.pp <- ptime.pp <- 
+    at.risk.pp <- n.cens.pp <- NULL
+  
+  if (comp_pp) {
+    ppTime <- proc.time()
+    setkeyv(x, c("lex.id", survScale))
+    comp_pp_weights(x, surv.scale = survScale, 
+                    breaks = breaks[[survScale]], haz = "haz", 
+                    style = "delta", verbose = FALSE)
+    setDT(x)
+    forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
+    
+    if (verbose) {
+      message("* popEpi::survtab: computed Pohar Perme weights; ", 
+              data.table::timetaken(ppTime))
+    }
+    
+    intelliCrop(x = x, breaks = breaks, allScales = allScales, cropStatuses = TRUE)
+    x <- intelliDrop(x, breaks = breaks, dropNegDur = TRUE, check = TRUE)
+    forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
+    
+    ppTime <- proc.time()
+    pp <- comp_pp_weighted_figures(x, haz = "haz", pp = "pp", by = "lex.id")
+    ppNames <- makeTempVarName(x, pre = names(pp))
+    x[, c(TF$ppNames) := TF$pp] ## note: TF$pp avoids conflicts
+    rm(pp)
+    
+    d.pp.2 <- ppNames[substr(ppNames, 1, 13) == "from0to1.pp.2"]
+    d.pp <- ppNames[substr(ppNames, 1, 11) == "from0to1.pp"]
+    d.pp <- setdiff(d.pp, d.pp.2)
+    d.exp.pp <- ppNames[substr(ppNames, 1, 8) == "d.exp.pp"]
+    ptime.pp <- ppNames[substr(ppNames, 1, 8) == "ptime.pp"]
+    n.cens.pp <- ppNames[substr(ppNames, 1, 11) == "from0to0.pp"]
+    n.cens.pp <- n.cens.pp[substr(n.cens.pp, 1,13) != "from0to0.pp.2"]
+    at.risk.pp <-  ppNames[substr(ppNames, 1, 10) == "at.risk.pp"]
+    d.exp.pp <-  ppNames[substr(ppNames, 1, 8) == "d.exp.pp"]
+    
+    if (verbose) {
+      message("* popEpi::survtab: computed Pohar Perme weighted counts and ",
+              "person-times; ", data.table::timetaken(ppTime))
+    }
+  }
+  
+  d.exp <- NULL
+  if (surv.type %in% c("surv.rel", "cif.rel") && "haz" %in% names(x)) {
+    d.exp <- makeTempVarName(x, pre = "d.exp_")
+    x[, c(TF$d.exp) := lex.dur * haz]
+  }
+  
+  ## aggregation ---------------------------------------------------------------
+  aggreTime <- proc.time()
+  
+  ## this includes time scale to compute survivals over
+  aggreVars <- c(prVars, adVars, survScale) 
+  
+  setDT(x)
+  forceLexisDT(x, breaks = breaks, allScales = allScales, key = TRUE)
+  
+  if (verbose) {
+    message("* popEpi::survtab: calling popEpi::aggre")
+  }
+  x <- aggre(x, by = aggreVars, verbose = verbose,
+             sum.values = c(d.exp, ppNames))
+  if (verbose) {
+    message("* popEpi::survtab: finished aggre call successfully")
+  }
+  setDT(x)
+  setattr(x, "class", c("aggre", "data.table", "data.frame"))
+  
+  if (verbose) {
+    message("* popEpi::survtab: done aggregating split Lexis data; ",
+            data.table::timetaken(aggreTime))
+  }
+  
+  ## neater column names -------------------------------------------------------
+  ## in case there are zero obs that are censored
+  censCols <- paste0("from", cens.values, "to", cens.values)
+  if (all(!censCols %in% names(x))) {
+    x[, c(censCols) := 0L]
+  }
+  
+  ## e.g. fromAlivetoDead -> Dead; looks better in survtab_ag output
+  evCols <- paste0("from", cens.values, "to", c(cens.values, event.values))
+  whEC <- which(evCols %in% names(x))
+  
+  if (sum(whEC)) {
+    setnames(x, evCols[whEC], 
+             as.character(c(cens.values, event.values)[whEC]))
+  }
+  
+  ## survtab_ag ----------------------------------------------------------------
+  dn <- intersect(event.values, names(x))
+  if (length(dn) == 0L) {
+    stop("Internal error: no event variables in work data. Complain to the ",
+         "package maintainer if you see this - unless there are no events ",
+         "in the data?")
+  }
+  n.cens <- intersect(cens.values, names(x))
+  
+  if (length(prVars) == 0L) {
+    prVars <- "1"
+  } 
+  
+  form <- as.formula(paste0(survScale, " ~ ", paste0(prVars, collapse = " + ")))
+  if (verbose) {
+    message("* popEpi::survtab: calling popEpi::survtab_ag")
+  }
+  st <- survtab_ag(data = x, 
+                   formula = TF$form,
+                   adjust = TF$adVars,
+                   
+                   weights = TF$weights, 
+                   
+                   d = TF$dn, pyrs = "pyrs", n = "at.risk",
+                   d.exp = TF$d.exp, n.cens = TF$n.cens,
+                   
+                   n.pp = TF$at.risk.pp,
+                   d.pp = TF$d.pp, d.exp.pp = TF$d.exp.pp, d.pp.2 = TF$d.pp.2, 
+                   n.cens.pp = TF$n.cens.pp, pyrs.pp = TF$ptime.pp,
+                   
+                   surv.type = surv.type,
+                   surv.method = surv.method,
+                   relsurv.method = relsurv.method,
+                   
+                   conf.type = conf.type,
+                   conf.level = conf.level,
+                   
+                   verbose = verbose)
+  if (verbose) {
+    message("* popEpi::survtab: successfully called popEpi::survtab_ag")
+  }
+  
+  ## attributes ----------------------------------------------------------------
+  attributes(st)$survtab.meta$call <- this_call
+  attributes(st)$survtab.meta$arguments$adjust <- adjust
+  attributes(st)$survtab.meta$arguments$conf.type <- conf.type
+  attributes(st)$survtab.meta$arguments$conf.level <- conf.level
+  
+  attributes(st)$survtab.meta$arguments$surv.type <- surv.type
+  attributes(st)$survtab.meta$arguments$surv.method <- surv.method
+  attributes(st)$survtab.meta$arguments$relsurv.method <- relsurv.method
+  
+  if (verbose) {
+    message("* popEpi::survtab: finished; ", data.table::timetaken(startTime))
+  }
+  return(st[])
+}
+# library(Epi)
+# library(popEpi)
+# dt <- copy(sire)[dg_date < ex_date,]
+# dt[, agegr := cut(dg_age, c(0,50,75,Inf))]
+# dt[, sex := rbinom(n = .N, size = 1, prob = 0.5)]
+# dt <- Lexis(data = dt, entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+#             exit = list(CAL = get.yrs(ex_date)), entry.status = 0L, exit.status = status, merge = TRUE)
+# pm <- copy(popEpi::popmort)
+# setnames(pm, c("agegroup", "year"), c("AGE", "CAL"))
+# st <- survtab(data = dt, formula = Surv(FUT, lex.Xst) ~ 1, #adjust = "agegr", 
+#                   # pophaz = pm,
+#                   surv.type = "surv.obs",
+#                   # weights = list(agegr = c(0.2,0.4,0.4)),
+#                   breaks = list(FUT = seq(0,5,1/12)))
+# st <- survtab(dt, print = NULL, #adjust = "agegr", 
+#                   # pophaz = pm,
+#                   surv.type = "surv.obs",
+#                   # weights = list(agegr = c(0.2,0.4,0.4)),
+#                   breaks = list(AGE = seq(0,100, 1)))
+# st <- survtab(dt, print = NULL, #adjust = "agegr", 
+#                   pophaz = pm,
+#                   surv.type = "surv.rel",
+#                   relsurv.method = "pp",
+#                   # weights = list(agegr = c(0.2,0.4,0.4)),
+#                   breaks = list(FUT = seq(0,5,1/12)))
+# st <- survtab(dt, print = NULL, adjust = c("sex","agegr"), 
+#                   pophaz = pm,
+#                   surv.type = "surv.rel",
+#                   relsurv.method = "pp",
+#                   weights = list(sex = c(0.5, 0.5), agegr = c(0.2,0.4,0.4)),
+#                   breaks = list(FUT = seq(0,5,1/12)))
+
+
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status %in% 1:2, pophaz = popmort, pp = TRUE,
+#               fot = seq(0, 5, 1/12))
+# pm2 <- copy(popEpi::popmort)
+# setnames(pm2, c("year", "agegroup"), c("per", "age"))
+# st <- survtab(ag, print = NULL, #adjust = c("sex","agegr"), 
+#                   pophaz = pm2,
+#                   surv.type = "surv.rel",
+#                   relsurv.method = "pp",
+#                   #weights = list(sex = c(0.5, 0.5), agegr = c(0.2,0.4,0.4)),
+#                   breaks = list(fot = seq(0,5,1/12)))
+detectEvents <- function(x, breaks, tol = .Machine$double.eps^0.5, by = "lex.id") {
+  ## INTENTION: given a Lexis object, determines which rows
+  ## have an event (a transition or end-point) within the window
+  ## determined by breaks (a list of breaks as supplied to e.g. splitMulti).
+  ## Usable with split and unsplit data, though it is best to do this
+  ## before splitting for efficiency.
+  ## NOTE: by should be a character vector specifying variables that identify
+  ## unique subjects; the idea is that each subject only has one end-point
+  ## and some transitions
+  ## NOTE: breaks should be a list of breaks or NULL; if it is NULL,
+  ## it is NOT checked whether observations were cut short by the breaks used.
+  ## observations cut short are not any kind of events.
+  ## OUTPUT: an integer vector coding events as follows:
+  ## 0: no event within breaks (row cut short by breaks or subject 
+  ##    has multiple rows, of which this is not an event)
+  ## 1: transition within breaks
+  ## 2: original end-point within breaks and no transition occured (i.e. censoring)
+  if (!is.data.table(x)) stop("x must be a data.table; if you see this, send the package maintainer an email")
+  # checkLexisData(x)
+  if (!inherits(x, "Lexis")) stop("data not a Lexis object")
+  if (!is.null(breaks)) {
+    checkBreaksList(x, breaks)
+    breaks[unlist(lapply(breaks, length)) == 0L] <- NULL
+  }
+  
+  ## R CMD CHECK appeasement
+  lex.Cst <- lex.Xst <- NULL
+  
+  tmp <- list()
+  oldKey <- key(x)
+  if (length(oldKey) == 0L) {
+    tmp$order <- makeTempVarName(x, pre = "order_")
+    on.exit(if (tmp$order %in% names(x)) setorderv(x, tmp$order), add = TRUE)
+    on.exit(setcolsnull(x, tmp$order, soft = TRUE), add = TRUE)
+    set(x, j = tmp$order, value = 1:nrow(x))
+  } else on.exit(setkeyv(x, oldKey), add = TRUE)
+    
+  
+  setkeyv(x, c(by, names(breaks)[1L]))
+  setkeyv(x, by)
+  ## rows that actually can be events: transitions and last rows by subject
+  whTr <- x[, lex.Cst != lex.Xst]
+  whLa <- !duplicated(x, fromLast = TRUE, by=key(x))
+  whEv <- whTr | whLa
+  
+  if (!is.null(breaks)) {
+    
+    splitScales <- names(breaks)
+    if (any(!splitScales %in% names(x))) stop("Following time scales missing from data that data was split by: ", paste0("'", setdiff(splitScales, names(x)), "'", collapse = ", "))
+    
+    brmax <- lapply(breaks, max)
+    brmin <- lapply(breaks, min)
+    
+    ## detect rows residing within breaks window
+    for (sc in splitScales) {
+      z <- (x$lex.dur + x[[sc]])[whEv]
+      tol_sc <- if (is.double(z)) tol else 0L
+      
+      ## NOTE: if max of orig values within breaks window, then all may be events
+      if (!(max(z) + tol_sc < brmax[[sc]])) whEv[whEv] <- z < brmax[[sc]] - tol_sc
+      if (!(min(z) - tol_sc > brmin[[sc]])) whEv[whEv] <- z > brmin[[sc]] + tol_sc
+      
+    }
+    ## whEv now indicates rows that may be events AND which reside within breaks window. 
+  }
+  
+  ## censored events are not transitions, but must reside within breaks window.
+  whCe <- whLa & !whTr & whEv
+  
+  ## need to add event indicator to data since it has been reordered,
+  ## reorder back old order, and return the event indicator.
+  tmp$ind <- makeTempVarName(x, pre = "event_indicator_")
+  on.exit(setcolsnull(x, delete = tmp$ind, soft = TRUE), add = TRUE)
+  evInd <- as.integer(whEv)
+  evInd <- ifelse(whCe, 2L, evInd)
+  set(x, j = tmp$ind, value = evInd)
+  
+  if (length(oldKey) == 0L) {
+    setkeyv(x, NULL)
+    setorderv(x, tmp$order)
+    set(x, j = tmp$order, value = NULL)
+  } else setkeyv(x, oldKey)
+  
+  
+  evInd <- x[[tmp$ind]]
+  set(x, j = tmp$ind, value = NULL)
+  on.exit(expr = {}, add = FALSE) ## removes on.exit expressions from earlier
+  
+  
+  if (!identical(oldKey, key(x))) stop("keys do not match at function end; send an email to package maintainer if you see this")
+  
+  evInd
+}
+
+
+
+
+
+
+
+
+
diff --git a/R/survival_utility_functions.R b/R/survival_utility_functions.R
index 96ac17f..1d372ae 100644
--- a/R/survival_utility_functions.R
+++ b/R/survival_utility_functions.R
@@ -1,639 +1,639 @@
-globalVariables(c("tab", "SE.A", "pp.table"))
-
-comp.st.surv <- function(surv.var = "p", surv.expr = "1-d/(n.eff-n.cens/2)", 
-                         SE.expr = "sqrt(p*(1-p)/(n.eff))", cumu = TRUE) {
-  
-  function(surv.table = tab, surv.by.vars = NULL) {
-    
-    tabname <- deparse(substitute(surv.table))
-    #     tabname <- "surv.table"    
-    by.expr <- NULL
-    if (!is.null(surv.by.vars)) {
-      surv.by.vars <- paste0("'", surv.by.vars, "'", collapse = " ,")
-      by.expr <- paste0(", by = c( ", surv.by.vars, " )")
-    }
-    
-    surv.expr <-  paste0(surv.var, " := ", surv.expr)
-    tab.surv.expr <- paste0(tabname, "[, ", surv.expr,  by.expr, "]")
-    #   surv.expr <- parse(text=surv.expr)
-    #   surv.var <- parse(text=surv.var)
-    SE.var <- paste0("SE.", surv.var)
-    SE.expr <- paste0(SE.var, " := ", SE.expr)
-    tab.SE.expr <- paste0(tabname, "[, ", SE.expr, ", ", by.expr, "]")
-    #   SE.expr <- parse(text=SE.expr)
-    #   SE.var <- parse(text=SE.var)
-    cumuexpr <- paste0(tabname, "[, ", surv.var, " := cumprod(", surv.var, ")", by.expr, "]" )
-    
-    
-    ## parent.frame(2): two steps upstream in environments
-    pe <- function(obj, ..., env=parent.frame(2)) {
-      eval(parse(text=obj), ..., envir=env)
-    }
-    
-    pe(tab.surv.expr)
-    
-    
-    if (cumu) pe(cumuexpr)
-    
-    ## standard error
-    pe(tab.SE.expr)
-    
-    ## zero survival leads to zero SE
-    minexpr <- paste0(tabname, "[",surv.var, "== 0, ", SE.var, " := 0]")
-    pe(minexpr)
-  }
-}
-
-
-comp.st.surv.obs.lif <- comp.st.surv(surv.var=  "surv.obs",
-                                     surv.expr= "1-d/n.eff", 
-                                     SE.expr=   "surv.obs*sqrt(  cumsum(  d/(n.eff*(n.eff-d))  )  )", ## old : sqrt(p*(1-p)/(n.eff))"
-                                     cumu = TRUE)
-
-comp.st.surv.obs.haz <- comp.st.surv(surv.var=  "surv.obs", 
-                                     surv.expr= "exp(-delta*d/pyrs)", 
-                                     SE.expr=   "surv.obs*sqrt(  cumsum(  delta^2*d/pyrs^2  )  )", ## old: sqrt(p*(1-p)/(n.eff))
-                                     cumu = TRUE)
-
-
-comp.st.r.e2.haz <- comp.st.surv(surv.var = "r.e2",
-                                 surv.expr = "exp(-delta*(d-d.exp)/pyrs)",
-                                 SE.expr = "SE.surv.obs/surv.exp",
-                                 cumu=TRUE)
-
-comp.st.r.e2.lif <- comp.st.surv(surv.var = "r.e2",
-                                 surv.expr = "1-(d-d.exp)/(n.eff)",
-                                 SE.expr = "SE.surv.obs/surv.exp",
-                                 cumu=TRUE)
-
-comp.st.r.pp.haz <- comp.st.surv(surv.var = "r.pp",
-                                 surv.expr= "exp(-delta*(d.pp - d.exp.pp)/pyrs.pp)",
-                                 SE.expr = "r.pp*sqrt(  cumsum(  delta^2*( d.pp.2)/ pyrs.pp^2  )  )",
-                                 cumu=TRUE)
-
-comp.st.r.pp.lif <- comp.st.surv(surv.var = "r.pp",
-                                 surv.expr = "1-(d.pp-d.exp.pp)/(n.eff.pp)",
-                                 SE.expr = "r.pp*sqrt(  cumsum(  d.pp.2/(n.eff^2)  )  )",
-                                 cumu=TRUE)
-
-
-## this function will calculate confidence intervals
-## for obs & rel & net survivals
-#' @import stats
-comp.st.conf.ints <- function(tab = pp.table, al=0.05, surv="r.pp", transform ="log-log") {
-  zlo <- as.character(qnorm(al/2))
-  zhi <- as.character(qnorm(1-al/2))
-  SE.surv <- paste0("SE.",surv)
-  surv.hi <- paste0(surv, ".hi")
-  surv.lo <- paste0(surv, ".lo")
-  
-  tmp_se <- makeTempVarName(names = names(tab), pre = "delta_method_SE_", length = 15L)
-  
-  pe <- function(...) {
-    eval(parse(text=paste0(...)), envir=tab)
-  }
-  
-  if (transform =="plain") {
-    ## assume S(t)~N(mu, sigma)
-    
-    ex <- paste0(surv,  " ", zlo, "*", SE.surv)
-    tab[, (surv.lo)  := pe(ex)]
-    ex <- paste0(surv, " +", zhi, "*", SE.surv)
-    tab[, (surv.hi)  := pe(ex)]
-    
-  } else if (transform =="log-log") {
-    ## assume log(H(t))~N(mu, sigma)
-    ## -> delta method SE: sqrt( SE^2 * (1/(log(SURV)*SURV))^2 )
-    
-    ex <- paste0(SE.surv,"/(abs(log(",surv,"))*",surv,")")
-    tab[,  (tmp_se) := pe(ex)]
-    
-    ex <- paste0(surv, "^exp(", zhi, "*", tmp_se,")")
-    tab[, (surv.lo)  := pe(ex)]
-    ex <- paste0(surv, "^exp(", zlo, "*", tmp_se,")")
-    tab[, (surv.hi)  := pe(ex)]
-    
-  } else if (transform =="log") {
-    ## assume log(S(t))~N(mu, sigma)
-    ## -> delta method SE: SE/SURV
-    
-    ex <- paste0(SE.surv,"/",surv)
-    tab[,  (tmp_se) := pe(ex)]
-    
-    ex <- paste0(surv, "*exp(", zlo, "*", tmp_se,")")
-    tab[, (surv.lo)  := pe(ex)]
-    ex <- paste0(surv, "*exp(", zhi, "*", tmp_se,")")
-    tab[, (surv.hi)  := pe(ex)]
-  }
-  if (tmp_se %in% names(tab)) tab[, (tmp_se) := NULL]
-  
-  ## zero SE means zero uncertainty means lo=hi=estimate
-  tab[tab[[SE.surv]] == 0, c(surv.lo, surv.hi) := .SD, .SDcols = surv]
-  
-  tab[]
-}
-
-
-
-# x <- Lexis(data=sire[1,], entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
-#            exit=list(per=get.yrs(ex_date)), exit.status=status)
-# x <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 1/12), per=1994:2013, age = 0:150))
-# x[, surv.int := cut(fot, seq(0, 5, 1/12) - .Machine$double.eps^0.5, labels = FALSE)]
-# x <- cutLowMerge(x, popmort, by.x = c("sex","per", "age"), 
-#                  by.y = c("sex", "year", "agegroup"), 
-#                  mid.scales = c("per", "age"), all.x = TRUE, all.y = FALSE)
-# comp_pp_weights(x, surv.scale = "fot", breaks = seq(0, 5, 1/12), haz = "haz", style = "delta")
-
-comp_pp_weights <- function(lex, surv.scale = "fot", breaks = NULL, haz = "haz", style = "delta", verbose = FALSE) {
-  ppTime <- proc.time()
-  ## input: a split Lexis object (data.table) and the time scale to compute
-  ## pp weights over; lex must also contain 'haz', the population
-  ## (expected) hazard level for each row
-  TF <- environment()
-  
-  lex.dur <- lex.id <- pp <- NULL ## APPEASE R CMD CHECK
-  
-  style <- match.arg(style, c("delta", "actual"))
-  if (!is.data.table(lex)) stop("lex must be a data.table")
-  
-  all_names_present(lex, c(haz, surv.scale, "lex.id", "lex.dur"))
-  
-  if ("pp" %in% names(lex)) stop("Variable named 'pp' existed in data when attempting to compute pohar-perme weights; 'pp' is reserved so you should delete or rename that variable")
-  
-  
-  if (!identical(key(lex), c("lex.id", surv.scale))) stop("lex must be a data.table keyed by lex.id and the survival time scale")
-  
-  
-  breaks <- sort(unique(breaks)) - .Machine$double.eps^0.5
-  
-  ## need a bunch of temporary variable names to compute pp weights
-  ## inside the data set without overwriting anything existing.
-  ## this will take about 0.1 secs.
-  tmpSI <- makeTempVarName(data = lex, pre = "surv.int_")
-  tmpSIstart <- makeTempVarName(data = lex, pre = "surv.int.start_")
-  tmpSIstop <- makeTempVarName(data = lex, pre = "surv.int.stop_")
-  tmpSIlength <- makeTempVarName(data = lex, pre = "surv.int.length_")
-  on.exit(setcolsnull(lex, delete = c(tmpSI, tmpSIstart, tmpSIstop, tmpSIlength, tmpPS, tmpPCS, tmpPCSM)), add = TRUE)
-  
-  set(lex, j = tmpSI, value = cut(lex[[surv.scale]], breaks, labels = FALSE))
-  set(lex, j = tmpSIstop, value = breaks[-1][lex[[tmpSI]]])
-  set(lex, j = tmpSIstart, value = breaks[-length(breaks)][lex[[tmpSI]]])
-  set(lex, j = tmpSIlength, value = lex[[tmpSIstop]] - lex[[tmpSIstart]])
-  
-  tmpPS <- makeTempVarName(data = lex, pre = "pop.surv_")
-  tmpPCS <- makeTempVarName(data = lex, pre = "pop.cumsurv_")
-  tmpPCSM <- makeTempVarName(data = lex, pre = "pop.cumsurv.mid_")
-
-  ## conditional survs
-  if (verbose) condSurvTime <- proc.time()
-  
-  set(lex, j = tmpPS, value = exp(-lex[[haz]]*lex$lex.dur))
-  ## till end of each interval...
-  lex[, c(tmpPCS) := list(exp(-cumsum(.SD[[1L]]*lex.dur))),
-      by = lex.id, .SDcols = c(haz, "lex.dur")]
-  ## till start of each interval
-  set(lex, j = tmpPCS, value = lex[[tmpPCS]] / lex[[tmpPS]]) 
-  if (verbose) cat("Time taken by computing expected survivals up to start of each interval for each lex.id: ", timetaken(condSurvTime), "\n")
-  
-  ## pohar-perme weighting by expected cumulative survival. approximation:
-  ## cumulative survival up to either middle of remaining surv.int (not individual-specific)
-  ## or up to middle of subject's follow-up in each row (individual-specific)
-  ## difference: e.g. 2 rows within a surv.int have either the same or different pp-weights
-  if (style == "actual") {
-    set(lex, j = tmpPCSM, value = lex[[tmpPCS]] * (lex[[tmpPS]])^0.5)
-  }
-  if (style == "delta") {
-    if (verbose) deltaTime <- proc.time()
-    
-    setkeyv(lex, c("lex.id", tmpSI))
-    ## expected survival up to middle of remaining time in surv.int
-    ## cumulation starting from first record for subject in each surv.int
-    
-    ## some records are the only one for a lex.id in a surv.int; these are easy
-    first_in_surv.int <- !duplicated(lex, fromLast = FALSE, by = key(lex))
-    last_in_surv.int <- !duplicated(lex, fromLast = TRUE, by = key(lex))
-    only_in_surv.int <- first_in_surv.int & last_in_surv.int
-    
-    lex[only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * exp(.SD[[2L]] * (.SD[[3L]] - .SD[[4L]])/2L),
-        .SDcols = c(tmpPCS, haz, tmpSIstop, surv.scale)]
-    
-    ## more complicated with many records in a surv.int per lex.id
-    if (any(!only_in_surv.int)) {
-      
-      tmpSImid <- makeTempVarName(lex, pre = "surv.int.mid_")
-      dist <- makeTempVarName(lex, pre = "dist_")
-      on.exit(setcolsnull(lex, delete = c(dist, tmpSImid)), add = TRUE)
-      
-      ## middle point of survival interval
-      set(lex, j = tmpSImid, value = (lex[[tmpSIstop]] - lex[[tmpSIstart]])/2L)
-      
-      ## distance from remaining surv.int mid-point starting from start of 
-      ## record; or at most lex.dur; for integration
-      set(lex, j = dist, value = pmin(lex$lex.dur, lex[[tmpSImid]]))
-      ## some records after mid-point can have negative fot.dist at this point
-      set(lex, j = dist, value = pmax(lex[[dist]], 0))
-      
-      ## some lex.id are censored / die before mid of surv.int; last record
-      ## must reach its fot.dist at least up to the mid (or be zero due to above)
-      lex[last_in_surv.int, c(dist) := pmax((.SD[[1L]] - .SD[[2L]])/2L, 0), 
-          .SDcols = c(tmpSIstop, surv.scale)]
-      
-      byTime <- proc.time()
-      
-      ## from start of first in surv.int till mid point
-      ## step by step for efficiency...
-      lex[!only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * .SD[[2L]], .SDcols = c(haz, dist)]
-      lex[!only_in_surv.int, c(tmpPCSM) := lapply(.SD, sum), 
-          .SDcols = c(tmpPCSM), by = c("lex.id", tmpSI)]
-      lex[!only_in_surv.int, c(tmpPCS) := .SD[1L], 
-          by = c("lex.id", tmpSI), .SDcols = c(tmpPCS)]
-      lex[!only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * exp(-.SD[[2L]]), 
-          .SDcols = c(tmpPCS, tmpPCSM)]
-      
-      ## todo: alternate faster method for integration!
-      setcolsnull(lex, delete = c(dist))
-      if (verbose) cat("Time taken by extra computation due to style 'delta': ", timetaken(deltaTime), "\n")
-    }
-    
-    rm(first_in_surv.int, last_in_surv.int, only_in_surv.int)
-    if (verbose) cat("Time taken by computation of Pohar-Perme weights: ", timetaken(ppTime), "\n")
-  }
-  
-  setkeyv(lex, c("lex.id", surv.scale))
-  
-  lex[, pp := 1/.SD, .SDcols = tmpPCSM]
-  
-  invisible(lex[])
-}
-
-comp_pp_weighted_figures <- function(lex, haz = "haz", pp = "pp", event.ind = NULL, by = "lex.id") {
-  ## PURPOSE: given a split Lexis object with a column of pre-computed
-  ## pohar-perme weights, computes pp-weighted:
-  ## * person-time (lex.dur*pp)
-  ## * at-risk indicator
-  ## * event counts
-  ## * event counts multiplied with pp^2
-  ## * expected event counts
-  ## events are transitions and end points as detected by detectEvents,
-  ## and include censorings.
-  ## OUTPUT: a DT of pp-weighted things.
-  
-  ## appease R CMD CHECK
-  lex.Cst <- lex.Xst <- NULL
-  
-  checkLexisData(lex, check.breaks = TRUE)
-  if (!is.data.table(lex)) stop("lex must be a data.table")
-  
-  all_names_present(lex, c(pp, haz, event.ind, by))
-  
-  ## NOTE: we want to avoid conflicts with possible variable names in lex
-  ## (e.g. an existing column named ptime.pp might conflict with ptime.pp char vec)
-  e <- environment()
-  
-  if (is.null(event.ind)) {
-    event.ind <- makeTempVarName(lex, pre = "event_indicator_")
-    on.exit(setcolsnull(lex, delete = event.ind, soft = TRUE), add = TRUE)
-    set(lex, j = event.ind, value = detectEvents(lex, breaks = attr(lex, "breaks"), by = by))
-    all_names_present(lex, event.ind)
-  }
-  ## data.table is probably faster in this than simply using vectors
-  idt <- data.table(1:2)
-  names(idt) <- event.ind
-  haveEvents <- sort(lex[idt, on = event.ind, which = TRUE])
-  evtab <- lex[haveEvents, .(obs = .N), by = list(lex.Cst, lex.Xst)]
-  set(evtab, j = "obs", value = NULL)
-  
-  events <- paste0("from", evtab$lex.Cst, "to", evtab$lex.Xst)
-  ppVars <- c(paste0(events, ".pp"), "ptime.pp", "d.exp.pp")
-  
-  
-#   ## build table to join with lex to limit to rows with events -----------------
-#   ## non-event-rows have zero valued pp-weighted event counts, naturally
-#   set(evtab, j = event.ind, value = NULL)
-#   evtab <- rbindlist(list(evtab, evtab))
-#   set(evtab, j = event.ind, value = rep(1:2, each = nrow(evtab)/2L))
-#   setkeyv(evtab, c("lex.Cst", "lex.Xst", event.ind))
-  
-  ## pp-weighted events --------------------------------------------------------
-  ## NOTE: a pp-weighted event is simply the pp weight where the event occurs
-  ## and zero otherwise (0L/1L times pp-weight)
-  
-  evN <- length(events)
-  evdt <- data.table(rn = rep(1:nrow(lex), times = evN))
-  set(evdt, j = "eventType", value = factor(rep(1:evN, each = nrow(lex)), levels = 1:evN, labels = events))
-  set(evdt, j = "pp", value = rep(lex[[pp]],length.out=nrow(evdt)))
-  
-  
-  ## need to still determine which rows are not their eventType's events -------
-  
-  rnVar <- makeTempVarName(lex, pre = "rowNum_")
-  on.exit(setcolsnull(lex, delete = rnVar, soft = TRUE), add = TRUE)
-  set(lex, j = rnVar, value = 1:nrow(lex))
-  
-  ## row numbers in lex by event type
-  rowNums <- lex[haveEvents][evtab, .(rowNum = .SD[[e$rnVar]]), by = .EACHI, on = c("lex.Cst", "lex.Xst"), .SDcols = rnVar]
-  ## multiply to accommodate expanded evdt data
-  
-  rowNum <- NULL ## appease R CMD CHECK
-  rowNums[, rowNum := rowNum+(.GRP-1L)*nrow(lex), by = list(lex.Cst, lex.Xst)]
-  noEvents <- setdiff(1:nrow(evdt), rowNums$rowNum)
-  
-  set(evdt, i = noEvents, j = "pp", value = 0)
-  
-  evdt <- dcast.data.table(evdt, rn ~ eventType, value.var = "pp")
-  
-  setorderv(evdt, "rn")
-  set(evdt, j = "rn", value = NULL)
-  
-  ## ptime.pp & d.exp.pp -------------------------------------------------------
-  set(evdt, j = "ptime.pp", value = lex$lex.dur * lex[[pp]])
-  set(evdt, j = "d.exp.pp", value = lex$lex.dur * lex[[haz]] * lex[[pp]])
-  
-  setcolorder(evdt, c("ptime.pp", "d.exp.pp", sort(events)))
-  setnames(evdt, events, paste0(events, ".pp"))
-  
-  ## pp-weighted at-risk indicators --------------------------------------------
-  ## these will be n.pp at the aggregate level.
-  set(evdt, j = "at.risk.pp", value = lex[[pp]]*1L)
-  
-  ## events multiplied with pp-weight again ------------------------------------
-  ## (i.e. event times pp squared)
-  
-  set(evdt, j = paste0(events, ".pp.2"), 
-      value = evdt[, .SD, .SDcols = paste0(events, ".pp")]*lex[[pp]])
-  
-  return(evdt[])
-  
-}
-
-
-
-test_empty_surv_ints <- function(x, by = NULL, sum.over = NULL, test.var = "pyrs", show.by = NULL) {
-  
-  x <- copy(x)
-  oc <- class(x)
-  setDT(x)
-  setattr(x, "class", oc)
-  
-  surv.int <- NULL ## APPEASE R CMD CHECK
-  
-  all_names_present(x, by, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
-  all_names_present(x, sum.over, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
-  all_names_present(x, test.var, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
-  
-  if (any(!sum.over %in% by)) stop("sum.over must be a subset of by.")
-  if (length(show.by) == 0L) show.by <- by
-  if (length(by) != length(show.by)) {
-    stop("Internal error: length(sum.over) != length(show.sum.over). ",
-         "If you see this, complain to the package maintainer.")
-  }
-  
-  wh_sum.to <- !by %in% sum.over
-  sum.to <- by[wh_sum.to]
-  show.sum.to <- show.by[wh_sum.to]
-  
-  if (length(sum.to) == 0L) sum.to <- show.sum.to <- NULL
-  
-  tmpTV <- makeTempVarName(x, pre = "testValues_")
-  tmpDiff <- makeTempVarName(x, pre = "diff_")
-  on.exit(setcolsnull(x, delete = c(tmpTV, tmpDiff)))
-  
-  ## first check empty surv.ints are all consecutive...
-  ## consecutiveness: e.g. out of 10 surv ints, 6-10 are empty.
-  ## non-consecutiveness: e.g. out of 10 surv ints, 6-8 are empty.
-  ## (then 9-10 will have NA estimates as well.)
-  setkeyv(x, c(by, "surv.int"))
-  
-  ct <- x[, lapply(.SD, sum), keyby = c(sum.to, "surv.int"), .SDcols = test.var]
-  setnames(ct, length(ct), tmpTV)
-  ct <- ct[ct[[tmpTV]] > 0L, diff(surv.int), keyby = eval(sum.to)]
-  setnames(ct, length(ct), tmpDiff)
-  ct <- ct[ct[[tmpDiff]] > 1L]
-  ## THE IDEA: if the difference in the number of the survival interval
-  ## is > 1, it means there is at least one row between two non-empty
-  ## intervals, i.e. non-consecutively.
-  
-  ## we keep non-consecutively bad surv.int stratas in entirety for inspection
-  if (nrow(ct) > 0L) {
-    msg <- paste0("The total person-time was zero in some survival intervals")
-    
-    if (!is.null(sum.to)) {
-      msg <- paste0(msg, ", when summed to the variable(s) ", 
-                    paste0("'", show.sum.to, "'", collapse = ", "),
-                    " (i.e. over all other variables, if any)") 
-    } else {
-      msg <- paste0(msg, " summed to the margins (over any stratifying ",
-                    "/ adjusting variables)")
-    }
-      
-    msg <- paste0(msg, " _non-consecutively_, i.e. some intervals after an ",
-                  "empty interval had person-time in them. ",
-                  "Keeping all survival ",
-                  "intervals with some estimates as NA for inspection.")
-    message(msg)
-  } else {
-    ## we leave out intervals that are empty consecutively (e.g. all from 5-10)
-    x[, c(tmpTV) := lapply(.SD, sum), by=c(sum.to, "surv.int"), .SDcols = test.var]
-    x <- x[x[[tmpTV]] > 0L]
-    setcolsnull(x, tmpTV)
-  }
-  
-  x
-}
-
-
-
-
-comp_e1 <- function(x, breaks, pophaz, survScale, by = NULL, id = "lex.id", immortal = TRUE, verbose = FALSE) {
-  ## INTENTION: given a Lexis data set x,
-  ## computes Ederer I expected survival curves till end of follow-up
-  ## by 'by' unless individual = TRUE.
-  TF <- environment()
-  
-  lex.dur <- haz <- surv.exp <- NULL # R CMD CHECK appeasement
-  ## check ---------------------------------------------------------------------
-  checkLexisData(x)
-  checkBreaksList(x, breaks)
-  ph <- data.table(pophaz)
-  tmpHaz <- makeTempVarName(x, pre = "pop_haz_")
-  setnames(ph, "haz", tmpHaz)
-  checkPophaz(x, ph, haz.name = tmpHaz)
-  
-  byErr <- paste0("Internal error (probably): work data did not have ",
-                  "variable(s) %%VARS%%. If your supplied data has them, ",
-                  "complain to the package maintainer.")
-  all_names_present(x, c(by, id, survScale), msg = byErr)
-  
-  if (length(id) != 1L) {
-    stop("Argument id must be of length 1.")
-  }
-  ## split ---------------------------------------------------------------------
-  pt <- proc.time()
-  oldBreaks <- attr(x, "breaks")
-  allScales <- attr(x, "time.scales")
-  if (!survScale %in% allScales) {
-    stop("survScale '", survScale, "' not a time scale in the Lexis object. ",
-         "(Possibly internal error - ensure you have that time scale in data. ",
-         "If not, complain to the package maintainer.")
-  }
-  keepVars <- c(allScales, "lex.dur", "lex.id", "lex.Cst", 
-                "lex.Xst", by, id, setdiff(names(ph), tmpHaz))
-  keepVars <- unique(keepVars)
-  y <- subsetDTorDF(x, select = keepVars)
-  y <- setDT(copy(y))
-  forceLexisDT(y, breaks = oldBreaks, allScales = allScales, key = TRUE)
-  
-  ## won't use statuses for anything
-  y[, c("lex.Cst", "lex.Xst") := NULL]
-  y[, c("lex.Cst", "lex.Xst") := 0L]
-  
-  if (immortal) {
-    ## set lex.dur to infinite. this assumes that the subjects never leave
-    ## follow-up (which Ederer I assumes)
-    storage.mode(y$lex.dur) <- "double"
-    id_last <- !duplicated(y, by = "lex.id", fromLast = TRUE)
-    y[TF$id_last, lex.dur := Inf]
-  }
-  
-  y <- intelliCrop(y, breaks = breaks, allScales = allScales)
-  y <- intelliDrop(y, breaks = breaks)
-  setDT(y)
-  forceLexisDT(y, breaks = oldBreaks, allScales = allScales, key = TRUE)
-  setkeyv(y, c(id, survScale))
-  
-  y <- splitMulti(y, breaks = breaks, drop = FALSE, merge = TRUE)
-  
-  if (verbose) cat("Time taken by splitting: ", timetaken(pt), ".\n", sep = "")
-  
-  ## merge pop haz -------------------------------------------------------------
-  pt <- proc.time()
-  mergeVars <- intersect(names(y), names(ph))
-  mergeScales <- intersect(allScales, mergeVars)
-  if (length(mergeScales) == 0L) mergeScales <- NULL
-  mergeCats <- setdiff(mergeVars, mergeScales)
-  if (length(mergeCats) == 0L) mergeCats <- NULL
-  
-  y <- cutLowMerge(y, ph, by = mergeVars, 
-                   mid.scales = mergeScales, old.nums = TRUE,
-                   all.x = TRUE, all.y = FALSE)
-  setDT(y)
-  if (verbose) cat("Time taken by merging pophaz: ", timetaken(pt), ".\n", sep = "")
-  
-  ## ederer I computation ------------------------------------------------------
-  ## prep temp surv.int var
-  tmpSI <- makeTempVarName(x, pre = "surv_int_")
-  setkeyv(y, c(by, id, allScales[1L]))
-  y[, c(tmpSI) := cut(y[[survScale]], breaks[[survScale]],
-                      right=FALSE,labels=FALSE)]
-  setkeyv(y, c(by, tmpSI))
-  
-  ## EDERER I: integral of the weighted average expected hazard,
-  ## where the weights are the subject-specific expected survival
-  ## probabilities.
-  ## 1) compute integral of hazard over an interval t_i by id
-  ##    (NOT cumulative hazard from zero till end of interval t_i)
-  ##    This sums over multiple rows a subject may have within one
-  ##    survival interval due to splitting by multiple time scales.
-  pt <- proc.time()
-  set(y, j = tmpHaz, value = y[[tmpHaz]]*y$lex.dur)
-  y <- y[, lapply(.SD, sum), keyby = eval(unique(c(by, id, tmpSI))), 
-         .SDcols = c(tmpHaz)]
-  setnames(y, ncol(y), tmpHaz)
-  
-  ## reverse temp names - need to be able to refer to haz without temp var
-  ## to enable correct cumsum() below
-  avoid <- c(names(y), tmpHaz, "surv.exp")
-  tmpBy <- makeTempVarName(names = avoid, pre = by)
-  tmpID <- makeTempVarName(names = c(avoid, tmpBy), pre = id)
-  if (length(by)) {
-    setnames(y, by, tmpBy)
-    if (id %in% by) tmpID <- id <- tmpBy[by == id]
-  } else {
-    tmpBy <- by <- NULL
-  }
-  setnames(y, id, tmpID)
-  setnames(y, tmpHaz, "haz")
-  if (verbose) cat("Time taken by 1): ", timetaken(pt), ".\n", sep = "")
-  
-  ## 2) expected cum.haz. over intervals t_1 -> t_i by id...
-  ##   (no cumulative exp.surv yet)
-  pt <- proc.time()
-  y[, surv.exp := cumsum(haz), by = eval(tmpID)]
-  
-  if (verbose) cat("Time taken by 2): ", timetaken(pt), ".\n", sep = "")
-  ## 3) cumulative surv.exp till end of interval t_i by id...
-  pt <- proc.time()
-  y[, surv.exp := exp(-surv.exp)]
-  
-  if (verbose) cat("Time taken by 3): ", timetaken(pt), ".\n", sep = "")
-  
-  ## 4) The Ederer I expected (marginal) survivals for intervals t_i 
-  pt <- proc.time()
-  y <- y[, .(surv.exp = mean(surv.exp)), by = eval(c(tmpBy, tmpSI))]
-  if (verbose) cat("Time taken by 4): ", timetaken(pt), ".\n", sep = "")
-  
-  if ("surv.exp" %in% by) {
-    by[by == "surv.exp"] <- makeTempVarName(y, pre = "surv.exp")
-  }
-  if (length(by)) setnames(y, tmpBy, by)
-  
-  setcolorder(y, c(by, tmpSI, "surv.exp"))
-  setkeyv(y, c(by, tmpSI))
-  setnames(y, tmpSI, survScale)
-  br <- breaks[[survScale]]
-  br <- br[-1]
-  y[, c(survScale) := br[y[[survScale]]]]
-  
-  
-  y[]
-}
-
-
-
-
-detectSurvivalTimeScale <- function(lex, values) {
-  
-  checkLexisData(lex)
-  
-  allScales <- attr(lex, "time.scales")
-  
-  allScVals <- lapply(allScales, function(ch) lex[[ch]])
-  names(allScVals) <- allScales
-  
-  whSurvScale <- lapply(allScVals, function(col) {
-    identical(col, values)
-  })
-  whSurvScale <- unlist(whSurvScale)
-  
-  if (sum(whSurvScale) == 0L) {
-    whSurvScale <- lapply(allScVals, function(col) {
-      isTRUE({
-        all.equal(col, values, scale = 1L, 
-                  check.attributes = FALSE,
-                  tolerance = .Machine$double.eps ^ 0.5)
-      })
-    })
-    whSurvScale <- unlist(whSurvScale)
-  }
-  if (sum(whSurvScale) == 0L) {
-    
-    dt <- as.data.table(allScVals)
-    dt <- cbind(dt, data.table(testValues = values))
-    on.exit(print(dt))
-    
-    stop("Could not determine which time scale was used. The formula MUST ",
-         "include the time scale used within a Surv() call (or a Surv object),",
-         " e.g. Surv(FUT, lex.Xst) ~ sex. Note that the 'time' argument is ",
-         "effectively (and exceptionally) used here to denote the times at ",
-         "the beginning of follow-up to identify the time scale existing in ",
-         "the supplied data to use. If you are sure you are mentioning a ",
-         "time scale in the formula in this manner, complain to the ",
-         "package maintainer. The table printed below contains the time ",
-         "scales tested against and the values that were supplied as the last ",
-         "column.")
-  }
-  survScale <- allScales[whSurvScale]
-  survScale
-  
-}
-
+globalVariables(c("tab", "SE.A", "pp.table"))
+
+comp.st.surv <- function(surv.var = "p", surv.expr = "1-d/(n.eff-n.cens/2)", 
+                         SE.expr = "sqrt(p*(1-p)/(n.eff))", cumu = TRUE) {
+  
+  function(surv.table = tab, surv.by.vars = NULL) {
+    
+    tabname <- deparse(substitute(surv.table))
+    #     tabname <- "surv.table"    
+    by.expr <- NULL
+    if (!is.null(surv.by.vars)) {
+      surv.by.vars <- paste0("'", surv.by.vars, "'", collapse = " ,")
+      by.expr <- paste0(", by = c( ", surv.by.vars, " )")
+    }
+    
+    surv.expr <-  paste0(surv.var, " := ", surv.expr)
+    tab.surv.expr <- paste0(tabname, "[, ", surv.expr,  by.expr, "]")
+    #   surv.expr <- parse(text=surv.expr)
+    #   surv.var <- parse(text=surv.var)
+    SE.var <- paste0("SE.", surv.var)
+    SE.expr <- paste0(SE.var, " := ", SE.expr)
+    tab.SE.expr <- paste0(tabname, "[, ", SE.expr, ", ", by.expr, "]")
+    #   SE.expr <- parse(text=SE.expr)
+    #   SE.var <- parse(text=SE.var)
+    cumuexpr <- paste0(tabname, "[, ", surv.var, " := cumprod(", surv.var, ")", by.expr, "]" )
+    
+    
+    ## parent.frame(2): two steps upstream in environments
+    pe <- function(obj, ..., env=parent.frame(2)) {
+      eval(parse(text=obj), ..., envir=env)
+    }
+    
+    pe(tab.surv.expr)
+    
+    
+    if (cumu) pe(cumuexpr)
+    
+    ## standard error
+    pe(tab.SE.expr)
+    
+    ## zero survival leads to zero SE
+    minexpr <- paste0(tabname, "[",surv.var, "== 0, ", SE.var, " := 0]")
+    pe(minexpr)
+  }
+}
+
+
+comp.st.surv.obs.lif <- comp.st.surv(surv.var=  "surv.obs",
+                                     surv.expr= "1-d/n.eff", 
+                                     SE.expr=   "surv.obs*sqrt(  cumsum(  d/(n.eff*(n.eff-d))  )  )", ## old : sqrt(p*(1-p)/(n.eff))"
+                                     cumu = TRUE)
+
+comp.st.surv.obs.haz <- comp.st.surv(surv.var=  "surv.obs", 
+                                     surv.expr= "exp(-delta*d/pyrs)", 
+                                     SE.expr=   "surv.obs*sqrt(  cumsum(  delta^2*d/pyrs^2  )  )", ## old: sqrt(p*(1-p)/(n.eff))
+                                     cumu = TRUE)
+
+
+comp.st.r.e2.haz <- comp.st.surv(surv.var = "r.e2",
+                                 surv.expr = "exp(-delta*(d-d.exp)/pyrs)",
+                                 SE.expr = "SE.surv.obs/surv.exp",
+                                 cumu=TRUE)
+
+comp.st.r.e2.lif <- comp.st.surv(surv.var = "r.e2",
+                                 surv.expr = "1-(d-d.exp)/(n.eff)",
+                                 SE.expr = "SE.surv.obs/surv.exp",
+                                 cumu=TRUE)
+
+comp.st.r.pp.haz <- comp.st.surv(surv.var = "r.pp",
+                                 surv.expr= "exp(-delta*(d.pp - d.exp.pp)/pyrs.pp)",
+                                 SE.expr = "r.pp*sqrt(  cumsum(  delta^2*( d.pp.2)/ pyrs.pp^2  )  )",
+                                 cumu=TRUE)
+
+comp.st.r.pp.lif <- comp.st.surv(surv.var = "r.pp",
+                                 surv.expr = "1-(d.pp-d.exp.pp)/(n.eff.pp)",
+                                 SE.expr = "r.pp*sqrt(  cumsum(  d.pp.2/(n.eff^2)  )  )",
+                                 cumu=TRUE)
+
+
+## this function will calculate confidence intervals
+## for obs & rel & net survivals
+#' @import stats
+comp.st.conf.ints <- function(tab = pp.table, al=0.05, surv="r.pp", transform ="log-log") {
+  zlo <- as.character(qnorm(al/2))
+  zhi <- as.character(qnorm(1-al/2))
+  SE.surv <- paste0("SE.",surv)
+  surv.hi <- paste0(surv, ".hi")
+  surv.lo <- paste0(surv, ".lo")
+  
+  tmp_se <- makeTempVarName(names = names(tab), pre = "delta_method_SE_", length = 15L)
+  
+  pe <- function(...) {
+    eval(parse(text=paste0(...)), envir=tab)
+  }
+  
+  if (transform =="plain") {
+    ## assume S(t)~N(mu, sigma)
+    
+    ex <- paste0(surv,  " ", zlo, "*", SE.surv)
+    tab[, (surv.lo)  := pe(ex)]
+    ex <- paste0(surv, " +", zhi, "*", SE.surv)
+    tab[, (surv.hi)  := pe(ex)]
+    
+  } else if (transform =="log-log") {
+    ## assume log(H(t))~N(mu, sigma)
+    ## -> delta method SE: sqrt( SE^2 * (1/(log(SURV)*SURV))^2 )
+    
+    ex <- paste0(SE.surv,"/(abs(log(",surv,"))*",surv,")")
+    tab[,  (tmp_se) := pe(ex)]
+    
+    ex <- paste0(surv, "^exp(", zhi, "*", tmp_se,")")
+    tab[, (surv.lo)  := pe(ex)]
+    ex <- paste0(surv, "^exp(", zlo, "*", tmp_se,")")
+    tab[, (surv.hi)  := pe(ex)]
+    
+  } else if (transform =="log") {
+    ## assume log(S(t))~N(mu, sigma)
+    ## -> delta method SE: SE/SURV
+    
+    ex <- paste0(SE.surv,"/",surv)
+    tab[,  (tmp_se) := pe(ex)]
+    
+    ex <- paste0(surv, "*exp(", zlo, "*", tmp_se,")")
+    tab[, (surv.lo)  := pe(ex)]
+    ex <- paste0(surv, "*exp(", zhi, "*", tmp_se,")")
+    tab[, (surv.hi)  := pe(ex)]
+  }
+  if (tmp_se %in% names(tab)) tab[, (tmp_se) := NULL]
+  
+  ## zero SE means zero uncertainty means lo=hi=estimate
+  tab[tab[[SE.surv]] == 0, c(surv.lo, surv.hi) := .SD, .SDcols = surv]
+  
+  tab[]
+}
+
+
+
+# x <- Lexis(data=sire[1,], entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
+#            exit=list(per=get.yrs(ex_date)), exit.status=status)
+# x <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 1/12), per=1994:2013, age = 0:150))
+# x[, surv.int := cut(fot, seq(0, 5, 1/12) - .Machine$double.eps^0.5, labels = FALSE)]
+# x <- cutLowMerge(x, popmort, by.x = c("sex","per", "age"), 
+#                  by.y = c("sex", "year", "agegroup"), 
+#                  mid.scales = c("per", "age"), all.x = TRUE, all.y = FALSE)
+# comp_pp_weights(x, surv.scale = "fot", breaks = seq(0, 5, 1/12), haz = "haz", style = "delta")
+
+comp_pp_weights <- function(lex, surv.scale = "fot", breaks = NULL, haz = "haz", style = "delta", verbose = FALSE) {
+  ppTime <- proc.time()
+  ## input: a split Lexis object (data.table) and the time scale to compute
+  ## pp weights over; lex must also contain 'haz', the population
+  ## (expected) hazard level for each row
+  TF <- environment()
+  
+  lex.dur <- lex.id <- pp <- NULL ## APPEASE R CMD CHECK
+  
+  style <- match.arg(style, c("delta", "actual"))
+  if (!is.data.table(lex)) stop("lex must be a data.table")
+  
+  all_names_present(lex, c(haz, surv.scale, "lex.id", "lex.dur"))
+  
+  if ("pp" %in% names(lex)) stop("Variable named 'pp' existed in data when attempting to compute pohar-perme weights; 'pp' is reserved so you should delete or rename that variable")
+  
+  
+  if (!identical(key(lex), c("lex.id", surv.scale))) stop("lex must be a data.table keyed by lex.id and the survival time scale")
+  
+  
+  breaks <- sort(unique(breaks)) - .Machine$double.eps^0.5
+  
+  ## need a bunch of temporary variable names to compute pp weights
+  ## inside the data set without overwriting anything existing.
+  ## this will take about 0.1 secs.
+  tmpSI <- makeTempVarName(data = lex, pre = "surv.int_")
+  tmpSIstart <- makeTempVarName(data = lex, pre = "surv.int.start_")
+  tmpSIstop <- makeTempVarName(data = lex, pre = "surv.int.stop_")
+  tmpSIlength <- makeTempVarName(data = lex, pre = "surv.int.length_")
+  on.exit(setcolsnull(lex, delete = c(tmpSI, tmpSIstart, tmpSIstop, tmpSIlength, tmpPS, tmpPCS, tmpPCSM)), add = TRUE)
+  
+  set(lex, j = tmpSI, value = cut(lex[[surv.scale]], breaks, labels = FALSE))
+  set(lex, j = tmpSIstop, value = breaks[-1][lex[[tmpSI]]])
+  set(lex, j = tmpSIstart, value = breaks[-length(breaks)][lex[[tmpSI]]])
+  set(lex, j = tmpSIlength, value = lex[[tmpSIstop]] - lex[[tmpSIstart]])
+  
+  tmpPS <- makeTempVarName(data = lex, pre = "pop.surv_")
+  tmpPCS <- makeTempVarName(data = lex, pre = "pop.cumsurv_")
+  tmpPCSM <- makeTempVarName(data = lex, pre = "pop.cumsurv.mid_")
+
+  ## conditional survs
+  if (verbose) condSurvTime <- proc.time()
+  
+  set(lex, j = tmpPS, value = exp(-lex[[haz]]*lex$lex.dur))
+  ## till end of each interval...
+  lex[, c(tmpPCS) := list(exp(-cumsum(.SD[[1L]]*lex.dur))),
+      by = lex.id, .SDcols = c(haz, "lex.dur")]
+  ## till start of each interval
+  set(lex, j = tmpPCS, value = lex[[tmpPCS]] / lex[[tmpPS]]) 
+  if (verbose) cat("Time taken by computing expected survivals up to start of each interval for each lex.id: ", timetaken(condSurvTime), "\n")
+  
+  ## pohar-perme weighting by expected cumulative survival. approximation:
+  ## cumulative survival up to either middle of remaining surv.int (not individual-specific)
+  ## or up to middle of subject's follow-up in each row (individual-specific)
+  ## difference: e.g. 2 rows within a surv.int have either the same or different pp-weights
+  if (style == "actual") {
+    set(lex, j = tmpPCSM, value = lex[[tmpPCS]] * (lex[[tmpPS]])^0.5)
+  }
+  if (style == "delta") {
+    if (verbose) deltaTime <- proc.time()
+    
+    setkeyv(lex, c("lex.id", tmpSI))
+    ## expected survival up to middle of remaining time in surv.int
+    ## cumulation starting from first record for subject in each surv.int
+    
+    ## some records are the only one for a lex.id in a surv.int; these are easy
+    first_in_surv.int <- !duplicated(lex, fromLast = FALSE, by = key(lex))
+    last_in_surv.int <- !duplicated(lex, fromLast = TRUE, by = key(lex))
+    only_in_surv.int <- first_in_surv.int & last_in_surv.int
+    
+    lex[only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * exp(.SD[[2L]] * (.SD[[3L]] - .SD[[4L]])/2L),
+        .SDcols = c(tmpPCS, haz, tmpSIstop, surv.scale)]
+    
+    ## more complicated with many records in a surv.int per lex.id
+    if (any(!only_in_surv.int)) {
+      
+      tmpSImid <- makeTempVarName(lex, pre = "surv.int.mid_")
+      dist <- makeTempVarName(lex, pre = "dist_")
+      on.exit(setcolsnull(lex, delete = c(dist, tmpSImid)), add = TRUE)
+      
+      ## middle point of survival interval
+      set(lex, j = tmpSImid, value = (lex[[tmpSIstop]] - lex[[tmpSIstart]])/2L)
+      
+      ## distance from remaining surv.int mid-point starting from start of 
+      ## record; or at most lex.dur; for integration
+      set(lex, j = dist, value = pmin(lex$lex.dur, lex[[tmpSImid]]))
+      ## some records after mid-point can have negative fot.dist at this point
+      set(lex, j = dist, value = pmax(lex[[dist]], 0))
+      
+      ## some lex.id are censored / die before mid of surv.int; last record
+      ## must reach its fot.dist at least up to the mid (or be zero due to above)
+      lex[last_in_surv.int, c(dist) := pmax((.SD[[1L]] - .SD[[2L]])/2L, 0), 
+          .SDcols = c(tmpSIstop, surv.scale)]
+      
+      byTime <- proc.time()
+      
+      ## from start of first in surv.int till mid point
+      ## step by step for efficiency...
+      lex[!only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * .SD[[2L]], .SDcols = c(haz, dist)]
+      lex[!only_in_surv.int, c(tmpPCSM) := lapply(.SD, sum), 
+          .SDcols = c(tmpPCSM), by = c("lex.id", tmpSI)]
+      lex[!only_in_surv.int, c(tmpPCS) := .SD[1L], 
+          by = c("lex.id", tmpSI), .SDcols = c(tmpPCS)]
+      lex[!only_in_surv.int, c(tmpPCSM) := .SD[[1L]] * exp(-.SD[[2L]]), 
+          .SDcols = c(tmpPCS, tmpPCSM)]
+      
+      ## todo: alternate faster method for integration!
+      setcolsnull(lex, delete = c(dist))
+      if (verbose) cat("Time taken by extra computation due to style 'delta': ", timetaken(deltaTime), "\n")
+    }
+    
+    rm(first_in_surv.int, last_in_surv.int, only_in_surv.int)
+    if (verbose) cat("Time taken by computation of Pohar-Perme weights: ", timetaken(ppTime), "\n")
+  }
+  
+  setkeyv(lex, c("lex.id", surv.scale))
+  
+  lex[, pp := 1/.SD, .SDcols = tmpPCSM]
+  
+  invisible(lex[])
+}
+
+comp_pp_weighted_figures <- function(lex, haz = "haz", pp = "pp", event.ind = NULL, by = "lex.id") {
+  ## PURPOSE: given a split Lexis object with a column of pre-computed
+  ## pohar-perme weights, computes pp-weighted:
+  ## * person-time (lex.dur*pp)
+  ## * at-risk indicator
+  ## * event counts
+  ## * event counts multiplied with pp^2
+  ## * expected event counts
+  ## events are transitions and end points as detected by detectEvents,
+  ## and include censorings.
+  ## OUTPUT: a DT of pp-weighted things.
+  
+  ## appease R CMD CHECK
+  lex.Cst <- lex.Xst <- NULL
+  
+  checkLexisData(lex, check.breaks = TRUE)
+  if (!is.data.table(lex)) stop("lex must be a data.table")
+  
+  all_names_present(lex, c(pp, haz, event.ind, by))
+  
+  ## NOTE: we want to avoid conflicts with possible variable names in lex
+  ## (e.g. an existing column named ptime.pp might conflict with ptime.pp char vec)
+  e <- environment()
+  
+  if (is.null(event.ind)) {
+    event.ind <- makeTempVarName(lex, pre = "event_indicator_")
+    on.exit(setcolsnull(lex, delete = event.ind, soft = TRUE), add = TRUE)
+    set(lex, j = event.ind, value = detectEvents(lex, breaks = attr(lex, "breaks"), by = by))
+    all_names_present(lex, event.ind)
+  }
+  ## data.table is probably faster in this than simply using vectors
+  idt <- data.table(1:2)
+  names(idt) <- event.ind
+  haveEvents <- sort(lex[idt, on = event.ind, which = TRUE])
+  evtab <- lex[haveEvents, .(obs = .N), by = list(lex.Cst, lex.Xst)]
+  set(evtab, j = "obs", value = NULL)
+  
+  events <- paste0("from", evtab$lex.Cst, "to", evtab$lex.Xst)
+  ppVars <- c(paste0(events, ".pp"), "ptime.pp", "d.exp.pp")
+  
+  
+#   ## build table to join with lex to limit to rows with events -----------------
+#   ## non-event-rows have zero valued pp-weighted event counts, naturally
+#   set(evtab, j = event.ind, value = NULL)
+#   evtab <- rbindlist(list(evtab, evtab))
+#   set(evtab, j = event.ind, value = rep(1:2, each = nrow(evtab)/2L))
+#   setkeyv(evtab, c("lex.Cst", "lex.Xst", event.ind))
+  
+  ## pp-weighted events --------------------------------------------------------
+  ## NOTE: a pp-weighted event is simply the pp weight where the event occurs
+  ## and zero otherwise (0L/1L times pp-weight)
+  
+  evN <- length(events)
+  evdt <- data.table(rn = rep(1:nrow(lex), times = evN))
+  set(evdt, j = "eventType", value = factor(rep(1:evN, each = nrow(lex)), levels = 1:evN, labels = events))
+  set(evdt, j = "pp", value = rep(lex[[pp]],length.out=nrow(evdt)))
+  
+  
+  ## need to still determine which rows are not their eventType's events -------
+  
+  rnVar <- makeTempVarName(lex, pre = "rowNum_")
+  on.exit(setcolsnull(lex, delete = rnVar, soft = TRUE), add = TRUE)
+  set(lex, j = rnVar, value = 1:nrow(lex))
+  
+  ## row numbers in lex by event type
+  rowNums <- lex[haveEvents][evtab, .(rowNum = .SD[[e$rnVar]]), by = .EACHI, on = c("lex.Cst", "lex.Xst"), .SDcols = rnVar]
+  ## multiply to accommodate expanded evdt data
+  
+  rowNum <- NULL ## appease R CMD CHECK
+  rowNums[, rowNum := rowNum+(.GRP-1L)*nrow(lex), by = list(lex.Cst, lex.Xst)]
+  noEvents <- setdiff(1:nrow(evdt), rowNums$rowNum)
+  
+  set(evdt, i = noEvents, j = "pp", value = 0)
+  
+  evdt <- dcast.data.table(evdt, rn ~ eventType, value.var = "pp")
+  
+  setorderv(evdt, "rn")
+  set(evdt, j = "rn", value = NULL)
+  
+  ## ptime.pp & d.exp.pp -------------------------------------------------------
+  set(evdt, j = "ptime.pp", value = lex$lex.dur * lex[[pp]])
+  set(evdt, j = "d.exp.pp", value = lex$lex.dur * lex[[haz]] * lex[[pp]])
+  
+  setcolorder(evdt, c("ptime.pp", "d.exp.pp", sort(events)))
+  setnames(evdt, events, paste0(events, ".pp"))
+  
+  ## pp-weighted at-risk indicators --------------------------------------------
+  ## these will be n.pp at the aggregate level.
+  set(evdt, j = "at.risk.pp", value = lex[[pp]]*1L)
+  
+  ## events multiplied with pp-weight again ------------------------------------
+  ## (i.e. event times pp squared)
+  
+  set(evdt, j = paste0(events, ".pp.2"), 
+      value = evdt[, .SD, .SDcols = paste0(events, ".pp")]*lex[[pp]])
+  
+  return(evdt[])
+  
+}
+
+
+
+test_empty_surv_ints <- function(x, by = NULL, sum.over = NULL, test.var = "pyrs", show.by = NULL) {
+  
+  x <- copy(x)
+  oc <- class(x)
+  setDT(x)
+  setattr(x, "class", oc)
+  
+  surv.int <- NULL ## APPEASE R CMD CHECK
+  
+  all_names_present(x, by, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
+  all_names_present(x, sum.over, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
+  all_names_present(x, test.var, msg = "Missing variable(s) %%VARS%% from data when inspected for empty survival intervals. If you see this, send a message to the package maintainer.")
+  
+  if (any(!sum.over %in% by)) stop("sum.over must be a subset of by.")
+  if (length(show.by) == 0L) show.by <- by
+  if (length(by) != length(show.by)) {
+    stop("Internal error: length(sum.over) != length(show.sum.over). ",
+         "If you see this, complain to the package maintainer.")
+  }
+  
+  wh_sum.to <- !by %in% sum.over
+  sum.to <- by[wh_sum.to]
+  show.sum.to <- show.by[wh_sum.to]
+  
+  if (length(sum.to) == 0L) sum.to <- show.sum.to <- NULL
+  
+  tmpTV <- makeTempVarName(x, pre = "testValues_")
+  tmpDiff <- makeTempVarName(x, pre = "diff_")
+  on.exit(setcolsnull(x, delete = c(tmpTV, tmpDiff)))
+  
+  ## first check empty surv.ints are all consecutive...
+  ## consecutiveness: e.g. out of 10 surv ints, 6-10 are empty.
+  ## non-consecutiveness: e.g. out of 10 surv ints, 6-8 are empty.
+  ## (then 9-10 will have NA estimates as well.)
+  setkeyv(x, c(by, "surv.int"))
+  
+  ct <- x[, lapply(.SD, sum), keyby = c(sum.to, "surv.int"), .SDcols = test.var]
+  setnames(ct, length(ct), tmpTV)
+  ct <- ct[ct[[tmpTV]] > 0L, diff(surv.int), keyby = eval(sum.to)]
+  setnames(ct, length(ct), tmpDiff)
+  ct <- ct[ct[[tmpDiff]] > 1L]
+  ## THE IDEA: if the difference in the number of the survival interval
+  ## is > 1, it means there is at least one row between two non-empty
+  ## intervals, i.e. non-consecutively.
+  
+  ## we keep non-consecutively bad surv.int stratas in entirety for inspection
+  if (nrow(ct) > 0L) {
+    msg <- paste0("The total person-time was zero in some survival intervals")
+    
+    if (!is.null(sum.to)) {
+      msg <- paste0(msg, ", when summed to the variable(s) ", 
+                    paste0("'", show.sum.to, "'", collapse = ", "),
+                    " (i.e. over all other variables, if any)") 
+    } else {
+      msg <- paste0(msg, " summed to the margins (over any stratifying ",
+                    "/ adjusting variables)")
+    }
+      
+    msg <- paste0(msg, " _non-consecutively_, i.e. some intervals after an ",
+                  "empty interval had person-time in them. ",
+                  "Keeping all survival ",
+                  "intervals with some estimates as NA for inspection.")
+    message(msg)
+  } else {
+    ## we leave out intervals that are empty consecutively (e.g. all from 5-10)
+    x[, c(tmpTV) := lapply(.SD, sum), by=c(sum.to, "surv.int"), .SDcols = test.var]
+    x <- x[x[[tmpTV]] > 0L]
+    setcolsnull(x, tmpTV)
+  }
+  
+  x
+}
+
+
+
+
+comp_e1 <- function(x, breaks, pophaz, survScale, by = NULL, id = "lex.id", immortal = TRUE, verbose = FALSE) {
+  ## INTENTION: given a Lexis data set x,
+  ## computes Ederer I expected survival curves till end of follow-up
+  ## by 'by' unless individual = TRUE.
+  TF <- environment()
+  
+  lex.dur <- haz <- surv.exp <- NULL # R CMD CHECK appeasement
+  ## check ---------------------------------------------------------------------
+  checkLexisData(x)
+  checkBreaksList(x, breaks)
+  ph <- data.table(pophaz)
+  tmpHaz <- makeTempVarName(x, pre = "pop_haz_")
+  setnames(ph, "haz", tmpHaz)
+  checkPophaz(x, ph, haz.name = tmpHaz)
+  
+  byErr <- paste0("Internal error (probably): work data did not have ",
+                  "variable(s) %%VARS%%. If your supplied data has them, ",
+                  "complain to the package maintainer.")
+  all_names_present(x, c(by, id, survScale), msg = byErr)
+  
+  if (length(id) != 1L) {
+    stop("Argument id must be of length 1.")
+  }
+  ## split ---------------------------------------------------------------------
+  pt <- proc.time()
+  oldBreaks <- attr(x, "breaks")
+  allScales <- attr(x, "time.scales")
+  if (!survScale %in% allScales) {
+    stop("survScale '", survScale, "' not a time scale in the Lexis object. ",
+         "(Possibly internal error - ensure you have that time scale in data. ",
+         "If not, complain to the package maintainer.")
+  }
+  keepVars <- c(allScales, "lex.dur", "lex.id", "lex.Cst", 
+                "lex.Xst", by, id, setdiff(names(ph), tmpHaz))
+  keepVars <- unique(keepVars)
+  y <- subsetDTorDF(x, select = keepVars)
+  y <- setDT(copy(y))
+  forceLexisDT(y, breaks = oldBreaks, allScales = allScales, key = TRUE)
+  
+  ## won't use statuses for anything
+  y[, c("lex.Cst", "lex.Xst") := NULL]
+  y[, c("lex.Cst", "lex.Xst") := 0L]
+  
+  if (immortal) {
+    ## set lex.dur to infinite. this assumes that the subjects never leave
+    ## follow-up (which Ederer I assumes)
+    storage.mode(y$lex.dur) <- "double"
+    id_last <- !duplicated(y, by = "lex.id", fromLast = TRUE)
+    y[TF$id_last, lex.dur := Inf]
+  }
+  
+  y <- intelliCrop(y, breaks = breaks, allScales = allScales)
+  y <- intelliDrop(y, breaks = breaks)
+  setDT(y)
+  forceLexisDT(y, breaks = oldBreaks, allScales = allScales, key = TRUE)
+  setkeyv(y, c(id, survScale))
+  
+  y <- splitMulti(y, breaks = breaks, drop = FALSE, merge = TRUE)
+  
+  if (verbose) cat("Time taken by splitting: ", timetaken(pt), ".\n", sep = "")
+  
+  ## merge pop haz -------------------------------------------------------------
+  pt <- proc.time()
+  mergeVars <- intersect(names(y), names(ph))
+  mergeScales <- intersect(allScales, mergeVars)
+  if (length(mergeScales) == 0L) mergeScales <- NULL
+  mergeCats <- setdiff(mergeVars, mergeScales)
+  if (length(mergeCats) == 0L) mergeCats <- NULL
+  
+  y <- cutLowMerge(y, ph, by = mergeVars, 
+                   mid.scales = mergeScales, old.nums = TRUE,
+                   all.x = TRUE, all.y = FALSE)
+  setDT(y)
+  if (verbose) cat("Time taken by merging pophaz: ", timetaken(pt), ".\n", sep = "")
+  
+  ## ederer I computation ------------------------------------------------------
+  ## prep temp surv.int var
+  tmpSI <- makeTempVarName(x, pre = "surv_int_")
+  setkeyv(y, c(by, id, allScales[1L]))
+  y[, c(tmpSI) := cut(y[[survScale]], breaks[[survScale]],
+                      right=FALSE,labels=FALSE)]
+  setkeyv(y, c(by, tmpSI))
+  
+  ## EDERER I: integral of the weighted average expected hazard,
+  ## where the weights are the subject-specific expected survival
+  ## probabilities.
+  ## 1) compute integral of hazard over an interval t_i by id
+  ##    (NOT cumulative hazard from zero till end of interval t_i)
+  ##    This sums over multiple rows a subject may have within one
+  ##    survival interval due to splitting by multiple time scales.
+  pt <- proc.time()
+  set(y, j = tmpHaz, value = y[[tmpHaz]]*y$lex.dur)
+  y <- y[, lapply(.SD, sum), keyby = eval(unique(c(by, id, tmpSI))), 
+         .SDcols = c(tmpHaz)]
+  setnames(y, ncol(y), tmpHaz)
+  
+  ## reverse temp names - need to be able to refer to haz without temp var
+  ## to enable correct cumsum() below
+  avoid <- c(names(y), tmpHaz, "surv.exp")
+  tmpBy <- makeTempVarName(names = avoid, pre = by)
+  tmpID <- makeTempVarName(names = c(avoid, tmpBy), pre = id)
+  if (length(by)) {
+    setnames(y, by, tmpBy)
+    if (id %in% by) tmpID <- id <- tmpBy[by == id]
+  } else {
+    tmpBy <- by <- NULL
+  }
+  setnames(y, id, tmpID)
+  setnames(y, tmpHaz, "haz")
+  if (verbose) cat("Time taken by 1): ", timetaken(pt), ".\n", sep = "")
+  
+  ## 2) expected cum.haz. over intervals t_1 -> t_i by id...
+  ##   (no cumulative exp.surv yet)
+  pt <- proc.time()
+  y[, surv.exp := cumsum(haz), by = eval(tmpID)]
+  
+  if (verbose) cat("Time taken by 2): ", timetaken(pt), ".\n", sep = "")
+  ## 3) cumulative surv.exp till end of interval t_i by id...
+  pt <- proc.time()
+  y[, surv.exp := exp(-surv.exp)]
+  
+  if (verbose) cat("Time taken by 3): ", timetaken(pt), ".\n", sep = "")
+  
+  ## 4) The Ederer I expected (marginal) survivals for intervals t_i 
+  pt <- proc.time()
+  y <- y[, .(surv.exp = mean(surv.exp)), by = eval(c(tmpBy, tmpSI))]
+  if (verbose) cat("Time taken by 4): ", timetaken(pt), ".\n", sep = "")
+  
+  if ("surv.exp" %in% by) {
+    by[by == "surv.exp"] <- makeTempVarName(y, pre = "surv.exp")
+  }
+  if (length(by)) setnames(y, tmpBy, by)
+  
+  setcolorder(y, c(by, tmpSI, "surv.exp"))
+  setkeyv(y, c(by, tmpSI))
+  setnames(y, tmpSI, survScale)
+  br <- breaks[[survScale]]
+  br <- br[-1]
+  y[, c(survScale) := br[y[[survScale]]]]
+  
+  
+  y[]
+}
+
+
+
+
+detectSurvivalTimeScale <- function(lex, values) {
+  
+  checkLexisData(lex)
+  
+  allScales <- attr(lex, "time.scales")
+  
+  allScVals <- lapply(allScales, function(ch) lex[[ch]])
+  names(allScVals) <- allScales
+  
+  whSurvScale <- lapply(allScVals, function(col) {
+    identical(col, values)
+  })
+  whSurvScale <- unlist(whSurvScale)
+  
+  if (sum(whSurvScale) == 0L) {
+    whSurvScale <- lapply(allScVals, function(col) {
+      isTRUE({
+        all.equal(col, values, scale = 1L, 
+                  check.attributes = FALSE,
+                  tolerance = .Machine$double.eps ^ 0.5)
+      })
+    })
+    whSurvScale <- unlist(whSurvScale)
+  }
+  if (sum(whSurvScale) == 0L) {
+    
+    dt <- as.data.table(allScVals)
+    dt <- cbind(dt, data.table(testValues = values))
+    on.exit(print(dt))
+    
+    stop("Could not determine which time scale was used. The formula MUST ",
+         "include the time scale used within a Surv() call (or a Surv object),",
+         " e.g. Surv(FUT, lex.Xst) ~ sex. Note that the 'time' argument is ",
+         "effectively (and exceptionally) used here to denote the times at ",
+         "the beginning of follow-up to identify the time scale existing in ",
+         "the supplied data to use. If you are sure you are mentioning a ",
+         "time scale in the formula in this manner, complain to the ",
+         "package maintainer. The table printed below contains the time ",
+         "scales tested against and the values that were supplied as the last ",
+         "column.")
+  }
+  survScale <- allScales[whSurvScale]
+  survScale
+  
+}
+
diff --git a/R/utility_functions.R b/R/utility_functions.R
index ccd3817..e3070e0 100644
--- a/R/utility_functions.R
+++ b/R/utility_functions.R
@@ -1,1322 +1,1322 @@
-
-
-#' @title Cast \code{data.table}/\code{data.frame} from long format to wide format
-#' @author Matti Rantanen, Joonas Miettinen
-#'
-#' @description 
-#' Convenience function for using \code{\link[data.table]{dcast.data.table}};
-#' inputs are character strings (names of variables) instead of a formula.
-#' 
-#' @param data a \code{data.table} or \code{data.frame}
-#' @param columns a character string vector; the (unique combinations of the) 
-#' levels of these variable will be different rows
-#' @param rows a character string vector; the (unique combinations of the) 
-#' levels of these variable will be different columns
-#' @param values a character string; the variable which will be represented
-#' on rows and columns as specified by \code{columns} and \code{rows}
-#' @import data.table
-#' @import stats
-#' @export cast_simple
-#' @details This function is just a small interface for \code{dcast} / 
-#' \code{dcast.data.table} and less flexible than the originals.
-#' 
-#' Note that all \code{data.table} objects are also \code{data.frame} 
-#' objects, but that each have their own \code{dcast} method.
-#' \code{\link[data.table]{dcast.data.table}} is faster.
-#' 
-#' If any values in \code{value.vars} need to be 
-#' aggregated, they are aggregated using \code{sum}.
-#' See \code{?dcast}.
-#' @return 
-#' A `data.table` just like `[data.table::dcast]`.
-#' 
-#' @examples 
-#' library("data.table")
-#' ## e.g. silly counts from a long-format table to a wide format
-#' test <- data.table::copy(popEpi::sire)
-#' test$dg_y <- year(test$dg_date)
-#' test$ex_y <- year(test$ex_date)
-#' tab <- ltable(test, c("dg_y","ex_y"))
-#' cast_simple(tab, columns='dg_y', rows="ex_y", values="obs")
-#' 
-
-
-cast_simple <- function(data=NULL, columns=NULL, rows=NULL, values=NULL) {
-  if (!is.data.frame(data)) stop("data needs be a data.frame or data.table")
-  if (is.null(data) || nrow(data) == 0L) stop("data NULL or has no rows")
-  
-  if (is.null(columns)) stop("columns cannot be NULL")
-  
-  msg <- paste0("Missing 'columns' variables: %%VARS%%")
-  all_names_present(data, columns, msg = msg)
-  msg <- paste0("Missing 'rows' variables: %%VARS%%")
-  all_names_present(data, rows, msg = msg)
-  msg <- paste0("Missing 'values' variables: %%VARS%%")
-  all_names_present(data, values, msg = msg)
-  
-  ## allow rows = NULL 
-  rowsNULL <- FALSE
-  if (is.null(rows)) rowsNULL <- TRUE
-  if (rowsNULL) rows <- "1"
-  
-  ## sometimes rows names appear to be like expressions, e.g. 'factor(V1)'
-  ## (and this function only uses string-column-names, so that's fine.)
-  actualRows <- rows
-  if (length(rows) > 1L || rows != "1") {
-    rows <- makeTempVarName(names = c(names(data), columns), 
-                            pre = paste0("RN", 1:length(rows)))
-    on.exit(setnames(data, rows, actualRows), add = TRUE)
-    setnames(data, actualRows, rows)
-  }
-  ## same for cols
-  actualCols <- columns
-  columns <- makeTempVarName(names = c(names(data), rows), 
-                             pre = paste0("CN", 1:length(columns)))
-  on.exit(setnames(data, columns, actualCols), add = TRUE)
-  setnames(data, actualCols, columns)
-  
-  form <- paste0(paste0(rows, collapse = " + "), " ~ ", 
-                 paste0(columns, collapse = " + "))
-  form <- as.formula(form)
-  
-  ## note: dcast probably usually finds the methods for data.frame / data.table,
-  ## but this method is more certain
-  if (is.data.table(data)) {
-    d <- dcast.data.table(data, formula = form, value.var=values, 
-                          drop=FALSE, fun.aggregate=sum)[]
-  } else {
-    d <- dcast(data, formula = form, value.var = values, 
-               drop = FALSE, fun.aggregate = sum)[]
-  }
-  if (rowsNULL) set(d, j = names(d)[1L], value = NULL)
-  wh_rows <- which(rows %in% names(d))
-  if (sum(wh_rows, na.rm = TRUE)) setnames(d, rows[wh_rows], actualRows[wh_rows])
-  
-  d
-}
-
-
-#' @title Convert NA's to zero in data.table
-#' @author Joonas Miettinen
-#' @description Given a \code{data.table DT}, replaces any \code{NA} values
-#' in the variables given in \code{vars} in \code{DT}. Takes a copy of the 
-#' original data and returns the modified copy.
-#' @import data.table
-#' @param DT \code{data.table} object
-#' @param vars a character string vector of variables names in \code{DT};
-#' if \code{NULL}, uses all variable names in \code{DT}
-#' @export na2zero
-#' @details Given a \code{data.table} object, converts \code{NA} values
-#' to numeric (double) zeros for all variables named in \code{vars} or
-#' all variables if \code{vars = NULL}.
-#' @return 
-#' A copy of `DT` where `NA` values have been replaced with zero.
-na2zero = function(DT, vars = NULL) { 
-  if (!data.table::is.data.table(DT)) stop("DT must be a data.table")
-  DT <- data.table::copy(DT)
-  
-  navars <- vars
-  if (is.null(navars)) navars <- names(DT)
-  all_names_present(DT, navars)
-  for (k in navars) {
-    DT[is.na(get(k)), (k) := 0]
-  }
-  
-  return(DT[])
-}
-
-
-#' @title Convert factor variable to numeric 
-#' @description Convert factor variable with numbers as levels into a numeric variable
-#' @param x a factor variable with numbers as levels
-#' @export fac2num
-#' @details
-#' For example, a factor with levels \code{c("5","7")} is converted into 
-#' a numeric variable with values \code{c(5,7)}.
-#' @seealso
-#' \code{\link{robust_values}}
-#' @source 
-#' \href{https://stackoverflow.com/questions/3418128/how-to-convert-a-factor-to-an-integer-numeric-without-a-loss-of-information}{Stackoverflow thread}
-#' @examples
-#' ## this is often not intended
-#' as.numeric(factor(c(5,7))) ## result: c(1,2)
-#' ## but this
-#' fac2num(factor(c(5,7))) ## result: c(5,7)
-#' 
-#' ## however
-#' as.numeric(factor(c("5","7","a"))) ## 1:3
-#' 
-#' suppressWarnings(
-#'   fac2num(factor(c("5","7","a"))) ## c(5,7,NA)
-#' )
-#' 
-#' 
-#' @return
-#' A numeric vector based on the levels of `x`.
-fac2num <- function(x) {
-  as.numeric(levels(x))[x]
-}
-
-
-
-
-#' @title Detect leap years
-#' @author Joonas Miettinen
-#' @description Given a vector or column of year values (numeric or integer), \code{\link{is_leap_year}} returns a vector of equal length
-#' of logical indicators, i.e. a vector where corresponding leap years have value TRUE, and FALSE otherwise.
-#' 
-#' @param years a vector or column of year values (numeric or integer)
-#' @examples
-#' ## can be used to assign new columns easily, e.g. a dummy indicator column
-#' df <- data.frame(yrs=c(1900,1904,2005,1995))
-#' df$lyd <- as.integer(is_leap_year(df$yrs))
-#' 
-#' ## mostly it is useful as a condition or to indicate which rows have leap years
-#' which(is_leap_year(df$yrs)) # 2
-#' df[is_leap_year(df$yrs),] # 2nd row
-#' 
-#' @export is_leap_year
-#' @return
-#' A `logical` vector where `TRUE` indicates a leap year.
-is_leap_year <- function(years) {
-  if (!is.numeric(years)) {
-    stop("years must be a numeric vector, preferably integer for speed. Use e.g. as.integer().")
-  }
-  
-  years <- try2int(years)
-  if (!is.integer(years)) stop("years could not be coerced to integer; don't use fractional years such as 2000.1234 but integers such as 2000")
-  
-  # divisible by four
-  isLeap <- years %% 4L == 0L
-  # not divisible by 100
-  isLeap <- isLeap & years %% 100L != 0L
-  # unless divisible by 400 also
-  isLeap <- isLeap | years %% 400L == 0L
-  isLeap
-  
-}
-#' @title Test if object is a \code{Date} object
-#' @description Tests if an object is a \code{Date} object and returns
-#' a logical vector of length 1. \code{IDate} objects are also 
-#' \code{Date} objects, but \code{date} objects from package \pkg{date}
-#' are not. 
-#' @author Joonas Miettinen
-#' @param obj object to test on
-#' @export is.Date
-#' @seealso
-#' \code{\link{get.yrs}}, \code{\link{is_leap_year}}, \code{\link{as.Date}}
-#' @return
-#' `TRUE` if `obj` is of class `"Date"` or `"IDate"`.
-is.Date <- function(obj) {
-  if (any(c("IDate", "Date") %in% class(obj))) {
-    return(TRUE)
-  }
-  return(FALSE)
-}
-
-
-#' @title Convert values to numeric robustly
-#' @author Joonas Miettinen
-#' 
-#' @param num.values values to convert to numeric
-#' @param force logical; if \code{TRUE}, returns a vector of values where values that cannot be interpreted as numeric are
-#' set to \code{NA}; if \code{FALSE}, returns the original vector and gives a warning if any value cannot be interpreted as
-#' numeric.
-#' @param messages logical; if \code{TRUE}, returns a message of what was done with the \code{num.values}
-#' @description Brute force solution for ensuring a variable is numeric by 
-#' coercing a variable of any type first to factor and then to numeric
-#' @export robust_values
-#' @import data.table
-#' @note
-#' Returns \code{NULL} if given \code{num.values} is \code{NULL}. 
-#' @examples
-#' ## this works
-#' values <- c("1", "3", "5")
-#' values <- robust_values(values)
-#' 
-#' ## this works
-#' values <- c("1", "3", "5", NA)
-#' values <- robust_values(values)
-#' 
-#' ## this returns originals and throws warnings
-#' values <- c("1", "3", "5", "a")
-#' suppressWarnings(
-#'   values <- robust_values(values)
-#' )
-#' 
-#' 
-#' ## this forces "a" to NA and works otherwise; throws warning about NAs
-#' values <- c("1", "3", "5", "a")
-#' suppressWarnings(
-#'   values <- robust_values(values, force=TRUE)
-#' )
-#' 
-#' @return
-#' A numeric vector.
-
-robust_values <- function(num.values, force = FALSE, messages = TRUE) {
-  a <- NULL
-  if (is.null(num.values)) {
-    return(NULL)
-  }
-  dt <- data.table(num.values)
-  nas <- dt[is.na(num.values), .N]
-  
-  suppressWarnings(
-    dt[,a := fac2num(factor(num.values))]
-  )
-  dt[, a := try2int(a)]
-  nas2 <- dt[is.na(a), .N]
-  
-  if (!force & nas2 > nas) {
-    if (messages) warning("since force = FALSE and NAs were created, returning original values")
-    return(dt$num.values)
-  }
-  if (force) {
-    if (nas2 > nas) {
-      if (messages) warning("some NAs were created")
-    }
-    return(dt$a)
-  }
-  
-  
-  return(dt$a)
-  
-  
-}
-
-#' @title Check if all names are present in given data
-#' @author Joonas Miettinen
-#' @param data dataset where the variable names should be found
-#' @param var.names a character vector of variable names, e.g.
-#' \code{c("var1", "var2")}
-#' @param stops logical, stop returns exception
-#' @param msg Custom message to return instead of default message.
-#' Special: include \code{\%\%VARS\%\%} in message string and the missing 
-#' variable names will be inserted there (quoted, separated by comma, e.g. 
-#' \code{'var1'}, \code{'var2'} --- no leading or tracing white space). 
-#' @description Given a character vector, checks if all names are present in \code{names(data)}.
-#' Throws error if \code{stops=TRUE}, else returns \code{FALSE} if some variable name is not present.
-#' @seealso
-#' \code{\link{robust_values}}
-#' @export all_names_present
-#' @return 
-#' `TRUE` if all `var.names` are in `data`, else `FALSE`,
-
-all_names_present <- function(data, var.names, stops = TRUE, msg = NULL) {
-  
-  if (!is.null(var.names) && !is.character(var.names)) {
-    stop("Argument 'var.names' must be NULL or a character vector of ",
-         "variable names.")
-  }
-  if (length(var.names) && any(is.na(var.names))) {
-    stop("There are ", sum(is.na(var.names)), " missing values in argument ",
-         "'var.names'. Please only supply non-NA values.")
-  }
-  
-  badNames <- setdiff(var.names, names(data))
-  if (length(badNames) == 0L) return(TRUE)
-  
-  badNames <- paste0("'", badNames, "'", collapse = ", ")
-  
-  if (is.null(msg)) msg <- paste0("Cannot proceed - following given variable name(s) not present in dataset '",
-                                  deparse(substitute(data)), "': ", badNames)
-  if (!is.character(msg) || length(msg) > 1L) stop("Argument 'msg' must be a character string vector of length one.") else
-    msg <- gsub(pattern = "%%VARS%%", replacement = badNames, x = msg)
-  if (!is.logical(stops) || length(stops) > 1L) stop("Argument 'stops' must be either TRUE or FALSE.")
-  
-  if (stops) stop(msg)
-  
-  return(FALSE)
-}
-
-
-#' @title Return lower_bound value from char string (20,30]
-#' @author Matti Rantanen
-#' @description selects lowest values of each factor after cut() based
-#' on that the value starts from index 2 and end in comma ",".
-#' @param cut is a character vector of elements "(20,60]"
-#' @export lower_bound
-#' @return
-#' A numeric vector.
-
-lower_bound <- function(cut) {
-  cut <- as.character(cut)
-  ind <- gregexpr(pattern=',',cut)
-  ind <- as.numeric(ind) - 1
-  t.sub <- as.numeric(substr(cut,2, ind))
-  return(t.sub)
-}
-
-
-#' @title Change output values from cut(..., labels = NULL) output
-#' @author Matti Rantanen
-#' @param t is a character vector of elements, e.g. "(20,60]"
-#' @param factor logical; TRUE returns informative character string, FALSE numeric (left value)
-#' @description Selects lowest values of each factor after cut() based
-#' on the assumption that the value starts from index 2 and end in comma ",".
-#' @details type = 'factor': "[50,52)" -> "50-51" OR "[50,51)" -> "50"
-#' 
-#' type = 'numeric': lowest bound in numeric.
-#' 
-#' @export cut_bound
-#' @return
-#' If `factor = TRUE`, returns a character vector; else returns a numeric 
-#' vector.
-#' @examples
-#' cut_bound("[1900, 1910)") ## "1900-1909"
-
-cut_bound <- function(t, factor=TRUE) {
-  if (!factor) {
-    t <- as.character(t)
-    ind <- gregexpr(pattern=',',t)
-    ind <- as.numeric(ind) - 1
-    t <- as.numeric(substr(t,2, ind))
-    return(t)
-  }
-  if (factor) {
-    t <- as.character(t)
-    t <- gsub(',', '-' , substr(t, 2, nchar(t) - 1) )
-    ind <-as.numeric( gregexpr(pattern='-',t) )
-    if (any(as.numeric( substr(t,1,ind-1) ) +1 == as.numeric( substr(t,ind+1,nchar(t))) ) ) {
-      t <- substr(t,1,ind-1)
-      return(t)
-    }
-    t
-    a <- substr(t, ind+1, nchar(t))
-    t <- sub(a, as.character(as.numeric(a)-1), t)
-    return(t)
-  }
-}
-
-
-
-
-#' @title Set the class of an object (convenience function for
-#'  \code{setattr(obj, "class", CLASS)}); can add instead of replace
-#' @description Sets the class of an object in place to \code{cl}
-#' by replacing or adding
-#' @param obj and object for which to set class
-#' @param cl class to set
-#' @param add if \code{TRUE}, adds \code{cl} to the 
-#' classes of the \code{obj}; otherwise replaces the class information
-#' @param add.place \code{"first"} or \code{"last"}; adds \code{cl}
-#' to the front or to the back of the \code{obj}'s class vector
-#' @author Joonas Miettinen
-setclass <- function(obj, cl, add=FALSE, add.place="first") {
-  match.arg(add.place, c("first","last"))
-  cl <- as.character(cl)
-  
-  if (add) {
-    old_classes <- attr(obj, "class")
-    
-    if (add.place=="first") {
-      setattr(obj, "class", c(cl, old_classes))
-    } else {
-      setattr(obj, "class", c(old_classes, cl))
-    }
-  } else {
-    setattr(obj, "class", cl)
-  }
-}
-
-
-
-
-#' @title Attempt coercion to integer
-#' @author James Arnold
-#' @description Attempts to convert a numeric object to integer, 
-#' but won't if loss of information is imminent (if values after decimal
-#' are not zero for even one value in \code{obj})
-#' @param obj a numeric vector
-#' @param tol tolerance; if each numeric value in \code{obj} deviate from
-#' the corresponding integers at most the value of \code{tol}, they are considered
-#' to be integers; e.g. by default \code{1 + .Machine$double.eps} is considered
-#' to be an integer but \code{1 + .Machine$double.eps^0.49} is not.
-#' @export try2int
-#' @return
-#' An `integer` vector if no information is lost in coercion; else `numeric` 
-#' vector.
-#' @source \href{https://stackoverflow.com/questions/3476782/how-to-check-if-the-number-is-integer}{Stackoverflow thread}
-try2int <- function(obj, tol = .Machine$double.eps^0.5) {
-  if (!is.numeric(obj)) stop("obj needs to be integer or double (numeric)")
-  if (is.integer(obj)) return(obj)
-  
-  test <- FALSE
-  
-  bad <- if (length(na.omit(obj)) == 0) TRUE else 
-    min(obj, na.rm = TRUE) == -Inf || max(obj, na.rm = TRUE) == Inf
-  if (bad) {
-    return(obj)
-  } else {
-    test <- max(abs(obj) %% 1, na.rm = TRUE) < tol
-  }
-  
-  if (is.na(test) || is.null(test)) test <- FALSE
-  
-  if (test) return(as.integer(obj))
-  
-  return(obj)
-  
-}
-
-
-#' @md
-#' @title Get rate and exact Poisson confidence intervals
-#' @author epitools
-#' @description Computes confidence intervals for Poisson rates
-#' @param x observed
-#' @param pt expected
-#' @param conf.level alpha level
-#' 
-#' @export poisson.ci
-#' 
-#' 
-#' @examples
-#' 
-#' poisson.ci(x = 4, pt = 5, conf.level = 0.95)
-#' @return
-#' A `data.frame` with columns
-#' 
-#' - `x`: arg `x`
-#' - `pt`: arg `pt`
-#' - `rate`: result of `x / pt`
-#' - `lower`: lower bound of CI
-#' - `upper`: upper bound of CI
-#' - `conf.level`: arg `conf.level`
-#' 
-poisson.ci <- function(x, pt = 1, conf.level = 0.95) {
-  xc <- cbind(x, conf.level, pt)
-  pt2 <- xc[, 3]
-  results <- matrix(NA, nrow(xc), 6)
-  f1 <- function(x, ans, alpha = alp) {
-    ppois(x, ans) - alpha/2
-  }
-  f2 <- function(x, ans, alpha = alp) 1 - ppois(x, ans) + dpois(x, ans) - alpha/2
-  for (i in 1:nrow(xc)) {
-    alp <- 1 - xc[i, 2]
-    interval <- c(0, xc[i, 1] * 5 + 4)
-    uci <- uniroot(f1, interval = interval, x = xc[i, 1])$root/pt2[i]
-    if (xc[i, 1] == 0) {
-      lci <- 0
-    }
-    else {
-      lci <- uniroot(f2, interval = interval, x = xc[i,1])$root/pt2[i]
-    }
-    results[i, ] <- c(xc[i, 1], pt2[i], xc[i, 1]/pt2[i], lci, uci, xc[i, 2])
-  }
-  coln <- c("x", "pt", "rate", "lower", "upper", "conf.level")
-  colnames(results) <- coln
-  data.frame(results)
-}
-
-
-#' @title Delete \code{data.table} columns if there
-#' @author Joonas Miettinen
-#' @description Deletes columns in a \code{data.table} conveniently.
-#' May only delete columns that are found silently. Sometimes useful in e.g.
-#' \code{on.exit} expressions.
-#' @param DT a \code{data.table}
-#' @param delete a character vector of column names to be deleted
-#' @param keep a character vector of column names to keep; 
-#' the rest will be removed; \code{keep} overrides \code{delete}
-#' @param colorder logical; if \code{TRUE}, also does \code{setcolorder} using
-#' \code{keep}
-#' @param soft logical; if \code{TRUE}, does not cause an error if any variable
-#' name in \code{keep} or \code{delete} is missing; \code{soft = FALSE} useful 
-#' for programming sometimes
-#' 
-#' 
-#' @export setcolsnull
-#' @return
-#' Always returns `NULL` invisibly.
-#' This function is called for its side effects.
-setcolsnull <- function(DT=NULL, delete=NULL, keep=NULL, colorder=FALSE, soft=TRUE) {
-  if (!is.data.table(DT)) stop("not a data.table")
-  if (!soft) {
-    all_names_present(DT, keep, msg = "Expected")
-    all_names_present(DT, delete)
-  }
-  del_cols <- NULL
-  del_cols <- intersect(delete, names(DT))
-  if (!is.null(keep)) {
-    del_cols <- setdiff(names(DT), keep)
-  }
-  if (length(del_cols) > 0) {
-    set(DT, j = (del_cols), value = NULL)
-  }
-  if (colorder) {
-    setcolorder(DT, intersect(keep, names(DT)))
-  }
-  return(invisible(NULL))
-}
-
-
-
-temp_var_names <- function(n = 1L, avoid = NULL, length = 10L) {
-  ## INTENTION: make temporary variable names that don't exist in
-  ## char vector "avoid", e.g. avoid = names(data).
-  if (n < 1L || !is.integer(n)) {
-    stop("n must an integer > 0")
-  }
-  if (length < 1L || !is.integer(length)) {
-    stop("length must an integer > 0")
-  }
-  if (!is.null(avoid)) avoid <- as.character(avoid)
-  
-  pool <- c(0:9, letters, LETTERS)
-  
-  formTemp <- function(int) {
-    v <- sample(x = pool, size = length, replace = TRUE)
-    paste0(v, collapse = "")
-  }
-  
-  l <- lapply(1:n, formTemp)
-  dupll <- duplicated(l) | l %in% avoid
-  tick <- 1L
-  while (any(dupll) && tick <= 100L) {
-    l[dupll] <- lapply(1:sum(dupll), formTemp)
-    dupll <- duplicated(l) | l %in% avoid
-    tick <- tick + 1L
-  }
-  if (tick >= 100L) {
-    stop("ran randomization 100 times and could not create unique temporary",
-         " names. Perhaps increase length?")
-  }
-  unlist(l)
-}
-
-#' @import stats
-makeTempVarName <- function(data=NULL, names=NULL, 
-                            pre=NULL, post=NULL, length = 10L) {
-  DN <- NULL
-  DN <- c(DN, names(data))
-  DN <- c(DN, names)
-  DN <- unique(DN)
-  
-  if (length(pre) != length(post) && length(post) > 0L && length(pre) > 0L) {
-    stop("Lengths of arguments 'pre' and 'post' differ (", length(pre), " vs. ",
-         length(post), "). (Tried to create temporary variables, so this is ",
-         "most likely an internal error and the pkg maintainer should be ",
-         "complained to.)")
-  }
-  useN <- max(length(pre), length(post), 1L)
-  useL <- length
-  tv <- temp_var_names(avoid = DN, n = useN, length = useL)
-  tv <- paste0(pre, tv, post)
-  tv
-}
-
-
-setDFpe <- function(x) {
-  ## intended to only be used to set data.table to data.frame in place
-  ## when option("popEpi.datatable") == FALSE
-  if (!is.data.table(x)) stop("only accepts data.table as input")
-  
-  cl <- class(x)
-  wh <- which(cl == "data.table")
-  cl = c(cl[1:(wh-1)], cl[(wh+1):length(cl)])
-  setattr(x, "class", cl)
-  
-  setattr(x, "sorted", NULL)
-  setattr(x, ".internal.selfref", NULL)
-}
-
-
-
-evalLogicalSubset <- function(data, substiset, n = 2, enclos = parent.frame(n)) {
-  ## NOTE: subset MUST be substitute()'d before using this function!
-  ## we allow substiset to be a logical condition only
-  ## ALWAYS returns a logical vector of length nrow(data)
-  
-  substiset <- eval(substiset, envir = data, enclos = enclos)
-  if (!is.null(substiset)) {
-    if (!is.logical(substiset)) stop("Expression to subset by must be a logical condition, e.g. var1 == 0, var1 %in% 1:2, var1 > 0, etc.")
-    substiset <- substiset & !is.na(substiset)
-    if (sum(substiset) == 0) stop("zero rows in data after subset")
-  } else {
-    substiset <- rep(TRUE, nrow(data))
-  }
-  substiset
-}
-
-
-#' @importFrom data.table setDT
-subsetDTorDF <- function(data, subset = NULL, select = NULL) {
-  ## INTENTION: subsetting either a data.table or a data.frame
-  ## and returning only selected variables for lazy people.
-  if (!is.data.frame(data)) {
-    stop("data must be a data.table/data.frame")
-  }
-  if (!inherits(subset, c("NULL", "logical"))) {
-    stop("subset must be a logical vector or NULL")
-  }
-  
-  expr <- "data[subset, select]"
-  if (is.data.table(data)) {
-    expr <- "data[subset, .SD, .SDcols = select]"
-  }
-  
-  if (!is.null(select)) {
-    all_names_present(data, select)
-  } else {
-    select <- names(data)
-  }
-  if (is.null(subset)) {
-    expr <- sub("^data\\[subset, ", "data[, ", expr)
-  }
-  
-  eval(parse(text = expr))
-}
-
-
-
-setDT2DF <- function(x) {
-  if (!is.data.table(x)) stop("only accepts data.table as input")
-  
-  cl <- class(x)
-  cl <- setdiff(cl, "data.table")
-  setattr(x, "class", cl)  
-  setattr(x, "sorted", NULL)
-  setattr(x, ".internal.selfref", NULL)
-  invisible(x)
-}
-
-setDF2DT <- function(x) {
-  if (!is.data.frame(x) || is.data.table(x)) stop("only accepts data.frame as input")
-  
-  cl <- class(x)
-  whDF <- which(cl == "data.frame")
-  cl <- c(cl[1:(whDF-1)], "data.table", "data.frame", cl[whDF:length(cl)])
-  
-  setattr(x, "class", cl)
-  alloc.col(x)
-  
-  invisible(x)
-}
-
-
-
-
-p.round <- function(p, dec=3) {
-  th <- eval( parse(text=paste0('1E-', dec ) ))
-  if( is.null(p)) return( '= NA') 
-  if( is.na(p))   return( '= NA') 
-  if( p < th ){
-    p <- paste0('< ', th  )
-  } else {
-    p <- paste0('= ', round(p, dec) )
-  }
-  p 
-}
-
-
-cutLow <- function(x, breaks, tol =  .Machine$double.eps^0.5) {
-  ## a cut function that returns the lower bounds of the cut intervals (as numeric) as levels
-  
-  breaks <- sort(breaks)
-  x <- cut(x + tol, right = FALSE, breaks = breaks, labels = FALSE)
-  x <- breaks[-length(breaks)][x]
-  x
-}
-
-
-
-
-setcols <- function(x, j, value) {
-  ## intention: add new columns to DT via modifying in place, and to DF
-  ## via DF$var <- value; both conserve memory (don't take copy of whole data)
-  
-  if (!is.data.frame(x)) stop("x must be a data.frame")
-  if (!is.list(value)) stop("value must be a list of values (columns to add)")
-  if (missing(j)) j <- names(value)
-  
-  if (!is.data.table(x)) {
-    x[j] <- value
-  } else {
-    set(x, j = j, value = value)
-  }
-  x
-}
-
-
-
-
-cutLowMerge <- function(x, y, by.x = by, by.y = by, by = NULL, all.x = all, all.y = all, all = FALSE, mid.scales = c("per", "age"), old.nums = TRUE) {
-  ## INTENTION: merges y to x by by.x & by.y after cutLow()'ing appropriate
-  ## variables in x so that y's values match with x's values
-  ## requirements;
-  ## * numeric variables in y correspond to lower limits of some intervals OR
-  ##   are group variables (e.g. sex = c(0,1))
-  ## inputs: two datas as in merge, preferably both data.table, and other args
-  ## to merge()
-  ## output: a data.table where y has been merged to x after cutLow()
-  ## example: merging popmort to a split Lexis object, where popmort's variables
-  ## correspond to at least some Lexis time scales
-  ## old.nums: return old numeric variable values used in cutLow()'ing?
-  ## mid.scales: use mid-point of interval when merging by these Lexis time scales
-  ## computed by adding + 0.5*lex.dur, which must exist
-  
-  if (!is.data.table(x)) {
-    stop("x must be a data.table")
-  }
-  
-  if ((is.null(by.x) && !is.null(by.y)) || (!is.null(by.x) && is.null(by.y))) {
-    stop("one but not both of by.x / by.y is NULL")
-  }
-  if (!is.null(by)) by.x <- by.y <- by 
-  
-  if (length(by.x) != length(by.y)) stop("lengths differ for by.y & by.x")
-  all_names_present(x, by.x)
-  all_names_present(y, by.y)
-  names(by.x) <- by.y
-  names(by.y) <- by.x
-  
-  if (length(mid.scales)>0) all_names_present(x, c("lex.dur", mid.scales))
-  
-  whScale <- by.x %in% mid.scales
-  xScales <- by.x[whScale]
-  yScales <- by.y[whScale]
-  
-  if (length(yScales) > 0) {
-    
-    oldVals <- copy(with(x, mget(xScales)))
-    on.exit(set(x, j = xScales, value = oldVals))
-    setattr(oldVals, "names", yScales)
-    
-    for (yVar in yScales) {
-      xVar <- xScales[yVar]
-      xBr <- sort(unique(y[[yVar]]))
-      xBr <- unique(c(xBr, Inf))
-      set(x, j = xVar, value = cutLow(x[[xVar]] + x$lex.dur*0.5, breaks = xBr))
-    }
-    
-  }
-  
-  ## ensure x retains order (no copy taken of it)
-  xKey <- key(x)
-  if (length(xKey) == 0) {
-    xKey <- makeTempVarName(x, pre = "sort_")
-    on.exit(if ("x" %in% ls()) setcolsnull(x, delete = xKey, soft = TRUE), add = TRUE)
-    on.exit(if ("z" %in% ls()) setcolsnull(z, delete = xKey, soft = TRUE), add = TRUE)
-    x[, (xKey) := 1:.N]
-  }
-  
-  if (any(duplicated(y, by = by.y))) {
-    stop("y is duplicated by the inferred/supplied by.y variables (",
-         paste0("'", by.y, "'", collapse = ", "), "). ",
-         "First ensure this is not so before proceeding.")
-  }
-  
-  ## avoid e.g. using merge.Lexis when x inherits Lexis
-  xClass <- class(x)
-  on.exit({
-    setattr(x, "class", xClass)
-    }, add = TRUE)
-  setattr(x, "class", c("data.table", "data.frame"))
-  
-  ## return old numeric values of variables that were cutLow()'d
-  ## by keeping them 
-  if (old.nums && length(xScales)) {
-    tmpXScales <- makeTempVarName(names = c(names(x), names(y)), pre = xScales)
-    set(x, j = tmpXScales, value = oldVals)
-    on.exit({
-      xOrder <- setdiff(names(x), tmpXScales)
-      setcolsnull(x, delete = xScales, soft = TRUE)
-      setnames(x, tmpXScales, xScales)
-      setcolorder(x, xOrder)
-      
-    }, add = TRUE)
-  }
-  
-  ## merge
-  z <- merge(x, y, by.x = by.x, by.y = by.y, 
-             all.x = all.x, all.y = all.y, all = all, 
-             sort = FALSE)
-  
-  setDT(z)
-  if (old.nums && length(xScales)) {
-    ## avoid warning due to coercing double to integer
-    set(z, j = xScales, value = NULL)
-    setnames(z, tmpXScales, xScales)
-  }
-  
-  zOrder <- intersect(names(x), names(z))
-  zOrder <- c(zOrder, setdiff(names(z), names(x)))
-  setcolorder(z, zOrder)
-  if (length(xKey) > 0) setkeyv(z, xKey)
-  z[]
-  
-}
-
-
-getOrigin <- function(x) {
-  ## input: Date, IDate, or date variable
-  ## output: the origin date in Date format,
-  ## the origin date being the date where the underlying index is zero.
-  if (inherits(x, "Date") || inherits(x, "IDate")) {
-    as.Date("1970-01-01")
-  } else if (inherits(x, "date")) {
-    as.Date("1960-01-01")
-  } else if (inherits(x, "dates")) {
-    as.Date(paste0(attr(x, "origin"), collapse = "-"), format = "%d-%m-%Y")
-  } else {
-    stop("class '", class(x), "' not supported; usage of Date recommended - see ?as.Date")
-  }
-  
-}
-
-promptYN <- function(q) {
-  
-  rl <- readline(prompt = paste0(q, " (Y/N) ::: "))
-  y <- c("y", "Y")
-  n <- c( "n", "N")
-  if (!rl %in% c(y,n)) {
-    cat("Answer must be one of the following (without ticks):", paste0("'",c(y, n),"'", collapse = ", "))
-    promptYN(q = q)
-  }
-  
-  if (rl %in% y) TRUE else FALSE
-  
-}
-
-
-
-oneWhitespace <- function(x) {
-  if (!is.character(x)) stop("x not a character")
-  x <- paste0(x, collapse = " ")
-  while(sum(grep(pattern = "  ", x = x))) {
-    x <- gsub(pattern = "  ", replacement = " ", x = x)
-  }
-  x
-}
-
-
-aliased_cols <- function(data, cols) {
-  
-  if (missing(cols)) cols <- names(data)
-  all_names_present(data, cols)
-  
-  if (length(cols) < 2L) return(invisible())
-  
-  x <- with(data, mget(cols))
-  x <- lapply(x, duplicated)
-  
-  sub_cols <- cols
-  tl <- list()
-  ## loop: each step reduce vector of names by one
-  ## to avoid testing the same variables twice (in both directions)
-  tick <- 0L
-  aliased <- FALSE
-  while (!aliased && length(sub_cols) > 1L && tick <= length(cols)) {
-    
-    currVar <- sub_cols[1L]
-    sub_cols <- setdiff(sub_cols, currVar)
-    tl[[currVar]] <- unlist(lapply(x[sub_cols], function(j) identical(x[[currVar]], j)))
-    aliased <- sum(tl[[currVar]])
-    
-    tick <- tick + 1L
-  }
-  
-  if (tick == length(cols)) warning("while loop went over the number of columns argument cols")
-  
-  ## result: list of logical vectors indicating if a column is aliased
-  ## with other columns
-  tl[vapply(tl, function(j) sum(j) == 0L, logical(1))] <- NULL
-  
-  if (length(tl) == 0L) return(invisible())
-  
-  ## take first vector for reporting
-  var <- names(tl)[1L]
-  aliases <- names(tl[[1L]])[tl[[1]]]
-  aliases <- paste0("'", aliases, "'", collapse = ", ")
-  stop("Variable '", var, "' is aliased with following variable(s): ", aliases, ".")
-  
-  invisible()
-}
-
-
-
-
-
-
-
-return_DT <- function() {
-  
-  x <- getOption("popEpi.datatable")
-  if (!is.null(x) && !is.logical(x)) {
-    stop("the option 'popEpi.datatable' must be either NULL or a logical ",
-         "value (TRUE / FALSE).")
-  }
-  if (is.null(x) || isTRUE(x)) {
-    return(TRUE)
-  }
-  return(FALSE)
-  
-}
-
-
-
-
-#' @title Create a Lexis Object with Follow-up Time, Period, and Age
-#' Time Scales
-#' @description 
-#' This is a simple wrapper around \code{\link[Epi]{Lexis}} for creating
-#' a \code{Lexis} object with the time scales \code{fot}, \code{per},
-#' and \code{age}.
-#' @param data a \code{data.frame}; mandatory
-#' @param birth the time of birth; A character string naming the variable in 
-#' data or an expression to evaluate - see 
-#' \link[=flexible_argument]{Flexible input}
-#' @param entry the time at entry to follow-up; supplied the 
-#' same way as \code{birth}
-#' @param exit the time at exit from follow-up; supplied the 
-#' same way as \code{birth}
-#' @param entry.status passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
-#' supplied the same way as \code{birth}
-#' @param exit.status passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
-#' supplied the same way as \code{birth}
-#' @param subset a logical condition to subset by before passing data
-#' and arguments to \code{\link[Epi]{Lexis}}
-#' @param ... additional optional arguments passed on to 
-#' \code{\link[Epi]{Lexis}}
-#' @return 
-#' A \code{Lexis} object with the usual columns that \code{Lexis} objects
-#' have, with time scale columns \code{fot}, \code{per}, and \code{age}.
-#' They are calculated as
-#' 
-#' \code{fot = entry - entry} (to ensure correct format, e.g. difftime)
-#' 
-#' \code{per = entry}
-#' 
-#' and 
-#' 
-#' \code{age = entry - birth}
-#' 
-#' @examples 
-#' 
-#' data("sire", package = "popEpi")
-#' 
-#' lex <- Lexis_fpa(sire, 
-#'                  birth = "bi_date", 
-#'                  entry = dg_date, 
-#'                  exit = ex_date + 1L,
-#'                  exit.status = "status")
-#' 
-#' ## some special cases
-#' myVar <- "bi_date"
-#' l <- list(myVar = "bi_date")
-#' sire$l <- sire$myVar <- 1
-#' 
-#' ## conflict: myVar taken from data when "bi_date" was intended
-#' lex <- Lexis_fpa(sire, 
-#'                  birth = myVar, 
-#'                  entry = dg_date, 
-#'                  exit = ex_date + 1L,
-#'                  exit.status = "status")
-#' 
-#' ## no conflict with names in data
-#' lex <- Lexis_fpa(sire, 
-#'                  birth = l$myVar, 
-#'                  entry = dg_date, 
-#'                  exit = ex_date + 1L,
-#'                  exit.status = "status")
-#' @export
-Lexis_fpa <- function(data, 
-                      birth = NULL, 
-                      entry = NULL, 
-                      exit = NULL, 
-                      entry.status = NULL, 
-                      exit.status = NULL, 
-                      subset = NULL,
-                      ...) {
-  if (!requireNamespace("Epi", quietly = TRUE)) {
-    stop("Install package Epi before using this function.")
-  }
-  TF <- environment()
-  PF <- parent.frame(1L)
-  
-  checkVars <- c("fot", "per", "age", 
-                 paste0("lex.", c("dur", "Xst", "Cst", "id")))
-  checkVars <- intersect(names(data), checkVars)
-  if (length(checkVars)) {
-    stop("Following variable name(s) reserved but exist in data: ",
-         paste0(checkVars, collapse = ", "))
-  }
-  
-  
-  sb <- substitute(subset)
-  subset <- evalLogicalSubset(data, sb, enclos = PF)
-  if (all(subset)) subset <- NULL
-  x <- subsetDTorDF(data = data, subset = subset)
-  setDT(x)
-  
-  an <- c("birth", "entry", "exit", "entry.status", "exit.status")
-  
-  l <- vector("list", length(an))
-  names(l) <- an
-  for (stri in an) {
-    e <- paste0("substitute(", stri, ", env = TF)")
-    e <- parse(text = e)[[1]]
-    e <- eval(e, envir = TF) ## e.g. result of substitute(birth)
-    e <- evalPopArg(data = x, arg = e, enclos = PF)[[1]]
-    l[[stri]] <- e
-  }
-  
-  l[sapply(l, is.null)] <- NULL
-  
-  missVars <- setdiff(c("birth", "entry", "exit"), names(l))
-  if (length(missVars)) {
-    stop("Following mandatory arguments were NULL: ",
-         paste0(missVars, collapse = ", "))
-  }
-  
-  fot <- l$entry - l$entry
-  per <- l$entry
-  age <- l$entry - l$birth
-  per_exit <- l$exit
-  
-  en <- list(fot = fot, per = per, age = age)
-  ex <- list(per = per_exit)
-  
-  al <- list(entry = en, exit = ex, entry.status = l$entry.status,
-             exit.status = l$exit.status, data = x)
-  al[sapply(al, is.null)] <- NULL
-  
-  do.call(Epi::Lexis, args = c(al, ...))
-}
-
-
-
-
-
-
-
-get_breaks <- function(x) {
-  UseMethod("get_breaks")
-}
-
-get_breaks.survtab <- function(x) {
-  
-  ss <- attributes(x)$survtab.meta$surv.scale
-  sb <- attributes(x)$survtab.meta$surv.breaks
-  
-  l <- list(sb)
-  names(l) <- ss
-  as.list(l)
-  
-}
-
-
-get_breaks.aggre <- function(x) {
-  
-  as.list(attributes(x)$aggre.meta$breaks)
-  
-}
-
-get_breaks.Lexis <- function(x) {
-  as.list(attributes(x)$breaks)
-}
-
-get_breaks.default <- function(x) {
-  NULL
-}
-
-
-select_breaks <- function(data, ...) {
-  UseMethod("select_breaks")
-}
-
-select_breaks.default <- function(data, ts, br = NULL, ...) {
-  br <- do_select_breaks(data = data, ts = ts, br = br)
-  if (is.null(br)) {
-    stop("Data did not contain breaks and no breaks were supplied ",
-         "by hand.")
-  }
-  br
-}
-
-select_breaks.aggre <- function(data, ts, br = NULL, ...) {
-  
-  
-  br <- do_select_breaks(data = data, ts = ts, br = br)
-  
-  select_breaks_subcheck(br, get_breaks(data)[[ts]], 
-                         "Manually supplied breaks were not a ",
-                         "subset of the breaks in aggre data. ",
-                         "Data has breaks as a result of being split and ",
-                         "aggregated; see ?as.aggre and ?aggre")
-  
-  if (is.null(br)) {
-    stop("aggre object did not contain breaks and no breaks were supplied ",
-         "by hand.")
-  }
-  
-  br
-}
-
-select_breaks.Lexis <- function(data, ts, br = NULL, ...) {
-  
-  checkLexisData(data)
-  
-  br <- do_select_breaks(data = data, ts = ts, br = br)
-  
-  select_breaks_subcheck(br, get_breaks(data)[[ts]], 
-                         "Manually supplied breaks were not a ",
-                         "subset of the breaks in Lexis data. ",
-                         "Data has breaks as a result of being a split Lexis ",
-                         "object; see ?Lexis and e.g. ?splitMulti")
-  
-  if (is.null(br)) {
-    stop("Lexis object did not contain breaks and no breaks were supplied ",
-         "by hand.")
-  }
-  bl <- list(br)
-  names(bl) <- ts
-  checkBreaksList(data, breaks = bl)
-  
-  br
-}
-
-
-select_breaks_subcheck <- function(b1, b2, ...) {
-  l1 <- list(b1)
-  l2 <- list(b2)
-  names(l1) <- names(l2) <- "TS"
-  
-  if (!is.null(b1) && !is.null(b2) && !all_breaks_in(l1, l2)) {
-    stop(...)
-  }
-}
-
-do_select_breaks <- function(data, ts, br = NULL) {
-  # @description selects breaks from data or from br depending on
-  # which one is NULL. If both exist, br must be a subset of the breaks
-  # in data.
-  
-  stopifnot(is.data.frame(data))
-  stopifnot(is.character(ts) && length(ts) == 1L && ts %in% names(data))
-  
-  dbr <- get_breaks(data)[[ts]]
-  
-  dbl <- list(dbr)
-  bl <- list(br)
-  names(dbl) <- names(bl) <- "TS"
-  
-  
-  
-  if (is.null(br)) br <- dbr
-  
-  br
-}
-
-
-
-
-breaks_in_data <- function(br, ts, data) {
-  ## note: last break does not usually appear in data, unless intentionally
-  ## limiting from e.g. 0:5 to 0:4
-  stopifnot(length(ts) == 1 && ts %in% names(data))
-  u <- unique(data[[ts]])
-  
-  br <- sort(unique(br))
-  if (length(br)<2) stop("There must be at least two breaks to form intervals")
-  
-  br <- if (max(br) <= max(u)) br else br[-length(br)]
-  all(br %in% u)
-  
-}
-
-
-
-
-
-is_named_list <- function(x) is.list(x) && length(unique(names(x))) == length(x)
-
-
-
-
-fuse_breakslists <- function(bl.old, bl.new, drop) {
-  # @description given two lists of breaks, uses all timescales found
-  # in both lists to fuse into one list. For common timescales an 
-  # interval-based subset is taken, so that the new always limits the old
-  # when drop = TRUE.
-  
-  stopifnot(
-    is_named_list(bl.old), is_named_list(bl.new)
-  )
-  
-  bl <- bl.old
-  new_scales <- setdiff(names(bl.old), names(bl.new))
-  if (length(new_scales)) {
-    bl[new_scales] <- bl.new[new_scales]
-  }
-  common_scales <- intersect(names(bl.old), names(bl.new))
-  if (length(common_scales)) {
-    
-    bl[common_scales] <- lapply(common_scales, function(time_scale) {
-      new <- bl.new[[time_scale]]
-      old <- bl.old[time_scale]
-      fuse <- sort(union(old, new))
-      if (drop) {
-        r.new <- range(new)
-        r.old <- range(old)
-        r <- c(max(r.new[1], r.old[1]), min(r.new[2], r.old[2]))
-        fuse <- fuse[between(fuse, r[1], r[2], incbounds = TRUE)]
-      }
-      fuse
-    })
-    
-  }
-  
-  bl
-  
-}
-
-
-
-
-
-
-set2 <- function(x, j, ...) {
-  cols_exst <- intersect(names(x), j)
-  old_order <- copy(names(x))
-  if (length(cols_exst)) {
-    set(x, j = cols_exst, value = NULL)
-  }
-  set(x = x, j = j, ...)
-  new_cols <- setdiff(names(x), old_order)
-  setcolorder(x, c(old_order, new_cols))
-  invisible(x)
-}
-
-
-
-
-
-mget_cols <- function(cols, data) {
-  
-  stopifnot(all(cols %in% names(data)))
-  
-  setDT(mget(x = cols, envir = as.environment(data), inherits = FALSE))
-}
-
-
-
-
-
-get_random_seed <- function() {
-  t <- Sys.time()
-  s <- as.numeric(t) %% as.integer(t)
-  nc <- nchar(s)
-  s <- as.integer(substr(s, nc-8, nc))
-  s
-}
-
-
-skip_normally <- function() {
-  requireNamespace("testthat")
-  if (!identical(Sys.getenv("popEpi_run_all_unit_tests"), "true")) {
-    testthat::skip("Unit tests skipped normally")
-  }
-}
+
+
+#' @title Cast \code{data.table}/\code{data.frame} from long format to wide format
+#' @author Matti Rantanen, Joonas Miettinen
+#'
+#' @description 
+#' Convenience function for using \code{\link[data.table]{dcast.data.table}};
+#' inputs are character strings (names of variables) instead of a formula.
+#' 
+#' @param data a \code{data.table} or \code{data.frame}
+#' @param columns a character string vector; the (unique combinations of the) 
+#' levels of these variable will be different rows
+#' @param rows a character string vector; the (unique combinations of the) 
+#' levels of these variable will be different columns
+#' @param values a character string; the variable which will be represented
+#' on rows and columns as specified by \code{columns} and \code{rows}
+#' @import data.table
+#' @import stats
+#' @export cast_simple
+#' @details This function is just a small interface for \code{dcast} / 
+#' \code{dcast.data.table} and less flexible than the originals.
+#' 
+#' Note that all \code{data.table} objects are also \code{data.frame} 
+#' objects, but that each have their own \code{dcast} method.
+#' \code{\link[data.table]{dcast.data.table}} is faster.
+#' 
+#' If any values in \code{value.vars} need to be 
+#' aggregated, they are aggregated using \code{sum}.
+#' See \code{?dcast}.
+#' @return 
+#' A `data.table` just like `[data.table::dcast]`.
+#' 
+#' @examples 
+#' library("data.table")
+#' ## e.g. silly counts from a long-format table to a wide format
+#' test <- data.table::copy(popEpi::sire)
+#' test$dg_y <- year(test$dg_date)
+#' test$ex_y <- year(test$ex_date)
+#' tab <- ltable(test, c("dg_y","ex_y"))
+#' cast_simple(tab, columns='dg_y', rows="ex_y", values="obs")
+#' 
+
+
+cast_simple <- function(data=NULL, columns=NULL, rows=NULL, values=NULL) {
+  if (!is.data.frame(data)) stop("data needs be a data.frame or data.table")
+  if (is.null(data) || nrow(data) == 0L) stop("data NULL or has no rows")
+  
+  if (is.null(columns)) stop("columns cannot be NULL")
+  
+  msg <- paste0("Missing 'columns' variables: %%VARS%%")
+  all_names_present(data, columns, msg = msg)
+  msg <- paste0("Missing 'rows' variables: %%VARS%%")
+  all_names_present(data, rows, msg = msg)
+  msg <- paste0("Missing 'values' variables: %%VARS%%")
+  all_names_present(data, values, msg = msg)
+  
+  ## allow rows = NULL 
+  rowsNULL <- FALSE
+  if (is.null(rows)) rowsNULL <- TRUE
+  if (rowsNULL) rows <- "1"
+  
+  ## sometimes rows names appear to be like expressions, e.g. 'factor(V1)'
+  ## (and this function only uses string-column-names, so that's fine.)
+  actualRows <- rows
+  if (length(rows) > 1L || rows != "1") {
+    rows <- makeTempVarName(names = c(names(data), columns), 
+                            pre = paste0("RN", 1:length(rows)))
+    on.exit(setnames(data, rows, actualRows), add = TRUE)
+    setnames(data, actualRows, rows)
+  }
+  ## same for cols
+  actualCols <- columns
+  columns <- makeTempVarName(names = c(names(data), rows), 
+                             pre = paste0("CN", 1:length(columns)))
+  on.exit(setnames(data, columns, actualCols), add = TRUE)
+  setnames(data, actualCols, columns)
+  
+  form <- paste0(paste0(rows, collapse = " + "), " ~ ", 
+                 paste0(columns, collapse = " + "))
+  form <- as.formula(form)
+  
+  ## note: dcast probably usually finds the methods for data.frame / data.table,
+  ## but this method is more certain
+  if (is.data.table(data)) {
+    d <- dcast.data.table(data, formula = form, value.var=values, 
+                          drop=FALSE, fun.aggregate=sum)[]
+  } else {
+    d <- dcast(data, formula = form, value.var = values, 
+               drop = FALSE, fun.aggregate = sum)[]
+  }
+  if (rowsNULL) set(d, j = names(d)[1L], value = NULL)
+  wh_rows <- which(rows %in% names(d))
+  if (sum(wh_rows, na.rm = TRUE)) setnames(d, rows[wh_rows], actualRows[wh_rows])
+  
+  d
+}
+
+
+#' @title Convert NA's to zero in data.table
+#' @author Joonas Miettinen
+#' @description Given a \code{data.table DT}, replaces any \code{NA} values
+#' in the variables given in \code{vars} in \code{DT}. Takes a copy of the 
+#' original data and returns the modified copy.
+#' @import data.table
+#' @param DT \code{data.table} object
+#' @param vars a character string vector of variables names in \code{DT};
+#' if \code{NULL}, uses all variable names in \code{DT}
+#' @export na2zero
+#' @details Given a \code{data.table} object, converts \code{NA} values
+#' to numeric (double) zeros for all variables named in \code{vars} or
+#' all variables if \code{vars = NULL}.
+#' @return 
+#' A copy of `DT` where `NA` values have been replaced with zero.
+na2zero = function(DT, vars = NULL) { 
+  if (!data.table::is.data.table(DT)) stop("DT must be a data.table")
+  DT <- data.table::copy(DT)
+  
+  navars <- vars
+  if (is.null(navars)) navars <- names(DT)
+  all_names_present(DT, navars)
+  for (k in navars) {
+    DT[is.na(get(k)), (k) := 0]
+  }
+  
+  return(DT[])
+}
+
+
+#' @title Convert factor variable to numeric 
+#' @description Convert factor variable with numbers as levels into a numeric variable
+#' @param x a factor variable with numbers as levels
+#' @export fac2num
+#' @details
+#' For example, a factor with levels \code{c("5","7")} is converted into 
+#' a numeric variable with values \code{c(5,7)}.
+#' @seealso
+#' \code{\link{robust_values}}
+#' @source 
+#' \href{https://stackoverflow.com/questions/3418128/how-to-convert-a-factor-to-an-integer-numeric-without-a-loss-of-information}{Stackoverflow thread}
+#' @examples
+#' ## this is often not intended
+#' as.numeric(factor(c(5,7))) ## result: c(1,2)
+#' ## but this
+#' fac2num(factor(c(5,7))) ## result: c(5,7)
+#' 
+#' ## however
+#' as.numeric(factor(c("5","7","a"))) ## 1:3
+#' 
+#' suppressWarnings(
+#'   fac2num(factor(c("5","7","a"))) ## c(5,7,NA)
+#' )
+#' 
+#' 
+#' @return
+#' A numeric vector based on the levels of `x`.
+fac2num <- function(x) {
+  as.numeric(levels(x))[x]
+}
+
+
+
+
+#' @title Detect leap years
+#' @author Joonas Miettinen
+#' @description Given a vector or column of year values (numeric or integer), \code{\link{is_leap_year}} returns a vector of equal length
+#' of logical indicators, i.e. a vector where corresponding leap years have value TRUE, and FALSE otherwise.
+#' 
+#' @param years a vector or column of year values (numeric or integer)
+#' @examples
+#' ## can be used to assign new columns easily, e.g. a dummy indicator column
+#' df <- data.frame(yrs=c(1900,1904,2005,1995))
+#' df$lyd <- as.integer(is_leap_year(df$yrs))
+#' 
+#' ## mostly it is useful as a condition or to indicate which rows have leap years
+#' which(is_leap_year(df$yrs)) # 2
+#' df[is_leap_year(df$yrs),] # 2nd row
+#' 
+#' @export is_leap_year
+#' @return
+#' A `logical` vector where `TRUE` indicates a leap year.
+is_leap_year <- function(years) {
+  if (!is.numeric(years)) {
+    stop("years must be a numeric vector, preferably integer for speed. Use e.g. as.integer().")
+  }
+  
+  years <- try2int(years)
+  if (!is.integer(years)) stop("years could not be coerced to integer; don't use fractional years such as 2000.1234 but integers such as 2000")
+  
+  # divisible by four
+  isLeap <- years %% 4L == 0L
+  # not divisible by 100
+  isLeap <- isLeap & years %% 100L != 0L
+  # unless divisible by 400 also
+  isLeap <- isLeap | years %% 400L == 0L
+  isLeap
+  
+}
+#' @title Test if object is a \code{Date} object
+#' @description Tests if an object is a \code{Date} object and returns
+#' a logical vector of length 1. \code{IDate} objects are also 
+#' \code{Date} objects, but \code{date} objects from package \pkg{date}
+#' are not. 
+#' @author Joonas Miettinen
+#' @param obj object to test on
+#' @export is.Date
+#' @seealso
+#' \code{\link{get.yrs}}, \code{\link{is_leap_year}}, \code{\link{as.Date}}
+#' @return
+#' `TRUE` if `obj` is of class `"Date"` or `"IDate"`.
+is.Date <- function(obj) {
+  if (any(c("IDate", "Date") %in% class(obj))) {
+    return(TRUE)
+  }
+  return(FALSE)
+}
+
+
+#' @title Convert values to numeric robustly
+#' @author Joonas Miettinen
+#' 
+#' @param num.values values to convert to numeric
+#' @param force logical; if \code{TRUE}, returns a vector of values where values that cannot be interpreted as numeric are
+#' set to \code{NA}; if \code{FALSE}, returns the original vector and gives a warning if any value cannot be interpreted as
+#' numeric.
+#' @param messages logical; if \code{TRUE}, returns a message of what was done with the \code{num.values}
+#' @description Brute force solution for ensuring a variable is numeric by 
+#' coercing a variable of any type first to factor and then to numeric
+#' @export robust_values
+#' @import data.table
+#' @note
+#' Returns \code{NULL} if given \code{num.values} is \code{NULL}. 
+#' @examples
+#' ## this works
+#' values <- c("1", "3", "5")
+#' values <- robust_values(values)
+#' 
+#' ## this works
+#' values <- c("1", "3", "5", NA)
+#' values <- robust_values(values)
+#' 
+#' ## this returns originals and throws warnings
+#' values <- c("1", "3", "5", "a")
+#' suppressWarnings(
+#'   values <- robust_values(values)
+#' )
+#' 
+#' 
+#' ## this forces "a" to NA and works otherwise; throws warning about NAs
+#' values <- c("1", "3", "5", "a")
+#' suppressWarnings(
+#'   values <- robust_values(values, force=TRUE)
+#' )
+#' 
+#' @return
+#' A numeric vector.
+
+robust_values <- function(num.values, force = FALSE, messages = TRUE) {
+  a <- NULL
+  if (is.null(num.values)) {
+    return(NULL)
+  }
+  dt <- data.table(num.values)
+  nas <- dt[is.na(num.values), .N]
+  
+  suppressWarnings(
+    dt[,a := fac2num(factor(num.values))]
+  )
+  dt[, a := try2int(a)]
+  nas2 <- dt[is.na(a), .N]
+  
+  if (!force & nas2 > nas) {
+    if (messages) warning("since force = FALSE and NAs were created, returning original values")
+    return(dt$num.values)
+  }
+  if (force) {
+    if (nas2 > nas) {
+      if (messages) warning("some NAs were created")
+    }
+    return(dt$a)
+  }
+  
+  
+  return(dt$a)
+  
+  
+}
+
+#' @title Check if all names are present in given data
+#' @author Joonas Miettinen
+#' @param data dataset where the variable names should be found
+#' @param var.names a character vector of variable names, e.g.
+#' \code{c("var1", "var2")}
+#' @param stops logical, stop returns exception
+#' @param msg Custom message to return instead of default message.
+#' Special: include \code{\%\%VARS\%\%} in message string and the missing 
+#' variable names will be inserted there (quoted, separated by comma, e.g. 
+#' \code{'var1'}, \code{'var2'} --- no leading or tracing white space). 
+#' @description Given a character vector, checks if all names are present in \code{names(data)}.
+#' Throws error if \code{stops=TRUE}, else returns \code{FALSE} if some variable name is not present.
+#' @seealso
+#' \code{\link{robust_values}}
+#' @export all_names_present
+#' @return 
+#' `TRUE` if all `var.names` are in `data`, else `FALSE`,
+
+all_names_present <- function(data, var.names, stops = TRUE, msg = NULL) {
+  
+  if (!is.null(var.names) && !is.character(var.names)) {
+    stop("Argument 'var.names' must be NULL or a character vector of ",
+         "variable names.")
+  }
+  if (length(var.names) && any(is.na(var.names))) {
+    stop("There are ", sum(is.na(var.names)), " missing values in argument ",
+         "'var.names'. Please only supply non-NA values.")
+  }
+  
+  badNames <- setdiff(var.names, names(data))
+  if (length(badNames) == 0L) return(TRUE)
+  
+  badNames <- paste0("'", badNames, "'", collapse = ", ")
+  
+  if (is.null(msg)) msg <- paste0("Cannot proceed - following given variable name(s) not present in dataset '",
+                                  deparse(substitute(data)), "': ", badNames)
+  if (!is.character(msg) || length(msg) > 1L) stop("Argument 'msg' must be a character string vector of length one.") else
+    msg <- gsub(pattern = "%%VARS%%", replacement = badNames, x = msg)
+  if (!is.logical(stops) || length(stops) > 1L) stop("Argument 'stops' must be either TRUE or FALSE.")
+  
+  if (stops) stop(msg)
+  
+  return(FALSE)
+}
+
+
+#' @title Return lower_bound value from char string (20,30]
+#' @author Matti Rantanen
+#' @description selects lowest values of each factor after cut() based
+#' on that the value starts from index 2 and end in comma ",".
+#' @param cut is a character vector of elements "(20,60]"
+#' @export lower_bound
+#' @return
+#' A numeric vector.
+
+lower_bound <- function(cut) {
+  cut <- as.character(cut)
+  ind <- gregexpr(pattern=',',cut)
+  ind <- as.numeric(ind) - 1
+  t.sub <- as.numeric(substr(cut,2, ind))
+  return(t.sub)
+}
+
+
+#' @title Change output values from cut(..., labels = NULL) output
+#' @author Matti Rantanen
+#' @param t is a character vector of elements, e.g. "(20,60]"
+#' @param factor logical; TRUE returns informative character string, FALSE numeric (left value)
+#' @description Selects lowest values of each factor after cut() based
+#' on the assumption that the value starts from index 2 and end in comma ",".
+#' @details type = 'factor': "[50,52)" -> "50-51" OR "[50,51)" -> "50"
+#' 
+#' type = 'numeric': lowest bound in numeric.
+#' 
+#' @export cut_bound
+#' @return
+#' If `factor = TRUE`, returns a character vector; else returns a numeric 
+#' vector.
+#' @examples
+#' cut_bound("[1900, 1910)") ## "1900-1909"
+
+cut_bound <- function(t, factor=TRUE) {
+  if (!factor) {
+    t <- as.character(t)
+    ind <- gregexpr(pattern=',',t)
+    ind <- as.numeric(ind) - 1
+    t <- as.numeric(substr(t,2, ind))
+    return(t)
+  }
+  if (factor) {
+    t <- as.character(t)
+    t <- gsub(',', '-' , substr(t, 2, nchar(t) - 1) )
+    ind <-as.numeric( gregexpr(pattern='-',t) )
+    if (any(as.numeric( substr(t,1,ind-1) ) +1 == as.numeric( substr(t,ind+1,nchar(t))) ) ) {
+      t <- substr(t,1,ind-1)
+      return(t)
+    }
+    t
+    a <- substr(t, ind+1, nchar(t))
+    t <- sub(a, as.character(as.numeric(a)-1), t)
+    return(t)
+  }
+}
+
+
+
+
+#' @title Set the class of an object (convenience function for
+#'  \code{setattr(obj, "class", CLASS)}); can add instead of replace
+#' @description Sets the class of an object in place to \code{cl}
+#' by replacing or adding
+#' @param obj and object for which to set class
+#' @param cl class to set
+#' @param add if \code{TRUE}, adds \code{cl} to the 
+#' classes of the \code{obj}; otherwise replaces the class information
+#' @param add.place \code{"first"} or \code{"last"}; adds \code{cl}
+#' to the front or to the back of the \code{obj}'s class vector
+#' @author Joonas Miettinen
+setclass <- function(obj, cl, add=FALSE, add.place="first") {
+  match.arg(add.place, c("first","last"))
+  cl <- as.character(cl)
+  
+  if (add) {
+    old_classes <- attr(obj, "class")
+    
+    if (add.place=="first") {
+      setattr(obj, "class", c(cl, old_classes))
+    } else {
+      setattr(obj, "class", c(old_classes, cl))
+    }
+  } else {
+    setattr(obj, "class", cl)
+  }
+}
+
+
+
+
+#' @title Attempt coercion to integer
+#' @author James Arnold
+#' @description Attempts to convert a numeric object to integer, 
+#' but won't if loss of information is imminent (if values after decimal
+#' are not zero for even one value in \code{obj})
+#' @param obj a numeric vector
+#' @param tol tolerance; if each numeric value in \code{obj} deviate from
+#' the corresponding integers at most the value of \code{tol}, they are considered
+#' to be integers; e.g. by default \code{1 + .Machine$double.eps} is considered
+#' to be an integer but \code{1 + .Machine$double.eps^0.49} is not.
+#' @export try2int
+#' @return
+#' An `integer` vector if no information is lost in coercion; else `numeric` 
+#' vector.
+#' @source \href{https://stackoverflow.com/questions/3476782/how-to-check-if-the-number-is-integer}{Stackoverflow thread}
+try2int <- function(obj, tol = .Machine$double.eps^0.5) {
+  if (!is.numeric(obj)) stop("obj needs to be integer or double (numeric)")
+  if (is.integer(obj)) return(obj)
+  
+  test <- FALSE
+  
+  bad <- if (length(na.omit(obj)) == 0) TRUE else 
+    min(obj, na.rm = TRUE) == -Inf || max(obj, na.rm = TRUE) == Inf
+  if (bad) {
+    return(obj)
+  } else {
+    test <- max(abs(obj) %% 1, na.rm = TRUE) < tol
+  }
+  
+  if (is.na(test) || is.null(test)) test <- FALSE
+  
+  if (test) return(as.integer(obj))
+  
+  return(obj)
+  
+}
+
+
+#' @md
+#' @title Get rate and exact Poisson confidence intervals
+#' @author epitools
+#' @description Computes confidence intervals for Poisson rates
+#' @param x observed
+#' @param pt expected
+#' @param conf.level alpha level
+#' 
+#' @export poisson.ci
+#' 
+#' 
+#' @examples
+#' 
+#' poisson.ci(x = 4, pt = 5, conf.level = 0.95)
+#' @return
+#' A `data.frame` with columns
+#' 
+#' - `x`: arg `x`
+#' - `pt`: arg `pt`
+#' - `rate`: result of `x / pt`
+#' - `lower`: lower bound of CI
+#' - `upper`: upper bound of CI
+#' - `conf.level`: arg `conf.level`
+#' 
+poisson.ci <- function(x, pt = 1, conf.level = 0.95) {
+  xc <- cbind(x, conf.level, pt)
+  pt2 <- xc[, 3]
+  results <- matrix(NA, nrow(xc), 6)
+  f1 <- function(x, ans, alpha = alp) {
+    ppois(x, ans) - alpha/2
+  }
+  f2 <- function(x, ans, alpha = alp) 1 - ppois(x, ans) + dpois(x, ans) - alpha/2
+  for (i in 1:nrow(xc)) {
+    alp <- 1 - xc[i, 2]
+    interval <- c(0, xc[i, 1] * 5 + 4)
+    uci <- uniroot(f1, interval = interval, x = xc[i, 1])$root/pt2[i]
+    if (xc[i, 1] == 0) {
+      lci <- 0
+    }
+    else {
+      lci <- uniroot(f2, interval = interval, x = xc[i,1])$root/pt2[i]
+    }
+    results[i, ] <- c(xc[i, 1], pt2[i], xc[i, 1]/pt2[i], lci, uci, xc[i, 2])
+  }
+  coln <- c("x", "pt", "rate", "lower", "upper", "conf.level")
+  colnames(results) <- coln
+  data.frame(results)
+}
+
+
+#' @title Delete \code{data.table} columns if there
+#' @author Joonas Miettinen
+#' @description Deletes columns in a \code{data.table} conveniently.
+#' May only delete columns that are found silently. Sometimes useful in e.g.
+#' \code{on.exit} expressions.
+#' @param DT a \code{data.table}
+#' @param delete a character vector of column names to be deleted
+#' @param keep a character vector of column names to keep; 
+#' the rest will be removed; \code{keep} overrides \code{delete}
+#' @param colorder logical; if \code{TRUE}, also does \code{setcolorder} using
+#' \code{keep}
+#' @param soft logical; if \code{TRUE}, does not cause an error if any variable
+#' name in \code{keep} or \code{delete} is missing; \code{soft = FALSE} useful 
+#' for programming sometimes
+#' 
+#' 
+#' @export setcolsnull
+#' @return
+#' Always returns `NULL` invisibly.
+#' This function is called for its side effects.
+setcolsnull <- function(DT=NULL, delete=NULL, keep=NULL, colorder=FALSE, soft=TRUE) {
+  if (!is.data.table(DT)) stop("not a data.table")
+  if (!soft) {
+    all_names_present(DT, keep, msg = "Expected")
+    all_names_present(DT, delete)
+  }
+  del_cols <- NULL
+  del_cols <- intersect(delete, names(DT))
+  if (!is.null(keep)) {
+    del_cols <- setdiff(names(DT), keep)
+  }
+  if (length(del_cols) > 0) {
+    set(DT, j = (del_cols), value = NULL)
+  }
+  if (colorder) {
+    setcolorder(DT, intersect(keep, names(DT)))
+  }
+  return(invisible(NULL))
+}
+
+
+
+temp_var_names <- function(n = 1L, avoid = NULL, length = 10L) {
+  ## INTENTION: make temporary variable names that don't exist in
+  ## char vector "avoid", e.g. avoid = names(data).
+  if (n < 1L || !is.integer(n)) {
+    stop("n must an integer > 0")
+  }
+  if (length < 1L || !is.integer(length)) {
+    stop("length must an integer > 0")
+  }
+  if (!is.null(avoid)) avoid <- as.character(avoid)
+  
+  pool <- c(0:9, letters, LETTERS)
+  
+  formTemp <- function(int) {
+    v <- sample(x = pool, size = length, replace = TRUE)
+    paste0(v, collapse = "")
+  }
+  
+  l <- lapply(1:n, formTemp)
+  dupll <- duplicated(l) | l %in% avoid
+  tick <- 1L
+  while (any(dupll) && tick <= 100L) {
+    l[dupll] <- lapply(1:sum(dupll), formTemp)
+    dupll <- duplicated(l) | l %in% avoid
+    tick <- tick + 1L
+  }
+  if (tick >= 100L) {
+    stop("ran randomization 100 times and could not create unique temporary",
+         " names. Perhaps increase length?")
+  }
+  unlist(l)
+}
+
+#' @import stats
+makeTempVarName <- function(data=NULL, names=NULL, 
+                            pre=NULL, post=NULL, length = 10L) {
+  DN <- NULL
+  DN <- c(DN, names(data))
+  DN <- c(DN, names)
+  DN <- unique(DN)
+  
+  if (length(pre) != length(post) && length(post) > 0L && length(pre) > 0L) {
+    stop("Lengths of arguments 'pre' and 'post' differ (", length(pre), " vs. ",
+         length(post), "). (Tried to create temporary variables, so this is ",
+         "most likely an internal error and the pkg maintainer should be ",
+         "complained to.)")
+  }
+  useN <- max(length(pre), length(post), 1L)
+  useL <- length
+  tv <- temp_var_names(avoid = DN, n = useN, length = useL)
+  tv <- paste0(pre, tv, post)
+  tv
+}
+
+
+setDFpe <- function(x) {
+  ## intended to only be used to set data.table to data.frame in place
+  ## when option("popEpi.datatable") == FALSE
+  if (!is.data.table(x)) stop("only accepts data.table as input")
+  
+  cl <- class(x)
+  wh <- which(cl == "data.table")
+  cl = c(cl[1:(wh-1)], cl[(wh+1):length(cl)])
+  setattr(x, "class", cl)
+  
+  setattr(x, "sorted", NULL)
+  setattr(x, ".internal.selfref", NULL)
+}
+
+
+
+evalLogicalSubset <- function(data, substiset, n = 2, enclos = parent.frame(n)) {
+  ## NOTE: subset MUST be substitute()'d before using this function!
+  ## we allow substiset to be a logical condition only
+  ## ALWAYS returns a logical vector of length nrow(data)
+  
+  substiset <- eval(substiset, envir = data, enclos = enclos)
+  if (!is.null(substiset)) {
+    if (!is.logical(substiset)) stop("Expression to subset by must be a logical condition, e.g. var1 == 0, var1 %in% 1:2, var1 > 0, etc.")
+    substiset <- substiset & !is.na(substiset)
+    if (sum(substiset) == 0) stop("zero rows in data after subset")
+  } else {
+    substiset <- rep(TRUE, nrow(data))
+  }
+  substiset
+}
+
+
+#' @importFrom data.table setDT
+subsetDTorDF <- function(data, subset = NULL, select = NULL) {
+  ## INTENTION: subsetting either a data.table or a data.frame
+  ## and returning only selected variables for lazy people.
+  if (!is.data.frame(data)) {
+    stop("data must be a data.table/data.frame")
+  }
+  if (!inherits(subset, c("NULL", "logical"))) {
+    stop("subset must be a logical vector or NULL")
+  }
+  
+  expr <- "data[subset, select]"
+  if (is.data.table(data)) {
+    expr <- "data[subset, .SD, .SDcols = select]"
+  }
+  
+  if (!is.null(select)) {
+    all_names_present(data, select)
+  } else {
+    select <- names(data)
+  }
+  if (is.null(subset)) {
+    expr <- sub("^data\\[subset, ", "data[, ", expr)
+  }
+  
+  eval(parse(text = expr))
+}
+
+
+
+setDT2DF <- function(x) {
+  if (!is.data.table(x)) stop("only accepts data.table as input")
+  
+  cl <- class(x)
+  cl <- setdiff(cl, "data.table")
+  setattr(x, "class", cl)  
+  setattr(x, "sorted", NULL)
+  setattr(x, ".internal.selfref", NULL)
+  invisible(x)
+}
+
+setDF2DT <- function(x) {
+  if (!is.data.frame(x) || is.data.table(x)) stop("only accepts data.frame as input")
+  
+  cl <- class(x)
+  whDF <- which(cl == "data.frame")
+  cl <- c(cl[1:(whDF-1)], "data.table", "data.frame", cl[whDF:length(cl)])
+  
+  setattr(x, "class", cl)
+  alloc.col(x)
+  
+  invisible(x)
+}
+
+
+
+
+p.round <- function(p, dec=3) {
+  th <- eval( parse(text=paste0('1E-', dec ) ))
+  if( is.null(p)) return( '= NA') 
+  if( is.na(p))   return( '= NA') 
+  if( p < th ){
+    p <- paste0('< ', th  )
+  } else {
+    p <- paste0('= ', round(p, dec) )
+  }
+  p 
+}
+
+
+cutLow <- function(x, breaks, tol =  .Machine$double.eps^0.5) {
+  ## a cut function that returns the lower bounds of the cut intervals (as numeric) as levels
+  
+  breaks <- sort(breaks)
+  x <- cut(x + tol, right = FALSE, breaks = breaks, labels = FALSE)
+  x <- breaks[-length(breaks)][x]
+  x
+}
+
+
+
+
+setcols <- function(x, j, value) {
+  ## intention: add new columns to DT via modifying in place, and to DF
+  ## via DF$var <- value; both conserve memory (don't take copy of whole data)
+  
+  if (!is.data.frame(x)) stop("x must be a data.frame")
+  if (!is.list(value)) stop("value must be a list of values (columns to add)")
+  if (missing(j)) j <- names(value)
+  
+  if (!is.data.table(x)) {
+    x[j] <- value
+  } else {
+    set(x, j = j, value = value)
+  }
+  x
+}
+
+
+
+
+cutLowMerge <- function(x, y, by.x = by, by.y = by, by = NULL, all.x = all, all.y = all, all = FALSE, mid.scales = c("per", "age"), old.nums = TRUE) {
+  ## INTENTION: merges y to x by by.x & by.y after cutLow()'ing appropriate
+  ## variables in x so that y's values match with x's values
+  ## requirements;
+  ## * numeric variables in y correspond to lower limits of some intervals OR
+  ##   are group variables (e.g. sex = c(0,1))
+  ## inputs: two datas as in merge, preferably both data.table, and other args
+  ## to merge()
+  ## output: a data.table where y has been merged to x after cutLow()
+  ## example: merging popmort to a split Lexis object, where popmort's variables
+  ## correspond to at least some Lexis time scales
+  ## old.nums: return old numeric variable values used in cutLow()'ing?
+  ## mid.scales: use mid-point of interval when merging by these Lexis time scales
+  ## computed by adding + 0.5*lex.dur, which must exist
+  
+  if (!is.data.table(x)) {
+    stop("x must be a data.table")
+  }
+  
+  if ((is.null(by.x) && !is.null(by.y)) || (!is.null(by.x) && is.null(by.y))) {
+    stop("one but not both of by.x / by.y is NULL")
+  }
+  if (!is.null(by)) by.x <- by.y <- by 
+  
+  if (length(by.x) != length(by.y)) stop("lengths differ for by.y & by.x")
+  all_names_present(x, by.x)
+  all_names_present(y, by.y)
+  names(by.x) <- by.y
+  names(by.y) <- by.x
+  
+  if (length(mid.scales)>0) all_names_present(x, c("lex.dur", mid.scales))
+  
+  whScale <- by.x %in% mid.scales
+  xScales <- by.x[whScale]
+  yScales <- by.y[whScale]
+  
+  if (length(yScales) > 0) {
+    
+    oldVals <- copy(with(x, mget(xScales)))
+    on.exit(set(x, j = xScales, value = oldVals))
+    setattr(oldVals, "names", yScales)
+    
+    for (yVar in yScales) {
+      xVar <- xScales[yVar]
+      xBr <- sort(unique(y[[yVar]]))
+      xBr <- unique(c(xBr, Inf))
+      set(x, j = xVar, value = cutLow(x[[xVar]] + x$lex.dur*0.5, breaks = xBr))
+    }
+    
+  }
+  
+  ## ensure x retains order (no copy taken of it)
+  xKey <- key(x)
+  if (length(xKey) == 0) {
+    xKey <- makeTempVarName(x, pre = "sort_")
+    on.exit(if ("x" %in% ls()) setcolsnull(x, delete = xKey, soft = TRUE), add = TRUE)
+    on.exit(if ("z" %in% ls()) setcolsnull(z, delete = xKey, soft = TRUE), add = TRUE)
+    x[, (xKey) := 1:.N]
+  }
+  
+  if (any(duplicated(y, by = by.y))) {
+    stop("y is duplicated by the inferred/supplied by.y variables (",
+         paste0("'", by.y, "'", collapse = ", "), "). ",
+         "First ensure this is not so before proceeding.")
+  }
+  
+  ## avoid e.g. using merge.Lexis when x inherits Lexis
+  xClass <- class(x)
+  on.exit({
+    setattr(x, "class", xClass)
+    }, add = TRUE)
+  setattr(x, "class", c("data.table", "data.frame"))
+  
+  ## return old numeric values of variables that were cutLow()'d
+  ## by keeping them 
+  if (old.nums && length(xScales)) {
+    tmpXScales <- makeTempVarName(names = c(names(x), names(y)), pre = xScales)
+    set(x, j = tmpXScales, value = oldVals)
+    on.exit({
+      xOrder <- setdiff(names(x), tmpXScales)
+      setcolsnull(x, delete = xScales, soft = TRUE)
+      setnames(x, tmpXScales, xScales)
+      setcolorder(x, xOrder)
+      
+    }, add = TRUE)
+  }
+  
+  ## merge
+  z <- merge(x, y, by.x = by.x, by.y = by.y, 
+             all.x = all.x, all.y = all.y, all = all, 
+             sort = FALSE)
+  
+  setDT(z)
+  if (old.nums && length(xScales)) {
+    ## avoid warning due to coercing double to integer
+    set(z, j = xScales, value = NULL)
+    setnames(z, tmpXScales, xScales)
+  }
+  
+  zOrder <- intersect(names(x), names(z))
+  zOrder <- c(zOrder, setdiff(names(z), names(x)))
+  setcolorder(z, zOrder)
+  if (length(xKey) > 0) setkeyv(z, xKey)
+  z[]
+  
+}
+
+
+getOrigin <- function(x) {
+  ## input: Date, IDate, or date variable
+  ## output: the origin date in Date format,
+  ## the origin date being the date where the underlying index is zero.
+  if (inherits(x, "Date") || inherits(x, "IDate")) {
+    as.Date("1970-01-01")
+  } else if (inherits(x, "date")) {
+    as.Date("1960-01-01")
+  } else if (inherits(x, "dates")) {
+    as.Date(paste0(attr(x, "origin"), collapse = "-"), format = "%d-%m-%Y")
+  } else {
+    stop("class '", class(x), "' not supported; usage of Date recommended - see ?as.Date")
+  }
+  
+}
+
+promptYN <- function(q) {
+  
+  rl <- readline(prompt = paste0(q, " (Y/N) ::: "))
+  y <- c("y", "Y")
+  n <- c( "n", "N")
+  if (!rl %in% c(y,n)) {
+    cat("Answer must be one of the following (without ticks):", paste0("'",c(y, n),"'", collapse = ", "))
+    promptYN(q = q)
+  }
+  
+  if (rl %in% y) TRUE else FALSE
+  
+}
+
+
+
+oneWhitespace <- function(x) {
+  if (!is.character(x)) stop("x not a character")
+  x <- paste0(x, collapse = " ")
+  while(sum(grep(pattern = "  ", x = x))) {
+    x <- gsub(pattern = "  ", replacement = " ", x = x)
+  }
+  x
+}
+
+
+aliased_cols <- function(data, cols) {
+  
+  if (missing(cols)) cols <- names(data)
+  all_names_present(data, cols)
+  
+  if (length(cols) < 2L) return(invisible())
+  
+  x <- with(data, mget(cols))
+  x <- lapply(x, duplicated)
+  
+  sub_cols <- cols
+  tl <- list()
+  ## loop: each step reduce vector of names by one
+  ## to avoid testing the same variables twice (in both directions)
+  tick <- 0L
+  aliased <- FALSE
+  while (!aliased && length(sub_cols) > 1L && tick <= length(cols)) {
+    
+    currVar <- sub_cols[1L]
+    sub_cols <- setdiff(sub_cols, currVar)
+    tl[[currVar]] <- unlist(lapply(x[sub_cols], function(j) identical(x[[currVar]], j)))
+    aliased <- sum(tl[[currVar]])
+    
+    tick <- tick + 1L
+  }
+  
+  if (tick == length(cols)) warning("while loop went over the number of columns argument cols")
+  
+  ## result: list of logical vectors indicating if a column is aliased
+  ## with other columns
+  tl[vapply(tl, function(j) sum(j) == 0L, logical(1))] <- NULL
+  
+  if (length(tl) == 0L) return(invisible())
+  
+  ## take first vector for reporting
+  var <- names(tl)[1L]
+  aliases <- names(tl[[1L]])[tl[[1]]]
+  aliases <- paste0("'", aliases, "'", collapse = ", ")
+  stop("Variable '", var, "' is aliased with following variable(s): ", aliases, ".")
+  
+  invisible()
+}
+
+
+
+
+
+
+
+return_DT <- function() {
+  
+  x <- getOption("popEpi.datatable")
+  if (!is.null(x) && !is.logical(x)) {
+    stop("the option 'popEpi.datatable' must be either NULL or a logical ",
+         "value (TRUE / FALSE).")
+  }
+  if (is.null(x) || isTRUE(x)) {
+    return(TRUE)
+  }
+  return(FALSE)
+  
+}
+
+
+
+
+#' @title Create a Lexis Object with Follow-up Time, Period, and Age
+#' Time Scales
+#' @description 
+#' This is a simple wrapper around \code{\link[Epi]{Lexis}} for creating
+#' a \code{Lexis} object with the time scales \code{fot}, \code{per},
+#' and \code{age}.
+#' @param data a \code{data.frame}; mandatory
+#' @param birth the time of birth; A character string naming the variable in 
+#' data or an expression to evaluate - see 
+#' \link[=flexible_argument]{Flexible input}
+#' @param entry the time at entry to follow-up; supplied the 
+#' same way as \code{birth}
+#' @param exit the time at exit from follow-up; supplied the 
+#' same way as \code{birth}
+#' @param entry.status passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
+#' supplied the same way as \code{birth}
+#' @param exit.status passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
+#' supplied the same way as \code{birth}
+#' @param subset a logical condition to subset by before passing data
+#' and arguments to \code{\link[Epi]{Lexis}}
+#' @param ... additional optional arguments passed on to 
+#' \code{\link[Epi]{Lexis}}
+#' @return 
+#' A \code{Lexis} object with the usual columns that \code{Lexis} objects
+#' have, with time scale columns \code{fot}, \code{per}, and \code{age}.
+#' They are calculated as
+#' 
+#' \code{fot = entry - entry} (to ensure correct format, e.g. difftime)
+#' 
+#' \code{per = entry}
+#' 
+#' and 
+#' 
+#' \code{age = entry - birth}
+#' 
+#' @examples 
+#' 
+#' data("sire", package = "popEpi")
+#' 
+#' lex <- Lexis_fpa(sire, 
+#'                  birth = "bi_date", 
+#'                  entry = dg_date, 
+#'                  exit = ex_date + 1L,
+#'                  exit.status = "status")
+#' 
+#' ## some special cases
+#' myVar <- "bi_date"
+#' l <- list(myVar = "bi_date")
+#' sire$l <- sire$myVar <- 1
+#' 
+#' ## conflict: myVar taken from data when "bi_date" was intended
+#' lex <- Lexis_fpa(sire, 
+#'                  birth = myVar, 
+#'                  entry = dg_date, 
+#'                  exit = ex_date + 1L,
+#'                  exit.status = "status")
+#' 
+#' ## no conflict with names in data
+#' lex <- Lexis_fpa(sire, 
+#'                  birth = l$myVar, 
+#'                  entry = dg_date, 
+#'                  exit = ex_date + 1L,
+#'                  exit.status = "status")
+#' @export
+Lexis_fpa <- function(data, 
+                      birth = NULL, 
+                      entry = NULL, 
+                      exit = NULL, 
+                      entry.status = NULL, 
+                      exit.status = NULL, 
+                      subset = NULL,
+                      ...) {
+  if (!requireNamespace("Epi", quietly = TRUE)) {
+    stop("Install package Epi before using this function.")
+  }
+  TF <- environment()
+  PF <- parent.frame(1L)
+  
+  checkVars <- c("fot", "per", "age", 
+                 paste0("lex.", c("dur", "Xst", "Cst", "id")))
+  checkVars <- intersect(names(data), checkVars)
+  if (length(checkVars)) {
+    stop("Following variable name(s) reserved but exist in data: ",
+         paste0(checkVars, collapse = ", "))
+  }
+  
+  
+  sb <- substitute(subset)
+  subset <- evalLogicalSubset(data, sb, enclos = PF)
+  if (all(subset)) subset <- NULL
+  x <- subsetDTorDF(data = data, subset = subset)
+  setDT(x)
+  
+  an <- c("birth", "entry", "exit", "entry.status", "exit.status")
+  
+  l <- vector("list", length(an))
+  names(l) <- an
+  for (stri in an) {
+    e <- paste0("substitute(", stri, ", env = TF)")
+    e <- parse(text = e)[[1]]
+    e <- eval(e, envir = TF) ## e.g. result of substitute(birth)
+    e <- evalPopArg(data = x, arg = e, enclos = PF)[[1]]
+    l[[stri]] <- e
+  }
+  
+  l[sapply(l, is.null)] <- NULL
+  
+  missVars <- setdiff(c("birth", "entry", "exit"), names(l))
+  if (length(missVars)) {
+    stop("Following mandatory arguments were NULL: ",
+         paste0(missVars, collapse = ", "))
+  }
+  
+  fot <- l$entry - l$entry
+  per <- l$entry
+  age <- l$entry - l$birth
+  per_exit <- l$exit
+  
+  en <- list(fot = fot, per = per, age = age)
+  ex <- list(per = per_exit)
+  
+  al <- list(entry = en, exit = ex, entry.status = l$entry.status,
+             exit.status = l$exit.status, data = x)
+  al[sapply(al, is.null)] <- NULL
+  
+  do.call(Epi::Lexis, args = c(al, ...))
+}
+
+
+
+
+
+
+
+get_breaks <- function(x) {
+  UseMethod("get_breaks")
+}
+
+get_breaks.survtab <- function(x) {
+  
+  ss <- attributes(x)$survtab.meta$surv.scale
+  sb <- attributes(x)$survtab.meta$surv.breaks
+  
+  l <- list(sb)
+  names(l) <- ss
+  as.list(l)
+  
+}
+
+
+get_breaks.aggre <- function(x) {
+  
+  as.list(attributes(x)$aggre.meta$breaks)
+  
+}
+
+get_breaks.Lexis <- function(x) {
+  as.list(attributes(x)$breaks)
+}
+
+get_breaks.default <- function(x) {
+  NULL
+}
+
+
+select_breaks <- function(data, ...) {
+  UseMethod("select_breaks")
+}
+
+select_breaks.default <- function(data, ts, br = NULL, ...) {
+  br <- do_select_breaks(data = data, ts = ts, br = br)
+  if (is.null(br)) {
+    stop("Data did not contain breaks and no breaks were supplied ",
+         "by hand.")
+  }
+  br
+}
+
+select_breaks.aggre <- function(data, ts, br = NULL, ...) {
+  
+  
+  br <- do_select_breaks(data = data, ts = ts, br = br)
+  
+  select_breaks_subcheck(br, get_breaks(data)[[ts]], 
+                         "Manually supplied breaks were not a ",
+                         "subset of the breaks in aggre data. ",
+                         "Data has breaks as a result of being split and ",
+                         "aggregated; see ?as.aggre and ?aggre")
+  
+  if (is.null(br)) {
+    stop("aggre object did not contain breaks and no breaks were supplied ",
+         "by hand.")
+  }
+  
+  br
+}
+
+select_breaks.Lexis <- function(data, ts, br = NULL, ...) {
+  
+  checkLexisData(data)
+  
+  br <- do_select_breaks(data = data, ts = ts, br = br)
+  
+  select_breaks_subcheck(br, get_breaks(data)[[ts]], 
+                         "Manually supplied breaks were not a ",
+                         "subset of the breaks in Lexis data. ",
+                         "Data has breaks as a result of being a split Lexis ",
+                         "object; see ?Lexis and e.g. ?splitMulti")
+  
+  if (is.null(br)) {
+    stop("Lexis object did not contain breaks and no breaks were supplied ",
+         "by hand.")
+  }
+  bl <- list(br)
+  names(bl) <- ts
+  checkBreaksList(data, breaks = bl)
+  
+  br
+}
+
+
+select_breaks_subcheck <- function(b1, b2, ...) {
+  l1 <- list(b1)
+  l2 <- list(b2)
+  names(l1) <- names(l2) <- "TS"
+  
+  if (!is.null(b1) && !is.null(b2) && !all_breaks_in(l1, l2)) {
+    stop(...)
+  }
+}
+
+do_select_breaks <- function(data, ts, br = NULL) {
+  # @description selects breaks from data or from br depending on
+  # which one is NULL. If both exist, br must be a subset of the breaks
+  # in data.
+  
+  stopifnot(is.data.frame(data))
+  stopifnot(is.character(ts) && length(ts) == 1L && ts %in% names(data))
+  
+  dbr <- get_breaks(data)[[ts]]
+  
+  dbl <- list(dbr)
+  bl <- list(br)
+  names(dbl) <- names(bl) <- "TS"
+  
+  
+  
+  if (is.null(br)) br <- dbr
+  
+  br
+}
+
+
+
+
+breaks_in_data <- function(br, ts, data) {
+  ## note: last break does not usually appear in data, unless intentionally
+  ## limiting from e.g. 0:5 to 0:4
+  stopifnot(length(ts) == 1 && ts %in% names(data))
+  u <- unique(data[[ts]])
+  
+  br <- sort(unique(br))
+  if (length(br)<2) stop("There must be at least two breaks to form intervals")
+  
+  br <- if (max(br) <= max(u)) br else br[-length(br)]
+  all(br %in% u)
+  
+}
+
+
+
+
+
+is_named_list <- function(x) is.list(x) && length(unique(names(x))) == length(x)
+
+
+
+
+fuse_breakslists <- function(bl.old, bl.new, drop) {
+  # @description given two lists of breaks, uses all timescales found
+  # in both lists to fuse into one list. For common timescales an 
+  # interval-based subset is taken, so that the new always limits the old
+  # when drop = TRUE.
+  
+  stopifnot(
+    is_named_list(bl.old), is_named_list(bl.new)
+  )
+  
+  bl <- bl.old
+  new_scales <- setdiff(names(bl.old), names(bl.new))
+  if (length(new_scales)) {
+    bl[new_scales] <- bl.new[new_scales]
+  }
+  common_scales <- intersect(names(bl.old), names(bl.new))
+  if (length(common_scales)) {
+    
+    bl[common_scales] <- lapply(common_scales, function(time_scale) {
+      new <- bl.new[[time_scale]]
+      old <- bl.old[time_scale]
+      fuse <- sort(union(old, new))
+      if (drop) {
+        r.new <- range(new)
+        r.old <- range(old)
+        r <- c(max(r.new[1], r.old[1]), min(r.new[2], r.old[2]))
+        fuse <- fuse[between(fuse, r[1], r[2], incbounds = TRUE)]
+      }
+      fuse
+    })
+    
+  }
+  
+  bl
+  
+}
+
+
+
+
+
+
+set2 <- function(x, j, ...) {
+  cols_exst <- intersect(names(x), j)
+  old_order <- copy(names(x))
+  if (length(cols_exst)) {
+    set(x, j = cols_exst, value = NULL)
+  }
+  set(x = x, j = j, ...)
+  new_cols <- setdiff(names(x), old_order)
+  setcolorder(x, c(old_order, new_cols))
+  invisible(x)
+}
+
+
+
+
+
+mget_cols <- function(cols, data) {
+  
+  stopifnot(all(cols %in% names(data)))
+  
+  setDT(mget(x = cols, envir = as.environment(data), inherits = FALSE))
+}
+
+
+
+
+
+get_random_seed <- function() {
+  t <- Sys.time()
+  s <- as.numeric(t) %% as.integer(t)
+  nc <- nchar(s)
+  s <- as.integer(substr(s, nc-8, nc))
+  s
+}
+
+
+skip_normally <- function() {
+  requireNamespace("testthat")
+  if (!identical(Sys.getenv("popEpi_run_all_unit_tests"), "true")) {
+    testthat::skip("Unit tests skipped normally")
+  }
+}
diff --git a/R/weighted_table.R b/R/weighted_table.R
index ca961e3..109ac4a 100644
--- a/R/weighted_table.R
+++ b/R/weighted_table.R
@@ -1,661 +1,661 @@
-# @title Make a \code{data.table} of Tabulated, Aggregated Values and Weights
-# @description An internal function that aggregates a table
-# and merges in weights.
-# @param data DF/DT; passed to \code{envir} in \code{eval}
-# @param values values to tabulate. Anything \code{evalPopArg} can evaluate.
-# @param print variables to tabulate by and include in \code{prVars} in attributes
-# @param adjust variables to tabulate by and include in \code{adVars} in attributes
-# @param formula a formula such as \code{fot ~ sex} or \code{Surv(fot, lex.Xst) ~ sex}
-# @param Surv.response logical, if \code{TRUE} throws error if response in
-# \code{formula} is not a \code{Surv} object and vice versa
-# @param by.other other variables to tabulate by and include 
-# in \code{boVars} in attributes
-# @param custom.levels a named list of values. When "inflating" the data
-# in the cross-join / cartesian join sense (try e.g. \code{merge(1:5, 1:2)}),
-# one can supply the levels to inflate by using this to ensure inflation is full.
-# E.g. data might only have levels present to do inflation analogous to
-# \code{merge(2:5, 1:2)} although \code{merge(1:5, 1:2)} is intended and 
-# needed. 
-# @param custom.levels.cut.low a character string vector of variable names.
-# These variables mentioned in \code{custom.levels} and existing in data
-# or first modified (in data) using \code{cutLow()} (essentially 
-# \code{cut()} with \code{right = FALSE} and returning the lower bounds
-# as values). Handy for aggregating data e.g. to survival intervals.
-# \strong{NOTE}: the appropriate elements in \code{custom.levels} for these 
-# variables must exceptionally contain an extra value as the roof used in
-# cutting, which will not be used in "inflating" the table using a merge.
-# See Examples.
-# @param weights a named list or long-form data.frame of weights. See Examples.
-# @param internal.weights.values the variable to use to compute internal
-# weights; only used if \code{weights = "internal"}.
-# @param enclos the enclosing environment passed on to \code{eval}. Variables
-# not found in \code{data} or searched for here.
-# @param NA.text a character string to display in a \code{warning}
-# if there are any rows with missing \code{values} or \code{adjust} values.
-# \strong{special:} key phrase \code{\%\%NA_COUNT\%\%} in text is replaced
-# with the count of missing observations.
-# E.g. \code{"Missing \%\%NA_COUNTS\%\% observations due to derpness."}
-# @examples 
-# library(data.table)
-# 
-# makeWeightsDT <- popEpi:::makeWeightsDT ## this avoids errors during tests
-# 
-# sire <- copy(popEpi::sire)
-# set.seed(1L)
-# sire$sex <- rbinom(nrow(sire), 1, 0.5)
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status %in% 1:2, pophaz = popmort, pp = FALSE,
-#               aggre = list(sex, agegr = cut(dg_age, c(0,50,75,Inf)), fot), 
-#               fot = seq(0, 5, 1/12))
-# ps <- quote(list(sex, fot))
-# as <- quote(list(agegr))
-# vs <- list(quote(list(pyrs, at.risk)))
-# ws <- list(agegr = c(0.2,0.4,0.4))
-# 
-# #### custom.levels usage
-# fb <- seq(0, 5-1/12, 1/12) ## exclude 5 as no row has that value
-# ag2 <- ag[fot > 0.5,]
-# # repeats fot intervals < 0.5 as empty rows
-# # may be the safest way to do this
-# dt <- makeWeightsDT(ag2, print = ps, adjust = as, 
-#                     values = vs, weights = ws,
-#                     custom.levels = list(fot = fb))
-# ## aggregate from intervals seq(0, 5, 1/12) to 0:5
-# fb2 <- 0:5 ## (this time we include 5 as the roof)       
-# dt <- makeWeightsDT(ag2, print = ps, adjust = as, 
-#                     values = vs, weights = ws,
-#                     custom.levels = list(fot = fb2),
-#                     custom.levels.cut.low = "fot")              
-#                     
-# 
-# #### use of enclos
-# TF <- environment()
-# gender <- factor(ag$sex)
-# dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
-#                     values = vs, weights = ws, enclos = TF)
-# ## or NULL: uses calling frame by default.
-# dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
-#                     values = vs, weights = ws,
-#                     enclos = NULL)
-# ## passing parent.fram(1) is the same thing (as below),
-# ## but won't pass in testing these examples somehow (but work in real life)
-# # dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
-# #                     values = vs, weights = ws,
-# #                     enclos = NULL)                  
-# 
-# #### formula usage
-# form <- Surv(fot, factor(from0to1))~gender
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = TRUE,
-#                     adjust = as, values = vs, weights = ws,
-#                     enclos = NULL)
-#                     
-# ## or
-# form <- Surv(fot, factor(from0to1))~gender + adjust(agegr)
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = TRUE,
-#                     adjust = NULL, values = vs, weights = ws,
-#                     enclos = NULL)
-#                     
-# ## or   
-# form <- from0to1 ~ fot + gender + adjust(agegr)
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
-#                     adjust = NULL, values = vs, weights = ws,
-#                     enclos = NULL)            
-# 
-# form <- from0to1 ~ fot + adjust(agegr) + adjust(sex)
-# ws2 <- list(agegr = c(0.33, 0.33, 0.33), sex = c(0.5, 0.5))
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
-#                     adjust = NULL, values = vs, weights = ws2,
-#                     enclos = NULL)
-# 
-# ## international standard pops
-# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-#               status = status %in% 1:2, pophaz = popmort, pp = FALSE,
-#               aggre = list(sex, agegr = cut(dg_age, c(seq(0, 85, 5), Inf)), fot), 
-#               fot = seq(0, 5, 1/12))
-#               
-# form <- from0to1 ~ fot + adjust(agegr)
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
-#                     adjust = NULL, values = vs, weights = "world_1966_18of5",
-#                     enclos = NULL)
-#                     
-# form <- from0to1 ~ fot + adjust(agegr, sex)
-# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
-#                     adjust = NULL, values = vs, 
-#                     weights = list(agegr = "nordic_2000_18of5", sex=c(1,1)),
-#                     enclos = NULL)
-makeWeightsDT <- function(data, values = NULL, 
-                          print = NULL, adjust = NULL,
-                          formula = NULL, Surv.response = TRUE,
-                          by.other = NULL, custom.levels = NULL, 
-                          custom.levels.cut.low = NULL, weights = NULL, 
-                          internal.weights.values = NULL, 
-                          enclos = NULL, NA.text = NULL) {
-  
-  # environmentalism -----------------------------------------------------------
-  TF <- environment()
-  PF <- parent.frame(1L)
-  if (missing(enclos) || is.null(enclos)) {
-    enclos <- PF 
-  } 
-  
-  enclos <- eval(enclos, envir = TF)
-  
-  if (!is.environment(enclos)) {
-    stop("Argument 'enclos' is not an environment. (Probably internal error, ",
-         "meaning you should complain to the package maintainer if you are not",
-         "doing something silly.)")
-  }
-  
-  THIS_CALL <- match.call()
-  
-  ## dataism -------------------------------------------------------------------
-  if (!is.data.frame(data)) stop("data must be a data.frame")
-  ## tmpDum for convenience. will be deleted in the end. (if no tabulating vars)
-  origData <- data
-  tmpDum <- makeTempVarName(origData, pre = "dummy_")
-  data <- data.table(rep(1L, nrow(origData)))
-  setnames(data, 1, tmpDum)
-
-  
-  # formula: vars to print and adjust by ---------------------------------------
-  
-  adSub <- adjust
-  adjust <- evalPopArg(data = origData, arg = adSub, DT = TRUE, enclos = enclos)
-  
-  if (!is.null(formula)) {
-    
-    foList <- usePopFormula(formula, adjust = adjust, 
-                            data = origData, enclos = enclos, 
-                            Surv.response = Surv.response)
-    print <- foList$print
-    adjust <- foList$adjust
-  } else {
-    
-    prSub <- substitute(print)
-    print <- evalPopArg(data = origData, arg = prSub, DT = TRUE, enclos = enclos)
-    
-    
-  }
-  
-  if (length(weights) && length(adjust)) {
-    checkWeights(weights, adjust = adjust)
-  }
-  
-  # variables to print by ----------------------------------------------------
-  prVars <- tmpDum
-  if (length(print) > 0) {
-    prVars <- names(print)
-    data[, c(prVars) := TF$print]
-    data[, c(tmpDum) := NULL]
-  } 
-  rm(print)
-  
-  # standardization ----------------------------------------------------------
-  ## note: adjust evaluated above with formula
-  adVars <- NULL
-  if (length(adjust) > 0) {
-    adVars <- names(adjust)
-    data[, c(adVars) := TF$adjust]
-  } 
-  rm(adjust)
-  
-  if (is.null(weights) && length(adVars)) {
-    stop("Variables to adjust by were defined but no weights were supplied.")
-  }
-  
-  if (!length(adVars)) {
-    
-    if (!is.null(weights)) {
-      message("NOTE: Weights ignored since no adjusting variables given")
-    }
-    
-    weights <- NULL
-  }
-  
-  # variables to sum -----------------------------------------------------------
-  if (!is.list(values)) stop("Argument 'values' must be a list ",
-                             "(internal error: complain to the package",
-                             "maintainer if you see this)")
-  values <- lapply(values, function(x) {
-    evalPopArg(data = origData, arg = x, DT = TRUE, enclos = enclos)
-  })
-  for (dt in setdiff(seq_along(values), 1L)) {
-    values[[1L]] <- cbind(values[[1L]], values[[dt]])
-  }
-  values <- values[[1L]]
-  vaVars <- NULL
-  if (nrow(values) != nrow(data)) {
-    stop("mismatch in numers of rows in data (", nrow(data), 
-         ") and 'values' (", nrow(values), "). If you see this message, ",
-         "complain to the package maintainer.")
-  }
-  
-  if (length(values) > 0) {
-    vaVars <- names(values)
-    data[, c(vaVars) := TF$values]
-  } else {
-    stop("no values given to sum!")
-  }
-  rm(values)
-  
-  # additionally, values to compute internal weights by: -----------------------
-  iwVar <- NULL
-  if (is.character(weights) && 
-      pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
-    iw <- substitute(internal.weights.values)
-    iw <- evalPopArg(data = origData, iw, DT = TRUE,
-                     enclos = PF, recursive = TRUE,
-                     types = c("character", "expression", "list", "NULL"))
-    
-    if (length(iw) > 1L) stop("Argument 'internal.weights.values' ",
-                              "must produce only one column.")
-    if (length(iw) == 1L && is.character(weights) && 
-        pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
-      iwVar <- makeTempVarName(names=c(names(data), names(origData)), pre = "iw_")
-      data[, c(iwVar) := TF$iw]
-    }
-    
-    if (length(iwVar) == 0L) {
-      stop("Requested computing internal weights, but no values to compute ",
-           "internals weights with were supplied (internal error: If you see ",
-           "this, complain to the package maintainer).")
-    }
-    rm(iw)
-  }
-  
-  
-  # other category vars to keep ------------------------------------------------
-  boSub <- by.other
-  by.other <- evalPopArg(data = origData, arg = boSub, DT = TRUE, enclos = enclos)
-  boVars <- NULL
-  if (length(by.other) > 0) {
-    boVars <- names(by.other)
-    data[, c(boVars) := TF$by.other]
-  } 
-  rm(by.other)
-  
-  # check for aliased columns --------------------------------------------------
-  aliased_cols(data, cols = c(prVars, adVars, boVars))
-  
-  # check for conflicting column names -----------------------------------------
-  dupCols <- c(prVars, adVars, boVars, vaVars, iwVar)
-  dupCols <- unique(dupCols[duplicated(dupCols)])
-  if (length(dupCols) > 0L) {
-    dupCols <- paste0("'", dupCols, "'", collapse = ", ")
-    stop("Following column names duplicated (columns created by arguments ", 
-         "print, adjust, etc.): ", dupCols, ". If you see this, please ensure ",
-         "you are not passing e.g. the same column to both for adjusting ",
-         "and stratification (printing).")
-  }
-  
-  # check for NA values --------------------------------------------------------
-  ## NOTE: NA values of print/by.other are OK. values/adjust are NOT.
-  
-    NAs <- data[, lapply(.SD, function(x) is.na(x)), .SDcols = c(vaVars, iwVar, adVars)]
-    NAs <- rowSums(NAs) > 0L
-    if (sum(NAs)) {
-      if (!is.null(NA.text)) {
-        NA.text <- gsub(x = NA.text, pattern = "%%NA_COUNT%%", 
-                        replacement = sum(NAs))
-        warning(NA.text)
-      }
-      data <- data[!NAs]
-    }
-
-  # inflate data ---------------------------------------------------------------
-  ## on the other hand we aggregate data to levels of print, adjust and 
-  ## by.other; on the other hand the data will always have tabulating variables
-  ## represented as cross-joined, e.g. merge(1:5, 1:2).
-  ## this means some rows might have zeros as values in the 'values'
-  ## columns.
-  ## (necessary for correct standardization with weights)
-  
-  ## NOTE: have to do CJ by hand: some levels of adjust or something may not
-  ## have each level of e.g. fot repeated!
-  
-  sortedLevs <- function(x) {
-    if (!is.factor(x)) return(sort(unique(x)))
-    
-    factor(levels(x), levels(x), levels(x))
-  }
-  cj <- list()
-  cj <- lapply(data[, .SD, .SDcols =  c(prVars, adVars, boVars)], sortedLevs)
-  
-  ## e.g. if data only has fot = seq(0, 4, 1/12), but want to display table
-  ## with fot = seq(0, 5, 1/12). Very important sometimes for handy usage
-  ## of weights.
-  if (length(custom.levels) > 0) cj[names(custom.levels)] <- custom.levels
-  
-  
-  ## SPECIAL: if e.g. a survival time scale with breaks seq(0, 5, 1/12)
-  ## is to be "compressed" to breaks 0:5, and the latter breaks were passed
-  ## via custom.levels, the following ensures e.g. intervals between 0 and 1
-  ## are aggregated to the same row in what follows after.
-  if (!is.null(custom.levels.cut.low)) {
-    cl_msg <- paste0("Internal error: tried to cut() variables in  ",
-                     "internally used work data that did not exist. ",
-                     "If you see this, complain to the ",
-                     "package maintainer. Bad variables: %%VARS%%.")
-    all_names_present(data, custom.levels.cut.low, msg = cl_msg)
-    all_names_present(cj, custom.levels.cut.low, msg = cl_msg)
-    
-    for (var in custom.levels.cut.low) {
-      set(data, j = var, value = cutLow(data[[var]], breaks = cj[[var]]))
-    }
-    
-    ## NOTE: if used cutlow(), then assume passed values via custom.levels
-    ## also contained the roof of the values which should not be repeated.
-    cj[custom.levels.cut.low] <- lapply(cj[custom.levels.cut.low],
-                                        function(elem) elem[-length(elem)])
-  }
-  
-  ## form data.table to merge by - the merge will inflate the data.
-  cj <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), cj)
-  
-  ## inflate & aggregate.
-  setkeyv(data, c(prVars, adVars, boVars))
-  data <- data[cj, lapply(.SD, sum), .SDcols = c(vaVars, iwVar), by = .EACHI]
-  
-  for (k in c(vaVars, iwVar)) {
-    data[is.na(get(k)), (k) := 0]
-  }
-  
-  setcolsnull(data, tmpDum)
-  prVars <- setdiff(prVars, tmpDum); if (length(prVars) == 0) prVars <- NULL
-  
-  ## merge in weights ----------------------------------------------------------
-  
-  if (!is.null(weights)) {
-    
-    if (is.list(weights) && !is.data.frame(weights)) {
-      ## in case one of the elements is a standardization scheme string,
-      ## such as "world_x_y". 
-      whChar <- which(unlist(lapply(weights, is.character)))
-      if (sum(whChar)) {
-        
-        weights[whChar] <- lapply(weights[whChar], function(string) {
-          ## expected to return data.frame with 1) age groups 2) weights
-          ## as columns.
-          stdr.weights(string)[[2]]
-        })
-        
-      }
-      
-      ## now list only contains numeric weights i hope.
-      
-    }
-    
-    ## NOTE: adjust used here to contain levels of adjust arguments only
-    adjust <- list()
-    if (length(adVars) > 0L) {
-      adjust <- lapply(data[, eval(adVars), with = FALSE], sortedLevs)
-    }
-    
-    if (is.character(weights)) {
-      
-      if (pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
-        
-        
-        all_names_present(data, iwVar,
-                          msg = paste0(
-                            "Internal error: expected to have variable ",
-                            "%%VARS%% in working data but didn't. Complain ",
-                            "to the pkg maintainer if you see this."
-                          ))
-        
-        weights <- lapply(seq_along(adjust), function(i) {
-          cn <- names(adjust)[i]
-          le <- unique(adjust[[i]])
-          le <- structure(list(le), names = cn)
-          data[le, lapply(.SD, sum), .SDcols = iwVar, by = .EACHI, on = cn][[iwVar]]
-        })
-        names(weights) <- names(adjust)
-        
-        
-        setcolsnull(data, iwVar)
-        setkeyv(data, c(prVars, adVars, boVars))
-      } else {
-        ## expected to return data.frame with 1) age groups 2) weights
-        ## as columns.
-        weights <- stdr.weights(weights)
-        weights <- weights[[2]]
-        
-      }
-    } 
-    
-    if (!is.data.frame(weights) && is.vector(weights)) {
-      ## note: lists are vectors
-      if (!is.list(weights)) {
-        weights <- list(weights) ## was a vector of values
-        setattr(weights, "names", adVars[1])
-      }
-      
-      weVars <- names(weights)
-      weights <- weights[names(adjust)]
-      
-      adjust <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), adjust)
-      weights <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), weights)
-      
-      weVars <- paste0(weVars, ".w")
-      setnames(weights, adVars, weVars)
-      weights[, (adVars) := adjust]
-      
-      set(weights, j = "weights", value = 1L)
-      for (k in weVars) {
-        set(weights, j = "weights", value = weights$weights * weights[[k]])
-      }
-      setcolsnull(weights, delete = weVars, soft = FALSE)
-      
-      ## NOTE: weights will be repeated for each level of print,
-      ## and for each level of print the weights must sum to one for things
-      ## to work.
-      weights[, "weights" := weights/sum(weights)]
-      
-    }
-    
-    if (!is.data.frame(weights)) {
-      stop("Something went wrong: 'weights' was not collated into a ",
-           "data.frame to merge with data. ",
-           "Blame the package maintainer please!")
-    }
-    ## at this points weights is a data.frame.
-    weights <- data.table(weights)
-    weights[, "weights" := as.double(weights)]
-    
-    ## ensure repetition by print levels if some adjust levels
-    ## that exist in weights do not exist in data.
-    ## NOTE: weights data.frame has at least as many levels as adjust column
-    ## in data (or it has more sometimes).
-    wm <- lapply(adVars, function(chStr) {
-      col <- weights[[chStr]]
-      if (is.factor(col)) return(levels(col))
-      sort(unique(col))
-      })
-    names(wm) <- adVars
-    if (length(prVars)) {
-      wm[prVars] <- lapply(prVars, function(chStr) {
-        col <- data[[chStr]]
-        if (is.factor(col)) return(levels(col))
-        sort(unique(col))
-      })
-    }
-    wm <- setDT(do.call(CJ, wm))
-    
-    weights <- merge(wm, weights, by = adVars, all.x = TRUE, all.y = TRUE)
-    
-    dt_robust_by(
-      'weights[, "weights" := weights/sum(weights), by = %%BY_VAR_NMS%%]',
-      by.var.nms = prVars
-    )
-    
-    data[i = weights, on = c(prVars, adVars), j = "weights" := weights]
-    
-    if (any(is.na(data$weights))) {
-      ## should not be any NAs since we checked for level congruence
-      ## in checkWeights
-      stop("Internal error: some weights were NA after merging to working ",
-           "data. Complain to the package maintainer if you see this.")
-    }
-    
-    
-  }
-  
-  setattr(data, "makeWeightsDT", list(prVars = prVars, adVars = adVars, 
-                                      boVars = boVars, vaVars = vaVars, 
-                                      NAs = NAs))
-  return(data[])
-  
-}
-
-checkCharWeights <- function(w) {
-  if (is.character(w)) {
-    if (length(w) != 1L) {
-      stop("weights supplied as a character string must be of length one.")
-    }
-    if (!pmatch(w, c("internal", "cohort"), nomatch = 0L)) {
-      stdr.weights(w)
-    }
-  }
-}
-
-checkWeights <- function(weights, adjust) {
-  ## INTENTION: given a list/DF/vector/string specifying weights
-  ## and a data.frame/list of the adjusting variables,
-  ## checks they are congruent and complains if not.
-  allowed_classes <- c("list","data.frame","integer","numeric","character",
-                       "NULL")
-  if (!any(class(weights) %in% allowed_classes)) {
-    stop("weights must be either a list, a data.frame, a numeric variable, ",
-         "or a character string specifing the weighting scheme to use. ",
-         "See ?direct_standardization for more information.") 
-  }
-  
-  if (is.list(weights) && !is.data.frame(weights) && 
-      length(adjust) != length(weights)) {
-    stop("Mismatch in numbers of variables (NOT necessarily in the numbers of ",
-         "levels/values within the variables) in adjust (", length(adjust), 
-         " variables) and weights (", length(weights)," variables); ",
-         "make sure each given weights vector has a corresponding ",
-         "variable in adjust and vice versa. ",
-         "See ?direct_standardization for more information.")
-  }
-  
-  if (is.list(weights)) {
-    isChar <- unlist(lapply(weights, is.character))
-    if (any(isChar)) {
-      lapply(weights[isChar], checkCharWeights)
-      weights[isChar] <- lapply(weights[isChar], function(string) {
-        if (pmatch(string, c("cohort", "internal"), nomatch = 0L)) {
-          stop("List of weights had 'cohort' or 'internal' as at least one ",
-               "element, which is currently not supported. ",
-               "See ?direct_standardization for more information.")
-        }
-        stdr.weights(string)[[2]]
-        })
-    }
-  }
-
-  if (is.character(weights)) {
-    checkCharWeights(weights)
-    if (pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
-      ## done checking since internal weights are pretty fool-proof.
-      return(invisible())
-    }
-    ## if not, pass along as vector of weights.
-    weights <- stdr.weights(weights)[[2]]
-  }
-  
-  if (is.numeric(weights)) { 
-    if (length(adjust) != 1L) {
-      stop("Weights is a numeric vector of weights, ",
-           "but there are more or less than one adjusting variable. ",
-           "See ?direct_standardization for more information.")
-    }
-    weights <- list(weights)
-    names(weights) <- names(adjust)
-  }
-  
-  ## by now either a list or a data.frame of weights...
-  adVars <- names(adjust)
-  weVars <- names(weights)
-  if (is.data.frame(weights)) {
-    if (!"weights" %in% weVars) {
-      stop("data.frame of weights did not have column named 'weights'. ",
-           "see ?direct_standardization for more information.")
-    }
-    weVars <- setdiff(weVars, "weights")
-  }
-  
-  badAdVars <- setdiff(adVars, weVars)
-  badWeVars <- setdiff(weVars, adVars)
-  if (length(badAdVars) > 0) {
-    stop("Mismatch in names of variables in adjust and weights; ",
-         "following adjust variables not mentioned in weights: ", 
-         paste0("'", badAdVars, "'", collapse = ", "))
-  }
-  
-  if (length(badWeVars) > 0) {
-    stop("Mismatch in names of variables in adjust and weights; ",
-         "following weights variables not mentioned in adjust: ",
-         paste0("'", badWeVars, "'", collapse = ", "))
-  }
-  
-  if (is.data.frame(weights)) {
-    
-    levDiff <- lapply(names(adjust), function(var) {
-      !all(adjust[[var]] %in% weights[[var]])
-    })
-    levDiff <- unlist(levDiff)
-    if (any(levDiff)) {
-      ## take only first conflicting variable for brevity of error message-
-      badVar <- names(adjust)[1]
-      badLevs <- setdiff(adjust[[badVar]], weights[[badVar]])
-      badLevs <- paste0("'", badLevs, "'", collapse = ", ")
-      stop("Missing levels in weights data.frame in variable '", badVar, "': ",
-           badLevs, ". These levels were found to exist in the corresponding ",
-           "adjusting variable. ",
-           "Usual suspects: adjusting variable is a factor and you ",
-           "only supplied weights for unique values in your data ",
-           "as opposed to the levels of the factor, which may contain levels ",
-           "that no row has. Try table(yourdata$yourvariable).")
-    }
-    
-  } else {
-    weights <- as.list(weights)
-    weights <- weights[adVars]
-    
-    ## check variable levels
-    adjust <- lapply(adjust, function(elem) {
-      if (is.factor(elem)) {
-        levels(elem)
-      } else {
-        sort(unique(elem))
-      }
-    })
-    
-    weLen <- unlist(lapply(weights, length))
-    adLen <- unlist(lapply(adjust, length))
-    badLen <- names(adjust)[weLen != adLen]
-    
-    if (length(badLen) > 0) {
-      stop("Mismatch in numbers of levels/unique values in adjusting variables ",
-           "and lengths of corresponding weights vectors. ",
-           "Names of mismatching variables: ", 
-           paste0("'", badLen, "'", collapse = ", "), ". There were ",
-           weLen[weLen != adLen], " weights and ", adLen[weLen != adLen], 
-           " adjusting variable levels.")
-    }
-  }
-  
-  
-  
-  invisible()
-}
-
-
-
-
-
-
-
-
+# @title Make a \code{data.table} of Tabulated, Aggregated Values and Weights
+# @description An internal function that aggregates a table
+# and merges in weights.
+# @param data DF/DT; passed to \code{envir} in \code{eval}
+# @param values values to tabulate. Anything \code{evalPopArg} can evaluate.
+# @param print variables to tabulate by and include in \code{prVars} in attributes
+# @param adjust variables to tabulate by and include in \code{adVars} in attributes
+# @param formula a formula such as \code{fot ~ sex} or \code{Surv(fot, lex.Xst) ~ sex}
+# @param Surv.response logical, if \code{TRUE} throws error if response in
+# \code{formula} is not a \code{Surv} object and vice versa
+# @param by.other other variables to tabulate by and include 
+# in \code{boVars} in attributes
+# @param custom.levels a named list of values. When "inflating" the data
+# in the cross-join / cartesian join sense (try e.g. \code{merge(1:5, 1:2)}),
+# one can supply the levels to inflate by using this to ensure inflation is full.
+# E.g. data might only have levels present to do inflation analogous to
+# \code{merge(2:5, 1:2)} although \code{merge(1:5, 1:2)} is intended and 
+# needed. 
+# @param custom.levels.cut.low a character string vector of variable names.
+# These variables mentioned in \code{custom.levels} and existing in data
+# or first modified (in data) using \code{cutLow()} (essentially 
+# \code{cut()} with \code{right = FALSE} and returning the lower bounds
+# as values). Handy for aggregating data e.g. to survival intervals.
+# \strong{NOTE}: the appropriate elements in \code{custom.levels} for these 
+# variables must exceptionally contain an extra value as the roof used in
+# cutting, which will not be used in "inflating" the table using a merge.
+# See Examples.
+# @param weights a named list or long-form data.frame of weights. See Examples.
+# @param internal.weights.values the variable to use to compute internal
+# weights; only used if \code{weights = "internal"}.
+# @param enclos the enclosing environment passed on to \code{eval}. Variables
+# not found in \code{data} or searched for here.
+# @param NA.text a character string to display in a \code{warning}
+# if there are any rows with missing \code{values} or \code{adjust} values.
+# \strong{special:} key phrase \code{\%\%NA_COUNT\%\%} in text is replaced
+# with the count of missing observations.
+# E.g. \code{"Missing \%\%NA_COUNTS\%\% observations due to derpness."}
+# @examples 
+# library(data.table)
+# 
+# makeWeightsDT <- popEpi:::makeWeightsDT ## this avoids errors during tests
+# 
+# sire <- copy(popEpi::sire)
+# set.seed(1L)
+# sire$sex <- rbinom(nrow(sire), 1, 0.5)
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status %in% 1:2, pophaz = popmort, pp = FALSE,
+#               aggre = list(sex, agegr = cut(dg_age, c(0,50,75,Inf)), fot), 
+#               fot = seq(0, 5, 1/12))
+# ps <- quote(list(sex, fot))
+# as <- quote(list(agegr))
+# vs <- list(quote(list(pyrs, at.risk)))
+# ws <- list(agegr = c(0.2,0.4,0.4))
+# 
+# #### custom.levels usage
+# fb <- seq(0, 5-1/12, 1/12) ## exclude 5 as no row has that value
+# ag2 <- ag[fot > 0.5,]
+# # repeats fot intervals < 0.5 as empty rows
+# # may be the safest way to do this
+# dt <- makeWeightsDT(ag2, print = ps, adjust = as, 
+#                     values = vs, weights = ws,
+#                     custom.levels = list(fot = fb))
+# ## aggregate from intervals seq(0, 5, 1/12) to 0:5
+# fb2 <- 0:5 ## (this time we include 5 as the roof)       
+# dt <- makeWeightsDT(ag2, print = ps, adjust = as, 
+#                     values = vs, weights = ws,
+#                     custom.levels = list(fot = fb2),
+#                     custom.levels.cut.low = "fot")              
+#                     
+# 
+# #### use of enclos
+# TF <- environment()
+# gender <- factor(ag$sex)
+# dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
+#                     values = vs, weights = ws, enclos = TF)
+# ## or NULL: uses calling frame by default.
+# dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
+#                     values = vs, weights = ws,
+#                     enclos = NULL)
+# ## passing parent.fram(1) is the same thing (as below),
+# ## but won't pass in testing these examples somehow (but work in real life)
+# # dt <- makeWeightsDT(ag, print = quote(gender), adjust = as, 
+# #                     values = vs, weights = ws,
+# #                     enclos = NULL)                  
+# 
+# #### formula usage
+# form <- Surv(fot, factor(from0to1))~gender
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = TRUE,
+#                     adjust = as, values = vs, weights = ws,
+#                     enclos = NULL)
+#                     
+# ## or
+# form <- Surv(fot, factor(from0to1))~gender + adjust(agegr)
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = TRUE,
+#                     adjust = NULL, values = vs, weights = ws,
+#                     enclos = NULL)
+#                     
+# ## or   
+# form <- from0to1 ~ fot + gender + adjust(agegr)
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
+#                     adjust = NULL, values = vs, weights = ws,
+#                     enclos = NULL)            
+# 
+# form <- from0to1 ~ fot + adjust(agegr) + adjust(sex)
+# ws2 <- list(agegr = c(0.33, 0.33, 0.33), sex = c(0.5, 0.5))
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
+#                     adjust = NULL, values = vs, weights = ws2,
+#                     enclos = NULL)
+# 
+# ## international standard pops
+# ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+#               status = status %in% 1:2, pophaz = popmort, pp = FALSE,
+#               aggre = list(sex, agegr = cut(dg_age, c(seq(0, 85, 5), Inf)), fot), 
+#               fot = seq(0, 5, 1/12))
+#               
+# form <- from0to1 ~ fot + adjust(agegr)
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
+#                     adjust = NULL, values = vs, weights = "world_1966_18of5",
+#                     enclos = NULL)
+#                     
+# form <- from0to1 ~ fot + adjust(agegr, sex)
+# dt <- makeWeightsDT(ag, formula = form, Surv.response = FALSE,
+#                     adjust = NULL, values = vs, 
+#                     weights = list(agegr = "nordic_2000_18of5", sex=c(1,1)),
+#                     enclos = NULL)
+makeWeightsDT <- function(data, values = NULL, 
+                          print = NULL, adjust = NULL,
+                          formula = NULL, Surv.response = TRUE,
+                          by.other = NULL, custom.levels = NULL, 
+                          custom.levels.cut.low = NULL, weights = NULL, 
+                          internal.weights.values = NULL, 
+                          enclos = NULL, NA.text = NULL) {
+  
+  # environmentalism -----------------------------------------------------------
+  TF <- environment()
+  PF <- parent.frame(1L)
+  if (missing(enclos) || is.null(enclos)) {
+    enclos <- PF 
+  } 
+  
+  enclos <- eval(enclos, envir = TF)
+  
+  if (!is.environment(enclos)) {
+    stop("Argument 'enclos' is not an environment. (Probably internal error, ",
+         "meaning you should complain to the package maintainer if you are not",
+         "doing something silly.)")
+  }
+  
+  THIS_CALL <- match.call()
+  
+  ## dataism -------------------------------------------------------------------
+  if (!is.data.frame(data)) stop("data must be a data.frame")
+  ## tmpDum for convenience. will be deleted in the end. (if no tabulating vars)
+  origData <- data
+  tmpDum <- makeTempVarName(origData, pre = "dummy_")
+  data <- data.table(rep(1L, nrow(origData)))
+  setnames(data, 1, tmpDum)
+
+  
+  # formula: vars to print and adjust by ---------------------------------------
+  
+  adSub <- adjust
+  adjust <- evalPopArg(data = origData, arg = adSub, DT = TRUE, enclos = enclos)
+  
+  if (!is.null(formula)) {
+    
+    foList <- usePopFormula(formula, adjust = adjust, 
+                            data = origData, enclos = enclos, 
+                            Surv.response = Surv.response)
+    print <- foList$print
+    adjust <- foList$adjust
+  } else {
+    
+    prSub <- substitute(print)
+    print <- evalPopArg(data = origData, arg = prSub, DT = TRUE, enclos = enclos)
+    
+    
+  }
+  
+  if (length(weights) && length(adjust)) {
+    checkWeights(weights, adjust = adjust)
+  }
+  
+  # variables to print by ----------------------------------------------------
+  prVars <- tmpDum
+  if (length(print) > 0) {
+    prVars <- names(print)
+    data[, c(prVars) := TF$print]
+    data[, c(tmpDum) := NULL]
+  } 
+  rm(print)
+  
+  # standardization ----------------------------------------------------------
+  ## note: adjust evaluated above with formula
+  adVars <- NULL
+  if (length(adjust) > 0) {
+    adVars <- names(adjust)
+    data[, c(adVars) := TF$adjust]
+  } 
+  rm(adjust)
+  
+  if (is.null(weights) && length(adVars)) {
+    stop("Variables to adjust by were defined but no weights were supplied.")
+  }
+  
+  if (!length(adVars)) {
+    
+    if (!is.null(weights)) {
+      message("NOTE: Weights ignored since no adjusting variables given")
+    }
+    
+    weights <- NULL
+  }
+  
+  # variables to sum -----------------------------------------------------------
+  if (!is.list(values)) stop("Argument 'values' must be a list ",
+                             "(internal error: complain to the package",
+                             "maintainer if you see this)")
+  values <- lapply(values, function(x) {
+    evalPopArg(data = origData, arg = x, DT = TRUE, enclos = enclos)
+  })
+  for (dt in setdiff(seq_along(values), 1L)) {
+    values[[1L]] <- cbind(values[[1L]], values[[dt]])
+  }
+  values <- values[[1L]]
+  vaVars <- NULL
+  if (nrow(values) != nrow(data)) {
+    stop("mismatch in numers of rows in data (", nrow(data), 
+         ") and 'values' (", nrow(values), "). If you see this message, ",
+         "complain to the package maintainer.")
+  }
+  
+  if (length(values) > 0) {
+    vaVars <- names(values)
+    data[, c(vaVars) := TF$values]
+  } else {
+    stop("no values given to sum!")
+  }
+  rm(values)
+  
+  # additionally, values to compute internal weights by: -----------------------
+  iwVar <- NULL
+  if (is.character(weights) && 
+      pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
+    iw <- substitute(internal.weights.values)
+    iw <- evalPopArg(data = origData, iw, DT = TRUE,
+                     enclos = PF, recursive = TRUE,
+                     types = c("character", "expression", "list", "NULL"))
+    
+    if (length(iw) > 1L) stop("Argument 'internal.weights.values' ",
+                              "must produce only one column.")
+    if (length(iw) == 1L && is.character(weights) && 
+        pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
+      iwVar <- makeTempVarName(names=c(names(data), names(origData)), pre = "iw_")
+      data[, c(iwVar) := TF$iw]
+    }
+    
+    if (length(iwVar) == 0L) {
+      stop("Requested computing internal weights, but no values to compute ",
+           "internals weights with were supplied (internal error: If you see ",
+           "this, complain to the package maintainer).")
+    }
+    rm(iw)
+  }
+  
+  
+  # other category vars to keep ------------------------------------------------
+  boSub <- by.other
+  by.other <- evalPopArg(data = origData, arg = boSub, DT = TRUE, enclos = enclos)
+  boVars <- NULL
+  if (length(by.other) > 0) {
+    boVars <- names(by.other)
+    data[, c(boVars) := TF$by.other]
+  } 
+  rm(by.other)
+  
+  # check for aliased columns --------------------------------------------------
+  aliased_cols(data, cols = c(prVars, adVars, boVars))
+  
+  # check for conflicting column names -----------------------------------------
+  dupCols <- c(prVars, adVars, boVars, vaVars, iwVar)
+  dupCols <- unique(dupCols[duplicated(dupCols)])
+  if (length(dupCols) > 0L) {
+    dupCols <- paste0("'", dupCols, "'", collapse = ", ")
+    stop("Following column names duplicated (columns created by arguments ", 
+         "print, adjust, etc.): ", dupCols, ". If you see this, please ensure ",
+         "you are not passing e.g. the same column to both for adjusting ",
+         "and stratification (printing).")
+  }
+  
+  # check for NA values --------------------------------------------------------
+  ## NOTE: NA values of print/by.other are OK. values/adjust are NOT.
+  
+    NAs <- data[, lapply(.SD, function(x) is.na(x)), .SDcols = c(vaVars, iwVar, adVars)]
+    NAs <- rowSums(NAs) > 0L
+    if (sum(NAs)) {
+      if (!is.null(NA.text)) {
+        NA.text <- gsub(x = NA.text, pattern = "%%NA_COUNT%%", 
+                        replacement = sum(NAs))
+        warning(NA.text)
+      }
+      data <- data[!NAs]
+    }
+
+  # inflate data ---------------------------------------------------------------
+  ## on the other hand we aggregate data to levels of print, adjust and 
+  ## by.other; on the other hand the data will always have tabulating variables
+  ## represented as cross-joined, e.g. merge(1:5, 1:2).
+  ## this means some rows might have zeros as values in the 'values'
+  ## columns.
+  ## (necessary for correct standardization with weights)
+  
+  ## NOTE: have to do CJ by hand: some levels of adjust or something may not
+  ## have each level of e.g. fot repeated!
+  
+  sortedLevs <- function(x) {
+    if (!is.factor(x)) return(sort(unique(x)))
+    
+    factor(levels(x), levels(x), levels(x))
+  }
+  cj <- list()
+  cj <- lapply(data[, .SD, .SDcols =  c(prVars, adVars, boVars)], sortedLevs)
+  
+  ## e.g. if data only has fot = seq(0, 4, 1/12), but want to display table
+  ## with fot = seq(0, 5, 1/12). Very important sometimes for handy usage
+  ## of weights.
+  if (length(custom.levels) > 0) cj[names(custom.levels)] <- custom.levels
+  
+  
+  ## SPECIAL: if e.g. a survival time scale with breaks seq(0, 5, 1/12)
+  ## is to be "compressed" to breaks 0:5, and the latter breaks were passed
+  ## via custom.levels, the following ensures e.g. intervals between 0 and 1
+  ## are aggregated to the same row in what follows after.
+  if (!is.null(custom.levels.cut.low)) {
+    cl_msg <- paste0("Internal error: tried to cut() variables in  ",
+                     "internally used work data that did not exist. ",
+                     "If you see this, complain to the ",
+                     "package maintainer. Bad variables: %%VARS%%.")
+    all_names_present(data, custom.levels.cut.low, msg = cl_msg)
+    all_names_present(cj, custom.levels.cut.low, msg = cl_msg)
+    
+    for (var in custom.levels.cut.low) {
+      set(data, j = var, value = cutLow(data[[var]], breaks = cj[[var]]))
+    }
+    
+    ## NOTE: if used cutlow(), then assume passed values via custom.levels
+    ## also contained the roof of the values which should not be repeated.
+    cj[custom.levels.cut.low] <- lapply(cj[custom.levels.cut.low],
+                                        function(elem) elem[-length(elem)])
+  }
+  
+  ## form data.table to merge by - the merge will inflate the data.
+  cj <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), cj)
+  
+  ## inflate & aggregate.
+  setkeyv(data, c(prVars, adVars, boVars))
+  data <- data[cj, lapply(.SD, sum), .SDcols = c(vaVars, iwVar), by = .EACHI]
+  
+  for (k in c(vaVars, iwVar)) {
+    data[is.na(get(k)), (k) := 0]
+  }
+  
+  setcolsnull(data, tmpDum)
+  prVars <- setdiff(prVars, tmpDum); if (length(prVars) == 0) prVars <- NULL
+  
+  ## merge in weights ----------------------------------------------------------
+  
+  if (!is.null(weights)) {
+    
+    if (is.list(weights) && !is.data.frame(weights)) {
+      ## in case one of the elements is a standardization scheme string,
+      ## such as "world_x_y". 
+      whChar <- which(unlist(lapply(weights, is.character)))
+      if (sum(whChar)) {
+        
+        weights[whChar] <- lapply(weights[whChar], function(string) {
+          ## expected to return data.frame with 1) age groups 2) weights
+          ## as columns.
+          stdr.weights(string)[[2]]
+        })
+        
+      }
+      
+      ## now list only contains numeric weights i hope.
+      
+    }
+    
+    ## NOTE: adjust used here to contain levels of adjust arguments only
+    adjust <- list()
+    if (length(adVars) > 0L) {
+      adjust <- lapply(data[, eval(adVars), with = FALSE], sortedLevs)
+    }
+    
+    if (is.character(weights)) {
+      
+      if (pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
+        
+        
+        all_names_present(data, iwVar,
+                          msg = paste0(
+                            "Internal error: expected to have variable ",
+                            "%%VARS%% in working data but didn't. Complain ",
+                            "to the pkg maintainer if you see this."
+                          ))
+        
+        weights <- lapply(seq_along(adjust), function(i) {
+          cn <- names(adjust)[i]
+          le <- unique(adjust[[i]])
+          le <- structure(list(le), names = cn)
+          data[le, lapply(.SD, sum), .SDcols = iwVar, by = .EACHI, on = cn][[iwVar]]
+        })
+        names(weights) <- names(adjust)
+        
+        
+        setcolsnull(data, iwVar)
+        setkeyv(data, c(prVars, adVars, boVars))
+      } else {
+        ## expected to return data.frame with 1) age groups 2) weights
+        ## as columns.
+        weights <- stdr.weights(weights)
+        weights <- weights[[2]]
+        
+      }
+    } 
+    
+    if (!is.data.frame(weights) && is.vector(weights)) {
+      ## note: lists are vectors
+      if (!is.list(weights)) {
+        weights <- list(weights) ## was a vector of values
+        setattr(weights, "names", adVars[1])
+      }
+      
+      weVars <- names(weights)
+      weights <- weights[names(adjust)]
+      
+      adjust <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), adjust)
+      weights <- do.call(function(...) CJ(..., unique = FALSE, sorted = FALSE), weights)
+      
+      weVars <- paste0(weVars, ".w")
+      setnames(weights, adVars, weVars)
+      weights[, (adVars) := adjust]
+      
+      set(weights, j = "weights", value = 1L)
+      for (k in weVars) {
+        set(weights, j = "weights", value = weights$weights * weights[[k]])
+      }
+      setcolsnull(weights, delete = weVars, soft = FALSE)
+      
+      ## NOTE: weights will be repeated for each level of print,
+      ## and for each level of print the weights must sum to one for things
+      ## to work.
+      weights[, "weights" := weights/sum(weights)]
+      
+    }
+    
+    if (!is.data.frame(weights)) {
+      stop("Something went wrong: 'weights' was not collated into a ",
+           "data.frame to merge with data. ",
+           "Blame the package maintainer please!")
+    }
+    ## at this points weights is a data.frame.
+    weights <- data.table(weights)
+    weights[, "weights" := as.double(weights)]
+    
+    ## ensure repetition by print levels if some adjust levels
+    ## that exist in weights do not exist in data.
+    ## NOTE: weights data.frame has at least as many levels as adjust column
+    ## in data (or it has more sometimes).
+    wm <- lapply(adVars, function(chStr) {
+      col <- weights[[chStr]]
+      if (is.factor(col)) return(levels(col))
+      sort(unique(col))
+      })
+    names(wm) <- adVars
+    if (length(prVars)) {
+      wm[prVars] <- lapply(prVars, function(chStr) {
+        col <- data[[chStr]]
+        if (is.factor(col)) return(levels(col))
+        sort(unique(col))
+      })
+    }
+    wm <- setDT(do.call(CJ, wm))
+    
+    weights <- merge(wm, weights, by = adVars, all.x = TRUE, all.y = TRUE)
+    
+    dt_robust_by(
+      'weights[, "weights" := weights/sum(weights), by = %%BY_VAR_NMS%%]',
+      by.var.nms = prVars
+    )
+    
+    data[i = weights, on = c(prVars, adVars), j = "weights" := weights]
+    
+    if (any(is.na(data$weights))) {
+      ## should not be any NAs since we checked for level congruence
+      ## in checkWeights
+      stop("Internal error: some weights were NA after merging to working ",
+           "data. Complain to the package maintainer if you see this.")
+    }
+    
+    
+  }
+  
+  setattr(data, "makeWeightsDT", list(prVars = prVars, adVars = adVars, 
+                                      boVars = boVars, vaVars = vaVars, 
+                                      NAs = NAs))
+  return(data[])
+  
+}
+
+checkCharWeights <- function(w) {
+  if (is.character(w)) {
+    if (length(w) != 1L) {
+      stop("weights supplied as a character string must be of length one.")
+    }
+    if (!pmatch(w, c("internal", "cohort"), nomatch = 0L)) {
+      stdr.weights(w)
+    }
+  }
+}
+
+checkWeights <- function(weights, adjust) {
+  ## INTENTION: given a list/DF/vector/string specifying weights
+  ## and a data.frame/list of the adjusting variables,
+  ## checks they are congruent and complains if not.
+  allowed_classes <- c("list","data.frame","integer","numeric","character",
+                       "NULL")
+  if (!any(class(weights) %in% allowed_classes)) {
+    stop("weights must be either a list, a data.frame, a numeric variable, ",
+         "or a character string specifing the weighting scheme to use. ",
+         "See ?direct_standardization for more information.") 
+  }
+  
+  if (is.list(weights) && !is.data.frame(weights) && 
+      length(adjust) != length(weights)) {
+    stop("Mismatch in numbers of variables (NOT necessarily in the numbers of ",
+         "levels/values within the variables) in adjust (", length(adjust), 
+         " variables) and weights (", length(weights)," variables); ",
+         "make sure each given weights vector has a corresponding ",
+         "variable in adjust and vice versa. ",
+         "See ?direct_standardization for more information.")
+  }
+  
+  if (is.list(weights)) {
+    isChar <- unlist(lapply(weights, is.character))
+    if (any(isChar)) {
+      lapply(weights[isChar], checkCharWeights)
+      weights[isChar] <- lapply(weights[isChar], function(string) {
+        if (pmatch(string, c("cohort", "internal"), nomatch = 0L)) {
+          stop("List of weights had 'cohort' or 'internal' as at least one ",
+               "element, which is currently not supported. ",
+               "See ?direct_standardization for more information.")
+        }
+        stdr.weights(string)[[2]]
+        })
+    }
+  }
+
+  if (is.character(weights)) {
+    checkCharWeights(weights)
+    if (pmatch(weights, c("internal", "cohort"), nomatch = 0L)) {
+      ## done checking since internal weights are pretty fool-proof.
+      return(invisible())
+    }
+    ## if not, pass along as vector of weights.
+    weights <- stdr.weights(weights)[[2]]
+  }
+  
+  if (is.numeric(weights)) { 
+    if (length(adjust) != 1L) {
+      stop("Weights is a numeric vector of weights, ",
+           "but there are more or less than one adjusting variable. ",
+           "See ?direct_standardization for more information.")
+    }
+    weights <- list(weights)
+    names(weights) <- names(adjust)
+  }
+  
+  ## by now either a list or a data.frame of weights...
+  adVars <- names(adjust)
+  weVars <- names(weights)
+  if (is.data.frame(weights)) {
+    if (!"weights" %in% weVars) {
+      stop("data.frame of weights did not have column named 'weights'. ",
+           "see ?direct_standardization for more information.")
+    }
+    weVars <- setdiff(weVars, "weights")
+  }
+  
+  badAdVars <- setdiff(adVars, weVars)
+  badWeVars <- setdiff(weVars, adVars)
+  if (length(badAdVars) > 0) {
+    stop("Mismatch in names of variables in adjust and weights; ",
+         "following adjust variables not mentioned in weights: ", 
+         paste0("'", badAdVars, "'", collapse = ", "))
+  }
+  
+  if (length(badWeVars) > 0) {
+    stop("Mismatch in names of variables in adjust and weights; ",
+         "following weights variables not mentioned in adjust: ",
+         paste0("'", badWeVars, "'", collapse = ", "))
+  }
+  
+  if (is.data.frame(weights)) {
+    
+    levDiff <- lapply(names(adjust), function(var) {
+      !all(adjust[[var]] %in% weights[[var]])
+    })
+    levDiff <- unlist(levDiff)
+    if (any(levDiff)) {
+      ## take only first conflicting variable for brevity of error message-
+      badVar <- names(adjust)[1]
+      badLevs <- setdiff(adjust[[badVar]], weights[[badVar]])
+      badLevs <- paste0("'", badLevs, "'", collapse = ", ")
+      stop("Missing levels in weights data.frame in variable '", badVar, "': ",
+           badLevs, ". These levels were found to exist in the corresponding ",
+           "adjusting variable. ",
+           "Usual suspects: adjusting variable is a factor and you ",
+           "only supplied weights for unique values in your data ",
+           "as opposed to the levels of the factor, which may contain levels ",
+           "that no row has. Try table(yourdata$yourvariable).")
+    }
+    
+  } else {
+    weights <- as.list(weights)
+    weights <- weights[adVars]
+    
+    ## check variable levels
+    adjust <- lapply(adjust, function(elem) {
+      if (is.factor(elem)) {
+        levels(elem)
+      } else {
+        sort(unique(elem))
+      }
+    })
+    
+    weLen <- unlist(lapply(weights, length))
+    adLen <- unlist(lapply(adjust, length))
+    badLen <- names(adjust)[weLen != adLen]
+    
+    if (length(badLen) > 0) {
+      stop("Mismatch in numbers of levels/unique values in adjusting variables ",
+           "and lengths of corresponding weights vectors. ",
+           "Names of mismatching variables: ", 
+           paste0("'", badLen, "'", collapse = ", "), ". There were ",
+           weLen[weLen != adLen], " weights and ", adLen[weLen != adLen], 
+           " adjusting variable levels.")
+    }
+  }
+  
+  
+  
+  invisible()
+}
+
+
+
+
+
+
+
+
diff --git a/README.md b/README.md
index 07f3def..cdea348 100644
--- a/README.md
+++ b/README.md
@@ -1,194 +1,194 @@
-[![Build
-Status](https://travis-ci.org/FinnishCancerRegistry/popEpi.png?branch=master)](https://travis-ci.org/FinnishCancerRegistry/popEpi)
-[![AppVeyor Build
-Status](https://ci.appveyor.com/api/projects/status/github/FinnishCancerRegistry/popEpi?branch=master&svg=true)](https://ci.appveyor.com/project/FinnishCancerRegistry/popepi)
-[![CRAN_Status_Badge](https://www.r-pkg.org/badges/version/popEpi)](https://cran.r-project.org/package=popEpi)
-[![codecov.io](https://codecov.io/github/FinnishCancerRegistry/popEpi/coverage.svg?branch=master)](https://codecov.io/github/FinnishCancerRegistry/popEpi?branch=master)
-[![CRAN_DLs_via_RStudio](https://cranlogs.r-pkg.org/badges/popEpi)](https://cran.r-project.org/package=popEpi)
-
-# popEpi: Epidemiology with population data
-
-The purpose of popEpi is to facilitate computing certain epidemiological
-statistics where population data is used. Current main attractions:
-
-## Splitting, merging population hazards, and aggregating
-
-the `lexpand` function allows users to split their subject-level
-follow-up data into sub-intervals along age, follow-up time and calendar
-time, merge corresponding population hazard information to those
-intervals, and to aggregate the resulting data if needed.
-
-``` r
-data(sire)
-sr <- sire[1,]
-print(sr)
-#>    sex    bi_date    dg_date    ex_date status   dg_age
-#> 1:   1 1952-05-27 1994-02-03 2012-12-31      0 41.68877
-```
-
-``` r
-x <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status %in% 1:2, 
-             fot = 0:5, per = 1994:2000)
-print(x)
-#>     lex.id      fot     per      age    lex.dur lex.Cst lex.Xst sex    bi_date
-#>  1:      1 0.000000 1994.09 41.68877 0.90958904       0       0   1 1952-05-27
-#>  2:      1 0.909589 1995.00 42.59836 0.09041096       0       0   1 1952-05-27
-#>  3:      1 1.000000 1995.09 42.68877 0.90958904       0       0   1 1952-05-27
-#>  4:      1 1.909589 1996.00 43.59836 0.09041096       0       0   1 1952-05-27
-#>  5:      1 2.000000 1996.09 43.68877 0.90958904       0       0   1 1952-05-27
-#>  6:      1 2.909589 1997.00 44.59836 0.09041096       0       0   1 1952-05-27
-#>  7:      1 3.000000 1997.09 44.68877 0.90958904       0       0   1 1952-05-27
-#>  8:      1 3.909589 1998.00 45.59836 0.09041096       0       0   1 1952-05-27
-#>  9:      1 4.000000 1998.09 45.68877 0.90958904       0       0   1 1952-05-27
-#> 10:      1 4.909589 1999.00 46.59836 0.09041096       0       0   1 1952-05-27
-#>        dg_date    ex_date status   dg_age
-#>  1: 1994-02-03 2012-12-31      0 41.68877
-#>  2: 1994-02-03 2012-12-31      0 41.68877
-#>  3: 1994-02-03 2012-12-31      0 41.68877
-#>  4: 1994-02-03 2012-12-31      0 41.68877
-#>  5: 1994-02-03 2012-12-31      0 41.68877
-#>  6: 1994-02-03 2012-12-31      0 41.68877
-#>  7: 1994-02-03 2012-12-31      0 41.68877
-#>  8: 1994-02-03 2012-12-31      0 41.68877
-#>  9: 1994-02-03 2012-12-31      0 41.68877
-#> 10: 1994-02-03 2012-12-31      0 41.68877
-```
-
-``` r
-data(popmort)
-x <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status %in% 1:2, 
-             fot = 0:5, per = 1994:2000, pophaz = popmort)
-print(x)
-#>     lex.id      fot     per      age    lex.dur lex.Cst lex.Xst sex    bi_date
-#>  1:      1 0.000000 1994.09 41.68877 0.90958904       0       0   1 1952-05-27
-#>  2:      1 0.909589 1995.00 42.59836 0.09041096       0       0   1 1952-05-27
-#>  3:      1 1.000000 1995.09 42.68877 0.90958904       0       0   1 1952-05-27
-#>  4:      1 1.909589 1996.00 43.59836 0.09041096       0       0   1 1952-05-27
-#>  5:      1 2.000000 1996.09 43.68877 0.90958904       0       0   1 1952-05-27
-#>  6:      1 2.909589 1997.00 44.59836 0.09041096       0       0   1 1952-05-27
-#>  7:      1 3.000000 1997.09 44.68877 0.90958904       0       0   1 1952-05-27
-#>  8:      1 3.909589 1998.00 45.59836 0.09041096       0       0   1 1952-05-27
-#>  9:      1 4.000000 1998.09 45.68877 0.90958904       0       0   1 1952-05-27
-#> 10:      1 4.909589 1999.00 46.59836 0.09041096       0       0   1 1952-05-27
-#>        dg_date    ex_date status   dg_age     pop.haz       pp
-#>  1: 1994-02-03 2012-12-31      0 41.68877 0.001170685 1.000651
-#>  2: 1994-02-03 2012-12-31      0 41.68877 0.001441038 1.000651
-#>  3: 1994-02-03 2012-12-31      0 41.68877 0.001200721 1.001856
-#>  4: 1994-02-03 2012-12-31      0 41.68877 0.001300846 1.001856
-#>  5: 1994-02-03 2012-12-31      0 41.68877 0.001400981 1.003207
-#>  6: 1994-02-03 2012-12-31      0 41.68877 0.002142293 1.003207
-#>  7: 1994-02-03 2012-12-31      0 41.68877 0.002202424 1.005067
-#>  8: 1994-02-03 2012-12-31      0 41.68877 0.001771568 1.005067
-#>  9: 1994-02-03 2012-12-31      0 41.68877 0.002222468 1.007277
-#> 10: 1994-02-03 2012-12-31      0 41.68877 0.002282603 1.007277
-```
-
-``` r
-a <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status %in% 1:2,
-             fot = 0:5, per = 1994:2000, aggre = list(fot, per))
-print(a)
-#>     fot  per       pyrs at.risk from0to0
-#>  1:   0 1994 0.90958904       0        0
-#>  2:   0 1995 0.09041096       1        0
-#>  3:   1 1995 0.90958904       0        0
-#>  4:   1 1996 0.09041096       1        0
-#>  5:   2 1996 0.90958904       0        0
-#>  6:   2 1997 0.09041096       1        0
-#>  7:   3 1997 0.90958904       0        0
-#>  8:   3 1998 0.09041096       1        0
-#>  9:   4 1998 0.90958904       0        0
-#> 10:   4 1999 0.09041096       1        1
-```
-
-## SIRs / SMRs
-
-One can make use of the `sir` function to estimate indirectly
-standardised incidence or mortality ratios (SIRs/SMRs). The data can be
-aggregated by `lexpand` or by other means. While `sir` is simple and
-flexible in itself, one may also use `sirspline` to fit spline functions
-for the effect of e.g. age as a continuous variable on SIRs.
-
-``` r
-data(popmort)
-data(sire)
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-#> dropped 16 rows where entry == exit
-
-se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup', 'year', 'sex'), print = 'fot')
-se
-#> SIR (adjusted by agegroup, year, sex) with 95% confidence intervals (profile) 
-#> Test for homogeneity: p < 0.001 
-#> 
-#>  Total sir: 3.08 (2.99-3.17)
-#>  Total observed: 4559
-#>  Total expected: 1482.13
-#>  Total person-years: 39906 
-#> 
-#> 
-#>    fot observed expected     pyrs  sir sir.lo sir.hi p_value
-#> 1:   0     4264  1214.54 34445.96 3.51   3.41   3.62   0.000
-#> 2:  10      295   267.59  5459.96 1.10   0.98   1.23   0.094
-```
-
-## (Relative) survival
-
-The `survtab` function computes observed, net/relative and
-cause-specific survivals as well as cumulative incidence functions for
-`Lexis` data. Any of the supported survival time functions can be easily
-adjusted by any number of categorical variables if needed.
-
-One can also use `survtab_ag` for aggregated data. This means the data
-does not have to be on the subject-level to compute survival time
-function estimates.
-
-``` r
-library(Epi)
-
-data(sibr)
-sire$cancer <- "rectal"
-sibr$cancer <- "breast"
-sr <- rbind(sire, sibr)
-
-sr$cancer <- factor(sr$cancer)
-sr <- sr[sr$dg_date < sr$ex_date, ]
-
-sr$status <- factor(sr$status, levels = 0:2, 
-                    labels = c("alive", "canD", "othD"))
-
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sr,
-           exit.status = status)
-#> NOTE: entry.status has been set to "alive" for all.
-
-st <- survtab(FUT ~ cancer, data = x,
-              breaks = list(FUT = seq(0, 5, 1/12)),
-              surv.type = "cif.obs")
-st
-#> 
-#> Call: 
-#>  survtab(formula = FUT ~ cancer, data = x, breaks = list(FUT = seq(0, 5, 1/12)), surv.type = "cif.obs") 
-#> 
-#> Type arguments: 
-#>  surv.type: cif.obs --- surv.method: hazard
-#>  
-#> Confidence interval arguments: 
-#>  level: 95 % --- transformation: log-log
-#>  
-#> Totals:
-#>  person-time:62120 --- events: 5375
-#>  
-#> Stratified by: 'cancer'
-#>    cancer Tstop surv.obs.lo surv.obs surv.obs.hi SE.surv.obs CIF_canD CIF_othD
-#> 1: breast   2.5      0.8804   0.8870      0.8933    0.003290   0.0687   0.0442
-#> 2: breast   5.0      0.7899   0.7986      0.8070    0.004368   0.1162   0.0852
-#> 3: rectal   2.5      0.6250   0.6359      0.6465    0.005480   0.2981   0.0660
-#> 4: rectal   5.0      0.5032   0.5148      0.5263    0.005901   0.3727   0.1125
-```
+[![Build
+Status](https://travis-ci.org/FinnishCancerRegistry/popEpi.png?branch=master)](https://travis-ci.org/FinnishCancerRegistry/popEpi)
+[![AppVeyor Build
+Status](https://ci.appveyor.com/api/projects/status/github/FinnishCancerRegistry/popEpi?branch=master&svg=true)](https://ci.appveyor.com/project/FinnishCancerRegistry/popepi)
+[![CRAN_Status_Badge](https://www.r-pkg.org/badges/version/popEpi)](https://cran.r-project.org/package=popEpi)
+[![codecov.io](https://codecov.io/github/FinnishCancerRegistry/popEpi/coverage.svg?branch=master)](https://codecov.io/github/FinnishCancerRegistry/popEpi?branch=master)
+[![CRAN_DLs_via_RStudio](https://cranlogs.r-pkg.org/badges/popEpi)](https://cran.r-project.org/package=popEpi)
+
+# popEpi: Epidemiology with population data
+
+The purpose of popEpi is to facilitate computing certain epidemiological
+statistics where population data is used. Current main attractions:
+
+## Splitting, merging population hazards, and aggregating
+
+the `lexpand` function allows users to split their subject-level
+follow-up data into sub-intervals along age, follow-up time and calendar
+time, merge corresponding population hazard information to those
+intervals, and to aggregate the resulting data if needed.
+
+``` r
+data(sire)
+sr <- sire[1,]
+print(sr)
+#>    sex    bi_date    dg_date    ex_date status   dg_age
+#> 1:   1 1952-05-27 1994-02-03 2012-12-31      0 41.68877
+```
+
+``` r
+x <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status %in% 1:2, 
+             fot = 0:5, per = 1994:2000)
+print(x)
+#>     lex.id      fot     per      age    lex.dur lex.Cst lex.Xst sex    bi_date
+#>  1:      1 0.000000 1994.09 41.68877 0.90958904       0       0   1 1952-05-27
+#>  2:      1 0.909589 1995.00 42.59836 0.09041096       0       0   1 1952-05-27
+#>  3:      1 1.000000 1995.09 42.68877 0.90958904       0       0   1 1952-05-27
+#>  4:      1 1.909589 1996.00 43.59836 0.09041096       0       0   1 1952-05-27
+#>  5:      1 2.000000 1996.09 43.68877 0.90958904       0       0   1 1952-05-27
+#>  6:      1 2.909589 1997.00 44.59836 0.09041096       0       0   1 1952-05-27
+#>  7:      1 3.000000 1997.09 44.68877 0.90958904       0       0   1 1952-05-27
+#>  8:      1 3.909589 1998.00 45.59836 0.09041096       0       0   1 1952-05-27
+#>  9:      1 4.000000 1998.09 45.68877 0.90958904       0       0   1 1952-05-27
+#> 10:      1 4.909589 1999.00 46.59836 0.09041096       0       0   1 1952-05-27
+#>        dg_date    ex_date status   dg_age
+#>  1: 1994-02-03 2012-12-31      0 41.68877
+#>  2: 1994-02-03 2012-12-31      0 41.68877
+#>  3: 1994-02-03 2012-12-31      0 41.68877
+#>  4: 1994-02-03 2012-12-31      0 41.68877
+#>  5: 1994-02-03 2012-12-31      0 41.68877
+#>  6: 1994-02-03 2012-12-31      0 41.68877
+#>  7: 1994-02-03 2012-12-31      0 41.68877
+#>  8: 1994-02-03 2012-12-31      0 41.68877
+#>  9: 1994-02-03 2012-12-31      0 41.68877
+#> 10: 1994-02-03 2012-12-31      0 41.68877
+```
+
+``` r
+data(popmort)
+x <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status %in% 1:2, 
+             fot = 0:5, per = 1994:2000, pophaz = popmort)
+print(x)
+#>     lex.id      fot     per      age    lex.dur lex.Cst lex.Xst sex    bi_date
+#>  1:      1 0.000000 1994.09 41.68877 0.90958904       0       0   1 1952-05-27
+#>  2:      1 0.909589 1995.00 42.59836 0.09041096       0       0   1 1952-05-27
+#>  3:      1 1.000000 1995.09 42.68877 0.90958904       0       0   1 1952-05-27
+#>  4:      1 1.909589 1996.00 43.59836 0.09041096       0       0   1 1952-05-27
+#>  5:      1 2.000000 1996.09 43.68877 0.90958904       0       0   1 1952-05-27
+#>  6:      1 2.909589 1997.00 44.59836 0.09041096       0       0   1 1952-05-27
+#>  7:      1 3.000000 1997.09 44.68877 0.90958904       0       0   1 1952-05-27
+#>  8:      1 3.909589 1998.00 45.59836 0.09041096       0       0   1 1952-05-27
+#>  9:      1 4.000000 1998.09 45.68877 0.90958904       0       0   1 1952-05-27
+#> 10:      1 4.909589 1999.00 46.59836 0.09041096       0       0   1 1952-05-27
+#>        dg_date    ex_date status   dg_age     pop.haz       pp
+#>  1: 1994-02-03 2012-12-31      0 41.68877 0.001170685 1.000651
+#>  2: 1994-02-03 2012-12-31      0 41.68877 0.001441038 1.000651
+#>  3: 1994-02-03 2012-12-31      0 41.68877 0.001200721 1.001856
+#>  4: 1994-02-03 2012-12-31      0 41.68877 0.001300846 1.001856
+#>  5: 1994-02-03 2012-12-31      0 41.68877 0.001400981 1.003207
+#>  6: 1994-02-03 2012-12-31      0 41.68877 0.002142293 1.003207
+#>  7: 1994-02-03 2012-12-31      0 41.68877 0.002202424 1.005067
+#>  8: 1994-02-03 2012-12-31      0 41.68877 0.001771568 1.005067
+#>  9: 1994-02-03 2012-12-31      0 41.68877 0.002222468 1.007277
+#> 10: 1994-02-03 2012-12-31      0 41.68877 0.002282603 1.007277
+```
+
+``` r
+a <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status %in% 1:2,
+             fot = 0:5, per = 1994:2000, aggre = list(fot, per))
+print(a)
+#>     fot  per       pyrs at.risk from0to0
+#>  1:   0 1994 0.90958904       0        0
+#>  2:   0 1995 0.09041096       1        0
+#>  3:   1 1995 0.90958904       0        0
+#>  4:   1 1996 0.09041096       1        0
+#>  5:   2 1996 0.90958904       0        0
+#>  6:   2 1997 0.09041096       1        0
+#>  7:   3 1997 0.90958904       0        0
+#>  8:   3 1998 0.09041096       1        0
+#>  9:   4 1998 0.90958904       0        0
+#> 10:   4 1999 0.09041096       1        1
+```
+
+## SIRs / SMRs
+
+One can make use of the `sir` function to estimate indirectly
+standardised incidence or mortality ratios (SIRs/SMRs). The data can be
+aggregated by `lexpand` or by other means. While `sir` is simple and
+flexible in itself, one may also use `sirspline` to fit spline functions
+for the effect of e.g. age as a continuous variable on SIRs.
+
+``` r
+data(popmort)
+data(sire)
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+#> dropped 16 rows where entry == exit
+
+se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup', 'year', 'sex'), print = 'fot')
+se
+#> SIR (adjusted by agegroup, year, sex) with 95% confidence intervals (profile) 
+#> Test for homogeneity: p < 0.001 
+#> 
+#>  Total sir: 3.08 (2.99-3.17)
+#>  Total observed: 4559
+#>  Total expected: 1482.13
+#>  Total person-years: 39906 
+#> 
+#> 
+#>    fot observed expected     pyrs  sir sir.lo sir.hi p_value
+#> 1:   0     4264  1214.54 34445.96 3.51   3.41   3.62   0.000
+#> 2:  10      295   267.59  5459.96 1.10   0.98   1.23   0.094
+```
+
+## (Relative) survival
+
+The `survtab` function computes observed, net/relative and
+cause-specific survivals as well as cumulative incidence functions for
+`Lexis` data. Any of the supported survival time functions can be easily
+adjusted by any number of categorical variables if needed.
+
+One can also use `survtab_ag` for aggregated data. This means the data
+does not have to be on the subject-level to compute survival time
+function estimates.
+
+``` r
+library(Epi)
+
+data(sibr)
+sire$cancer <- "rectal"
+sibr$cancer <- "breast"
+sr <- rbind(sire, sibr)
+
+sr$cancer <- factor(sr$cancer)
+sr <- sr[sr$dg_date < sr$ex_date, ]
+
+sr$status <- factor(sr$status, levels = 0:2, 
+                    labels = c("alive", "canD", "othD"))
+
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sr,
+           exit.status = status)
+#> NOTE: entry.status has been set to "alive" for all.
+
+st <- survtab(FUT ~ cancer, data = x,
+              breaks = list(FUT = seq(0, 5, 1/12)),
+              surv.type = "cif.obs")
+st
+#> 
+#> Call: 
+#>  survtab(formula = FUT ~ cancer, data = x, breaks = list(FUT = seq(0, 5, 1/12)), surv.type = "cif.obs") 
+#> 
+#> Type arguments: 
+#>  surv.type: cif.obs --- surv.method: hazard
+#>  
+#> Confidence interval arguments: 
+#>  level: 95 % --- transformation: log-log
+#>  
+#> Totals:
+#>  person-time:62120 --- events: 5375
+#>  
+#> Stratified by: 'cancer'
+#>    cancer Tstop surv.obs.lo surv.obs surv.obs.hi SE.surv.obs CIF_canD CIF_othD
+#> 1: breast   2.5      0.8804   0.8870      0.8933    0.003290   0.0687   0.0442
+#> 2: breast   5.0      0.7899   0.7986      0.8070    0.004368   0.1162   0.0852
+#> 3: rectal   2.5      0.6250   0.6359      0.6465    0.005480   0.2981   0.0660
+#> 4: rectal   5.0      0.5032   0.5148      0.5263    0.005901   0.3727   0.1125
+```
diff --git a/build/partial.rdb b/build/partial.rdb
index bff827b..d53b128 100644
Binary files a/build/partial.rdb and b/build/partial.rdb differ
diff --git a/build/vignette.rds b/build/vignette.rds
index 2ac0443..3031c67 100644
Binary files a/build/vignette.rds and b/build/vignette.rds differ
diff --git a/debian/changelog b/debian/changelog
index 0dd7dbe..add68ac 100644
--- a/debian/changelog
+++ b/debian/changelog
@@ -1,3 +1,9 @@
+r-cran-popepi (0.4.10+git20221003.1.6b03392+dfsg-1) UNRELEASED; urgency=low
+
+  * New upstream snapshot.
+
+ -- Debian Janitor <janitor@jelmer.uk>  Thu, 19 Jan 2023 13:56:49 -0000
+
 r-cran-popepi (0.4.10+dfsg-1) unstable; urgency=medium
 
   * Disable reprotest
diff --git a/inst/WORDLIST b/inst/WORDLIST
index 8307025..d579873 100644
--- a/inst/WORDLIST
+++ b/inst/WORDLIST
@@ -1,109 +1,109 @@
-YPLL 
-FinnishCancerRegistry
-Xst
-var
-wald
-underfitting
-Timo
-survtab
-survmean
-Surv
-std
-Stackoverflow
-splitMulti
-splitLexisDT
-splitLexis
-specified
-SMRs
-SMR's
-SMR
-sibr
-specified
-Vol
-sirspline
-SIRs
-SIR
-SIR's
-RStudio
-RSRs
-RSR
-robustified
-Riccardo
-relpois
-pyrs
-popmort
-pophaz
-Pokhrel
-Poisson
-Pohar
-pkg
-Perme
-obs
-Nordcan
-NA's
-mstate 
-mis
-Maja
-lex
-Janez
-io
-ICSS
-IARC
-hotfix
-haz
-Hakulinen
-Github
-github
-frac
-fot
-etc
-Estève
-epitools
-Epi
-EdererII
-Ederer
-Dyban
-Dyba
-DT
-doi
-DLs
-difftime
-dg
-DF
-cutLexis
-Correa
-Corazziari
-com
-coh
-codecov
-CIFs
-CIF
-censorings
-Capocaccia
-Breslow
-args
-AppVeyor
-Altman
-adj
-Esteve
-df
-dt
-dX
-ratetable
-seq
-Pohar
-Perme
-Dickman
-Ederer
-Esteve
-Hakulinen
-Heise
-Sloggett
-aggre
-arg
-dcast
-IDate
-obj
-sim
-Stat
-vars
+YPLL 
+FinnishCancerRegistry
+Xst
+var
+wald
+underfitting
+Timo
+survtab
+survmean
+Surv
+std
+Stackoverflow
+splitMulti
+splitLexisDT
+splitLexis
+specified
+SMRs
+SMR's
+SMR
+sibr
+specified
+Vol
+sirspline
+SIRs
+SIR
+SIR's
+RStudio
+RSRs
+RSR
+robustified
+Riccardo
+relpois
+pyrs
+popmort
+pophaz
+Pokhrel
+Poisson
+Pohar
+pkg
+Perme
+obs
+Nordcan
+NA's
+mstate 
+mis
+Maja
+lex
+Janez
+io
+ICSS
+IARC
+hotfix
+haz
+Hakulinen
+Github
+github
+frac
+fot
+etc
+Estève
+epitools
+Epi
+EdererII
+Ederer
+Dyban
+Dyba
+DT
+doi
+DLs
+difftime
+dg
+DF
+cutLexis
+Correa
+Corazziari
+com
+coh
+codecov
+CIFs
+CIF
+censorings
+Capocaccia
+Breslow
+args
+AppVeyor
+Altman
+adj
+Esteve
+df
+dt
+dX
+ratetable
+seq
+Pohar
+Perme
+Dickman
+Ederer
+Esteve
+Hakulinen
+Heise
+Sloggett
+aggre
+arg
+dcast
+IDate
+obj
+sim
+Stat
+vars
diff --git a/inst/doc/sir.R b/inst/doc/sir.R
index df4b8cb..67f4460 100644
--- a/inst/doc/sir.R
+++ b/inst/doc/sir.R
@@ -1,63 +1,63 @@
-## ---- echo=TRUE, warning=FALSE, message=FALSE---------------------------------
-library(popEpi)
-library(Epi)
-
-## -----------------------------------------------------------------------------
-data(sire)
-data(popmort)
-c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-
-## -----------------------------------------------------------------------------
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-
-## ---- fig.height=3, fig.width=6-----------------------------------------------
-plot(se, col = 2:3)
-title('SMR for follow-up categories')
-
-## ---- fig.height=5, fig.width=6-----------------------------------------------
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines=FALSE)
-
-st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines = TRUE)
-
-plot(sf, col=2, log=TRUE)
-title('Splines fitted in different models')
-
-plot(st, col=4, log=TRUE)
-title('Splines are dependent')
-
-## ---- results='hide', fig.height=5, fig.width=6-------------------------------
-c$year.cat <- ifelse(c$year < 2002, 1, 2)
-sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup'), print = 'year.cat')
-plot(sy, log=TRUE)
-legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
-
-## -----------------------------------------------------------------------------
-print(sy)
-
+## ---- echo=TRUE, warning=FALSE, message=FALSE---------------------------------
+library(popEpi)
+library(Epi)
+
+## -----------------------------------------------------------------------------
+data(sire)
+data(popmort)
+c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+
+## -----------------------------------------------------------------------------
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+
+## ---- fig.height=3, fig.width=6-----------------------------------------------
+plot(se, col = 2:3)
+title('SMR for follow-up categories')
+
+## ---- fig.height=5, fig.width=6-----------------------------------------------
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines=FALSE)
+
+st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines = TRUE)
+
+plot(sf, col=2, log=TRUE)
+title('Splines fitted in different models')
+
+plot(st, col=4, log=TRUE)
+title('Splines are dependent')
+
+## ---- results='hide', fig.height=5, fig.width=6-------------------------------
+c$year.cat <- ifelse(c$year < 2002, 1, 2)
+sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup'), print = 'year.cat')
+plot(sy, log=TRUE)
+legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
+
+## -----------------------------------------------------------------------------
+print(sy)
+
diff --git a/inst/doc/sir.Rmd b/inst/doc/sir.Rmd
index f125f05..97887c4 100644
--- a/inst/doc/sir.Rmd
+++ b/inst/doc/sir.Rmd
@@ -1,158 +1,158 @@
----
-title: "SMR Vignette"
-author: "Matti Rantanen"
-date: "`r Sys.Date()`"
-output: 
-  html_document:
-    fig_caption: yes
-    toc: true
-    toc_depth: 2
-vignette: >
-  %\VignetteEngine{knitr::rmarkdown}
-  %\VignetteIndexEntry{Standardised incidence and mortality ratios}
-  %\usepackage[utf8]{inputenc}
----
-
-```{r, echo=TRUE, warning=FALSE, message=FALSE}
-library(popEpi)
-library(Epi)
-```
-
-# Introduction
-
-Standardized incidence ratio (SIR) or mortality ratio (SMR) is a ratio of observed and expected cases. Observed cases is the absolute number of cases in the cohort. The expected cases are derived by multiplying the cohort person-years with reference populations rate. The rate should be stratified or adjusted by confounding factors. Usually these are age group, gender, calendar period and possibly a cancer type or other confounding variable. Also a social economic status or area variable can be used.
-
-In reference population the expected rate in strata $j$ is $\lambda_j = d_j$ / $n_j$, where $d_j$ is observed cases and $n_j$ is observed person years. Now the SIR can be written as a ratio
-$$
-SIR = \frac{ \sum d_j }{\sum n_j \lambda_j} = \frac{D}{E} 
-$$
-where $D$ is the observed cases in cohort population and $E$ is the expected number. Univariate confidence intervals are based on exact values of Poisson distribution and the formula for p-value is
-$$
-\chi^2 = \frac{ (|O - E| -0.5)^2 }{E}.
-$$
-Modelled SIR is a Poisson regression model with log-link and cohorts person-years as a offset.
-
-The homogeneity of SIR's can be tested using a likelihood ratio test in Poisson modelled SIRs.
-
-The same workflow applies for standardised mortality ratios.
-
-# Splines
-
-A continuous spline function can be fitted for time variables, e.g. age-group. Idea of the splines is to smooth the SMR estimates and do inference from the curve figure. This requires pre-defined knots/nodes that are used to fit the spline curve. Selecting the number of knots and knot places is a very subjective matter and there are three options to pass spline knots to function.
-
-It's good practice to try between different knot settings for realistic spline estimates. Overfitting might cause unintentional artefacts in the estimate and underfitting might smooth away interesting patterns.
-
-The spline variable should be as continuous as possible, say from 18 to 100 time points. But when splitting time in too narrow intervals, random variation might occur in the expected or population rate values. Therefore it's also possible to do two variables for age or period: first with wider intervals for standardisation and second with narrow intervals for the spline.
-
-## Knots
-
-There are three options to for assigning knots to the spline:
-
-1. A vector of numbers of knots for each spline variable. Number of knots includes the boundary knots, so that the minimum number of knots is 2, which is a log linear association. The knots are placed automatically using the quantiles of observed cases.
-
-2. A list of vectors of predefined knot places. Number of vectors needs to match the length of spline variables. And each vector has to have at least the minimum and maximum for boundary knots.
-
-3. NULL will automatically finds the optimal number of knots based on AIC. Knots are placed according the quantiles of observed cases. This is usually a good place to start the fitting process.
-
-Number of knots and knot places are always found in output. 
-
-# SMR
-
-## Mortality: External cohort and popmort data
-
-Estimate SMR of a simulated cohort of Finnish female rectal cancer patients, `sire`.
-Death rates for each age, period and sex is available in `popmort` dataset.
-
-For more information about the dataset see `help(popmort)` and `help(sire)`.
-
-```{r}
-data(sire)
-data(popmort)
-c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-```
-
-SMR's for other causes is 1 for both follow-up intervals. Also the p-value suggest that there is no heterogeneity between SMR estimates (p=0.735).
-
-
-The total mortality can be estimated by modifying the `status` argument. Now we want to account all deaths, i.e. status is 1 or 2.
-
-```{r}
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-```
-
-Now the estimates for follow-up intervals seems to differ significantly, p = 0. Plotting SMR (S3-method for `sir`-object) is easily done using default plot-function.
-
-```{r, fig.height=3, fig.width=6}
-plot(se, col = 2:3)
-title('SMR for follow-up categories')
-```
-
-
-## splines
-
-
-Lets fit splines for the follow-up time and age group using two different options: the splines are fitted in different model and in same model, `dependent.splines`.
-
-```{r, fig.height=5, fig.width=6}
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines=FALSE)
-
-st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines = TRUE)
-
-plot(sf, col=2, log=TRUE)
-title('Splines fitted in different models')
-
-plot(st, col=4, log=TRUE)
-title('Splines are dependent')
-```
-
-In dependent spline the `fot` is the ratio with zero time as reference point. Reference points can be altered. Here age group profile is assumed to be same for every follow-up time. SMR is 0.2 times from 0 to 10 years of follow-up. 
-
-
-Splines can also be stratified using the `print` argument. For example we split the death time in two time periods and test if the age group splines are equal.
-
-```{r, results='hide', fig.height=5, fig.width=6}
-c$year.cat <- ifelse(c$year < 2002, 1, 2)
-sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup'), print = 'year.cat')
-plot(sy, log=TRUE)
-legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
-```
-
-For category before 2002 the SMR seems to be higher after the age of 50. Also the p-value (<0.0001) indicates that there is a difference in age group trends before and after year 2002. P-value is a likelihood ratio test that compares models where splines are fitted together and separately.
-
-```{r}
-print(sy)
-```
-
-
-
-
-
-
-
+---
+title: "SMR Vignette"
+author: "Matti Rantanen"
+date: "`r Sys.Date()`"
+output: 
+  html_document:
+    fig_caption: yes
+    toc: true
+    toc_depth: 2
+vignette: >
+  %\VignetteEngine{knitr::rmarkdown}
+  %\VignetteIndexEntry{Standardised incidence and mortality ratios}
+  %\usepackage[utf8]{inputenc}
+---
+
+```{r, echo=TRUE, warning=FALSE, message=FALSE}
+library(popEpi)
+library(Epi)
+```
+
+# Introduction
+
+Standardized incidence ratio (SIR) or mortality ratio (SMR) is a ratio of observed and expected cases. Observed cases is the absolute number of cases in the cohort. The expected cases are derived by multiplying the cohort person-years with reference populations rate. The rate should be stratified or adjusted by confounding factors. Usually these are age group, gender, calendar period and possibly a cancer type or other confounding variable. Also a social economic status or area variable can be used.
+
+In reference population the expected rate in strata $j$ is $\lambda_j = d_j$ / $n_j$, where $d_j$ is observed cases and $n_j$ is observed person years. Now the SIR can be written as a ratio
+$$
+SIR = \frac{ \sum d_j }{\sum n_j \lambda_j} = \frac{D}{E} 
+$$
+where $D$ is the observed cases in cohort population and $E$ is the expected number. Univariate confidence intervals are based on exact values of Poisson distribution and the formula for p-value is
+$$
+\chi^2 = \frac{ (|O - E| -0.5)^2 }{E}.
+$$
+Modelled SIR is a Poisson regression model with log-link and cohorts person-years as a offset.
+
+The homogeneity of SIR's can be tested using a likelihood ratio test in Poisson modelled SIRs.
+
+The same workflow applies for standardised mortality ratios.
+
+# Splines
+
+A continuous spline function can be fitted for time variables, e.g. age-group. Idea of the splines is to smooth the SMR estimates and do inference from the curve figure. This requires pre-defined knots/nodes that are used to fit the spline curve. Selecting the number of knots and knot places is a very subjective matter and there are three options to pass spline knots to function.
+
+It's good practice to try between different knot settings for realistic spline estimates. Overfitting might cause unintentional artefacts in the estimate and underfitting might smooth away interesting patterns.
+
+The spline variable should be as continuous as possible, say from 18 to 100 time points. But when splitting time in too narrow intervals, random variation might occur in the expected or population rate values. Therefore it's also possible to do two variables for age or period: first with wider intervals for standardisation and second with narrow intervals for the spline.
+
+## Knots
+
+There are three options to for assigning knots to the spline:
+
+1. A vector of numbers of knots for each spline variable. Number of knots includes the boundary knots, so that the minimum number of knots is 2, which is a log linear association. The knots are placed automatically using the quantiles of observed cases.
+
+2. A list of vectors of predefined knot places. Number of vectors needs to match the length of spline variables. And each vector has to have at least the minimum and maximum for boundary knots.
+
+3. NULL will automatically finds the optimal number of knots based on AIC. Knots are placed according the quantiles of observed cases. This is usually a good place to start the fitting process.
+
+Number of knots and knot places are always found in output. 
+
+# SMR
+
+## Mortality: External cohort and popmort data
+
+Estimate SMR of a simulated cohort of Finnish female rectal cancer patients, `sire`.
+Death rates for each age, period and sex is available in `popmort` dataset.
+
+For more information about the dataset see `help(popmort)` and `help(sire)`.
+
+```{r}
+data(sire)
+data(popmort)
+c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+```
+
+SMR's for other causes is 1 for both follow-up intervals. Also the p-value suggest that there is no heterogeneity between SMR estimates (p=0.735).
+
+
+The total mortality can be estimated by modifying the `status` argument. Now we want to account all deaths, i.e. status is 1 or 2.
+
+```{r}
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+```
+
+Now the estimates for follow-up intervals seems to differ significantly, p = 0. Plotting SMR (S3-method for `sir`-object) is easily done using default plot-function.
+
+```{r, fig.height=3, fig.width=6}
+plot(se, col = 2:3)
+title('SMR for follow-up categories')
+```
+
+
+## splines
+
+
+Lets fit splines for the follow-up time and age group using two different options: the splines are fitted in different model and in same model, `dependent.splines`.
+
+```{r, fig.height=5, fig.width=6}
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines=FALSE)
+
+st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines = TRUE)
+
+plot(sf, col=2, log=TRUE)
+title('Splines fitted in different models')
+
+plot(st, col=4, log=TRUE)
+title('Splines are dependent')
+```
+
+In dependent spline the `fot` is the ratio with zero time as reference point. Reference points can be altered. Here age group profile is assumed to be same for every follow-up time. SMR is 0.2 times from 0 to 10 years of follow-up. 
+
+
+Splines can also be stratified using the `print` argument. For example we split the death time in two time periods and test if the age group splines are equal.
+
+```{r, results='hide', fig.height=5, fig.width=6}
+c$year.cat <- ifelse(c$year < 2002, 1, 2)
+sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup'), print = 'year.cat')
+plot(sy, log=TRUE)
+legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
+```
+
+For category before 2002 the SMR seems to be higher after the age of 50. Also the p-value (<0.0001) indicates that there is a difference in age group trends before and after year 2002. P-value is a likelihood ratio test that compares models where splines are fitted together and separately.
+
+```{r}
+print(sy)
+```
+
+
+
+
+
+
+
diff --git a/inst/doc/sir.html b/inst/doc/sir.html
new file mode 100644
index 0000000..c121cfc
--- /dev/null
+++ b/inst/doc/sir.html
@@ -0,0 +1,649 @@
+<!DOCTYPE html>
+
+<html>
+
+<head>
+
+<meta charset="utf-8" />
+<meta name="generator" content="pandoc" />
+<meta http-equiv="X-UA-Compatible" content="IE=EDGE" />
+
+
+<meta name="author" content="Matti Rantanen" />
+
+<meta name="date" content="2023-01-19" />
+
+<title>SMR Vignette</title>
+
+<script>// Pandoc 2.9 adds attributes on both header and div. We remove the former (to
+// be compatible with the behavior of Pandoc < 2.8).
+document.addEventListener('DOMContentLoaded', function(e) {
+  var hs = document.querySelectorAll("div.section[class*='level'] > :first-child");
+  var i, h, a;
+  for (i = 0; i < hs.length; i++) {
+    h = hs[i];
+    if (!/^h[1-6]$/i.test(h.tagName)) continue;  // it should be a header h1-h6
+    a = h.attributes;
+    while (a.length > 0) h.removeAttribute(a[0].name);
+  }
+});
+</script>
+<script>/*! jQuery v3.6.1 | (c) OpenJS Foundation and other contributors | jquery.org/license */
+!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(e.document)return t(e);throw new Error("jQuery requires a window with a document")}:t(e)}("undefined"!=typeof window?window:this,function(w,R){"use strict";function v(e){return"function"==typeof e&&"number"!=typeof e.nodeType&&"function"!=typeof e.item}function g(e){return null!=e&&e===e.window}var t=[],M=Object.getPrototypeOf,s=t.slice,I=t.flat?function(e){return t.flat.call(e)}:function(e){return t.concat.apply([],e)},W=t.push,F=t.indexOf,$={},B=$.toString,_=$.hasOwnProperty,z=_.toString,U=z.call(Object),y={},T=w.document,X={type:!0,src:!0,nonce:!0,noModule:!0};function V(e,t,n){var r,i,o=(n=n||T).createElement("script");if(o.text=e,t)for(r in X)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function h(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?$[B.call(e)]||"object":typeof e}var e="3.6.1",C=function(e,t){return new C.fn.init(e,t)};function G(e){var t=!!e&&"length"in e&&e.length,n=h(e);return!v(e)&&!g(e)&&("array"===n||0===t||"number"==typeof t&&0<t&&t-1 in e)}C.fn=C.prototype={jquery:e,constructor:C,length:0,toArray:function(){return s.call(this)},get:function(e){return null==e?s.call(this):e<0?this[e+this.length]:this[e]},pushStack:function(e){e=C.merge(this.constructor(),e);return e.prevObject=this,e},each:function(e){return C.each(this,e)},map:function(n){return this.pushStack(C.map(this,function(e,t){return n.call(e,t,e)}))},slice:function(){return this.pushStack(s.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},even:function(){return this.pushStack(C.grep(this,function(e,t){return(t+1)%2}))},odd:function(){return this.pushStack(C.grep(this,function(e,t){return t%2}))},eq:function(e){var t=this.length,e=+e+(e<0?t:0);return this.pushStack(0<=e&&e<t?[this[e]]:[])},end:function(){return this.prevObject||this.constructor()},push:W,sort:t.sort,splice:t.splice},C.extend=C.fn.extend=function(){var e,t,n,r,i,o=arguments[0]||{},a=1,s=arguments.length,u=!1;for("boolean"==typeof o&&(u=o,o=arguments[a]||{},a++),"object"==typeof o||v(o)||(o={}),a===s&&(o=this,a--);a<s;a++)if(null!=(e=arguments[a]))for(t in e)n=e[t],"__proto__"!==t&&o!==n&&(u&&n&&(C.isPlainObject(n)||(r=Array.isArray(n)))?(i=o[t],i=r&&!Array.isArray(i)?[]:r||C.isPlainObject(i)?i:{},r=!1,o[t]=C.extend(u,i,n)):void 0!==n&&(o[t]=n));return o},C.extend({expando:"jQuery"+(e+Math.random()).replace(/\D/g,""),isReady:!0,error:function(e){throw new Error(e)},noop:function(){},isPlainObject:function(e){return!(!e||"[object Object]"!==B.call(e))&&(!(e=M(e))||"function"==typeof(e=_.call(e,"constructor")&&e.constructor)&&z.call(e)===U)},isEmptyObject:function(e){for(var t in e)return!1;return!0},globalEval:function(e,t,n){V(e,{nonce:t&&t.nonce},n)},each:function(e,t){var n,r=0;if(G(e)){for(n=e.length;r<n;r++)if(!1===t.call(e[r],r,e[r]))break}else for(r in e)if(!1===t.call(e[r],r,e[r]))break;return e},makeArray:function(e,t){t=t||[];return null!=e&&(G(Object(e))?C.merge(t,"string"==typeof e?[e]:e):W.call(t,e)),t},inArray:function(e,t,n){return null==t?-1:F.call(t,e,n)},merge:function(e,t){for(var n=+t.length,r=0,i=e.length;r<n;r++)e[i++]=t[r];return e.length=i,e},grep:function(e,t,n){for(var r=[],i=0,o=e.length,a=!n;i<o;i++)!t(e[i],i)!=a&&r.push(e[i]);return r},map:function(e,t,n){var r,i,o=0,a=[];if(G(e))for(r=e.length;o<r;o++)null!=(i=t(e[o],o,n))&&a.push(i);else for(o in e)null!=(i=t(e[o],o,n))&&a.push(i);return I(a)},guid:1,support:y}),"function"==typeof Symbol&&(C.fn[Symbol.iterator]=t[Symbol.iterator]),C.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(e,t){$["[object "+t+"]"]=t.toLowerCase()});function r(e,t,n){var r=[],i=void 0!==n;while((e=e[t])&&9!==e.nodeType)if(1===e.nodeType){if(i&&C(e).is(n))break;r.push(e)}return r}function Y(e,t){for(var n=[];e;e=e.nextSibling)1===e.nodeType&&e!==t&&n.push(e);return n}var e=function(R){function c(e,t){return e="0x"+e.slice(1)-65536,t||(e<0?String.fromCharCode(65536+e):String.fromCharCode(e>>10|55296,1023&e|56320))}function M(e,t){return t?"\0"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e}function I(){T()}var e,p,b,o,W,d,F,$,w,u,l,T,C,n,E,h,r,i,g,S="sizzle"+ +new Date,f=R.document,k=0,B=0,_=q(),z=q(),U=q(),y=q(),X=function(e,t){return e===t&&(l=!0),0},V={}.hasOwnProperty,t=[],G=t.pop,Y=t.push,A=t.push,Q=t.slice,v=function(e,t){for(var n=0,r=e.length;n<r;n++)if(e[n]===t)return n;return-1},J="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",a="[\\x20\\t\\r\\n\\f]",s="(?:\\\\[\\da-fA-F]{1,6}"+a+"?|\\\\[^\\r\\n\\f]|[\\w-]|[^\0-\\x7f])+",K="\\["+a+"*("+s+")(?:"+a+"*([*^$|!~]?=)"+a+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+s+"))|)"+a+"*\\]",Z=":("+s+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+K+")*)|.*)\\)|)",ee=new RegExp(a+"+","g"),m=new RegExp("^"+a+"+|((?:^|[^\\\\])(?:\\\\.)*)"+a+"+$","g"),te=new RegExp("^"+a+"*,"+a+"*"),ne=new RegExp("^"+a+"*([>+~]|"+a+")"+a+"*"),re=new RegExp(a+"|>"),ie=new RegExp(Z),oe=new RegExp("^"+s+"$"),x={ID:new RegExp("^#("+s+")"),CLASS:new RegExp("^\\.("+s+")"),TAG:new RegExp("^("+s+"|[*])"),ATTR:new RegExp("^"+K),PSEUDO:new RegExp("^"+Z),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+a+"*(even|odd|(([+-]|)(\\d*)n|)"+a+"*(?:([+-]|)"+a+"*(\\d+)|))"+a+"*\\)|)","i"),bool:new RegExp("^(?:"+J+")$","i"),needsContext:new RegExp("^"+a+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+a+"*((?:-\\d)?\\d*)"+a+"*\\)|)(?=[^-]|$)","i")},ae=/HTML$/i,se=/^(?:input|select|textarea|button)$/i,ue=/^h\d$/i,N=/^[^{]+\{\s*\[native \w/,le=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ce=/[+~]/,j=new RegExp("\\\\[\\da-fA-F]{1,6}"+a+"?|\\\\([^\\r\\n\\f])","g"),fe=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,pe=ve(function(e){return!0===e.disabled&&"fieldset"===e.nodeName.toLowerCase()},{dir:"parentNode",next:"legend"});try{A.apply(t=Q.call(f.childNodes),f.childNodes),t[f.childNodes.length].nodeType}catch(e){A={apply:t.length?function(e,t){Y.apply(e,Q.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function D(t,e,n,r){var i,o,a,s,u,l,c=e&&e.ownerDocument,f=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==f&&9!==f&&11!==f)return n;if(!r&&(T(e),e=e||C,E)){if(11!==f&&(s=le.exec(t)))if(i=s[1]){if(9===f){if(!(l=e.getElementById(i)))return n;if(l.id===i)return n.push(l),n}else if(c&&(l=c.getElementById(i))&&g(e,l)&&l.id===i)return n.push(l),n}else{if(s[2])return A.apply(n,e.getElementsByTagName(t)),n;if((i=s[3])&&p.getElementsByClassName&&e.getElementsByClassName)return A.apply(n,e.getElementsByClassName(i)),n}if(p.qsa&&!y[t+" "]&&(!h||!h.test(t))&&(1!==f||"object"!==e.nodeName.toLowerCase())){if(l=t,c=e,1===f&&(re.test(t)||ne.test(t))){(c=ce.test(t)&&ye(e.parentNode)||e)===e&&p.scope||((a=e.getAttribute("id"))?a=a.replace(fe,M):e.setAttribute("id",a=S)),o=(u=d(t)).length;while(o--)u[o]=(a?"#"+a:":scope")+" "+P(u[o]);l=u.join(",")}try{return A.apply(n,c.querySelectorAll(l)),n}catch(e){y(t,!0)}finally{a===S&&e.removeAttribute("id")}}}return $(t.replace(m,"$1"),e,n,r)}function q(){var n=[];function r(e,t){return n.push(e+" ")>b.cacheLength&&delete r[n.shift()],r[e+" "]=t}return r}function L(e){return e[S]=!0,e}function H(e){var t=C.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t)}}function de(e,t){var n=e.split("|"),r=n.length;while(r--)b.attrHandle[n[r]]=t}function he(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function ge(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&pe(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function O(a){return L(function(o){return o=+o,L(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function ye(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}for(e in p=D.support={},W=D.isXML=function(e){var t=e&&e.namespaceURI,e=e&&(e.ownerDocument||e).documentElement;return!ae.test(t||e&&e.nodeName||"HTML")},T=D.setDocument=function(e){var e=e?e.ownerDocument||e:f;return e!=C&&9===e.nodeType&&e.documentElement&&(n=(C=e).documentElement,E=!W(C),f!=C&&(e=C.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",I,!1):e.attachEvent&&e.attachEvent("onunload",I)),p.scope=H(function(e){return n.appendChild(e).appendChild(C.createElement("div")),"undefined"!=typeof e.querySelectorAll&&!e.querySelectorAll(":scope fieldset div").length}),p.attributes=H(function(e){return e.className="i",!e.getAttribute("className")}),p.getElementsByTagName=H(function(e){return e.appendChild(C.createComment("")),!e.getElementsByTagName("*").length}),p.getElementsByClassName=N.test(C.getElementsByClassName),p.getById=H(function(e){return n.appendChild(e).id=S,!C.getElementsByName||!C.getElementsByName(S).length}),p.getById?(b.filter.ID=function(e){var t=e.replace(j,c);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E)return(t=t.getElementById(e))?[t]:[]}):(b.filter.ID=function(e){var t=e.replace(j,c);return function(e){e="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return e&&e.value===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=p.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):p.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"!==e)return o;while(n=o[i++])1===n.nodeType&&r.push(n);return r},b.find.CLASS=p.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&E)return t.getElementsByClassName(e)},r=[],h=[],(p.qsa=N.test(C.querySelectorAll))&&(H(function(e){var t;n.appendChild(e).innerHTML="<a id='"+S+"'></a><select id='"+S+"-\r\\' msallowcapture=''><option selected=''></option></select>",e.querySelectorAll("[msallowcapture^='']").length&&h.push("[*^$]="+a+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||h.push("\\["+a+"*(?:value|"+J+")"),e.querySelectorAll("[id~="+S+"-]").length||h.push("~="),(t=C.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||h.push("\\["+a+"*name"+a+"*="+a+"*(?:''|\"\")"),e.querySelectorAll(":checked").length||h.push(":checked"),e.querySelectorAll("a#"+S+"+*").length||h.push(".#.+[+~]"),e.querySelectorAll("\\\f"),h.push("[\\r\\n\\f]")}),H(function(e){e.innerHTML="<a href='' disabled='disabled'></a><select disabled='disabled'><option/></select>";var t=C.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&h.push("name"+a+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&h.push(":enabled",":disabled"),n.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&h.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),h.push(",.*:")})),(p.matchesSelector=N.test(i=n.matches||n.webkitMatchesSelector||n.mozMatchesSelector||n.oMatchesSelector||n.msMatchesSelector))&&H(function(e){p.disconnectedMatch=i.call(e,"*"),i.call(e,"[s!='']:x"),r.push("!=",Z)}),h=h.length&&new RegExp(h.join("|")),r=r.length&&new RegExp(r.join("|")),e=N.test(n.compareDocumentPosition),g=e||N.test(n.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,t=t&&t.parentNode;return e===t||!(!t||1!==t.nodeType||!(n.contains?n.contains(t):e.compareDocumentPosition&&16&e.compareDocumentPosition(t)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},X=e?function(e,t){var n;return e===t?(l=!0,0):(n=!e.compareDocumentPosition-!t.compareDocumentPosition)||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!p.sortDetached&&t.compareDocumentPosition(e)===n?e==C||e.ownerDocument==f&&g(f,e)?-1:t==C||t.ownerDocument==f&&g(f,t)?1:u?v(u,e)-v(u,t):0:4&n?-1:1)}:function(e,t){if(e===t)return l=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e==C?-1:t==C?1:i?-1:o?1:u?v(u,e)-v(u,t):0;if(i===o)return he(e,t);n=e;while(n=n.parentNode)a.unshift(n);n=t;while(n=n.parentNode)s.unshift(n);while(a[r]===s[r])r++;return r?he(a[r],s[r]):a[r]==f?-1:s[r]==f?1:0}),C},D.matches=function(e,t){return D(e,null,null,t)},D.matchesSelector=function(e,t){if(T(e),p.matchesSelector&&E&&!y[t+" "]&&(!r||!r.test(t))&&(!h||!h.test(t)))try{var n=i.call(e,t);if(n||p.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){y(t,!0)}return 0<D(t,C,null,[e]).length},D.contains=function(e,t){return(e.ownerDocument||e)!=C&&T(e),g(e,t)},D.attr=function(e,t){(e.ownerDocument||e)!=C&&T(e);var n=b.attrHandle[t.toLowerCase()],n=n&&V.call(b.attrHandle,t.toLowerCase())?n(e,t,!E):void 0;return void 0!==n?n:p.attributes||!E?e.getAttribute(t):(n=e.getAttributeNode(t))&&n.specified?n.value:null},D.escape=function(e){return(e+"").replace(fe,M)},D.error=function(e){throw new Error("Syntax error, unrecognized expression: "+e)},D.uniqueSort=function(e){var t,n=[],r=0,i=0;if(l=!p.detectDuplicates,u=!p.sortStable&&e.slice(0),e.sort(X),l){while(t=e[i++])t===e[i]&&(r=n.push(i));while(r--)e.splice(n[r],1)}return u=null,e},o=D.getText=function(e){var t,n="",r=0,i=e.nodeType;if(i){if(1===i||9===i||11===i){if("string"==typeof e.textContent)return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=o(e)}else if(3===i||4===i)return e.nodeValue}else while(t=e[r++])n+=o(t);return n},(b=D.selectors={cacheLength:50,createPseudo:L,match:x,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(j,c),e[3]=(e[3]||e[4]||e[5]||"").replace(j,c),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||D.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&D.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return x.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&ie.test(n)&&(t=d(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(j,c).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=_[e+" "];return t||(t=new RegExp("(^|"+a+")"+e+"("+a+"|$)"))&&_(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(t,n,r){return function(e){e=D.attr(e,t);return null==e?"!="===n:!n||(e+="","="===n?e===r:"!="===n?e!==r:"^="===n?r&&0===e.indexOf(r):"*="===n?r&&-1<e.indexOf(r):"$="===n?r&&e.slice(-r.length)===r:"~="===n?-1<(" "+e.replace(ee," ")+" ").indexOf(r):"|="===n&&(e===r||e.slice(0,r.length+1)===r+"-"))}},CHILD:function(h,e,t,g,y){var m="nth"!==h.slice(0,3),v="last"!==h.slice(-4),x="of-type"===e;return 1===g&&0===y?function(e){return!!e.parentNode}:function(e,t,n){var r,i,o,a,s,u,l=m!=v?"nextSibling":"previousSibling",c=e.parentNode,f=x&&e.nodeName.toLowerCase(),p=!n&&!x,d=!1;if(c){if(m){while(l){a=e;while(a=a[l])if(x?a.nodeName.toLowerCase()===f:1===a.nodeType)return!1;u=l="only"===h&&!u&&"nextSibling"}return!0}if(u=[v?c.firstChild:c.lastChild],v&&p){d=(s=(r=(i=(o=(a=c)[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]||[])[0]===k&&r[1])&&r[2],a=s&&c.childNodes[s];while(a=++s&&a&&a[l]||(d=s=0)||u.pop())if(1===a.nodeType&&++d&&a===e){i[h]=[k,s,d];break}}else if(!1===(d=p?s=(r=(i=(o=(a=e)[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]||[])[0]===k&&r[1]:d))while(a=++s&&a&&a[l]||(d=s=0)||u.pop())if((x?a.nodeName.toLowerCase()===f:1===a.nodeType)&&++d&&(p&&((i=(o=a[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]=[k,d]),a===e))break;return(d-=y)===g||d%g==0&&0<=d/g}}},PSEUDO:function(e,o){var t,a=b.pseudos[e]||b.setFilters[e.toLowerCase()]||D.error("unsupported pseudo: "+e);return a[S]?a(o):1<a.length?(t=[e,e,"",o],b.setFilters.hasOwnProperty(e.toLowerCase())?L(function(e,t){var n,r=a(e,o),i=r.length;while(i--)e[n=v(e,r[i])]=!(t[n]=r[i])}):function(e){return a(e,0,t)}):a}},pseudos:{not:L(function(e){var r=[],i=[],s=F(e.replace(m,"$1"));return s[S]?L(function(e,t,n,r){var i,o=s(e,null,r,[]),a=e.length;while(a--)(i=o[a])&&(e[a]=!(t[a]=i))}):function(e,t,n){return r[0]=e,s(r,null,n,i),r[0]=null,!i.pop()}}),has:L(function(t){return function(e){return 0<D(t,e).length}}),contains:L(function(t){return t=t.replace(j,c),function(e){return-1<(e.textContent||o(e)).indexOf(t)}}),lang:L(function(n){return oe.test(n||"")||D.error("unsupported lang: "+n),n=n.replace(j,c).toLowerCase(),function(e){var t;do{if(t=E?e.lang:e.getAttribute("xml:lang")||e.getAttribute("lang"))return(t=t.toLowerCase())===n||0===t.indexOf(n+"-")}while((e=e.parentNode)&&1===e.nodeType);return!1}}),target:function(e){var t=R.location&&R.location.hash;return t&&t.slice(1)===e.id},root:function(e){return e===n},focus:function(e){return e===C.activeElement&&(!C.hasFocus||C.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},enabled:ge(!1),disabled:ge(!0),checked:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&!!e.checked||"option"===t&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,!0===e.selected},empty:function(e){for(e=e.firstChild;e;e=e.nextSibling)if(e.nodeType<6)return!1;return!0},parent:function(e){return!b.pseudos.empty(e)},header:function(e){return ue.test(e.nodeName)},input:function(e){return se.test(e.nodeName)},button:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&"button"===e.type||"button"===t},text:function(e){return"input"===e.nodeName.toLowerCase()&&"text"===e.type&&(null==(e=e.getAttribute("type"))||"text"===e.toLowerCase())},first:O(function(){return[0]}),last:O(function(e,t){return[t-1]}),eq:O(function(e,t,n){return[n<0?n+t:n]}),even:O(function(e,t){for(var n=0;n<t;n+=2)e.push(n);return e}),odd:O(function(e,t){for(var n=1;n<t;n+=2)e.push(n);return e}),lt:O(function(e,t,n){for(var r=n<0?n+t:t<n?t:n;0<=--r;)e.push(r);return e}),gt:O(function(e,t,n){for(var r=n<0?n+t:n;++r<t;)e.push(r);return e})}}).pseudos.nth=b.pseudos.eq,{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})b.pseudos[e]=function(t){return function(e){return"input"===e.nodeName.toLowerCase()&&e.type===t}}(e);for(e in{submit:!0,reset:!0})b.pseudos[e]=function(n){return function(e){var t=e.nodeName.toLowerCase();return("input"===t||"button"===t)&&e.type===n}}(e);function me(){}function P(e){for(var t=0,n=e.length,r="";t<n;t++)r+=e[t].value;return r}function ve(a,e,t){var s=e.dir,u=e.next,l=u||s,c=t&&"parentNode"===l,f=B++;return e.first?function(e,t,n){while(e=e[s])if(1===e.nodeType||c)return a(e,t,n);return!1}:function(e,t,n){var r,i,o=[k,f];if(n){while(e=e[s])if((1===e.nodeType||c)&&a(e,t,n))return!0}else while(e=e[s])if(1===e.nodeType||c)if(i=(i=e[S]||(e[S]={}))[e.uniqueID]||(i[e.uniqueID]={}),u&&u===e.nodeName.toLowerCase())e=e[s]||e;else{if((r=i[l])&&r[0]===k&&r[1]===f)return o[2]=r[2];if((i[l]=o)[2]=a(e,t,n))return!0}return!1}}function xe(i){return 1<i.length?function(e,t,n){var r=i.length;while(r--)if(!i[r](e,t,n))return!1;return!0}:i[0]}function be(e,t,n,r,i){for(var o,a=[],s=0,u=e.length,l=null!=t;s<u;s++)!(o=e[s])||n&&!n(o,r,i)||(a.push(o),l&&t.push(s));return a}function we(d,h,g,y,m,e){return y&&!y[S]&&(y=we(y)),m&&!m[S]&&(m=we(m,e)),L(function(e,t,n,r){var i,o,a,s=[],u=[],l=t.length,c=e||function(e,t,n){for(var r=0,i=t.length;r<i;r++)D(e,t[r],n);return n}(h||"*",n.nodeType?[n]:n,[]),f=!d||!e&&h?c:be(c,s,d,n,r),p=g?m||(e?d:l||y)?[]:t:f;if(g&&g(f,p,n,r),y){i=be(p,u),y(i,[],n,r),o=i.length;while(o--)(a=i[o])&&(p[u[o]]=!(f[u[o]]=a))}if(e){if(m||d){if(m){i=[],o=p.length;while(o--)(a=p[o])&&i.push(f[o]=a);m(null,p=[],i,r)}o=p.length;while(o--)(a=p[o])&&-1<(i=m?v(e,a):s[o])&&(e[i]=!(t[i]=a))}}else p=be(p===t?p.splice(l,p.length):p),m?m(null,t,p,r):A.apply(t,p)})}function Te(y,m){function e(e,t,n,r,i){var o,a,s,u=0,l="0",c=e&&[],f=[],p=w,d=e||x&&b.find.TAG("*",i),h=k+=null==p?1:Math.random()||.1,g=d.length;for(i&&(w=t==C||t||i);l!==g&&null!=(o=d[l]);l++){if(x&&o){a=0,t||o.ownerDocument==C||(T(o),n=!E);while(s=y[a++])if(s(o,t||C,n)){r.push(o);break}i&&(k=h)}v&&((o=!s&&o)&&u--,e&&c.push(o))}if(u+=l,v&&l!==u){a=0;while(s=m[a++])s(c,f,t,n);if(e){if(0<u)while(l--)c[l]||f[l]||(f[l]=G.call(r));f=be(f)}A.apply(r,f),i&&!e&&0<f.length&&1<u+m.length&&D.uniqueSort(r)}return i&&(k=h,w=p),c}var v=0<m.length,x=0<y.length;return v?L(e):e}return me.prototype=b.filters=b.pseudos,b.setFilters=new me,d=D.tokenize=function(e,t){var n,r,i,o,a,s,u,l=z[e+" "];if(l)return t?0:l.slice(0);a=e,s=[],u=b.preFilter;while(a){for(o in n&&!(r=te.exec(a))||(r&&(a=a.slice(r[0].length)||a),s.push(i=[])),n=!1,(r=ne.exec(a))&&(n=r.shift(),i.push({value:n,type:r[0].replace(m," ")}),a=a.slice(n.length)),b.filter)!(r=x[o].exec(a))||u[o]&&!(r=u[o](r))||(n=r.shift(),i.push({value:n,type:o,matches:r}),a=a.slice(n.length));if(!n)break}return t?a.length:a?D.error(e):z(e,s).slice(0)},F=D.compile=function(e,t){var n,r=[],i=[],o=U[e+" "];if(!o){n=(t=t||d(e)).length;while(n--)((o=function e(t){for(var r,n,i,o=t.length,a=b.relative[t[0].type],s=a||b.relative[" "],u=a?1:0,l=ve(function(e){return e===r},s,!0),c=ve(function(e){return-1<v(r,e)},s,!0),f=[function(e,t,n){return e=!a&&(n||t!==w)||((r=t).nodeType?l:c)(e,t,n),r=null,e}];u<o;u++)if(n=b.relative[t[u].type])f=[ve(xe(f),n)];else{if((n=b.filter[t[u].type].apply(null,t[u].matches))[S]){for(i=++u;i<o;i++)if(b.relative[t[i].type])break;return we(1<u&&xe(f),1<u&&P(t.slice(0,u-1).concat({value:" "===t[u-2].type?"*":""})).replace(m,"$1"),n,u<i&&e(t.slice(u,i)),i<o&&e(t=t.slice(i)),i<o&&P(t))}f.push(n)}return xe(f)}(t[n]))[S]?r:i).push(o);(o=U(e,Te(i,r))).selector=e}return o},$=D.select=function(e,t,n,r){var i,o,a,s,u="function"==typeof e&&e,l=!r&&d(e=u.selector||e);if(n=n||[],1===l.length){if(2<(o=l[0]=l[0].slice(0)).length&&"ID"===(a=o[0]).type&&9===t.nodeType&&E&&b.relative[o[1].type]){if(!(t=(b.find.ID(a.matches[0].replace(j,c),t)||[])[0]))return n;u&&(t=t.parentNode),e=e.slice(o.shift().value.length)}i=x.needsContext.test(e)?0:o.length;while(i--){if(a=o[i],b.relative[s=a.type])break;if((s=b.find[s])&&(r=s(a.matches[0].replace(j,c),ce.test(o[0].type)&&ye(t.parentNode)||t))){if(o.splice(i,1),e=r.length&&P(o))break;return A.apply(n,r),n}}}return(u||F(e,l))(r,t,!E,n,!t||ce.test(e)&&ye(t.parentNode)||t),n},p.sortStable=S.split("").sort(X).join("")===S,p.detectDuplicates=!!l,T(),p.sortDetached=H(function(e){return 1&e.compareDocumentPosition(C.createElement("fieldset"))}),H(function(e){return e.innerHTML="<a href='#'></a>","#"===e.firstChild.getAttribute("href")})||de("type|href|height|width",function(e,t,n){if(!n)return e.getAttribute(t,"type"===t.toLowerCase()?1:2)}),p.attributes&&H(function(e){return e.innerHTML="<input/>",e.firstChild.setAttribute("value",""),""===e.firstChild.getAttribute("value")})||de("value",function(e,t,n){if(!n&&"input"===e.nodeName.toLowerCase())return e.defaultValue}),H(function(e){return null==e.getAttribute("disabled")})||de(J,function(e,t,n){if(!n)return!0===e[t]?t.toLowerCase():(n=e.getAttributeNode(t))&&n.specified?n.value:null}),D}(w),Q=(C.find=e,C.expr=e.selectors,C.expr[":"]=C.expr.pseudos,C.uniqueSort=C.unique=e.uniqueSort,C.text=e.getText,C.isXMLDoc=e.isXML,C.contains=e.contains,C.escapeSelector=e.escape,C.expr.match.needsContext);function u(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()}var J=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function K(e,n,r){return v(n)?C.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?C.grep(e,function(e){return e===n!==r}):"string"!=typeof n?C.grep(e,function(e){return-1<F.call(n,e)!==r}):C.filter(n,e,r)}C.filter=function(e,t,n){var r=t[0];return n&&(e=":not("+e+")"),1===t.length&&1===r.nodeType?C.find.matchesSelector(r,e)?[r]:[]:C.find.matches(e,C.grep(t,function(e){return 1===e.nodeType}))},C.fn.extend({find:function(e){var t,n,r=this.length,i=this;if("string"!=typeof e)return this.pushStack(C(e).filter(function(){for(t=0;t<r;t++)if(C.contains(i[t],this))return!0}));for(n=this.pushStack([]),t=0;t<r;t++)C.find(e,i[t],n);return 1<r?C.uniqueSort(n):n},filter:function(e){return this.pushStack(K(this,e||[],!1))},not:function(e){return this.pushStack(K(this,e||[],!0))},is:function(e){return!!K(this,"string"==typeof e&&Q.test(e)?C(e):e||[],!1).length}});var Z,ee=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,te=((C.fn.init=function(e,t,n){if(e){if(n=n||Z,"string"!=typeof e)return e.nodeType?(this[0]=e,this.length=1,this):v(e)?void 0!==n.ready?n.ready(e):e(C):C.makeArray(e,this);if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:ee.exec(e))||!r[1]&&t)return(!t||t.jquery?t||n:this.constructor(t)).find(e);if(r[1]){if(t=t instanceof C?t[0]:t,C.merge(this,C.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:T,!0)),J.test(r[1])&&C.isPlainObject(t))for(var r in t)v(this[r])?this[r](t[r]):this.attr(r,t[r])}else(n=T.getElementById(r[2]))&&(this[0]=n,this.length=1)}return this}).prototype=C.fn,Z=C(T),/^(?:parents|prev(?:Until|All))/),ne={children:!0,contents:!0,next:!0,prev:!0};function re(e,t){while((e=e[t])&&1!==e.nodeType);return e}C.fn.extend({has:function(e){var t=C(e,this),n=t.length;return this.filter(function(){for(var e=0;e<n;e++)if(C.contains(this,t[e]))return!0})},closest:function(e,t){var n,r=0,i=this.length,o=[],a="string"!=typeof e&&C(e);if(!Q.test(e))for(;r<i;r++)for(n=this[r];n&&n!==t;n=n.parentNode)if(n.nodeType<11&&(a?-1<a.index(n):1===n.nodeType&&C.find.matchesSelector(n,e))){o.push(n);break}return this.pushStack(1<o.length?C.uniqueSort(o):o)},index:function(e){return e?"string"==typeof e?F.call(C(e),this[0]):F.call(this,e.jquery?e[0]:e):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(e,t){return this.pushStack(C.uniqueSort(C.merge(this.get(),C(e,t))))},addBack:function(e){return this.add(null==e?this.prevObject:this.prevObject.filter(e))}}),C.each({parent:function(e){e=e.parentNode;return e&&11!==e.nodeType?e:null},parents:function(e){return r(e,"parentNode")},parentsUntil:function(e,t,n){return r(e,"parentNode",n)},next:function(e){return re(e,"nextSibling")},prev:function(e){return re(e,"previousSibling")},nextAll:function(e){return r(e,"nextSibling")},prevAll:function(e){return r(e,"previousSibling")},nextUntil:function(e,t,n){return r(e,"nextSibling",n)},prevUntil:function(e,t,n){return r(e,"previousSibling",n)},siblings:function(e){return Y((e.parentNode||{}).firstChild,e)},children:function(e){return Y(e.firstChild)},contents:function(e){return null!=e.contentDocument&&M(e.contentDocument)?e.contentDocument:(u(e,"template")&&(e=e.content||e),C.merge([],e.childNodes))}},function(r,i){C.fn[r]=function(e,t){var n=C.map(this,i,e);return(t="Until"!==r.slice(-5)?e:t)&&"string"==typeof t&&(n=C.filter(t,n)),1<this.length&&(ne[r]||C.uniqueSort(n),te.test(r)&&n.reverse()),this.pushStack(n)}});var E=/[^\x20\t\r\n\f]+/g;function c(e){return e}function ie(e){throw e}function oe(e,t,n,r){var i;try{e&&v(i=e.promise)?i.call(e).done(t).fail(n):e&&v(i=e.then)?i.call(e,t,n):t.apply(void 0,[e].slice(r))}catch(e){n.apply(void 0,[e])}}C.Callbacks=function(r){var e,n;r="string"==typeof r?(e=r,n={},C.each(e.match(E)||[],function(e,t){n[t]=!0}),n):C.extend({},r);function i(){for(s=s||r.once,a=o=!0;l.length;c=-1){t=l.shift();while(++c<u.length)!1===u[c].apply(t[0],t[1])&&r.stopOnFalse&&(c=u.length,t=!1)}r.memory||(t=!1),o=!1,s&&(u=t?[]:"")}var o,t,a,s,u=[],l=[],c=-1,f={add:function(){return u&&(t&&!o&&(c=u.length-1,l.push(t)),function n(e){C.each(e,function(e,t){v(t)?r.unique&&f.has(t)||u.push(t):t&&t.length&&"string"!==h(t)&&n(t)})}(arguments),t&&!o&&i()),this},remove:function(){return C.each(arguments,function(e,t){var n;while(-1<(n=C.inArray(t,u,n)))u.splice(n,1),n<=c&&c--}),this},has:function(e){return e?-1<C.inArray(e,u):0<u.length},empty:function(){return u=u&&[],this},disable:function(){return s=l=[],u=t="",this},disabled:function(){return!u},lock:function(){return s=l=[],t||o||(u=t=""),this},locked:function(){return!!s},fireWith:function(e,t){return s||(t=[e,(t=t||[]).slice?t.slice():t],l.push(t),o||i()),this},fire:function(){return f.fireWith(this,arguments),this},fired:function(){return!!a}};return f},C.extend({Deferred:function(e){var o=[["notify","progress",C.Callbacks("memory"),C.Callbacks("memory"),2],["resolve","done",C.Callbacks("once memory"),C.Callbacks("once memory"),0,"resolved"],["reject","fail",C.Callbacks("once memory"),C.Callbacks("once memory"),1,"rejected"]],i="pending",a={state:function(){return i},always:function(){return s.done(arguments).fail(arguments),this},catch:function(e){return a.then(null,e)},pipe:function(){var i=arguments;return C.Deferred(function(r){C.each(o,function(e,t){var n=v(i[t[4]])&&i[t[4]];s[t[1]](function(){var e=n&&n.apply(this,arguments);e&&v(e.promise)?e.promise().progress(r.notify).done(r.resolve).fail(r.reject):r[t[0]+"With"](this,n?[e]:arguments)})}),i=null}).promise()},then:function(t,n,r){var u=0;function l(i,o,a,s){return function(){function e(){var e,t;if(!(i<u)){if((e=a.apply(n,r))===o.promise())throw new TypeError("Thenable self-resolution");t=e&&("object"==typeof e||"function"==typeof e)&&e.then,v(t)?s?t.call(e,l(u,o,c,s),l(u,o,ie,s)):(u++,t.call(e,l(u,o,c,s),l(u,o,ie,s),l(u,o,c,o.notifyWith))):(a!==c&&(n=void 0,r=[e]),(s||o.resolveWith)(n,r))}}var n=this,r=arguments,t=s?e:function(){try{e()}catch(e){C.Deferred.exceptionHook&&C.Deferred.exceptionHook(e,t.stackTrace),u<=i+1&&(a!==ie&&(n=void 0,r=[e]),o.rejectWith(n,r))}};i?t():(C.Deferred.getStackHook&&(t.stackTrace=C.Deferred.getStackHook()),w.setTimeout(t))}}return C.Deferred(function(e){o[0][3].add(l(0,e,v(r)?r:c,e.notifyWith)),o[1][3].add(l(0,e,v(t)?t:c)),o[2][3].add(l(0,e,v(n)?n:ie))}).promise()},promise:function(e){return null!=e?C.extend(e,a):a}},s={};return C.each(o,function(e,t){var n=t[2],r=t[5];a[t[1]]=n.add,r&&n.add(function(){i=r},o[3-e][2].disable,o[3-e][3].disable,o[0][2].lock,o[0][3].lock),n.add(t[3].fire),s[t[0]]=function(){return s[t[0]+"With"](this===s?void 0:this,arguments),this},s[t[0]+"With"]=n.fireWith}),a.promise(s),e&&e.call(s,s),s},when:function(e){function t(t){return function(e){i[t]=this,o[t]=1<arguments.length?s.call(arguments):e,--n||a.resolveWith(i,o)}}var n=arguments.length,r=n,i=Array(r),o=s.call(arguments),a=C.Deferred();if(n<=1&&(oe(e,a.done(t(r)).resolve,a.reject,!n),"pending"===a.state()||v(o[r]&&o[r].then)))return a.then();while(r--)oe(o[r],t(r),a.reject);return a.promise()}});var ae=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/,se=(C.Deferred.exceptionHook=function(e,t){w.console&&w.console.warn&&e&&ae.test(e.name)&&w.console.warn("jQuery.Deferred exception: "+e.message,e.stack,t)},C.readyException=function(e){w.setTimeout(function(){throw e})},C.Deferred());function ue(){T.removeEventListener("DOMContentLoaded",ue),w.removeEventListener("load",ue),C.ready()}C.fn.ready=function(e){return se.then(e).catch(function(e){C.readyException(e)}),this},C.extend({isReady:!1,readyWait:1,ready:function(e){(!0===e?--C.readyWait:C.isReady)||(C.isReady=!0)!==e&&0<--C.readyWait||se.resolveWith(T,[C])}}),C.ready.then=se.then,"complete"===T.readyState||"loading"!==T.readyState&&!T.documentElement.doScroll?w.setTimeout(C.ready):(T.addEventListener("DOMContentLoaded",ue),w.addEventListener("load",ue));function f(e,t,n,r,i,o,a){var s=0,u=e.length,l=null==n;if("object"===h(n))for(s in i=!0,n)f(e,t,s,n[s],!0,o,a);else if(void 0!==r&&(i=!0,v(r)||(a=!0),t=l?a?(t.call(e,r),null):(l=t,function(e,t,n){return l.call(C(e),n)}):t))for(;s<u;s++)t(e[s],n,a?r:r.call(e[s],s,t(e[s],n)));return i?e:l?t.call(e):u?t(e[0],n):o}var le=/^-ms-/,ce=/-([a-z])/g;function fe(e,t){return t.toUpperCase()}function x(e){return e.replace(le,"ms-").replace(ce,fe)}function m(e){return 1===e.nodeType||9===e.nodeType||!+e.nodeType}function pe(){this.expando=C.expando+pe.uid++}pe.uid=1,pe.prototype={cache:function(e){var t=e[this.expando];return t||(t={},m(e)&&(e.nodeType?e[this.expando]=t:Object.defineProperty(e,this.expando,{value:t,configurable:!0}))),t},set:function(e,t,n){var r,i=this.cache(e);if("string"==typeof t)i[x(t)]=n;else for(r in t)i[x(r)]=t[r];return i},get:function(e,t){return void 0===t?this.cache(e):e[this.expando]&&e[this.expando][x(t)]},access:function(e,t,n){return void 0===t||t&&"string"==typeof t&&void 0===n?this.get(e,t):(this.set(e,t,n),void 0!==n?n:t)},remove:function(e,t){var n,r=e[this.expando];if(void 0!==r){if(void 0!==t){n=(t=Array.isArray(t)?t.map(x):(t=x(t))in r?[t]:t.match(E)||[]).length;while(n--)delete r[t[n]]}void 0!==t&&!C.isEmptyObject(r)||(e.nodeType?e[this.expando]=void 0:delete e[this.expando])}},hasData:function(e){e=e[this.expando];return void 0!==e&&!C.isEmptyObject(e)}};var b=new pe,l=new pe,de=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,he=/[A-Z]/g;function ge(e,t,n){var r,i;if(void 0===n&&1===e.nodeType)if(r="data-"+t.replace(he,"-$&").toLowerCase(),"string"==typeof(n=e.getAttribute(r))){try{n="true"===(i=n)||"false"!==i&&("null"===i?null:i===+i+""?+i:de.test(i)?JSON.parse(i):i)}catch(e){}l.set(e,t,n)}else n=void 0;return n}C.extend({hasData:function(e){return l.hasData(e)||b.hasData(e)},data:function(e,t,n){return l.access(e,t,n)},removeData:function(e,t){l.remove(e,t)},_data:function(e,t,n){return b.access(e,t,n)},_removeData:function(e,t){b.remove(e,t)}}),C.fn.extend({data:function(n,e){var t,r,i,o=this[0],a=o&&o.attributes;if(void 0!==n)return"object"==typeof n?this.each(function(){l.set(this,n)}):f(this,function(e){var t;if(o&&void 0===e)return void 0!==(t=l.get(o,n))||void 0!==(t=ge(o,n))?t:void 0;this.each(function(){l.set(this,n,e)})},null,e,1<arguments.length,null,!0);if(this.length&&(i=l.get(o),1===o.nodeType&&!b.get(o,"hasDataAttrs"))){t=a.length;while(t--)a[t]&&0===(r=a[t].name).indexOf("data-")&&(r=x(r.slice(5)),ge(o,r,i[r]));b.set(o,"hasDataAttrs",!0)}return i},removeData:function(e){return this.each(function(){l.remove(this,e)})}}),C.extend({queue:function(e,t,n){var r;if(e)return r=b.get(e,t=(t||"fx")+"queue"),n&&(!r||Array.isArray(n)?r=b.access(e,t,C.makeArray(n)):r.push(n)),r||[]},dequeue:function(e,t){t=t||"fx";var n=C.queue(e,t),r=n.length,i=n.shift(),o=C._queueHooks(e,t);"inprogress"===i&&(i=n.shift(),r--),i&&("fx"===t&&n.unshift("inprogress"),delete o.stop,i.call(e,function(){C.dequeue(e,t)},o)),!r&&o&&o.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return b.get(e,n)||b.access(e,n,{empty:C.Callbacks("once memory").add(function(){b.remove(e,[t+"queue",n])})})}}),C.fn.extend({queue:function(t,n){var e=2;return"string"!=typeof t&&(n=t,t="fx",e--),arguments.length<e?C.queue(this[0],t):void 0===n?this:this.each(function(){var e=C.queue(this,t,n);C._queueHooks(this,t),"fx"===t&&"inprogress"!==e[0]&&C.dequeue(this,t)})},dequeue:function(e){return this.each(function(){C.dequeue(this,e)})},clearQueue:function(e){return this.queue(e||"fx",[])},promise:function(e,t){function n(){--i||o.resolveWith(a,[a])}var r,i=1,o=C.Deferred(),a=this,s=this.length;"string"!=typeof e&&(t=e,e=void 0),e=e||"fx";while(s--)(r=b.get(a[s],e+"queueHooks"))&&r.empty&&(i++,r.empty.add(n));return n(),o.promise(t)}});function ye(e,t){return"none"===(e=t||e).style.display||""===e.style.display&&k(e)&&"none"===C.css(e,"display")}var e=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,me=new RegExp("^(?:([+-])=|)("+e+")([a-z%]*)$","i"),p=["Top","Right","Bottom","Left"],S=T.documentElement,k=function(e){return C.contains(e.ownerDocument,e)},ve={composed:!0};S.getRootNode&&(k=function(e){return C.contains(e.ownerDocument,e)||e.getRootNode(ve)===e.ownerDocument});function xe(e,t,n,r){var i,o,a=20,s=r?function(){return r.cur()}:function(){return C.css(e,t,"")},u=s(),l=n&&n[3]||(C.cssNumber[t]?"":"px"),c=e.nodeType&&(C.cssNumber[t]||"px"!==l&&+u)&&me.exec(C.css(e,t));if(c&&c[3]!==l){l=l||c[3],c=+(u/=2)||1;while(a--)C.style(e,t,c+l),(1-o)*(1-(o=s()/u||.5))<=0&&(a=0),c/=o;C.style(e,t,(c*=2)+l),n=n||[]}return n&&(c=+c||+u||0,i=n[1]?c+(n[1]+1)*n[2]:+n[2],r&&(r.unit=l,r.start=c,r.end=i)),i}var be={};function A(e,t){for(var n,r,i,o,a,s=[],u=0,l=e.length;u<l;u++)(r=e[u]).style&&(n=r.style.display,t?("none"===n&&(s[u]=b.get(r,"display")||null,s[u]||(r.style.display="")),""===r.style.display&&ye(r)&&(s[u]=(a=o=void 0,o=(i=r).ownerDocument,i=i.nodeName,(a=be[i])||(o=o.body.appendChild(o.createElement(i)),a=C.css(o,"display"),o.parentNode.removeChild(o),be[i]=a="none"===a?"block":a),a))):"none"!==n&&(s[u]="none",b.set(r,"display",n)));for(u=0;u<l;u++)null!=s[u]&&(e[u].style.display=s[u]);return e}C.fn.extend({show:function(){return A(this,!0)},hide:function(){return A(this)},toggle:function(e){return"boolean"==typeof e?e?this.show():this.hide():this.each(function(){ye(this)?C(this).show():C(this).hide()})}});var we=/^(?:checkbox|radio)$/i,Te=/<([a-z][^\/\0>\x20\t\r\n\f]*)/i,Ce=/^$|^module$|\/(?:java|ecma)script/i,N=(L=T.createDocumentFragment().appendChild(T.createElement("div")),(o=T.createElement("input")).setAttribute("type","radio"),o.setAttribute("checked","checked"),o.setAttribute("name","t"),L.appendChild(o),y.checkClone=L.cloneNode(!0).cloneNode(!0).lastChild.checked,L.innerHTML="<textarea>x</textarea>",y.noCloneChecked=!!L.cloneNode(!0).lastChild.defaultValue,L.innerHTML="<option></option>",y.option=!!L.lastChild,{thead:[1,"<table>","</table>"],col:[2,"<table><colgroup>","</colgroup></table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:[0,"",""]});function j(e,t){var n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[];return void 0===t||t&&u(e,t)?C.merge([e],n):n}function Ee(e,t){for(var n=0,r=e.length;n<r;n++)b.set(e[n],"globalEval",!t||b.get(t[n],"globalEval"))}N.tbody=N.tfoot=N.colgroup=N.caption=N.thead,N.th=N.td,y.option||(N.optgroup=N.option=[1,"<select multiple='multiple'>","</select>"]);var Se=/<|&#?\w+;/;function ke(e,t,n,r,i){for(var o,a,s,u,l,c=t.createDocumentFragment(),f=[],p=0,d=e.length;p<d;p++)if((o=e[p])||0===o)if("object"===h(o))C.merge(f,o.nodeType?[o]:o);else if(Se.test(o)){a=a||c.appendChild(t.createElement("div")),s=(Te.exec(o)||["",""])[1].toLowerCase(),s=N[s]||N._default,a.innerHTML=s[1]+C.htmlPrefilter(o)+s[2],l=s[0];while(l--)a=a.lastChild;C.merge(f,a.childNodes),(a=c.firstChild).textContent=""}else f.push(t.createTextNode(o));c.textContent="",p=0;while(o=f[p++])if(r&&-1<C.inArray(o,r))i&&i.push(o);else if(u=k(o),a=j(c.appendChild(o),"script"),u&&Ee(a),n){l=0;while(o=a[l++])Ce.test(o.type||"")&&n.push(o)}return c}var Ae=/^([^.]*)(?:\.(.+)|)/;function n(){return!0}function d(){return!1}function Ne(e,t){return e===function(){try{return T.activeElement}catch(e){}}()==("focus"===t)}function je(e,t,n,r,i,o){var a,s;if("object"==typeof t){for(s in"string"!=typeof n&&(r=r||n,n=void 0),t)je(e,s,n,r,t[s],o);return e}if(null==r&&null==i?(i=n,r=n=void 0):null==i&&("string"==typeof n?(i=r,r=void 0):(i=r,r=n,n=void 0)),!1===i)i=d;else if(!i)return e;return 1===o&&(a=i,(i=function(e){return C().off(e),a.apply(this,arguments)}).guid=a.guid||(a.guid=C.guid++)),e.each(function(){C.event.add(this,t,i,r,n)})}function De(e,i,o){o?(b.set(e,i,!1),C.event.add(e,i,{namespace:!1,handler:function(e){var t,n,r=b.get(this,i);if(1&e.isTrigger&&this[i]){if(r.length)(C.event.special[i]||{}).delegateType&&e.stopPropagation();else if(r=s.call(arguments),b.set(this,i,r),t=o(this,i),this[i](),r!==(n=b.get(this,i))||t?b.set(this,i,!1):n={},r!==n)return e.stopImmediatePropagation(),e.preventDefault(),n&&n.value}else r.length&&(b.set(this,i,{value:C.event.trigger(C.extend(r[0],C.Event.prototype),r.slice(1),this)}),e.stopImmediatePropagation())}})):void 0===b.get(e,i)&&C.event.add(e,i,n)}C.event={global:{},add:function(t,e,n,r,i){var o,a,s,u,l,c,f,p,d,h=b.get(t);if(m(t)){n.handler&&(n=(o=n).handler,i=o.selector),i&&C.find.matchesSelector(S,i),n.guid||(n.guid=C.guid++),(s=h.events)||(s=h.events=Object.create(null)),(a=h.handle)||(a=h.handle=function(e){return"undefined"!=typeof C&&C.event.triggered!==e.type?C.event.dispatch.apply(t,arguments):void 0}),u=(e=(e||"").match(E)||[""]).length;while(u--)f=d=(p=Ae.exec(e[u])||[])[1],p=(p[2]||"").split(".").sort(),f&&(l=C.event.special[f]||{},f=(i?l.delegateType:l.bindType)||f,l=C.event.special[f]||{},d=C.extend({type:f,origType:d,data:r,handler:n,guid:n.guid,selector:i,needsContext:i&&C.expr.match.needsContext.test(i),namespace:p.join(".")},o),(c=s[f])||((c=s[f]=[]).delegateCount=0,l.setup&&!1!==l.setup.call(t,r,p,a)||t.addEventListener&&t.addEventListener(f,a)),l.add&&(l.add.call(t,d),d.handler.guid||(d.handler.guid=n.guid)),i?c.splice(c.delegateCount++,0,d):c.push(d),C.event.global[f]=!0)}},remove:function(e,t,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,y=b.hasData(e)&&b.get(e);if(y&&(u=y.events)){l=(t=(t||"").match(E)||[""]).length;while(l--)if(d=g=(s=Ae.exec(t[l])||[])[1],h=(s[2]||"").split(".").sort(),d){f=C.event.special[d]||{},p=u[d=(r?f.delegateType:f.bindType)||d]||[],s=s[2]&&new RegExp("(^|\\.)"+h.join("\\.(?:.*\\.|)")+"(\\.|$)"),a=o=p.length;while(o--)c=p[o],!i&&g!==c.origType||n&&n.guid!==c.guid||s&&!s.test(c.namespace)||r&&r!==c.selector&&("**"!==r||!c.selector)||(p.splice(o,1),c.selector&&p.delegateCount--,f.remove&&f.remove.call(e,c));a&&!p.length&&(f.teardown&&!1!==f.teardown.call(e,h,y.handle)||C.removeEvent(e,d,y.handle),delete u[d])}else for(d in u)C.event.remove(e,d+t[l],n,r,!0);C.isEmptyObject(u)&&b.remove(e,"handle events")}},dispatch:function(e){var t,n,r,i,o,a=new Array(arguments.length),s=C.event.fix(e),e=(b.get(this,"events")||Object.create(null))[s.type]||[],u=C.event.special[s.type]||{};for(a[0]=s,t=1;t<arguments.length;t++)a[t]=arguments[t];if(s.delegateTarget=this,!u.preDispatch||!1!==u.preDispatch.call(this,s)){o=C.event.handlers.call(this,s,e),t=0;while((r=o[t++])&&!s.isPropagationStopped()){s.currentTarget=r.elem,n=0;while((i=r.handlers[n++])&&!s.isImmediatePropagationStopped())s.rnamespace&&!1!==i.namespace&&!s.rnamespace.test(i.namespace)||(s.handleObj=i,s.data=i.data,void 0!==(i=((C.event.special[i.origType]||{}).handle||i.handler).apply(r.elem,a))&&!1===(s.result=i)&&(s.preventDefault(),s.stopPropagation()))}return u.postDispatch&&u.postDispatch.call(this,s),s.result}},handlers:function(e,t){var n,r,i,o,a,s=[],u=t.delegateCount,l=e.target;if(u&&l.nodeType&&!("click"===e.type&&1<=e.button))for(;l!==this;l=l.parentNode||this)if(1===l.nodeType&&("click"!==e.type||!0!==l.disabled)){for(o=[],a={},n=0;n<u;n++)void 0===a[i=(r=t[n]).selector+" "]&&(a[i]=r.needsContext?-1<C(i,this).index(l):C.find(i,this,null,[l]).length),a[i]&&o.push(r);o.length&&s.push({elem:l,handlers:o})}return l=this,u<t.length&&s.push({elem:l,handlers:t.slice(u)}),s},addProp:function(t,e){Object.defineProperty(C.Event.prototype,t,{enumerable:!0,configurable:!0,get:v(e)?function(){if(this.originalEvent)return e(this.originalEvent)}:function(){if(this.originalEvent)return this.originalEvent[t]},set:function(e){Object.defineProperty(this,t,{enumerable:!0,configurable:!0,writable:!0,value:e})}})},fix:function(e){return e[C.expando]?e:new C.Event(e)},special:{load:{noBubble:!0},click:{setup:function(e){e=this||e;return we.test(e.type)&&e.click&&u(e,"input")&&De(e,"click",n),!1},trigger:function(e){e=this||e;return we.test(e.type)&&e.click&&u(e,"input")&&De(e,"click"),!0},_default:function(e){e=e.target;return we.test(e.type)&&e.click&&u(e,"input")&&b.get(e,"click")||u(e,"a")}},beforeunload:{postDispatch:function(e){void 0!==e.result&&e.originalEvent&&(e.originalEvent.returnValue=e.result)}}}},C.removeEvent=function(e,t,n){e.removeEventListener&&e.removeEventListener(t,n)},C.Event=function(e,t){if(!(this instanceof C.Event))return new C.Event(e,t);e&&e.type?(this.originalEvent=e,this.type=e.type,this.isDefaultPrevented=e.defaultPrevented||void 0===e.defaultPrevented&&!1===e.returnValue?n:d,this.target=e.target&&3===e.target.nodeType?e.target.parentNode:e.target,this.currentTarget=e.currentTarget,this.relatedTarget=e.relatedTarget):this.type=e,t&&C.extend(this,t),this.timeStamp=e&&e.timeStamp||Date.now(),this[C.expando]=!0},C.Event.prototype={constructor:C.Event,isDefaultPrevented:d,isPropagationStopped:d,isImmediatePropagationStopped:d,isSimulated:!1,preventDefault:function(){var e=this.originalEvent;this.isDefaultPrevented=n,e&&!this.isSimulated&&e.preventDefault()},stopPropagation:function(){var e=this.originalEvent;this.isPropagationStopped=n,e&&!this.isSimulated&&e.stopPropagation()},stopImmediatePropagation:function(){var e=this.originalEvent;this.isImmediatePropagationStopped=n,e&&!this.isSimulated&&e.stopImmediatePropagation(),this.stopPropagation()}},C.each({altKey:!0,bubbles:!0,cancelable:!0,changedTouches:!0,ctrlKey:!0,detail:!0,eventPhase:!0,metaKey:!0,pageX:!0,pageY:!0,shiftKey:!0,view:!0,char:!0,code:!0,charCode:!0,key:!0,keyCode:!0,button:!0,buttons:!0,clientX:!0,clientY:!0,offsetX:!0,offsetY:!0,pointerId:!0,pointerType:!0,screenX:!0,screenY:!0,targetTouches:!0,toElement:!0,touches:!0,which:!0},C.event.addProp),C.each({focus:"focusin",blur:"focusout"},function(t,e){C.event.special[t]={setup:function(){return De(this,t,Ne),!1},trigger:function(){return De(this,t),!0},_default:function(e){return b.get(e.target,t)},delegateType:e}}),C.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(e,i){C.event.special[e]={delegateType:i,bindType:i,handle:function(e){var t,n=e.relatedTarget,r=e.handleObj;return n&&(n===this||C.contains(this,n))||(e.type=r.origType,t=r.handler.apply(this,arguments),e.type=i),t}}}),C.fn.extend({on:function(e,t,n,r){return je(this,e,t,n,r)},one:function(e,t,n,r){return je(this,e,t,n,r,1)},off:function(e,t,n){var r,i;if(e&&e.preventDefault&&e.handleObj)r=e.handleObj,C(e.delegateTarget).off(r.namespace?r.origType+"."+r.namespace:r.origType,r.selector,r.handler);else{if("object"!=typeof e)return!1!==t&&"function"!=typeof t||(n=t,t=void 0),!1===n&&(n=d),this.each(function(){C.event.remove(this,e,n,t)});for(i in e)this.off(i,t,e[i])}return this}});var qe=/<script|<style|<link/i,Le=/checked\s*(?:[^=]|=\s*.checked.)/i,He=/^\s*<!\[CDATA\[|\]\]>\s*$/g;function Oe(e,t){return u(e,"table")&&u(11!==t.nodeType?t:t.firstChild,"tr")&&C(e).children("tbody")[0]||e}function Pe(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function Re(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Me(e,t){var n,r,i,o;if(1===t.nodeType){if(b.hasData(e)&&(o=b.get(e).events))for(i in b.remove(t,"handle events"),o)for(n=0,r=o[i].length;n<r;n++)C.event.add(t,i,o[i][n]);l.hasData(e)&&(e=l.access(e),e=C.extend({},e),l.set(t,e))}}function D(n,r,i,o){r=I(r);var e,t,a,s,u,l,c=0,f=n.length,p=f-1,d=r[0],h=v(d);if(h||1<f&&"string"==typeof d&&!y.checkClone&&Le.test(d))return n.each(function(e){var t=n.eq(e);h&&(r[0]=d.call(this,e,t.html())),D(t,r,i,o)});if(f&&(t=(e=ke(r,n[0].ownerDocument,!1,n,o)).firstChild,1===e.childNodes.length&&(e=t),t||o)){for(s=(a=C.map(j(e,"script"),Pe)).length;c<f;c++)u=e,c!==p&&(u=C.clone(u,!0,!0),s&&C.merge(a,j(u,"script"))),i.call(n[c],u,c);if(s)for(l=a[a.length-1].ownerDocument,C.map(a,Re),c=0;c<s;c++)u=a[c],Ce.test(u.type||"")&&!b.access(u,"globalEval")&&C.contains(l,u)&&(u.src&&"module"!==(u.type||"").toLowerCase()?C._evalUrl&&!u.noModule&&C._evalUrl(u.src,{nonce:u.nonce||u.getAttribute("nonce")},l):V(u.textContent.replace(He,""),u,l))}return n}function Ie(e,t,n){for(var r,i=t?C.filter(t,e):e,o=0;null!=(r=i[o]);o++)n||1!==r.nodeType||C.cleanData(j(r)),r.parentNode&&(n&&k(r)&&Ee(j(r,"script")),r.parentNode.removeChild(r));return e}C.extend({htmlPrefilter:function(e){return e},clone:function(e,t,n){var r,i,o,a,s,u,l,c=e.cloneNode(!0),f=k(e);if(!(y.noCloneChecked||1!==e.nodeType&&11!==e.nodeType||C.isXMLDoc(e)))for(a=j(c),r=0,i=(o=j(e)).length;r<i;r++)s=o[r],u=a[r],l=void 0,"input"===(l=u.nodeName.toLowerCase())&&we.test(s.type)?u.checked=s.checked:"input"!==l&&"textarea"!==l||(u.defaultValue=s.defaultValue);if(t)if(n)for(o=o||j(e),a=a||j(c),r=0,i=o.length;r<i;r++)Me(o[r],a[r]);else Me(e,c);return 0<(a=j(c,"script")).length&&Ee(a,!f&&j(e,"script")),c},cleanData:function(e){for(var t,n,r,i=C.event.special,o=0;void 0!==(n=e[o]);o++)if(m(n)){if(t=n[b.expando]){if(t.events)for(r in t.events)i[r]?C.event.remove(n,r):C.removeEvent(n,r,t.handle);n[b.expando]=void 0}n[l.expando]&&(n[l.expando]=void 0)}}}),C.fn.extend({detach:function(e){return Ie(this,e,!0)},remove:function(e){return Ie(this,e)},text:function(e){return f(this,function(e){return void 0===e?C.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=e)})},null,e,arguments.length)},append:function(){return D(this,arguments,function(e){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||Oe(this,e).appendChild(e)})},prepend:function(){return D(this,arguments,function(e){var t;1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(t=Oe(this,e)).insertBefore(e,t.firstChild)})},before:function(){return D(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this)})},after:function(){return D(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this.nextSibling)})},empty:function(){for(var e,t=0;null!=(e=this[t]);t++)1===e.nodeType&&(C.cleanData(j(e,!1)),e.textContent="");return this},clone:function(e,t){return e=null!=e&&e,t=null==t?e:t,this.map(function(){return C.clone(this,e,t)})},html:function(e){return f(this,function(e){var t=this[0]||{},n=0,r=this.length;if(void 0===e&&1===t.nodeType)return t.innerHTML;if("string"==typeof e&&!qe.test(e)&&!N[(Te.exec(e)||["",""])[1].toLowerCase()]){e=C.htmlPrefilter(e);try{for(;n<r;n++)1===(t=this[n]||{}).nodeType&&(C.cleanData(j(t,!1)),t.innerHTML=e);t=0}catch(e){}}t&&this.empty().append(e)},null,e,arguments.length)},replaceWith:function(){var n=[];return D(this,arguments,function(e){var t=this.parentNode;C.inArray(this,n)<0&&(C.cleanData(j(this)),t&&t.replaceChild(e,this))},n)}}),C.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(e,a){C.fn[e]=function(e){for(var t,n=[],r=C(e),i=r.length-1,o=0;o<=i;o++)t=o===i?this:this.clone(!0),C(r[o])[a](t),W.apply(n,t.get());return this.pushStack(n)}});function We(e){var t=e.ownerDocument.defaultView;return(t=t&&t.opener?t:w).getComputedStyle(e)}function Fe(e,t,n){var r,i={};for(r in t)i[r]=e.style[r],e.style[r]=t[r];for(r in n=n.call(e),t)e.style[r]=i[r];return n}var $e,Be,_e,ze,Ue,Xe,Ve,i,Ge=new RegExp("^("+e+")(?!px)[a-z%]+$","i"),Ye=/^--/,Qe=new RegExp(p.join("|"),"i"),o="[\\x20\\t\\r\\n\\f]",Je=new RegExp("^"+o+"+|((?:^|[^\\\\])(?:\\\\.)*)"+o+"+$","g");function Ke(){var e;i&&(Ve.style.cssText="position:absolute;left:-11111px;width:60px;margin-top:1px;padding:0;border:0",i.style.cssText="position:relative;display:block;box-sizing:border-box;overflow:scroll;margin:auto;border:1px;padding:1px;width:60%;top:1%",S.appendChild(Ve).appendChild(i),e=w.getComputedStyle(i),$e="1%"!==e.top,Xe=12===Ze(e.marginLeft),i.style.right="60%",ze=36===Ze(e.right),Be=36===Ze(e.width),i.style.position="absolute",_e=12===Ze(i.offsetWidth/3),S.removeChild(Ve),i=null)}function Ze(e){return Math.round(parseFloat(e))}function et(e,t,n){var r,i=Ye.test(t),o=e.style;return(n=n||We(e))&&(r=n.getPropertyValue(t)||n[t],""!==(r=i?r.replace(Je,"$1"):r)||k(e)||(r=C.style(e,t)),!y.pixelBoxStyles()&&Ge.test(r)&&Qe.test(t)&&(i=o.width,e=o.minWidth,t=o.maxWidth,o.minWidth=o.maxWidth=o.width=r,r=n.width,o.width=i,o.minWidth=e,o.maxWidth=t)),void 0!==r?r+"":r}function tt(e,t){return{get:function(){if(!e())return(this.get=t).apply(this,arguments);delete this.get}}}Ve=T.createElement("div"),(i=T.createElement("div")).style&&(i.style.backgroundClip="content-box",i.cloneNode(!0).style.backgroundClip="",y.clearCloneStyle="content-box"===i.style.backgroundClip,C.extend(y,{boxSizingReliable:function(){return Ke(),Be},pixelBoxStyles:function(){return Ke(),ze},pixelPosition:function(){return Ke(),$e},reliableMarginLeft:function(){return Ke(),Xe},scrollboxSize:function(){return Ke(),_e},reliableTrDimensions:function(){var e,t,n;return null==Ue&&(e=T.createElement("table"),t=T.createElement("tr"),n=T.createElement("div"),e.style.cssText="position:absolute;left:-11111px;border-collapse:separate",t.style.cssText="border:1px solid",t.style.height="1px",n.style.height="9px",n.style.display="block",S.appendChild(e).appendChild(t).appendChild(n),n=w.getComputedStyle(t),Ue=parseInt(n.height,10)+parseInt(n.borderTopWidth,10)+parseInt(n.borderBottomWidth,10)===t.offsetHeight,S.removeChild(e)),Ue}}));var nt=["Webkit","Moz","ms"],rt=T.createElement("div").style,it={};function ot(e){var t=C.cssProps[e]||it[e];return t||(e in rt?e:it[e]=function(e){var t=e[0].toUpperCase()+e.slice(1),n=nt.length;while(n--)if((e=nt[n]+t)in rt)return e}(e)||e)}var at=/^(none|table(?!-c[ea]).+)/,st={position:"absolute",visibility:"hidden",display:"block"},ut={letterSpacing:"0",fontWeight:"400"};function lt(e,t,n){var r=me.exec(t);return r?Math.max(0,r[2]-(n||0))+(r[3]||"px"):t}function ct(e,t,n,r,i,o){var a="width"===t?1:0,s=0,u=0;if(n===(r?"border":"content"))return 0;for(;a<4;a+=2)"margin"===n&&(u+=C.css(e,n+p[a],!0,i)),r?("content"===n&&(u-=C.css(e,"padding"+p[a],!0,i)),"margin"!==n&&(u-=C.css(e,"border"+p[a]+"Width",!0,i))):(u+=C.css(e,"padding"+p[a],!0,i),"padding"!==n?u+=C.css(e,"border"+p[a]+"Width",!0,i):s+=C.css(e,"border"+p[a]+"Width",!0,i));return!r&&0<=o&&(u+=Math.max(0,Math.ceil(e["offset"+t[0].toUpperCase()+t.slice(1)]-o-u-s-.5))||0),u}function ft(e,t,n){var r=We(e),i=(!y.boxSizingReliable()||n)&&"border-box"===C.css(e,"boxSizing",!1,r),o=i,a=et(e,t,r),s="offset"+t[0].toUpperCase()+t.slice(1);if(Ge.test(a)){if(!n)return a;a="auto"}return(!y.boxSizingReliable()&&i||!y.reliableTrDimensions()&&u(e,"tr")||"auto"===a||!parseFloat(a)&&"inline"===C.css(e,"display",!1,r))&&e.getClientRects().length&&(i="border-box"===C.css(e,"boxSizing",!1,r),(o=s in e)&&(a=e[s])),(a=parseFloat(a)||0)+ct(e,t,n||(i?"border":"content"),o,r,a)+"px"}function a(e,t,n,r,i){return new a.prototype.init(e,t,n,r,i)}C.extend({cssHooks:{opacity:{get:function(e,t){if(t)return""===(t=et(e,"opacity"))?"1":t}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,gridArea:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnStart:!0,gridRow:!0,gridRowEnd:!0,gridRowStart:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{},style:function(e,t,n,r){if(e&&3!==e.nodeType&&8!==e.nodeType&&e.style){var i,o,a,s=x(t),u=Ye.test(t),l=e.style;if(u||(t=ot(s)),a=C.cssHooks[t]||C.cssHooks[s],void 0===n)return a&&"get"in a&&void 0!==(i=a.get(e,!1,r))?i:l[t];"string"===(o=typeof n)&&(i=me.exec(n))&&i[1]&&(n=xe(e,t,i),o="number"),null!=n&&n==n&&("number"!==o||u||(n+=i&&i[3]||(C.cssNumber[s]?"":"px")),y.clearCloneStyle||""!==n||0!==t.indexOf("background")||(l[t]="inherit"),a&&"set"in a&&void 0===(n=a.set(e,n,r))||(u?l.setProperty(t,n):l[t]=n))}},css:function(e,t,n,r){var i,o=x(t);return Ye.test(t)||(t=ot(o)),"normal"===(i=void 0===(i=(o=C.cssHooks[t]||C.cssHooks[o])&&"get"in o?o.get(e,!0,n):i)?et(e,t,r):i)&&t in ut&&(i=ut[t]),(""===n||n)&&(o=parseFloat(i),!0===n||isFinite(o))?o||0:i}}),C.each(["height","width"],function(e,a){C.cssHooks[a]={get:function(e,t,n){if(t)return!at.test(C.css(e,"display"))||e.getClientRects().length&&e.getBoundingClientRect().width?ft(e,a,n):Fe(e,st,function(){return ft(e,a,n)})},set:function(e,t,n){var r=We(e),i=!y.scrollboxSize()&&"absolute"===r.position,o=(i||n)&&"border-box"===C.css(e,"boxSizing",!1,r),n=n?ct(e,a,n,o,r):0;return o&&i&&(n-=Math.ceil(e["offset"+a[0].toUpperCase()+a.slice(1)]-parseFloat(r[a])-ct(e,a,"border",!1,r)-.5)),n&&(o=me.exec(t))&&"px"!==(o[3]||"px")&&(e.style[a]=t,t=C.css(e,a)),lt(0,t,n)}}}),C.cssHooks.marginLeft=tt(y.reliableMarginLeft,function(e,t){if(t)return(parseFloat(et(e,"marginLeft"))||e.getBoundingClientRect().left-Fe(e,{marginLeft:0},function(){return e.getBoundingClientRect().left}))+"px"}),C.each({margin:"",padding:"",border:"Width"},function(i,o){C.cssHooks[i+o]={expand:function(e){for(var t=0,n={},r="string"==typeof e?e.split(" "):[e];t<4;t++)n[i+p[t]+o]=r[t]||r[t-2]||r[0];return n}},"margin"!==i&&(C.cssHooks[i+o].set=lt)}),C.fn.extend({css:function(e,t){return f(this,function(e,t,n){var r,i,o={},a=0;if(Array.isArray(t)){for(r=We(e),i=t.length;a<i;a++)o[t[a]]=C.css(e,t[a],!1,r);return o}return void 0!==n?C.style(e,t,n):C.css(e,t)},e,t,1<arguments.length)}}),((C.Tween=a).prototype={constructor:a,init:function(e,t,n,r,i,o){this.elem=e,this.prop=n,this.easing=i||C.easing._default,this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=o||(C.cssNumber[n]?"":"px")},cur:function(){var e=a.propHooks[this.prop];return(e&&e.get?e:a.propHooks._default).get(this)},run:function(e){var t,n=a.propHooks[this.prop];return this.options.duration?this.pos=t=C.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):this.pos=t=e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),(n&&n.set?n:a.propHooks._default).set(this),this}}).init.prototype=a.prototype,(a.propHooks={_default:{get:function(e){return 1!==e.elem.nodeType||null!=e.elem[e.prop]&&null==e.elem.style[e.prop]?e.elem[e.prop]:(e=C.css(e.elem,e.prop,""))&&"auto"!==e?e:0},set:function(e){C.fx.step[e.prop]?C.fx.step[e.prop](e):1!==e.elem.nodeType||!C.cssHooks[e.prop]&&null==e.elem.style[ot(e.prop)]?e.elem[e.prop]=e.now:C.style(e.elem,e.prop,e.now+e.unit)}}}).scrollTop=a.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},C.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2},_default:"swing"},C.fx=a.prototype.init,C.fx.step={};var q,pt,L,dt=/^(?:toggle|show|hide)$/,ht=/queueHooks$/;function gt(){pt&&(!1===T.hidden&&w.requestAnimationFrame?w.requestAnimationFrame(gt):w.setTimeout(gt,C.fx.interval),C.fx.tick())}function yt(){return w.setTimeout(function(){q=void 0}),q=Date.now()}function mt(e,t){var n,r=0,i={height:e};for(t=t?1:0;r<4;r+=2-t)i["margin"+(n=p[r])]=i["padding"+n]=e;return t&&(i.opacity=i.width=e),i}function vt(e,t,n){for(var r,i=(H.tweeners[t]||[]).concat(H.tweeners["*"]),o=0,a=i.length;o<a;o++)if(r=i[o].call(n,t,e))return r}function H(i,e,t){var n,o,r,a,s,u,l,c=0,f=H.prefilters.length,p=C.Deferred().always(function(){delete d.elem}),d=function(){if(!o){for(var e=q||yt(),e=Math.max(0,h.startTime+h.duration-e),t=1-(e/h.duration||0),n=0,r=h.tweens.length;n<r;n++)h.tweens[n].run(t);if(p.notifyWith(i,[h,t,e]),t<1&&r)return e;r||p.notifyWith(i,[h,1,0]),p.resolveWith(i,[h])}return!1},h=p.promise({elem:i,props:C.extend({},e),opts:C.extend(!0,{specialEasing:{},easing:C.easing._default},t),originalProperties:e,originalOptions:t,startTime:q||yt(),duration:t.duration,tweens:[],createTween:function(e,t){t=C.Tween(i,h.opts,e,t,h.opts.specialEasing[e]||h.opts.easing);return h.tweens.push(t),t},stop:function(e){var t=0,n=e?h.tweens.length:0;if(!o){for(o=!0;t<n;t++)h.tweens[t].run(1);e?(p.notifyWith(i,[h,1,0]),p.resolveWith(i,[h,e])):p.rejectWith(i,[h,e])}return this}}),g=h.props,y=g,m=h.opts.specialEasing;for(r in y)if(s=m[a=x(r)],u=y[r],Array.isArray(u)&&(s=u[1],u=y[r]=u[0]),r!==a&&(y[a]=u,delete y[r]),(l=C.cssHooks[a])&&"expand"in l)for(r in u=l.expand(u),delete y[a],u)r in y||(y[r]=u[r],m[r]=s);else m[a]=s;for(;c<f;c++)if(n=H.prefilters[c].call(h,i,g,h.opts))return v(n.stop)&&(C._queueHooks(h.elem,h.opts.queue).stop=n.stop.bind(n)),n;return C.map(g,vt,h),v(h.opts.start)&&h.opts.start.call(i,h),h.progress(h.opts.progress).done(h.opts.done,h.opts.complete).fail(h.opts.fail).always(h.opts.always),C.fx.timer(C.extend(d,{elem:i,anim:h,queue:h.opts.queue})),h}C.Animation=C.extend(H,{tweeners:{"*":[function(e,t){var n=this.createTween(e,t);return xe(n.elem,e,me.exec(t),n),n}]},tweener:function(e,t){for(var n,r=0,i=(e=v(e)?(t=e,["*"]):e.match(E)).length;r<i;r++)n=e[r],H.tweeners[n]=H.tweeners[n]||[],H.tweeners[n].unshift(t)},prefilters:[function(e,t,n){var r,i,o,a,s,u,l,c="width"in t||"height"in t,f=this,p={},d=e.style,h=e.nodeType&&ye(e),g=b.get(e,"fxshow");for(r in n.queue||(null==(a=C._queueHooks(e,"fx")).unqueued&&(a.unqueued=0,s=a.empty.fire,a.empty.fire=function(){a.unqueued||s()}),a.unqueued++,f.always(function(){f.always(function(){a.unqueued--,C.queue(e,"fx").length||a.empty.fire()})})),t)if(i=t[r],dt.test(i)){if(delete t[r],o=o||"toggle"===i,i===(h?"hide":"show")){if("show"!==i||!g||void 0===g[r])continue;h=!0}p[r]=g&&g[r]||C.style(e,r)}if((u=!C.isEmptyObject(t))||!C.isEmptyObject(p))for(r in c&&1===e.nodeType&&(n.overflow=[d.overflow,d.overflowX,d.overflowY],null==(l=g&&g.display)&&(l=b.get(e,"display")),"none"===(c=C.css(e,"display"))&&(l?c=l:(A([e],!0),l=e.style.display||l,c=C.css(e,"display"),A([e]))),("inline"===c||"inline-block"===c&&null!=l)&&"none"===C.css(e,"float")&&(u||(f.done(function(){d.display=l}),null==l&&(c=d.display,l="none"===c?"":c)),d.display="inline-block")),n.overflow&&(d.overflow="hidden",f.always(function(){d.overflow=n.overflow[0],d.overflowX=n.overflow[1],d.overflowY=n.overflow[2]})),u=!1,p)u||(g?"hidden"in g&&(h=g.hidden):g=b.access(e,"fxshow",{display:l}),o&&(g.hidden=!h),h&&A([e],!0),f.done(function(){for(r in h||A([e]),b.remove(e,"fxshow"),p)C.style(e,r,p[r])})),u=vt(h?g[r]:0,r,f),r in g||(g[r]=u.start,h&&(u.end=u.start,u.start=0))}],prefilter:function(e,t){t?H.prefilters.unshift(e):H.prefilters.push(e)}}),C.speed=function(e,t,n){var r=e&&"object"==typeof e?C.extend({},e):{complete:n||!n&&t||v(e)&&e,duration:e,easing:n&&t||t&&!v(t)&&t};return C.fx.off?r.duration=0:"number"!=typeof r.duration&&(r.duration in C.fx.speeds?r.duration=C.fx.speeds[r.duration]:r.duration=C.fx.speeds._default),null!=r.queue&&!0!==r.queue||(r.queue="fx"),r.old=r.complete,r.complete=function(){v(r.old)&&r.old.call(this),r.queue&&C.dequeue(this,r.queue)},r},C.fn.extend({fadeTo:function(e,t,n,r){return this.filter(ye).css("opacity",0).show().end().animate({opacity:t},e,n,r)},animate:function(t,e,n,r){function i(){var e=H(this,C.extend({},t),a);(o||b.get(this,"finish"))&&e.stop(!0)}var o=C.isEmptyObject(t),a=C.speed(e,n,r);return i.finish=i,o||!1===a.queue?this.each(i):this.queue(a.queue,i)},stop:function(i,e,o){function a(e){var t=e.stop;delete e.stop,t(o)}return"string"!=typeof i&&(o=e,e=i,i=void 0),e&&this.queue(i||"fx",[]),this.each(function(){var e=!0,t=null!=i&&i+"queueHooks",n=C.timers,r=b.get(this);if(t)r[t]&&r[t].stop&&a(r[t]);else for(t in r)r[t]&&r[t].stop&&ht.test(t)&&a(r[t]);for(t=n.length;t--;)n[t].elem!==this||null!=i&&n[t].queue!==i||(n[t].anim.stop(o),e=!1,n.splice(t,1));!e&&o||C.dequeue(this,i)})},finish:function(a){return!1!==a&&(a=a||"fx"),this.each(function(){var e,t=b.get(this),n=t[a+"queue"],r=t[a+"queueHooks"],i=C.timers,o=n?n.length:0;for(t.finish=!0,C.queue(this,a,[]),r&&r.stop&&r.stop.call(this,!0),e=i.length;e--;)i[e].elem===this&&i[e].queue===a&&(i[e].anim.stop(!0),i.splice(e,1));for(e=0;e<o;e++)n[e]&&n[e].finish&&n[e].finish.call(this);delete t.finish})}}),C.each(["toggle","show","hide"],function(e,r){var i=C.fn[r];C.fn[r]=function(e,t,n){return null==e||"boolean"==typeof e?i.apply(this,arguments):this.animate(mt(r,!0),e,t,n)}}),C.each({slideDown:mt("show"),slideUp:mt("hide"),slideToggle:mt("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(e,r){C.fn[e]=function(e,t,n){return this.animate(r,e,t,n)}}),C.timers=[],C.fx.tick=function(){var e,t=0,n=C.timers;for(q=Date.now();t<n.length;t++)(e=n[t])()||n[t]!==e||n.splice(t--,1);n.length||C.fx.stop(),q=void 0},C.fx.timer=function(e){C.timers.push(e),C.fx.start()},C.fx.interval=13,C.fx.start=function(){pt||(pt=!0,gt())},C.fx.stop=function(){pt=null},C.fx.speeds={slow:600,fast:200,_default:400},C.fn.delay=function(r,e){return r=C.fx&&C.fx.speeds[r]||r,this.queue(e=e||"fx",function(e,t){var n=w.setTimeout(e,r);t.stop=function(){w.clearTimeout(n)}})},L=T.createElement("input"),e=T.createElement("select").appendChild(T.createElement("option")),L.type="checkbox",y.checkOn=""!==L.value,y.optSelected=e.selected,(L=T.createElement("input")).value="t",L.type="radio",y.radioValue="t"===L.value;var xt,bt=C.expr.attrHandle,wt=(C.fn.extend({attr:function(e,t){return f(this,C.attr,e,t,1<arguments.length)},removeAttr:function(e){return this.each(function(){C.removeAttr(this,e)})}}),C.extend({attr:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return"undefined"==typeof e.getAttribute?C.prop(e,t,n):(1===o&&C.isXMLDoc(e)||(i=C.attrHooks[t.toLowerCase()]||(C.expr.match.bool.test(t)?xt:void 0)),void 0!==n?null===n?void C.removeAttr(e,t):i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:(e.setAttribute(t,n+""),n):!(i&&"get"in i&&null!==(r=i.get(e,t)))&&null==(r=C.find.attr(e,t))?void 0:r)},attrHooks:{type:{set:function(e,t){var n;if(!y.radioValue&&"radio"===t&&u(e,"input"))return n=e.value,e.setAttribute("type",t),n&&(e.value=n),t}}},removeAttr:function(e,t){var n,r=0,i=t&&t.match(E);if(i&&1===e.nodeType)while(n=i[r++])e.removeAttribute(n)}}),xt={set:function(e,t,n){return!1===t?C.removeAttr(e,n):e.setAttribute(n,n),n}},C.each(C.expr.match.bool.source.match(/\w+/g),function(e,t){var a=bt[t]||C.find.attr;bt[t]=function(e,t,n){var r,i,o=t.toLowerCase();return n||(i=bt[o],bt[o]=r,r=null!=a(e,t,n)?o:null,bt[o]=i),r}}),/^(?:input|select|textarea|button)$/i),Tt=/^(?:a|area)$/i;function O(e){return(e.match(E)||[]).join(" ")}function P(e){return e.getAttribute&&e.getAttribute("class")||""}function Ct(e){return Array.isArray(e)?e:"string"==typeof e&&e.match(E)||[]}C.fn.extend({prop:function(e,t){return f(this,C.prop,e,t,1<arguments.length)},removeProp:function(e){return this.each(function(){delete this[C.propFix[e]||e]})}}),C.extend({prop:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return 1===o&&C.isXMLDoc(e)||(t=C.propFix[t]||t,i=C.propHooks[t]),void 0!==n?i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:e[t]=n:i&&"get"in i&&null!==(r=i.get(e,t))?r:e[t]},propHooks:{tabIndex:{get:function(e){var t=C.find.attr(e,"tabindex");return t?parseInt(t,10):wt.test(e.nodeName)||Tt.test(e.nodeName)&&e.href?0:-1}}},propFix:{for:"htmlFor",class:"className"}}),y.optSelected||(C.propHooks.selected={get:function(e){e=e.parentNode;return e&&e.parentNode&&e.parentNode.selectedIndex,null},set:function(e){e=e.parentNode;e&&(e.selectedIndex,e.parentNode&&e.parentNode.selectedIndex)}}),C.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){C.propFix[this.toLowerCase()]=this}),C.fn.extend({addClass:function(t){var e,n,r,i,o,a;return v(t)?this.each(function(e){C(this).addClass(t.call(this,e,P(this)))}):(e=Ct(t)).length?this.each(function(){if(r=P(this),n=1===this.nodeType&&" "+O(r)+" "){for(o=0;o<e.length;o++)i=e[o],n.indexOf(" "+i+" ")<0&&(n+=i+" ");a=O(n),r!==a&&this.setAttribute("class",a)}}):this},removeClass:function(t){var e,n,r,i,o,a;return v(t)?this.each(function(e){C(this).removeClass(t.call(this,e,P(this)))}):arguments.length?(e=Ct(t)).length?this.each(function(){if(r=P(this),n=1===this.nodeType&&" "+O(r)+" "){for(o=0;o<e.length;o++){i=e[o];while(-1<n.indexOf(" "+i+" "))n=n.replace(" "+i+" "," ")}a=O(n),r!==a&&this.setAttribute("class",a)}}):this:this.attr("class","")},toggleClass:function(t,n){var e,r,i,o,a=typeof t,s="string"==a||Array.isArray(t);return v(t)?this.each(function(e){C(this).toggleClass(t.call(this,e,P(this),n),n)}):"boolean"==typeof n&&s?n?this.addClass(t):this.removeClass(t):(e=Ct(t),this.each(function(){if(s)for(o=C(this),i=0;i<e.length;i++)r=e[i],o.hasClass(r)?o.removeClass(r):o.addClass(r);else void 0!==t&&"boolean"!=a||((r=P(this))&&b.set(this,"__className__",r),this.setAttribute&&this.setAttribute("class",!r&&!1!==t&&b.get(this,"__className__")||""))}))},hasClass:function(e){var t,n=0,r=" "+e+" ";while(t=this[n++])if(1===t.nodeType&&-1<(" "+O(P(t))+" ").indexOf(r))return!0;return!1}});function Et(e){e.stopPropagation()}var St=/\r/g,kt=(C.fn.extend({val:function(t){var n,e,r,i=this[0];return arguments.length?(r=v(t),this.each(function(e){1===this.nodeType&&(null==(e=r?t.call(this,e,C(this).val()):t)?e="":"number"==typeof e?e+="":Array.isArray(e)&&(e=C.map(e,function(e){return null==e?"":e+""})),(n=C.valHooks[this.type]||C.valHooks[this.nodeName.toLowerCase()])&&"set"in n&&void 0!==n.set(this,e,"value")||(this.value=e))})):i?(n=C.valHooks[i.type]||C.valHooks[i.nodeName.toLowerCase()])&&"get"in n&&void 0!==(e=n.get(i,"value"))?e:"string"==typeof(e=i.value)?e.replace(St,""):null==e?"":e:void 0}}),C.extend({valHooks:{option:{get:function(e){var t=C.find.attr(e,"value");return null!=t?t:O(C.text(e))}},select:{get:function(e){for(var t,n=e.options,r=e.selectedIndex,i="select-one"===e.type,o=i?null:[],a=i?r+1:n.length,s=r<0?a:i?r:0;s<a;s++)if(((t=n[s]).selected||s===r)&&!t.disabled&&(!t.parentNode.disabled||!u(t.parentNode,"optgroup"))){if(t=C(t).val(),i)return t;o.push(t)}return o},set:function(e,t){var n,r,i=e.options,o=C.makeArray(t),a=i.length;while(a--)((r=i[a]).selected=-1<C.inArray(C.valHooks.option.get(r),o))&&(n=!0);return n||(e.selectedIndex=-1),o}}}}),C.each(["radio","checkbox"],function(){C.valHooks[this]={set:function(e,t){if(Array.isArray(t))return e.checked=-1<C.inArray(C(e).val(),t)}},y.checkOn||(C.valHooks[this].get=function(e){return null===e.getAttribute("value")?"on":e.value})}),y.focusin="onfocusin"in w,/^(?:focusinfocus|focusoutblur)$/),At=(C.extend(C.event,{trigger:function(e,t,n,r){var i,o,a,s,u,l,c,f=[n||T],p=_.call(e,"type")?e.type:e,d=_.call(e,"namespace")?e.namespace.split("."):[],h=c=o=n=n||T;if(3!==n.nodeType&&8!==n.nodeType&&!kt.test(p+C.event.triggered)&&(-1<p.indexOf(".")&&(p=(d=p.split(".")).shift(),d.sort()),s=p.indexOf(":")<0&&"on"+p,(e=e[C.expando]?e:new C.Event(p,"object"==typeof e&&e)).isTrigger=r?2:3,e.namespace=d.join("."),e.rnamespace=e.namespace?new RegExp("(^|\\.)"+d.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,e.result=void 0,e.target||(e.target=n),t=null==t?[e]:C.makeArray(t,[e]),l=C.event.special[p]||{},r||!l.trigger||!1!==l.trigger.apply(n,t))){if(!r&&!l.noBubble&&!g(n)){for(a=l.delegateType||p,kt.test(a+p)||(h=h.parentNode);h;h=h.parentNode)f.push(h),o=h;o===(n.ownerDocument||T)&&f.push(o.defaultView||o.parentWindow||w)}i=0;while((h=f[i++])&&!e.isPropagationStopped())c=h,e.type=1<i?a:l.bindType||p,(u=(b.get(h,"events")||Object.create(null))[e.type]&&b.get(h,"handle"))&&u.apply(h,t),(u=s&&h[s])&&u.apply&&m(h)&&(e.result=u.apply(h,t),!1===e.result&&e.preventDefault());return e.type=p,r||e.isDefaultPrevented()||l._default&&!1!==l._default.apply(f.pop(),t)||!m(n)||s&&v(n[p])&&!g(n)&&((o=n[s])&&(n[s]=null),C.event.triggered=p,e.isPropagationStopped()&&c.addEventListener(p,Et),n[p](),e.isPropagationStopped()&&c.removeEventListener(p,Et),C.event.triggered=void 0,o&&(n[s]=o)),e.result}},simulate:function(e,t,n){n=C.extend(new C.Event,n,{type:e,isSimulated:!0});C.event.trigger(n,null,t)}}),C.fn.extend({trigger:function(e,t){return this.each(function(){C.event.trigger(e,t,this)})},triggerHandler:function(e,t){var n=this[0];if(n)return C.event.trigger(e,t,n,!0)}}),y.focusin||C.each({focus:"focusin",blur:"focusout"},function(n,r){function i(e){C.event.simulate(r,e.target,C.event.fix(e))}C.event.special[r]={setup:function(){var e=this.ownerDocument||this.document||this,t=b.access(e,r);t||e.addEventListener(n,i,!0),b.access(e,r,(t||0)+1)},teardown:function(){var e=this.ownerDocument||this.document||this,t=b.access(e,r)-1;t?b.access(e,r,t):(e.removeEventListener(n,i,!0),b.remove(e,r))}}}),w.location),Nt={guid:Date.now()},jt=/\?/,Dt=(C.parseXML=function(e){var t,n;if(!e||"string"!=typeof e)return null;try{t=(new w.DOMParser).parseFromString(e,"text/xml")}catch(e){}return n=t&&t.getElementsByTagName("parsererror")[0],t&&!n||C.error("Invalid XML: "+(n?C.map(n.childNodes,function(e){return e.textContent}).join("\n"):e)),t},/\[\]$/),qt=/\r?\n/g,Lt=/^(?:submit|button|image|reset|file)$/i,Ht=/^(?:input|select|textarea|keygen)/i;C.param=function(e,t){function n(e,t){t=v(t)?t():t,i[i.length]=encodeURIComponent(e)+"="+encodeURIComponent(null==t?"":t)}var r,i=[];if(null==e)return"";if(Array.isArray(e)||e.jquery&&!C.isPlainObject(e))C.each(e,function(){n(this.name,this.value)});else for(r in e)!function n(r,e,i,o){if(Array.isArray(e))C.each(e,function(e,t){i||Dt.test(r)?o(r,t):n(r+"["+("object"==typeof t&&null!=t?e:"")+"]",t,i,o)});else if(i||"object"!==h(e))o(r,e);else for(var t in e)n(r+"["+t+"]",e[t],i,o)}(r,e[r],t,n);return i.join("&")},C.fn.extend({serialize:function(){return C.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var e=C.prop(this,"elements");return e?C.makeArray(e):this}).filter(function(){var e=this.type;return this.name&&!C(this).is(":disabled")&&Ht.test(this.nodeName)&&!Lt.test(e)&&(this.checked||!we.test(e))}).map(function(e,t){var n=C(this).val();return null==n?null:Array.isArray(n)?C.map(n,function(e){return{name:t.name,value:e.replace(qt,"\r\n")}}):{name:t.name,value:n.replace(qt,"\r\n")}}).get()}});var Ot=/%20/g,Pt=/#.*$/,Rt=/([?&])_=[^&]*/,Mt=/^(.*?):[ \t]*([^\r\n]*)$/gm,It=/^(?:GET|HEAD)$/,Wt=/^\/\//,Ft={},$t={},Bt="*/".concat("*"),_t=T.createElement("a");function zt(o){return function(e,t){"string"!=typeof e&&(t=e,e="*");var n,r=0,i=e.toLowerCase().match(E)||[];if(v(t))while(n=i[r++])"+"===n[0]?(n=n.slice(1)||"*",(o[n]=o[n]||[]).unshift(t)):(o[n]=o[n]||[]).push(t)}}function Ut(t,r,i,o){var a={},s=t===$t;function u(e){var n;return a[e]=!0,C.each(t[e]||[],function(e,t){t=t(r,i,o);return"string"!=typeof t||s||a[t]?s?!(n=t):void 0:(r.dataTypes.unshift(t),u(t),!1)}),n}return u(r.dataTypes[0])||!a["*"]&&u("*")}function Xt(e,t){var n,r,i=C.ajaxSettings.flatOptions||{};for(n in t)void 0!==t[n]&&((i[n]?e:r=r||{})[n]=t[n]);return r&&C.extend(!0,e,r),e}_t.href=At.href,C.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:At.href,type:"GET",isLocal:/^(?:about|app|app-storage|.+-extension|file|res|widget):$/.test(At.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Bt,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":JSON.parse,"text xml":C.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(e,t){return t?Xt(Xt(e,C.ajaxSettings),t):Xt(C.ajaxSettings,e)},ajaxPrefilter:zt(Ft),ajaxTransport:zt($t),ajax:function(e,t){"object"==typeof e&&(t=e,e=void 0);var u,l,c,n,f,p,d,r,h=C.ajaxSetup({},t=t||{}),g=h.context||h,y=h.context&&(g.nodeType||g.jquery)?C(g):C.event,m=C.Deferred(),v=C.Callbacks("once memory"),x=h.statusCode||{},i={},o={},a="canceled",b={readyState:0,getResponseHeader:function(e){var t;if(p){if(!n){n={};while(t=Mt.exec(c))n[t[1].toLowerCase()+" "]=(n[t[1].toLowerCase()+" "]||[]).concat(t[2])}t=n[e.toLowerCase()+" "]}return null==t?null:t.join(", ")},getAllResponseHeaders:function(){return p?c:null},setRequestHeader:function(e,t){return null==p&&(e=o[e.toLowerCase()]=o[e.toLowerCase()]||e,i[e]=t),this},overrideMimeType:function(e){return null==p&&(h.mimeType=e),this},statusCode:function(e){if(e)if(p)b.always(e[b.status]);else for(var t in e)x[t]=[x[t],e[t]];return this},abort:function(e){e=e||a;return u&&u.abort(e),s(0,e),this}};if(m.promise(b),h.url=((e||h.url||At.href)+"").replace(Wt,At.protocol+"//"),h.type=t.method||t.type||h.method||h.type,h.dataTypes=(h.dataType||"*").toLowerCase().match(E)||[""],null==h.crossDomain){e=T.createElement("a");try{e.href=h.url,e.href=e.href,h.crossDomain=_t.protocol+"//"+_t.host!=e.protocol+"//"+e.host}catch(e){h.crossDomain=!0}}if(h.data&&h.processData&&"string"!=typeof h.data&&(h.data=C.param(h.data,h.traditional)),Ut(Ft,h,t,b),!p){for(r in(d=C.event&&h.global)&&0==C.active++&&C.event.trigger("ajaxStart"),h.type=h.type.toUpperCase(),h.hasContent=!It.test(h.type),l=h.url.replace(Pt,""),h.hasContent?h.data&&h.processData&&0===(h.contentType||"").indexOf("application/x-www-form-urlencoded")&&(h.data=h.data.replace(Ot,"+")):(e=h.url.slice(l.length),h.data&&(h.processData||"string"==typeof h.data)&&(l+=(jt.test(l)?"&":"?")+h.data,delete h.data),!1===h.cache&&(l=l.replace(Rt,"$1"),e=(jt.test(l)?"&":"?")+"_="+Nt.guid+++e),h.url=l+e),h.ifModified&&(C.lastModified[l]&&b.setRequestHeader("If-Modified-Since",C.lastModified[l]),C.etag[l]&&b.setRequestHeader("If-None-Match",C.etag[l])),(h.data&&h.hasContent&&!1!==h.contentType||t.contentType)&&b.setRequestHeader("Content-Type",h.contentType),b.setRequestHeader("Accept",h.dataTypes[0]&&h.accepts[h.dataTypes[0]]?h.accepts[h.dataTypes[0]]+("*"!==h.dataTypes[0]?", "+Bt+"; q=0.01":""):h.accepts["*"]),h.headers)b.setRequestHeader(r,h.headers[r]);if(h.beforeSend&&(!1===h.beforeSend.call(g,b,h)||p))return b.abort();if(a="abort",v.add(h.complete),b.done(h.success),b.fail(h.error),u=Ut($t,h,t,b)){if(b.readyState=1,d&&y.trigger("ajaxSend",[b,h]),p)return b;h.async&&0<h.timeout&&(f=w.setTimeout(function(){b.abort("timeout")},h.timeout));try{p=!1,u.send(i,s)}catch(e){if(p)throw e;s(-1,e)}}else s(-1,"No Transport")}return b;function s(e,t,n,r){var i,o,a,s=t;p||(p=!0,f&&w.clearTimeout(f),u=void 0,c=r||"",b.readyState=0<e?4:0,r=200<=e&&e<300||304===e,n&&(a=function(e,t,n){var r,i,o,a,s=e.contents,u=e.dataTypes;while("*"===u[0])u.shift(),void 0===r&&(r=e.mimeType||t.getResponseHeader("Content-Type"));if(r)for(i in s)if(s[i]&&s[i].test(r)){u.unshift(i);break}if(u[0]in n)o=u[0];else{for(i in n){if(!u[0]||e.converters[i+" "+u[0]]){o=i;break}a=a||i}o=o||a}if(o)return o!==u[0]&&u.unshift(o),n[o]}(h,b,n)),!r&&-1<C.inArray("script",h.dataTypes)&&C.inArray("json",h.dataTypes)<0&&(h.converters["text script"]=function(){}),a=function(e,t,n,r){var i,o,a,s,u,l={},c=e.dataTypes.slice();if(c[1])for(a in e.converters)l[a.toLowerCase()]=e.converters[a];o=c.shift();while(o)if(e.responseFields[o]&&(n[e.responseFields[o]]=t),!u&&r&&e.dataFilter&&(t=e.dataFilter(t,e.dataType)),u=o,o=c.shift())if("*"===o)o=u;else if("*"!==u&&u!==o){if(!(a=l[u+" "+o]||l["* "+o]))for(i in l)if((s=i.split(" "))[1]===o&&(a=l[u+" "+s[0]]||l["* "+s[0]])){!0===a?a=l[i]:!0!==l[i]&&(o=s[0],c.unshift(s[1]));break}if(!0!==a)if(a&&e.throws)t=a(t);else try{t=a(t)}catch(e){return{state:"parsererror",error:a?e:"No conversion from "+u+" to "+o}}}return{state:"success",data:t}}(h,a,b,r),r?(h.ifModified&&((n=b.getResponseHeader("Last-Modified"))&&(C.lastModified[l]=n),(n=b.getResponseHeader("etag"))&&(C.etag[l]=n)),204===e||"HEAD"===h.type?s="nocontent":304===e?s="notmodified":(s=a.state,i=a.data,r=!(o=a.error))):(o=s,!e&&s||(s="error",e<0&&(e=0))),b.status=e,b.statusText=(t||s)+"",r?m.resolveWith(g,[i,s,b]):m.rejectWith(g,[b,s,o]),b.statusCode(x),x=void 0,d&&y.trigger(r?"ajaxSuccess":"ajaxError",[b,h,r?i:o]),v.fireWith(g,[b,s]),d&&(y.trigger("ajaxComplete",[b,h]),--C.active||C.event.trigger("ajaxStop")))}},getJSON:function(e,t,n){return C.get(e,t,n,"json")},getScript:function(e,t){return C.get(e,void 0,t,"script")}}),C.each(["get","post"],function(e,i){C[i]=function(e,t,n,r){return v(t)&&(r=r||n,n=t,t=void 0),C.ajax(C.extend({url:e,type:i,dataType:r,data:t,success:n},C.isPlainObject(e)&&e))}}),C.ajaxPrefilter(function(e){for(var t in e.headers)"content-type"===t.toLowerCase()&&(e.contentType=e.headers[t]||"")}),C._evalUrl=function(e,t,n){return C.ajax({url:e,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,converters:{"text script":function(){}},dataFilter:function(e){C.globalEval(e,t,n)}})},C.fn.extend({wrapAll:function(e){return this[0]&&(v(e)&&(e=e.call(this[0])),e=C(e,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&e.insertBefore(this[0]),e.map(function(){var e=this;while(e.firstElementChild)e=e.firstElementChild;return e}).append(this)),this},wrapInner:function(n){return v(n)?this.each(function(e){C(this).wrapInner(n.call(this,e))}):this.each(function(){var e=C(this),t=e.contents();t.length?t.wrapAll(n):e.append(n)})},wrap:function(t){var n=v(t);return this.each(function(e){C(this).wrapAll(n?t.call(this,e):t)})},unwrap:function(e){return this.parent(e).not("body").each(function(){C(this).replaceWith(this.childNodes)}),this}}),C.expr.pseudos.hidden=function(e){return!C.expr.pseudos.visible(e)},C.expr.pseudos.visible=function(e){return!!(e.offsetWidth||e.offsetHeight||e.getClientRects().length)},C.ajaxSettings.xhr=function(){try{return new w.XMLHttpRequest}catch(e){}};var Vt={0:200,1223:204},Gt=C.ajaxSettings.xhr(),Yt=(y.cors=!!Gt&&"withCredentials"in Gt,y.ajax=Gt=!!Gt,C.ajaxTransport(function(i){var o,a;if(y.cors||Gt&&!i.crossDomain)return{send:function(e,t){var n,r=i.xhr();if(r.open(i.type,i.url,i.async,i.username,i.password),i.xhrFields)for(n in i.xhrFields)r[n]=i.xhrFields[n];for(n in i.mimeType&&r.overrideMimeType&&r.overrideMimeType(i.mimeType),i.crossDomain||e["X-Requested-With"]||(e["X-Requested-With"]="XMLHttpRequest"),e)r.setRequestHeader(n,e[n]);o=function(e){return function(){o&&(o=a=r.onload=r.onerror=r.onabort=r.ontimeout=r.onreadystatechange=null,"abort"===e?r.abort():"error"===e?"number"!=typeof r.status?t(0,"error"):t(r.status,r.statusText):t(Vt[r.status]||r.status,r.statusText,"text"!==(r.responseType||"text")||"string"!=typeof r.responseText?{binary:r.response}:{text:r.responseText},r.getAllResponseHeaders()))}},r.onload=o(),a=r.onerror=r.ontimeout=o("error"),void 0!==r.onabort?r.onabort=a:r.onreadystatechange=function(){4===r.readyState&&w.setTimeout(function(){o&&a()})},o=o("abort");try{r.send(i.hasContent&&i.data||null)}catch(e){if(o)throw e}},abort:function(){o&&o()}}}),C.ajaxPrefilter(function(e){e.crossDomain&&(e.contents.script=!1)}),C.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(e){return C.globalEval(e),e}}}),C.ajaxPrefilter("script",function(e){void 0===e.cache&&(e.cache=!1),e.crossDomain&&(e.type="GET")}),C.ajaxTransport("script",function(n){var r,i;if(n.crossDomain||n.scriptAttrs)return{send:function(e,t){r=C("<script>").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),T.head.appendChild(r[0])},abort:function(){i&&i()}}}),[]),Qt=/(=)\?(?=&|$)|\?\?/,Jt=(C.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Yt.pop()||C.expando+"_"+Nt.guid++;return this[e]=!0,e}}),C.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Qt.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Qt.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=v(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Qt,"$1"+r):!1!==e.jsonp&&(e.url+=(jt.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||C.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=w[r],w[r]=function(){o=arguments},n.always(function(){void 0===i?C(w).removeProp(r):w[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,Yt.push(r)),o&&v(i)&&i(o[0]),o=i=void 0}),"script"}),y.createHTMLDocument=((o=T.implementation.createHTMLDocument("").body).innerHTML="<form></form><form></form>",2===o.childNodes.length),C.parseHTML=function(e,t,n){var r;return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(y.createHTMLDocument?((r=(t=T.implementation.createHTMLDocument("")).createElement("base")).href=T.location.href,t.head.appendChild(r)):t=T),r=!n&&[],(n=J.exec(e))?[t.createElement(n[1])]:(n=ke([e],t,r),r&&r.length&&C(r).remove(),C.merge([],n.childNodes)))},C.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1<s&&(r=O(e.slice(s)),e=e.slice(0,s)),v(t)?(n=t,t=void 0):t&&"object"==typeof t&&(i="POST"),0<a.length&&C.ajax({url:e,type:i||"GET",dataType:"html",data:t}).done(function(e){o=arguments,a.html(r?C("<div>").append(C.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},C.expr.pseudos.animated=function(t){return C.grep(C.timers,function(e){return t===e.elem}).length},C.offset={setOffset:function(e,t,n){var r,i,o,a,s=C.css(e,"position"),u=C(e),l={};"static"===s&&(e.style.position="relative"),o=u.offset(),r=C.css(e,"top"),a=C.css(e,"left"),s=("absolute"===s||"fixed"===s)&&-1<(r+a).indexOf("auto")?(i=(s=u.position()).top,s.left):(i=parseFloat(r)||0,parseFloat(a)||0),null!=(t=v(t)?t.call(e,n,C.extend({},o)):t).top&&(l.top=t.top-o.top+i),null!=t.left&&(l.left=t.left-o.left+s),"using"in t?t.using.call(e,l):u.css(l)}},C.fn.extend({offset:function(t){var e,n;return arguments.length?void 0===t?this:this.each(function(e){C.offset.setOffset(this,t,e)}):(n=this[0])?n.getClientRects().length?(e=n.getBoundingClientRect(),n=n.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===C.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===C.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=C(e).offset()).top+=C.css(e,"borderTopWidth",!0),i.left+=C.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-C.css(r,"marginTop",!0),left:t.left-i.left-C.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===C.css(e,"position"))e=e.offsetParent;return e||S})}}),C.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;C.fn[t]=function(e){return f(this,function(e,t,n){var r;if(g(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),C.each(["top","left"],function(e,n){C.cssHooks[n]=tt(y.pixelPosition,function(e,t){if(t)return t=et(e,n),Ge.test(t)?C(e).position()[n]+"px":t})}),C.each({Height:"height",Width:"width"},function(a,s){C.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){C.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return f(this,function(e,t,n){var r;return g(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?C.css(e,t,i):C.style(e,t,n,i)},s,n?e:void 0,n)}})}),C.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){C.fn[t]=function(e){return this.on(t,e)}}),C.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),C.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){C.fn[n]=function(e,t){return 0<arguments.length?this.on(n,null,e,t):this.trigger(n)}}),/^[\s\uFEFF\xA0]+|([^\s\uFEFF\xA0])[\s\uFEFF\xA0]+$/g),Kt=(C.proxy=function(e,t){var n,r;if("string"==typeof t&&(r=e[t],t=e,e=r),v(e))return n=s.call(arguments,2),(r=function(){return e.apply(t||this,n.concat(s.call(arguments)))}).guid=e.guid=e.guid||C.guid++,r},C.holdReady=function(e){e?C.readyWait++:C.ready(!0)},C.isArray=Array.isArray,C.parseJSON=JSON.parse,C.nodeName=u,C.isFunction=v,C.isWindow=g,C.camelCase=x,C.type=h,C.now=Date.now,C.isNumeric=function(e){var t=C.type(e);return("number"===t||"string"===t)&&!isNaN(e-parseFloat(e))},C.trim=function(e){return null==e?"":(e+"").replace(Jt,"$1")},"function"==typeof define&&define.amd&&define("jquery",[],function(){return C}),w.jQuery),Zt=w.$;return C.noConflict=function(e){return w.$===C&&(w.$=Zt),e&&w.jQuery===C&&(w.jQuery=Kt),C},"undefined"==typeof R&&(w.jQuery=w.$=C),C});
+</script>
+<meta name="viewport" content="width=device-width, initial-scale=1" />
+<style type="text/css">  html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,hgroup,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block;vertical-align:baseline}audio:not([controls]){display:none;height:0}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}abbr[title]{border-bottom:none;text-decoration:underline;text-decoration:underline dotted}b,strong{font-weight:bold}dfn{font-style:italic}h1{font-size:2em;margin:.67em 0}mark{background:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-0.5em}sub{bottom:-0.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:1em 40px}hr{box-sizing:content-box;height:0}pre{overflow:auto}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}button,input,optgroup,select,textarea{color:inherit;font:inherit;margin:0}button{overflow:visible}button,select{text-transform:none}button,html input[type="button"],input[type="reset"],input[type="submit"]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}input{line-height:normal}input[type="checkbox"],input[type="radio"]{box-sizing:border-box;padding:0}input[type="number"]::-webkit-inner-spin-button,input[type="number"]::-webkit-outer-spin-button{height:auto}input[type="search"]{-webkit-appearance:textfield;box-sizing:content-box}input[type="search"]::-webkit-search-cancel-button,input[type="search"]::-webkit-search-decoration{-webkit-appearance:none}fieldset{border:1px solid #c0c0c0;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}textarea{overflow:auto}optgroup{font-weight:bold}table{border-collapse:collapse;border-spacing:0}td,th{padding:0} @media print{*,*:before,*:after{color:#000!important;text-shadow:none!important;background:transparent!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:"(" attr(href) ")"}abbr[title]:after{content:"(" attr(title) ")"}a[href^="#"]:after,a[href^="javascript:"]:after{content:""}pre,blockquote{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100%!important}p,h2,h3{orphans:3;widows:3}h2,h3{page-break-after:avoid}.navbar{display:none}.btn>.caret,.dropup>.btn>.caret{border-top-color:#000!important}.label{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered th,.table-bordered td{border:1px solid #ddd!important}}@font-face{font-family:"Glyphicons Halflings";src:url(data:application/vnd.ms-fontobject;base64,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);src:url(data:application/vnd.ms-fontobject;base64,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) format("embedded-opentype"),url(data:font/woff2;base64,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) format("woff2"),url(data:application/font-woff;base64,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) format("woff"),url(data:application/font-sfnt;base64,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JbwmFCZ+JuYnUCe8J/goNChwKKwpoCnMKiYqSiqEKworeiwILGgsuizsLRwtiC30LiguZi6iLtgvDi9GL34vsi/4MD4whDDSMRIxYDGuMegyJDJeMpoy3jMiMz4zaDO2NBg0YDSoNNI1LDWeNeg2PjZ8Ntw3GjdON5I31DgQOEI4hjjIOQo5SjmIOcw6HDpsOpo63jugO9w8GDxQPKI8+D0yPew+Oj6MPtQ/KD9uP6o/+kBIQIBAxkECQX5CGEKoQu5DGENCQ3ZDoEPKRBBEYESuRPZFWkW2RgZGdEa0RvZHNkd2R7ZH9kgWSDJITkhqSIZIzEkSSThJXkmESapKAkouSlIAAQAAARcApwARAAAAAAACAAAAAQABAAAAQAAuAAAAAAAAABAAxgABAAAAAAATABIAAAADAAEECQAAAGoAEgADAAEECQABACgAfAADAAEECQACAA4ApAADAAEECQADAEwAsgADAAEECQAEADgA/gADAAEECQAFAHgBNgADAAEECQAGADYBrgADAAEECQAIABYB5AADAAEECQAJABYB+gADAAEECQALACQCEAADAAEECQAMACQCNAADAAEECQATACQCWAADAAEECQDIABYCfAADAAEECQDJADACkgADAAEECdkDABoCwnd3dy5nbHlwaGljb25zLmNvbQBDAG8AcAB5AHIAaQBnAGgAdAAgAKkAIAAyADAAMQA0ACAAYgB5ACAASgBhAG4AIABLAG8AdgBhAHIAaQBrAC4AIABBAGwAbAAgAHIAaQBnAGgAdABzACAAcgBlAHMAZQByAHYAZQBkAC4ARwBMAFkAUABIAEkAQwBPAE4AUwAgAEgAYQBsAGYAbABpAG4AZwBzAFIAZQBnAHUAbABhAHIAMQAuADAAMAA5ADsAVQBLAFcATgA7AEcATABZAFAASABJAEMATwBOAFMASABhAGwAZgBsAGkAbgBnAHMALQBSAGUAZwB1AGwAYQByAEcATABZAFAASABJAEMATwBOAFMAIABIAGEAbABmAGwAaQBuAGcAcwAgAFIAZQBnAHUAbABhAHIAVgBlAHIAcwBpAG8AbgAgADEALgAwADAAOQA7AFAAUwAgADAAMAAxAC4AMAAwADkAOwBoAG8AdABjAG8AbgB2ACAAMQAuADAALgA3ADAAOwBtAGEAawBlAG8AdABmAC4AbABpAGIAMgAuADUALgA1ADgAMwAyADkARwBMAFkAUABIAEkAQwBPAE4AUwBIAGEAbABmAGwAaQBuAGcAcwAtAFIAZQBnAHUAbABhAHIASgBhAG4AIABLAG8AdgBhAHIAaQBrAEoAYQBuACAASwBvAHYAYQByAGkAawB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQBXAGUAYgBmAG8AbgB0ACAAMQAuADAAVwBlAGQAIABPAGMAdAAgADIAOQAgADAANgA6ADMANgA6ADAANwAgADIAMAAxADQARgBvAG4AdAAgAFMAcQB1AGkAcgByAGUAbAAAAAIAAAAAAAD/tQAyAAAAAAAAAAAAAAAAAAAAAAAAAAABFwAAAQIBAwADAA0ADgEEAJYBBQEGAQcBCAEJAQoBCwEMAQ0BDgEPARABEQESARMA7wEUARUBFgEXARgBGQEaARsBHAEdAR4BHwEgASEBIgEjASQBJQEmAScBKAEpASoBKwEsAS0BLgEvATABMQEyATMBNAE1ATYBNwE4ATkBOgE7ATwBPQE+AT8BQAFBAUIBQwFEAUUBRgFHAUgBSQFKAUsBTAFNAU4BTwFQAVEBUgFTAVQBVQFWAVcBWAFZAVoBWwFcAV0BXgFfAWABYQFiAWMBZAFlAWYBZwFoAWkBagFrAWwBbQFuAW8BcAFxAXIBcwF0AXUBdgF3AXgBeQF6AXsBfAF9AX4BfwGAAYEBggGDAYQBhQGGAYcBiAGJAYoBiwGMAY0BjgGPAZABkQGSAZMBlAGVAZYBlwGYAZkBmgGbAZwBnQGeAZ8BoAGhAaIBowGkAaUBpgGnAagBqQGqAasBrAGtAa4BrwGwAbEBsgGzAbQBtQG2AbcBuAG5AboBuwG8Ab0BvgG/AcABwQHCAcMBxAHFAcYBxwHIAckBygHLAcwBzQHOAc8B0AHRAdIB0wHUAdUB1gHXAdgB2QHaAdsB3AHdAd4B3wHgAeEB4gHjAeQB5QHmAecB6AHpAeoB6wHsAe0B7gHvAfAB8QHyAfMB9AH1AfYB9wH4AfkB+gH7AfwB/QH+Af8CAAIBAgICAwIEAgUCBgIHAggCCQIKAgsCDAINAg4CDwIQAhECEgZnbHlwaDEGZ2x5cGgyB3VuaTAwQTAHdW5pMjAwMAd1bmkyMDAxB3VuaTIwMDIHdW5pMjAwMwd1bmkyMDA0B3VuaTIwMDUHdW5pMjAwNgd1bmkyMDA3B3VuaTIwMDgHdW5pMjAwOQd1bmkyMDBBB3VuaTIwMkYHdW5pMjA1RgRFdXJvB3VuaTIwQkQHdW5pMjMxQgd1bmkyNUZDB3VuaTI2MDEHdW5pMjZGQQd1bmkyNzA5B3VuaTI3MEYHdW5pRTAwMQd1bmlFMDAyB3VuaUUwMDMHdW5pRTAwNQd1bmlFMDA2B3VuaUUwMDcHdW5pRTAwOAd1bmlFMDA5B3VuaUUwMTAHdW5pRTAxMQd1bmlFMDEyB3VuaUUwMTMHdW5pRTAxNAd1bmlFMDE1B3VuaUUwMTYHdW5pRTAxNwd1bmlFMDE4B3VuaUUwMTkHdW5pRTAyMAd1bmlFMDIxB3VuaUUwMjIHdW5pRTAyMwd1bmlFMDI0B3VuaUUwMjUHdW5pRTAyNgd1bmlFMDI3B3VuaUUwMjgHdW5pRTAyOQd1bmlFMDMwB3VuaUUwMzEHdW5pRTAzMgd1bmlFMDMzB3VuaUUwMzQHdW5pRTAzNQd1bmlFMDM2B3VuaUUwMzcHdW5pRTAzOAd1bmlFMDM5B3VuaUUwNDAHdW5pRTA0MQd1bmlFMDQyB3VuaUUwNDMHdW5pRTA0NAd1bmlFMDQ1B3VuaUUwNDYHdW5pRTA0Nwd1bmlFMDQ4B3VuaUUwNDkHdW5pRTA1MAd1bmlFMDUxB3VuaUUwNTIHdW5pRTA1Mwd1bmlFMDU0B3VuaUUwNTUHdW5pRTA1Ngd1bmlFMDU3B3VuaUUwNTgHdW5pRTA1OQd1bmlFMDYwB3VuaUUwNjIHdW5pRTA2Mwd1bmlFMDY0B3VuaUUwNjUHdW5pRTA2Ngd1bmlFMDY3B3VuaUUwNjgHdW5pRTA2OQd1bmlFMDcwB3VuaUUwNzEHdW5pRTA3Mgd1bmlFMDczB3VuaUUwNzQHdW5pRTA3NQd1bmlFMDc2B3VuaUUwNzcHdW5pRTA3OAd1bmlFMDc5B3VuaUUwODAHdW5pRTA4MQd1bmlFMDgyB3VuaUUwODMHdW5pRTA4NAd1bmlFMDg1B3VuaUUwODYHdW5pRTA4Nwd1bmlFMDg4B3VuaUUwODkHdW5pRTA5MAd1bmlFMDkxB3VuaUUwOTIHdW5pRTA5Mwd1bmlFMDk0B3VuaUUwOTUHdW5pRTA5Ngd1bmlFMDk3B3VuaUUxMDEHdW5pRTEwMgd1bmlFMTAzB3VuaUUxMDQHdW5pRTEwNQd1bmlFMTA2B3VuaUUxMDcHdW5pRTEwOAd1bmlFMTA5B3VuaUUxMTAHdW5pRTExMQd1bmlFMTEyB3VuaUUxMTMHdW5pRTExNAd1bmlFMTE1B3VuaUUxMTYHdW5pRTExNwd1bmlFMTE4B3VuaUUxMTkHdW5pRTEyMAd1bmlFMTIxB3VuaUUxMjIHdW5pRTEyMwd1bmlFMTI0B3VuaUUxMjUHdW5pRTEyNgd1bmlFMTI3B3VuaUUxMjgHdW5pRTEyOQd1bmlFMTMwB3VuaUUxMzEHdW5pRTEzMgd1bmlFMTMzB3VuaUUxMzQHdW5pRTEzNQd1bmlFMTM2B3VuaUUxMzcHdW5pRTEzOAd1bmlFMTM5B3VuaUUxNDAHdW5pRTE0MQd1bmlFMTQyB3VuaUUxNDMHdW5pRTE0NAd1bmlFMTQ1B3VuaUUxNDYHdW5pRTE0OAd1bmlFMTQ5B3VuaUUxNTAHdW5pRTE1MQd1bmlFMTUyB3VuaUUxNTMHdW5pRTE1NAd1bmlFMTU1B3VuaUUxNTYHdW5pRTE1Nwd1bmlFMTU4B3VuaUUxNTkHdW5pRTE2MAd1bmlFMTYxB3VuaUUxNjIHdW5pRTE2Mwd1bmlFMTY0B3VuaUUxNjUHdW5pRTE2Ngd1bmlFMTY3B3VuaUUxNjgHdW5pRTE2OQd1bmlFMTcwB3VuaUUxNzEHdW5pRTE3Mgd1bmlFMTczB3VuaUUxNzQHdW5pRTE3NQd1bmlFMTc2B3VuaUUxNzcHdW5pRTE3OAd1bmlFMTc5B3VuaUUxODAHdW5pRTE4MQd1bmlFMTgyB3VuaUUxODMHdW5pRTE4NAd1bmlFMTg1B3VuaUUxODYHdW5pRTE4Nwd1bmlFMTg4B3VuaUUxODkHdW5pRTE5MAd1bmlFMTkxB3VuaUUxOTIHdW5pRTE5Mwd1bmlFMTk0B3VuaUUxOTUHdW5pRTE5Nwd1bmlFMTk4B3VuaUUxOTkHdW5pRTIwMAd1bmlFMjAxB3VuaUUyMDIHdW5pRTIwMwd1bmlFMjA0B3VuaUUyMDUHdW5pRTIwNgd1bmlFMjA5B3VuaUUyMTAHdW5pRTIxMQd1bmlFMjEyB3VuaUUyMTMHdW5pRTIxNAd1bmlFMjE1B3VuaUUyMTYHdW5pRTIxOAd1bmlFMjE5B3VuaUUyMjEHdW5pRTIyMwd1bmlFMjI0B3VuaUUyMjUHdW5pRTIyNgd1bmlFMjI3B3VuaUUyMzAHdW5pRTIzMQd1bmlFMjMyB3VuaUUyMzMHdW5pRTIzNAd1bmlFMjM1B3VuaUUyMzYHdW5pRTIzNwd1bmlFMjM4B3VuaUUyMzkHdW5pRTI0MAd1bmlFMjQxB3VuaUUyNDIHdW5pRTI0Mwd1bmlFMjQ0B3VuaUUyNDUHdW5pRTI0Ngd1bmlFMjQ3B3VuaUUyNDgHdW5pRTI0OQd1bmlFMjUwB3VuaUUyNTEHdW5pRTI1Mgd1bmlFMjUzB3VuaUUyNTQHdW5pRTI1NQd1bmlFMjU2B3VuaUUyNTcHdW5pRTI1OAd1bmlFMjU5B3VuaUUyNjAHdW5pRjhGRgZ1MUY1MTEGdTFGNkFBAAAAAAFUUMMXAAA=) format("truetype"),url() format("svg")}.glyphicon{position:relative;top:1px;display:inline-block;font-family:"Glyphicons Halflings";font-style:normal;font-weight:400;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.glyphicon-asterisk:before{content:"\002a"}.glyphicon-plus:before{content:"\002b"}.glyphicon-euro:before,.glyphicon-eur:before{content:"\20ac"}.glyphicon-minus:before{content:"\2212"}.glyphicon-cloud:before{content:"\2601"}.glyphicon-envelope:before{content:"\2709"}.glyphicon-pencil:before{content:"\270f"}.glyphicon-glass:before{content:"\e001"}.glyphicon-music:before{content:"\e002"}.glyphicon-search:before{content:"\e003"}.glyphicon-heart:before{content:"\e005"}.glyphicon-star:before{content:"\e006"}.glyphicon-star-empty:before{content:"\e007"}.glyphicon-user:before{content:"\e008"}.glyphicon-film:before{content:"\e009"}.glyphicon-th-large:before{content:"\e010"}.glyphicon-th:before{content:"\e011"}.glyphicon-th-list:before{content:"\e012"}.glyphicon-ok:before{content:"\e013"}.glyphicon-remove:before{content:"\e014"}.glyphicon-zoom-in:before{content:"\e015"}.glyphicon-zoom-out:before{content:"\e016"}.glyphicon-off:before{content:"\e017"}.glyphicon-signal:before{content:"\e018"}.glyphicon-cog:before{content:"\e019"}.glyphicon-trash:before{content:"\e020"}.glyphicon-home:before{content:"\e021"}.glyphicon-file:before{content:"\e022"}.glyphicon-time:before{content:"\e023"}.glyphicon-road:before{content:"\e024"}.glyphicon-download-alt:before{content:"\e025"}.glyphicon-download:before{content:"\e026"}.glyphicon-upload:before{content:"\e027"}.glyphicon-inbox:before{content:"\e028"}.glyphicon-play-circle:before{content:"\e029"}.glyphicon-repeat:before{content:"\e030"}.glyphicon-refresh:before{content:"\e031"}.glyphicon-list-alt:before{content:"\e032"}.glyphicon-lock:before{content:"\e033"}.glyphicon-flag:before{content:"\e034"}.glyphicon-headphones:before{content:"\e035"}.glyphicon-volume-off:before{content:"\e036"}.glyphicon-volume-down:before{content:"\e037"}.glyphicon-volume-up:before{content:"\e038"}.glyphicon-qrcode:before{content:"\e039"}.glyphicon-barcode:before{content:"\e040"}.glyphicon-tag:before{content:"\e041"}.glyphicon-tags:before{content:"\e042"}.glyphicon-book:before{content:"\e043"}.glyphicon-bookmark:before{content:"\e044"}.glyphicon-print:before{content:"\e045"}.glyphicon-camera:before{content:"\e046"}.glyphicon-font:before{content:"\e047"}.glyphicon-bold:before{content:"\e048"}.glyphicon-italic:before{content:"\e049"}.glyphicon-text-height:before{content:"\e050"}.glyphicon-text-width:before{content:"\e051"}.glyphicon-align-left:before{content:"\e052"}.glyphicon-align-center:before{content:"\e053"}.glyphicon-align-right:before{content:"\e054"}.glyphicon-align-justify:before{content:"\e055"}.glyphicon-list:before{content:"\e056"}.glyphicon-indent-left:before{content:"\e057"}.glyphicon-indent-right:before{content:"\e058"}.glyphicon-facetime-video:before{content:"\e059"}.glyphicon-picture:before{content:"\e060"}.glyphicon-map-marker:before{content:"\e062"}.glyphicon-adjust:before{content:"\e063"}.glyphicon-tint:before{content:"\e064"}.glyphicon-edit:before{content:"\e065"}.glyphicon-share:before{content:"\e066"}.glyphicon-check:before{content:"\e067"}.glyphicon-move:before{content:"\e068"}.glyphicon-step-backward:before{content:"\e069"}.glyphicon-fast-backward:before{content:"\e070"}.glyphicon-backward:before{content:"\e071"}.glyphicon-play:before{content:"\e072"}.glyphicon-pause:before{content:"\e073"}.glyphicon-stop:before{content:"\e074"}.glyphicon-forward:before{content:"\e075"}.glyphicon-fast-forward:before{content:"\e076"}.glyphicon-step-forward:before{content:"\e077"}.glyphicon-eject:before{content:"\e078"}.glyphicon-chevron-left:before{content:"\e079"}.glyphicon-chevron-right:before{content:"\e080"}.glyphicon-plus-sign:before{content:"\e081"}.glyphicon-minus-sign:before{content:"\e082"}.glyphicon-remove-sign:before{content:"\e083"}.glyphicon-ok-sign:before{content:"\e084"}.glyphicon-question-sign:before{content:"\e085"}.glyphicon-info-sign:before{content:"\e086"}.glyphicon-screenshot:before{content:"\e087"}.glyphicon-remove-circle:before{content:"\e088"}.glyphicon-ok-circle:before{content:"\e089"}.glyphicon-ban-circle:before{content:"\e090"}.glyphicon-arrow-left:before{content:"\e091"}.glyphicon-arrow-right:before{content:"\e092"}.glyphicon-arrow-up:before{content:"\e093"}.glyphicon-arrow-down:before{content:"\e094"}.glyphicon-share-alt:before{content:"\e095"}.glyphicon-resize-full:before{content:"\e096"}.glyphicon-resize-small:before{content:"\e097"}.glyphicon-exclamation-sign:before{content:"\e101"}.glyphicon-gift:before{content:"\e102"}.glyphicon-leaf:before{content:"\e103"}.glyphicon-fire:before{content:"\e104"}.glyphicon-eye-open:before{content:"\e105"}.glyphicon-eye-close:before{content:"\e106"}.glyphicon-warning-sign:before{content:"\e107"}.glyphicon-plane:before{content:"\e108"}.glyphicon-calendar:before{content:"\e109"}.glyphicon-random:before{content:"\e110"}.glyphicon-comment:before{content:"\e111"}.glyphicon-magnet:before{content:"\e112"}.glyphicon-chevron-up:before{content:"\e113"}.glyphicon-chevron-down:before{content:"\e114"}.glyphicon-retweet:before{content:"\e115"}.glyphicon-shopping-cart:before{content:"\e116"}.glyphicon-folder-close:before{content:"\e117"}.glyphicon-folder-open:before{content:"\e118"}.glyphicon-resize-vertical:before{content:"\e119"}.glyphicon-resize-horizontal:before{content:"\e120"}.glyphicon-hdd:before{content:"\e121"}.glyphicon-bullhorn:before{content:"\e122"}.glyphicon-bell:before{content:"\e123"}.glyphicon-certificate:before{content:"\e124"}.glyphicon-thumbs-up:before{content:"\e125"}.glyphicon-thumbs-down:before{content:"\e126"}.glyphicon-hand-right:before{content:"\e127"}.glyphicon-hand-left:before{content:"\e128"}.glyphicon-hand-up:before{content:"\e129"}.glyphicon-hand-down:before{content:"\e130"}.glyphicon-circle-arrow-right:before{content:"\e131"}.glyphicon-circle-arrow-left:before{content:"\e132"}.glyphicon-circle-arrow-up:before{content:"\e133"}.glyphicon-circle-arrow-down:before{content:"\e134"}.glyphicon-globe:before{content:"\e135"}.glyphicon-wrench:before{content:"\e136"}.glyphicon-tasks:before{content:"\e137"}.glyphicon-filter:before{content:"\e138"}.glyphicon-briefcase:before{content:"\e139"}.glyphicon-fullscreen:before{content:"\e140"}.glyphicon-dashboard:before{content:"\e141"}.glyphicon-paperclip:before{content:"\e142"}.glyphicon-heart-empty:before{content:"\e143"}.glyphicon-link:before{content:"\e144"}.glyphicon-phone:before{content:"\e145"}.glyphicon-pushpin:before{content:"\e146"}.glyphicon-usd:before{content:"\e148"}.glyphicon-gbp:before{content:"\e149"}.glyphicon-sort:before{content:"\e150"}.glyphicon-sort-by-alphabet:before{content:"\e151"}.glyphicon-sort-by-alphabet-alt:before{content:"\e152"}.glyphicon-sort-by-order:before{content:"\e153"}.glyphicon-sort-by-order-alt:before{content:"\e154"}.glyphicon-sort-by-attributes:before{content:"\e155"}.glyphicon-sort-by-attributes-alt:before{content:"\e156"}.glyphicon-unchecked:before{content:"\e157"}.glyphicon-expand:before{content:"\e158"}.glyphicon-collapse-down:before{content:"\e159"}.glyphicon-collapse-up:before{content:"\e160"}.glyphicon-log-in:before{content:"\e161"}.glyphicon-flash:before{content:"\e162"}.glyphicon-log-out:before{content:"\e163"}.glyphicon-new-window:before{content:"\e164"}.glyphicon-record:before{content:"\e165"}.glyphicon-save:before{content:"\e166"}.glyphicon-open:before{content:"\e167"}.glyphicon-saved:before{content:"\e168"}.glyphicon-import:before{content:"\e169"}.glyphicon-export:before{content:"\e170"}.glyphicon-send:before{content:"\e171"}.glyphicon-floppy-disk:before{content:"\e172"}.glyphicon-floppy-saved:before{content:"\e173"}.glyphicon-floppy-remove:before{content:"\e174"}.glyphicon-floppy-save:before{content:"\e175"}.glyphicon-floppy-open:before{content:"\e176"}.glyphicon-credit-card:before{content:"\e177"}.glyphicon-transfer:before{content:"\e178"}.glyphicon-cutlery:before{content:"\e179"}.glyphicon-header:before{content:"\e180"}.glyphicon-compressed:before{content:"\e181"}.glyphicon-earphone:before{content:"\e182"}.glyphicon-phone-alt:before{content:"\e183"}.glyphicon-tower:before{content:"\e184"}.glyphicon-stats:before{content:"\e185"}.glyphicon-sd-video:before{content:"\e186"}.glyphicon-hd-video:before{content:"\e187"}.glyphicon-subtitles:before{content:"\e188"}.glyphicon-sound-stereo:before{content:"\e189"}.glyphicon-sound-dolby:before{content:"\e190"}.glyphicon-sound-5-1:before{content:"\e191"}.glyphicon-sound-6-1:before{content:"\e192"}.glyphicon-sound-7-1:before{content:"\e193"}.glyphicon-copyright-mark:before{content:"\e194"}.glyphicon-registration-mark:before{content:"\e195"}.glyphicon-cloud-download:before{content:"\e197"}.glyphicon-cloud-upload:before{content:"\e198"}.glyphicon-tree-conifer:before{content:"\e199"}.glyphicon-tree-deciduous:before{content:"\e200"}.glyphicon-cd:before{content:"\e201"}.glyphicon-save-file:before{content:"\e202"}.glyphicon-open-file:before{content:"\e203"}.glyphicon-level-up:before{content:"\e204"}.glyphicon-copy:before{content:"\e205"}.glyphicon-paste:before{content:"\e206"}.glyphicon-alert:before{content:"\e209"}.glyphicon-equalizer:before{content:"\e210"}.glyphicon-king:before{content:"\e211"}.glyphicon-queen:before{content:"\e212"}.glyphicon-pawn:before{content:"\e213"}.glyphicon-bishop:before{content:"\e214"}.glyphicon-knight:before{content:"\e215"}.glyphicon-baby-formula:before{content:"\e216"}.glyphicon-tent:before{content:"\26fa"}.glyphicon-blackboard:before{content:"\e218"}.glyphicon-bed:before{content:"\e219"}.glyphicon-apple:before{content:"\f8ff"}.glyphicon-erase:before{content:"\e221"}.glyphicon-hourglass:before{content:"\231b"}.glyphicon-lamp:before{content:"\e223"}.glyphicon-duplicate:before{content:"\e224"}.glyphicon-piggy-bank:before{content:"\e225"}.glyphicon-scissors:before{content:"\e226"}.glyphicon-bitcoin:before{content:"\e227"}.glyphicon-btc:before{content:"\e227"}.glyphicon-xbt:before{content:"\e227"}.glyphicon-yen:before{content:"\00a5"}.glyphicon-jpy:before{content:"\00a5"}.glyphicon-ruble:before{content:"\20bd"}.glyphicon-rub:before{content:"\20bd"}.glyphicon-scale:before{content:"\e230"}.glyphicon-ice-lolly:before{content:"\e231"}.glyphicon-ice-lolly-tasted:before{content:"\e232"}.glyphicon-education:before{content:"\e233"}.glyphicon-option-horizontal:before{content:"\e234"}.glyphicon-option-vertical:before{content:"\e235"}.glyphicon-menu-hamburger:before{content:"\e236"}.glyphicon-modal-window:before{content:"\e237"}.glyphicon-oil:before{content:"\e238"}.glyphicon-grain:before{content:"\e239"}.glyphicon-sunglasses:before{content:"\e240"}.glyphicon-text-size:before{content:"\e241"}.glyphicon-text-color:before{content:"\e242"}.glyphicon-text-background:before{content:"\e243"}.glyphicon-object-align-top:before{content:"\e244"}.glyphicon-object-align-bottom:before{content:"\e245"}.glyphicon-object-align-horizontal:before{content:"\e246"}.glyphicon-object-align-left:before{content:"\e247"}.glyphicon-object-align-vertical:before{content:"\e248"}.glyphicon-object-align-right:before{content:"\e249"}.glyphicon-triangle-right:before{content:"\e250"}.glyphicon-triangle-left:before{content:"\e251"}.glyphicon-triangle-bottom:before{content:"\e252"}.glyphicon-triangle-top:before{content:"\e253"}.glyphicon-console:before{content:"\e254"}.glyphicon-superscript:before{content:"\e255"}.glyphicon-subscript:before{content:"\e256"}.glyphicon-menu-left:before{content:"\e257"}.glyphicon-menu-right:before{content:"\e258"}.glyphicon-menu-down:before{content:"\e259"}.glyphicon-menu-up:before{content:"\e260"}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}*:before,*:after{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}input,button,select,textarea{font-family:inherit;font-size:inherit;line-height:inherit}a{color:#337ab7;text-decoration:none}a:hover,a:focus{color:#23527c;text-decoration:underline}a:focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}figure{margin:0}img{vertical-align:middle}.img-responsive,.thumbnail>img,.thumbnail a>img,.carousel-inner>.item>img,.carousel-inner>.item>a>img{display:block;max-width:100%;height:auto}.img-rounded{border-radius:6px}.img-thumbnail{padding:4px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:all .2s ease-in-out;-o-transition:all .2s ease-in-out;transition:all .2s ease-in-out;display:inline-block;max-width:100%;height:auto}.img-circle{border-radius:50%}hr{margin-top:20px;margin-bottom:20px;border:0;border-top:1px solid #eee}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;margin:0;overflow:visible;clip:auto}[role="button"]{cursor:pointer}h1,h2,h3,h4,h5,h6,.h1,.h2,.h3,.h4,.h5,.h6{font-family:inherit;font-weight:500;line-height:1.1;color:inherit}h1 small,h2 small,h3 small,h4 small,h5 small,h6 small,.h1 small,.h2 small,.h3 small,.h4 small,.h5 small,.h6 small,h1 .small,h2 .small,h3 .small,h4 .small,h5 .small,h6 .small,.h1 .small,.h2 .small,.h3 .small,.h4 .small,.h5 .small,.h6 .small{font-weight:400;line-height:1;color:#777}h1,.h1,h2,.h2,h3,.h3{margin-top:20px;margin-bottom:10px}h1 small,.h1 small,h2 small,.h2 small,h3 small,.h3 small,h1 .small,.h1 .small,h2 .small,.h2 .small,h3 .small,.h3 .small{font-size:65%}h4,.h4,h5,.h5,h6,.h6{margin-top:10px;margin-bottom:10px}h4 small,.h4 small,h5 small,.h5 small,h6 small,.h6 small,h4 .small,.h4 .small,h5 .small,.h5 .small,h6 .small,.h6 .small{font-size:75%}h1,.h1{font-size:36px}h2,.h2{font-size:30px}h3,.h3{font-size:24px}h4,.h4{font-size:18px}h5,.h5{font-size:14px}h6,.h6{font-size:12px}p{margin:0 0 10px}.lead{margin-bottom:20px;font-size:16px;font-weight:300;line-height:1.4}@media(min-width:768px){.lead{font-size:21px}}small,.small{font-size:85%}mark,.mark{padding:.2em;background-color:#fcf8e3}.text-left{text-align:left}.text-right{text-align:right}.text-center{text-align:center}.text-justify{text-align:justify}.text-nowrap{white-space:nowrap}.text-lowercase{text-transform:lowercase}.text-uppercase{text-transform:uppercase}.text-capitalize{text-transform:capitalize}.text-muted{color:#777}.text-primary{color:#337ab7}a.text-primary:hover,a.text-primary:focus{color:#286090}.text-success{color:#3c763d}a.text-success:hover,a.text-success:focus{color:#2b542c}.text-info{color:#31708f}a.text-info:hover,a.text-info:focus{color:#245269}.text-warning{color:#8a6d3b}a.text-warning:hover,a.text-warning:focus{color:#66512c}.text-danger{color:#a94442}a.text-danger:hover,a.text-danger:focus{color:#843534}.bg-primary{color:#fff;background-color:#337ab7}a.bg-primary:hover,a.bg-primary:focus{background-color:#286090}.bg-success{background-color:#dff0d8}a.bg-success:hover,a.bg-success:focus{background-color:#c1e2b3}.bg-info{background-color:#d9edf7}a.bg-info:hover,a.bg-info:focus{background-color:#afd9ee}.bg-warning{background-color:#fcf8e3}a.bg-warning:hover,a.bg-warning:focus{background-color:#f7ecb5}.bg-danger{background-color:#f2dede}a.bg-danger:hover,a.bg-danger:focus{background-color:#e4b9b9}.page-header{padding-bottom:9px;margin:40px 0 20px;border-bottom:1px solid #eee}ul,ol{margin-top:0;margin-bottom:10px}ul ul,ol ul,ul ol,ol ol{margin-bottom:0}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none;margin-left:-5px}.list-inline>li{display:inline-block;padding-right:5px;padding-left:5px}dl{margin-top:0;margin-bottom:20px}dt,dd{line-height:1.42857143}dt{font-weight:700}dd{margin-left:0}@media(min-width:768px){.dl-horizontal dt{float:left;width:160px;clear:left;text-align:right;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.dl-horizontal dd{margin-left:180px}}abbr[title],abbr[data-original-title]{cursor:help}.initialism{font-size:90%;text-transform:uppercase}blockquote{padding:10px 20px;margin:0 0 20px;font-size:17.5px;border-left:5px solid #eee}blockquote p:last-child,blockquote ul:last-child,blockquote ol:last-child{margin-bottom:0}blockquote footer,blockquote small,blockquote .small{display:block;font-size:80%;line-height:1.42857143;color:#777}blockquote footer:before,blockquote small:before,blockquote .small:before{content:"\2014 \00A0"}.blockquote-reverse,blockquote.pull-right{padding-right:15px;padding-left:0;text-align:right;border-right:5px solid #eee;border-left:0}.blockquote-reverse footer:before,blockquote.pull-right footer:before,.blockquote-reverse small:before,blockquote.pull-right small:before,.blockquote-reverse .small:before,blockquote.pull-right .small:before{content:""}.blockquote-reverse footer:after,blockquote.pull-right footer:after,.blockquote-reverse small:after,blockquote.pull-right small:after,.blockquote-reverse .small:after,blockquote.pull-right .small:after{content:"\00A0 \2014"}address{margin-bottom:20px;font-style:normal;line-height:1.42857143}code,kbd,pre,samp{font-family:Menlo,Monaco,Consolas,"Courier New",monospace}code{padding:2px 4px;font-size:90%;color:#c7254e;background-color:#f9f2f4;border-radius:4px}kbd{padding:2px 4px;font-size:90%;color:#fff;background-color:#333;border-radius:3px;box-shadow:inset 0 -1px 0 rgba(0,0,0,0.25)}kbd kbd{padding:0;font-size:100%;font-weight:700;box-shadow:none}pre{display:block;padding:9.5px;margin:0 0 10px;font-size:13px;line-height:1.42857143;color:#333;word-break:break-all;word-wrap:break-word;background-color:#f5f5f5;border:1px solid #ccc;border-radius:4px}pre code{padding:0;font-size:inherit;color:inherit;white-space:pre-wrap;background-color:transparent;border-radius:0}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media(min-width:768px){.container{width:750px}}@media(min-width:992px){.container{width:970px}}@media(min-width:1200px){.container{width:1170px}}.container-fluid{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{margin-right:-15px;margin-left:-15px}.row-no-gutters{margin-right:0;margin-left:0}.row-no-gutters [class*="col-"]{padding-right:0;padding-left:0}.col-xs-1,.col-sm-1,.col-md-1,.col-lg-1,.col-xs-2,.col-sm-2,.col-md-2,.col-lg-2,.col-xs-3,.col-sm-3,.col-md-3,.col-lg-3,.col-xs-4,.col-sm-4,.col-md-4,.col-lg-4,.col-xs-5,.col-sm-5,.col-md-5,.col-lg-5,.col-xs-6,.col-sm-6,.col-md-6,.col-lg-6,.col-xs-7,.col-sm-7,.col-md-7,.col-lg-7,.col-xs-8,.col-sm-8,.col-md-8,.col-lg-8,.col-xs-9,.col-sm-9,.col-md-9,.col-lg-9,.col-xs-10,.col-sm-10,.col-md-10,.col-lg-10,.col-xs-11,.col-sm-11,.col-md-11,.col-lg-11,.col-xs-12,.col-sm-12,.col-md-12,.col-lg-12{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-1,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9,.col-xs-10,.col-xs-11,.col-xs-12{float:left}.col-xs-12{width:100%}.col-xs-11{width:91.66666667%}.col-xs-10{width:83.33333333%}.col-xs-9{width:75%}.col-xs-8{width:66.66666667%}.col-xs-7{width:58.33333333%}.col-xs-6{width:50%}.col-xs-5{width:41.66666667%}.col-xs-4{width:33.33333333%}.col-xs-3{width:25%}.col-xs-2{width:16.66666667%}.col-xs-1{width:8.33333333%}.col-xs-pull-12{right:100%}.col-xs-pull-11{right:91.66666667%}.col-xs-pull-10{right:83.33333333%}.col-xs-pull-9{right:75%}.col-xs-pull-8{right:66.66666667%}.col-xs-pull-7{right:58.33333333%}.col-xs-pull-6{right:50%}.col-xs-pull-5{right:41.66666667%}.col-xs-pull-4{right:33.33333333%}.col-xs-pull-3{right:25%}.col-xs-pull-2{right:16.66666667%}.col-xs-pull-1{right:8.33333333%}.col-xs-pull-0{right:auto}.col-xs-push-12{left:100%}.col-xs-push-11{left:91.66666667%}.col-xs-push-10{left:83.33333333%}.col-xs-push-9{left:75%}.col-xs-push-8{left:66.66666667%}.col-xs-push-7{left:58.33333333%}.col-xs-push-6{left:50%}.col-xs-push-5{left:41.66666667%}.col-xs-push-4{left:33.33333333%}.col-xs-push-3{left:25%}.col-xs-push-2{left:16.66666667%}.col-xs-push-1{left:8.33333333%}.col-xs-push-0{left:auto}.col-xs-offset-12{margin-left:100%}.col-xs-offset-11{margin-left:91.66666667%}.col-xs-offset-10{margin-left:83.33333333%}.col-xs-offset-9{margin-left:75%}.col-xs-offset-8{margin-left:66.66666667%}.col-xs-offset-7{margin-left:58.33333333%}.col-xs-offset-6{margin-left:50%}.col-xs-offset-5{margin-left:41.66666667%}.col-xs-offset-4{margin-left:33.33333333%}.col-xs-offset-3{margin-left:25%}.col-xs-offset-2{margin-left:16.66666667%}.col-xs-offset-1{margin-left:8.33333333%}.col-xs-offset-0{margin-left:0}@media(min-width:768px){.col-sm-1,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-10,.col-sm-11,.col-sm-12{float:left}.col-sm-12{width:100%}.col-sm-11{width:91.66666667%}.col-sm-10{width:83.33333333%}.col-sm-9{width:75%}.col-sm-8{width:66.66666667%}.col-sm-7{width:58.33333333%}.col-sm-6{width:50%}.col-sm-5{width:41.66666667%}.col-sm-4{width:33.33333333%}.col-sm-3{width:25%}.col-sm-2{width:16.66666667%}.col-sm-1{width:8.33333333%}.col-sm-pull-12{right:100%}.col-sm-pull-11{right:91.66666667%}.col-sm-pull-10{right:83.33333333%}.col-sm-pull-9{right:75%}.col-sm-pull-8{right:66.66666667%}.col-sm-pull-7{right:58.33333333%}.col-sm-pull-6{right:50%}.col-sm-pull-5{right:41.66666667%}.col-sm-pull-4{right:33.33333333%}.col-sm-pull-3{right:25%}.col-sm-pull-2{right:16.66666667%}.col-sm-pull-1{right:8.33333333%}.col-sm-pull-0{right:auto}.col-sm-push-12{left:100%}.col-sm-push-11{left:91.66666667%}.col-sm-push-10{left:83.33333333%}.col-sm-push-9{left:75%}.col-sm-push-8{left:66.66666667%}.col-sm-push-7{left:58.33333333%}.col-sm-push-6{left:50%}.col-sm-push-5{left:41.66666667%}.col-sm-push-4{left:33.33333333%}.col-sm-push-3{left:25%}.col-sm-push-2{left:16.66666667%}.col-sm-push-1{left:8.33333333%}.col-sm-push-0{left:auto}.col-sm-offset-12{margin-left:100%}.col-sm-offset-11{margin-left:91.66666667%}.col-sm-offset-10{margin-left:83.33333333%}.col-sm-offset-9{margin-left:75%}.col-sm-offset-8{margin-left:66.66666667%}.col-sm-offset-7{margin-left:58.33333333%}.col-sm-offset-6{margin-left:50%}.col-sm-offset-5{margin-left:41.66666667%}.col-sm-offset-4{margin-left:33.33333333%}.col-sm-offset-3{margin-left:25%}.col-sm-offset-2{margin-left:16.66666667%}.col-sm-offset-1{margin-left:8.33333333%}.col-sm-offset-0{margin-left:0}}@media(min-width:992px){.col-md-1,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-10,.col-md-11,.col-md-12{float:left}.col-md-12{width:100%}.col-md-11{width:91.66666667%}.col-md-10{width:83.33333333%}.col-md-9{width:75%}.col-md-8{width:66.66666667%}.col-md-7{width:58.33333333%}.col-md-6{width:50%}.col-md-5{width:41.66666667%}.col-md-4{width:33.33333333%}.col-md-3{width:25%}.col-md-2{width:16.66666667%}.col-md-1{width:8.33333333%}.col-md-pull-12{right:100%}.col-md-pull-11{right:91.66666667%}.col-md-pull-10{right:83.33333333%}.col-md-pull-9{right:75%}.col-md-pull-8{right:66.66666667%}.col-md-pull-7{right:58.33333333%}.col-md-pull-6{right:50%}.col-md-pull-5{right:41.66666667%}.col-md-pull-4{right:33.33333333%}.col-md-pull-3{right:25%}.col-md-pull-2{right:16.66666667%}.col-md-pull-1{right:8.33333333%}.col-md-pull-0{right:auto}.col-md-push-12{left:100%}.col-md-push-11{left:91.66666667%}.col-md-push-10{left:83.33333333%}.col-md-push-9{left:75%}.col-md-push-8{left:66.66666667%}.col-md-push-7{left:58.33333333%}.col-md-push-6{left:50%}.col-md-push-5{left:41.66666667%}.col-md-push-4{left:33.33333333%}.col-md-push-3{left:25%}.col-md-push-2{left:16.66666667%}.col-md-push-1{left:8.33333333%}.col-md-push-0{left:auto}.col-md-offset-12{margin-left:100%}.col-md-offset-11{margin-left:91.66666667%}.col-md-offset-10{margin-left:83.33333333%}.col-md-offset-9{margin-left:75%}.col-md-offset-8{margin-left:66.66666667%}.col-md-offset-7{margin-left:58.33333333%}.col-md-offset-6{margin-left:50%}.col-md-offset-5{margin-left:41.66666667%}.col-md-offset-4{margin-left:33.33333333%}.col-md-offset-3{margin-left:25%}.col-md-offset-2{margin-left:16.66666667%}.col-md-offset-1{margin-left:8.33333333%}.col-md-offset-0{margin-left:0}}@media(min-width:1200px){.col-lg-1,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-10,.col-lg-11,.col-lg-12{float:left}.col-lg-12{width:100%}.col-lg-11{width:91.66666667%}.col-lg-10{width:83.33333333%}.col-lg-9{width:75%}.col-lg-8{width:66.66666667%}.col-lg-7{width:58.33333333%}.col-lg-6{width:50%}.col-lg-5{width:41.66666667%}.col-lg-4{width:33.33333333%}.col-lg-3{width:25%}.col-lg-2{width:16.66666667%}.col-lg-1{width:8.33333333%}.col-lg-pull-12{right:100%}.col-lg-pull-11{right:91.66666667%}.col-lg-pull-10{right:83.33333333%}.col-lg-pull-9{right:75%}.col-lg-pull-8{right:66.66666667%}.col-lg-pull-7{right:58.33333333%}.col-lg-pull-6{right:50%}.col-lg-pull-5{right:41.66666667%}.col-lg-pull-4{right:33.33333333%}.col-lg-pull-3{right:25%}.col-lg-pull-2{right:16.66666667%}.col-lg-pull-1{right:8.33333333%}.col-lg-pull-0{right:auto}.col-lg-push-12{left:100%}.col-lg-push-11{left:91.66666667%}.col-lg-push-10{left:83.33333333%}.col-lg-push-9{left:75%}.col-lg-push-8{left:66.66666667%}.col-lg-push-7{left:58.33333333%}.col-lg-push-6{left:50%}.col-lg-push-5{left:41.66666667%}.col-lg-push-4{left:33.33333333%}.col-lg-push-3{left:25%}.col-lg-push-2{left:16.66666667%}.col-lg-push-1{left:8.33333333%}.col-lg-push-0{left:auto}.col-lg-offset-12{margin-left:100%}.col-lg-offset-11{margin-left:91.66666667%}.col-lg-offset-10{margin-left:83.33333333%}.col-lg-offset-9{margin-left:75%}.col-lg-offset-8{margin-left:66.66666667%}.col-lg-offset-7{margin-left:58.33333333%}.col-lg-offset-6{margin-left:50%}.col-lg-offset-5{margin-left:41.66666667%}.col-lg-offset-4{margin-left:33.33333333%}.col-lg-offset-3{margin-left:25%}.col-lg-offset-2{margin-left:16.66666667%}.col-lg-offset-1{margin-left:8.33333333%}.col-lg-offset-0{margin-left:0}}table{background-color:transparent}table col[class*="col-"]{position:static;display:table-column;float:none}table td[class*="col-"],table th[class*="col-"]{position:static;display:table-cell;float:none}caption{padding-top:8px;padding-bottom:8px;color:#777;text-align:left}th{text-align:left}.table{width:100%;max-width:100%;margin-bottom:20px}.table>thead>tr>th,.table>tbody>tr>th,.table>tfoot>tr>th,.table>thead>tr>td,.table>tbody>tr>td,.table>tfoot>tr>td{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table>thead>tr>th{vertical-align:bottom;border-bottom:2px solid #ddd}.table>caption+thead>tr:first-child>th,.table>colgroup+thead>tr:first-child>th,.table>thead:first-child>tr:first-child>th,.table>caption+thead>tr:first-child>td,.table>colgroup+thead>tr:first-child>td,.table>thead:first-child>tr:first-child>td{border-top:0}.table>tbody+tbody{border-top:2px solid #ddd}.table .table{background-color:#fff}.table-condensed>thead>tr>th,.table-condensed>tbody>tr>th,.table-condensed>tfoot>tr>th,.table-condensed>thead>tr>td,.table-condensed>tbody>tr>td,.table-condensed>tfoot>tr>td{padding:5px}.table-bordered{border:1px solid #ddd}.table-bordered>thead>tr>th,.table-bordered>tbody>tr>th,.table-bordered>tfoot>tr>th,.table-bordered>thead>tr>td,.table-bordered>tbody>tr>td,.table-bordered>tfoot>tr>td{border:1px solid #ddd}.table-bordered>thead>tr>th,.table-bordered>thead>tr>td{border-bottom-width:2px}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.table-hover>tbody>tr:hover{background-color:#f5f5f5}.table>thead>tr>td.active,.table>tbody>tr>td.active,.table>tfoot>tr>td.active,.table>thead>tr>th.active,.table>tbody>tr>th.active,.table>tfoot>tr>th.active,.table>thead>tr.active>td,.table>tbody>tr.active>td,.table>tfoot>tr.active>td,.table>thead>tr.active>th,.table>tbody>tr.active>th,.table>tfoot>tr.active>th{background-color:#f5f5f5}.table-hover>tbody>tr>td.active:hover,.table-hover>tbody>tr>th.active:hover,.table-hover>tbody>tr.active:hover>td,.table-hover>tbody>tr:hover>.active,.table-hover>tbody>tr.active:hover>th{background-color:#e8e8e8}.table>thead>tr>td.success,.table>tbody>tr>td.success,.table>tfoot>tr>td.success,.table>thead>tr>th.success,.table>tbody>tr>th.success,.table>tfoot>tr>th.success,.table>thead>tr.success>td,.table>tbody>tr.success>td,.table>tfoot>tr.success>td,.table>thead>tr.success>th,.table>tbody>tr.success>th,.table>tfoot>tr.success>th{background-color:#dff0d8}.table-hover>tbody>tr>td.success:hover,.table-hover>tbody>tr>th.success:hover,.table-hover>tbody>tr.success:hover>td,.table-hover>tbody>tr:hover>.success,.table-hover>tbody>tr.success:hover>th{background-color:#d0e9c6}.table>thead>tr>td.info,.table>tbody>tr>td.info,.table>tfoot>tr>td.info,.table>thead>tr>th.info,.table>tbody>tr>th.info,.table>tfoot>tr>th.info,.table>thead>tr.info>td,.table>tbody>tr.info>td,.table>tfoot>tr.info>td,.table>thead>tr.info>th,.table>tbody>tr.info>th,.table>tfoot>tr.info>th{background-color:#d9edf7}.table-hover>tbody>tr>td.info:hover,.table-hover>tbody>tr>th.info:hover,.table-hover>tbody>tr.info:hover>td,.table-hover>tbody>tr:hover>.info,.table-hover>tbody>tr.info:hover>th{background-color:#c4e3f3}.table>thead>tr>td.warning,.table>tbody>tr>td.warning,.table>tfoot>tr>td.warning,.table>thead>tr>th.warning,.table>tbody>tr>th.warning,.table>tfoot>tr>th.warning,.table>thead>tr.warning>td,.table>tbody>tr.warning>td,.table>tfoot>tr.warning>td,.table>thead>tr.warning>th,.table>tbody>tr.warning>th,.table>tfoot>tr.warning>th{background-color:#fcf8e3}.table-hover>tbody>tr>td.warning:hover,.table-hover>tbody>tr>th.warning:hover,.table-hover>tbody>tr.warning:hover>td,.table-hover>tbody>tr:hover>.warning,.table-hover>tbody>tr.warning:hover>th{background-color:#faf2cc}.table>thead>tr>td.danger,.table>tbody>tr>td.danger,.table>tfoot>tr>td.danger,.table>thead>tr>th.danger,.table>tbody>tr>th.danger,.table>tfoot>tr>th.danger,.table>thead>tr.danger>td,.table>tbody>tr.danger>td,.table>tfoot>tr.danger>td,.table>thead>tr.danger>th,.table>tbody>tr.danger>th,.table>tfoot>tr.danger>th{background-color:#f2dede}.table-hover>tbody>tr>td.danger:hover,.table-hover>tbody>tr>th.danger:hover,.table-hover>tbody>tr.danger:hover>td,.table-hover>tbody>tr:hover>.danger,.table-hover>tbody>tr.danger:hover>th{background-color:#ebcccc}.table-responsive{min-height:.01%;overflow-x:auto}@media screen and (max-width:767px){.table-responsive{width:100%;margin-bottom:15px;overflow-y:hidden;-ms-overflow-style:-ms-autohiding-scrollbar;border:1px solid #ddd}.table-responsive>.table{margin-bottom:0}.table-responsive>.table>thead>tr>th,.table-responsive>.table>tbody>tr>th,.table-responsive>.table>tfoot>tr>th,.table-responsive>.table>thead>tr>td,.table-responsive>.table>tbody>tr>td,.table-responsive>.table>tfoot>tr>td{white-space:nowrap}.table-responsive>.table-bordered{border:0}.table-responsive>.table-bordered>thead>tr>th:first-child,.table-responsive>.table-bordered>tbody>tr>th:first-child,.table-responsive>.table-bordered>tfoot>tr>th:first-child,.table-responsive>.table-bordered>thead>tr>td:first-child,.table-responsive>.table-bordered>tbody>tr>td:first-child,.table-responsive>.table-bordered>tfoot>tr>td:first-child{border-left:0}.table-responsive>.table-bordered>thead>tr>th:last-child,.table-responsive>.table-bordered>tbody>tr>th:last-child,.table-responsive>.table-bordered>tfoot>tr>th:last-child,.table-responsive>.table-bordered>thead>tr>td:last-child,.table-responsive>.table-bordered>tbody>tr>td:last-child,.table-responsive>.table-bordered>tfoot>tr>td:last-child{border-right:0}.table-responsive>.table-bordered>tbody>tr:last-child>th,.table-responsive>.table-bordered>tfoot>tr:last-child>th,.table-responsive>.table-bordered>tbody>tr:last-child>td,.table-responsive>.table-bordered>tfoot>tr:last-child>td{border-bottom:0}}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;padding:0;margin-bottom:20px;font-size:21px;line-height:inherit;color:#333;border:0;border-bottom:1px solid #e5e5e5}label{display:inline-block;max-width:100%;margin-bottom:5px;font-weight:700}input[type="search"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;-webkit-appearance:none;appearance:none}input[type="radio"],input[type="checkbox"]{margin:4px 0 0;margin-top:1px \9;line-height:normal}input[type="radio"][disabled],input[type="checkbox"][disabled],input[type="radio"].disabled,input[type="checkbox"].disabled,fieldset[disabled] input[type="radio"],fieldset[disabled] input[type="checkbox"]{cursor:not-allowed}input[type="file"]{display:block}input[type="range"]{display:block;width:100%}select[multiple],select[size]{height:auto}input[type="file"]:focus,input[type="radio"]:focus,input[type="checkbox"]:focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}output{display:block;padding-top:7px;font-size:14px;line-height:1.42857143;color:#555}.form-control{display:block;width:100%;height:34px;padding:6px 12px;font-size:14px;line-height:1.42857143;color:#555;background-color:#fff;background-image:none;border:1px solid #ccc;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);-webkit-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;-o-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s}.form-control:focus{border-color:#66afe9;outline:0;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,0.6);box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,0.6)}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.form-control::-webkit-input-placeholder{color:#999}.form-control::-ms-expand{background-color:transparent;border:0}.form-control[disabled],.form-control[readonly],fieldset[disabled] .form-control{background-color:#eee;opacity:1}.form-control[disabled],fieldset[disabled] .form-control{cursor:not-allowed}textarea.form-control{height:auto}@media screen and (-webkit-min-device-pixel-ratio:0){input[type="date"].form-control,input[type="time"].form-control,input[type="datetime-local"].form-control,input[type="month"].form-control{line-height:34px}input[type="date"].input-sm,input[type="time"].input-sm,input[type="datetime-local"].input-sm,input[type="month"].input-sm,.input-group-sm input[type="date"],.input-group-sm input[type="time"],.input-group-sm input[type="datetime-local"],.input-group-sm input[type="month"]{line-height:30px}input[type="date"].input-lg,input[type="time"].input-lg,input[type="datetime-local"].input-lg,input[type="month"].input-lg,.input-group-lg input[type="date"],.input-group-lg input[type="time"],.input-group-lg input[type="datetime-local"],.input-group-lg input[type="month"]{line-height:46px}}.form-group{margin-bottom:15px}.radio,.checkbox{position:relative;display:block;margin-top:10px;margin-bottom:10px}.radio.disabled label,.checkbox.disabled label,fieldset[disabled] .radio label,fieldset[disabled] .checkbox label{cursor:not-allowed}.radio label,.checkbox label{min-height:20px;padding-left:20px;margin-bottom:0;font-weight:400;cursor:pointer}.radio input[type="radio"],.radio-inline input[type="radio"],.checkbox input[type="checkbox"],.checkbox-inline input[type="checkbox"]{position:absolute;margin-top:4px \9;margin-left:-20px}.radio+.radio,.checkbox+.checkbox{margin-top:-5px}.radio-inline,.checkbox-inline{position:relative;display:inline-block;padding-left:20px;margin-bottom:0;font-weight:400;vertical-align:middle;cursor:pointer}.radio-inline.disabled,.checkbox-inline.disabled,fieldset[disabled] .radio-inline,fieldset[disabled] .checkbox-inline{cursor:not-allowed}.radio-inline+.radio-inline,.checkbox-inline+.checkbox-inline{margin-top:0;margin-left:10px}.form-control-static{min-height:34px;padding-top:7px;padding-bottom:7px;margin-bottom:0}.form-control-static.input-lg,.form-control-static.input-sm{padding-right:0;padding-left:0}.input-sm{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-sm{height:30px;line-height:30px}textarea.input-sm,select[multiple].input-sm{height:auto}.form-group-sm .form-control{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.form-group-sm select.form-control{height:30px;line-height:30px}.form-group-sm textarea.form-control,.form-group-sm select[multiple].form-control{height:auto}.form-group-sm .form-control-static{height:30px;min-height:32px;padding:6px 10px;font-size:12px;line-height:1.5}.input-lg{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-lg{height:46px;line-height:46px}textarea.input-lg,select[multiple].input-lg{height:auto}.form-group-lg .form-control{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.form-group-lg select.form-control{height:46px;line-height:46px}.form-group-lg textarea.form-control,.form-group-lg select[multiple].form-control{height:auto}.form-group-lg .form-control-static{height:46px;min-height:38px;padding:11px 16px;font-size:18px;line-height:1.3333333}.has-feedback{position:relative}.has-feedback .form-control{padding-right:42.5px}.form-control-feedback{position:absolute;top:0;right:0;z-index:2;display:block;width:34px;height:34px;line-height:34px;text-align:center;pointer-events:none}.input-lg+.form-control-feedback,.input-group-lg+.form-control-feedback,.form-group-lg .form-control+.form-control-feedback{width:46px;height:46px;line-height:46px}.input-sm+.form-control-feedback,.input-group-sm+.form-control-feedback,.form-group-sm .form-control+.form-control-feedback{width:30px;height:30px;line-height:30px}.has-success .help-block,.has-success .control-label,.has-success .radio,.has-success .checkbox,.has-success .radio-inline,.has-success .checkbox-inline,.has-success.radio label,.has-success.checkbox label,.has-success.radio-inline label,.has-success.checkbox-inline label{color:#3c763d}.has-success .form-control{border-color:#3c763d;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-success .form-control:focus{border-color:#2b542c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #67b168;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #67b168}.has-success .input-group-addon{color:#3c763d;background-color:#dff0d8;border-color:#3c763d}.has-success .form-control-feedback{color:#3c763d}.has-warning .help-block,.has-warning .control-label,.has-warning .radio,.has-warning .checkbox,.has-warning .radio-inline,.has-warning .checkbox-inline,.has-warning.radio label,.has-warning.checkbox label,.has-warning.radio-inline label,.has-warning.checkbox-inline label{color:#8a6d3b}.has-warning .form-control{border-color:#8a6d3b;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-warning .form-control:focus{border-color:#66512c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #c0a16b;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #c0a16b}.has-warning .input-group-addon{color:#8a6d3b;background-color:#fcf8e3;border-color:#8a6d3b}.has-warning .form-control-feedback{color:#8a6d3b}.has-error .help-block,.has-error .control-label,.has-error .radio,.has-error .checkbox,.has-error .radio-inline,.has-error .checkbox-inline,.has-error.radio label,.has-error.checkbox label,.has-error.radio-inline label,.has-error.checkbox-inline label{color:#a94442}.has-error .form-control{border-color:#a94442;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-error .form-control:focus{border-color:#843534;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #ce8483;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #ce8483}.has-error .input-group-addon{color:#a94442;background-color:#f2dede;border-color:#a94442}.has-error .form-control-feedback{color:#a94442}.has-feedback label ~ .form-control-feedback{top:25px}.has-feedback label.sr-only ~ .form-control-feedback{top:0}.help-block{display:block;margin-top:5px;margin-bottom:10px;color:#737373}@media(min-width:768px){.form-inline .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-static{display:inline-block}.form-inline .input-group{display:inline-table;vertical-align:middle}.form-inline .input-group .input-group-addon,.form-inline .input-group .input-group-btn,.form-inline .input-group .form-control{width:auto}.form-inline .input-group>.form-control{width:100%}.form-inline .control-label{margin-bottom:0;vertical-align:middle}.form-inline .radio,.form-inline .checkbox{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.form-inline .radio label,.form-inline .checkbox label{padding-left:0}.form-inline .radio input[type="radio"],.form-inline .checkbox input[type="checkbox"]{position:relative;margin-left:0}.form-inline .has-feedback .form-control-feedback{top:0}}.form-horizontal .radio,.form-horizontal .checkbox,.form-horizontal .radio-inline,.form-horizontal .checkbox-inline{padding-top:7px;margin-top:0;margin-bottom:0}.form-horizontal .radio,.form-horizontal .checkbox{min-height:27px}.form-horizontal .form-group{margin-right:-15px;margin-left:-15px}@media(min-width:768px){.form-horizontal .control-label{padding-top:7px;margin-bottom:0;text-align:right}}.form-horizontal .has-feedback .form-control-feedback{right:15px}@media(min-width:768px){.form-horizontal .form-group-lg .control-label{padding-top:11px;font-size:18px}}@media(min-width:768px){.form-horizontal .form-group-sm .control-label{padding-top:6px;font-size:12px}}.btn{display:inline-block;margin-bottom:0;font-weight:normal;text-align:center;white-space:nowrap;vertical-align:middle;touch-action:manipulation;cursor:pointer;background-image:none;border:1px solid transparent;padding:6px 12px;font-size:14px;line-height:1.42857143;border-radius:4px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.btn:focus,.btn:active:focus,.btn.active:focus,.btn.focus,.btn:active.focus,.btn.active.focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.btn:hover,.btn:focus,.btn.focus{color:#333;text-decoration:none}.btn:active,.btn.active{background-image:none;outline:0;-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,0.125);box-shadow:inset 0 3px 5px rgba(0,0,0,0.125)}.btn.disabled,.btn[disabled],fieldset[disabled] .btn{cursor:not-allowed;filter:alpha(opacity=65);opacity:.65;-webkit-box-shadow:none;box-shadow:none}a.btn.disabled,fieldset[disabled] a.btn{pointer-events:none}.btn-default{color:#333;background-color:#fff;border-color:#ccc}.btn-default:focus,.btn-default.focus{color:#333;background-color:#e6e6e6;border-color:#8c8c8c}.btn-default:hover{color:#333;background-color:#e6e6e6;border-color:#adadad}.btn-default:active,.btn-default.active,.open>.dropdown-toggle.btn-default{color:#333;background-color:#e6e6e6;background-image:none;border-color:#adadad}.btn-default:active:hover,.btn-default.active:hover,.open>.dropdown-toggle.btn-default:hover,.btn-default:active:focus,.btn-default.active:focus,.open>.dropdown-toggle.btn-default:focus,.btn-default:active.focus,.btn-default.active.focus,.open>.dropdown-toggle.btn-default.focus{color:#333;background-color:#d4d4d4;border-color:#8c8c8c}.btn-default.disabled:hover,.btn-default[disabled]:hover,fieldset[disabled] .btn-default:hover,.btn-default.disabled:focus,.btn-default[disabled]:focus,fieldset[disabled] .btn-default:focus,.btn-default.disabled.focus,.btn-default[disabled].focus,fieldset[disabled] .btn-default.focus{background-color:#fff;border-color:#ccc}.btn-default .badge{color:#fff;background-color:#333}.btn-primary{color:#fff;background-color:#337ab7;border-color:#2e6da4}.btn-primary:focus,.btn-primary.focus{color:#fff;background-color:#286090;border-color:#122b40}.btn-primary:hover{color:#fff;background-color:#286090;border-color:#204d74}.btn-primary:active,.btn-primary.active,.open>.dropdown-toggle.btn-primary{color:#fff;background-color:#286090;background-image:none;border-color:#204d74}.btn-primary:active:hover,.btn-primary.active:hover,.open>.dropdown-toggle.btn-primary:hover,.btn-primary:active:focus,.btn-primary.active:focus,.open>.dropdown-toggle.btn-primary:focus,.btn-primary:active.focus,.btn-primary.active.focus,.open>.dropdown-toggle.btn-primary.focus{color:#fff;background-color:#204d74;border-color:#122b40}.btn-primary.disabled:hover,.btn-primary[disabled]:hover,fieldset[disabled] .btn-primary:hover,.btn-primary.disabled:focus,.btn-primary[disabled]:focus,fieldset[disabled] .btn-primary:focus,.btn-primary.disabled.focus,.btn-primary[disabled].focus,fieldset[disabled] .btn-primary.focus{background-color:#337ab7;border-color:#2e6da4}.btn-primary .badge{color:#337ab7;background-color:#fff}.btn-success{color:#fff;background-color:#5cb85c;border-color:#4cae4c}.btn-success:focus,.btn-success.focus{color:#fff;background-color:#449d44;border-color:#255625}.btn-success:hover{color:#fff;background-color:#449d44;border-color:#398439}.btn-success:active,.btn-success.active,.open>.dropdown-toggle.btn-success{color:#fff;background-color:#449d44;background-image:none;border-color:#398439}.btn-success:active:hover,.btn-success.active:hover,.open>.dropdown-toggle.btn-success:hover,.btn-success:active:focus,.btn-success.active:focus,.open>.dropdown-toggle.btn-success:focus,.btn-success:active.focus,.btn-success.active.focus,.open>.dropdown-toggle.btn-success.focus{color:#fff;background-color:#398439;border-color:#255625}.btn-success.disabled:hover,.btn-success[disabled]:hover,fieldset[disabled] .btn-success:hover,.btn-success.disabled:focus,.btn-success[disabled]:focus,fieldset[disabled] .btn-success:focus,.btn-success.disabled.focus,.btn-success[disabled].focus,fieldset[disabled] .btn-success.focus{background-color:#5cb85c;border-color:#4cae4c}.btn-success .badge{color:#5cb85c;background-color:#fff}.btn-info{color:#fff;background-color:#5bc0de;border-color:#46b8da}.btn-info:focus,.btn-info.focus{color:#fff;background-color:#31b0d5;border-color:#1b6d85}.btn-info:hover{color:#fff;background-color:#31b0d5;border-color:#269abc}.btn-info:active,.btn-info.active,.open>.dropdown-toggle.btn-info{color:#fff;background-color:#31b0d5;background-image:none;border-color:#269abc}.btn-info:active:hover,.btn-info.active:hover,.open>.dropdown-toggle.btn-info:hover,.btn-info:active:focus,.btn-info.active:focus,.open>.dropdown-toggle.btn-info:focus,.btn-info:active.focus,.btn-info.active.focus,.open>.dropdown-toggle.btn-info.focus{color:#fff;background-color:#269abc;border-color:#1b6d85}.btn-info.disabled:hover,.btn-info[disabled]:hover,fieldset[disabled] .btn-info:hover,.btn-info.disabled:focus,.btn-info[disabled]:focus,fieldset[disabled] .btn-info:focus,.btn-info.disabled.focus,.btn-info[disabled].focus,fieldset[disabled] .btn-info.focus{background-color:#5bc0de;border-color:#46b8da}.btn-info .badge{color:#5bc0de;background-color:#fff}.btn-warning{color:#fff;background-color:#f0ad4e;border-color:#eea236}.btn-warning:focus,.btn-warning.focus{color:#fff;background-color:#ec971f;border-color:#985f0d}.btn-warning:hover{color:#fff;background-color:#ec971f;border-color:#d58512}.btn-warning:active,.btn-warning.active,.open>.dropdown-toggle.btn-warning{color:#fff;background-color:#ec971f;background-image:none;border-color:#d58512}.btn-warning:active:hover,.btn-warning.active:hover,.open>.dropdown-toggle.btn-warning:hover,.btn-warning:active:focus,.btn-warning.active:focus,.open>.dropdown-toggle.btn-warning:focus,.btn-warning:active.focus,.btn-warning.active.focus,.open>.dropdown-toggle.btn-warning.focus{color:#fff;background-color:#d58512;border-color:#985f0d}.btn-warning.disabled:hover,.btn-warning[disabled]:hover,fieldset[disabled] .btn-warning:hover,.btn-warning.disabled:focus,.btn-warning[disabled]:focus,fieldset[disabled] .btn-warning:focus,.btn-warning.disabled.focus,.btn-warning[disabled].focus,fieldset[disabled] .btn-warning.focus{background-color:#f0ad4e;border-color:#eea236}.btn-warning .badge{color:#f0ad4e;background-color:#fff}.btn-danger{color:#fff;background-color:#d9534f;border-color:#d43f3a}.btn-danger:focus,.btn-danger.focus{color:#fff;background-color:#c9302c;border-color:#761c19}.btn-danger:hover{color:#fff;background-color:#c9302c;border-color:#ac2925}.btn-danger:active,.btn-danger.active,.open>.dropdown-toggle.btn-danger{color:#fff;background-color:#c9302c;background-image:none;border-color:#ac2925}.btn-danger:active:hover,.btn-danger.active:hover,.open>.dropdown-toggle.btn-danger:hover,.btn-danger:active:focus,.btn-danger.active:focus,.open>.dropdown-toggle.btn-danger:focus,.btn-danger:active.focus,.btn-danger.active.focus,.open>.dropdown-toggle.btn-danger.focus{color:#fff;background-color:#ac2925;border-color:#761c19}.btn-danger.disabled:hover,.btn-danger[disabled]:hover,fieldset[disabled] .btn-danger:hover,.btn-danger.disabled:focus,.btn-danger[disabled]:focus,fieldset[disabled] .btn-danger:focus,.btn-danger.disabled.focus,.btn-danger[disabled].focus,fieldset[disabled] .btn-danger.focus{background-color:#d9534f;border-color:#d43f3a}.btn-danger .badge{color:#d9534f;background-color:#fff}.btn-link{font-weight:400;color:#337ab7;border-radius:0}.btn-link,.btn-link:active,.btn-link.active,.btn-link[disabled],fieldset[disabled] .btn-link{background-color:transparent;-webkit-box-shadow:none;box-shadow:none}.btn-link,.btn-link:hover,.btn-link:focus,.btn-link:active{border-color:transparent}.btn-link:hover,.btn-link:focus{color:#23527c;text-decoration:underline;background-color:transparent}.btn-link[disabled]:hover,fieldset[disabled] .btn-link:hover,.btn-link[disabled]:focus,fieldset[disabled] .btn-link:focus{color:#777;text-decoration:none}.btn-lg,.btn-group-lg>.btn{padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.btn-sm,.btn-group-sm>.btn{padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.btn-xs,.btn-group-xs>.btn{padding:1px 5px;font-size:12px;line-height:1.5;border-radius:3px}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:5px}input[type="submit"].btn-block,input[type="reset"].btn-block,input[type="button"].btn-block{width:100%}.fade{opacity:0;-webkit-transition:opacity .15s linear;-o-transition:opacity .15s linear;transition:opacity .15s linear}.fade.in{opacity:1}.collapse{display:none}.collapse.in{display:block}tr.collapse.in{display:table-row}tbody.collapse.in{display:table-row-group}.collapsing{position:relative;height:0;overflow:hidden;-webkit-transition-property:height,visibility;transition-property:height,visibility;-webkit-transition-duration:.35s;transition-duration:.35s;-webkit-transition-timing-function:ease;transition-timing-function:ease}.caret{display:inline-block;width:0;height:0;margin-left:2px;vertical-align:middle;border-top:4px dashed;border-top:4px solid \9;border-right:4px solid transparent;border-left:4px solid transparent}.dropup,.dropdown{position:relative}.dropdown-toggle:focus{outline:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:160px;padding:5px 0;margin:2px 0 0;font-size:14px;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,0.15);border-radius:4px;-webkit-box-shadow:0 6px 12px rgba(0,0,0,0.175);box-shadow:0 6px 12px rgba(0,0,0,0.175)}.dropdown-menu.pull-right{right:0;left:auto}.dropdown-menu .divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.dropdown-menu>li>a{display:block;padding:3px 20px;clear:both;font-weight:400;line-height:1.42857143;color:#333;white-space:nowrap}.dropdown-menu>li>a:hover,.dropdown-menu>li>a:focus{color:#262626;text-decoration:none;background-color:#f5f5f5}.dropdown-menu>.active>a,.dropdown-menu>.active>a:hover,.dropdown-menu>.active>a:focus{color:#fff;text-decoration:none;background-color:#337ab7;outline:0}.dropdown-menu>.disabled>a,.dropdown-menu>.disabled>a:hover,.dropdown-menu>.disabled>a:focus{color:#777}.dropdown-menu>.disabled>a:hover,.dropdown-menu>.disabled>a:focus{text-decoration:none;cursor:not-allowed;background-color:transparent;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false)}.open>.dropdown-menu{display:block}.open>a{outline:0}.dropdown-menu-right{right:0;left:auto}.dropdown-menu-left{right:auto;left:0}.dropdown-header{display:block;padding:3px 20px;font-size:12px;line-height:1.42857143;color:#777;white-space:nowrap}.dropdown-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:990}.pull-right>.dropdown-menu{right:0;left:auto}.dropup .caret,.navbar-fixed-bottom .dropdown .caret{content:"";border-top:0;border-bottom:4px dashed;border-bottom:4px solid \9}.dropup .dropdown-menu,.navbar-fixed-bottom .dropdown .dropdown-menu{top:auto;bottom:100%;margin-bottom:2px}@media(min-width:768px){.navbar-right .dropdown-menu{right:0;left:auto}.navbar-right .dropdown-menu-left{right:auto;left:0}}.btn-group,.btn-group-vertical{position:relative;display:inline-block;vertical-align:middle}.btn-group>.btn,.btn-group-vertical>.btn{position:relative;float:left}.btn-group>.btn:hover,.btn-group-vertical>.btn:hover,.btn-group>.btn:focus,.btn-group-vertical>.btn:focus,.btn-group>.btn:active,.btn-group-vertical>.btn:active,.btn-group>.btn.active,.btn-group-vertical>.btn.active{z-index:2}.btn-group .btn+.btn,.btn-group .btn+.btn-group,.btn-group .btn-group+.btn,.btn-group .btn-group+.btn-group{margin-left:-1px}.btn-toolbar{margin-left:-5px}.btn-toolbar .btn,.btn-toolbar .btn-group,.btn-toolbar .input-group{float:left}.btn-toolbar>.btn,.btn-toolbar>.btn-group,.btn-toolbar>.input-group{margin-left:5px}.btn-group>.btn:not(:first-child):not(:last-child):not(.dropdown-toggle){border-radius:0}.btn-group>.btn:first-child{margin-left:0}.btn-group>.btn:first-child:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn:last-child:not(:first-child),.btn-group>.dropdown-toggle:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.btn-group>.btn-group{float:left}.btn-group>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-bottom-left-radius:0}.btn-group .dropdown-toggle:active,.btn-group.open .dropdown-toggle{outline:0}.btn-group>.btn+.dropdown-toggle{padding-right:8px;padding-left:8px}.btn-group>.btn-lg+.dropdown-toggle{padding-right:12px;padding-left:12px}.btn-group.open .dropdown-toggle{-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,0.125);box-shadow:inset 0 3px 5px rgba(0,0,0,0.125)}.btn-group.open .dropdown-toggle.btn-link{-webkit-box-shadow:none;box-shadow:none}.btn .caret{margin-left:0}.btn-lg .caret{border-width:5px 5px 0;border-bottom-width:0}.dropup .btn-lg .caret{border-width:0 5px 5px}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group,.btn-group-vertical>.btn-group>.btn{display:block;float:none;width:100%;max-width:100%}.btn-group-vertical>.btn-group>.btn{float:none}.btn-group-vertical>.btn+.btn,.btn-group-vertical>.btn+.btn-group,.btn-group-vertical>.btn-group+.btn,.btn-group-vertical>.btn-group+.btn-group{margin-top:-1px;margin-left:0}.btn-group-vertical>.btn:not(:first-child):not(:last-child){border-radius:0}.btn-group-vertical>.btn:first-child:not(:last-child){border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn:last-child:not(:first-child){border-top-left-radius:0;border-top-right-radius:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}.btn-group-vertical>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group-vertical>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group-vertical>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-top-right-radius:0}.btn-group-justified{display:table;width:100%;table-layout:fixed;border-collapse:separate}.btn-group-justified>.btn,.btn-group-justified>.btn-group{display:table-cell;float:none;width:1%}.btn-group-justified>.btn-group .btn{width:100%}.btn-group-justified>.btn-group .dropdown-menu{left:auto}[data-toggle="buttons"]>.btn input[type="radio"],[data-toggle="buttons"]>.btn-group>.btn input[type="radio"],[data-toggle="buttons"]>.btn input[type="checkbox"],[data-toggle="buttons"]>.btn-group>.btn input[type="checkbox"]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:table;border-collapse:separate}.input-group[class*="col-"]{float:none;padding-right:0;padding-left:0}.input-group .form-control{position:relative;z-index:2;float:left;width:100%;margin-bottom:0}.input-group .form-control:focus{z-index:3}.input-group-lg>.form-control,.input-group-lg>.input-group-addon,.input-group-lg>.input-group-btn>.btn{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-group-lg>.form-control,select.input-group-lg>.input-group-addon,select.input-group-lg>.input-group-btn>.btn{height:46px;line-height:46px}textarea.input-group-lg>.form-control,textarea.input-group-lg>.input-group-addon,textarea.input-group-lg>.input-group-btn>.btn,select[multiple].input-group-lg>.form-control,select[multiple].input-group-lg>.input-group-addon,select[multiple].input-group-lg>.input-group-btn>.btn{height:auto}.input-group-sm>.form-control,.input-group-sm>.input-group-addon,.input-group-sm>.input-group-btn>.btn{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-group-sm>.form-control,select.input-group-sm>.input-group-addon,select.input-group-sm>.input-group-btn>.btn{height:30px;line-height:30px}textarea.input-group-sm>.form-control,textarea.input-group-sm>.input-group-addon,textarea.input-group-sm>.input-group-btn>.btn,select[multiple].input-group-sm>.form-control,select[multiple].input-group-sm>.input-group-addon,select[multiple].input-group-sm>.input-group-btn>.btn{height:auto}.input-group-addon,.input-group-btn,.input-group .form-control{display:table-cell}.input-group-addon:not(:first-child):not(:last-child),.input-group-btn:not(:first-child):not(:last-child),.input-group .form-control:not(:first-child):not(:last-child){border-radius:0}.input-group-addon,.input-group-btn{width:1%;white-space:nowrap;vertical-align:middle}.input-group-addon{padding:6px 12px;font-size:14px;font-weight:400;line-height:1;color:#555;text-align:center;background-color:#eee;border:1px solid #ccc;border-radius:4px}.input-group-addon.input-sm{padding:5px 10px;font-size:12px;border-radius:3px}.input-group-addon.input-lg{padding:10px 16px;font-size:18px;border-radius:6px}.input-group-addon input[type="radio"],.input-group-addon input[type="checkbox"]{margin-top:0}.input-group .form-control:first-child,.input-group-addon:first-child,.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group>.btn,.input-group-btn:first-child>.dropdown-toggle,.input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group-btn:last-child>.btn-group:not(:last-child)>.btn{border-top-right-radius:0;border-bottom-right-radius:0}.input-group-addon:first-child{border-right:0}.input-group .form-control:last-child,.input-group-addon:last-child,.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group>.btn,.input-group-btn:last-child>.dropdown-toggle,.input-group-btn:first-child>.btn:not(:first-child),.input-group-btn:first-child>.btn-group:not(:first-child)>.btn{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-addon:last-child{border-left:0}.input-group-btn{position:relative;font-size:0;white-space:nowrap}.input-group-btn>.btn{position:relative}.input-group-btn>.btn+.btn{margin-left:-1px}.input-group-btn>.btn:hover,.input-group-btn>.btn:focus,.input-group-btn>.btn:active{z-index:2}.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group{margin-right:-1px}.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group{z-index:2;margin-left:-1px}.nav{padding-left:0;margin-bottom:0;list-style:none}.nav>li{position:relative;display:block}.nav>li>a{position:relative;display:block;padding:10px 15px}.nav>li>a:hover,.nav>li>a:focus{text-decoration:none;background-color:#eee}.nav>li.disabled>a{color:#777}.nav>li.disabled>a:hover,.nav>li.disabled>a:focus{color:#777;text-decoration:none;cursor:not-allowed;background-color:transparent}.nav .open>a,.nav .open>a:hover,.nav .open>a:focus{background-color:#eee;border-color:#337ab7}.nav .nav-divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.nav>li>a>img{max-width:none}.nav-tabs{border-bottom:1px solid #ddd}.nav-tabs>li{float:left;margin-bottom:-1px}.nav-tabs>li>a{margin-right:2px;line-height:1.42857143;border:1px solid transparent;border-radius:4px 4px 0 0}.nav-tabs>li>a:hover{border-color:#eee #eee #ddd}.nav-tabs>li.active>a,.nav-tabs>li.active>a:hover,.nav-tabs>li.active>a:focus{color:#555;cursor:default;background-color:#fff;border:1px solid #ddd;border-bottom-color:transparent}.nav-tabs.nav-justified{width:100%;border-bottom:0}.nav-tabs.nav-justified>li{float:none}.nav-tabs.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-tabs.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media(min-width:768px){.nav-tabs.nav-justified>li{display:table-cell;width:1%}.nav-tabs.nav-justified>li>a{margin-bottom:0}}.nav-tabs.nav-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:hover,.nav-tabs.nav-justified>.active>a:focus{border:1px solid #ddd}@media(min-width:768px){.nav-tabs.nav-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:hover,.nav-tabs.nav-justified>.active>a:focus{border-bottom-color:#fff}}.nav-pills>li{float:left}.nav-pills>li>a{border-radius:4px}.nav-pills>li+li{margin-left:2px}.nav-pills>li.active>a,.nav-pills>li.active>a:hover,.nav-pills>li.active>a:focus{color:#fff;background-color:#337ab7}.nav-stacked>li{float:none}.nav-stacked>li+li{margin-top:2px;margin-left:0}.nav-justified{width:100%}.nav-justified>li{float:none}.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media(min-width:768px){.nav-justified>li{display:table-cell;width:1%}.nav-justified>li>a{margin-bottom:0}}.nav-tabs-justified{border-bottom:0}.nav-tabs-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:hover,.nav-tabs-justified>.active>a:focus{border:1px solid #ddd}@media(min-width:768px){.nav-tabs-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:hover,.nav-tabs-justified>.active>a:focus{border-bottom-color:#fff}}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.navbar{position:relative;min-height:50px;margin-bottom:20px;border:1px solid transparent}@media(min-width:768px){.navbar{border-radius:4px}}@media(min-width:768px){.navbar-header{float:left}}.navbar-collapse{padding-right:15px;padding-left:15px;overflow-x:visible;border-top:1px solid transparent;box-shadow:inset 0 1px 0 rgba(255,255,255,0.1);-webkit-overflow-scrolling:touch}.navbar-collapse.in{overflow-y:auto}@media(min-width:768px){.navbar-collapse{width:auto;border-top:0;box-shadow:none}.navbar-collapse.collapse{display:block!important;height:auto!important;padding-bottom:0;overflow:visible!important}.navbar-collapse.in{overflow-y:visible}.navbar-fixed-top .navbar-collapse,.navbar-static-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{padding-right:0;padding-left:0}}.navbar-fixed-top,.navbar-fixed-bottom{position:fixed;right:0;left:0;z-index:1030}.navbar-fixed-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{max-height:340px}@media(max-device-width:480px) and (orientation:landscape){.navbar-fixed-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{max-height:200px}}@media(min-width:768px){.navbar-fixed-top,.navbar-fixed-bottom{border-radius:0}}.navbar-fixed-top{top:0;border-width:0 0 1px}.navbar-fixed-bottom{bottom:0;margin-bottom:0;border-width:1px 0 0}.container>.navbar-header,.container-fluid>.navbar-header,.container>.navbar-collapse,.container-fluid>.navbar-collapse{margin-right:-15px;margin-left:-15px}@media(min-width:768px){.container>.navbar-header,.container-fluid>.navbar-header,.container>.navbar-collapse,.container-fluid>.navbar-collapse{margin-right:0;margin-left:0}}.navbar-static-top{z-index:1000;border-width:0 0 1px}@media(min-width:768px){.navbar-static-top{border-radius:0}}.navbar-brand{float:left;height:50px;padding:15px 15px;font-size:18px;line-height:20px}.navbar-brand:hover,.navbar-brand:focus{text-decoration:none}.navbar-brand>img{display:block}@media(min-width:768px){.navbar>.container .navbar-brand,.navbar>.container-fluid .navbar-brand{margin-left:-15px}}.navbar-toggle{position:relative;float:right;padding:9px 10px;margin-right:15px;margin-top:8px;margin-bottom:8px;background-color:transparent;background-image:none;border:1px solid transparent;border-radius:4px}.navbar-toggle:focus{outline:0}.navbar-toggle .icon-bar{display:block;width:22px;height:2px;border-radius:1px}.navbar-toggle .icon-bar+.icon-bar{margin-top:4px}@media(min-width:768px){.navbar-toggle{display:none}}.navbar-nav{margin:7.5px -15px}.navbar-nav>li>a{padding-top:10px;padding-bottom:10px;line-height:20px}@media(max-width:767px){.navbar-nav .open .dropdown-menu{position:static;float:none;width:auto;margin-top:0;background-color:transparent;border:0;box-shadow:none}.navbar-nav .open .dropdown-menu>li>a,.navbar-nav .open .dropdown-menu .dropdown-header{padding:5px 15px 5px 25px}.navbar-nav .open .dropdown-menu>li>a{line-height:20px}.navbar-nav .open .dropdown-menu>li>a:hover,.navbar-nav .open .dropdown-menu>li>a:focus{background-image:none}}@media(min-width:768px){.navbar-nav{float:left;margin:0}.navbar-nav>li{float:left}.navbar-nav>li>a{padding-top:15px;padding-bottom:15px}}.navbar-form{padding:10px 15px;margin-right:-15px;margin-left:-15px;border-top:1px solid transparent;border-bottom:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);margin-top:8px;margin-bottom:8px}@media(min-width:768px){.navbar-form .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.navbar-form .form-control{display:inline-block;width:auto;vertical-align:middle}.navbar-form .form-control-static{display:inline-block}.navbar-form .input-group{display:inline-table;vertical-align:middle}.navbar-form .input-group .input-group-addon,.navbar-form .input-group .input-group-btn,.navbar-form .input-group .form-control{width:auto}.navbar-form .input-group>.form-control{width:100%}.navbar-form .control-label{margin-bottom:0;vertical-align:middle}.navbar-form .radio,.navbar-form .checkbox{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.navbar-form .radio label,.navbar-form .checkbox label{padding-left:0}.navbar-form .radio input[type="radio"],.navbar-form .checkbox input[type="checkbox"]{position:relative;margin-left:0}.navbar-form .has-feedback .form-control-feedback{top:0}}@media(max-width:767px){.navbar-form .form-group{margin-bottom:5px}.navbar-form .form-group:last-child{margin-bottom:0}}@media(min-width:768px){.navbar-form{width:auto;padding-top:0;padding-bottom:0;margin-right:0;margin-left:0;border:0;-webkit-box-shadow:none;box-shadow:none}}.navbar-nav>li>.dropdown-menu{margin-top:0;border-top-left-radius:0;border-top-right-radius:0}.navbar-fixed-bottom .navbar-nav>li>.dropdown-menu{margin-bottom:0;border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.navbar-btn{margin-top:8px;margin-bottom:8px}.navbar-btn.btn-sm{margin-top:10px;margin-bottom:10px}.navbar-btn.btn-xs{margin-top:14px;margin-bottom:14px}.navbar-text{margin-top:15px;margin-bottom:15px}@media(min-width:768px){.navbar-text{float:left;margin-right:15px;margin-left:15px}}@media(min-width:768px){.navbar-left{float:left!important}.navbar-right{float:right!important;margin-right:-15px}.navbar-right ~ .navbar-right{margin-right:0}}.navbar-default{background-color:#f8f8f8;border-color:#e7e7e7}.navbar-default .navbar-brand{color:#777}.navbar-default .navbar-brand:hover,.navbar-default .navbar-brand:focus{color:#5e5e5e;background-color:transparent}.navbar-default .navbar-text{color:#777}.navbar-default .navbar-nav>li>a{color:#777}.navbar-default .navbar-nav>li>a:hover,.navbar-default .navbar-nav>li>a:focus{color:#333;background-color:transparent}.navbar-default .navbar-nav>.active>a,.navbar-default .navbar-nav>.active>a:hover,.navbar-default .navbar-nav>.active>a:focus{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav>.disabled>a,.navbar-default .navbar-nav>.disabled>a:hover,.navbar-default .navbar-nav>.disabled>a:focus{color:#ccc;background-color:transparent}.navbar-default .navbar-nav>.open>a,.navbar-default .navbar-nav>.open>a:hover,.navbar-default .navbar-nav>.open>a:focus{color:#555;background-color:#e7e7e7}@media(max-width:767px){.navbar-default .navbar-nav .open .dropdown-menu>li>a{color:#777}.navbar-default .navbar-nav .open .dropdown-menu>li>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>li>a:focus{color:#333;background-color:transparent}.navbar-default .navbar-nav .open .dropdown-menu>.active>a,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:focus{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:focus{color:#ccc;background-color:transparent}}.navbar-default .navbar-toggle{border-color:#ddd}.navbar-default .navbar-toggle:hover,.navbar-default .navbar-toggle:focus{background-color:#ddd}.navbar-default .navbar-toggle .icon-bar{background-color:#888}.navbar-default .navbar-collapse,.navbar-default .navbar-form{border-color:#e7e7e7}.navbar-default .navbar-link{color:#777}.navbar-default .navbar-link:hover{color:#333}.navbar-default .btn-link{color:#777}.navbar-default .btn-link:hover,.navbar-default .btn-link:focus{color:#333}.navbar-default .btn-link[disabled]:hover,fieldset[disabled] .navbar-default .btn-link:hover,.navbar-default .btn-link[disabled]:focus,fieldset[disabled] .navbar-default .btn-link:focus{color:#ccc}.navbar-inverse{background-color:#222;border-color:#080808}.navbar-inverse .navbar-brand{color:#9d9d9d}.navbar-inverse .navbar-brand:hover,.navbar-inverse .navbar-brand:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-text{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a:hover,.navbar-inverse .navbar-nav>li>a:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav>.active>a,.navbar-inverse .navbar-nav>.active>a:hover,.navbar-inverse .navbar-nav>.active>a:focus{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav>.disabled>a,.navbar-inverse .navbar-nav>.disabled>a:hover,.navbar-inverse .navbar-nav>.disabled>a:focus{color:#444;background-color:transparent}.navbar-inverse .navbar-nav>.open>a,.navbar-inverse .navbar-nav>.open>a:hover,.navbar-inverse .navbar-nav>.open>a:focus{color:#fff;background-color:#080808}@media(max-width:767px){.navbar-inverse .navbar-nav .open .dropdown-menu>.dropdown-header{border-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu .divider{background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:focus{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:focus{color:#444;background-color:transparent}}.navbar-inverse .navbar-toggle{border-color:#333}.navbar-inverse .navbar-toggle:hover,.navbar-inverse .navbar-toggle:focus{background-color:#333}.navbar-inverse .navbar-toggle .icon-bar{background-color:#fff}.navbar-inverse .navbar-collapse,.navbar-inverse .navbar-form{border-color:#101010}.navbar-inverse .navbar-link{color:#9d9d9d}.navbar-inverse .navbar-link:hover{color:#fff}.navbar-inverse .btn-link{color:#9d9d9d}.navbar-inverse .btn-link:hover,.navbar-inverse .btn-link:focus{color:#fff}.navbar-inverse .btn-link[disabled]:hover,fieldset[disabled] .navbar-inverse .btn-link:hover,.navbar-inverse .btn-link[disabled]:focus,fieldset[disabled] .navbar-inverse .btn-link:focus{color:#444}.breadcrumb{padding:8px 15px;margin-bottom:20px;list-style:none;background-color:#f5f5f5;border-radius:4px}.breadcrumb>li{display:inline-block}.breadcrumb>li+li:before{padding:0 5px;color:#ccc;content:"/\00a0"}.breadcrumb>.active{color:#777}.pagination{display:inline-block;padding-left:0;margin:20px 0;border-radius:4px}.pagination>li{display:inline}.pagination>li>a,.pagination>li>span{position:relative;float:left;padding:6px 12px;margin-left:-1px;line-height:1.42857143;color:#337ab7;text-decoration:none;background-color:#fff;border:1px solid #ddd}.pagination>li>a:hover,.pagination>li>span:hover,.pagination>li>a:focus,.pagination>li>span:focus{z-index:2;color:#23527c;background-color:#eee;border-color:#ddd}.pagination>li:first-child>a,.pagination>li:first-child>span{margin-left:0;border-top-left-radius:4px;border-bottom-left-radius:4px}.pagination>li:last-child>a,.pagination>li:last-child>span{border-top-right-radius:4px;border-bottom-right-radius:4px}.pagination>.active>a,.pagination>.active>span,.pagination>.active>a:hover,.pagination>.active>span:hover,.pagination>.active>a:focus,.pagination>.active>span:focus{z-index:3;color:#fff;cursor:default;background-color:#337ab7;border-color:#337ab7}.pagination>.disabled>span,.pagination>.disabled>span:hover,.pagination>.disabled>span:focus,.pagination>.disabled>a,.pagination>.disabled>a:hover,.pagination>.disabled>a:focus{color:#777;cursor:not-allowed;background-color:#fff;border-color:#ddd}.pagination-lg>li>a,.pagination-lg>li>span{padding:10px 16px;font-size:18px;line-height:1.3333333}.pagination-lg>li:first-child>a,.pagination-lg>li:first-child>span{border-top-left-radius:6px;border-bottom-left-radius:6px}.pagination-lg>li:last-child>a,.pagination-lg>li:last-child>span{border-top-right-radius:6px;border-bottom-right-radius:6px}.pagination-sm>li>a,.pagination-sm>li>span{padding:5px 10px;font-size:12px;line-height:1.5}.pagination-sm>li:first-child>a,.pagination-sm>li:first-child>span{border-top-left-radius:3px;border-bottom-left-radius:3px}.pagination-sm>li:last-child>a,.pagination-sm>li:last-child>span{border-top-right-radius:3px;border-bottom-right-radius:3px}.pager{padding-left:0;margin:20px 0;text-align:center;list-style:none}.pager li{display:inline}.pager li>a,.pager li>span{display:inline-block;padding:5px 14px;background-color:#fff;border:1px solid #ddd;border-radius:15px}.pager li>a:hover,.pager li>a:focus{text-decoration:none;background-color:#eee}.pager .next>a,.pager .next>span{float:right}.pager .previous>a,.pager .previous>span{float:left}.pager .disabled>a,.pager .disabled>a:hover,.pager .disabled>a:focus,.pager .disabled>span{color:#777;cursor:not-allowed;background-color:#fff}.label{display:inline;padding:.2em .6em .3em;font-size:75%;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25em}a.label:hover,a.label:focus{color:#fff;text-decoration:none;cursor:pointer}.label:empty{display:none}.btn .label{position:relative;top:-1px}.label-default{background-color:#777}.label-default[href]:hover,.label-default[href]:focus{background-color:#5e5e5e}.label-primary{background-color:#337ab7}.label-primary[href]:hover,.label-primary[href]:focus{background-color:#286090}.label-success{background-color:#5cb85c}.label-success[href]:hover,.label-success[href]:focus{background-color:#449d44}.label-info{background-color:#5bc0de}.label-info[href]:hover,.label-info[href]:focus{background-color:#31b0d5}.label-warning{background-color:#f0ad4e}.label-warning[href]:hover,.label-warning[href]:focus{background-color:#ec971f}.label-danger{background-color:#d9534f}.label-danger[href]:hover,.label-danger[href]:focus{background-color:#c9302c}.badge{display:inline-block;min-width:10px;padding:3px 7px;font-size:12px;font-weight:bold;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:middle;background-color:#777;border-radius:10px}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.btn-xs .badge,.btn-group-xs>.btn .badge{top:0;padding:1px 5px}a.badge:hover,a.badge:focus{color:#fff;text-decoration:none;cursor:pointer}.list-group-item.active>.badge,.nav-pills>.active>a>.badge{color:#337ab7;background-color:#fff}.list-group-item>.badge{float:right}.list-group-item>.badge+.badge{margin-right:5px}.nav-pills>li>a>.badge{margin-left:3px}.jumbotron{padding-top:30px;padding-bottom:30px;margin-bottom:30px;color:inherit;background-color:#eee}.jumbotron h1,.jumbotron .h1{color:inherit}.jumbotron p{margin-bottom:15px;font-size:21px;font-weight:200}.jumbotron>hr{border-top-color:#d5d5d5}.container .jumbotron,.container-fluid .jumbotron{padding-right:15px;padding-left:15px;border-radius:6px}.jumbotron .container{max-width:100%}@media screen and (min-width:768px){.jumbotron{padding-top:48px;padding-bottom:48px}.container .jumbotron,.container-fluid .jumbotron{padding-right:60px;padding-left:60px}.jumbotron h1,.jumbotron .h1{font-size:63px}}.thumbnail{display:block;padding:4px;margin-bottom:20px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:border .2s ease-in-out;-o-transition:border .2s ease-in-out;transition:border .2s ease-in-out}.thumbnail>img,.thumbnail a>img{margin-right:auto;margin-left:auto}a.thumbnail:hover,a.thumbnail:focus,a.thumbnail.active{border-color:#337ab7}.thumbnail .caption{padding:9px;color:#333}.alert{padding:15px;margin-bottom:20px;border:1px solid transparent;border-radius:4px}.alert h4{margin-top:0;color:inherit}.alert .alert-link{font-weight:bold}.alert>p,.alert>ul{margin-bottom:0}.alert>p+p{margin-top:5px}.alert-dismissable,.alert-dismissible{padding-right:35px}.alert-dismissable .close,.alert-dismissible .close{position:relative;top:-2px;right:-21px;color:inherit}.alert-success{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.alert-success hr{border-top-color:#c9e2b3}.alert-success .alert-link{color:#2b542c}.alert-info{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.alert-info hr{border-top-color:#a6e1ec}.alert-info .alert-link{color:#245269}.alert-warning{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.alert-warning hr{border-top-color:#f7e1b5}.alert-warning .alert-link{color:#66512c}.alert-danger{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.alert-danger hr{border-top-color:#e4b9c0}.alert-danger .alert-link{color:#843534}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}.progress{height:20px;margin-bottom:20px;overflow:hidden;background-color:#f5f5f5;border-radius:4px;-webkit-box-shadow:inset 0 1px 2px rgba(0,0,0,0.1);box-shadow:inset 0 1px 2px rgba(0,0,0,0.1)}.progress-bar{float:left;width:0;height:100%;font-size:12px;line-height:20px;color:#fff;text-align:center;background-color:#337ab7;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,0.15);box-shadow:inset 0 -1px 0 rgba(0,0,0,0.15);-webkit-transition:width .6s ease;-o-transition:width .6s ease;transition:width .6s ease}.progress-striped .progress-bar,.progress-bar-striped{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-size:40px 40px}.progress.active .progress-bar,.progress-bar.active{-webkit-animation:progress-bar-stripes 2s linear infinite;-o-animation:progress-bar-stripes 2s linear infinite;animation:progress-bar-stripes 2s linear infinite}.progress-bar-success{background-color:#5cb85c}.progress-striped .progress-bar-success{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-info{background-color:#5bc0de}.progress-striped .progress-bar-info{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-warning{background-color:#f0ad4e}.progress-striped .progress-bar-warning{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-danger{background-color:#d9534f}.progress-striped .progress-bar-danger{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.media{margin-top:15px}.media:first-child{margin-top:0}.media,.media-body{overflow:hidden;zoom:1}.media-body{width:10000px}.media-object{display:block}.media-object.img-thumbnail{max-width:none}.media-right,.media>.pull-right{padding-left:10px}.media-left,.media>.pull-left{padding-right:10px}.media-left,.media-right,.media-body{display:table-cell;vertical-align:top}.media-middle{vertical-align:middle}.media-bottom{vertical-align:bottom}.media-heading{margin-top:0;margin-bottom:5px}.media-list{padding-left:0;list-style:none}.list-group{padding-left:0;margin-bottom:20px}.list-group-item{position:relative;display:block;padding:10px 15px;margin-bottom:-1px;background-color:#fff;border:1px solid #ddd}.list-group-item:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}.list-group-item.disabled,.list-group-item.disabled:hover,.list-group-item.disabled:focus{color:#777;cursor:not-allowed;background-color:#eee}.list-group-item.disabled .list-group-item-heading,.list-group-item.disabled:hover .list-group-item-heading,.list-group-item.disabled:focus .list-group-item-heading{color:inherit}.list-group-item.disabled .list-group-item-text,.list-group-item.disabled:hover .list-group-item-text,.list-group-item.disabled:focus .list-group-item-text{color:#777}.list-group-item.active,.list-group-item.active:hover,.list-group-item.active:focus{z-index:2;color:#fff;background-color:#337ab7;border-color:#337ab7}.list-group-item.active .list-group-item-heading,.list-group-item.active:hover .list-group-item-heading,.list-group-item.active:focus .list-group-item-heading,.list-group-item.active .list-group-item-heading>small,.list-group-item.active:hover .list-group-item-heading>small,.list-group-item.active:focus .list-group-item-heading>small,.list-group-item.active .list-group-item-heading>.small,.list-group-item.active:hover .list-group-item-heading>.small,.list-group-item.active:focus .list-group-item-heading>.small{color:inherit}.list-group-item.active .list-group-item-text,.list-group-item.active:hover .list-group-item-text,.list-group-item.active:focus .list-group-item-text{color:#c7ddef}a.list-group-item,button.list-group-item{color:#555}a.list-group-item .list-group-item-heading,button.list-group-item .list-group-item-heading{color:#333}a.list-group-item:hover,button.list-group-item:hover,a.list-group-item:focus,button.list-group-item:focus{color:#555;text-decoration:none;background-color:#f5f5f5}button.list-group-item{width:100%;text-align:left}.list-group-item-success{color:#3c763d;background-color:#dff0d8}a.list-group-item-success,button.list-group-item-success{color:#3c763d}a.list-group-item-success .list-group-item-heading,button.list-group-item-success .list-group-item-heading{color:inherit}a.list-group-item-success:hover,button.list-group-item-success:hover,a.list-group-item-success:focus,button.list-group-item-success:focus{color:#3c763d;background-color:#d0e9c6}a.list-group-item-success.active,button.list-group-item-success.active,a.list-group-item-success.active:hover,button.list-group-item-success.active:hover,a.list-group-item-success.active:focus,button.list-group-item-success.active:focus{color:#fff;background-color:#3c763d;border-color:#3c763d}.list-group-item-info{color:#31708f;background-color:#d9edf7}a.list-group-item-info,button.list-group-item-info{color:#31708f}a.list-group-item-info .list-group-item-heading,button.list-group-item-info .list-group-item-heading{color:inherit}a.list-group-item-info:hover,button.list-group-item-info:hover,a.list-group-item-info:focus,button.list-group-item-info:focus{color:#31708f;background-color:#c4e3f3}a.list-group-item-info.active,button.list-group-item-info.active,a.list-group-item-info.active:hover,button.list-group-item-info.active:hover,a.list-group-item-info.active:focus,button.list-group-item-info.active:focus{color:#fff;background-color:#31708f;border-color:#31708f}.list-group-item-warning{color:#8a6d3b;background-color:#fcf8e3}a.list-group-item-warning,button.list-group-item-warning{color:#8a6d3b}a.list-group-item-warning .list-group-item-heading,button.list-group-item-warning .list-group-item-heading{color:inherit}a.list-group-item-warning:hover,button.list-group-item-warning:hover,a.list-group-item-warning:focus,button.list-group-item-warning:focus{color:#8a6d3b;background-color:#faf2cc}a.list-group-item-warning.active,button.list-group-item-warning.active,a.list-group-item-warning.active:hover,button.list-group-item-warning.active:hover,a.list-group-item-warning.active:focus,button.list-group-item-warning.active:focus{color:#fff;background-color:#8a6d3b;border-color:#8a6d3b}.list-group-item-danger{color:#a94442;background-color:#f2dede}a.list-group-item-danger,button.list-group-item-danger{color:#a94442}a.list-group-item-danger .list-group-item-heading,button.list-group-item-danger .list-group-item-heading{color:inherit}a.list-group-item-danger:hover,button.list-group-item-danger:hover,a.list-group-item-danger:focus,button.list-group-item-danger:focus{color:#a94442;background-color:#ebcccc}a.list-group-item-danger.active,button.list-group-item-danger.active,a.list-group-item-danger.active:hover,button.list-group-item-danger.active:hover,a.list-group-item-danger.active:focus,button.list-group-item-danger.active:focus{color:#fff;background-color:#a94442;border-color:#a94442}.list-group-item-heading{margin-top:0;margin-bottom:5px}.list-group-item-text{margin-bottom:0;line-height:1.3}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,0.05);box-shadow:0 1px 1px rgba(0,0,0,0.05)}.panel-body{padding:15px}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel-heading>.dropdown .dropdown-toggle{color:inherit}.panel-title{margin-top:0;margin-bottom:0;font-size:16px;color:inherit}.panel-title>a,.panel-title>small,.panel-title>.small,.panel-title>small>a,.panel-title>.small>a{color:inherit}.panel-footer{padding:10px 15px;background-color:#f5f5f5;border-top:1px solid #ddd;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.list-group,.panel>.panel-collapse>.list-group{margin-bottom:0}.panel>.list-group .list-group-item,.panel>.panel-collapse>.list-group .list-group-item{border-width:1px 0;border-radius:0}.panel>.list-group:first-child .list-group-item:first-child,.panel>.panel-collapse>.list-group:first-child .list-group-item:first-child{border-top:0;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.list-group:last-child .list-group-item:last-child,.panel>.panel-collapse>.list-group:last-child .list-group-item:last-child{border-bottom:0;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.panel-heading+.panel-collapse>.list-group .list-group-item:first-child{border-top-left-radius:0;border-top-right-radius:0}.panel-heading+.list-group .list-group-item:first-child{border-top-width:0}.list-group+.panel-footer{border-top-width:0}.panel>.table,.panel>.table-responsive>.table,.panel>.panel-collapse>.table{margin-bottom:0}.panel>.table caption,.panel>.table-responsive>.table caption,.panel>.panel-collapse>.table caption{padding-right:15px;padding-left:15px}.panel>.table:first-child,.panel>.table-responsive:first-child>.table:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:first-child{border-top-left-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:last-child{border-top-right-radius:3px}.panel>.table:last-child,.panel>.table-responsive:last-child>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:first-child{border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:last-child{border-bottom-right-radius:3px}.panel>.panel-body+.table,.panel>.panel-body+.table-responsive,.panel>.table+.panel-body,.panel>.table-responsive+.panel-body{border-top:1px solid #ddd}.panel>.table>tbody:first-child>tr:first-child th,.panel>.table>tbody:first-child>tr:first-child td{border-top:0}.panel>.table-bordered,.panel>.table-responsive>.table-bordered{border:0}.panel>.table-bordered>thead>tr>th:first-child,.panel>.table-responsive>.table-bordered>thead>tr>th:first-child,.panel>.table-bordered>tbody>tr>th:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:first-child,.panel>.table-bordered>tfoot>tr>th:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:first-child,.panel>.table-bordered>thead>tr>td:first-child,.panel>.table-responsive>.table-bordered>thead>tr>td:first-child,.panel>.table-bordered>tbody>tr>td:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:first-child,.panel>.table-bordered>tfoot>tr>td:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:first-child{border-left:0}.panel>.table-bordered>thead>tr>th:last-child,.panel>.table-responsive>.table-bordered>thead>tr>th:last-child,.panel>.table-bordered>tbody>tr>th:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:last-child,.panel>.table-bordered>tfoot>tr>th:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:last-child,.panel>.table-bordered>thead>tr>td:last-child,.panel>.table-responsive>.table-bordered>thead>tr>td:last-child,.panel>.table-bordered>tbody>tr>td:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:last-child,.panel>.table-bordered>tfoot>tr>td:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:last-child{border-right:0}.panel>.table-bordered>thead>tr:first-child>td,.panel>.table-responsive>.table-bordered>thead>tr:first-child>td,.panel>.table-bordered>tbody>tr:first-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>td,.panel>.table-bordered>thead>tr:first-child>th,.panel>.table-responsive>.table-bordered>thead>tr:first-child>th,.panel>.table-bordered>tbody>tr:first-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>th{border-bottom:0}.panel>.table-bordered>tbody>tr:last-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>td,.panel>.table-bordered>tfoot>tr:last-child>td,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>td,.panel>.table-bordered>tbody>tr:last-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>th,.panel>.table-bordered>tfoot>tr:last-child>th,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}.panel>.table-responsive{margin-bottom:0;border:0}.panel-group{margin-bottom:20px}.panel-group .panel{margin-bottom:0;border-radius:4px}.panel-group .panel+.panel{margin-top:5px}.panel-group .panel-heading{border-bottom:0}.panel-group .panel-heading+.panel-collapse>.panel-body,.panel-group .panel-heading+.panel-collapse>.list-group{border-top:1px solid #ddd}.panel-group .panel-footer{border-top:0}.panel-group .panel-footer+.panel-collapse .panel-body{border-bottom:1px solid #ddd}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.panel-default>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ddd}.panel-default>.panel-heading .badge{color:#f5f5f5;background-color:#333}.panel-default>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ddd}.panel-primary{border-color:#337ab7}.panel-primary>.panel-heading{color:#fff;background-color:#337ab7;border-color:#337ab7}.panel-primary>.panel-heading+.panel-collapse>.panel-body{border-top-color:#337ab7}.panel-primary>.panel-heading .badge{color:#337ab7;background-color:#fff}.panel-primary>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#337ab7}.panel-success{border-color:#d6e9c6}.panel-success>.panel-heading{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.panel-success>.panel-heading+.panel-collapse>.panel-body{border-top-color:#d6e9c6}.panel-success>.panel-heading .badge{color:#dff0d8;background-color:#3c763d}.panel-success>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#d6e9c6}.panel-info{border-color:#bce8f1}.panel-info>.panel-heading{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.panel-info>.panel-heading+.panel-collapse>.panel-body{border-top-color:#bce8f1}.panel-info>.panel-heading .badge{color:#d9edf7;background-color:#31708f}.panel-info>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#bce8f1}.panel-warning{border-color:#faebcc}.panel-warning>.panel-heading{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.panel-warning>.panel-heading+.panel-collapse>.panel-body{border-top-color:#faebcc}.panel-warning>.panel-heading .badge{color:#fcf8e3;background-color:#8a6d3b}.panel-warning>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#faebcc}.panel-danger{border-color:#ebccd1}.panel-danger>.panel-heading{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.panel-danger>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ebccd1}.panel-danger>.panel-heading .badge{color:#f2dede;background-color:#a94442}.panel-danger>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ebccd1}.embed-responsive{position:relative;display:block;height:0;padding:0;overflow:hidden}.embed-responsive .embed-responsive-item,.embed-responsive iframe,.embed-responsive embed,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-16by9{padding-bottom:56.25%}.embed-responsive-4by3{padding-bottom:75%}.well{min-height:20px;padding:19px;margin-bottom:20px;background-color:#f5f5f5;border:1px solid #e3e3e3;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.05);box-shadow:inset 0 1px 1px rgba(0,0,0,0.05)}.well blockquote{border-color:#ddd;border-color:rgba(0,0,0,0.15)}.well-lg{padding:24px;border-radius:6px}.well-sm{padding:9px;border-radius:3px}.close{float:right;font-size:21px;font-weight:bold;line-height:1;color:#000;text-shadow:0 1px 0 #fff;filter:alpha(opacity=20);opacity:.2}.close:hover,.close:focus{color:#000;text-decoration:none;cursor:pointer;filter:alpha(opacity=50);opacity:.5}button.close{padding:0;cursor:pointer;background:transparent;border:0;-webkit-appearance:none;appearance:none}.modal-open{overflow:hidden}.modal{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1050;display:none;overflow:hidden;-webkit-overflow-scrolling:touch;outline:0}.modal.fade .modal-dialog{-webkit-transform:translate(0,-25%);-ms-transform:translate(0,-25%);-o-transform:translate(0,-25%);transform:translate(0,-25%);-webkit-transition:-webkit-transform .3s ease-out;-moz-transition:-moz-transform .3s ease-out;-o-transition:-o-transform .3s ease-out;transition:transform .3s ease-out}.modal.in .modal-dialog{-webkit-transform:translate(0,0);-ms-transform:translate(0,0);-o-transform:translate(0,0);transform:translate(0,0)}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal-dialog{position:relative;width:auto;margin:10px}.modal-content{position:relative;background-color:#fff;background-clip:padding-box;border:1px solid #999;border:1px solid rgba(0,0,0,0.2);border-radius:6px;-webkit-box-shadow:0 3px 9px rgba(0,0,0,0.5);box-shadow:0 3px 9px rgba(0,0,0,0.5);outline:0}.modal-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1040;background-color:#000}.modal-backdrop.fade{filter:alpha(opacity=0);opacity:0}.modal-backdrop.in{filter:alpha(opacity=50);opacity:.5}.modal-header{padding:15px;border-bottom:1px solid #e5e5e5}.modal-header .close{margin-top:-2px}.modal-title{margin:0;line-height:1.42857143}.modal-body{position:relative;padding:15px}.modal-footer{padding:15px;text-align:right;border-top:1px solid #e5e5e5}.modal-footer .btn+.btn{margin-bottom:0;margin-left:5px}.modal-footer .btn-group .btn+.btn{margin-left:-1px}.modal-footer .btn-block+.btn-block{margin-left:0}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media(min-width:768px){.modal-dialog{width:600px;margin:30px auto}.modal-content{-webkit-box-shadow:0 5px 15px rgba(0,0,0,0.5);box-shadow:0 5px 15px rgba(0,0,0,0.5)}.modal-sm{width:300px}}@media(min-width:992px){.modal-lg{width:900px}}.tooltip{position:absolute;z-index:1070;display:block;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-style:normal;font-weight:400;line-height:1.42857143;line-break:auto;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;word-wrap:normal;white-space:normal;font-size:12px;filter:alpha(opacity=0);opacity:0}.tooltip.in{filter:alpha(opacity=90);opacity:.9}.tooltip.top{padding:5px 0;margin-top:-3px}.tooltip.right{padding:0 5px;margin-left:3px}.tooltip.bottom{padding:5px 0;margin-top:3px}.tooltip.left{padding:0 5px;margin-left:-3px}.tooltip.top .tooltip-arrow{bottom:0;left:50%;margin-left:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-left .tooltip-arrow{right:5px;bottom:0;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-right .tooltip-arrow{bottom:0;left:5px;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.right .tooltip-arrow{top:50%;left:0;margin-top:-5px;border-width:5px 5px 5px 0;border-right-color:#000}.tooltip.left .tooltip-arrow{top:50%;right:0;margin-top:-5px;border-width:5px 0 5px 5px;border-left-color:#000}.tooltip.bottom .tooltip-arrow{top:0;left:50%;margin-left:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-left .tooltip-arrow{top:0;right:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-right .tooltip-arrow{top:0;left:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip-inner{max-width:200px;padding:3px 8px;color:#fff;text-align:center;background-color:#000;border-radius:4px}.tooltip-arrow{position:absolute;width:0;height:0;border-color:transparent;border-style:solid}.popover{position:absolute;top:0;left:0;z-index:1060;display:none;max-width:276px;padding:1px;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-style:normal;font-weight:400;line-height:1.42857143;line-break:auto;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;word-wrap:normal;white-space:normal;font-size:14px;background-color:#fff;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,0.2);border-radius:6px;-webkit-box-shadow:0 5px 10px rgba(0,0,0,0.2);box-shadow:0 5px 10px rgba(0,0,0,0.2)}.popover.top{margin-top:-10px}.popover.right{margin-left:10px}.popover.bottom{margin-top:10px}.popover.left{margin-left:-10px}.popover>.arrow{border-width:11px}.popover>.arrow,.popover>.arrow:after{position:absolute;display:block;width:0;height:0;border-color:transparent;border-style:solid}.popover>.arrow:after{content:"";border-width:10px}.popover.top>.arrow{bottom:-11px;left:50%;margin-left:-11px;border-top-color:#999;border-top-color:rgba(0,0,0,0.25);border-bottom-width:0}.popover.top>.arrow:after{bottom:1px;margin-left:-10px;content:" ";border-top-color:#fff;border-bottom-width:0}.popover.right>.arrow{top:50%;left:-11px;margin-top:-11px;border-right-color:#999;border-right-color:rgba(0,0,0,0.25);border-left-width:0}.popover.right>.arrow:after{bottom:-10px;left:1px;content:" ";border-right-color:#fff;border-left-width:0}.popover.bottom>.arrow{top:-11px;left:50%;margin-left:-11px;border-top-width:0;border-bottom-color:#999;border-bottom-color:rgba(0,0,0,0.25)}.popover.bottom>.arrow:after{top:1px;margin-left:-10px;content:" ";border-top-width:0;border-bottom-color:#fff}.popover.left>.arrow{top:50%;right:-11px;margin-top:-11px;border-right-width:0;border-left-color:#999;border-left-color:rgba(0,0,0,0.25)}.popover.left>.arrow:after{right:1px;bottom:-10px;content:" ";border-right-width:0;border-left-color:#fff}.popover-title{padding:8px 14px;margin:0;font-size:14px;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-radius:5px 5px 0 0}.popover-content{padding:9px 14px}.carousel{position:relative}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner>.item{position:relative;display:none;-webkit-transition:.6s ease-in-out left;-o-transition:.6s ease-in-out left;transition:.6s ease-in-out left}.carousel-inner>.item>img,.carousel-inner>.item>a>img{line-height:1}@media all and (transform-3d),(-webkit-transform-3d){.carousel-inner>.item{-webkit-transition:-webkit-transform .6s ease-in-out;-moz-transition:-moz-transform .6s ease-in-out;-o-transition:-o-transform .6s ease-in-out;transition:transform .6s ease-in-out;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;backface-visibility:hidden;-webkit-perspective:1000px;-moz-perspective:1000px;perspective:1000px}.carousel-inner>.item.next,.carousel-inner>.item.active.right{-webkit-transform:translate3d(100%,0,0);transform:translate3d(100%,0,0);left:0}.carousel-inner>.item.prev,.carousel-inner>.item.active.left{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0);left:0}.carousel-inner>.item.next.left,.carousel-inner>.item.prev.right,.carousel-inner>.item.active{-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0);left:0}}.carousel-inner>.active,.carousel-inner>.next,.carousel-inner>.prev{display:block}.carousel-inner>.active{left:0}.carousel-inner>.next,.carousel-inner>.prev{position:absolute;top:0;width:100%}.carousel-inner>.next{left:100%}.carousel-inner>.prev{left:-100%}.carousel-inner>.next.left,.carousel-inner>.prev.right{left:0}.carousel-inner>.active.left{left:-100%}.carousel-inner>.active.right{left:100%}.carousel-control{position:absolute;top:0;bottom:0;left:0;width:15%;font-size:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,0.6);background-color:rgba(0,0,0,0);filter:alpha(opacity=50);opacity:.5}.carousel-control.left{background-image:-webkit-linear-gradient(left,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);background-image:linear-gradient(to right,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#80000000',endColorstr='#00000000',GradientType=1);background-repeat:repeat-x}.carousel-control.right{right:0;left:auto;background-image:-webkit-linear-gradient(left,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);background-image:linear-gradient(to right,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#00000000',endColorstr='#80000000',GradientType=1);background-repeat:repeat-x}.carousel-control:hover,.carousel-control:focus{color:#fff;text-decoration:none;outline:0;filter:alpha(opacity=90);opacity:.9}.carousel-control .icon-prev,.carousel-control .icon-next,.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right{position:absolute;top:50%;z-index:5;display:inline-block;margin-top:-10px}.carousel-control .icon-prev,.carousel-control .glyphicon-chevron-left{left:50%;margin-left:-10px}.carousel-control .icon-next,.carousel-control .glyphicon-chevron-right{right:50%;margin-right:-10px}.carousel-control .icon-prev,.carousel-control .icon-next{width:20px;height:20px;font-family:serif;line-height:1}.carousel-control .icon-prev:before{content:"\2039"}.carousel-control .icon-next:before{content:"\203a"}.carousel-indicators{position:absolute;bottom:10px;left:50%;z-index:15;width:60%;padding-left:0;margin-left:-30%;text-align:center;list-style:none}.carousel-indicators li{display:inline-block;width:10px;height:10px;margin:1px;text-indent:-999px;cursor:pointer;background-color:#000 \9;background-color:rgba(0,0,0,0);border:1px solid #fff;border-radius:10px}.carousel-indicators .active{width:12px;height:12px;margin:0;background-color:#fff}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,0.6)}.carousel-caption .btn{text-shadow:none}@media screen and (min-width:768px){.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-prev,.carousel-control .icon-next{width:30px;height:30px;margin-top:-10px;font-size:30px}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{margin-left:-10px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{margin-right:-10px}.carousel-caption{right:20%;left:20%;padding-bottom:30px}.carousel-indicators{bottom:20px}}.clearfix:before,.clearfix:after,.dl-horizontal dd:before,.dl-horizontal dd:after,.container:before,.container:after,.container-fluid:before,.container-fluid:after,.row:before,.row:after,.form-horizontal .form-group:before,.form-horizontal .form-group:after,.btn-toolbar:before,.btn-toolbar:after,.btn-group-vertical>.btn-group:before,.btn-group-vertical>.btn-group:after,.nav:before,.nav:after,.navbar:before,.navbar:after,.navbar-header:before,.navbar-header:after,.navbar-collapse:before,.navbar-collapse:after,.pager:before,.pager:after,.panel-body:before,.panel-body:after,.modal-header:before,.modal-header:after,.modal-footer:before,.modal-footer:after{display:table;content:" "}.clearfix:after,.dl-horizontal dd:after,.container:after,.container-fluid:after,.row:after,.form-horizontal .form-group:after,.btn-toolbar:after,.btn-group-vertical>.btn-group:after,.nav:after,.navbar:after,.navbar-header:after,.navbar-collapse:after,.pager:after,.panel-body:after,.modal-header:after,.modal-footer:after{clear:both}.center-block{display:block;margin-right:auto;margin-left:auto}.pull-right{float:right!important}.pull-left{float:left!important}.hide{display:none!important}.show{display:block!important}.invisible{visibility:hidden}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.hidden{display:none!important}.affix{position:fixed}@-ms-viewport{width:device-width}.visible-xs,.visible-sm,.visible-md,.visible-lg{display:none!important}.visible-xs-block,.visible-xs-inline,.visible-xs-inline-block,.visible-sm-block,.visible-sm-inline,.visible-sm-inline-block,.visible-md-block,.visible-md-inline,.visible-md-inline-block,.visible-lg-block,.visible-lg-inline,.visible-lg-inline-block{display:none!important}@media(max-width:767px){.visible-xs{display:block!important}table.visible-xs{display:table!important}tr.visible-xs{display:table-row!important}th.visible-xs,td.visible-xs{display:table-cell!important}}@media(max-width:767px){.visible-xs-block{display:block!important}}@media(max-width:767px){.visible-xs-inline{display:inline!important}}@media(max-width:767px){.visible-xs-inline-block{display:inline-block!important}}@media(min-width:768px) and (max-width:991px){.visible-sm{display:block!important}table.visible-sm{display:table!important}tr.visible-sm{display:table-row!important}th.visible-sm,td.visible-sm{display:table-cell!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-block{display:block!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-inline{display:inline!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-inline-block{display:inline-block!important}}@media(min-width:992px) and (max-width:1199px){.visible-md{display:block!important}table.visible-md{display:table!important}tr.visible-md{display:table-row!important}th.visible-md,td.visible-md{display:table-cell!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-block{display:block!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-inline{display:inline!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-inline-block{display:inline-block!important}}@media(min-width:1200px){.visible-lg{display:block!important}table.visible-lg{display:table!important}tr.visible-lg{display:table-row!important}th.visible-lg,td.visible-lg{display:table-cell!important}}@media(min-width:1200px){.visible-lg-block{display:block!important}}@media(min-width:1200px){.visible-lg-inline{display:inline!important}}@media(min-width:1200px){.visible-lg-inline-block{display:inline-block!important}}@media(max-width:767px){.hidden-xs{display:none!important}}@media(min-width:768px) and (max-width:991px){.hidden-sm{display:none!important}}@media(min-width:992px) and (max-width:1199px){.hidden-md{display:none!important}}@media(min-width:1200px){.hidden-lg{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:block!important}table.visible-print{display:table!important}tr.visible-print{display:table-row!important}th.visible-print,td.visible-print{display:table-cell!important}}.visible-print-block{display:none!important}@media print{.visible-print-block{display:block!important}}.visible-print-inline{display:none!important}@media print{.visible-print-inline{display:inline!important}}.visible-print-inline-block{display:none!important}@media print{.visible-print-inline-block{display:inline-block!important}}@media print{.hidden-print{display:none!important}}</style>
+<script>if(typeof jQuery==="undefined"){throw new Error("Bootstrap's JavaScript requires jQuery")}+function($){"use strict";var version=$.fn.jquery.split(" ")[0].split(".");if(version[0]<2&&version[1]<9||version[0]==1&&version[1]==9&&version[2]<1||version[0]>3){throw new Error("Bootstrap's JavaScript requires jQuery version 1.9.1 or higher, but lower than version 4")}}(jQuery);+function($){"use strict";function transitionEnd(){var el=document.createElement("bootstrap");var transEndEventNames={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"};for(var name in transEndEventNames){if(el.style[name]!==undefined){return{end:transEndEventNames[name]}}}return false}$.fn.emulateTransitionEnd=function(duration){var called=false;var $el=this;$(this).one("bsTransitionEnd",function(){called=true});var callback=function(){if(!called)$($el).trigger($.support.transition.end)};setTimeout(callback,duration);return this};$(function(){$.support.transition=transitionEnd();if(!$.support.transition)return;$.event.special.bsTransitionEnd={bindType:$.support.transition.end,delegateType:$.support.transition.end,handle:function(e){if($(e.target).is(this))return e.handleObj.handler.apply(this,arguments)}}})}(jQuery);+function($){"use strict";var dismiss='[data-dismiss="alert"]';var Alert=function(el){$(el).on("click",dismiss,this.close)};Alert.VERSION="3.4.1";Alert.TRANSITION_DURATION=150;Alert.prototype.close=function(e){var $this=$(this);var selector=$this.attr("data-target");if(!selector){selector=$this.attr("href");selector=selector&&selector.replace(/.*(?=#[^\s]*$)/,"")}selector=selector==="#"?[]:selector;var $parent=$(document).find(selector);if(e)e.preventDefault();if(!$parent.length){$parent=$this.closest(".alert")}$parent.trigger(e=$.Event("close.bs.alert"));if(e.isDefaultPrevented())return;$parent.removeClass("in");function removeElement(){$parent.detach().trigger("closed.bs.alert").remove()}$.support.transition&&$parent.hasClass("fade")?$parent.one("bsTransitionEnd",removeElement).emulateTransitionEnd(Alert.TRANSITION_DURATION):removeElement()};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.alert");if(!data)$this.data("bs.alert",data=new Alert(this));if(typeof option=="string")data[option].call($this)})}var old=$.fn.alert;$.fn.alert=Plugin;$.fn.alert.Constructor=Alert;$.fn.alert.noConflict=function(){$.fn.alert=old;return this};$(document).on("click.bs.alert.data-api",dismiss,Alert.prototype.close)}(jQuery);+function($){"use strict";var Button=function(element,options){this.$element=$(element);this.options=$.extend({},Button.DEFAULTS,options);this.isLoading=false};Button.VERSION="3.4.1";Button.DEFAULTS={loadingText:"loading..."};Button.prototype.setState=function(state){var d="disabled";var $el=this.$element;var val=$el.is("input")?"val":"html";var data=$el.data();state+="Text";if(data.resetText==null)$el.data("resetText",$el[val]());setTimeout($.proxy(function(){$el[val](data[state]==null?this.options[state]:data[state]);if(state=="loadingText"){this.isLoading=true;$el.addClass(d).attr(d,d).prop(d,true)}else if(this.isLoading){this.isLoading=false;$el.removeClass(d).removeAttr(d).prop(d,false)}},this),0)};Button.prototype.toggle=function(){var changed=true;var $parent=this.$element.closest('[data-toggle="buttons"]');if($parent.length){var $input=this.$element.find("input");if($input.prop("type")=="radio"){if($input.prop("checked"))changed=false;$parent.find(".active").removeClass("active");this.$element.addClass("active")}else if($input.prop("type")=="checkbox"){if($input.prop("checked")!==this.$element.hasClass("active"))changed=false;this.$element.toggleClass("active")}$input.prop("checked",this.$element.hasClass("active"));if(changed)$input.trigger("change")}else{this.$element.attr("aria-pressed",!this.$element.hasClass("active"));this.$element.toggleClass("active")}};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.button");var options=typeof option=="object"&&option;if(!data)$this.data("bs.button",data=new Button(this,options));if(option=="toggle")data.toggle();else if(option)data.setState(option)})}var old=$.fn.button;$.fn.button=Plugin;$.fn.button.Constructor=Button;$.fn.button.noConflict=function(){$.fn.button=old;return this};$(document).on("click.bs.button.data-api",'[data-toggle^="button"]',function(e){var $btn=$(e.target).closest(".btn");Plugin.call($btn,"toggle");if(!$(e.target).is('input[type="radio"], input[type="checkbox"]')){e.preventDefault();if($btn.is("input,button"))$btn.trigger("focus");else $btn.find("input:visible,button:visible").first().trigger("focus")}}).on("focus.bs.button.data-api blur.bs.button.data-api",'[data-toggle^="button"]',function(e){$(e.target).closest(".btn").toggleClass("focus",/^focus(in)?$/.test(e.type))})}(jQuery);+function($){"use strict";var Carousel=function(element,options){this.$element=$(element);this.$indicators=this.$element.find(".carousel-indicators");this.options=options;this.paused=null;this.sliding=null;this.interval=null;this.$active=null;this.$items=null;this.options.keyboard&&this.$element.on("keydown.bs.carousel",$.proxy(this.keydown,this));this.options.pause=="hover"&&!("ontouchstart"in document.documentElement)&&this.$element.on("mouseenter.bs.carousel",$.proxy(this.pause,this)).on("mouseleave.bs.carousel",$.proxy(this.cycle,this))};Carousel.VERSION="3.4.1";Carousel.TRANSITION_DURATION=600;Carousel.DEFAULTS={interval:5e3,pause:"hover",wrap:true,keyboard:true};Carousel.prototype.keydown=function(e){if(/input|textarea/i.test(e.target.tagName))return;switch(e.which){case 37:this.prev();break;case 39:this.next();break;default:return}e.preventDefault()};Carousel.prototype.cycle=function(e){e||(this.paused=false);this.interval&&clearInterval(this.interval);this.options.interval&&!this.paused&&(this.interval=setInterval($.proxy(this.next,this),this.options.interval));return this};Carousel.prototype.getItemIndex=function(item){this.$items=item.parent().children(".item");return this.$items.index(item||this.$active)};Carousel.prototype.getItemForDirection=function(direction,active){var activeIndex=this.getItemIndex(active);var willWrap=direction=="prev"&&activeIndex===0||direction=="next"&&activeIndex==this.$items.length-1;if(willWrap&&!this.options.wrap)return active;var delta=direction=="prev"?-1:1;var itemIndex=(activeIndex+delta)%this.$items.length;return this.$items.eq(itemIndex)};Carousel.prototype.to=function(pos){var that=this;var activeIndex=this.getItemIndex(this.$active=this.$element.find(".item.active"));if(pos>this.$items.length-1||pos<0)return;if(this.sliding)return this.$element.one("slid.bs.carousel",function(){that.to(pos)});if(activeIndex==pos)return this.pause().cycle();return this.slide(pos>activeIndex?"next":"prev",this.$items.eq(pos))};Carousel.prototype.pause=function(e){e||(this.paused=true);if(this.$element.find(".next, .prev").length&&$.support.transition){this.$element.trigger($.support.transition.end);this.cycle(true)}this.interval=clearInterval(this.interval);return this};Carousel.prototype.next=function(){if(this.sliding)return;return this.slide("next")};Carousel.prototype.prev=function(){if(this.sliding)return;return this.slide("prev")};Carousel.prototype.slide=function(type,next){var $active=this.$element.find(".item.active");var $next=next||this.getItemForDirection(type,$active);var isCycling=this.interval;var direction=type=="next"?"left":"right";var that=this;if($next.hasClass("active"))return this.sliding=false;var relatedTarget=$next[0];var slideEvent=$.Event("slide.bs.carousel",{relatedTarget:relatedTarget,direction:direction});this.$element.trigger(slideEvent);if(slideEvent.isDefaultPrevented())return;this.sliding=true;isCycling&&this.pause();if(this.$indicators.length){this.$indicators.find(".active").removeClass("active");var $nextIndicator=$(this.$indicators.children()[this.getItemIndex($next)]);$nextIndicator&&$nextIndicator.addClass("active")}var slidEvent=$.Event("slid.bs.carousel",{relatedTarget:relatedTarget,direction:direction});if($.support.transition&&this.$element.hasClass("slide")){$next.addClass(type);if(typeof $next==="object"&&$next.length){$next[0].offsetWidth}$active.addClass(direction);$next.addClass(direction);$active.one("bsTransitionEnd",function(){$next.removeClass([type,direction].join(" ")).addClass("active");$active.removeClass(["active",direction].join(" "));that.sliding=false;setTimeout(function(){that.$element.trigger(slidEvent)},0)}).emulateTransitionEnd(Carousel.TRANSITION_DURATION)}else{$active.removeClass("active");$next.addClass("active");this.sliding=false;this.$element.trigger(slidEvent)}isCycling&&this.cycle();return this};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.carousel");var options=$.extend({},Carousel.DEFAULTS,$this.data(),typeof option=="object"&&option);var action=typeof option=="string"?option:options.slide;if(!data)$this.data("bs.carousel",data=new Carousel(this,options));if(typeof option=="number")data.to(option);else if(action)data[action]();else if(options.interval)data.pause().cycle()})}var old=$.fn.carousel;$.fn.carousel=Plugin;$.fn.carousel.Constructor=Carousel;$.fn.carousel.noConflict=function(){$.fn.carousel=old;return this};var clickHandler=function(e){var $this=$(this);var href=$this.attr("href");if(href){href=href.replace(/.*(?=#[^\s]+$)/,"")}var target=$this.attr("data-target")||href;var $target=$(document).find(target);if(!$target.hasClass("carousel"))return;var options=$.extend({},$target.data(),$this.data());var slideIndex=$this.attr("data-slide-to");if(slideIndex)options.interval=false;Plugin.call($target,options);if(slideIndex){$target.data("bs.carousel").to(slideIndex)}e.preventDefault()};$(document).on("click.bs.carousel.data-api","[data-slide]",clickHandler).on("click.bs.carousel.data-api","[data-slide-to]",clickHandler);$(window).on("load",function(){$('[data-ride="carousel"]').each(function(){var $carousel=$(this);Plugin.call($carousel,$carousel.data())})})}(jQuery);+function($){"use strict";var Collapse=function(element,options){this.$element=$(element);this.options=$.extend({},Collapse.DEFAULTS,options);this.$trigger=$('[data-toggle="collapse"][href="#'+element.id+'"],'+'[data-toggle="collapse"][data-target="#'+element.id+'"]');this.transitioning=null;if(this.options.parent){this.$parent=this.getParent()}else{this.addAriaAndCollapsedClass(this.$element,this.$trigger)}if(this.options.toggle)this.toggle()};Collapse.VERSION="3.4.1";Collapse.TRANSITION_DURATION=350;Collapse.DEFAULTS={toggle:true};Collapse.prototype.dimension=function(){var hasWidth=this.$element.hasClass("width");return hasWidth?"width":"height"};Collapse.prototype.show=function(){if(this.transitioning||this.$element.hasClass("in"))return;var activesData;var actives=this.$parent&&this.$parent.children(".panel").children(".in, .collapsing");if(actives&&actives.length){activesData=actives.data("bs.collapse");if(activesData&&activesData.transitioning)return}var startEvent=$.Event("show.bs.collapse");this.$element.trigger(startEvent);if(startEvent.isDefaultPrevented())return;if(actives&&actives.length){Plugin.call(actives,"hide");activesData||actives.data("bs.collapse",null)}var dimension=this.dimension();this.$element.removeClass("collapse").addClass("collapsing")[dimension](0).attr("aria-expanded",true);this.$trigger.removeClass("collapsed").attr("aria-expanded",true);this.transitioning=1;var complete=function(){this.$element.removeClass("collapsing").addClass("collapse in")[dimension]("");this.transitioning=0;this.$element.trigger("shown.bs.collapse")};if(!$.support.transition)return complete.call(this);var scrollSize=$.camelCase(["scroll",dimension].join("-"));this.$element.one("bsTransitionEnd",$.proxy(complete,this)).emulateTransitionEnd(Collapse.TRANSITION_DURATION)[dimension](this.$element[0][scrollSize])};Collapse.prototype.hide=function(){if(this.transitioning||!this.$element.hasClass("in"))return;var startEvent=$.Event("hide.bs.collapse");this.$element.trigger(startEvent);if(startEvent.isDefaultPrevented())return;var dimension=this.dimension();this.$element[dimension](this.$element[dimension]())[0].offsetHeight;this.$element.addClass("collapsing").removeClass("collapse in").attr("aria-expanded",false);this.$trigger.addClass("collapsed").attr("aria-expanded",false);this.transitioning=1;var complete=function(){this.transitioning=0;this.$element.removeClass("collapsing").addClass("collapse").trigger("hidden.bs.collapse")};if(!$.support.transition)return complete.call(this);this.$element[dimension](0).one("bsTransitionEnd",$.proxy(complete,this)).emulateTransitionEnd(Collapse.TRANSITION_DURATION)};Collapse.prototype.toggle=function(){this[this.$element.hasClass("in")?"hide":"show"]()};Collapse.prototype.getParent=function(){return $(document).find(this.options.parent).find('[data-toggle="collapse"][data-parent="'+this.options.parent+'"]').each($.proxy(function(i,element){var $element=$(element);this.addAriaAndCollapsedClass(getTargetFromTrigger($element),$element)},this)).end()};Collapse.prototype.addAriaAndCollapsedClass=function($element,$trigger){var isOpen=$element.hasClass("in");$element.attr("aria-expanded",isOpen);$trigger.toggleClass("collapsed",!isOpen).attr("aria-expanded",isOpen)};function getTargetFromTrigger($trigger){var href;var target=$trigger.attr("data-target")||(href=$trigger.attr("href"))&&href.replace(/.*(?=#[^\s]+$)/,"");return $(document).find(target)}function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.collapse");var options=$.extend({},Collapse.DEFAULTS,$this.data(),typeof option=="object"&&option);if(!data&&options.toggle&&/show|hide/.test(option))options.toggle=false;if(!data)$this.data("bs.collapse",data=new Collapse(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.collapse;$.fn.collapse=Plugin;$.fn.collapse.Constructor=Collapse;$.fn.collapse.noConflict=function(){$.fn.collapse=old;return this};$(document).on("click.bs.collapse.data-api",'[data-toggle="collapse"]',function(e){var $this=$(this);if(!$this.attr("data-target"))e.preventDefault();var $target=getTargetFromTrigger($this);var data=$target.data("bs.collapse");var option=data?"toggle":$this.data();Plugin.call($target,option)})}(jQuery);+function($){"use strict";var backdrop=".dropdown-backdrop";var toggle='[data-toggle="dropdown"]';var Dropdown=function(element){$(element).on("click.bs.dropdown",this.toggle)};Dropdown.VERSION="3.4.1";function getParent($this){var selector=$this.attr("data-target");if(!selector){selector=$this.attr("href");selector=selector&&/#[A-Za-z]/.test(selector)&&selector.replace(/.*(?=#[^\s]*$)/,"")}var $parent=selector!=="#"?$(document).find(selector):null;return $parent&&$parent.length?$parent:$this.parent()}function clearMenus(e){if(e&&e.which===3)return;$(backdrop).remove();$(toggle).each(function(){var $this=$(this);var $parent=getParent($this);var relatedTarget={relatedTarget:this};if(!$parent.hasClass("open"))return;if(e&&e.type=="click"&&/input|textarea/i.test(e.target.tagName)&&$.contains($parent[0],e.target))return;$parent.trigger(e=$.Event("hide.bs.dropdown",relatedTarget));if(e.isDefaultPrevented())return;$this.attr("aria-expanded","false");$parent.removeClass("open").trigger($.Event("hidden.bs.dropdown",relatedTarget))})}Dropdown.prototype.toggle=function(e){var $this=$(this);if($this.is(".disabled, :disabled"))return;var $parent=getParent($this);var isActive=$parent.hasClass("open");clearMenus();if(!isActive){if("ontouchstart"in document.documentElement&&!$parent.closest(".navbar-nav").length){$(document.createElement("div")).addClass("dropdown-backdrop").insertAfter($(this)).on("click",clearMenus)}var relatedTarget={relatedTarget:this};$parent.trigger(e=$.Event("show.bs.dropdown",relatedTarget));if(e.isDefaultPrevented())return;$this.trigger("focus").attr("aria-expanded","true");$parent.toggleClass("open").trigger($.Event("shown.bs.dropdown",relatedTarget))}return false};Dropdown.prototype.keydown=function(e){if(!/(38|40|27|32)/.test(e.which)||/input|textarea/i.test(e.target.tagName))return;var $this=$(this);e.preventDefault();e.stopPropagation();if($this.is(".disabled, :disabled"))return;var $parent=getParent($this);var isActive=$parent.hasClass("open");if(!isActive&&e.which!=27||isActive&&e.which==27){if(e.which==27)$parent.find(toggle).trigger("focus");return $this.trigger("click")}var desc=" li:not(.disabled):visible a";var $items=$parent.find(".dropdown-menu"+desc);if(!$items.length)return;var index=$items.index(e.target);if(e.which==38&&index>0)index--;if(e.which==40&&index<$items.length-1)index++;if(!~index)index=0;$items.eq(index).trigger("focus")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.dropdown");if(!data)$this.data("bs.dropdown",data=new Dropdown(this));if(typeof option=="string")data[option].call($this)})}var old=$.fn.dropdown;$.fn.dropdown=Plugin;$.fn.dropdown.Constructor=Dropdown;$.fn.dropdown.noConflict=function(){$.fn.dropdown=old;return this};$(document).on("click.bs.dropdown.data-api",clearMenus).on("click.bs.dropdown.data-api",".dropdown form",function(e){e.stopPropagation()}).on("click.bs.dropdown.data-api",toggle,Dropdown.prototype.toggle).on("keydown.bs.dropdown.data-api",toggle,Dropdown.prototype.keydown).on("keydown.bs.dropdown.data-api",".dropdown-menu",Dropdown.prototype.keydown)}(jQuery);+function($){"use strict";var Modal=function(element,options){this.options=options;this.$body=$(document.body);this.$element=$(element);this.$dialog=this.$element.find(".modal-dialog");this.$backdrop=null;this.isShown=null;this.originalBodyPad=null;this.scrollbarWidth=0;this.ignoreBackdropClick=false;this.fixedContent=".navbar-fixed-top, .navbar-fixed-bottom";if(this.options.remote){this.$element.find(".modal-content").load(this.options.remote,$.proxy(function(){this.$element.trigger("loaded.bs.modal")},this))}};Modal.VERSION="3.4.1";Modal.TRANSITION_DURATION=300;Modal.BACKDROP_TRANSITION_DURATION=150;Modal.DEFAULTS={backdrop:true,keyboard:true,show:true};Modal.prototype.toggle=function(_relatedTarget){return this.isShown?this.hide():this.show(_relatedTarget)};Modal.prototype.show=function(_relatedTarget){var that=this;var e=$.Event("show.bs.modal",{relatedTarget:_relatedTarget});this.$element.trigger(e);if(this.isShown||e.isDefaultPrevented())return;this.isShown=true;this.checkScrollbar();this.setScrollbar();this.$body.addClass("modal-open");this.escape();this.resize();this.$element.on("click.dismiss.bs.modal",'[data-dismiss="modal"]',$.proxy(this.hide,this));this.$dialog.on("mousedown.dismiss.bs.modal",function(){that.$element.one("mouseup.dismiss.bs.modal",function(e){if($(e.target).is(that.$element))that.ignoreBackdropClick=true})});this.backdrop(function(){var transition=$.support.transition&&that.$element.hasClass("fade");if(!that.$element.parent().length){that.$element.appendTo(that.$body)}that.$element.show().scrollTop(0);that.adjustDialog();if(transition){that.$element[0].offsetWidth}that.$element.addClass("in");that.enforceFocus();var e=$.Event("shown.bs.modal",{relatedTarget:_relatedTarget});transition?that.$dialog.one("bsTransitionEnd",function(){that.$element.trigger("focus").trigger(e)}).emulateTransitionEnd(Modal.TRANSITION_DURATION):that.$element.trigger("focus").trigger(e)})};Modal.prototype.hide=function(e){if(e)e.preventDefault();e=$.Event("hide.bs.modal");this.$element.trigger(e);if(!this.isShown||e.isDefaultPrevented())return;this.isShown=false;this.escape();this.resize();$(document).off("focusin.bs.modal");this.$element.removeClass("in").off("click.dismiss.bs.modal").off("mouseup.dismiss.bs.modal");this.$dialog.off("mousedown.dismiss.bs.modal");$.support.transition&&this.$element.hasClass("fade")?this.$element.one("bsTransitionEnd",$.proxy(this.hideModal,this)).emulateTransitionEnd(Modal.TRANSITION_DURATION):this.hideModal()};Modal.prototype.enforceFocus=function(){$(document).off("focusin.bs.modal").on("focusin.bs.modal",$.proxy(function(e){if(document!==e.target&&this.$element[0]!==e.target&&!this.$element.has(e.target).length){this.$element.trigger("focus")}},this))};Modal.prototype.escape=function(){if(this.isShown&&this.options.keyboard){this.$element.on("keydown.dismiss.bs.modal",$.proxy(function(e){e.which==27&&this.hide()},this))}else if(!this.isShown){this.$element.off("keydown.dismiss.bs.modal")}};Modal.prototype.resize=function(){if(this.isShown){$(window).on("resize.bs.modal",$.proxy(this.handleUpdate,this))}else{$(window).off("resize.bs.modal")}};Modal.prototype.hideModal=function(){var that=this;this.$element.hide();this.backdrop(function(){that.$body.removeClass("modal-open");that.resetAdjustments();that.resetScrollbar();that.$element.trigger("hidden.bs.modal")})};Modal.prototype.removeBackdrop=function(){this.$backdrop&&this.$backdrop.remove();this.$backdrop=null};Modal.prototype.backdrop=function(callback){var that=this;var animate=this.$element.hasClass("fade")?"fade":"";if(this.isShown&&this.options.backdrop){var doAnimate=$.support.transition&&animate;this.$backdrop=$(document.createElement("div")).addClass("modal-backdrop "+animate).appendTo(this.$body);this.$element.on("click.dismiss.bs.modal",$.proxy(function(e){if(this.ignoreBackdropClick){this.ignoreBackdropClick=false;return}if(e.target!==e.currentTarget)return;this.options.backdrop=="static"?this.$element[0].focus():this.hide()},this));if(doAnimate)this.$backdrop[0].offsetWidth;this.$backdrop.addClass("in");if(!callback)return;doAnimate?this.$backdrop.one("bsTransitionEnd",callback).emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION):callback()}else if(!this.isShown&&this.$backdrop){this.$backdrop.removeClass("in");var callbackRemove=function(){that.removeBackdrop();callback&&callback()};$.support.transition&&this.$element.hasClass("fade")?this.$backdrop.one("bsTransitionEnd",callbackRemove).emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION):callbackRemove()}else if(callback){callback()}};Modal.prototype.handleUpdate=function(){this.adjustDialog()};Modal.prototype.adjustDialog=function(){var modalIsOverflowing=this.$element[0].scrollHeight>document.documentElement.clientHeight;this.$element.css({paddingLeft:!this.bodyIsOverflowing&&modalIsOverflowing?this.scrollbarWidth:"",paddingRight:this.bodyIsOverflowing&&!modalIsOverflowing?this.scrollbarWidth:""})};Modal.prototype.resetAdjustments=function(){this.$element.css({paddingLeft:"",paddingRight:""})};Modal.prototype.checkScrollbar=function(){var fullWindowWidth=window.innerWidth;if(!fullWindowWidth){var documentElementRect=document.documentElement.getBoundingClientRect();fullWindowWidth=documentElementRect.right-Math.abs(documentElementRect.left)}this.bodyIsOverflowing=document.body.clientWidth<fullWindowWidth;this.scrollbarWidth=this.measureScrollbar()};Modal.prototype.setScrollbar=function(){var bodyPad=parseInt(this.$body.css("padding-right")||0,10);this.originalBodyPad=document.body.style.paddingRight||"";var scrollbarWidth=this.scrollbarWidth;if(this.bodyIsOverflowing){this.$body.css("padding-right",bodyPad+scrollbarWidth);$(this.fixedContent).each(function(index,element){var actualPadding=element.style.paddingRight;var calculatedPadding=$(element).css("padding-right");$(element).data("padding-right",actualPadding).css("padding-right",parseFloat(calculatedPadding)+scrollbarWidth+"px")})}};Modal.prototype.resetScrollbar=function(){this.$body.css("padding-right",this.originalBodyPad);$(this.fixedContent).each(function(index,element){var padding=$(element).data("padding-right");$(element).removeData("padding-right");element.style.paddingRight=padding?padding:""})};Modal.prototype.measureScrollbar=function(){var scrollDiv=document.createElement("div");scrollDiv.className="modal-scrollbar-measure";this.$body.append(scrollDiv);var scrollbarWidth=scrollDiv.offsetWidth-scrollDiv.clientWidth;this.$body[0].removeChild(scrollDiv);return scrollbarWidth};function Plugin(option,_relatedTarget){return this.each(function(){var $this=$(this);var data=$this.data("bs.modal");var options=$.extend({},Modal.DEFAULTS,$this.data(),typeof option=="object"&&option);if(!data)$this.data("bs.modal",data=new Modal(this,options));if(typeof option=="string")data[option](_relatedTarget);else if(options.show)data.show(_relatedTarget)})}var old=$.fn.modal;$.fn.modal=Plugin;$.fn.modal.Constructor=Modal;$.fn.modal.noConflict=function(){$.fn.modal=old;return this};$(document).on("click.bs.modal.data-api",'[data-toggle="modal"]',function(e){var $this=$(this);var href=$this.attr("href");var target=$this.attr("data-target")||href&&href.replace(/.*(?=#[^\s]+$)/,"");var $target=$(document).find(target);var option=$target.data("bs.modal")?"toggle":$.extend({remote:!/#/.test(href)&&href},$target.data(),$this.data());if($this.is("a"))e.preventDefault();$target.one("show.bs.modal",function(showEvent){if(showEvent.isDefaultPrevented())return;$target.one("hidden.bs.modal",function(){$this.is(":visible")&&$this.trigger("focus")})});Plugin.call($target,option,this)})}(jQuery);+function($){"use strict";var DISALLOWED_ATTRIBUTES=["sanitize","whiteList","sanitizeFn"];var uriAttrs=["background","cite","href","itemtype","longdesc","poster","src","xlink:href"];var ARIA_ATTRIBUTE_PATTERN=/^aria-[\w-]*$/i;var DefaultWhitelist={"*":["class","dir","id","lang","role",ARIA_ATTRIBUTE_PATTERN],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],div:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]};var SAFE_URL_PATTERN=/^(?:(?:https?|mailto|ftp|tel|file):|[^&:/?#]*(?:[/?#]|$))/gi;var DATA_URL_PATTERN=/^data:(?:image\/(?:bmp|gif|jpeg|jpg|png|tiff|webp)|video\/(?:mpeg|mp4|ogg|webm)|audio\/(?:mp3|oga|ogg|opus));base64,[a-z0-9+/]+=*$/i;function allowedAttribute(attr,allowedAttributeList){var attrName=attr.nodeName.toLowerCase();if($.inArray(attrName,allowedAttributeList)!==-1){if($.inArray(attrName,uriAttrs)!==-1){return Boolean(attr.nodeValue.match(SAFE_URL_PATTERN)||attr.nodeValue.match(DATA_URL_PATTERN))}return true}var regExp=$(allowedAttributeList).filter(function(index,value){return value instanceof RegExp});for(var i=0,l=regExp.length;i<l;i++){if(attrName.match(regExp[i])){return true}}return false}function sanitizeHtml(unsafeHtml,whiteList,sanitizeFn){if(unsafeHtml.length===0){return unsafeHtml}if(sanitizeFn&&typeof sanitizeFn==="function"){return sanitizeFn(unsafeHtml)}if(!document.implementation||!document.implementation.createHTMLDocument){return unsafeHtml}var createdDocument=document.implementation.createHTMLDocument("sanitization");createdDocument.body.innerHTML=unsafeHtml;var whitelistKeys=$.map(whiteList,function(el,i){return i});var elements=$(createdDocument.body).find("*");for(var i=0,len=elements.length;i<len;i++){var el=elements[i];var elName=el.nodeName.toLowerCase();if($.inArray(elName,whitelistKeys)===-1){el.parentNode.removeChild(el);continue}var attributeList=$.map(el.attributes,function(el){return el});var whitelistedAttributes=[].concat(whiteList["*"]||[],whiteList[elName]||[]);for(var j=0,len2=attributeList.length;j<len2;j++){if(!allowedAttribute(attributeList[j],whitelistedAttributes)){el.removeAttribute(attributeList[j].nodeName)}}}return createdDocument.body.innerHTML}var Tooltip=function(element,options){this.type=null;this.options=null;this.enabled=null;this.timeout=null;this.hoverState=null;this.$element=null;this.inState=null;this.init("tooltip",element,options)};Tooltip.VERSION="3.4.1";Tooltip.TRANSITION_DURATION=150;Tooltip.DEFAULTS={animation:true,placement:"top",selector:false,template:'<div class="tooltip" role="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>',trigger:"hover focus",title:"",delay:0,html:false,container:false,viewport:{selector:"body",padding:0},sanitize:true,sanitizeFn:null,whiteList:DefaultWhitelist};Tooltip.prototype.init=function(type,element,options){this.enabled=true;this.type=type;this.$element=$(element);this.options=this.getOptions(options);this.$viewport=this.options.viewport&&$(document).find($.isFunction(this.options.viewport)?this.options.viewport.call(this,this.$element):this.options.viewport.selector||this.options.viewport);this.inState={click:false,hover:false,focus:false};if(this.$element[0]instanceof document.constructor&&!this.options.selector){throw new Error("`selector` option must be specified when initializing "+this.type+" on the window.document object!")}var triggers=this.options.trigger.split(" ");for(var i=triggers.length;i--;){var trigger=triggers[i];if(trigger=="click"){this.$element.on("click."+this.type,this.options.selector,$.proxy(this.toggle,this))}else if(trigger!="manual"){var eventIn=trigger=="hover"?"mouseenter":"focusin";var eventOut=trigger=="hover"?"mouseleave":"focusout";this.$element.on(eventIn+"."+this.type,this.options.selector,$.proxy(this.enter,this));this.$element.on(eventOut+"."+this.type,this.options.selector,$.proxy(this.leave,this))}}this.options.selector?this._options=$.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()};Tooltip.prototype.getDefaults=function(){return Tooltip.DEFAULTS};Tooltip.prototype.getOptions=function(options){var dataAttributes=this.$element.data();for(var dataAttr in dataAttributes){if(dataAttributes.hasOwnProperty(dataAttr)&&$.inArray(dataAttr,DISALLOWED_ATTRIBUTES)!==-1){delete dataAttributes[dataAttr]}}options=$.extend({},this.getDefaults(),dataAttributes,options);if(options.delay&&typeof options.delay=="number"){options.delay={show:options.delay,hide:options.delay}}if(options.sanitize){options.template=sanitizeHtml(options.template,options.whiteList,options.sanitizeFn)}return options};Tooltip.prototype.getDelegateOptions=function(){var options={};var defaults=this.getDefaults();this._options&&$.each(this._options,function(key,value){if(defaults[key]!=value)options[key]=value});return options};Tooltip.prototype.enter=function(obj){var self=obj instanceof this.constructor?obj:$(obj.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(obj.currentTarget,this.getDelegateOptions());$(obj.currentTarget).data("bs."+this.type,self)}if(obj instanceof $.Event){self.inState[obj.type=="focusin"?"focus":"hover"]=true}if(self.tip().hasClass("in")||self.hoverState=="in"){self.hoverState="in";return}clearTimeout(self.timeout);self.hoverState="in";if(!self.options.delay||!self.options.delay.show)return self.show();self.timeout=setTimeout(function(){if(self.hoverState=="in")self.show()},self.options.delay.show)};Tooltip.prototype.isInStateTrue=function(){for(var key in this.inState){if(this.inState[key])return true}return false};Tooltip.prototype.leave=function(obj){var self=obj instanceof this.constructor?obj:$(obj.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(obj.currentTarget,this.getDelegateOptions());$(obj.currentTarget).data("bs."+this.type,self)}if(obj instanceof $.Event){self.inState[obj.type=="focusout"?"focus":"hover"]=false}if(self.isInStateTrue())return;clearTimeout(self.timeout);self.hoverState="out";if(!self.options.delay||!self.options.delay.hide)return self.hide();self.timeout=setTimeout(function(){if(self.hoverState=="out")self.hide()},self.options.delay.hide)};Tooltip.prototype.show=function(){var e=$.Event("show.bs."+this.type);if(this.hasContent()&&this.enabled){this.$element.trigger(e);var inDom=$.contains(this.$element[0].ownerDocument.documentElement,this.$element[0]);if(e.isDefaultPrevented()||!inDom)return;var that=this;var $tip=this.tip();var tipId=this.getUID(this.type);this.setContent();$tip.attr("id",tipId);this.$element.attr("aria-describedby",tipId);if(this.options.animation)$tip.addClass("fade");var placement=typeof this.options.placement=="function"?this.options.placement.call(this,$tip[0],this.$element[0]):this.options.placement;var autoToken=/\s?auto?\s?/i;var autoPlace=autoToken.test(placement);if(autoPlace)placement=placement.replace(autoToken,"")||"top";$tip.detach().css({top:0,left:0,display:"block"}).addClass(placement).data("bs."+this.type,this);this.options.container?$tip.appendTo($(document).find(this.options.container)):$tip.insertAfter(this.$element);this.$element.trigger("inserted.bs."+this.type);var pos=this.getPosition();var actualWidth=$tip[0].offsetWidth;var actualHeight=$tip[0].offsetHeight;if(autoPlace){var orgPlacement=placement;var viewportDim=this.getPosition(this.$viewport);placement=placement=="bottom"&&pos.bottom+actualHeight>viewportDim.bottom?"top":placement=="top"&&pos.top-actualHeight<viewportDim.top?"bottom":placement=="right"&&pos.right+actualWidth>viewportDim.width?"left":placement=="left"&&pos.left-actualWidth<viewportDim.left?"right":placement;$tip.removeClass(orgPlacement).addClass(placement)}var calculatedOffset=this.getCalculatedOffset(placement,pos,actualWidth,actualHeight);this.applyPlacement(calculatedOffset,placement);var complete=function(){var prevHoverState=that.hoverState;that.$element.trigger("shown.bs."+that.type);that.hoverState=null;if(prevHoverState=="out")that.leave(that)};$.support.transition&&this.$tip.hasClass("fade")?$tip.one("bsTransitionEnd",complete).emulateTransitionEnd(Tooltip.TRANSITION_DURATION):complete()}};Tooltip.prototype.applyPlacement=function(offset,placement){var $tip=this.tip();var width=$tip[0].offsetWidth;var height=$tip[0].offsetHeight;var marginTop=parseInt($tip.css("margin-top"),10);var marginLeft=parseInt($tip.css("margin-left"),10);if(isNaN(marginTop))marginTop=0;if(isNaN(marginLeft))marginLeft=0;offset.top+=marginTop;offset.left+=marginLeft;$.offset.setOffset($tip[0],$.extend({using:function(props){$tip.css({top:Math.round(props.top),left:Math.round(props.left)})}},offset),0);$tip.addClass("in");var actualWidth=$tip[0].offsetWidth;var actualHeight=$tip[0].offsetHeight;if(placement=="top"&&actualHeight!=height){offset.top=offset.top+height-actualHeight}var delta=this.getViewportAdjustedDelta(placement,offset,actualWidth,actualHeight);if(delta.left)offset.left+=delta.left;else offset.top+=delta.top;var isVertical=/top|bottom/.test(placement);var arrowDelta=isVertical?delta.left*2-width+actualWidth:delta.top*2-height+actualHeight;var arrowOffsetPosition=isVertical?"offsetWidth":"offsetHeight";$tip.offset(offset);this.replaceArrow(arrowDelta,$tip[0][arrowOffsetPosition],isVertical)};Tooltip.prototype.replaceArrow=function(delta,dimension,isVertical){this.arrow().css(isVertical?"left":"top",50*(1-delta/dimension)+"%").css(isVertical?"top":"left","")};Tooltip.prototype.setContent=function(){var $tip=this.tip();var title=this.getTitle();if(this.options.html){if(this.options.sanitize){title=sanitizeHtml(title,this.options.whiteList,this.options.sanitizeFn)}$tip.find(".tooltip-inner").html(title)}else{$tip.find(".tooltip-inner").text(title)}$tip.removeClass("fade in top bottom left right")};Tooltip.prototype.hide=function(callback){var that=this;var $tip=$(this.$tip);var e=$.Event("hide.bs."+this.type);function complete(){if(that.hoverState!="in")$tip.detach();if(that.$element){that.$element.removeAttr("aria-describedby").trigger("hidden.bs."+that.type)}callback&&callback()}this.$element.trigger(e);if(e.isDefaultPrevented())return;$tip.removeClass("in");$.support.transition&&$tip.hasClass("fade")?$tip.one("bsTransitionEnd",complete).emulateTransitionEnd(Tooltip.TRANSITION_DURATION):complete();this.hoverState=null;return this};Tooltip.prototype.fixTitle=function(){var $e=this.$element;if($e.attr("title")||typeof $e.attr("data-original-title")!="string"){$e.attr("data-original-title",$e.attr("title")||"").attr("title","")}};Tooltip.prototype.hasContent=function(){return this.getTitle()};Tooltip.prototype.getPosition=function($element){$element=$element||this.$element;var el=$element[0];var isBody=el.tagName=="BODY";var elRect=el.getBoundingClientRect();if(elRect.width==null){elRect=$.extend({},elRect,{width:elRect.right-elRect.left,height:elRect.bottom-elRect.top})}var isSvg=window.SVGElement&&el instanceof window.SVGElement;var elOffset=isBody?{top:0,left:0}:isSvg?null:$element.offset();var scroll={scroll:isBody?document.documentElement.scrollTop||document.body.scrollTop:$element.scrollTop()};var outerDims=isBody?{width:$(window).width(),height:$(window).height()}:null;return $.extend({},elRect,scroll,outerDims,elOffset)};Tooltip.prototype.getCalculatedOffset=function(placement,pos,actualWidth,actualHeight){return placement=="bottom"?{top:pos.top+pos.height,left:pos.left+pos.width/2-actualWidth/2}:placement=="top"?{top:pos.top-actualHeight,left:pos.left+pos.width/2-actualWidth/2}:placement=="left"?{top:pos.top+pos.height/2-actualHeight/2,left:pos.left-actualWidth}:{top:pos.top+pos.height/2-actualHeight/2,left:pos.left+pos.width}};Tooltip.prototype.getViewportAdjustedDelta=function(placement,pos,actualWidth,actualHeight){var delta={top:0,left:0};if(!this.$viewport)return delta;var viewportPadding=this.options.viewport&&this.options.viewport.padding||0;var viewportDimensions=this.getPosition(this.$viewport);if(/right|left/.test(placement)){var topEdgeOffset=pos.top-viewportPadding-viewportDimensions.scroll;var bottomEdgeOffset=pos.top+viewportPadding-viewportDimensions.scroll+actualHeight;if(topEdgeOffset<viewportDimensions.top){delta.top=viewportDimensions.top-topEdgeOffset}else if(bottomEdgeOffset>viewportDimensions.top+viewportDimensions.height){delta.top=viewportDimensions.top+viewportDimensions.height-bottomEdgeOffset}}else{var leftEdgeOffset=pos.left-viewportPadding;var rightEdgeOffset=pos.left+viewportPadding+actualWidth;if(leftEdgeOffset<viewportDimensions.left){delta.left=viewportDimensions.left-leftEdgeOffset}else if(rightEdgeOffset>viewportDimensions.right){delta.left=viewportDimensions.left+viewportDimensions.width-rightEdgeOffset}}return delta};Tooltip.prototype.getTitle=function(){var title;var $e=this.$element;var o=this.options;title=$e.attr("data-original-title")||(typeof o.title=="function"?o.title.call($e[0]):o.title);return title};Tooltip.prototype.getUID=function(prefix){do{prefix+=~~(Math.random()*1e6)}while(document.getElementById(prefix));return prefix};Tooltip.prototype.tip=function(){if(!this.$tip){this.$tip=$(this.options.template);if(this.$tip.length!=1){throw new Error(this.type+" `template` option must consist of exactly 1 top-level element!")}}return this.$tip};Tooltip.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")};Tooltip.prototype.enable=function(){this.enabled=true};Tooltip.prototype.disable=function(){this.enabled=false};Tooltip.prototype.toggleEnabled=function(){this.enabled=!this.enabled};Tooltip.prototype.toggle=function(e){var self=this;if(e){self=$(e.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(e.currentTarget,this.getDelegateOptions());$(e.currentTarget).data("bs."+this.type,self)}}if(e){self.inState.click=!self.inState.click;if(self.isInStateTrue())self.enter(self);else self.leave(self)}else{self.tip().hasClass("in")?self.leave(self):self.enter(self)}};Tooltip.prototype.destroy=function(){var that=this;clearTimeout(this.timeout);this.hide(function(){that.$element.off("."+that.type).removeData("bs."+that.type);if(that.$tip){that.$tip.detach()}that.$tip=null;that.$arrow=null;that.$viewport=null;that.$element=null})};Tooltip.prototype.sanitizeHtml=function(unsafeHtml){return sanitizeHtml(unsafeHtml,this.options.whiteList,this.options.sanitizeFn)};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.tooltip");var options=typeof option=="object"&&option;if(!data&&/destroy|hide/.test(option))return;if(!data)$this.data("bs.tooltip",data=new Tooltip(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.tooltip;$.fn.tooltip=Plugin;$.fn.tooltip.Constructor=Tooltip;$.fn.tooltip.noConflict=function(){$.fn.tooltip=old;return this}}(jQuery);+function($){"use strict";var Popover=function(element,options){this.init("popover",element,options)};if(!$.fn.tooltip)throw new Error("Popover requires tooltip.js");Popover.VERSION="3.4.1";Popover.DEFAULTS=$.extend({},$.fn.tooltip.Constructor.DEFAULTS,{placement:"right",trigger:"click",content:"",template:'<div class="popover" role="tooltip"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'});Popover.prototype=$.extend({},$.fn.tooltip.Constructor.prototype);Popover.prototype.constructor=Popover;Popover.prototype.getDefaults=function(){return Popover.DEFAULTS};Popover.prototype.setContent=function(){var $tip=this.tip();var title=this.getTitle();var content=this.getContent();if(this.options.html){var typeContent=typeof content;if(this.options.sanitize){title=this.sanitizeHtml(title);if(typeContent==="string"){content=this.sanitizeHtml(content)}}$tip.find(".popover-title").html(title);$tip.find(".popover-content").children().detach().end()[typeContent==="string"?"html":"append"](content)}else{$tip.find(".popover-title").text(title);$tip.find(".popover-content").children().detach().end().text(content)}$tip.removeClass("fade top bottom left right in");if(!$tip.find(".popover-title").html())$tip.find(".popover-title").hide()};Popover.prototype.hasContent=function(){return this.getTitle()||this.getContent()};Popover.prototype.getContent=function(){var $e=this.$element;var o=this.options;return $e.attr("data-content")||(typeof o.content=="function"?o.content.call($e[0]):o.content)};Popover.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".arrow")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.popover");var options=typeof option=="object"&&option;if(!data&&/destroy|hide/.test(option))return;if(!data)$this.data("bs.popover",data=new Popover(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.popover;$.fn.popover=Plugin;$.fn.popover.Constructor=Popover;$.fn.popover.noConflict=function(){$.fn.popover=old;return this}}(jQuery);+function($){"use strict";function ScrollSpy(element,options){this.$body=$(document.body);this.$scrollElement=$(element).is(document.body)?$(window):$(element);this.options=$.extend({},ScrollSpy.DEFAULTS,options);this.selector=(this.options.target||"")+" .nav li > a";this.offsets=[];this.targets=[];this.activeTarget=null;this.scrollHeight=0;this.$scrollElement.on("scroll.bs.scrollspy",$.proxy(this.process,this));this.refresh();this.process()}ScrollSpy.VERSION="3.4.1";ScrollSpy.DEFAULTS={offset:10};ScrollSpy.prototype.getScrollHeight=function(){return this.$scrollElement[0].scrollHeight||Math.max(this.$body[0].scrollHeight,document.documentElement.scrollHeight)};ScrollSpy.prototype.refresh=function(){var that=this;var offsetMethod="offset";var offsetBase=0;this.offsets=[];this.targets=[];this.scrollHeight=this.getScrollHeight();if(!$.isWindow(this.$scrollElement[0])){offsetMethod="position";offsetBase=this.$scrollElement.scrollTop()}this.$body.find(this.selector).map(function(){var $el=$(this);var href=$el.data("target")||$el.attr("href");var $href=/^#./.test(href)&&$(href);return $href&&$href.length&&$href.is(":visible")&&[[$href[offsetMethod]().top+offsetBase,href]]||null}).sort(function(a,b){return a[0]-b[0]}).each(function(){that.offsets.push(this[0]);that.targets.push(this[1])})};ScrollSpy.prototype.process=function(){var scrollTop=this.$scrollElement.scrollTop()+this.options.offset;var scrollHeight=this.getScrollHeight();var maxScroll=this.options.offset+scrollHeight-this.$scrollElement.height();var offsets=this.offsets;var targets=this.targets;var activeTarget=this.activeTarget;var i;if(this.scrollHeight!=scrollHeight){this.refresh()}if(scrollTop>=maxScroll){return activeTarget!=(i=targets[targets.length-1])&&this.activate(i)}if(activeTarget&&scrollTop<offsets[0]){this.activeTarget=null;return this.clear()}for(i=offsets.length;i--;){activeTarget!=targets[i]&&scrollTop>=offsets[i]&&(offsets[i+1]===undefined||scrollTop<offsets[i+1])&&this.activate(targets[i])}};ScrollSpy.prototype.activate=function(target){this.activeTarget=target;this.clear();var selector=this.selector+'[data-target="'+target+'"],'+this.selector+'[href="'+target+'"]';var active=$(selector).parents("li").addClass("active");if(active.parent(".dropdown-menu").length){active=active.closest("li.dropdown").addClass("active")}active.trigger("activate.bs.scrollspy")};ScrollSpy.prototype.clear=function(){$(this.selector).parentsUntil(this.options.target,".active").removeClass("active")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.scrollspy");var options=typeof option=="object"&&option;if(!data)$this.data("bs.scrollspy",data=new ScrollSpy(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.scrollspy;$.fn.scrollspy=Plugin;$.fn.scrollspy.Constructor=ScrollSpy;$.fn.scrollspy.noConflict=function(){$.fn.scrollspy=old;return this};$(window).on("load.bs.scrollspy.data-api",function(){$('[data-spy="scroll"]').each(function(){var $spy=$(this);Plugin.call($spy,$spy.data())})})}(jQuery);+function($){"use strict";var Tab=function(element){this.element=$(element)};Tab.VERSION="3.4.1";Tab.TRANSITION_DURATION=150;Tab.prototype.show=function(){var $this=this.element;var $ul=$this.closest("ul:not(.dropdown-menu)");var selector=$this.data("target");if(!selector){selector=$this.attr("href");selector=selector&&selector.replace(/.*(?=#[^\s]*$)/,"")}if($this.parent("li").hasClass("active"))return;var $previous=$ul.find(".active:last a");var hideEvent=$.Event("hide.bs.tab",{relatedTarget:$this[0]});var showEvent=$.Event("show.bs.tab",{relatedTarget:$previous[0]});$previous.trigger(hideEvent);$this.trigger(showEvent);if(showEvent.isDefaultPrevented()||hideEvent.isDefaultPrevented())return;var $target=$(document).find(selector);this.activate($this.closest("li"),$ul);this.activate($target,$target.parent(),function(){$previous.trigger({type:"hidden.bs.tab",relatedTarget:$this[0]});$this.trigger({type:"shown.bs.tab",relatedTarget:$previous[0]})})};Tab.prototype.activate=function(element,container,callback){var $active=container.find("> .active");var transition=callback&&$.support.transition&&($active.length&&$active.hasClass("fade")||!!container.find("> .fade").length);function next(){$active.removeClass("active").find("> .dropdown-menu > .active").removeClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",false);element.addClass("active").find('[data-toggle="tab"]').attr("aria-expanded",true);if(transition){element[0].offsetWidth;element.addClass("in")}else{element.removeClass("fade")}if(element.parent(".dropdown-menu").length){element.closest("li.dropdown").addClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",true)}callback&&callback()}$active.length&&transition?$active.one("bsTransitionEnd",next).emulateTransitionEnd(Tab.TRANSITION_DURATION):next();$active.removeClass("in")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.tab");if(!data)$this.data("bs.tab",data=new Tab(this));if(typeof option=="string")data[option]()})}var old=$.fn.tab;$.fn.tab=Plugin;$.fn.tab.Constructor=Tab;$.fn.tab.noConflict=function(){$.fn.tab=old;return this};var clickHandler=function(e){e.preventDefault();Plugin.call($(this),"show")};$(document).on("click.bs.tab.data-api",'[data-toggle="tab"]',clickHandler).on("click.bs.tab.data-api",'[data-toggle="pill"]',clickHandler)}(jQuery);+function($){"use strict";var Affix=function(element,options){this.options=$.extend({},Affix.DEFAULTS,options);var target=this.options.target===Affix.DEFAULTS.target?$(this.options.target):$(document).find(this.options.target);this.$target=target.on("scroll.bs.affix.data-api",$.proxy(this.checkPosition,this)).on("click.bs.affix.data-api",$.proxy(this.checkPositionWithEventLoop,this));this.$element=$(element);this.affixed=null;this.unpin=null;this.pinnedOffset=null;this.checkPosition()};Affix.VERSION="3.4.1";Affix.RESET="affix affix-top affix-bottom";Affix.DEFAULTS={offset:0,target:window};Affix.prototype.getState=function(scrollHeight,height,offsetTop,offsetBottom){var scrollTop=this.$target.scrollTop();var position=this.$element.offset();var targetHeight=this.$target.height();if(offsetTop!=null&&this.affixed=="top")return scrollTop<offsetTop?"top":false;if(this.affixed=="bottom"){if(offsetTop!=null)return scrollTop+this.unpin<=position.top?false:"bottom";return scrollTop+targetHeight<=scrollHeight-offsetBottom?false:"bottom"}var initializing=this.affixed==null;var colliderTop=initializing?scrollTop:position.top;var colliderHeight=initializing?targetHeight:height;if(offsetTop!=null&&scrollTop<=offsetTop)return"top";if(offsetBottom!=null&&colliderTop+colliderHeight>=scrollHeight-offsetBottom)return"bottom";return false};Affix.prototype.getPinnedOffset=function(){if(this.pinnedOffset)return this.pinnedOffset;this.$element.removeClass(Affix.RESET).addClass("affix");var scrollTop=this.$target.scrollTop();var position=this.$element.offset();return this.pinnedOffset=position.top-scrollTop};Affix.prototype.checkPositionWithEventLoop=function(){setTimeout($.proxy(this.checkPosition,this),1)};Affix.prototype.checkPosition=function(){if(!this.$element.is(":visible"))return;var height=this.$element.height();var offset=this.options.offset;var offsetTop=offset.top;var offsetBottom=offset.bottom;var scrollHeight=Math.max($(document).height(),$(document.body).height());if(typeof offset!="object")offsetBottom=offsetTop=offset;if(typeof offsetTop=="function")offsetTop=offset.top(this.$element);if(typeof offsetBottom=="function")offsetBottom=offset.bottom(this.$element);var affix=this.getState(scrollHeight,height,offsetTop,offsetBottom);if(this.affixed!=affix){if(this.unpin!=null)this.$element.css("top","");var affixType="affix"+(affix?"-"+affix:"");var e=$.Event(affixType+".bs.affix");this.$element.trigger(e);if(e.isDefaultPrevented())return;this.affixed=affix;this.unpin=affix=="bottom"?this.getPinnedOffset():null;this.$element.removeClass(Affix.RESET).addClass(affixType).trigger(affixType.replace("affix","affixed")+".bs.affix")}if(affix=="bottom"){this.$element.offset({top:scrollHeight-height-offsetBottom})}};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.affix");var options=typeof option=="object"&&option;if(!data)$this.data("bs.affix",data=new Affix(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.affix;$.fn.affix=Plugin;$.fn.affix.Constructor=Affix;$.fn.affix.noConflict=function(){$.fn.affix=old;return this};$(window).on("load",function(){$('[data-spy="affix"]').each(function(){var $spy=$(this);var data=$spy.data();data.offset=data.offset||{};if(data.offsetBottom!=null)data.offset.bottom=data.offsetBottom;if(data.offsetTop!=null)data.offset.top=data.offsetTop;Plugin.call($spy,data)})})}(jQuery);</script>
+<script>!function(e, i) {
+    var l, m, t, n = e.html5 || {}, r = /^<|^(?:button|map|select|textarea|object|iframe|option|optgroup)$/i, a = /^(?:a|b|code|div|fieldset|h1|h2|h3|h4|h5|h6|i|label|li|ol|p|q|span|strong|style|table|tbody|td|th|tr|ul)$/i, o = "_html5shiv", c = 0, s = {};
+    try {
+        var d = i.createElement("a");
+        d.innerHTML = "<xyz></xyz>", l = "hidden" in d, m = 1 == d.childNodes.length || (i.createElement("a"), 
+        void 0 === (t = i.createDocumentFragment()).cloneNode || void 0 === t.createDocumentFragment || void 0 === t.createElement);
+    } catch (e) {
+        m = l = !0;
+    }
+    function u() {
+        var e = p.elements;
+        return "string" == typeof e ? e.split(" ") : e;
+    }
+    function h(e) {
+        var t = s[e[o]];
+        return t || (t = {}, c++, e[o] = c, s[c] = t), t;
+    }
+    function f(e, t, n) {
+        return t = t || i, m ? t.createElement(e) : !(t = (n = n || h(t)).cache[e] ? n.cache[e].cloneNode() : a.test(e) ? (n.cache[e] = n.createElem(e)).cloneNode() : n.createElem(e)).canHaveChildren || r.test(e) || t.tagUrn ? t : n.frag.appendChild(t);
+    }
+    function g(e) {
+        var t, n, r, a, o, c = h(e = e || i);
+        return !p.shivCSS || l || c.hasCSS || (c.hasCSS = (n = "article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}mark{background:#FF0;color:#000}template{display:none}", 
+        r = (t = e).createElement("p"), t = t.getElementsByTagName("head")[0] || t.documentElement, 
+        r.innerHTML = "x<style>" + n + "</style>", !!t.insertBefore(r.lastChild, t.firstChild))), 
+        m || (a = e, (o = c).cache || (o.cache = {}, o.createElem = a.createElement, 
+        o.createFrag = a.createDocumentFragment, o.frag = o.createFrag()), a.createElement = function(e) {
+            return p.shivMethods ? f(e, a, o) : o.createElem(e);
+        }, a.createDocumentFragment = Function("h,f", "return function(){var n=f.cloneNode(),c=n.createElement;h.shivMethods&&(" + u().join().replace(/[\w\-:]+/g, function(e) {
+            return o.createElem(e), o.frag.createElement(e), 'c("' + e + '")';
+        }) + ");return n}")(p, o.frag)), e;
+    }
+    var p = {
+        elements: n.elements || "abbr article aside audio bdi canvas data datalist details dialog figcaption figure footer header hgroup main mark meter nav output picture progress section summary template time video",
+        version: "3.7.3",
+        shivCSS: !1 !== n.shivCSS,
+        supportsUnknownElements: m,
+        shivMethods: !1 !== n.shivMethods,
+        type: "default",
+        shivDocument: g,
+        createElement: f,
+        createDocumentFragment: function(e, t) {
+            if (e = e || i, m) return e.createDocumentFragment();
+            for (var n = (t = t || h(e)).frag.cloneNode(), r = 0, a = u(), o = a.length; r < o; r++) n.createElement(a[r]);
+            return n;
+        },
+        addElements: function(e, t) {
+            var n = p.elements;
+            "string" != typeof n && (n = n.join(" ")), "string" != typeof e && (e = e.join(" ")), 
+            p.elements = n + " " + e, g(t);
+        }
+    };
+    e.html5 = p, g(i), "object" == typeof module && module.exports && (module.exports = p);
+}("undefined" != typeof window ? window : this, document);</script>
+<script>(function(c){c.matchMedia=c.matchMedia||function(c,e){var h,r=c.documentElement,t=r.firstElementChild||r.firstChild,m=c.createElement("body"),f=c.createElement("div");f.id="mq-test-1";f.style.cssText="position:absolute;top:-100em";m.style.background="none";m.appendChild(f);return function(c){f.innerHTML='&shy;<style media="'+c+'"> #mq-test-1 { width: 42px; }</style>';r.insertBefore(m,t);h=42===f.offsetWidth;r.removeChild(m);return{matches:h,media:c}}}(c.document)})(this);
+(function(c){function D(){y(!0)}var e={};c.respond=e;e.update=function(){};var h=[],r=function(){var d=!1;try{d=new c.XMLHttpRequest}catch(b){d=new c.ActiveXObject("Microsoft.XMLHTTP")}return function(){return d}}(),t=function(d,b){var a=r();a&&(a.open("GET",d,!0),a.onreadystatechange=function(){4!==a.readyState||200!==a.status&&304!==a.status||b(a.responseText)},4!==a.readyState&&a.send(null))},m=function(d){return d.replace(e.regex.minmaxwh,"").match(e.regex.other)};e.ajax=t;e.queue=h;e.unsupportedmq=
+m;e.regex={media:/@media[^\{]+\{([^\{\}]*\{[^\}\{]*\})+/gi,keyframes:/@(?:\-(?:o|moz|webkit)\-)?keyframes[^\{]+\{(?:[^\{\}]*\{[^\}\{]*\})+[^\}]*\}/gi,comments:/\/\*[^*]*\*+([^/][^*]*\*+)*\//gi,urls:/(url\()['"]?([^\/\)'"][^:\)'"]+)['"]?(\))/g,findStyles:/@media *([^\{]+)\{([\S\s]+?)$/,only:/(only\s+)?([a-zA-Z]+)\s?/,minw:/\(\s*min\-width\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/,maxw:/\(\s*max\-width\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/,minmaxwh:/\(\s*m(in|ax)\-(height|width)\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/gi,
+other:/\([^\)]*\)/g};e.mediaQueriesSupported=c.matchMedia&&null!==c.matchMedia("only all")&&c.matchMedia("only all").matches;if(!e.mediaQueriesSupported){var f=c.document,k=f.documentElement,u=[],v=[],n=[],z={},w=f.getElementsByTagName("head")[0]||k,I=f.getElementsByTagName("base")[0],x=w.getElementsByTagName("link"),A,E,B,C=function(){var d,b=f.createElement("div"),a=f.body,c=k.style.fontSize,e=a&&a.style.fontSize,g=!1;b.style.cssText="position:absolute;font-size:1em;width:1em";a||(a=g=f.createElement("body"),
+a.style.background="none");k.style.fontSize="100%";a.style.fontSize="100%";a.appendChild(b);g&&k.insertBefore(a,k.firstChild);d=b.offsetWidth;g?k.removeChild(a):a.removeChild(b);k.style.fontSize=c;e&&(a.style.fontSize=e);return d=B=parseFloat(d)},y=function(d){var b=k.clientWidth,a="CSS1Compat"===f.compatMode&&b||f.body.clientWidth||b,b={},e=x[x.length-1],q=(new Date).getTime();if(d&&A&&30>q-A)c.clearTimeout(E),E=c.setTimeout(y,30);else{A=q;for(var g in u)if(u.hasOwnProperty(g)){d=u[g];var q=d.minw,
+s=d.maxw,l=null===q,h=null===s;q&&(q=parseFloat(q)*(-1<q.indexOf("em")?B||C():1));s&&(s=parseFloat(s)*(-1<s.indexOf("em")?B||C():1));if(!d.hasquery||(!l||!h)&&(l||a>=q)&&(h||a<=s))b[d.media]||(b[d.media]=[]),b[d.media].push(v[d.rules])}for(var p in n)n.hasOwnProperty(p)&&n[p]&&n[p].parentNode===w&&w.removeChild(n[p]);n.length=0;for(var m in b)b.hasOwnProperty(m)&&(g=f.createElement("style"),p=b[m].join("\n"),g.type="text/css",g.media=m,w.insertBefore(g,e.nextSibling),g.styleSheet?g.styleSheet.cssText=
+p:g.appendChild(f.createTextNode(p)),n.push(g))}},F=function(d,b,a){var c=d.replace(e.regex.comments,"").replace(e.regex.keyframes,"").match(e.regex.media),f=c&&c.length||0;b=b.substring(0,b.lastIndexOf("/"));var g=!f&&a;b.length&&(b+="/");g&&(f=1);for(var h=0;h<f;h++){var l,k,p;g?(l=a,v.push(d.replace(e.regex.urls,"$1"+b+"$2$3"))):(l=c[h].match(e.regex.findStyles)&&RegExp.$1,v.push(RegExp.$2&&RegExp.$2.replace(e.regex.urls,"$1"+b+"$2$3")));k=l.split(",");p=k.length;for(var n=0;n<p;n++)l=k[n],m(l)||
+u.push({media:l.split("(")[0].match(e.regex.only)&&RegExp.$2||"all",rules:v.length-1,hasquery:-1<l.indexOf("("),minw:l.match(e.regex.minw)&&parseFloat(RegExp.$1)+(RegExp.$2||""),maxw:l.match(e.regex.maxw)&&parseFloat(RegExp.$1)+(RegExp.$2||"")})}y()},G=function(){if(h.length){var d=h.shift();t(d.href,function(b){F(b,d.href,d.media);z[d.href]=!0;c.setTimeout(function(){G()},0)})}},H=function(){for(var d=0;d<x.length;d++){var b=x[d],a=b.href,e=b.media,f=b.rel&&"stylesheet"===b.rel.toLowerCase();if(a&&
+f&&!z[a])if(b.styleSheet&&b.styleSheet.rawCssText)F(b.styleSheet.rawCssText,a,e),z[a]=!0;else if(!/^([a-zA-Z:]*\/\/)/.test(a)&&!I||a.replace(RegExp.$1,"").split("/")[0]===c.location.host)"//"===a.substring(0,2)&&(a=c.location.protocol+a),h.push({href:a,media:e})}G()};H();e.update=H;e.getEmValue=C;c.addEventListener?c.addEventListener("resize",D,!1):c.attachEvent&&c.attachEvent("onresize",D)}})(this);
+</script>
+<style>h1 {font-size: 34px;}
+       h1.title {font-size: 38px;}
+       h2 {font-size: 30px;}
+       h3 {font-size: 24px;}
+       h4 {font-size: 18px;}
+       h5 {font-size: 16px;}
+       h6 {font-size: 12px;}
+       code {color: inherit; background-color: rgba(0, 0, 0, 0.04);}
+       pre:not([class]) { background-color: white }</style>
+<script>
+
+/**
+ * jQuery Plugin: Sticky Tabs
+ *
+ * @author Aidan Lister <aidan@php.net>
+ * adapted by Ruben Arslan to activate parent tabs too
+ * http://www.aidanlister.com/2014/03/persisting-the-tab-state-in-bootstrap/
+ */
+(function($) {
+  "use strict";
+  $.fn.rmarkdownStickyTabs = function() {
+    var context = this;
+    // Show the tab corresponding with the hash in the URL, or the first tab
+    var showStuffFromHash = function() {
+      var hash = window.location.hash;
+      var selector = hash ? 'a[href="' + hash + '"]' : 'li.active > a';
+      var $selector = $(selector, context);
+      if($selector.data('toggle') === "tab") {
+        $selector.tab('show');
+        // walk up the ancestors of this element, show any hidden tabs
+        $selector.parents('.section.tabset').each(function(i, elm) {
+          var link = $('a[href="#' + $(elm).attr('id') + '"]');
+          if(link.data('toggle') === "tab") {
+            link.tab("show");
+          }
+        });
+      }
+    };
+
+
+    // Set the correct tab when the page loads
+    showStuffFromHash(context);
+
+    // Set the correct tab when a user uses their back/forward button
+    $(window).on('hashchange', function() {
+      showStuffFromHash(context);
+    });
+
+    // Change the URL when tabs are clicked
+    $('a', context).on('click', function(e) {
+      history.pushState(null, null, this.href);
+      showStuffFromHash(context);
+    });
+
+    return this;
+  };
+}(jQuery));
+
+window.buildTabsets = function(tocID) {
+
+  // build a tabset from a section div with the .tabset class
+  function buildTabset(tabset) {
+
+    // check for fade and pills options
+    var fade = tabset.hasClass("tabset-fade");
+    var pills = tabset.hasClass("tabset-pills");
+    var navClass = pills ? "nav-pills" : "nav-tabs";
+
+    // determine the heading level of the tabset and tabs
+    var match = tabset.attr('class').match(/level(\d) /);
+    if (match === null)
+      return;
+    var tabsetLevel = Number(match[1]);
+    var tabLevel = tabsetLevel + 1;
+
+    // find all subheadings immediately below
+    var tabs = tabset.find("div.section.level" + tabLevel);
+    if (!tabs.length)
+      return;
+
+    // create tablist and tab-content elements
+    var tabList = $('<ul class="nav ' + navClass + '" role="tablist"></ul>');
+    $(tabs[0]).before(tabList);
+    var tabContent = $('<div class="tab-content"></div>');
+    $(tabs[0]).before(tabContent);
+
+    // build the tabset
+    var activeTab = 0;
+    tabs.each(function(i) {
+
+      // get the tab div
+      var tab = $(tabs[i]);
+
+      // get the id then sanitize it for use with bootstrap tabs
+      var id = tab.attr('id');
+
+      // see if this is marked as the active tab
+      if (tab.hasClass('active'))
+        activeTab = i;
+
+      // remove any table of contents entries associated with
+      // this ID (since we'll be removing the heading element)
+      $("div#" + tocID + " li a[href='#" + id + "']").parent().remove();
+
+      // sanitize the id for use with bootstrap tabs
+      id = id.replace(/[.\/?&!#<>]/g, '').replace(/\s/g, '_');
+      tab.attr('id', id);
+
+      // get the heading element within it, grab it's text, then remove it
+      var heading = tab.find('h' + tabLevel + ':first');
+      var headingText = heading.html();
+      heading.remove();
+
+      // build and append the tab list item
+      var a = $('<a role="tab" data-toggle="tab">' + headingText + '</a>');
+      a.attr('href', '#' + id);
+      a.attr('aria-controls', id);
+      var li = $('<li role="presentation"></li>');
+      li.append(a);
+      tabList.append(li);
+
+      // set it's attributes
+      tab.attr('role', 'tabpanel');
+      tab.addClass('tab-pane');
+      tab.addClass('tabbed-pane');
+      if (fade)
+        tab.addClass('fade');
+
+      // move it into the tab content div
+      tab.detach().appendTo(tabContent);
+    });
+
+    // set active tab
+    $(tabList.children('li')[activeTab]).addClass('active');
+    var active = $(tabContent.children('div.section')[activeTab]);
+    active.addClass('active');
+    if (fade)
+      active.addClass('in');
+
+    if (tabset.hasClass("tabset-sticky"))
+      tabset.rmarkdownStickyTabs();
+  }
+
+  // convert section divs with the .tabset class to tabsets
+  var tabsets = $("div.section.tabset");
+  tabsets.each(function(i) {
+    buildTabset($(tabsets[i]));
+  });
+};
+
+</script>
+<style type="text/css">.hljs-literal {
+color: #990073;
+}
+.hljs-number {
+color: #099;
+}
+.hljs-comment {
+color: #998;
+font-style: italic;
+}
+.hljs-keyword {
+color: #900;
+font-weight: bold;
+}
+.hljs-string {
+color: #d14;
+}
+</style>
+<script>var hljs=function(factory){var globalObject=typeof window==="object"&&window||typeof self==="object"&&self;if(typeof exports!=="undefined"&&!exports.nodeType){return factory(exports)}else if(globalObject){globalObject.hljs=factory({});if(typeof define==="function"&&define.amd){define([],function(){return globalObject.hljs})}return globalObject.hljs}}(function(hljs){var showedUpgradeWarning=false;var ArrayProto=[],objectKeys=Object.keys;var languages=Object.create(null),aliases=Object.create(null);var SAFE_MODE=true;var noHighlightRe=/^(no-?highlight|plain|text)$/i,languagePrefixRe=/\blang(?:uage)?-([\w-]+)\b/i,fixMarkupRe=/((^(<[^>]+>|\t|)+|(?:\n)))/gm;var API_REPLACES;var spanEndTag="</span>";var LANGUAGE_NOT_FOUND="Could not find the language '{}', did you forget to load/include a language module?";var options={hideUpgradeWarningAcceptNoSupportOrSecurityUpdates:false,classPrefix:"hljs-",tabReplace:null,useBR:false,languages:undefined};var COMMON_KEYWORDS="of and for in not or if then".split(" ");function escape(value){return value.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;")}function tag(node){return node.nodeName.toLowerCase()}function testRe(re,lexeme){var match=re&&re.exec(lexeme);return match&&match.index===0}function isNotHighlighted(language){return noHighlightRe.test(language)}function blockLanguage(block){var i,match,length,_class;var classes=block.className+" ";classes+=block.parentNode?block.parentNode.className:"";match=languagePrefixRe.exec(classes);if(match){var language=getLanguage(match[1]);if(!language){console.warn(LANGUAGE_NOT_FOUND.replace("{}",match[1]));console.warn("Falling back to no-highlight mode for this block.",block)}return language?match[1]:"no-highlight"}classes=classes.split(/\s+/);for(i=0,length=classes.length;i<length;i++){_class=classes[i];if(isNotHighlighted(_class)||getLanguage(_class)){return _class}}}function inherit(parent){var key;var result={};var objects=Array.prototype.slice.call(arguments,1);for(key in parent)result[key]=parent[key];objects.forEach(function(obj){for(key in obj)result[key]=obj[key]});return result}function nodeStream(node){var result=[];(function _nodeStream(node,offset){for(var child=node.firstChild;child;child=child.nextSibling){if(child.nodeType===3)offset+=child.nodeValue.length;else if(child.nodeType===1){result.push({event:"start",offset:offset,node:child});offset=_nodeStream(child,offset);if(!tag(child).match(/br|hr|img|input/)){result.push({event:"stop",offset:offset,node:child})}}}return offset})(node,0);return result}function mergeStreams(original,highlighted,value){var processed=0;var result="";var nodeStack=[];function selectStream(){if(!original.length||!highlighted.length){return original.length?original:highlighted}if(original[0].offset!==highlighted[0].offset){return original[0].offset<highlighted[0].offset?original:highlighted}return highlighted[0].event==="start"?original:highlighted}function open(node){function attr_str(a){return" "+a.nodeName+'="'+escape(a.value).replace(/"/g,"&quot;")+'"'}result+="<"+tag(node)+ArrayProto.map.call(node.attributes,attr_str).join("")+">"}function close(node){result+="</"+tag(node)+">"}function render(event){(event.event==="start"?open:close)(event.node)}while(original.length||highlighted.length){var stream=selectStream();result+=escape(value.substring(processed,stream[0].offset));processed=stream[0].offset;if(stream===original){nodeStack.reverse().forEach(close);do{render(stream.splice(0,1)[0]);stream=selectStream()}while(stream===original&&stream.length&&stream[0].offset===processed);nodeStack.reverse().forEach(open)}else{if(stream[0].event==="start"){nodeStack.push(stream[0].node)}else{nodeStack.pop()}render(stream.splice(0,1)[0])}}return result+escape(value.substr(processed))}function dependencyOnParent(mode){if(!mode)return false;return mode.endsWithParent||dependencyOnParent(mode.starts)}function expand_or_clone_mode(mode){if(mode.variants&&!mode.cached_variants){mode.cached_variants=mode.variants.map(function(variant){return inherit(mode,{variants:null},variant)})}if(mode.cached_variants)return mode.cached_variants;if(dependencyOnParent(mode))return[inherit(mode,{starts:mode.starts?inherit(mode.starts):null})];if(Object.isFrozen(mode))return[inherit(mode)];return[mode]}function restoreLanguageApi(obj){if(API_REPLACES&&!obj.langApiRestored){obj.langApiRestored=true;for(var key in API_REPLACES){if(obj[key]){obj[API_REPLACES[key]]=obj[key]}}(obj.contains||[]).concat(obj.variants||[]).forEach(restoreLanguageApi)}}function compileKeywords(rawKeywords,case_insensitive){var compiled_keywords={};if(typeof rawKeywords==="string"){splitAndCompile("keyword",rawKeywords)}else{objectKeys(rawKeywords).forEach(function(className){splitAndCompile(className,rawKeywords[className])})}return compiled_keywords;function splitAndCompile(className,str){if(case_insensitive){str=str.toLowerCase()}str.split(" ").forEach(function(keyword){var pair=keyword.split("|");compiled_keywords[pair[0]]=[className,scoreForKeyword(pair[0],pair[1])]})}}function scoreForKeyword(keyword,providedScore){if(providedScore)return Number(providedScore);return commonKeyword(keyword)?0:1}function commonKeyword(word){return COMMON_KEYWORDS.indexOf(word.toLowerCase())!=-1}function compileLanguage(language){function reStr(re){return re&&re.source||re}function langRe(value,global){return new RegExp(reStr(value),"m"+(language.case_insensitive?"i":"")+(global?"g":""))}function reCountMatchGroups(re){return new RegExp(re.toString()+"|").exec("").length-1}function joinRe(regexps,separator){var backreferenceRe=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./;var numCaptures=0;var ret="";for(var i=0;i<regexps.length;i++){numCaptures+=1;var offset=numCaptures;var re=reStr(regexps[i]);if(i>0){ret+=separator}ret+="(";while(re.length>0){var match=backreferenceRe.exec(re);if(match==null){ret+=re;break}ret+=re.substring(0,match.index);re=re.substring(match.index+match[0].length);if(match[0][0]=="\\"&&match[1]){ret+="\\"+String(Number(match[1])+offset)}else{ret+=match[0];if(match[0]=="("){numCaptures++}}}ret+=")"}return ret}function buildModeRegex(mode){var matchIndexes={};var matcherRe;var regexes=[];var matcher={};var matchAt=1;function addRule(rule,regex){matchIndexes[matchAt]=rule;regexes.push([rule,regex]);matchAt+=reCountMatchGroups(regex)+1}var term;for(var i=0;i<mode.contains.length;i++){var re;term=mode.contains[i];if(term.beginKeywords){re="\\.?(?:"+term.begin+")\\.?"}else{re=term.begin}addRule(term,re)}if(mode.terminator_end)addRule("end",mode.terminator_end);if(mode.illegal)addRule("illegal",mode.illegal);var terminators=regexes.map(function(el){return el[1]});matcherRe=langRe(joinRe(terminators,"|"),true);matcher.lastIndex=0;matcher.exec=function(s){var rule;if(regexes.length===0)return null;matcherRe.lastIndex=matcher.lastIndex;var match=matcherRe.exec(s);if(!match){return null}for(var i=0;i<match.length;i++){if(match[i]!=undefined&&matchIndexes[""+i]!=undefined){rule=matchIndexes[""+i];break}}if(typeof rule==="string"){match.type=rule;match.extra=[mode.illegal,mode.terminator_end]}else{match.type="begin";match.rule=rule}return match};return matcher}function compileMode(mode,parent){if(mode.compiled)return;mode.compiled=true;mode.keywords=mode.keywords||mode.beginKeywords;if(mode.keywords)mode.keywords=compileKeywords(mode.keywords,language.case_insensitive);mode.lexemesRe=langRe(mode.lexemes||/\w+/,true);if(parent){if(mode.beginKeywords){mode.begin="\\b("+mode.beginKeywords.split(" ").join("|")+")\\b"}if(!mode.begin)mode.begin=/\B|\b/;mode.beginRe=langRe(mode.begin);if(mode.endSameAsBegin)mode.end=mode.begin;if(!mode.end&&!mode.endsWithParent)mode.end=/\B|\b/;if(mode.end)mode.endRe=langRe(mode.end);mode.terminator_end=reStr(mode.end)||"";if(mode.endsWithParent&&parent.terminator_end)mode.terminator_end+=(mode.end?"|":"")+parent.terminator_end}if(mode.illegal)mode.illegalRe=langRe(mode.illegal);if(mode.relevance==null)mode.relevance=1;if(!mode.contains){mode.contains=[]}mode.contains=Array.prototype.concat.apply([],mode.contains.map(function(c){return expand_or_clone_mode(c==="self"?mode:c)}));mode.contains.forEach(function(c){compileMode(c,mode)});if(mode.starts){compileMode(mode.starts,parent)}mode.terminators=buildModeRegex(mode)}if(language.contains&&language.contains.indexOf("self")!=-1){if(!SAFE_MODE){throw new Error("ERR: contains `self` is not supported at the top-level of a language.  See documentation.")}else{language.contains=language.contains.filter(function(mode){return mode!="self"})}}compileMode(language)}function hideUpgradeWarning(){if(options.hideUpgradeWarningAcceptNoSupportOrSecurityUpdates)return true;if(typeof process==="object"&&typeof process.env==="object"&&process.env["HLJS_HIDE_UPGRADE_WARNING"])return true}function highlight(languageName,code,ignore_illegals,continuation){if(!hideUpgradeWarning()){if(!showedUpgradeWarning){showedUpgradeWarning=true;console.log("Version 9 of Highlight.js has reached EOL and is no longer supported.\n"+"Please upgrade or ask whatever dependency you are using to upgrade.\n"+"https://github.com/highlightjs/highlight.js/issues/2877")}}var codeToHighlight=code;function escapeRe(value){return new RegExp(value.replace(/[-\/\\^$*+?.()|[\]{}]/g,"\\$&"),"m")}function endOfMode(mode,lexeme){if(testRe(mode.endRe,lexeme)){while(mode.endsParent&&mode.parent){mode=mode.parent}return mode}if(mode.endsWithParent){return endOfMode(mode.parent,lexeme)}}function keywordMatch(mode,match){var match_str=language.case_insensitive?match[0].toLowerCase():match[0];return mode.keywords.hasOwnProperty(match_str)&&mode.keywords[match_str]}function buildSpan(className,insideSpan,leaveOpen,noPrefix){if(!leaveOpen&&insideSpan==="")return"";if(!className)return insideSpan;var classPrefix=noPrefix?"":options.classPrefix,openSpan='<span class="'+classPrefix,closeSpan=leaveOpen?"":spanEndTag;openSpan+=className+'">';return openSpan+insideSpan+closeSpan}function processKeywords(){var keyword_match,last_index,match,result;if(!top.keywords)return escape(mode_buffer);result="";last_index=0;top.lexemesRe.lastIndex=0;match=top.lexemesRe.exec(mode_buffer);while(match){result+=escape(mode_buffer.substring(last_index,match.index));keyword_match=keywordMatch(top,match);if(keyword_match){relevance+=keyword_match[1];result+=buildSpan(keyword_match[0],escape(match[0]))}else{result+=escape(match[0])}last_index=top.lexemesRe.lastIndex;match=top.lexemesRe.exec(mode_buffer)}return result+escape(mode_buffer.substr(last_index))}function processSubLanguage(){var explicit=typeof top.subLanguage==="string";if(explicit&&!languages[top.subLanguage]){return escape(mode_buffer)}var result=explicit?highlight(top.subLanguage,mode_buffer,true,continuations[top.subLanguage]):highlightAuto(mode_buffer,top.subLanguage.length?top.subLanguage:undefined);if(top.relevance>0){relevance+=result.relevance}if(explicit){continuations[top.subLanguage]=result.top}return buildSpan(result.language,result.value,false,true)}function processBuffer(){result+=top.subLanguage!=null?processSubLanguage():processKeywords();mode_buffer=""}function startNewMode(mode){result+=mode.className?buildSpan(mode.className,"",true):"";top=Object.create(mode,{parent:{value:top}})}function doBeginMatch(match){var lexeme=match[0];var new_mode=match.rule;if(new_mode&&new_mode.endSameAsBegin){new_mode.endRe=escapeRe(lexeme)}if(new_mode.skip){mode_buffer+=lexeme}else{if(new_mode.excludeBegin){mode_buffer+=lexeme}processBuffer();if(!new_mode.returnBegin&&!new_mode.excludeBegin){mode_buffer=lexeme}}startNewMode(new_mode);return new_mode.returnBegin?0:lexeme.length}function doEndMatch(match){var lexeme=match[0];var matchPlusRemainder=codeToHighlight.substr(match.index);var end_mode=endOfMode(top,matchPlusRemainder);if(!end_mode){return}var origin=top;if(origin.skip){mode_buffer+=lexeme}else{if(!(origin.returnEnd||origin.excludeEnd)){mode_buffer+=lexeme}processBuffer();if(origin.excludeEnd){mode_buffer=lexeme}}do{if(top.className){result+=spanEndTag}if(!top.skip&&!top.subLanguage){relevance+=top.relevance}top=top.parent}while(top!==end_mode.parent);if(end_mode.starts){if(end_mode.endSameAsBegin){end_mode.starts.endRe=end_mode.endRe}startNewMode(end_mode.starts)}return origin.returnEnd?0:lexeme.length}var lastMatch={};function processLexeme(text_before_match,match){var lexeme=match&&match[0];mode_buffer+=text_before_match;if(lexeme==null){processBuffer();return 0}if(lastMatch.type=="begin"&&match.type=="end"&&lastMatch.index==match.index&&lexeme===""){mode_buffer+=codeToHighlight.slice(match.index,match.index+1);return 1}if(lastMatch.type==="illegal"&&lexeme===""){mode_buffer+=codeToHighlight.slice(match.index,match.index+1);return 1}lastMatch=match;if(match.type==="begin"){return doBeginMatch(match)}else if(match.type==="illegal"&&!ignore_illegals){throw new Error('Illegal lexeme "'+lexeme+'" for mode "'+(top.className||"<unnamed>")+'"')}else if(match.type==="end"){var processed=doEndMatch(match);if(processed!=undefined)return processed}mode_buffer+=lexeme;return lexeme.length}var language=getLanguage(languageName);if(!language){console.error(LANGUAGE_NOT_FOUND.replace("{}",languageName));throw new Error('Unknown language: "'+languageName+'"')}compileLanguage(language);var top=continuation||language;var continuations={};var result="",current;for(current=top;current!==language;current=current.parent){if(current.className){result=buildSpan(current.className,"",true)+result}}var mode_buffer="";var relevance=0;try{var match,count,index=0;while(true){top.terminators.lastIndex=index;match=top.terminators.exec(codeToHighlight);if(!match)break;count=processLexeme(codeToHighlight.substring(index,match.index),match);index=match.index+count}processLexeme(codeToHighlight.substr(index));for(current=top;current.parent;current=current.parent){if(current.className){result+=spanEndTag}}return{relevance:relevance,value:result,illegal:false,language:languageName,top:top}}catch(err){if(err.message&&err.message.indexOf("Illegal")!==-1){return{illegal:true,relevance:0,value:escape(codeToHighlight)}}else if(SAFE_MODE){return{relevance:0,value:escape(codeToHighlight),language:languageName,top:top,errorRaised:err}}else{throw err}}}function highlightAuto(code,languageSubset){languageSubset=languageSubset||options.languages||objectKeys(languages);var result={relevance:0,value:escape(code)};var second_best=result;languageSubset.filter(getLanguage).filter(autoDetection).forEach(function(name){var current=highlight(name,code,false);current.language=name;if(current.relevance>second_best.relevance){second_best=current}if(current.relevance>result.relevance){second_best=result;result=current}});if(second_best.language){result.second_best=second_best}return result}function fixMarkup(value){if(!(options.tabReplace||options.useBR)){return value}return value.replace(fixMarkupRe,function(match,p1){if(options.useBR&&match==="\n"){return"<br>"}else if(options.tabReplace){return p1.replace(/\t/g,options.tabReplace)}return""})}function buildClassName(prevClassName,currentLang,resultLang){var language=currentLang?aliases[currentLang]:resultLang,result=[prevClassName.trim()];if(!prevClassName.match(/\bhljs\b/)){result.push("hljs")}if(prevClassName.indexOf(language)===-1){result.push(language)}return result.join(" ").trim()}function highlightBlock(block){var node,originalStream,result,resultNode,text;var language=blockLanguage(block);if(isNotHighlighted(language))return;if(options.useBR){node=document.createElement("div");node.innerHTML=block.innerHTML.replace(/\n/g,"").replace(/<br[ \/]*>/g,"\n")}else{node=block}text=node.textContent;result=language?highlight(language,text,true):highlightAuto(text);originalStream=nodeStream(node);if(originalStream.length){resultNode=document.createElement("div");resultNode.innerHTML=result.value;result.value=mergeStreams(originalStream,nodeStream(resultNode),text)}result.value=fixMarkup(result.value);block.innerHTML=result.value;block.className=buildClassName(block.className,language,result.language);block.result={language:result.language,re:result.relevance};if(result.second_best){block.second_best={language:result.second_best.language,re:result.second_best.relevance}}}function configure(user_options){options=inherit(options,user_options)}function initHighlighting(){if(initHighlighting.called)return;initHighlighting.called=true;var blocks=document.querySelectorAll("pre code");ArrayProto.forEach.call(blocks,highlightBlock)}function initHighlightingOnLoad(){window.addEventListener("DOMContentLoaded",initHighlighting,false);window.addEventListener("load",initHighlighting,false)}var PLAINTEXT_LANGUAGE={disableAutodetect:true};function registerLanguage(name,language){var lang;try{lang=language(hljs)}catch(error){console.error("Language definition for '{}' could not be registered.".replace("{}",name));if(!SAFE_MODE){throw error}else{console.error(error)}lang=PLAINTEXT_LANGUAGE}languages[name]=lang;restoreLanguageApi(lang);lang.rawDefinition=language.bind(null,hljs);if(lang.aliases){lang.aliases.forEach(function(alias){aliases[alias]=name})}}function listLanguages(){return objectKeys(languages)}function requireLanguage(name){var lang=getLanguage(name);if(lang){return lang}var err=new Error("The '{}' language is required, but not loaded.".replace("{}",name));throw err}function getLanguage(name){name=(name||"").toLowerCase();return languages[name]||languages[aliases[name]]}function autoDetection(name){var lang=getLanguage(name);return lang&&!lang.disableAutodetect}hljs.highlight=highlight;hljs.highlightAuto=highlightAuto;hljs.fixMarkup=fixMarkup;hljs.highlightBlock=highlightBlock;hljs.configure=configure;hljs.initHighlighting=initHighlighting;hljs.initHighlightingOnLoad=initHighlightingOnLoad;hljs.registerLanguage=registerLanguage;hljs.listLanguages=listLanguages;hljs.getLanguage=getLanguage;hljs.requireLanguage=requireLanguage;hljs.autoDetection=autoDetection;hljs.inherit=inherit;hljs.debugMode=function(){SAFE_MODE=false};hljs.IDENT_RE="[a-zA-Z]\\w*";hljs.UNDERSCORE_IDENT_RE="[a-zA-Z_]\\w*";hljs.NUMBER_RE="\\b\\d+(\\.\\d+)?";hljs.C_NUMBER_RE="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)";hljs.BINARY_NUMBER_RE="\\b(0b[01]+)";hljs.RE_STARTERS_RE="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~";hljs.BACKSLASH_ESCAPE={begin:"\\\\[\\s\\S]",relevance:0};hljs.APOS_STRING_MODE={className:"string",begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};hljs.QUOTE_STRING_MODE={className:"string",begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};hljs.PHRASAL_WORDS_MODE={begin:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/};hljs.COMMENT=function(begin,end,inherits){var mode=hljs.inherit({className:"comment",begin:begin,end:end,contains:[]},inherits||{});mode.contains.push(hljs.PHRASAL_WORDS_MODE);mode.contains.push({className:"doctag",begin:"(?:TODO|FIXME|NOTE|BUG|XXX):",relevance:0});return mode};hljs.C_LINE_COMMENT_MODE=hljs.COMMENT("//","$");hljs.C_BLOCK_COMMENT_MODE=hljs.COMMENT("/\\*","\\*/");hljs.HASH_COMMENT_MODE=hljs.COMMENT("#","$");hljs.NUMBER_MODE={className:"number",begin:hljs.NUMBER_RE,relevance:0};hljs.C_NUMBER_MODE={className:"number",begin:hljs.C_NUMBER_RE,relevance:0};hljs.BINARY_NUMBER_MODE={className:"number",begin:hljs.BINARY_NUMBER_RE,relevance:0};hljs.CSS_NUMBER_MODE={className:"number",begin:hljs.NUMBER_RE+"("+"%|em|ex|ch|rem"+"|vw|vh|vmin|vmax"+"|cm|mm|in|pt|pc|px"+"|deg|grad|rad|turn"+"|s|ms"+"|Hz|kHz"+"|dpi|dpcm|dppx"+")?",relevance:0};hljs.REGEXP_MODE={className:"regexp",begin:/\//,end:/\/[gimuy]*/,illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE,{begin:/\[/,end:/\]/,relevance:0,contains:[hljs.BACKSLASH_ESCAPE]}]};hljs.TITLE_MODE={className:"title",begin:hljs.IDENT_RE,relevance:0};hljs.UNDERSCORE_TITLE_MODE={className:"title",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0};hljs.METHOD_GUARD={begin:"\\.\\s*"+hljs.UNDERSCORE_IDENT_RE,relevance:0};var constants=[hljs.BACKSLASH_ESCAPE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.PHRASAL_WORDS_MODE,hljs.COMMENT,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE,hljs.C_NUMBER_MODE,hljs.BINARY_NUMBER_MODE,hljs.CSS_NUMBER_MODE,hljs.REGEXP_MODE,hljs.TITLE_MODE,hljs.UNDERSCORE_TITLE_MODE,hljs.METHOD_GUARD];constants.forEach(function(obj){deepFreeze(obj)});function deepFreeze(o){Object.freeze(o);var objIsFunction=typeof o==="function";Object.getOwnPropertyNames(o).forEach(function(prop){if(o.hasOwnProperty(prop)&&o[prop]!==null&&(typeof o[prop]==="object"||typeof o[prop]==="function")&&(objIsFunction?prop!=="caller"&&prop!=="callee"&&prop!=="arguments":true)&&!Object.isFrozen(o[prop])){deepFreeze(o[prop])}});return o}return hljs});hljs.registerLanguage("hsp",function(hljs){return{case_insensitive:true,lexemes:/[\w\._]+/,keywords:"goto gosub return break repeat loop continue wait await dim sdim foreach dimtype dup dupptr end stop newmod delmod mref run exgoto on mcall assert logmes newlab resume yield onexit onerror onkey onclick oncmd exist delete mkdir chdir dirlist bload bsave bcopy memfile if else poke wpoke lpoke getstr chdpm memexpand memcpy memset notesel noteadd notedel noteload notesave randomize noteunsel noteget split strrep setease button chgdisp exec dialog mmload mmplay mmstop mci pset pget syscolor mes print title pos circle cls font sysfont objsize picload color palcolor palette redraw width gsel gcopy gzoom gmode bmpsave hsvcolor getkey listbox chkbox combox input mesbox buffer screen bgscr mouse objsel groll line clrobj boxf objprm objmode stick grect grotate gsquare gradf objimage objskip objenable celload celdiv celput newcom querycom delcom cnvstow comres axobj winobj sendmsg comevent comevarg sarrayconv callfunc cnvwtos comevdisp libptr system hspstat hspver stat cnt err strsize looplev sublev iparam wparam lparam refstr refdval int rnd strlen length length2 length3 length4 vartype gettime peek wpeek lpeek varptr varuse noteinfo instr abs limit getease str strmid strf getpath strtrim sin cos tan atan sqrt double absf expf logf limitf powf geteasef mousex mousey mousew hwnd hinstance hdc ginfo objinfo dirinfo sysinfo thismod __hspver__ __hsp30__ __date__ __time__ __line__ __file__ _debug __hspdef__ and or xor not screen_normal screen_palette screen_hide screen_fixedsize screen_tool screen_frame gmode_gdi gmode_mem gmode_rgb0 gmode_alpha gmode_rgb0alpha gmode_add gmode_sub gmode_pixela ginfo_mx ginfo_my ginfo_act ginfo_sel ginfo_wx1 ginfo_wy1 ginfo_wx2 ginfo_wy2 ginfo_vx ginfo_vy ginfo_sizex ginfo_sizey ginfo_winx ginfo_winy ginfo_mesx ginfo_mesy ginfo_r ginfo_g ginfo_b ginfo_paluse ginfo_dispx ginfo_dispy ginfo_cx ginfo_cy ginfo_intid ginfo_newid ginfo_sx ginfo_sy objinfo_mode objinfo_bmscr objinfo_hwnd notemax notesize dir_cur dir_exe dir_win dir_sys dir_cmdline dir_desktop dir_mydoc dir_tv font_normal font_bold font_italic font_underline font_strikeout font_antialias objmode_normal objmode_guifont objmode_usefont gsquare_grad msgothic msmincho do until while wend for next _break _continue switch case default swbreak swend ddim ldim alloc m_pi rad2deg deg2rad ease_linear ease_quad_in ease_quad_out ease_quad_inout ease_cubic_in ease_cubic_out ease_cubic_inout ease_quartic_in ease_quartic_out ease_quartic_inout ease_bounce_in ease_bounce_out ease_bounce_inout ease_shake_in ease_shake_out ease_shake_inout ease_loop",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"string",begin:'{"',end:'"}',contains:[hljs.BACKSLASH_ESCAPE]},hljs.COMMENT(";","$",{relevance:0}),{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"addion cfunc cmd cmpopt comfunc const defcfunc deffunc define else endif enum epack func global if ifdef ifndef include modcfunc modfunc modinit modterm module pack packopt regcmd runtime undef usecom uselib"},contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"}),hljs.NUMBER_MODE,hljs.C_NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"symbol",begin:"^\\*(\\w+|@)"},hljs.NUMBER_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("applescript",function(hljs){var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:""});var PARAMS={className:"params",begin:"\\(",end:"\\)",contains:["self",hljs.C_NUMBER_MODE,STRING]};var COMMENT_MODE_1=hljs.COMMENT("--","$");var COMMENT_MODE_2=hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self",COMMENT_MODE_1]});var COMMENTS=[COMMENT_MODE_1,COMMENT_MODE_2,hljs.HASH_COMMENT_MODE];return{aliases:["osascript"],keywords:{keyword:"about above after against and around as at back before beginning "+"behind below beneath beside between but by considering "+"contain contains continue copy div does eighth else end equal "+"equals error every exit fifth first for fourth from front "+"get given global if ignoring in into is it its last local me "+"middle mod my ninth not of on onto or over prop property put ref "+"reference repeat returning script second set seventh since "+"sixth some tell tenth that the|0 then third through thru "+"timeout times to transaction try until where while whose with "+"without",literal:"AppleScript false linefeed return pi quote result space tab true",built_in:"alias application boolean class constant date file integer list "+"number real record string text "+"activate beep count delay launch log offset read round "+"run say summarize write "+"character characters contents day frontmost id item length "+"month name paragraph paragraphs rest reverse running time version "+"weekday word words year"},contains:[STRING,hljs.C_NUMBER_MODE,{className:"built_in",begin:"\\b(clipboard info|the clipboard|info for|list (disks|folder)|"+"mount volume|path to|(close|open for) access|(get|set) eof|"+"current date|do shell script|get volume settings|random number|"+"set volume|system attribute|system info|time to GMT|"+"(load|run|store) script|scripting components|"+"ASCII (character|number)|localized string|"+"choose (application|color|file|file name|"+"folder|from list|remote application|URL)|"+"display (alert|dialog))\\b|^\\s*return\\b"},{className:"literal",begin:"\\b(text item delimiters|current application|missing value)\\b"},{className:"keyword",begin:"\\b(apart from|aside from|instead of|out of|greater than|"+"isn't|(doesn't|does not) (equal|come before|come after|contain)|"+"(greater|less) than( or equal)?|(starts?|ends|begins?) with|"+"contained by|comes (before|after)|a (ref|reference)|POSIX file|"+"POSIX path|(date|time) string|quoted form)\\b"},{beginKeywords:"on",illegal:"[${=;\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]}].concat(COMMENTS),illegal:"//|->|=>|\\[\\["}});hljs.registerLanguage("matlab",function(hljs){var TRANSPOSE_RE="('|\\.')+";var TRANSPOSE={relevance:0,contains:[{begin:TRANSPOSE_RE}]};return{keywords:{keyword:"break case catch classdef continue else elseif end enumerated events for function "+"global if methods otherwise parfor persistent properties return spmd switch try while",built_in:"sin sind sinh asin asind asinh cos cosd cosh acos acosd acosh tan tand tanh atan "+"atand atan2 atanh sec secd sech asec asecd asech csc cscd csch acsc acscd acsch cot "+"cotd coth acot acotd acoth hypot exp expm1 log log1p log10 log2 pow2 realpow reallog "+"realsqrt sqrt nthroot nextpow2 abs angle complex conj imag real unwrap isreal "+"cplxpair fix floor ceil round mod rem sign airy besselj bessely besselh besseli "+"besselk beta betainc betaln ellipj ellipke erf erfc erfcx erfinv expint gamma "+"gammainc gammaln psi legendre cross dot factor isprime primes gcd lcm rat rats perms "+"nchoosek factorial cart2sph cart2pol pol2cart sph2cart hsv2rgb rgb2hsv zeros ones "+"eye repmat rand randn linspace logspace freqspace meshgrid accumarray size length "+"ndims numel disp isempty isequal isequalwithequalnans cat reshape diag blkdiag tril "+"triu fliplr flipud flipdim rot90 find sub2ind ind2sub bsxfun ndgrid permute ipermute "+"shiftdim circshift squeeze isscalar isvector ans eps realmax realmin pi i inf nan "+"isnan isinf isfinite j why compan gallery hadamard hankel hilb invhilb magic pascal "+"rosser toeplitz vander wilkinson max min nanmax nanmin mean nanmean type table "+"readtable writetable sortrows sort figure plot plot3 scatter scatter3 cellfun "+"legend intersect ismember procrustes hold num2cell "},illegal:'(//|"|#|/\\*|\\s+/\\w+)',contains:[{className:"function",beginKeywords:"function",end:"$",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",variants:[{begin:"\\(",end:"\\)"},{begin:"\\[",end:"\\]"}]}]},{className:"built_in",begin:/true|false/,relevance:0,starts:TRANSPOSE},{begin:"[a-zA-Z][a-zA-Z_0-9]*"+TRANSPOSE_RE,relevance:0},{className:"number",begin:hljs.C_NUMBER_RE,relevance:0,starts:TRANSPOSE},{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE,{begin:"''"}]},{begin:/\]|}|\)/,relevance:0,starts:TRANSPOSE},{className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE,{begin:'""'}],starts:TRANSPOSE},hljs.COMMENT("^\\s*\\%\\{\\s*$","^\\s*\\%\\}\\s*$"),hljs.COMMENT("\\%","$")]}});hljs.registerLanguage("gml",function(hljs){var GML_KEYWORDS={keyword:"begin end if then else while do for break continue with until "+"repeat exit and or xor not return mod div switch case default var "+"globalvar enum #macro #region #endregion",built_in:"is_real is_string is_array is_undefined is_int32 is_int64 "+"is_ptr is_vec3 is_vec4 is_matrix is_bool typeof "+"variable_global_exists variable_global_get variable_global_set "+"variable_instance_exists variable_instance_get variable_instance_set "+"variable_instance_get_names array_length_1d array_length_2d "+"array_height_2d array_equals array_create array_copy random "+"random_range irandom irandom_range random_set_seed random_get_seed "+"randomize randomise choose abs round floor ceil sign frac sqrt sqr "+"exp ln log2 log10 sin cos tan arcsin arccos arctan arctan2 dsin dcos "+"dtan darcsin darccos darctan darctan2 degtorad radtodeg power logn "+"min max mean median clamp lerp dot_product dot_product_3d "+"dot_product_normalised dot_product_3d_normalised "+"dot_product_normalized dot_product_3d_normalized math_set_epsilon "+"math_get_epsilon angle_difference point_distance_3d point_distance "+"point_direction lengthdir_x lengthdir_y real string int64 ptr "+"string_format chr ansi_char ord string_length string_byte_length "+"string_pos string_copy string_char_at string_ord_at string_byte_at "+"string_set_byte_at string_delete string_insert string_lower "+"string_upper string_repeat string_letters string_digits "+"string_lettersdigits string_replace string_replace_all string_count "+"string_hash_to_newline clipboard_has_text clipboard_set_text "+"clipboard_get_text date_current_datetime date_create_datetime "+"date_valid_datetime date_inc_year date_inc_month date_inc_week "+"date_inc_day date_inc_hour date_inc_minute date_inc_second "+"date_get_year date_get_month date_get_week date_get_day "+"date_get_hour date_get_minute date_get_second date_get_weekday "+"date_get_day_of_year date_get_hour_of_year date_get_minute_of_year "+"date_get_second_of_year date_year_span date_month_span "+"date_week_span date_day_span date_hour_span date_minute_span "+"date_second_span date_compare_datetime date_compare_date "+"date_compare_time date_date_of date_time_of date_datetime_string "+"date_date_string date_time_string date_days_in_month "+"date_days_in_year date_leap_year date_is_today date_set_timezone "+"date_get_timezone game_set_speed game_get_speed motion_set "+"motion_add place_free place_empty place_meeting place_snapped "+"move_random move_snap move_towards_point move_contact_solid "+"move_contact_all move_outside_solid move_outside_all "+"move_bounce_solid move_bounce_all move_wrap distance_to_point "+"distance_to_object position_empty position_meeting path_start "+"path_end mp_linear_step mp_potential_step mp_linear_step_object "+"mp_potential_step_object mp_potential_settings mp_linear_path "+"mp_potential_path mp_linear_path_object mp_potential_path_object "+"mp_grid_create mp_grid_destroy mp_grid_clear_all mp_grid_clear_cell "+"mp_grid_clear_rectangle mp_grid_add_cell mp_grid_get_cell "+"mp_grid_add_rectangle mp_grid_add_instances mp_grid_path "+"mp_grid_draw mp_grid_to_ds_grid collision_point collision_rectangle "+"collision_circle collision_ellipse collision_line "+"collision_point_list collision_rectangle_list collision_circle_list "+"collision_ellipse_list collision_line_list instance_position_list "+"instance_place_list point_in_rectangle "+"point_in_triangle point_in_circle rectangle_in_rectangle "+"rectangle_in_triangle rectangle_in_circle instance_find "+"instance_exists instance_number instance_position instance_nearest "+"instance_furthest instance_place instance_create_depth "+"instance_create_layer instance_copy instance_change instance_destroy "+"position_destroy position_change instance_id_get "+"instance_deactivate_all instance_deactivate_object "+"instance_deactivate_region instance_activate_all "+"instance_activate_object instance_activate_region room_goto "+"room_goto_previous room_goto_next room_previous room_next "+"room_restart game_end game_restart game_load game_save "+"game_save_buffer game_load_buffer event_perform event_user "+"event_perform_object event_inherited show_debug_message "+"show_debug_overlay debug_event debug_get_callstack alarm_get "+"alarm_set font_texture_page_size keyboard_set_map keyboard_get_map "+"keyboard_unset_map keyboard_check keyboard_check_pressed "+"keyboard_check_released keyboard_check_direct keyboard_get_numlock "+"keyboard_set_numlock keyboard_key_press keyboard_key_release "+"keyboard_clear io_clear mouse_check_button "+"mouse_check_button_pressed mouse_check_button_released "+"mouse_wheel_up mouse_wheel_down mouse_clear draw_self draw_sprite "+"draw_sprite_pos draw_sprite_ext draw_sprite_stretched "+"draw_sprite_stretched_ext draw_sprite_tiled draw_sprite_tiled_ext "+"draw_sprite_part draw_sprite_part_ext draw_sprite_general draw_clear "+"draw_clear_alpha draw_point draw_line draw_line_width draw_rectangle "+"draw_roundrect draw_roundrect_ext draw_triangle draw_circle "+"draw_ellipse draw_set_circle_precision draw_arrow draw_button "+"draw_path draw_healthbar draw_getpixel draw_getpixel_ext "+"draw_set_colour draw_set_color draw_set_alpha draw_get_colour "+"draw_get_color draw_get_alpha merge_colour make_colour_rgb "+"make_colour_hsv colour_get_red colour_get_green colour_get_blue "+"colour_get_hue colour_get_saturation colour_get_value merge_color "+"make_color_rgb make_color_hsv color_get_red color_get_green "+"color_get_blue color_get_hue color_get_saturation color_get_value "+"merge_color screen_save screen_save_part draw_set_font "+"draw_set_halign draw_set_valign draw_text draw_text_ext string_width "+"string_height string_width_ext string_height_ext "+"draw_text_transformed draw_text_ext_transformed draw_text_colour "+"draw_text_ext_colour draw_text_transformed_colour "+"draw_text_ext_transformed_colour draw_text_color draw_text_ext_color "+"draw_text_transformed_color draw_text_ext_transformed_color "+"draw_point_colour draw_line_colour draw_line_width_colour "+"draw_rectangle_colour draw_roundrect_colour "+"draw_roundrect_colour_ext draw_triangle_colour draw_circle_colour "+"draw_ellipse_colour draw_point_color draw_line_color "+"draw_line_width_color draw_rectangle_color draw_roundrect_color "+"draw_roundrect_color_ext draw_triangle_color draw_circle_color "+"draw_ellipse_color draw_primitive_begin draw_vertex "+"draw_vertex_colour draw_vertex_color draw_primitive_end "+"sprite_get_uvs font_get_uvs sprite_get_texture font_get_texture "+"texture_get_width texture_get_height texture_get_uvs "+"draw_primitive_begin_texture draw_vertex_texture "+"draw_vertex_texture_colour draw_vertex_texture_color "+"texture_global_scale surface_create surface_create_ext "+"surface_resize surface_free surface_exists surface_get_width "+"surface_get_height surface_get_texture surface_set_target "+"surface_set_target_ext surface_reset_target surface_depth_disable "+"surface_get_depth_disable draw_surface draw_surface_stretched "+"draw_surface_tiled draw_surface_part draw_surface_ext "+"draw_surface_stretched_ext draw_surface_tiled_ext "+"draw_surface_part_ext draw_surface_general surface_getpixel "+"surface_getpixel_ext surface_save surface_save_part surface_copy "+"surface_copy_part application_surface_draw_enable "+"application_get_position application_surface_enable "+"application_surface_is_enabled display_get_width display_get_height "+"display_get_orientation display_get_gui_width display_get_gui_height "+"display_reset display_mouse_get_x display_mouse_get_y "+"display_mouse_set display_set_ui_visibility "+"window_set_fullscreen window_get_fullscreen "+"window_set_caption window_set_min_width window_set_max_width "+"window_set_min_height window_set_max_height window_get_visible_rects "+"window_get_caption window_set_cursor window_get_cursor "+"window_set_colour window_get_colour window_set_color "+"window_get_color window_set_position window_set_size "+"window_set_rectangle window_center window_get_x window_get_y "+"window_get_width window_get_height window_mouse_get_x "+"window_mouse_get_y window_mouse_set window_view_mouse_get_x "+"window_view_mouse_get_y window_views_mouse_get_x "+"window_views_mouse_get_y audio_listener_position "+"audio_listener_velocity audio_listener_orientation "+"audio_emitter_position audio_emitter_create audio_emitter_free "+"audio_emitter_exists audio_emitter_pitch audio_emitter_velocity "+"audio_emitter_falloff audio_emitter_gain audio_play_sound "+"audio_play_sound_on audio_play_sound_at audio_stop_sound "+"audio_resume_music audio_music_is_playing audio_resume_sound "+"audio_pause_sound audio_pause_music audio_channel_num "+"audio_sound_length audio_get_type audio_falloff_set_model "+"audio_play_music audio_stop_music audio_master_gain audio_music_gain "+"audio_sound_gain audio_sound_pitch audio_stop_all audio_resume_all "+"audio_pause_all audio_is_playing audio_is_paused audio_exists "+"audio_sound_set_track_position audio_sound_get_track_position "+"audio_emitter_get_gain audio_emitter_get_pitch audio_emitter_get_x "+"audio_emitter_get_y audio_emitter_get_z audio_emitter_get_vx "+"audio_emitter_get_vy audio_emitter_get_vz "+"audio_listener_set_position audio_listener_set_velocity "+"audio_listener_set_orientation audio_listener_get_data "+"audio_set_master_gain audio_get_master_gain audio_sound_get_gain "+"audio_sound_get_pitch audio_get_name audio_sound_set_track_position "+"audio_sound_get_track_position audio_create_stream "+"audio_destroy_stream audio_create_sync_group "+"audio_destroy_sync_group audio_play_in_sync_group "+"audio_start_sync_group audio_stop_sync_group audio_pause_sync_group "+"audio_resume_sync_group audio_sync_group_get_track_pos "+"audio_sync_group_debug audio_sync_group_is_playing audio_debug "+"audio_group_load audio_group_unload audio_group_is_loaded "+"audio_group_load_progress audio_group_name audio_group_stop_all "+"audio_group_set_gain audio_create_buffer_sound "+"audio_free_buffer_sound audio_create_play_queue "+"audio_free_play_queue audio_queue_sound audio_get_recorder_count "+"audio_get_recorder_info audio_start_recording audio_stop_recording "+"audio_sound_get_listener_mask audio_emitter_get_listener_mask "+"audio_get_listener_mask audio_sound_set_listener_mask "+"audio_emitter_set_listener_mask audio_set_listener_mask "+"audio_get_listener_count audio_get_listener_info audio_system "+"show_message show_message_async clickable_add clickable_add_ext "+"clickable_change clickable_change_ext clickable_delete "+"clickable_exists clickable_set_style show_question "+"show_question_async get_integer get_string get_integer_async "+"get_string_async get_login_async get_open_filename get_save_filename "+"get_open_filename_ext get_save_filename_ext show_error "+"highscore_clear highscore_add highscore_value highscore_name "+"draw_highscore sprite_exists sprite_get_name sprite_get_number "+"sprite_get_width sprite_get_height sprite_get_xoffset "+"sprite_get_yoffset sprite_get_bbox_left sprite_get_bbox_right "+"sprite_get_bbox_top sprite_get_bbox_bottom sprite_save "+"sprite_save_strip sprite_set_cache_size sprite_set_cache_size_ext "+"sprite_get_tpe sprite_prefetch sprite_prefetch_multi sprite_flush "+"sprite_flush_multi sprite_set_speed sprite_get_speed_type "+"sprite_get_speed font_exists font_get_name font_get_fontname "+"font_get_bold font_get_italic font_get_first font_get_last "+"font_get_size font_set_cache_size path_exists path_get_name "+"path_get_length path_get_time path_get_kind path_get_closed "+"path_get_precision path_get_number path_get_point_x path_get_point_y "+"path_get_point_speed path_get_x path_get_y path_get_speed "+"script_exists script_get_name timeline_add timeline_delete "+"timeline_clear timeline_exists timeline_get_name "+"timeline_moment_clear timeline_moment_add_script timeline_size "+"timeline_max_moment object_exists object_get_name object_get_sprite "+"object_get_solid object_get_visible object_get_persistent "+"object_get_mask object_get_parent object_get_physics "+"object_is_ancestor room_exists room_get_name sprite_set_offset "+"sprite_duplicate sprite_assign sprite_merge sprite_add "+"sprite_replace sprite_create_from_surface sprite_add_from_surface "+"sprite_delete sprite_set_alpha_from_sprite sprite_collision_mask "+"font_add_enable_aa font_add_get_enable_aa font_add font_add_sprite "+"font_add_sprite_ext font_replace font_replace_sprite "+"font_replace_sprite_ext font_delete path_set_kind path_set_closed "+"path_set_precision path_add path_assign path_duplicate path_append "+"path_delete path_add_point path_insert_point path_change_point "+"path_delete_point path_clear_points path_reverse path_mirror "+"path_flip path_rotate path_rescale path_shift script_execute "+"object_set_sprite object_set_solid object_set_visible "+"object_set_persistent object_set_mask room_set_width room_set_height "+"room_set_persistent room_set_background_colour "+"room_set_background_color room_set_view room_set_viewport "+"room_get_viewport room_set_view_enabled room_add room_duplicate "+"room_assign room_instance_add room_instance_clear room_get_camera "+"room_set_camera asset_get_index asset_get_type "+"file_text_open_from_string file_text_open_read file_text_open_write "+"file_text_open_append file_text_close file_text_write_string "+"file_text_write_real file_text_writeln file_text_read_string "+"file_text_read_real file_text_readln file_text_eof file_text_eoln "+"file_exists file_delete file_rename file_copy directory_exists "+"directory_create directory_destroy file_find_first file_find_next "+"file_find_close file_attributes filename_name filename_path "+"filename_dir filename_drive filename_ext filename_change_ext "+"file_bin_open file_bin_rewrite file_bin_close file_bin_position "+"file_bin_size file_bin_seek file_bin_write_byte file_bin_read_byte "+"parameter_count parameter_string environment_get_variable "+"ini_open_from_string ini_open ini_close ini_read_string "+"ini_read_real ini_write_string ini_write_real ini_key_exists "+"ini_section_exists ini_key_delete ini_section_delete "+"ds_set_precision ds_exists ds_stack_create ds_stack_destroy "+"ds_stack_clear ds_stack_copy ds_stack_size ds_stack_empty "+"ds_stack_push ds_stack_pop ds_stack_top ds_stack_write ds_stack_read "+"ds_queue_create ds_queue_destroy ds_queue_clear ds_queue_copy "+"ds_queue_size ds_queue_empty ds_queue_enqueue ds_queue_dequeue "+"ds_queue_head ds_queue_tail ds_queue_write ds_queue_read "+"ds_list_create ds_list_destroy ds_list_clear ds_list_copy "+"ds_list_size ds_list_empty ds_list_add ds_list_insert "+"ds_list_replace ds_list_delete ds_list_find_index ds_list_find_value "+"ds_list_mark_as_list ds_list_mark_as_map ds_list_sort "+"ds_list_shuffle ds_list_write ds_list_read ds_list_set ds_map_create "+"ds_map_destroy ds_map_clear ds_map_copy ds_map_size ds_map_empty "+"ds_map_add ds_map_add_list ds_map_add_map ds_map_replace "+"ds_map_replace_map ds_map_replace_list ds_map_delete ds_map_exists "+"ds_map_find_value ds_map_find_previous ds_map_find_next "+"ds_map_find_first ds_map_find_last ds_map_write ds_map_read "+"ds_map_secure_save ds_map_secure_load ds_map_secure_load_buffer "+"ds_map_secure_save_buffer ds_map_set ds_priority_create "+"ds_priority_destroy ds_priority_clear ds_priority_copy "+"ds_priority_size ds_priority_empty ds_priority_add "+"ds_priority_change_priority ds_priority_find_priority "+"ds_priority_delete_value ds_priority_delete_min ds_priority_find_min "+"ds_priority_delete_max ds_priority_find_max ds_priority_write "+"ds_priority_read ds_grid_create ds_grid_destroy ds_grid_copy "+"ds_grid_resize ds_grid_width ds_grid_height ds_grid_clear "+"ds_grid_set ds_grid_add ds_grid_multiply ds_grid_set_region "+"ds_grid_add_region ds_grid_multiply_region ds_grid_set_disk "+"ds_grid_add_disk ds_grid_multiply_disk ds_grid_set_grid_region "+"ds_grid_add_grid_region ds_grid_multiply_grid_region ds_grid_get "+"ds_grid_get_sum ds_grid_get_max ds_grid_get_min ds_grid_get_mean "+"ds_grid_get_disk_sum ds_grid_get_disk_min ds_grid_get_disk_max "+"ds_grid_get_disk_mean ds_grid_value_exists ds_grid_value_x "+"ds_grid_value_y ds_grid_value_disk_exists ds_grid_value_disk_x "+"ds_grid_value_disk_y ds_grid_shuffle ds_grid_write ds_grid_read "+"ds_grid_sort ds_grid_set ds_grid_get effect_create_below "+"effect_create_above effect_clear part_type_create part_type_destroy "+"part_type_exists part_type_clear part_type_shape part_type_sprite "+"part_type_size part_type_scale part_type_orientation part_type_life "+"part_type_step part_type_death part_type_speed part_type_direction "+"part_type_gravity part_type_colour1 part_type_colour2 "+"part_type_colour3 part_type_colour_mix part_type_colour_rgb "+"part_type_colour_hsv part_type_color1 part_type_color2 "+"part_type_color3 part_type_color_mix part_type_color_rgb "+"part_type_color_hsv part_type_alpha1 part_type_alpha2 "+"part_type_alpha3 part_type_blend part_system_create "+"part_system_create_layer part_system_destroy part_system_exists "+"part_system_clear part_system_draw_order part_system_depth "+"part_system_position part_system_automatic_update "+"part_system_automatic_draw part_system_update part_system_drawit "+"part_system_get_layer part_system_layer part_particles_create "+"part_particles_create_colour part_particles_create_color "+"part_particles_clear part_particles_count part_emitter_create "+"part_emitter_destroy part_emitter_destroy_all part_emitter_exists "+"part_emitter_clear part_emitter_region part_emitter_burst "+"part_emitter_stream external_call external_define external_free "+"window_handle window_device matrix_get matrix_set "+"matrix_build_identity matrix_build matrix_build_lookat "+"matrix_build_projection_ortho matrix_build_projection_perspective "+"matrix_build_projection_perspective_fov matrix_multiply "+"matrix_transform_vertex matrix_stack_push matrix_stack_pop "+"matrix_stack_multiply matrix_stack_set matrix_stack_clear "+"matrix_stack_top matrix_stack_is_empty browser_input_capture "+"os_get_config os_get_info os_get_language os_get_region "+"os_lock_orientation display_get_dpi_x display_get_dpi_y "+"display_set_gui_size display_set_gui_maximise "+"display_set_gui_maximize device_mouse_dbclick_enable "+"display_set_timing_method display_get_timing_method "+"display_set_sleep_margin display_get_sleep_margin virtual_key_add "+"virtual_key_hide virtual_key_delete virtual_key_show "+"draw_enable_drawevent draw_enable_swf_aa draw_set_swf_aa_level "+"draw_get_swf_aa_level draw_texture_flush draw_flush "+"gpu_set_blendenable gpu_set_ztestenable gpu_set_zfunc "+"gpu_set_zwriteenable gpu_set_lightingenable gpu_set_fog "+"gpu_set_cullmode gpu_set_blendmode gpu_set_blendmode_ext "+"gpu_set_blendmode_ext_sepalpha gpu_set_colorwriteenable "+"gpu_set_colourwriteenable gpu_set_alphatestenable "+"gpu_set_alphatestref gpu_set_alphatestfunc gpu_set_texfilter "+"gpu_set_texfilter_ext gpu_set_texrepeat gpu_set_texrepeat_ext "+"gpu_set_tex_filter gpu_set_tex_filter_ext gpu_set_tex_repeat "+"gpu_set_tex_repeat_ext gpu_set_tex_mip_filter "+"gpu_set_tex_mip_filter_ext gpu_set_tex_mip_bias "+"gpu_set_tex_mip_bias_ext gpu_set_tex_min_mip gpu_set_tex_min_mip_ext "+"gpu_set_tex_max_mip gpu_set_tex_max_mip_ext gpu_set_tex_max_aniso "+"gpu_set_tex_max_aniso_ext gpu_set_tex_mip_enable "+"gpu_set_tex_mip_enable_ext gpu_get_blendenable gpu_get_ztestenable "+"gpu_get_zfunc gpu_get_zwriteenable gpu_get_lightingenable "+"gpu_get_fog gpu_get_cullmode gpu_get_blendmode gpu_get_blendmode_ext "+"gpu_get_blendmode_ext_sepalpha gpu_get_blendmode_src "+"gpu_get_blendmode_dest gpu_get_blendmode_srcalpha "+"gpu_get_blendmode_destalpha gpu_get_colorwriteenable "+"gpu_get_colourwriteenable gpu_get_alphatestenable "+"gpu_get_alphatestref gpu_get_alphatestfunc gpu_get_texfilter "+"gpu_get_texfilter_ext gpu_get_texrepeat gpu_get_texrepeat_ext "+"gpu_get_tex_filter gpu_get_tex_filter_ext gpu_get_tex_repeat "+"gpu_get_tex_repeat_ext gpu_get_tex_mip_filter "+"gpu_get_tex_mip_filter_ext gpu_get_tex_mip_bias "+"gpu_get_tex_mip_bias_ext gpu_get_tex_min_mip gpu_get_tex_min_mip_ext "+"gpu_get_tex_max_mip gpu_get_tex_max_mip_ext gpu_get_tex_max_aniso "+"gpu_get_tex_max_aniso_ext gpu_get_tex_mip_enable "+"gpu_get_tex_mip_enable_ext gpu_push_state gpu_pop_state "+"gpu_get_state gpu_set_state draw_light_define_ambient "+"draw_light_define_direction draw_light_define_point "+"draw_light_enable draw_set_lighting draw_light_get_ambient "+"draw_light_get draw_get_lighting shop_leave_rating url_get_domain "+"url_open url_open_ext url_open_full get_timer achievement_login "+"achievement_logout achievement_post achievement_increment "+"achievement_post_score achievement_available "+"achievement_show_achievements achievement_show_leaderboards "+"achievement_load_friends achievement_load_leaderboard "+"achievement_send_challenge achievement_load_progress "+"achievement_reset achievement_login_status achievement_get_pic "+"achievement_show_challenge_notifications achievement_get_challenges "+"achievement_event achievement_show achievement_get_info "+"cloud_file_save cloud_string_save cloud_synchronise ads_enable "+"ads_disable ads_setup ads_engagement_launch ads_engagement_available "+"ads_engagement_active ads_event ads_event_preload "+"ads_set_reward_callback ads_get_display_height ads_get_display_width "+"ads_move ads_interstitial_available ads_interstitial_display "+"device_get_tilt_x device_get_tilt_y device_get_tilt_z "+"device_is_keypad_open device_mouse_check_button "+"device_mouse_check_button_pressed device_mouse_check_button_released "+"device_mouse_x device_mouse_y device_mouse_raw_x device_mouse_raw_y "+"device_mouse_x_to_gui device_mouse_y_to_gui iap_activate iap_status "+"iap_enumerate_products iap_restore_all iap_acquire iap_consume "+"iap_product_details iap_purchase_details facebook_init "+"facebook_login facebook_status facebook_graph_request "+"facebook_dialog facebook_logout facebook_launch_offerwall "+"facebook_post_message facebook_send_invite facebook_user_id "+"facebook_accesstoken facebook_check_permission "+"facebook_request_read_permissions "+"facebook_request_publish_permissions gamepad_is_supported "+"gamepad_get_device_count gamepad_is_connected "+"gamepad_get_description gamepad_get_button_threshold "+"gamepad_set_button_threshold gamepad_get_axis_deadzone "+"gamepad_set_axis_deadzone gamepad_button_count gamepad_button_check "+"gamepad_button_check_pressed gamepad_button_check_released "+"gamepad_button_value gamepad_axis_count gamepad_axis_value "+"gamepad_set_vibration gamepad_set_colour gamepad_set_color "+"os_is_paused window_has_focus code_is_compiled http_get "+"http_get_file http_post_string http_request json_encode json_decode "+"zip_unzip load_csv base64_encode base64_decode md5_string_unicode "+"md5_string_utf8 md5_file os_is_network_connected sha1_string_unicode "+"sha1_string_utf8 sha1_file os_powersave_enable analytics_event "+"analytics_event_ext win8_livetile_tile_notification "+"win8_livetile_tile_clear win8_livetile_badge_notification "+"win8_livetile_badge_clear win8_livetile_queue_enable "+"win8_secondarytile_pin win8_secondarytile_badge_notification "+"win8_secondarytile_delete win8_livetile_notification_begin "+"win8_livetile_notification_secondary_begin "+"win8_livetile_notification_expiry win8_livetile_notification_tag "+"win8_livetile_notification_text_add "+"win8_livetile_notification_image_add win8_livetile_notification_end "+"win8_appbar_enable win8_appbar_add_element "+"win8_appbar_remove_element win8_settingscharm_add_entry "+"win8_settingscharm_add_html_entry win8_settingscharm_add_xaml_entry "+"win8_settingscharm_set_xaml_property "+"win8_settingscharm_get_xaml_property win8_settingscharm_remove_entry "+"win8_share_image win8_share_screenshot win8_share_file "+"win8_share_url win8_share_text win8_search_enable "+"win8_search_disable win8_search_add_suggestions "+"win8_device_touchscreen_available win8_license_initialize_sandbox "+"win8_license_trial_version winphone_license_trial_version "+"winphone_tile_title winphone_tile_count winphone_tile_back_title "+"winphone_tile_back_content winphone_tile_back_content_wide "+"winphone_tile_front_image winphone_tile_front_image_small "+"winphone_tile_front_image_wide winphone_tile_back_image "+"winphone_tile_back_image_wide winphone_tile_background_colour "+"winphone_tile_background_color winphone_tile_icon_image "+"winphone_tile_small_icon_image winphone_tile_wide_content "+"winphone_tile_cycle_images winphone_tile_small_background_image "+"physics_world_create physics_world_gravity "+"physics_world_update_speed physics_world_update_iterations "+"physics_world_draw_debug physics_pause_enable physics_fixture_create "+"physics_fixture_set_kinematic physics_fixture_set_density "+"physics_fixture_set_awake physics_fixture_set_restitution "+"physics_fixture_set_friction physics_fixture_set_collision_group "+"physics_fixture_set_sensor physics_fixture_set_linear_damping "+"physics_fixture_set_angular_damping physics_fixture_set_circle_shape "+"physics_fixture_set_box_shape physics_fixture_set_edge_shape "+"physics_fixture_set_polygon_shape physics_fixture_set_chain_shape "+"physics_fixture_add_point physics_fixture_bind "+"physics_fixture_bind_ext physics_fixture_delete physics_apply_force "+"physics_apply_impulse physics_apply_angular_impulse "+"physics_apply_local_force physics_apply_local_impulse "+"physics_apply_torque physics_mass_properties physics_draw_debug "+"physics_test_overlap physics_remove_fixture physics_set_friction "+"physics_set_density physics_set_restitution physics_get_friction "+"physics_get_density physics_get_restitution "+"physics_joint_distance_create physics_joint_rope_create "+"physics_joint_revolute_create physics_joint_prismatic_create "+"physics_joint_pulley_create physics_joint_wheel_create "+"physics_joint_weld_create physics_joint_friction_create "+"physics_joint_gear_create physics_joint_enable_motor "+"physics_joint_get_value physics_joint_set_value physics_joint_delete "+"physics_particle_create physics_particle_delete "+"physics_particle_delete_region_circle "+"physics_particle_delete_region_box "+"physics_particle_delete_region_poly physics_particle_set_flags "+"physics_particle_set_category_flags physics_particle_draw "+"physics_particle_draw_ext physics_particle_count "+"physics_particle_get_data physics_particle_get_data_particle "+"physics_particle_group_begin physics_particle_group_circle "+"physics_particle_group_box physics_particle_group_polygon "+"physics_particle_group_add_point physics_particle_group_end "+"physics_particle_group_join physics_particle_group_delete "+"physics_particle_group_count physics_particle_group_get_data "+"physics_particle_group_get_mass physics_particle_group_get_inertia "+"physics_particle_group_get_centre_x "+"physics_particle_group_get_centre_y physics_particle_group_get_vel_x "+"physics_particle_group_get_vel_y physics_particle_group_get_ang_vel "+"physics_particle_group_get_x physics_particle_group_get_y "+"physics_particle_group_get_angle physics_particle_set_group_flags "+"physics_particle_get_group_flags physics_particle_get_max_count "+"physics_particle_get_radius physics_particle_get_density "+"physics_particle_get_damping physics_particle_get_gravity_scale "+"physics_particle_set_max_count physics_particle_set_radius "+"physics_particle_set_density physics_particle_set_damping "+"physics_particle_set_gravity_scale network_create_socket "+"network_create_socket_ext network_create_server "+"network_create_server_raw network_connect network_connect_raw "+"network_send_packet network_send_raw network_send_broadcast "+"network_send_udp network_send_udp_raw network_set_timeout "+"network_set_config network_resolve network_destroy buffer_create "+"buffer_write buffer_read buffer_seek buffer_get_surface "+"buffer_set_surface buffer_delete buffer_exists buffer_get_type "+"buffer_get_alignment buffer_poke buffer_peek buffer_save "+"buffer_save_ext buffer_load buffer_load_ext buffer_load_partial "+"buffer_copy buffer_fill buffer_get_size buffer_tell buffer_resize "+"buffer_md5 buffer_sha1 buffer_base64_encode buffer_base64_decode "+"buffer_base64_decode_ext buffer_sizeof buffer_get_address "+"buffer_create_from_vertex_buffer "+"buffer_create_from_vertex_buffer_ext buffer_copy_from_vertex_buffer "+"buffer_async_group_begin buffer_async_group_option "+"buffer_async_group_end buffer_load_async buffer_save_async "+"gml_release_mode gml_pragma steam_activate_overlay "+"steam_is_overlay_enabled steam_is_overlay_activated "+"steam_get_persona_name steam_initialised "+"steam_is_cloud_enabled_for_app steam_is_cloud_enabled_for_account "+"steam_file_persisted steam_get_quota_total steam_get_quota_free "+"steam_file_write steam_file_write_file steam_file_read "+"steam_file_delete steam_file_exists steam_file_size steam_file_share "+"steam_is_screenshot_requested steam_send_screenshot "+"steam_is_user_logged_on steam_get_user_steam_id steam_user_owns_dlc "+"steam_user_installed_dlc steam_set_achievement steam_get_achievement "+"steam_clear_achievement steam_set_stat_int steam_set_stat_float "+"steam_set_stat_avg_rate steam_get_stat_int steam_get_stat_float "+"steam_get_stat_avg_rate steam_reset_all_stats "+"steam_reset_all_stats_achievements steam_stats_ready "+"steam_create_leaderboard steam_upload_score steam_upload_score_ext "+"steam_download_scores_around_user steam_download_scores "+"steam_download_friends_scores steam_upload_score_buffer "+"steam_upload_score_buffer_ext steam_current_game_language "+"steam_available_languages steam_activate_overlay_browser "+"steam_activate_overlay_user steam_activate_overlay_store "+"steam_get_user_persona_name steam_get_app_id "+"steam_get_user_account_id steam_ugc_download steam_ugc_create_item "+"steam_ugc_start_item_update steam_ugc_set_item_title "+"steam_ugc_set_item_description steam_ugc_set_item_visibility "+"steam_ugc_set_item_tags steam_ugc_set_item_content "+"steam_ugc_set_item_preview steam_ugc_submit_item_update "+"steam_ugc_get_item_update_progress steam_ugc_subscribe_item "+"steam_ugc_unsubscribe_item steam_ugc_num_subscribed_items "+"steam_ugc_get_subscribed_items steam_ugc_get_item_install_info "+"steam_ugc_get_item_update_info steam_ugc_request_item_details "+"steam_ugc_create_query_user steam_ugc_create_query_user_ex "+"steam_ugc_create_query_all steam_ugc_create_query_all_ex "+"steam_ugc_query_set_cloud_filename_filter "+"steam_ugc_query_set_match_any_tag steam_ugc_query_set_search_text "+"steam_ugc_query_set_ranked_by_trend_days "+"steam_ugc_query_add_required_tag steam_ugc_query_add_excluded_tag "+"steam_ugc_query_set_return_long_description "+"steam_ugc_query_set_return_total_only "+"steam_ugc_query_set_allow_cached_response steam_ugc_send_query "+"shader_set shader_get_name shader_reset shader_current "+"shader_is_compiled shader_get_sampler_index shader_get_uniform "+"shader_set_uniform_i shader_set_uniform_i_array shader_set_uniform_f "+"shader_set_uniform_f_array shader_set_uniform_matrix "+"shader_set_uniform_matrix_array shader_enable_corner_id "+"texture_set_stage texture_get_texel_width texture_get_texel_height "+"shaders_are_supported vertex_format_begin vertex_format_end "+"vertex_format_delete vertex_format_add_position "+"vertex_format_add_position_3d vertex_format_add_colour "+"vertex_format_add_color vertex_format_add_normal "+"vertex_format_add_texcoord vertex_format_add_textcoord "+"vertex_format_add_custom vertex_create_buffer "+"vertex_create_buffer_ext vertex_delete_buffer vertex_begin "+"vertex_end vertex_position vertex_position_3d vertex_colour "+"vertex_color vertex_argb vertex_texcoord vertex_normal vertex_float1 "+"vertex_float2 vertex_float3 vertex_float4 vertex_ubyte4 "+"vertex_submit vertex_freeze vertex_get_number vertex_get_buffer_size "+"vertex_create_buffer_from_buffer "+"vertex_create_buffer_from_buffer_ext push_local_notification "+"push_get_first_local_notification push_get_next_local_notification "+"push_cancel_local_notification skeleton_animation_set "+"skeleton_animation_get skeleton_animation_mix "+"skeleton_animation_set_ext skeleton_animation_get_ext "+"skeleton_animation_get_duration skeleton_animation_get_frames "+"skeleton_animation_clear skeleton_skin_set skeleton_skin_get "+"skeleton_attachment_set skeleton_attachment_get "+"skeleton_attachment_create skeleton_collision_draw_set "+"skeleton_bone_data_get skeleton_bone_data_set "+"skeleton_bone_state_get skeleton_bone_state_set skeleton_get_minmax "+"skeleton_get_num_bounds skeleton_get_bounds "+"skeleton_animation_get_frame skeleton_animation_set_frame "+"draw_skeleton draw_skeleton_time draw_skeleton_instance "+"draw_skeleton_collision skeleton_animation_list skeleton_skin_list "+"skeleton_slot_data layer_get_id layer_get_id_at_depth "+"layer_get_depth layer_create layer_destroy layer_destroy_instances "+"layer_add_instance layer_has_instance layer_set_visible "+"layer_get_visible layer_exists layer_x layer_y layer_get_x "+"layer_get_y layer_hspeed layer_vspeed layer_get_hspeed "+"layer_get_vspeed layer_script_begin layer_script_end layer_shader "+"layer_get_script_begin layer_get_script_end layer_get_shader "+"layer_set_target_room layer_get_target_room layer_reset_target_room "+"layer_get_all layer_get_all_elements layer_get_name layer_depth "+"layer_get_element_layer layer_get_element_type layer_element_move "+"layer_force_draw_depth layer_is_draw_depth_forced "+"layer_get_forced_depth layer_background_get_id "+"layer_background_exists layer_background_create "+"layer_background_destroy layer_background_visible "+"layer_background_change layer_background_sprite "+"layer_background_htiled layer_background_vtiled "+"layer_background_stretch layer_background_yscale "+"layer_background_xscale layer_background_blend "+"layer_background_alpha layer_background_index layer_background_speed "+"layer_background_get_visible layer_background_get_sprite "+"layer_background_get_htiled layer_background_get_vtiled "+"layer_background_get_stretch layer_background_get_yscale "+"layer_background_get_xscale layer_background_get_blend "+"layer_background_get_alpha layer_background_get_index "+"layer_background_get_speed layer_sprite_get_id layer_sprite_exists "+"layer_sprite_create layer_sprite_destroy layer_sprite_change "+"layer_sprite_index layer_sprite_speed layer_sprite_xscale "+"layer_sprite_yscale layer_sprite_angle layer_sprite_blend "+"layer_sprite_alpha layer_sprite_x layer_sprite_y "+"layer_sprite_get_sprite layer_sprite_get_index "+"layer_sprite_get_speed layer_sprite_get_xscale "+"layer_sprite_get_yscale layer_sprite_get_angle "+"layer_sprite_get_blend layer_sprite_get_alpha layer_sprite_get_x "+"layer_sprite_get_y layer_tilemap_get_id layer_tilemap_exists "+"layer_tilemap_create layer_tilemap_destroy tilemap_tileset tilemap_x "+"tilemap_y tilemap_set tilemap_set_at_pixel tilemap_get_tileset "+"tilemap_get_tile_width tilemap_get_tile_height tilemap_get_width "+"tilemap_get_height tilemap_get_x tilemap_get_y tilemap_get "+"tilemap_get_at_pixel tilemap_get_cell_x_at_pixel "+"tilemap_get_cell_y_at_pixel tilemap_clear draw_tilemap draw_tile "+"tilemap_set_global_mask tilemap_get_global_mask tilemap_set_mask "+"tilemap_get_mask tilemap_get_frame tile_set_empty tile_set_index "+"tile_set_flip tile_set_mirror tile_set_rotate tile_get_empty "+"tile_get_index tile_get_flip tile_get_mirror tile_get_rotate "+"layer_tile_exists layer_tile_create layer_tile_destroy "+"layer_tile_change layer_tile_xscale layer_tile_yscale "+"layer_tile_blend layer_tile_alpha layer_tile_x layer_tile_y "+"layer_tile_region layer_tile_visible layer_tile_get_sprite "+"layer_tile_get_xscale layer_tile_get_yscale layer_tile_get_blend "+"layer_tile_get_alpha layer_tile_get_x layer_tile_get_y "+"layer_tile_get_region layer_tile_get_visible "+"layer_instance_get_instance instance_activate_layer "+"instance_deactivate_layer camera_create camera_create_view "+"camera_destroy camera_apply camera_get_active camera_get_default "+"camera_set_default camera_set_view_mat camera_set_proj_mat "+"camera_set_update_script camera_set_begin_script "+"camera_set_end_script camera_set_view_pos camera_set_view_size "+"camera_set_view_speed camera_set_view_border camera_set_view_angle "+"camera_set_view_target camera_get_view_mat camera_get_proj_mat "+"camera_get_update_script camera_get_begin_script "+"camera_get_end_script camera_get_view_x camera_get_view_y "+"camera_get_view_width camera_get_view_height camera_get_view_speed_x "+"camera_get_view_speed_y camera_get_view_border_x "+"camera_get_view_border_y camera_get_view_angle "+"camera_get_view_target view_get_camera view_get_visible "+"view_get_xport view_get_yport view_get_wport view_get_hport "+"view_get_surface_id view_set_camera view_set_visible view_set_xport "+"view_set_yport view_set_wport view_set_hport view_set_surface_id "+"gesture_drag_time gesture_drag_distance gesture_flick_speed "+"gesture_double_tap_time gesture_double_tap_distance "+"gesture_pinch_distance gesture_pinch_angle_towards "+"gesture_pinch_angle_away gesture_rotate_time gesture_rotate_angle "+"gesture_tap_count gesture_get_drag_time gesture_get_drag_distance "+"gesture_get_flick_speed gesture_get_double_tap_time "+"gesture_get_double_tap_distance gesture_get_pinch_distance "+"gesture_get_pinch_angle_towards gesture_get_pinch_angle_away "+"gesture_get_rotate_time gesture_get_rotate_angle "+"gesture_get_tap_count keyboard_virtual_show keyboard_virtual_hide "+"keyboard_virtual_status keyboard_virtual_height",literal:"self other all noone global local undefined pointer_invalid "+"pointer_null path_action_stop path_action_restart "+"path_action_continue path_action_reverse true false pi GM_build_date "+"GM_version GM_runtime_version  timezone_local timezone_utc "+"gamespeed_fps gamespeed_microseconds  ev_create ev_destroy ev_step "+"ev_alarm ev_keyboard ev_mouse ev_collision ev_other ev_draw "+"ev_draw_begin ev_draw_end ev_draw_pre ev_draw_post ev_keypress "+"ev_keyrelease ev_trigger ev_left_button ev_right_button "+"ev_middle_button ev_no_button ev_left_press ev_right_press "+"ev_middle_press ev_left_release ev_right_release ev_middle_release "+"ev_mouse_enter ev_mouse_leave ev_mouse_wheel_up ev_mouse_wheel_down "+"ev_global_left_button ev_global_right_button ev_global_middle_button "+"ev_global_left_press ev_global_right_press ev_global_middle_press "+"ev_global_left_release ev_global_right_release "+"ev_global_middle_release ev_joystick1_left ev_joystick1_right "+"ev_joystick1_up ev_joystick1_down ev_joystick1_button1 "+"ev_joystick1_button2 ev_joystick1_button3 ev_joystick1_button4 "+"ev_joystick1_button5 ev_joystick1_button6 ev_joystick1_button7 "+"ev_joystick1_button8 ev_joystick2_left ev_joystick2_right "+"ev_joystick2_up ev_joystick2_down ev_joystick2_button1 "+"ev_joystick2_button2 ev_joystick2_button3 ev_joystick2_button4 "+"ev_joystick2_button5 ev_joystick2_button6 ev_joystick2_button7 "+"ev_joystick2_button8 ev_outside ev_boundary ev_game_start "+"ev_game_end ev_room_start ev_room_end ev_no_more_lives "+"ev_animation_end ev_end_of_path ev_no_more_health ev_close_button "+"ev_user0 ev_user1 ev_user2 ev_user3 ev_user4 ev_user5 ev_user6 "+"ev_user7 ev_user8 ev_user9 ev_user10 ev_user11 ev_user12 ev_user13 "+"ev_user14 ev_user15 ev_step_normal ev_step_begin ev_step_end ev_gui "+"ev_gui_begin ev_gui_end ev_cleanup ev_gesture ev_gesture_tap "+"ev_gesture_double_tap ev_gesture_drag_start ev_gesture_dragging "+"ev_gesture_drag_end ev_gesture_flick ev_gesture_pinch_start "+"ev_gesture_pinch_in ev_gesture_pinch_out ev_gesture_pinch_end "+"ev_gesture_rotate_start ev_gesture_rotating ev_gesture_rotate_end "+"ev_global_gesture_tap ev_global_gesture_double_tap "+"ev_global_gesture_drag_start ev_global_gesture_dragging "+"ev_global_gesture_drag_end ev_global_gesture_flick "+"ev_global_gesture_pinch_start ev_global_gesture_pinch_in "+"ev_global_gesture_pinch_out ev_global_gesture_pinch_end "+"ev_global_gesture_rotate_start ev_global_gesture_rotating "+"ev_global_gesture_rotate_end vk_nokey vk_anykey vk_enter vk_return "+"vk_shift vk_control vk_alt vk_escape vk_space vk_backspace vk_tab "+"vk_pause vk_printscreen vk_left vk_right vk_up vk_down vk_home "+"vk_end vk_delete vk_insert vk_pageup vk_pagedown vk_f1 vk_f2 vk_f3 "+"vk_f4 vk_f5 vk_f6 vk_f7 vk_f8 vk_f9 vk_f10 vk_f11 vk_f12 vk_numpad0 "+"vk_numpad1 vk_numpad2 vk_numpad3 vk_numpad4 vk_numpad5 vk_numpad6 "+"vk_numpad7 vk_numpad8 vk_numpad9 vk_divide vk_multiply vk_subtract "+"vk_add vk_decimal vk_lshift vk_lcontrol vk_lalt vk_rshift "+"vk_rcontrol vk_ralt  mb_any mb_none mb_left mb_right mb_middle "+"c_aqua c_black c_blue c_dkgray c_fuchsia c_gray c_green c_lime "+"c_ltgray c_maroon c_navy c_olive c_purple c_red c_silver c_teal "+"c_white c_yellow c_orange fa_left fa_center fa_right fa_top "+"fa_middle fa_bottom pr_pointlist pr_linelist pr_linestrip "+"pr_trianglelist pr_trianglestrip pr_trianglefan bm_complex bm_normal "+"bm_add bm_max bm_subtract bm_zero bm_one bm_src_colour "+"bm_inv_src_colour bm_src_color bm_inv_src_color bm_src_alpha "+"bm_inv_src_alpha bm_dest_alpha bm_inv_dest_alpha bm_dest_colour "+"bm_inv_dest_colour bm_dest_color bm_inv_dest_color bm_src_alpha_sat "+"tf_point tf_linear tf_anisotropic mip_off mip_on mip_markedonly "+"audio_falloff_none audio_falloff_inverse_distance "+"audio_falloff_inverse_distance_clamped audio_falloff_linear_distance "+"audio_falloff_linear_distance_clamped "+"audio_falloff_exponent_distance "+"audio_falloff_exponent_distance_clamped audio_old_system "+"audio_new_system audio_mono audio_stereo audio_3d cr_default cr_none "+"cr_arrow cr_cross cr_beam cr_size_nesw cr_size_ns cr_size_nwse "+"cr_size_we cr_uparrow cr_hourglass cr_drag cr_appstart cr_handpoint "+"cr_size_all spritespeed_framespersecond "+"spritespeed_framespergameframe asset_object asset_unknown "+"asset_sprite asset_sound asset_room asset_path asset_script "+"asset_font asset_timeline asset_tiles asset_shader fa_readonly "+"fa_hidden fa_sysfile fa_volumeid fa_directory fa_archive  "+"ds_type_map ds_type_list ds_type_stack ds_type_queue ds_type_grid "+"ds_type_priority ef_explosion ef_ring ef_ellipse ef_firework "+"ef_smoke ef_smokeup ef_star ef_spark ef_flare ef_cloud ef_rain "+"ef_snow pt_shape_pixel pt_shape_disk pt_shape_square pt_shape_line "+"pt_shape_star pt_shape_circle pt_shape_ring pt_shape_sphere "+"pt_shape_flare pt_shape_spark pt_shape_explosion pt_shape_cloud "+"pt_shape_smoke pt_shape_snow ps_distr_linear ps_distr_gaussian "+"ps_distr_invgaussian ps_shape_rectangle ps_shape_ellipse "+"ps_shape_diamond ps_shape_line ty_real ty_string dll_cdecl "+"dll_stdcall matrix_view matrix_projection matrix_world os_win32 "+"os_windows os_macosx os_ios os_android os_symbian os_linux "+"os_unknown os_winphone os_tizen os_win8native "+"os_wiiu os_3ds  os_psvita os_bb10 os_ps4 os_xboxone "+"os_ps3 os_xbox360 os_uwp os_tvos os_switch "+"browser_not_a_browser browser_unknown browser_ie browser_firefox "+"browser_chrome browser_safari browser_safari_mobile browser_opera "+"browser_tizen browser_edge browser_windows_store browser_ie_mobile  "+"device_ios_unknown device_ios_iphone device_ios_iphone_retina "+"device_ios_ipad device_ios_ipad_retina device_ios_iphone5 "+"device_ios_iphone6 device_ios_iphone6plus device_emulator "+"device_tablet display_landscape display_landscape_flipped "+"display_portrait display_portrait_flipped tm_sleep tm_countvsyncs "+"of_challenge_win of_challen ge_lose of_challenge_tie "+"leaderboard_type_number leaderboard_type_time_mins_secs "+"cmpfunc_never cmpfunc_less cmpfunc_equal cmpfunc_lessequal "+"cmpfunc_greater cmpfunc_notequal cmpfunc_greaterequal cmpfunc_always "+"cull_noculling cull_clockwise cull_counterclockwise lighttype_dir "+"lighttype_point iap_ev_storeload iap_ev_product iap_ev_purchase "+"iap_ev_consume iap_ev_restore iap_storeload_ok iap_storeload_failed "+"iap_status_uninitialised iap_status_unavailable iap_status_loading "+"iap_status_available iap_status_processing iap_status_restoring "+"iap_failed iap_unavailable iap_available iap_purchased iap_canceled "+"iap_refunded fb_login_default fb_login_fallback_to_webview "+"fb_login_no_fallback_to_webview fb_login_forcing_webview "+"fb_login_use_system_account fb_login_forcing_safari  "+"phy_joint_anchor_1_x phy_joint_anchor_1_y phy_joint_anchor_2_x "+"phy_joint_anchor_2_y phy_joint_reaction_force_x "+"phy_joint_reaction_force_y phy_joint_reaction_torque "+"phy_joint_motor_speed phy_joint_angle phy_joint_motor_torque "+"phy_joint_max_motor_torque phy_joint_translation phy_joint_speed "+"phy_joint_motor_force phy_joint_max_motor_force phy_joint_length_1 "+"phy_joint_length_2 phy_joint_damping_ratio phy_joint_frequency "+"phy_joint_lower_angle_limit phy_joint_upper_angle_limit "+"phy_joint_angle_limits phy_joint_max_length phy_joint_max_torque "+"phy_joint_max_force phy_debug_render_aabb "+"phy_debug_render_collision_pairs phy_debug_render_coms "+"phy_debug_render_core_shapes phy_debug_render_joints "+"phy_debug_render_obb phy_debug_render_shapes  "+"phy_particle_flag_water phy_particle_flag_zombie "+"phy_particle_flag_wall phy_particle_flag_spring "+"phy_particle_flag_elastic phy_particle_flag_viscous "+"phy_particle_flag_powder phy_particle_flag_tensile "+"phy_particle_flag_colourmixing phy_particle_flag_colormixing "+"phy_particle_group_flag_solid phy_particle_group_flag_rigid "+"phy_particle_data_flag_typeflags phy_particle_data_flag_position "+"phy_particle_data_flag_velocity phy_particle_data_flag_colour "+"phy_particle_data_flag_color phy_particle_data_flag_category  "+"achievement_our_info achievement_friends_info "+"achievement_leaderboard_info achievement_achievement_info "+"achievement_filter_all_players achievement_filter_friends_only "+"achievement_filter_favorites_only "+"achievement_type_achievement_challenge "+"achievement_type_score_challenge achievement_pic_loaded  "+"achievement_show_ui achievement_show_profile "+"achievement_show_leaderboard achievement_show_achievement "+"achievement_show_bank achievement_show_friend_picker "+"achievement_show_purchase_prompt network_socket_tcp "+"network_socket_udp network_socket_bluetooth network_type_connect "+"network_type_disconnect network_type_data "+"network_type_non_blocking_connect network_config_connect_timeout "+"network_config_use_non_blocking_socket "+"network_config_enable_reliable_udp "+"network_config_disable_reliable_udp buffer_fixed buffer_grow "+"buffer_wrap buffer_fast buffer_vbuffer buffer_network buffer_u8 "+"buffer_s8 buffer_u16 buffer_s16 buffer_u32 buffer_s32 buffer_u64 "+"buffer_f16 buffer_f32 buffer_f64 buffer_bool buffer_text "+"buffer_string buffer_surface_copy buffer_seek_start "+"buffer_seek_relative buffer_seek_end "+"buffer_generalerror buffer_outofspace buffer_outofbounds "+"buffer_invalidtype  text_type button_type input_type ANSI_CHARSET "+"DEFAULT_CHARSET EASTEUROPE_CHARSET RUSSIAN_CHARSET SYMBOL_CHARSET "+"SHIFTJIS_CHARSET HANGEUL_CHARSET GB2312_CHARSET CHINESEBIG5_CHARSET "+"JOHAB_CHARSET HEBREW_CHARSET ARABIC_CHARSET GREEK_CHARSET "+"TURKISH_CHARSET VIETNAMESE_CHARSET THAI_CHARSET MAC_CHARSET "+"BALTIC_CHARSET OEM_CHARSET  gp_face1 gp_face2 gp_face3 gp_face4 "+"gp_shoulderl gp_shoulderr gp_shoulderlb gp_shoulderrb gp_select "+"gp_start gp_stickl gp_stickr gp_padu gp_padd gp_padl gp_padr "+"gp_axislh gp_axislv gp_axisrh gp_axisrv ov_friends ov_community "+"ov_players ov_settings ov_gamegroup ov_achievements lb_sort_none "+"lb_sort_ascending lb_sort_descending lb_disp_none lb_disp_numeric "+"lb_disp_time_sec lb_disp_time_ms ugc_result_success "+"ugc_filetype_community ugc_filetype_microtrans ugc_visibility_public "+"ugc_visibility_friends_only ugc_visibility_private "+"ugc_query_RankedByVote ugc_query_RankedByPublicationDate "+"ugc_query_AcceptedForGameRankedByAcceptanceDate "+"ugc_query_RankedByTrend "+"ugc_query_FavoritedByFriendsRankedByPublicationDate "+"ugc_query_CreatedByFriendsRankedByPublicationDate "+"ugc_query_RankedByNumTimesReported "+"ugc_query_CreatedByFollowedUsersRankedByPublicationDate "+"ugc_query_NotYetRated ugc_query_RankedByTotalVotesAsc "+"ugc_query_RankedByVotesUp ugc_query_RankedByTextSearch "+"ugc_sortorder_CreationOrderDesc ugc_sortorder_CreationOrderAsc "+"ugc_sortorder_TitleAsc ugc_sortorder_LastUpdatedDesc "+"ugc_sortorder_SubscriptionDateDesc ugc_sortorder_VoteScoreDesc "+"ugc_sortorder_ForModeration ugc_list_Published ugc_list_VotedOn "+"ugc_list_VotedUp ugc_list_VotedDown ugc_list_WillVoteLater "+"ugc_list_Favorited ugc_list_Subscribed ugc_list_UsedOrPlayed "+"ugc_list_Followed ugc_match_Items ugc_match_Items_Mtx "+"ugc_match_Items_ReadyToUse ugc_match_Collections ugc_match_Artwork "+"ugc_match_Videos ugc_match_Screenshots ugc_match_AllGuides "+"ugc_match_WebGuides ugc_match_IntegratedGuides "+"ugc_match_UsableInGame ugc_match_ControllerBindings  "+"vertex_usage_position vertex_usage_colour vertex_usage_color "+"vertex_usage_normal vertex_usage_texcoord vertex_usage_textcoord "+"vertex_usage_blendweight vertex_usage_blendindices "+"vertex_usage_psize vertex_usage_tangent vertex_usage_binormal "+"vertex_usage_fog vertex_usage_depth vertex_usage_sample "+"vertex_type_float1 vertex_type_float2 vertex_type_float3 "+"vertex_type_float4 vertex_type_colour vertex_type_color "+"vertex_type_ubyte4 layerelementtype_undefined "+"layerelementtype_background layerelementtype_instance "+"layerelementtype_oldtilemap layerelementtype_sprite "+"layerelementtype_tilemap layerelementtype_particlesystem "+"layerelementtype_tile tile_rotate tile_flip tile_mirror "+"tile_index_mask kbv_type_default kbv_type_ascii kbv_type_url "+"kbv_type_email kbv_type_numbers kbv_type_phone kbv_type_phone_name "+"kbv_returnkey_default kbv_returnkey_go kbv_returnkey_google "+"kbv_returnkey_join kbv_returnkey_next kbv_returnkey_route "+"kbv_returnkey_search kbv_returnkey_send kbv_returnkey_yahoo "+"kbv_returnkey_done kbv_returnkey_continue kbv_returnkey_emergency "+"kbv_autocapitalize_none kbv_autocapitalize_words "+"kbv_autocapitalize_sentences kbv_autocapitalize_characters",symbol:"argument_relative argument argument0 argument1 argument2 "+"argument3 argument4 argument5 argument6 argument7 argument8 "+"argument9 argument10 argument11 argument12 argument13 argument14 "+"argument15 argument_count x y xprevious yprevious xstart ystart "+"hspeed vspeed direction speed friction gravity gravity_direction "+"path_index path_position path_positionprevious path_speed "+"path_scale path_orientation path_endaction object_index id solid "+"persistent mask_index instance_count instance_id room_speed fps "+"fps_real current_time current_year current_month current_day "+"current_weekday current_hour current_minute current_second alarm "+"timeline_index timeline_position timeline_speed timeline_running "+"timeline_loop room room_first room_last room_width room_height "+"room_caption room_persistent score lives health show_score "+"show_lives show_health caption_score caption_lives caption_health "+"event_type event_number event_object event_action "+"application_surface gamemaker_pro gamemaker_registered "+"gamemaker_version error_occurred error_last debug_mode "+"keyboard_key keyboard_lastkey keyboard_lastchar keyboard_string "+"mouse_x mouse_y mouse_button mouse_lastbutton cursor_sprite "+"visible sprite_index sprite_width sprite_height sprite_xoffset "+"sprite_yoffset image_number image_index image_speed depth "+"image_xscale image_yscale image_angle image_alpha image_blend "+"bbox_left bbox_right bbox_top bbox_bottom layer background_colour  "+"background_showcolour background_color background_showcolor "+"view_enabled view_current view_visible view_xview view_yview "+"view_wview view_hview view_xport view_yport view_wport view_hport "+"view_angle view_hborder view_vborder view_hspeed view_vspeed "+"view_object view_surface_id view_camera game_id game_display_name "+"game_project_name game_save_id working_directory temp_directory "+"program_directory browser_width browser_height os_type os_device "+"os_browser os_version display_aa async_load delta_time "+"webgl_enabled event_data iap_data phy_rotation phy_position_x "+"phy_position_y phy_angular_velocity phy_linear_velocity_x "+"phy_linear_velocity_y phy_speed_x phy_speed_y phy_speed "+"phy_angular_damping phy_linear_damping phy_bullet "+"phy_fixed_rotation phy_active phy_mass phy_inertia phy_com_x "+"phy_com_y phy_dynamic phy_kinematic phy_sleeping "+"phy_collision_points phy_collision_x phy_collision_y "+"phy_col_normal_x phy_col_normal_y phy_position_xprevious "+"phy_position_yprevious"};return{aliases:["gml","GML"],case_insensitive:false,keywords:GML_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("smali",function(hljs){var smali_instr_low_prio=["add","and","cmp","cmpg","cmpl","const","div","double","float","goto","if","int","long","move","mul","neg","new","nop","not","or","rem","return","shl","shr","sput","sub","throw","ushr","xor"];var smali_instr_high_prio=["aget","aput","array","check","execute","fill","filled","goto/16","goto/32","iget","instance","invoke","iput","monitor","packed","sget","sparse"];var smali_keywords=["transient","constructor","abstract","final","synthetic","public","private","protected","static","bridge","system"];return{aliases:["smali"],contains:[{className:"string",begin:'"',end:'"',relevance:0},hljs.COMMENT("#","$",{relevance:0}),{className:"keyword",variants:[{begin:"\\s*\\.end\\s[a-zA-Z0-9]*"},{begin:"^[ ]*\\.[a-zA-Z]*",relevance:0},{begin:"\\s:[a-zA-Z_0-9]*",relevance:0},{begin:"\\s("+smali_keywords.join("|")+")"}]},{className:"built_in",variants:[{begin:"\\s("+smali_instr_low_prio.join("|")+")\\s"},{begin:"\\s("+smali_instr_low_prio.join("|")+")((\\-|/)[a-zA-Z0-9]+)+\\s",relevance:10},{begin:"\\s("+smali_instr_high_prio.join("|")+")((\\-|/)[a-zA-Z0-9]+)*\\s",relevance:10}]},{className:"class",begin:"L[^(;:\n]*;",relevance:0},{begin:"[vp][0-9]+"}]}});hljs.registerLanguage("lsl",function(hljs){var LSL_STRING_ESCAPE_CHARS={className:"subst",begin:/\\[tn"\\]/};var LSL_STRINGS={className:"string",begin:'"',end:'"',contains:[LSL_STRING_ESCAPE_CHARS]};var LSL_NUMBERS={className:"number",begin:hljs.C_NUMBER_RE};var LSL_CONSTANTS={className:"literal",variants:[{begin:"\\b(?:PI|TWO_PI|PI_BY_TWO|DEG_TO_RAD|RAD_TO_DEG|SQRT2)\\b"},{begin:"\\b(?:XP_ERROR_(?:EXPERIENCES_DISABLED|EXPERIENCE_(?:DISABLED|SUSPENDED)|INVALID_(?:EXPERIENCE|PARAMETERS)|KEY_NOT_FOUND|MATURITY_EXCEEDED|NONE|NOT_(?:FOUND|PERMITTED(?:_LAND)?)|NO_EXPERIENCE|QUOTA_EXCEEDED|RETRY_UPDATE|STORAGE_EXCEPTION|STORE_DISABLED|THROTTLED|UNKNOWN_ERROR)|JSON_APPEND|STATUS_(?:PHYSICS|ROTATE_[XYZ]|PHANTOM|SANDBOX|BLOCK_GRAB(?:_OBJECT)?|(?:DIE|RETURN)_AT_EDGE|CAST_SHADOWS|OK|MALFORMED_PARAMS|TYPE_MISMATCH|BOUNDS_ERROR|NOT_(?:FOUND|SUPPORTED)|INTERNAL_ERROR|WHITELIST_FAILED)|AGENT(?:_(?:BY_(?:LEGACY_|USER)NAME|FLYING|ATTACHMENTS|SCRIPTED|MOUSELOOK|SITTING|ON_OBJECT|AWAY|WALKING|IN_AIR|TYPING|CROUCHING|BUSY|ALWAYS_RUN|AUTOPILOT|LIST_(?:PARCEL(?:_OWNER)?|REGION)))?|CAMERA_(?:PITCH|DISTANCE|BEHINDNESS_(?:ANGLE|LAG)|(?:FOCUS|POSITION)(?:_(?:THRESHOLD|LOCKED|LAG))?|FOCUS_OFFSET|ACTIVE)|ANIM_ON|LOOP|REVERSE|PING_PONG|SMOOTH|ROTATE|SCALE|ALL_SIDES|LINK_(?:ROOT|SET|ALL_(?:OTHERS|CHILDREN)|THIS)|ACTIVE|PASS(?:IVE|_(?:ALWAYS|IF_NOT_HANDLED|NEVER))|SCRIPTED|CONTROL_(?:FWD|BACK|(?:ROT_)?(?:LEFT|RIGHT)|UP|DOWN|(?:ML_)?LBUTTON)|PERMISSION_(?:RETURN_OBJECTS|DEBIT|OVERRIDE_ANIMATIONS|SILENT_ESTATE_MANAGEMENT|TAKE_CONTROLS|TRIGGER_ANIMATION|ATTACH|CHANGE_LINKS|(?:CONTROL|TRACK)_CAMERA|TELEPORT)|INVENTORY_(?:TEXTURE|SOUND|OBJECT|SCRIPT|LANDMARK|CLOTHING|NOTECARD|BODYPART|ANIMATION|GESTURE|ALL|NONE)|CHANGED_(?:INVENTORY|COLOR|SHAPE|SCALE|TEXTURE|LINK|ALLOWED_DROP|OWNER|REGION(?:_START)?|TELEPORT|MEDIA)|OBJECT_(?:CLICK_ACTION|HOVER_HEIGHT|LAST_OWNER_ID|(?:PHYSICS|SERVER|STREAMING)_COST|UNKNOWN_DETAIL|CHARACTER_TIME|PHANTOM|PHYSICS|TEMP_(?:ATTACHED|ON_REZ)|NAME|DESC|POS|PRIM_(?:COUNT|EQUIVALENCE)|RETURN_(?:PARCEL(?:_OWNER)?|REGION)|REZZER_KEY|ROO?T|VELOCITY|OMEGA|OWNER|GROUP(?:_TAG)?|CREATOR|ATTACHED_(?:POINT|SLOTS_AVAILABLE)|RENDER_WEIGHT|(?:BODY_SHAPE|PATHFINDING)_TYPE|(?:RUNNING|TOTAL)_SCRIPT_COUNT|TOTAL_INVENTORY_COUNT|SCRIPT_(?:MEMORY|TIME))|TYPE_(?:INTEGER|FLOAT|STRING|KEY|VECTOR|ROTATION|INVALID)|(?:DEBUG|PUBLIC)_CHANNEL|ATTACH_(?:AVATAR_CENTER|CHEST|HEAD|BACK|PELVIS|MOUTH|CHIN|NECK|NOSE|BELLY|[LR](?:SHOULDER|HAND|FOOT|EAR|EYE|[UL](?:ARM|LEG)|HIP)|(?:LEFT|RIGHT)_PEC|HUD_(?:CENTER_[12]|TOP_(?:RIGHT|CENTER|LEFT)|BOTTOM(?:_(?:RIGHT|LEFT))?)|[LR]HAND_RING1|TAIL_(?:BASE|TIP)|[LR]WING|FACE_(?:JAW|[LR]EAR|[LR]EYE|TOUNGE)|GROIN|HIND_[LR]FOOT)|LAND_(?:LEVEL|RAISE|LOWER|SMOOTH|NOISE|REVERT)|DATA_(?:ONLINE|NAME|BORN|SIM_(?:POS|STATUS|RATING)|PAYINFO)|PAYMENT_INFO_(?:ON_FILE|USED)|REMOTE_DATA_(?:CHANNEL|REQUEST|REPLY)|PSYS_(?:PART_(?:BF_(?:ZERO|ONE(?:_MINUS_(?:DEST_COLOR|SOURCE_(ALPHA|COLOR)))?|DEST_COLOR|SOURCE_(ALPHA|COLOR))|BLEND_FUNC_(DEST|SOURCE)|FLAGS|(?:START|END)_(?:COLOR|ALPHA|SCALE|GLOW)|MAX_AGE|(?:RIBBON|WIND|INTERP_(?:COLOR|SCALE)|BOUNCE|FOLLOW_(?:SRC|VELOCITY)|TARGET_(?:POS|LINEAR)|EMISSIVE)_MASK)|SRC_(?:MAX_AGE|PATTERN|ANGLE_(?:BEGIN|END)|BURST_(?:RATE|PART_COUNT|RADIUS|SPEED_(?:MIN|MAX))|ACCEL|TEXTURE|TARGET_KEY|OMEGA|PATTERN_(?:DROP|EXPLODE|ANGLE(?:_CONE(?:_EMPTY)?)?)))|VEHICLE_(?:REFERENCE_FRAME|TYPE_(?:NONE|SLED|CAR|BOAT|AIRPLANE|BALLOON)|(?:LINEAR|ANGULAR)_(?:FRICTION_TIMESCALE|MOTOR_DIRECTION)|LINEAR_MOTOR_OFFSET|HOVER_(?:HEIGHT|EFFICIENCY|TIMESCALE)|BUOYANCY|(?:LINEAR|ANGULAR)_(?:DEFLECTION_(?:EFFICIENCY|TIMESCALE)|MOTOR_(?:DECAY_)?TIMESCALE)|VERTICAL_ATTRACTION_(?:EFFICIENCY|TIMESCALE)|BANKING_(?:EFFICIENCY|MIX|TIMESCALE)|FLAG_(?:NO_DEFLECTION_UP|LIMIT_(?:ROLL_ONLY|MOTOR_UP)|HOVER_(?:(?:WATER|TERRAIN|UP)_ONLY|GLOBAL_HEIGHT)|MOUSELOOK_(?:STEER|BANK)|CAMERA_DECOUPLED))|PRIM_(?:ALLOW_UNSIT|ALPHA_MODE(?:_(?:BLEND|EMISSIVE|MASK|NONE))?|NORMAL|SPECULAR|TYPE(?:_(?:BOX|CYLINDER|PRISM|SPHERE|TORUS|TUBE|RING|SCULPT))?|HOLE_(?:DEFAULT|CIRCLE|SQUARE|TRIANGLE)|MATERIAL(?:_(?:STONE|METAL|GLASS|WOOD|FLESH|PLASTIC|RUBBER))?|SHINY_(?:NONE|LOW|MEDIUM|HIGH)|BUMP_(?:NONE|BRIGHT|DARK|WOOD|BARK|BRICKS|CHECKER|CONCRETE|TILE|STONE|DISKS|GRAVEL|BLOBS|SIDING|LARGETILE|STUCCO|SUCTION|WEAVE)|TEXGEN_(?:DEFAULT|PLANAR)|SCRIPTED_SIT_ONLY|SCULPT_(?:TYPE_(?:SPHERE|TORUS|PLANE|CYLINDER|MASK)|FLAG_(?:MIRROR|INVERT))|PHYSICS(?:_(?:SHAPE_(?:CONVEX|NONE|PRIM|TYPE)))?|(?:POS|ROT)_LOCAL|SLICE|TEXT|FLEXIBLE|POINT_LIGHT|TEMP_ON_REZ|PHANTOM|POSITION|SIT_TARGET|SIZE|ROTATION|TEXTURE|NAME|OMEGA|DESC|LINK_TARGET|COLOR|BUMP_SHINY|FULLBRIGHT|TEXGEN|GLOW|MEDIA_(?:ALT_IMAGE_ENABLE|CONTROLS|(?:CURRENT|HOME)_URL|AUTO_(?:LOOP|PLAY|SCALE|ZOOM)|FIRST_CLICK_INTERACT|(?:WIDTH|HEIGHT)_PIXELS|WHITELIST(?:_ENABLE)?|PERMS_(?:INTERACT|CONTROL)|PARAM_MAX|CONTROLS_(?:STANDARD|MINI)|PERM_(?:NONE|OWNER|GROUP|ANYONE)|MAX_(?:URL_LENGTH|WHITELIST_(?:SIZE|COUNT)|(?:WIDTH|HEIGHT)_PIXELS)))|MASK_(?:BASE|OWNER|GROUP|EVERYONE|NEXT)|PERM_(?:TRANSFER|MODIFY|COPY|MOVE|ALL)|PARCEL_(?:MEDIA_COMMAND_(?:STOP|PAUSE|PLAY|LOOP|TEXTURE|URL|TIME|AGENT|UNLOAD|AUTO_ALIGN|TYPE|SIZE|DESC|LOOP_SET)|FLAG_(?:ALLOW_(?:FLY|(?:GROUP_)?SCRIPTS|LANDMARK|TERRAFORM|DAMAGE|CREATE_(?:GROUP_)?OBJECTS)|USE_(?:ACCESS_(?:GROUP|LIST)|BAN_LIST|LAND_PASS_LIST)|LOCAL_SOUND_ONLY|RESTRICT_PUSHOBJECT|ALLOW_(?:GROUP|ALL)_OBJECT_ENTRY)|COUNT_(?:TOTAL|OWNER|GROUP|OTHER|SELECTED|TEMP)|DETAILS_(?:NAME|DESC|OWNER|GROUP|AREA|ID|SEE_AVATARS))|LIST_STAT_(?:MAX|MIN|MEAN|MEDIAN|STD_DEV|SUM(?:_SQUARES)?|NUM_COUNT|GEOMETRIC_MEAN|RANGE)|PAY_(?:HIDE|DEFAULT)|REGION_FLAG_(?:ALLOW_DAMAGE|FIXED_SUN|BLOCK_TERRAFORM|SANDBOX|DISABLE_(?:COLLISIONS|PHYSICS)|BLOCK_FLY|ALLOW_DIRECT_TELEPORT|RESTRICT_PUSHOBJECT)|HTTP_(?:METHOD|MIMETYPE|BODY_(?:MAXLENGTH|TRUNCATED)|CUSTOM_HEADER|PRAGMA_NO_CACHE|VERBOSE_THROTTLE|VERIFY_CERT)|SIT_(?:INVALID_(?:AGENT|LINK_OBJECT)|NO(?:T_EXPERIENCE|_(?:ACCESS|EXPERIENCE_PERMISSION|SIT_TARGET)))|STRING_(?:TRIM(?:_(?:HEAD|TAIL))?)|CLICK_ACTION_(?:NONE|TOUCH|SIT|BUY|PAY|OPEN(?:_MEDIA)?|PLAY|ZOOM)|TOUCH_INVALID_FACE|PROFILE_(?:NONE|SCRIPT_MEMORY)|RC_(?:DATA_FLAGS|DETECT_PHANTOM|GET_(?:LINK_NUM|NORMAL|ROOT_KEY)|MAX_HITS|REJECT_(?:TYPES|AGENTS|(?:NON)?PHYSICAL|LAND))|RCERR_(?:CAST_TIME_EXCEEDED|SIM_PERF_LOW|UNKNOWN)|ESTATE_ACCESS_(?:ALLOWED_(?:AGENT|GROUP)_(?:ADD|REMOVE)|BANNED_AGENT_(?:ADD|REMOVE))|DENSITY|FRICTION|RESTITUTION|GRAVITY_MULTIPLIER|KFM_(?:COMMAND|CMD_(?:PLAY|STOP|PAUSE)|MODE|FORWARD|LOOP|PING_PONG|REVERSE|DATA|ROTATION|TRANSLATION)|ERR_(?:GENERIC|PARCEL_PERMISSIONS|MALFORMED_PARAMS|RUNTIME_PERMISSIONS|THROTTLED)|CHARACTER_(?:CMD_(?:(?:SMOOTH_)?STOP|JUMP)|DESIRED_(?:TURN_)?SPEED|RADIUS|STAY_WITHIN_PARCEL|LENGTH|ORIENTATION|ACCOUNT_FOR_SKIPPED_FRAMES|AVOIDANCE_MODE|TYPE(?:_(?:[ABCD]|NONE))?|MAX_(?:DECEL|TURN_RADIUS|(?:ACCEL|SPEED)))|PURSUIT_(?:OFFSET|FUZZ_FACTOR|GOAL_TOLERANCE|INTERCEPT)|REQUIRE_LINE_OF_SIGHT|FORCE_DIRECT_PATH|VERTICAL|HORIZONTAL|AVOID_(?:CHARACTERS|DYNAMIC_OBSTACLES|NONE)|PU_(?:EVADE_(?:HIDDEN|SPOTTED)|FAILURE_(?:DYNAMIC_PATHFINDING_DISABLED|INVALID_(?:GOAL|START)|NO_(?:NAVMESH|VALID_DESTINATION)|OTHER|TARGET_GONE|(?:PARCEL_)?UNREACHABLE)|(?:GOAL|SLOWDOWN_DISTANCE)_REACHED)|TRAVERSAL_TYPE(?:_(?:FAST|NONE|SLOW))?|CONTENT_TYPE_(?:ATOM|FORM|HTML|JSON|LLSD|RSS|TEXT|XHTML|XML)|GCNP_(?:RADIUS|STATIC)|(?:PATROL|WANDER)_PAUSE_AT_WAYPOINTS|OPT_(?:AVATAR|CHARACTER|EXCLUSION_VOLUME|LEGACY_LINKSET|MATERIAL_VOLUME|OTHER|STATIC_OBSTACLE|WALKABLE)|SIM_STAT_PCT_CHARS_STEPPED)\\b"},{begin:"\\b(?:FALSE|TRUE)\\b"},{begin:"\\b(?:ZERO_ROTATION)\\b"},{begin:"\\b(?:EOF|JSON_(?:ARRAY|DELETE|FALSE|INVALID|NULL|NUMBER|OBJECT|STRING|TRUE)|NULL_KEY|TEXTURE_(?:BLANK|DEFAULT|MEDIA|PLYWOOD|TRANSPARENT)|URL_REQUEST_(?:GRANTED|DENIED))\\b"},{begin:"\\b(?:ZERO_VECTOR|TOUCH_INVALID_(?:TEXCOORD|VECTOR))\\b"}]};var LSL_FUNCTIONS={className:"built_in",begin:"\\b(?:ll(?:AgentInExperience|(?:Create|DataSize|Delete|KeyCount|Keys|Read|Update)KeyValue|GetExperience(?:Details|ErrorMessage)|ReturnObjectsBy(?:ID|Owner)|Json(?:2List|[GS]etValue|ValueType)|Sin|Cos|Tan|Atan2|Sqrt|Pow|Abs|Fabs|Frand|Floor|Ceil|Round|Vec(?:Mag|Norm|Dist)|Rot(?:Between|2(?:Euler|Fwd|Left|Up))|(?:Euler|Axes)2Rot|Whisper|(?:Region|Owner)?Say|Shout|Listen(?:Control|Remove)?|Sensor(?:Repeat|Remove)?|Detected(?:Name|Key|Owner|Type|Pos|Vel|Grab|Rot|Group|LinkNumber)|Die|Ground|Wind|(?:[GS]et)(?:AnimationOverride|MemoryLimit|PrimMediaParams|ParcelMusicURL|Object(?:Desc|Name)|PhysicsMaterial|Status|Scale|Color|Alpha|Texture|Pos|Rot|Force|Torque)|ResetAnimationOverride|(?:Scale|Offset|Rotate)Texture|(?:Rot)?Target(?:Remove)?|(?:Stop)?MoveToTarget|Apply(?:Rotational)?Impulse|Set(?:KeyframedMotion|ContentType|RegionPos|(?:Angular)?Velocity|Buoyancy|HoverHeight|ForceAndTorque|TimerEvent|ScriptState|Damage|TextureAnim|Sound(?:Queueing|Radius)|Vehicle(?:Type|(?:Float|Vector|Rotation)Param)|(?:Touch|Sit)?Text|Camera(?:Eye|At)Offset|PrimitiveParams|ClickAction|Link(?:Alpha|Color|PrimitiveParams(?:Fast)?|Texture(?:Anim)?|Camera|Media)|RemoteScriptAccessPin|PayPrice|LocalRot)|ScaleByFactor|Get(?:(?:Max|Min)ScaleFactor|ClosestNavPoint|StaticPath|SimStats|Env|PrimitiveParams|Link(?:PrimitiveParams|Number(?:OfSides)?|Key|Name|Media)|HTTPHeader|FreeURLs|Object(?:Details|PermMask|PrimCount)|Parcel(?:MaxPrims|Details|Prim(?:Count|Owners))|Attached(?:List)?|(?:SPMax|Free|Used)Memory|Region(?:Name|TimeDilation|FPS|Corner|AgentCount)|Root(?:Position|Rotation)|UnixTime|(?:Parcel|Region)Flags|(?:Wall|GMT)clock|SimulatorHostname|BoundingBox|GeometricCenter|Creator|NumberOf(?:Prims|NotecardLines|Sides)|Animation(?:List)?|(?:Camera|Local)(?:Pos|Rot)|Vel|Accel|Omega|Time(?:stamp|OfDay)|(?:Object|CenterOf)?Mass|MassMKS|Energy|Owner|(?:Owner)?Key|SunDirection|Texture(?:Offset|Scale|Rot)|Inventory(?:Number|Name|Key|Type|Creator|PermMask)|Permissions(?:Key)?|StartParameter|List(?:Length|EntryType)|Date|Agent(?:Size|Info|Language|List)|LandOwnerAt|NotecardLine|Script(?:Name|State))|(?:Get|Reset|GetAndReset)Time|PlaySound(?:Slave)?|LoopSound(?:Master|Slave)?|(?:Trigger|Stop|Preload)Sound|(?:(?:Get|Delete)Sub|Insert)String|To(?:Upper|Lower)|Give(?:InventoryList|Money)|RezObject|(?:Stop)?LookAt|Sleep|CollisionFilter|(?:Take|Release)Controls|DetachFromAvatar|AttachToAvatar(?:Temp)?|InstantMessage|(?:GetNext)?Email|StopHover|MinEventDelay|RotLookAt|String(?:Length|Trim)|(?:Start|Stop)Animation|TargetOmega|Request(?:Experience)?Permissions|(?:Create|Break)Link|BreakAllLinks|(?:Give|Remove)Inventory|Water|PassTouches|Request(?:Agent|Inventory)Data|TeleportAgent(?:Home|GlobalCoords)?|ModifyLand|CollisionSound|ResetScript|MessageLinked|PushObject|PassCollisions|AxisAngle2Rot|Rot2(?:Axis|Angle)|A(?:cos|sin)|AngleBetween|AllowInventoryDrop|SubStringIndex|List2(?:CSV|Integer|Json|Float|String|Key|Vector|Rot|List(?:Strided)?)|DeleteSubList|List(?:Statistics|Sort|Randomize|(?:Insert|Find|Replace)List)|EdgeOfWorld|AdjustSoundVolume|Key2Name|TriggerSoundLimited|EjectFromLand|(?:CSV|ParseString)2List|OverMyLand|SameGroup|UnSit|Ground(?:Slope|Normal|Contour)|GroundRepel|(?:Set|Remove)VehicleFlags|SitOnLink|(?:AvatarOn)?(?:Link)?SitTarget|Script(?:Danger|Profiler)|Dialog|VolumeDetect|ResetOtherScript|RemoteLoadScriptPin|(?:Open|Close)RemoteDataChannel|SendRemoteData|RemoteDataReply|(?:Integer|String)ToBase64|XorBase64|Log(?:10)?|Base64To(?:String|Integer)|ParseStringKeepNulls|RezAtRoot|RequestSimulatorData|ForceMouselook|(?:Load|Release|(?:E|Une)scape)URL|ParcelMedia(?:CommandList|Query)|ModPow|MapDestination|(?:RemoveFrom|AddTo|Reset)Land(?:Pass|Ban)List|(?:Set|Clear)CameraParams|HTTP(?:Request|Response)|TextBox|DetectedTouch(?:UV|Face|Pos|(?:N|Bin)ormal|ST)|(?:MD5|SHA1|DumpList2)String|Request(?:Secure)?URL|Clear(?:Prim|Link)Media|(?:Link)?ParticleSystem|(?:Get|Request)(?:Username|DisplayName)|RegionSayTo|CastRay|GenerateKey|TransferLindenDollars|ManageEstateAccess|(?:Create|Delete)Character|ExecCharacterCmd|Evade|FleeFrom|NavigateTo|PatrolPoints|Pursue|UpdateCharacter|WanderWithin))\\b"};return{illegal:":",contains:[LSL_STRINGS,{className:"comment",variants:[hljs.COMMENT("//","$"),hljs.COMMENT("/\\*","\\*/")],relevance:0},LSL_NUMBERS,{className:"section",variants:[{begin:"\\b(?:state|default)\\b"},{begin:"\\b(?:state_(?:entry|exit)|touch(?:_(?:start|end))?|(?:land_)?collision(?:_(?:start|end))?|timer|listen|(?:no_)?sensor|control|(?:not_)?at_(?:rot_)?target|money|email|experience_permissions(?:_denied)?|run_time_permissions|changed|attach|dataserver|moving_(?:start|end)|link_message|(?:on|object)_rez|remote_data|http_re(?:sponse|quest)|path_update|transaction_result)\\b"}]},LSL_FUNCTIONS,LSL_CONSTANTS,{className:"type",begin:"\\b(?:integer|float|string|key|vector|quaternion|rotation|list)\\b"}]}});hljs.registerLanguage("markdown",function(hljs){return{aliases:["md","mkdown","mkd"],contains:[{className:"section",variants:[{begin:"^#{1,6}",end:"$"},{begin:"^.+?\\n[=-]{2,}$"}]},{begin:"<",end:">",subLanguage:"xml",relevance:0},{className:"bullet",begin:"^\\s*([*+-]|(\\d+\\.))\\s+"},{className:"strong",begin:"[*_]{2}.+?[*_]{2}"},{className:"emphasis",variants:[{begin:"\\*.+?\\*"},{begin:"_.+?_",relevance:0}]},{className:"quote",begin:"^>\\s+",end:"$"},{className:"code",variants:[{begin:"^```\\w*\\s*$",end:"^```[ ]*$"},{begin:"`.+?`"},{begin:"^( {4}|\\t)",end:"$",relevance:0}]},{begin:"^[-\\*]{3,}",end:"$"},{begin:"\\[.+?\\][\\(\\[].*?[\\)\\]]",returnBegin:true,contains:[{className:"string",begin:"\\[",end:"\\]",excludeBegin:true,returnEnd:true,relevance:0},{className:"link",begin:"\\]\\(",end:"\\)",excludeBegin:true,excludeEnd:true},{className:"symbol",begin:"\\]\\[",end:"\\]",excludeBegin:true,excludeEnd:true}],relevance:10},{begin:/^\[[^\n]+\]:/,returnBegin:true,contains:[{className:"symbol",begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true},{className:"link",begin:/:\s*/,end:/$/,excludeBegin:true}]}]}});hljs.registerLanguage("dart",function(hljs){var SUBST={className:"subst",variants:[{begin:"\\$[A-Za-z0-9_]+"}]};var BRACED_SUBST={className:"subst",variants:[{begin:"\\${",end:"}"}],keywords:"true false null this is new super"};var STRING={className:"string",variants:[{begin:"r'''",end:"'''"},{begin:'r"""',end:'"""'},{begin:"r'",end:"'",illegal:"\\n"},{begin:'r"',end:'"',illegal:"\\n"},{begin:"'''",end:"'''",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:'"""',end:'"""',contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]}]};BRACED_SUBST.contains=[hljs.C_NUMBER_MODE,STRING];var KEYWORDS={keyword:"abstract as assert async await break case catch class const continue covariant default deferred do "+"dynamic else enum export extends extension external factory false final finally for Function get hide if "+"implements import in inferface is library mixin new null on operator part rethrow return set show static "+"super switch sync this throw true try typedef var void while with yield",built_in:"Comparable DateTime Duration Function Iterable Iterator List Map Match Null Object Pattern RegExp Set "+"Stopwatch String StringBuffer StringSink Symbol Type Uri bool double dynamic int num print "+"Element ElementList document querySelector querySelectorAll window"};return{keywords:KEYWORDS,contains:[STRING,hljs.COMMENT("/\\*\\*","\\*/",{subLanguage:"markdown"}),hljs.COMMENT("///+\\s*","$",{contains:[{subLanguage:"markdown",begin:".",end:"$"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"},{begin:"=>"}]}});hljs.registerLanguage("xml",function(hljs){var XML_IDENT_RE="[A-Za-z0-9\\._:-]+";var XML_ENTITIES={className:"symbol",begin:"&[a-z]+;|&#[0-9]+;|&#x[a-f0-9]+;"};var XML_META_KEYWORDS={begin:"\\s",contains:[{className:"meta-keyword",begin:"#?[a-z_][a-z1-9_-]+",illegal:"\\n"}]};var XML_META_PAR_KEYWORDS=hljs.inherit(XML_META_KEYWORDS,{begin:"\\(",end:"\\)"});var APOS_META_STRING_MODE=hljs.inherit(hljs.APOS_STRING_MODE,{className:"meta-string"});var QUOTE_META_STRING_MODE=hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"});var TAG_INTERNALS={endsWithParent:true,illegal:/</,relevance:0,contains:[{className:"attr",begin:XML_IDENT_RE,relevance:0},{begin:/=\s*/,relevance:0,contains:[{className:"string",endsParent:true,variants:[{begin:/"/,end:/"/,contains:[XML_ENTITIES]},{begin:/'/,end:/'/,contains:[XML_ENTITIES]},{begin:/[^\s"'=<>`]+/}]}]}]};return{aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist","wsf","svg"],case_insensitive:true,contains:[{className:"meta",begin:"<![a-z]",end:">",relevance:10,contains:[XML_META_KEYWORDS,QUOTE_META_STRING_MODE,APOS_META_STRING_MODE,XML_META_PAR_KEYWORDS,{begin:"\\[",end:"\\]",contains:[{className:"meta",begin:"<![a-z]",end:">",contains:[XML_META_KEYWORDS,XML_META_PAR_KEYWORDS,QUOTE_META_STRING_MODE,APOS_META_STRING_MODE]}]}]},hljs.COMMENT("\x3c!--","--\x3e",{relevance:10}),{begin:"<\\!\\[CDATA\\[",end:"\\]\\]>",relevance:10},XML_ENTITIES,{className:"meta",begin:/<\?xml/,end:/\?>/,relevance:10},{begin:/<\?(php)?/,end:/\?>/,subLanguage:"php",contains:[{begin:"/\\*",end:"\\*/",skip:true},{begin:'b"',end:'"',skip:true},{begin:"b'",end:"'",skip:true},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null,className:null,contains:null,skip:true}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null,className:null,contains:null,skip:true})]},{className:"tag",begin:"<style(?=\\s|>)",end:">",keywords:{name:"style"},contains:[TAG_INTERNALS],starts:{end:"</style>",returnEnd:true,subLanguage:["css","xml"]}},{className:"tag",begin:"<script(?=\\s|>)",end:">",keywords:{name:"script"},contains:[TAG_INTERNALS],starts:{end:"<\/script>",returnEnd:true,subLanguage:["actionscript","javascript","handlebars","xml"]}},{className:"tag",begin:"</?",end:"/?>",contains:[{className:"name",begin:/[^\/><\s]+/,relevance:0},TAG_INTERNALS]}]}});hljs.registerLanguage("parser3",function(hljs){var CURLY_SUBCOMMENT=hljs.COMMENT("{","}",{contains:["self"]});return{subLanguage:"xml",relevance:0,contains:[hljs.COMMENT("^#","$"),hljs.COMMENT("\\^rem{","}",{relevance:10,contains:[CURLY_SUBCOMMENT]}),{className:"meta",begin:"^@(?:BASE|USE|CLASS|OPTIONS)$",relevance:10},{className:"title",begin:"@[\\w\\-]+\\[[\\w^;\\-]*\\](?:\\[[\\w^;\\-]*\\])?(?:.*)$"},{className:"variable",begin:"\\$\\{?[\\w\\-\\.\\:]+\\}?"},{className:"keyword",begin:"\\^[\\w\\-\\.\\:]+"},{className:"number",begin:"\\^#[0-9a-fA-F]+"},hljs.C_NUMBER_MODE]}});hljs.registerLanguage("sql",function(hljs){var COMMENT_MODE=hljs.COMMENT("--","$");return{case_insensitive:true,illegal:/[<>{}*]/,contains:[{beginKeywords:"begin end start commit rollback savepoint lock alter create drop rename call "+"delete do handler insert load replace select truncate update set show pragma grant "+"merge describe use explain help declare prepare execute deallocate release "+"unlock purge reset change stop analyze cache flush optimize repair kill "+"install uninstall checksum restore check backup revoke comment values with",end:/;/,endsWithParent:true,lexemes:/[\w\.]+/,keywords:{keyword:"as abort abs absolute acc acce accep accept access accessed accessible account acos action activate add "+"addtime admin administer advanced advise aes_decrypt aes_encrypt after agent aggregate ali alia alias "+"all allocate allow alter always analyze ancillary and anti any anydata anydataset anyschema anytype apply "+"archive archived archivelog are as asc ascii asin assembly assertion associate asynchronous at atan "+"atn2 attr attri attrib attribu attribut attribute attributes audit authenticated authentication authid "+"authors auto autoallocate autodblink autoextend automatic availability avg backup badfile basicfile "+"before begin beginning benchmark between bfile bfile_base big bigfile bin binary_double binary_float "+"binlog bit_and bit_count bit_length bit_or bit_xor bitmap blob_base block blocksize body both bound "+"bucket buffer_cache buffer_pool build bulk by byte byteordermark bytes cache caching call calling cancel "+"capacity cascade cascaded case cast catalog category ceil ceiling chain change changed char_base "+"char_length character_length characters characterset charindex charset charsetform charsetid check "+"checksum checksum_agg child choose chr chunk class cleanup clear client clob clob_base clone close "+"cluster_id cluster_probability cluster_set clustering coalesce coercibility col collate collation "+"collect colu colum column column_value columns columns_updated comment commit compact compatibility "+"compiled complete composite_limit compound compress compute concat concat_ws concurrent confirm conn "+"connec connect connect_by_iscycle connect_by_isleaf connect_by_root connect_time connection "+"consider consistent constant constraint constraints constructor container content contents context "+"contributors controlfile conv convert convert_tz corr corr_k corr_s corresponding corruption cos cost "+"count count_big counted covar_pop covar_samp cpu_per_call cpu_per_session crc32 create creation "+"critical cross cube cume_dist curdate current current_date current_time current_timestamp current_user "+"cursor curtime customdatum cycle data database databases datafile datafiles datalength date_add "+"date_cache date_format date_sub dateadd datediff datefromparts datename datepart datetime2fromparts "+"day day_to_second dayname dayofmonth dayofweek dayofyear days db_role_change dbtimezone ddl deallocate "+"declare decode decompose decrement decrypt deduplicate def defa defau defaul default defaults "+"deferred defi defin define degrees delayed delegate delete delete_all delimited demand dense_rank "+"depth dequeue des_decrypt des_encrypt des_key_file desc descr descri describ describe descriptor "+"deterministic diagnostics difference dimension direct_load directory disable disable_all "+"disallow disassociate discardfile disconnect diskgroup distinct distinctrow distribute distributed div "+"do document domain dotnet double downgrade drop dumpfile duplicate duration each edition editionable "+"editions element ellipsis else elsif elt empty enable enable_all enclosed encode encoding encrypt "+"end end-exec endian enforced engine engines enqueue enterprise entityescaping eomonth error errors "+"escaped evalname evaluate event eventdata events except exception exceptions exchange exclude excluding "+"execu execut execute exempt exists exit exp expire explain explode export export_set extended extent external "+"external_1 external_2 externally extract failed failed_login_attempts failover failure far fast "+"feature_set feature_value fetch field fields file file_name_convert filesystem_like_logging final "+"finish first first_value fixed flash_cache flashback floor flush following follows for forall force foreign "+"form forma format found found_rows freelist freelists freepools fresh from from_base64 from_days "+"ftp full function general generated get get_format get_lock getdate getutcdate global global_name "+"globally go goto grant grants greatest group group_concat group_id grouping grouping_id groups "+"gtid_subtract guarantee guard handler hash hashkeys having hea head headi headin heading heap help hex "+"hierarchy high high_priority hosts hour hours http id ident_current ident_incr ident_seed identified "+"identity idle_time if ifnull ignore iif ilike ilm immediate import in include including increment "+"index indexes indexing indextype indicator indices inet6_aton inet6_ntoa inet_aton inet_ntoa infile "+"initial initialized initially initrans inmemory inner innodb input insert install instance instantiable "+"instr interface interleaved intersect into invalidate invisible is is_free_lock is_ipv4 is_ipv4_compat "+"is_not is_not_null is_used_lock isdate isnull isolation iterate java join json json_exists "+"keep keep_duplicates key keys kill language large last last_day last_insert_id last_value lateral lax lcase "+"lead leading least leaves left len lenght length less level levels library like like2 like4 likec limit "+"lines link list listagg little ln load load_file lob lobs local localtime localtimestamp locate "+"locator lock locked log log10 log2 logfile logfiles logging logical logical_reads_per_call "+"logoff logon logs long loop low low_priority lower lpad lrtrim ltrim main make_set makedate maketime "+"managed management manual map mapping mask master master_pos_wait match matched materialized max "+"maxextents maximize maxinstances maxlen maxlogfiles maxloghistory maxlogmembers maxsize maxtrans "+"md5 measures median medium member memcompress memory merge microsecond mid migration min minextents "+"minimum mining minus minute minutes minvalue missing mod mode model modification modify module monitoring month "+"months mount move movement multiset mutex name name_const names nan national native natural nav nchar "+"nclob nested never new newline next nextval no no_write_to_binlog noarchivelog noaudit nobadfile "+"nocheck nocompress nocopy nocycle nodelay nodiscardfile noentityescaping noguarantee nokeep nologfile "+"nomapping nomaxvalue nominimize nominvalue nomonitoring none noneditionable nonschema noorder "+"nopr nopro noprom nopromp noprompt norely noresetlogs noreverse normal norowdependencies noschemacheck "+"noswitch not nothing notice notnull notrim novalidate now nowait nth_value nullif nulls num numb numbe "+"nvarchar nvarchar2 object ocicoll ocidate ocidatetime ociduration ociinterval ociloblocator ocinumber "+"ociref ocirefcursor ocirowid ocistring ocitype oct octet_length of off offline offset oid oidindex old "+"on online only opaque open operations operator optimal optimize option optionally or oracle oracle_date "+"oradata ord ordaudio orddicom orddoc order ordimage ordinality ordvideo organization orlany orlvary "+"out outer outfile outline output over overflow overriding package pad parallel parallel_enable "+"parameters parent parse partial partition partitions pascal passing password password_grace_time "+"password_lock_time password_reuse_max password_reuse_time password_verify_function patch path patindex "+"pctincrease pctthreshold pctused pctversion percent percent_rank percentile_cont percentile_disc "+"performance period period_add period_diff permanent physical pi pipe pipelined pivot pluggable plugin "+"policy position post_transaction pow power pragma prebuilt precedes preceding precision prediction "+"prediction_cost prediction_details prediction_probability prediction_set prepare present preserve "+"prior priority private private_sga privileges procedural procedure procedure_analyze processlist "+"profiles project prompt protection public publishingservername purge quarter query quick quiesce quota "+"quotename radians raise rand range rank raw read reads readsize rebuild record records "+"recover recovery recursive recycle redo reduced ref reference referenced references referencing refresh "+"regexp_like register regr_avgx regr_avgy regr_count regr_intercept regr_r2 regr_slope regr_sxx regr_sxy "+"reject rekey relational relative relaylog release release_lock relies_on relocate rely rem remainder rename "+"repair repeat replace replicate replication required reset resetlogs resize resource respect restore "+"restricted result result_cache resumable resume retention return returning returns reuse reverse revoke "+"right rlike role roles rollback rolling rollup round row row_count rowdependencies rowid rownum rows "+"rtrim rules safe salt sample save savepoint sb1 sb2 sb4 scan schema schemacheck scn scope scroll "+"sdo_georaster sdo_topo_geometry search sec_to_time second seconds section securefile security seed segment select "+"self semi sequence sequential serializable server servererror session session_user sessions_per_user set "+"sets settings sha sha1 sha2 share shared shared_pool short show shrink shutdown si_averagecolor "+"si_colorhistogram si_featurelist si_positionalcolor si_stillimage si_texture siblings sid sign sin "+"size size_t sizes skip slave sleep smalldatetimefromparts smallfile snapshot some soname sort soundex "+"source space sparse spfile split sql sql_big_result sql_buffer_result sql_cache sql_calc_found_rows "+"sql_small_result sql_variant_property sqlcode sqldata sqlerror sqlname sqlstate sqrt square standalone "+"standby start starting startup statement static statistics stats_binomial_test stats_crosstab "+"stats_ks_test stats_mode stats_mw_test stats_one_way_anova stats_t_test_ stats_t_test_indep "+"stats_t_test_one stats_t_test_paired stats_wsr_test status std stddev stddev_pop stddev_samp stdev "+"stop storage store stored str str_to_date straight_join strcmp strict string struct stuff style subdate "+"subpartition subpartitions substitutable substr substring subtime subtring_index subtype success sum "+"suspend switch switchoffset switchover sync synchronous synonym sys sys_xmlagg sysasm sysaux sysdate "+"sysdatetimeoffset sysdba sysoper system system_user sysutcdatetime table tables tablespace tablesample tan tdo "+"template temporary terminated tertiary_weights test than then thread through tier ties time time_format "+"time_zone timediff timefromparts timeout timestamp timestampadd timestampdiff timezone_abbr "+"timezone_minute timezone_region to to_base64 to_date to_days to_seconds todatetimeoffset trace tracking "+"transaction transactional translate translation treat trigger trigger_nestlevel triggers trim truncate "+"try_cast try_convert try_parse type ub1 ub2 ub4 ucase unarchived unbounded uncompress "+"under undo unhex unicode uniform uninstall union unique unix_timestamp unknown unlimited unlock unnest unpivot "+"unrecoverable unsafe unsigned until untrusted unusable unused update updated upgrade upped upper upsert "+"url urowid usable usage use use_stored_outlines user user_data user_resources users using utc_date "+"utc_timestamp uuid uuid_short validate validate_password_strength validation valist value values var "+"var_samp varcharc vari varia variab variabl variable variables variance varp varraw varrawc varray "+"verify version versions view virtual visible void wait wallet warning warnings week weekday weekofyear "+"wellformed when whene whenev wheneve whenever where while whitespace window with within without work wrapped "+"xdb xml xmlagg xmlattributes xmlcast xmlcolattval xmlelement xmlexists xmlforest xmlindex xmlnamespaces "+"xmlpi xmlquery xmlroot xmlschema xmlserialize xmltable xmltype xor year year_to_month years yearweek",literal:"true false null unknown",built_in:"array bigint binary bit blob bool boolean char character date dec decimal float int int8 integer interval number "+"numeric real record serial serial8 smallint text time timestamp tinyint varchar varchar2 varying void"},contains:[{className:"string",begin:"'",end:"'",contains:[{begin:"''"}]},{className:"string",begin:'"',end:'"',contains:[{begin:'""'}]},{className:"string",begin:"`",end:"`"},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,hljs.HASH_COMMENT_MODE]},hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("perl",function(hljs){var PERL_KEYWORDS="getpwent getservent quotemeta msgrcv scalar kill dbmclose undef lc "+"ma syswrite tr send umask sysopen shmwrite vec qx utime local oct semctl localtime "+"readpipe do return format read sprintf dbmopen pop getpgrp not getpwnam rewinddir qq"+"fileno qw endprotoent wait sethostent bless s|0 opendir continue each sleep endgrent "+"shutdown dump chomp connect getsockname die socketpair close flock exists index shmget"+"sub for endpwent redo lstat msgctl setpgrp abs exit select print ref gethostbyaddr "+"unshift fcntl syscall goto getnetbyaddr join gmtime symlink semget splice x|0 "+"getpeername recv log setsockopt cos last reverse gethostbyname getgrnam study formline "+"endhostent times chop length gethostent getnetent pack getprotoent getservbyname rand "+"mkdir pos chmod y|0 substr endnetent printf next open msgsnd readdir use unlink "+"getsockopt getpriority rindex wantarray hex system getservbyport endservent int chr "+"untie rmdir prototype tell listen fork shmread ucfirst setprotoent else sysseek link "+"getgrgid shmctl waitpid unpack getnetbyname reset chdir grep split require caller "+"lcfirst until warn while values shift telldir getpwuid my getprotobynumber delete and "+"sort uc defined srand accept package seekdir getprotobyname semop our rename seek if q|0 "+"chroot sysread setpwent no crypt getc chown sqrt write setnetent setpriority foreach "+"tie sin msgget map stat getlogin unless elsif truncate exec keys glob tied closedir"+"ioctl socket readlink eval xor readline binmode setservent eof ord bind alarm pipe "+"atan2 getgrent exp time push setgrent gt lt or ne m|0 break given say state when";var SUBST={className:"subst",begin:"[$@]\\{",end:"\\}",keywords:PERL_KEYWORDS};var METHOD={begin:"->{",end:"}"};var VAR={variants:[{begin:/\$\d/},{begin:/[\$%@](\^\w\b|#\w+(::\w+)*|{\w+}|\w+(::\w*)*)/},{begin:/[\$%@][^\s\w{]/,relevance:0}]};var STRING_CONTAINS=[hljs.BACKSLASH_ESCAPE,SUBST,VAR];var PERL_DEFAULT_CONTAINS=[VAR,hljs.HASH_COMMENT_MODE,hljs.COMMENT("^\\=\\w","\\=cut",{endsWithParent:true}),METHOD,{className:"string",contains:STRING_CONTAINS,variants:[{begin:"q[qwxr]?\\s*\\(",end:"\\)",relevance:5},{begin:"q[qwxr]?\\s*\\[",end:"\\]",relevance:5},{begin:"q[qwxr]?\\s*\\{",end:"\\}",relevance:5},{begin:"q[qwxr]?\\s*\\|",end:"\\|",relevance:5},{begin:"q[qwxr]?\\s*\\<",end:"\\>",relevance:5},{begin:"qw\\s+q",end:"q",relevance:5},{begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"',end:'"'},{begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"{\\w+}",contains:[],relevance:0},{begin:"-?\\w+\\s*\\=\\>",contains:[],relevance:0}]},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},{begin:"(\\/\\/|"+hljs.RE_STARTERS_RE+"|\\b(split|return|print|reverse|grep)\\b)\\s*",keywords:"split return print reverse grep",relevance:0,contains:[hljs.HASH_COMMENT_MODE,{className:"regexp",begin:"(s|tr|y)/(\\\\.|[^/])*/(\\\\.|[^/])*/[a-z]*",relevance:10},{className:"regexp",begin:"(m|qr)?/",end:"/[a-z]*",contains:[hljs.BACKSLASH_ESCAPE],relevance:0}]},{className:"function",beginKeywords:"sub",end:"(\\s*\\(.*?\\))?[;{]",excludeEnd:true,relevance:5,contains:[hljs.TITLE_MODE]},{begin:"-\\w\\b",relevance:0},{begin:"^__DATA__$",end:"^__END__$",subLanguage:"mojolicious",contains:[{begin:"^@@.*",end:"$",className:"comment"}]}];SUBST.contains=PERL_DEFAULT_CONTAINS;METHOD.contains=PERL_DEFAULT_CONTAINS;return{aliases:["pl","pm"],lexemes:/[\w\.]+/,keywords:PERL_KEYWORDS,contains:PERL_DEFAULT_CONTAINS}});hljs.registerLanguage("gauss",function(hljs){var KEYWORDS={keyword:"bool break call callexe checkinterrupt clear clearg closeall cls comlog compile "+"continue create debug declare delete disable dlibrary dllcall do dos ed edit else "+"elseif enable end endfor endif endp endo errorlog errorlogat expr external fn "+"for format goto gosub graph if keyword let lib library line load loadarray loadexe "+"loadf loadk loadm loadp loads loadx local locate loopnextindex lprint lpwidth lshow "+"matrix msym ndpclex new open output outwidth plot plotsym pop prcsn print "+"printdos proc push retp return rndcon rndmod rndmult rndseed run save saveall screen "+"scroll setarray show sparse stop string struct system trace trap threadfor "+"threadendfor threadbegin threadjoin threadstat threadend until use while winprint "+"ne ge le gt lt and xor or not eq eqv",built_in:"abs acf aconcat aeye amax amean AmericanBinomCall AmericanBinomCall_Greeks AmericanBinomCall_ImpVol "+"AmericanBinomPut AmericanBinomPut_Greeks AmericanBinomPut_ImpVol AmericanBSCall AmericanBSCall_Greeks "+"AmericanBSCall_ImpVol AmericanBSPut AmericanBSPut_Greeks AmericanBSPut_ImpVol amin amult annotationGetDefaults "+"annotationSetBkd annotationSetFont annotationSetLineColor annotationSetLineStyle annotationSetLineThickness "+"annualTradingDays arccos arcsin areshape arrayalloc arrayindex arrayinit arraytomat asciiload asclabel astd "+"astds asum atan atan2 atranspose axmargin balance band bandchol bandcholsol bandltsol bandrv bandsolpd bar "+"base10 begwind besselj bessely beta box boxcox cdfBeta cdfBetaInv cdfBinomial cdfBinomialInv cdfBvn cdfBvn2 "+"cdfBvn2e cdfCauchy cdfCauchyInv cdfChic cdfChii cdfChinc cdfChincInv cdfExp cdfExpInv cdfFc cdfFnc cdfFncInv "+"cdfGam cdfGenPareto cdfHyperGeo cdfLaplace cdfLaplaceInv cdfLogistic cdfLogisticInv cdfmControlCreate cdfMvn "+"cdfMvn2e cdfMvnce cdfMvne cdfMvt2e cdfMvtce cdfMvte cdfN cdfN2 cdfNc cdfNegBinomial cdfNegBinomialInv cdfNi "+"cdfPoisson cdfPoissonInv cdfRayleigh cdfRayleighInv cdfTc cdfTci cdfTnc cdfTvn cdfWeibull cdfWeibullInv cdir "+"ceil ChangeDir chdir chiBarSquare chol choldn cholsol cholup chrs close code cols colsf combinate combinated "+"complex con cond conj cons ConScore contour conv convertsatostr convertstrtosa corrm corrms corrvc corrx corrxs "+"cos cosh counts countwts crossprd crout croutp csrcol csrlin csvReadM csvReadSA cumprodc cumsumc curve cvtos "+"datacreate datacreatecomplex datalist dataload dataloop dataopen datasave date datestr datestring datestrymd "+"dayinyr dayofweek dbAddDatabase dbClose dbCommit dbCreateQuery dbExecQuery dbGetConnectOptions dbGetDatabaseName "+"dbGetDriverName dbGetDrivers dbGetHostName dbGetLastErrorNum dbGetLastErrorText dbGetNumericalPrecPolicy "+"dbGetPassword dbGetPort dbGetTableHeaders dbGetTables dbGetUserName dbHasFeature dbIsDriverAvailable dbIsOpen "+"dbIsOpenError dbOpen dbQueryBindValue dbQueryClear dbQueryCols dbQueryExecPrepared dbQueryFetchAllM dbQueryFetchAllSA "+"dbQueryFetchOneM dbQueryFetchOneSA dbQueryFinish dbQueryGetBoundValue dbQueryGetBoundValues dbQueryGetField "+"dbQueryGetLastErrorNum dbQueryGetLastErrorText dbQueryGetLastInsertID dbQueryGetLastQuery dbQueryGetPosition "+"dbQueryIsActive dbQueryIsForwardOnly dbQueryIsNull dbQueryIsSelect dbQueryIsValid dbQueryPrepare dbQueryRows "+"dbQuerySeek dbQuerySeekFirst dbQuerySeekLast dbQuerySeekNext dbQuerySeekPrevious dbQuerySetForwardOnly "+"dbRemoveDatabase dbRollback dbSetConnectOptions dbSetDatabaseName dbSetHostName dbSetNumericalPrecPolicy "+"dbSetPort dbSetUserName dbTransaction DeleteFile delif delrows denseToSp denseToSpRE denToZero design det detl "+"dfft dffti diag diagrv digamma doswin DOSWinCloseall DOSWinOpen dotfeq dotfeqmt dotfge dotfgemt dotfgt dotfgtmt "+"dotfle dotflemt dotflt dotfltmt dotfne dotfnemt draw drop dsCreate dstat dstatmt dstatmtControlCreate dtdate dtday "+"dttime dttodtv dttostr dttoutc dtvnormal dtvtodt dtvtoutc dummy dummybr dummydn eig eigh eighv eigv elapsedTradingDays "+"endwind envget eof eqSolve eqSolvemt eqSolvemtControlCreate eqSolvemtOutCreate eqSolveset erf erfc erfccplx erfcplx error "+"etdays ethsec etstr EuropeanBinomCall EuropeanBinomCall_Greeks EuropeanBinomCall_ImpVol EuropeanBinomPut "+"EuropeanBinomPut_Greeks EuropeanBinomPut_ImpVol EuropeanBSCall EuropeanBSCall_Greeks EuropeanBSCall_ImpVol "+"EuropeanBSPut EuropeanBSPut_Greeks EuropeanBSPut_ImpVol exctsmpl exec execbg exp extern eye fcheckerr fclearerr feq "+"feqmt fflush fft ffti fftm fftmi fftn fge fgemt fgets fgetsa fgetsat fgetst fgt fgtmt fileinfo filesa fle flemt "+"floor flt fltmt fmod fne fnemt fonts fopen formatcv formatnv fputs fputst fseek fstrerror ftell ftocv ftos ftostrC "+"gamma gammacplx gammaii gausset gdaAppend gdaCreate gdaDStat gdaDStatMat gdaGetIndex gdaGetName gdaGetNames gdaGetOrders "+"gdaGetType gdaGetTypes gdaGetVarInfo gdaIsCplx gdaLoad gdaPack gdaRead gdaReadByIndex gdaReadSome gdaReadSparse "+"gdaReadStruct gdaReportVarInfo gdaSave gdaUpdate gdaUpdateAndPack gdaVars gdaWrite gdaWrite32 gdaWriteSome getarray "+"getdims getf getGAUSShome getmatrix getmatrix4D getname getnamef getNextTradingDay getNextWeekDay getnr getorders "+"getpath getPreviousTradingDay getPreviousWeekDay getRow getscalar3D getscalar4D getTrRow getwind glm gradcplx gradMT "+"gradMTm gradMTT gradMTTm gradp graphprt graphset hasimag header headermt hess hessMT hessMTg hessMTgw hessMTm "+"hessMTmw hessMTT hessMTTg hessMTTgw hessMTTm hessMTw hessp hist histf histp hsec imag indcv indexcat indices indices2 "+"indicesf indicesfn indnv indsav integrate1d integrateControlCreate intgrat2 intgrat3 inthp1 inthp2 inthp3 inthp4 "+"inthpControlCreate intquad1 intquad2 intquad3 intrleav intrleavsa intrsect intsimp inv invpd invswp iscplx iscplxf "+"isden isinfnanmiss ismiss key keyav keyw lag lag1 lagn lapEighb lapEighi lapEighvb lapEighvi lapgEig lapgEigh lapgEighv "+"lapgEigv lapgSchur lapgSvdcst lapgSvds lapgSvdst lapSvdcusv lapSvds lapSvdusv ldlp ldlsol linSolve listwise ln lncdfbvn "+"lncdfbvn2 lncdfmvn lncdfn lncdfn2 lncdfnc lnfact lngammacplx lnpdfmvn lnpdfmvt lnpdfn lnpdft loadd loadstruct loadwind "+"loess loessmt loessmtControlCreate log loglog logx logy lower lowmat lowmat1 ltrisol lu lusol machEpsilon make makevars "+"makewind margin matalloc matinit mattoarray maxbytes maxc maxindc maxv maxvec mbesselei mbesselei0 mbesselei1 mbesseli "+"mbesseli0 mbesseli1 meanc median mergeby mergevar minc minindc minv miss missex missrv moment momentd movingave "+"movingaveExpwgt movingaveWgt nextindex nextn nextnevn nextwind ntos null null1 numCombinations ols olsmt olsmtControlCreate "+"olsqr olsqr2 olsqrmt ones optn optnevn orth outtyp pacf packedToSp packr parse pause pdfCauchy pdfChi pdfExp pdfGenPareto "+"pdfHyperGeo pdfLaplace pdfLogistic pdfn pdfPoisson pdfRayleigh pdfWeibull pi pinv pinvmt plotAddArrow plotAddBar plotAddBox "+"plotAddHist plotAddHistF plotAddHistP plotAddPolar plotAddScatter plotAddShape plotAddTextbox plotAddTS plotAddXY plotArea "+"plotBar plotBox plotClearLayout plotContour plotCustomLayout plotGetDefaults plotHist plotHistF plotHistP plotLayout "+"plotLogLog plotLogX plotLogY plotOpenWindow plotPolar plotSave plotScatter plotSetAxesPen plotSetBar plotSetBarFill "+"plotSetBarStacked plotSetBkdColor plotSetFill plotSetGrid plotSetLegend plotSetLineColor plotSetLineStyle plotSetLineSymbol "+"plotSetLineThickness plotSetNewWindow plotSetTitle plotSetWhichYAxis plotSetXAxisShow plotSetXLabel plotSetXRange "+"plotSetXTicInterval plotSetXTicLabel plotSetYAxisShow plotSetYLabel plotSetYRange plotSetZAxisShow plotSetZLabel "+"plotSurface plotTS plotXY polar polychar polyeval polygamma polyint polymake polymat polymroot polymult polyroot "+"pqgwin previousindex princomp printfm printfmt prodc psi putarray putf putvals pvCreate pvGetIndex pvGetParNames "+"pvGetParVector pvLength pvList pvPack pvPacki pvPackm pvPackmi pvPacks pvPacksi pvPacksm pvPacksmi pvPutParVector "+"pvTest pvUnpack QNewton QNewtonmt QNewtonmtControlCreate QNewtonmtOutCreate QNewtonSet QProg QProgmt QProgmtInCreate "+"qqr qqre qqrep qr qre qrep qrsol qrtsol qtyr qtyre qtyrep quantile quantiled qyr qyre qyrep qz rank rankindx readr "+"real reclassify reclassifyCuts recode recserar recsercp recserrc rerun rescale reshape rets rev rfft rffti rfftip rfftn "+"rfftnp rfftp rndBernoulli rndBeta rndBinomial rndCauchy rndChiSquare rndCon rndCreateState rndExp rndGamma rndGeo rndGumbel "+"rndHyperGeo rndi rndKMbeta rndKMgam rndKMi rndKMn rndKMnb rndKMp rndKMu rndKMvm rndLaplace rndLCbeta rndLCgam rndLCi rndLCn "+"rndLCnb rndLCp rndLCu rndLCvm rndLogNorm rndMTu rndMVn rndMVt rndn rndnb rndNegBinomial rndp rndPoisson rndRayleigh "+"rndStateSkip rndu rndvm rndWeibull rndWishart rotater round rows rowsf rref sampleData satostrC saved saveStruct savewind "+"scale scale3d scalerr scalinfnanmiss scalmiss schtoc schur searchsourcepath seekr select selif seqa seqm setdif setdifsa "+"setvars setvwrmode setwind shell shiftr sin singleindex sinh sleep solpd sortc sortcc sortd sorthc sorthcc sortind "+"sortindc sortmc sortr sortrc spBiconjGradSol spChol spConjGradSol spCreate spDenseSubmat spDiagRvMat spEigv spEye spLDL "+"spline spLU spNumNZE spOnes spreadSheetReadM spreadSheetReadSA spreadSheetWrite spScale spSubmat spToDense spTrTDense "+"spTScalar spZeros sqpSolve sqpSolveMT sqpSolveMTControlCreate sqpSolveMTlagrangeCreate sqpSolveMToutCreate sqpSolveSet "+"sqrt statements stdc stdsc stocv stof strcombine strindx strlen strput strrindx strsect strsplit strsplitPad strtodt "+"strtof strtofcplx strtriml strtrimr strtrunc strtruncl strtruncpad strtruncr submat subscat substute subvec sumc sumr "+"surface svd svd1 svd2 svdcusv svds svdusv sysstate tab tan tanh tempname "+"time timedt timestr timeutc title tkf2eps tkf2ps tocart todaydt toeplitz token topolar trapchk "+"trigamma trimr trunc type typecv typef union unionsa uniqindx uniqindxsa unique uniquesa upmat upmat1 upper utctodt "+"utctodtv utrisol vals varCovMS varCovXS varget vargetl varmall varmares varput varputl vartypef vcm vcms vcx vcxs "+"vec vech vecr vector vget view viewxyz vlist vnamecv volume vput vread vtypecv wait waitc walkindex where window "+"writer xlabel xlsGetSheetCount xlsGetSheetSize xlsGetSheetTypes xlsMakeRange xlsReadM xlsReadSA xlsWrite xlsWriteM "+"xlsWriteSA xpnd xtics xy xyz ylabel ytics zeros zeta zlabel ztics cdfEmpirical dot h5create h5open h5read h5readAttribute "+"h5write h5writeAttribute ldl plotAddErrorBar plotAddSurface plotCDFEmpirical plotSetColormap plotSetContourLabels "+"plotSetLegendFont plotSetTextInterpreter plotSetXTicCount plotSetYTicCount plotSetZLevels powerm strjoin sylvester "+"strtrim",literal:"DB_AFTER_LAST_ROW DB_ALL_TABLES DB_BATCH_OPERATIONS DB_BEFORE_FIRST_ROW DB_BLOB DB_EVENT_NOTIFICATIONS "+"DB_FINISH_QUERY DB_HIGH_PRECISION DB_LAST_INSERT_ID DB_LOW_PRECISION_DOUBLE DB_LOW_PRECISION_INT32 "+"DB_LOW_PRECISION_INT64 DB_LOW_PRECISION_NUMBERS DB_MULTIPLE_RESULT_SETS DB_NAMED_PLACEHOLDERS "+"DB_POSITIONAL_PLACEHOLDERS DB_PREPARED_QUERIES DB_QUERY_SIZE DB_SIMPLE_LOCKING DB_SYSTEM_TABLES DB_TABLES "+"DB_TRANSACTIONS DB_UNICODE DB_VIEWS __STDIN __STDOUT __STDERR __FILE_DIR"};var AT_COMMENT_MODE=hljs.COMMENT("@","@");var PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"define definecs|10 undef ifdef ifndef iflight ifdllcall ifmac ifos2win ifunix else endif lineson linesoff srcfile srcline"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",end:"$",keywords:{"meta-keyword":"include"},contains:[{className:"meta-string",begin:'"',end:'"',illegal:"\\n"}]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE]};var STRUCT_TYPE={begin:/\bstruct\s+/,end:/\s/,keywords:"struct",contains:[{className:"type",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0}]};var PARSE_PARAMS=[{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,endsWithParent:true,relevance:0,contains:[{className:"literal",begin:/\.\.\./},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,STRUCT_TYPE]}];var FUNCTION_DEF={className:"title",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0};var DEFINITION=function(beginKeywords,end,inherits){var mode=hljs.inherit({className:"function",beginKeywords:beginKeywords,end:end,excludeEnd:true,contains:[].concat(PARSE_PARAMS)},inherits||{});mode.contains.push(FUNCTION_DEF);mode.contains.push(hljs.C_NUMBER_MODE);mode.contains.push(hljs.C_BLOCK_COMMENT_MODE);mode.contains.push(AT_COMMENT_MODE);return mode};var BUILT_IN_REF={className:"built_in",begin:"\\b("+KEYWORDS.built_in.split(" ").join("|")+")\\b"};var STRING_REF={className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE],relevance:0};var FUNCTION_REF={begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,keywords:KEYWORDS,relevance:0,contains:[{beginKeywords:KEYWORDS.keyword},BUILT_IN_REF,{className:"built_in",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0}]};var FUNCTION_REF_PARAMS={begin:/\(/,end:/\)/,relevance:0,keywords:{built_in:KEYWORDS.built_in,literal:KEYWORDS.literal},contains:[hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,BUILT_IN_REF,FUNCTION_REF,STRING_REF,"self"]};FUNCTION_REF.contains.push(FUNCTION_REF_PARAMS);return{aliases:["gss"],case_insensitive:true,keywords:KEYWORDS,illegal:/(\{[%#]|[%#]\}| <- )/,contains:[hljs.C_NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,STRING_REF,PREPROCESSOR,{className:"keyword",begin:/\bexternal (matrix|string|array|sparse matrix|struct|proc|keyword|fn)/},DEFINITION("proc keyword",";"),DEFINITION("fn","="),{beginKeywords:"for threadfor",end:/;/,relevance:0,contains:[hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,FUNCTION_REF_PARAMS]},{variants:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\."+hljs.UNDERSCORE_IDENT_RE},{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*="}],relevance:0},FUNCTION_REF,STRUCT_TYPE]}});hljs.registerLanguage("awk",function(hljs){var VARIABLE={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var KEYWORDS="BEGIN END if else while do for in break continue delete next nextfile function func exit|10";var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:/(u|b)?r?'''/,end:/'''/,relevance:10},{begin:/(u|b)?r?"""/,end:/"""/,relevance:10},{begin:/(u|r|ur)'/,end:/'/,relevance:10},{begin:/(u|r|ur)"/,end:/"/,relevance:10},{begin:/(b|br)'/,end:/'/},{begin:/(b|br)"/,end:/"/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};return{keywords:{keyword:KEYWORDS},contains:[VARIABLE,STRING,hljs.REGEXP_MODE,hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("yaml",function(hljs){var LITERALS="true false yes no null";var KEY={className:"attr",variants:[{begin:"\\w[\\w :\\/.-]*:(?=[ \t]|$)"},{begin:'"\\w[\\w :\\/.-]*":(?=[ \t]|$)'},{begin:"'\\w[\\w :\\/.-]*':(?=[ \t]|$)"}]};var TEMPLATE_VARIABLES={className:"template-variable",variants:[{begin:"{{",end:"}}"},{begin:"%{",end:"}"}]};var STRING={className:"string",relevance:0,variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/\S+/}],contains:[hljs.BACKSLASH_ESCAPE,TEMPLATE_VARIABLES]};return{case_insensitive:true,aliases:["yml","YAML","yaml"],contains:[KEY,{className:"meta",begin:"^---s*$",relevance:10},{className:"string",begin:"[\\|>]([0-9]?[+-])?[ ]*\\n( *)[\\S ]+\\n(\\2[\\S ]+\\n?)*"},{begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:"!"+hljs.UNDERSCORE_IDENT_RE},{className:"type",begin:"!!"+hljs.UNDERSCORE_IDENT_RE},{className:"meta",begin:"&"+hljs.UNDERSCORE_IDENT_RE+"$"},{className:"meta",begin:"\\*"+hljs.UNDERSCORE_IDENT_RE+"$"},{className:"bullet",begin:"\\-(?=[ ]|$)",relevance:0},hljs.HASH_COMMENT_MODE,{beginKeywords:LITERALS,keywords:{literal:LITERALS}},{className:"number",begin:hljs.C_NUMBER_RE+"\\b"},STRING]}});hljs.registerLanguage("tap",function(hljs){return{case_insensitive:true,contains:[hljs.HASH_COMMENT_MODE,{className:"meta",variants:[{begin:"^TAP version (\\d+)$"},{begin:"^1\\.\\.(\\d+)$"}]},{begin:"(s+)?---$",end:"\\.\\.\\.$",subLanguage:"yaml",relevance:0},{className:"number",begin:" (\\d+) "},{className:"symbol",variants:[{begin:"^ok"},{begin:"^not ok"}]}]}});hljs.registerLanguage("rib",function(hljs){return{keywords:"ArchiveRecord AreaLightSource Atmosphere Attribute AttributeBegin AttributeEnd Basis "+"Begin Blobby Bound Clipping ClippingPlane Color ColorSamples ConcatTransform Cone "+"CoordinateSystem CoordSysTransform CropWindow Curves Cylinder DepthOfField Detail "+"DetailRange Disk Displacement Display End ErrorHandler Exposure Exterior Format "+"FrameAspectRatio FrameBegin FrameEnd GeneralPolygon GeometricApproximation Geometry "+"Hider Hyperboloid Identity Illuminate Imager Interior LightSource "+"MakeCubeFaceEnvironment MakeLatLongEnvironment MakeShadow MakeTexture Matte "+"MotionBegin MotionEnd NuPatch ObjectBegin ObjectEnd ObjectInstance Opacity Option "+"Orientation Paraboloid Patch PatchMesh Perspective PixelFilter PixelSamples "+"PixelVariance Points PointsGeneralPolygons PointsPolygons Polygon Procedural Projection "+"Quantize ReadArchive RelativeDetail ReverseOrientation Rotate Scale ScreenWindow "+"ShadingInterpolation ShadingRate Shutter Sides Skew SolidBegin SolidEnd Sphere "+"SubdivisionMesh Surface TextureCoordinates Torus Transform TransformBegin TransformEnd "+"TransformPoints Translate TrimCurve WorldBegin WorldEnd",illegal:"</",contains:[hljs.HASH_COMMENT_MODE,hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]}});hljs.registerLanguage("prolog",function(hljs){var ATOM={begin:/[a-z][A-Za-z0-9_]*/,relevance:0};var VAR={className:"symbol",variants:[{begin:/[A-Z][a-zA-Z0-9_]*/},{begin:/_[A-Za-z0-9_]*/}],relevance:0};var PARENTED={begin:/\(/,end:/\)/,relevance:0};var LIST={begin:/\[/,end:/\]/};var LINE_COMMENT={className:"comment",begin:/%/,end:/$/,contains:[hljs.PHRASAL_WORDS_MODE]};var BACKTICK_STRING={className:"string",begin:/`/,end:/`/,contains:[hljs.BACKSLASH_ESCAPE]};var CHAR_CODE={className:"string",begin:/0\'(\\\'|.)/};var SPACE_CODE={className:"string",begin:/0\'\\s/};var PRED_OP={begin:/:-/};var inner=[ATOM,VAR,PARENTED,PRED_OP,LIST,LINE_COMMENT,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,BACKTICK_STRING,CHAR_CODE,SPACE_CODE,hljs.C_NUMBER_MODE];PARENTED.contains=inner;LIST.contains=inner;return{contains:inner.concat([{begin:/\.$/}])}});hljs.registerLanguage("actionscript",function(hljs){var IDENT_RE="[a-zA-Z_$][a-zA-Z0-9_$]*";var IDENT_FUNC_RETURN_TYPE_RE="([*]|[a-zA-Z_$][a-zA-Z0-9_$]*)";var AS3_REST_ARG_MODE={className:"rest_arg",begin:"[.]{3}",end:IDENT_RE,relevance:10};return{aliases:["as"],keywords:{keyword:"as break case catch class const continue default delete do dynamic each "+"else extends final finally for function get if implements import in include "+"instanceof interface internal is namespace native new override package private "+"protected public return set static super switch this throw try typeof use var void "+"while with",literal:"true false null undefined"},contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"class",beginKeywords:"package",end:"{",contains:[hljs.TITLE_MODE]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,contains:[{beginKeywords:"extends implements"},hljs.TITLE_MODE]},{className:"meta",beginKeywords:"import include",end:";",keywords:{"meta-keyword":"import include"}},{className:"function",beginKeywords:"function",end:"[{;]",excludeEnd:true,illegal:"\\S",contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AS3_REST_ARG_MODE]},{begin:":\\s*"+IDENT_FUNC_RETURN_TYPE_RE}]},hljs.METHOD_GUARD],illegal:/#/}});hljs.registerLanguage("ebnf",function(hljs){var commentMode=hljs.COMMENT(/\(\*/,/\*\)/);var nonTerminalMode={className:"attribute",begin:/^[ ]*[a-zA-Z][a-zA-Z-_]*([\s-_]+[a-zA-Z][a-zA-Z]*)*/};var specialSequenceMode={className:"meta",begin:/\?.*\?/};var ruleBodyMode={begin:/=/,end:/[.;]/,contains:[commentMode,specialSequenceMode,{className:"string",variants:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"`",end:"`"}]}]};return{illegal:/\S/,contains:[commentMode,nonTerminalMode,ruleBodyMode]}});hljs.registerLanguage("fix",function(hljs){return{contains:[{begin:/[^\u2401\u0001]+/,end:/[\u2401\u0001]/,excludeEnd:true,returnBegin:true,returnEnd:false,contains:[{begin:/([^\u2401\u0001=]+)/,end:/=([^\u2401\u0001=]+)/,returnEnd:true,returnBegin:false,className:"attr"},{begin:/=/,end:/([\u2401\u0001])/,excludeEnd:true,excludeBegin:true,className:"string"}]}],case_insensitive:true}});hljs.registerLanguage("bnf",function(hljs){return{contains:[{className:"attribute",begin:/</,end:/>/},{begin:/::=/,starts:{end:/$/,contains:[{begin:/</,end:/>/},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]}}]}});hljs.registerLanguage("isbl",function(hljs){var UNDERSCORE_IDENT_RE="[A-Za-zА-Яа-яёЁ_!][A-Za-zА-Яа-яёЁ_0-9]*";var FUNCTION_NAME_IDENT_RE="[A-Za-zА-Яа-яёЁ_][A-Za-zА-Яа-яёЁ_0-9]*";var KEYWORD="and и else иначе endexcept endfinally endforeach конецвсе endif конецесли endwhile конецпока "+"except exitfor finally foreach все if если in в not не or или try while пока ";var sysres_constants="SYSRES_CONST_ACCES_RIGHT_TYPE_EDIT "+"SYSRES_CONST_ACCES_RIGHT_TYPE_FULL "+"SYSRES_CONST_ACCES_RIGHT_TYPE_VIEW "+"SYSRES_CONST_ACCESS_MODE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_NO_ACCESS_VIEW "+"SYSRES_CONST_ACCESS_NO_ACCESS_VIEW_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_ADD_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_ADD_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_CHANGE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_CHANGE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_DELETE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_DELETE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_EXECUTE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_EXECUTE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_NO_ACCESS_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_NO_ACCESS_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_RATIFY_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_RATIFY_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_TYPE_CHANGE "+"SYSRES_CONST_ACCESS_TYPE_CHANGE_CODE "+"SYSRES_CONST_ACCESS_TYPE_EXISTS "+"SYSRES_CONST_ACCESS_TYPE_EXISTS_CODE "+"SYSRES_CONST_ACCESS_TYPE_FULL "+"SYSRES_CONST_ACCESS_TYPE_FULL_CODE "+"SYSRES_CONST_ACCESS_TYPE_VIEW "+"SYSRES_CONST_ACCESS_TYPE_VIEW_CODE "+"SYSRES_CONST_ACTION_TYPE_ABORT "+"SYSRES_CONST_ACTION_TYPE_ACCEPT "+"SYSRES_CONST_ACTION_TYPE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ADD_ATTACHMENT "+"SYSRES_CONST_ACTION_TYPE_CHANGE_CARD "+"SYSRES_CONST_ACTION_TYPE_CHANGE_KIND "+"SYSRES_CONST_ACTION_TYPE_CHANGE_STORAGE "+"SYSRES_CONST_ACTION_TYPE_CONTINUE "+"SYSRES_CONST_ACTION_TYPE_COPY "+"SYSRES_CONST_ACTION_TYPE_CREATE "+"SYSRES_CONST_ACTION_TYPE_CREATE_VERSION "+"SYSRES_CONST_ACTION_TYPE_DELETE "+"SYSRES_CONST_ACTION_TYPE_DELETE_ATTACHMENT "+"SYSRES_CONST_ACTION_TYPE_DELETE_VERSION "+"SYSRES_CONST_ACTION_TYPE_DISABLE_DELEGATE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ENABLE_DELEGATE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_CERTIFICATE "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_CERTIFICATE_AND_PASSWORD "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_PASSWORD "+"SYSRES_CONST_ACTION_TYPE_EXPORT_WITH_LOCK "+"SYSRES_CONST_ACTION_TYPE_EXPORT_WITHOUT_LOCK "+"SYSRES_CONST_ACTION_TYPE_IMPORT_WITH_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_IMPORT_WITHOUT_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_LIFE_CYCLE_STAGE "+"SYSRES_CONST_ACTION_TYPE_LOCK "+"SYSRES_CONST_ACTION_TYPE_LOCK_FOR_SERVER "+"SYSRES_CONST_ACTION_TYPE_LOCK_MODIFY "+"SYSRES_CONST_ACTION_TYPE_MARK_AS_READED "+"SYSRES_CONST_ACTION_TYPE_MARK_AS_UNREADED "+"SYSRES_CONST_ACTION_TYPE_MODIFY "+"SYSRES_CONST_ACTION_TYPE_MODIFY_CARD "+"SYSRES_CONST_ACTION_TYPE_MOVE_TO_ARCHIVE "+"SYSRES_CONST_ACTION_TYPE_OFF_ENCRYPTION "+"SYSRES_CONST_ACTION_TYPE_PASSWORD_CHANGE "+"SYSRES_CONST_ACTION_TYPE_PERFORM "+"SYSRES_CONST_ACTION_TYPE_RECOVER_FROM_LOCAL_COPY "+"SYSRES_CONST_ACTION_TYPE_RESTART "+"SYSRES_CONST_ACTION_TYPE_RESTORE_FROM_ARCHIVE "+"SYSRES_CONST_ACTION_TYPE_REVISION "+"SYSRES_CONST_ACTION_TYPE_SEND_BY_MAIL "+"SYSRES_CONST_ACTION_TYPE_SIGN "+"SYSRES_CONST_ACTION_TYPE_START "+"SYSRES_CONST_ACTION_TYPE_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_UNLOCK_FROM_SERVER "+"SYSRES_CONST_ACTION_TYPE_VERSION_STATE "+"SYSRES_CONST_ACTION_TYPE_VERSION_VISIBILITY "+"SYSRES_CONST_ACTION_TYPE_VIEW "+"SYSRES_CONST_ACTION_TYPE_VIEW_SHADOW_COPY "+"SYSRES_CONST_ACTION_TYPE_WORKFLOW_DESCRIPTION_MODIFY "+"SYSRES_CONST_ACTION_TYPE_WRITE_HISTORY "+"SYSRES_CONST_ACTIVE_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_ADD_REFERENCE_MODE_NAME "+"SYSRES_CONST_ADDITION_REQUISITE_CODE "+"SYSRES_CONST_ADDITIONAL_PARAMS_REQUISITE_CODE "+"SYSRES_CONST_ADITIONAL_JOB_END_DATE_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_READ_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_START_DATE_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_STATE_REQUISITE_NAME "+"SYSRES_CONST_ADMINISTRATION_HISTORY_ADDING_USER_TO_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_ADDING_USER_TO_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_COMP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_COMP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_USER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_USER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_CREATION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_CREATION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_DELETION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_DELETION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_COMP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_COMP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_FROM_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_FROM_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_RESTRICTION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_RESTRICTION_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_PRIVILEGE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_PRIVILEGE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_RIGHTS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_RIGHTS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_MAIN_SERVER_CHANGED_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_MAIN_SERVER_CHANGED_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_PUBLIC_CHANGED_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_PUBLIC_CHANGED_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_RESTRICTION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_RESTRICTION_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_PRIVILEGE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_PRIVILEGE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_RIGHTS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_RIGHTS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_CREATION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_CREATION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_DELETION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_DELETION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_CATEGORY_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_CATEGORY_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_COMP_TITLE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_COMP_TITLE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_FULL_NAME_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_FULL_NAME_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_PARENT_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_PARENT_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_AUTH_TYPE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_AUTH_TYPE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_LOGIN_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_LOGIN_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_STATUS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_STATUS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_USER_PASSWORD_CHANGE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_USER_PASSWORD_CHANGE_ACTION "+"SYSRES_CONST_ALL_ACCEPT_CONDITION_RUS "+"SYSRES_CONST_ALL_USERS_GROUP "+"SYSRES_CONST_ALL_USERS_GROUP_NAME "+"SYSRES_CONST_ALL_USERS_SERVER_GROUP_NAME "+"SYSRES_CONST_ALLOWED_ACCESS_TYPE_CODE "+"SYSRES_CONST_ALLOWED_ACCESS_TYPE_NAME "+"SYSRES_CONST_APP_VIEWER_TYPE_REQUISITE_CODE "+"SYSRES_CONST_APPROVING_SIGNATURE_NAME "+"SYSRES_CONST_APPROVING_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_ASSISTANT_SUBSTITUE_TYPE "+"SYSRES_CONST_ASSISTANT_SUBSTITUE_TYPE_CODE "+"SYSRES_CONST_ATTACH_TYPE_COMPONENT_TOKEN "+"SYSRES_CONST_ATTACH_TYPE_DOC "+"SYSRES_CONST_ATTACH_TYPE_EDOC "+"SYSRES_CONST_ATTACH_TYPE_FOLDER "+"SYSRES_CONST_ATTACH_TYPE_JOB "+"SYSRES_CONST_ATTACH_TYPE_REFERENCE "+"SYSRES_CONST_ATTACH_TYPE_TASK "+"SYSRES_CONST_AUTH_ENCODED_PASSWORD "+"SYSRES_CONST_AUTH_ENCODED_PASSWORD_CODE "+"SYSRES_CONST_AUTH_NOVELL "+"SYSRES_CONST_AUTH_PASSWORD "+"SYSRES_CONST_AUTH_PASSWORD_CODE "+"SYSRES_CONST_AUTH_WINDOWS "+"SYSRES_CONST_AUTHENTICATING_SIGNATURE_NAME "+"SYSRES_CONST_AUTHENTICATING_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_AUTO_ENUM_METHOD_FLAG "+"SYSRES_CONST_AUTO_NUMERATION_CODE "+"SYSRES_CONST_AUTO_STRONG_ENUM_METHOD_FLAG "+"SYSRES_CONST_AUTOTEXT_NAME_REQUISITE_CODE "+"SYSRES_CONST_AUTOTEXT_TEXT_REQUISITE_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_ALL "+"SYSRES_CONST_AUTOTEXT_USAGE_ALL_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_SIGN "+"SYSRES_CONST_AUTOTEXT_USAGE_SIGN_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_WORK "+"SYSRES_CONST_AUTOTEXT_USAGE_WORK_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ANYWHERE_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ON_SIGNING_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ON_WORK_CODE "+"SYSRES_CONST_BEGIN_DATE_REQUISITE_CODE "+"SYSRES_CONST_BLACK_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_BLUE_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_BTN_PART "+"SYSRES_CONST_CALCULATED_ROLE_TYPE_CODE "+"SYSRES_CONST_CALL_TYPE_VARIABLE_BUTTON_VALUE "+"SYSRES_CONST_CALL_TYPE_VARIABLE_PROGRAM_VALUE "+"SYSRES_CONST_CANCEL_MESSAGE_FUNCTION_RESULT "+"SYSRES_CONST_CARD_PART "+"SYSRES_CONST_CARD_REFERENCE_MODE_NAME "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_ENCRYPT_VALUE "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_SIGN_AND_ENCRYPT_VALUE "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_SIGN_VALUE "+"SYSRES_CONST_CHECK_PARAM_VALUE_DATE_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_FLOAT_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_INTEGER_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_PICK_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_REEFRENCE_PARAM_TYPE "+"SYSRES_CONST_CLOSED_RECORD_FLAG_VALUE_FEMININE "+"SYSRES_CONST_CLOSED_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_ADMIN "+"SYSRES_CONST_CODE_COMPONENT_TYPE_DEVELOPER "+"SYSRES_CONST_CODE_COMPONENT_TYPE_DOCS "+"SYSRES_CONST_CODE_COMPONENT_TYPE_EDOC_CARDS "+"SYSRES_CONST_CODE_COMPONENT_TYPE_EXTERNAL_EXECUTABLE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_OTHER "+"SYSRES_CONST_CODE_COMPONENT_TYPE_REFERENCE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_REPORT "+"SYSRES_CONST_CODE_COMPONENT_TYPE_SCRIPT "+"SYSRES_CONST_CODE_COMPONENT_TYPE_URL "+"SYSRES_CONST_CODE_REQUISITE_ACCESS "+"SYSRES_CONST_CODE_REQUISITE_CODE "+"SYSRES_CONST_CODE_REQUISITE_COMPONENT "+"SYSRES_CONST_CODE_REQUISITE_DESCRIPTION "+"SYSRES_CONST_CODE_REQUISITE_EXCLUDE_COMPONENT "+"SYSRES_CONST_CODE_REQUISITE_RECORD "+"SYSRES_CONST_COMMENT_REQ_CODE "+"SYSRES_CONST_COMMON_SETTINGS_REQUISITE_CODE "+"SYSRES_CONST_COMP_CODE_GRD "+"SYSRES_CONST_COMPONENT_GROUP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_COMPONENT_TYPE_ADMIN_COMPONENTS "+"SYSRES_CONST_COMPONENT_TYPE_DEVELOPER_COMPONENTS "+"SYSRES_CONST_COMPONENT_TYPE_DOCS "+"SYSRES_CONST_COMPONENT_TYPE_EDOC_CARDS "+"SYSRES_CONST_COMPONENT_TYPE_EDOCS "+"SYSRES_CONST_COMPONENT_TYPE_EXTERNAL_EXECUTABLE "+"SYSRES_CONST_COMPONENT_TYPE_OTHER "+"SYSRES_CONST_COMPONENT_TYPE_REFERENCE_TYPES "+"SYSRES_CONST_COMPONENT_TYPE_REFERENCES "+"SYSRES_CONST_COMPONENT_TYPE_REPORTS "+"SYSRES_CONST_COMPONENT_TYPE_SCRIPTS "+"SYSRES_CONST_COMPONENT_TYPE_URL "+"SYSRES_CONST_COMPONENTS_REMOTE_SERVERS_VIEW_CODE "+"SYSRES_CONST_CONDITION_BLOCK_DESCRIPTION "+"SYSRES_CONST_CONST_FIRM_STATUS_COMMON "+"SYSRES_CONST_CONST_FIRM_STATUS_INDIVIDUAL "+"SYSRES_CONST_CONST_NEGATIVE_VALUE "+"SYSRES_CONST_CONST_POSITIVE_VALUE "+"SYSRES_CONST_CONST_SERVER_STATUS_DONT_REPLICATE "+"SYSRES_CONST_CONST_SERVER_STATUS_REPLICATE "+"SYSRES_CONST_CONTENTS_REQUISITE_CODE "+"SYSRES_CONST_DATA_TYPE_BOOLEAN "+"SYSRES_CONST_DATA_TYPE_DATE "+"SYSRES_CONST_DATA_TYPE_FLOAT "+"SYSRES_CONST_DATA_TYPE_INTEGER "+"SYSRES_CONST_DATA_TYPE_PICK "+"SYSRES_CONST_DATA_TYPE_REFERENCE "+"SYSRES_CONST_DATA_TYPE_STRING "+"SYSRES_CONST_DATA_TYPE_TEXT "+"SYSRES_CONST_DATA_TYPE_VARIANT "+"SYSRES_CONST_DATE_CLOSE_REQ_CODE "+"SYSRES_CONST_DATE_FORMAT_DATE_ONLY_CHAR "+"SYSRES_CONST_DATE_OPEN_REQ_CODE "+"SYSRES_CONST_DATE_REQUISITE "+"SYSRES_CONST_DATE_REQUISITE_CODE "+"SYSRES_CONST_DATE_REQUISITE_NAME "+"SYSRES_CONST_DATE_REQUISITE_TYPE "+"SYSRES_CONST_DATE_TYPE_CHAR "+"SYSRES_CONST_DATETIME_FORMAT_VALUE "+"SYSRES_CONST_DEA_ACCESS_RIGHTS_ACTION_CODE "+"SYSRES_CONST_DESCRIPTION_LOCALIZE_ID_REQUISITE_CODE "+"SYSRES_CONST_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_DET1_PART "+"SYSRES_CONST_DET2_PART "+"SYSRES_CONST_DET3_PART "+"SYSRES_CONST_DET4_PART "+"SYSRES_CONST_DET5_PART "+"SYSRES_CONST_DET6_PART "+"SYSRES_CONST_DETAIL_DATASET_KEY_REQUISITE_CODE "+"SYSRES_CONST_DETAIL_PICK_REQUISITE_CODE "+"SYSRES_CONST_DETAIL_REQ_CODE "+"SYSRES_CONST_DO_NOT_USE_ACCESS_TYPE_CODE "+"SYSRES_CONST_DO_NOT_USE_ACCESS_TYPE_NAME "+"SYSRES_CONST_DO_NOT_USE_ON_VIEW_ACCESS_TYPE_CODE "+"SYSRES_CONST_DO_NOT_USE_ON_VIEW_ACCESS_TYPE_NAME "+"SYSRES_CONST_DOCUMENT_STORAGES_CODE "+"SYSRES_CONST_DOCUMENT_TEMPLATES_TYPE_NAME "+"SYSRES_CONST_DOUBLE_REQUISITE_CODE "+"SYSRES_CONST_EDITOR_CLOSE_FILE_OBSERV_TYPE_CODE "+"SYSRES_CONST_EDITOR_CLOSE_PROCESS_OBSERV_TYPE_CODE "+"SYSRES_CONST_EDITOR_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_APPLICATION_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_CREATE_SEVERAL_PROCESSES_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_EXTENSION_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_OBSERVER_BY_PROCESS_TYPE "+"SYSRES_CONST_EDITORS_REFERENCE_CODE "+"SYSRES_CONST_EDITORS_REPLACE_SPEC_CHARS_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_USE_PLUGINS_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_VIEW_DOCUMENT_OPENED_TO_EDIT_CODE "+"SYSRES_CONST_EDOC_CARD_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_CARD_TYPES_LINK_REQUISITE_CODE "+"SYSRES_CONST_EDOC_CERTIFICATE_AND_PASSWORD_ENCODE_CODE "+"SYSRES_CONST_EDOC_CERTIFICATE_ENCODE_CODE "+"SYSRES_CONST_EDOC_DATE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_KIND_REFERENCE_CODE "+"SYSRES_CONST_EDOC_KINDS_BY_TEMPLATE_ACTION_CODE "+"SYSRES_CONST_EDOC_MANAGE_ACCESS_CODE "+"SYSRES_CONST_EDOC_NONE_ENCODE_CODE "+"SYSRES_CONST_EDOC_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_EDOC_PASSWORD_ENCODE_CODE "+"SYSRES_CONST_EDOC_READONLY_ACCESS_CODE "+"SYSRES_CONST_EDOC_SHELL_LIFE_TYPE_VIEW_VALUE "+"SYSRES_CONST_EDOC_SIZE_RESTRICTION_PRIORITY_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_CHECK_ACCESS_RIGHTS_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_COMPUTER_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_DATABASE_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_EDIT_IN_STORAGE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_LOCAL_PATH_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_SHARED_SOURCE_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_TYPES_REFERENCE_CODE "+"SYSRES_CONST_EDOC_VERSION_ACTIVE_STAGE_CODE "+"SYSRES_CONST_EDOC_VERSION_DESIGN_STAGE_CODE "+"SYSRES_CONST_EDOC_VERSION_OBSOLETE_STAGE_CODE "+"SYSRES_CONST_EDOC_WRITE_ACCES_CODE "+"SYSRES_CONST_EDOCUMENT_CARD_REQUISITES_REFERENCE_CODE_SELECTED_REQUISITE "+"SYSRES_CONST_ENCODE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_END_DATE_REQUISITE_CODE "+"SYSRES_CONST_ENUMERATION_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EXECUTE_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_EXECUTIVE_FILE_STORAGE_TYPE "+"SYSRES_CONST_EXIST_CONST "+"SYSRES_CONST_EXIST_VALUE "+"SYSRES_CONST_EXPORT_LOCK_TYPE_ASK "+"SYSRES_CONST_EXPORT_LOCK_TYPE_WITH_LOCK "+"SYSRES_CONST_EXPORT_LOCK_TYPE_WITHOUT_LOCK "+"SYSRES_CONST_EXPORT_VERSION_TYPE_ASK "+"SYSRES_CONST_EXPORT_VERSION_TYPE_LAST "+"SYSRES_CONST_EXPORT_VERSION_TYPE_LAST_ACTIVE "+"SYSRES_CONST_EXTENSION_REQUISITE_CODE "+"SYSRES_CONST_FILTER_NAME_REQUISITE_CODE "+"SYSRES_CONST_FILTER_REQUISITE_CODE "+"SYSRES_CONST_FILTER_TYPE_COMMON_CODE "+"SYSRES_CONST_FILTER_TYPE_COMMON_NAME "+"SYSRES_CONST_FILTER_TYPE_USER_CODE "+"SYSRES_CONST_FILTER_TYPE_USER_NAME "+"SYSRES_CONST_FILTER_VALUE_REQUISITE_NAME "+"SYSRES_CONST_FLOAT_NUMBER_FORMAT_CHAR "+"SYSRES_CONST_FLOAT_REQUISITE_TYPE "+"SYSRES_CONST_FOLDER_AUTHOR_VALUE "+"SYSRES_CONST_FOLDER_KIND_ANY_OBJECTS "+"SYSRES_CONST_FOLDER_KIND_COMPONENTS "+"SYSRES_CONST_FOLDER_KIND_EDOCS "+"SYSRES_CONST_FOLDER_KIND_JOBS "+"SYSRES_CONST_FOLDER_KIND_TASKS "+"SYSRES_CONST_FOLDER_TYPE_COMMON "+"SYSRES_CONST_FOLDER_TYPE_COMPONENT "+"SYSRES_CONST_FOLDER_TYPE_FAVORITES "+"SYSRES_CONST_FOLDER_TYPE_INBOX "+"SYSRES_CONST_FOLDER_TYPE_OUTBOX "+"SYSRES_CONST_FOLDER_TYPE_QUICK_LAUNCH "+"SYSRES_CONST_FOLDER_TYPE_SEARCH "+"SYSRES_CONST_FOLDER_TYPE_SHORTCUTS "+"SYSRES_CONST_FOLDER_TYPE_USER "+"SYSRES_CONST_FROM_DICTIONARY_ENUM_METHOD_FLAG "+"SYSRES_CONST_FULL_SUBSTITUTE_TYPE "+"SYSRES_CONST_FULL_SUBSTITUTE_TYPE_CODE "+"SYSRES_CONST_FUNCTION_CANCEL_RESULT "+"SYSRES_CONST_FUNCTION_CATEGORY_SYSTEM "+"SYSRES_CONST_FUNCTION_CATEGORY_USER "+"SYSRES_CONST_FUNCTION_FAILURE_RESULT "+"SYSRES_CONST_FUNCTION_SAVE_RESULT "+"SYSRES_CONST_GENERATED_REQUISITE "+"SYSRES_CONST_GREEN_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_GROUP_ACCOUNT_TYPE_VALUE_CODE "+"SYSRES_CONST_GROUP_CATEGORY_NORMAL_CODE "+"SYSRES_CONST_GROUP_CATEGORY_NORMAL_NAME "+"SYSRES_CONST_GROUP_CATEGORY_SERVICE_CODE "+"SYSRES_CONST_GROUP_CATEGORY_SERVICE_NAME "+"SYSRES_CONST_GROUP_COMMON_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_GROUP_FULL_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_RIGHTS_T_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVER_CODES_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVER_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVICE_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_GROUP_USER_REQUISITE_CODE "+"SYSRES_CONST_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_GROUPS_REQUISITE_CODE "+"SYSRES_CONST_HIDDEN_MODE_NAME "+"SYSRES_CONST_HIGH_LVL_REQUISITE_CODE "+"SYSRES_CONST_HISTORY_ACTION_CREATE_CODE "+"SYSRES_CONST_HISTORY_ACTION_DELETE_CODE "+"SYSRES_CONST_HISTORY_ACTION_EDIT_CODE "+"SYSRES_CONST_HOUR_CHAR "+"SYSRES_CONST_ID_REQUISITE_CODE "+"SYSRES_CONST_IDSPS_REQUISITE_CODE "+"SYSRES_CONST_IMAGE_MODE_COLOR "+"SYSRES_CONST_IMAGE_MODE_GREYSCALE "+"SYSRES_CONST_IMAGE_MODE_MONOCHROME "+"SYSRES_CONST_IMPORTANCE_HIGH "+"SYSRES_CONST_IMPORTANCE_LOW "+"SYSRES_CONST_IMPORTANCE_NORMAL "+"SYSRES_CONST_IN_DESIGN_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_INCOMING_WORK_RULE_TYPE_CODE "+"SYSRES_CONST_INT_REQUISITE "+"SYSRES_CONST_INT_REQUISITE_TYPE "+"SYSRES_CONST_INTEGER_NUMBER_FORMAT_CHAR "+"SYSRES_CONST_INTEGER_TYPE_CHAR "+"SYSRES_CONST_IS_GENERATED_REQUISITE_NEGATIVE_VALUE "+"SYSRES_CONST_IS_PUBLIC_ROLE_REQUISITE_CODE "+"SYSRES_CONST_IS_REMOTE_USER_NEGATIVE_VALUE "+"SYSRES_CONST_IS_REMOTE_USER_POSITIVE_VALUE "+"SYSRES_CONST_IS_STORED_REQUISITE_NEGATIVE_VALUE "+"SYSRES_CONST_IS_STORED_REQUISITE_STORED_VALUE "+"SYSRES_CONST_ITALIC_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_JOB_BLOCK_DESCRIPTION "+"SYSRES_CONST_JOB_KIND_CONTROL_JOB "+"SYSRES_CONST_JOB_KIND_JOB "+"SYSRES_CONST_JOB_KIND_NOTICE "+"SYSRES_CONST_JOB_STATE_ABORTED "+"SYSRES_CONST_JOB_STATE_COMPLETE "+"SYSRES_CONST_JOB_STATE_WORKING "+"SYSRES_CONST_KIND_REQUISITE_CODE "+"SYSRES_CONST_KIND_REQUISITE_NAME "+"SYSRES_CONST_KINDS_CREATE_SHADOW_COPIES_REQUISITE_CODE "+"SYSRES_CONST_KINDS_DEFAULT_EDOC_LIFE_STAGE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALL_TEPLATES_ALLOWED_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALLOW_LIFE_CYCLE_STAGE_CHANGING_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALLOW_MULTIPLE_ACTIVE_VERSIONS_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_SHARE_ACCES_RIGHTS_BY_DEFAULT_CODE "+"SYSRES_CONST_KINDS_EDOC_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_TYPE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_SIGNERS_REQUISITES_CODE "+"SYSRES_CONST_KOD_INPUT_TYPE "+"SYSRES_CONST_LAST_UPDATE_DATE_REQUISITE_CODE "+"SYSRES_CONST_LIFE_CYCLE_START_STAGE_REQUISITE_CODE "+"SYSRES_CONST_LILAC_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_LINK_OBJECT_KIND_COMPONENT "+"SYSRES_CONST_LINK_OBJECT_KIND_DOCUMENT "+"SYSRES_CONST_LINK_OBJECT_KIND_EDOC "+"SYSRES_CONST_LINK_OBJECT_KIND_FOLDER "+"SYSRES_CONST_LINK_OBJECT_KIND_JOB "+"SYSRES_CONST_LINK_OBJECT_KIND_REFERENCE "+"SYSRES_CONST_LINK_OBJECT_KIND_TASK "+"SYSRES_CONST_LINK_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_LIST_REFERENCE_MODE_NAME "+"SYSRES_CONST_LOCALIZATION_DICTIONARY_MAIN_VIEW_CODE "+"SYSRES_CONST_MAIN_VIEW_CODE "+"SYSRES_CONST_MANUAL_ENUM_METHOD_FLAG "+"SYSRES_CONST_MASTER_COMP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_MASTER_TABLE_REC_ID_REQUISITE_CODE "+"SYSRES_CONST_MAXIMIZED_MODE_NAME "+"SYSRES_CONST_ME_VALUE "+"SYSRES_CONST_MESSAGE_ATTENTION_CAPTION "+"SYSRES_CONST_MESSAGE_CONFIRMATION_CAPTION "+"SYSRES_CONST_MESSAGE_ERROR_CAPTION "+"SYSRES_CONST_MESSAGE_INFORMATION_CAPTION "+"SYSRES_CONST_MINIMIZED_MODE_NAME "+"SYSRES_CONST_MINUTE_CHAR "+"SYSRES_CONST_MODULE_REQUISITE_CODE "+"SYSRES_CONST_MONITORING_BLOCK_DESCRIPTION "+"SYSRES_CONST_MONTH_FORMAT_VALUE "+"SYSRES_CONST_NAME_LOCALIZE_ID_REQUISITE_CODE "+"SYSRES_CONST_NAME_REQUISITE_CODE "+"SYSRES_CONST_NAME_SINGULAR_REQUISITE_CODE "+"SYSRES_CONST_NAMEAN_INPUT_TYPE "+"SYSRES_CONST_NEGATIVE_PICK_VALUE "+"SYSRES_CONST_NEGATIVE_VALUE "+"SYSRES_CONST_NO "+"SYSRES_CONST_NO_PICK_VALUE "+"SYSRES_CONST_NO_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_NO_VALUE "+"SYSRES_CONST_NONE_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_NONOPERATING_RECORD_FLAG_VALUE "+"SYSRES_CONST_NONOPERATING_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_NORMAL_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_NORMAL_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_NORMAL_MODE_NAME "+"SYSRES_CONST_NOT_ALLOWED_ACCESS_TYPE_CODE "+"SYSRES_CONST_NOT_ALLOWED_ACCESS_TYPE_NAME "+"SYSRES_CONST_NOTE_REQUISITE_CODE "+"SYSRES_CONST_NOTICE_BLOCK_DESCRIPTION "+"SYSRES_CONST_NUM_REQUISITE "+"SYSRES_CONST_NUM_STR_REQUISITE_CODE "+"SYSRES_CONST_NUMERATION_AUTO_NOT_STRONG "+"SYSRES_CONST_NUMERATION_AUTO_STRONG "+"SYSRES_CONST_NUMERATION_FROM_DICTONARY "+"SYSRES_CONST_NUMERATION_MANUAL "+"SYSRES_CONST_NUMERIC_TYPE_CHAR "+"SYSRES_CONST_NUMREQ_REQUISITE_CODE "+"SYSRES_CONST_OBSOLETE_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_CODE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_FEMININE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_OPTIONAL_FORM_COMP_REQCODE_PREFIX "+"SYSRES_CONST_ORANGE_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_ORIGINALREF_REQUISITE_CODE "+"SYSRES_CONST_OURFIRM_REF_CODE "+"SYSRES_CONST_OURFIRM_REQUISITE_CODE "+"SYSRES_CONST_OURFIRM_VAR "+"SYSRES_CONST_OUTGOING_WORK_RULE_TYPE_CODE "+"SYSRES_CONST_PICK_NEGATIVE_RESULT "+"SYSRES_CONST_PICK_POSITIVE_RESULT "+"SYSRES_CONST_PICK_REQUISITE "+"SYSRES_CONST_PICK_REQUISITE_TYPE "+"SYSRES_CONST_PICK_TYPE_CHAR "+"SYSRES_CONST_PLAN_STATUS_REQUISITE_CODE "+"SYSRES_CONST_PLATFORM_VERSION_COMMENT "+"SYSRES_CONST_PLUGINS_SETTINGS_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_POSITIVE_PICK_VALUE "+"SYSRES_CONST_POWER_TO_CREATE_ACTION_CODE "+"SYSRES_CONST_POWER_TO_SIGN_ACTION_CODE "+"SYSRES_CONST_PRIORITY_REQUISITE_CODE "+"SYSRES_CONST_QUALIFIED_TASK_TYPE "+"SYSRES_CONST_QUALIFIED_TASK_TYPE_CODE "+"SYSRES_CONST_RECSTAT_REQUISITE_CODE "+"SYSRES_CONST_RED_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_REF_ID_T_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_REF_REQUISITE "+"SYSRES_CONST_REF_REQUISITE_TYPE "+"SYSRES_CONST_REF_REQUISITES_REFERENCE_CODE_SELECTED_REQUISITE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_CREATE_ACTION_CODE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_DELETE_ACTION_CODE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_MODIFY_ACTION_CODE "+"SYSRES_CONST_REFERENCE_TYPE_CHAR "+"SYSRES_CONST_REFERENCE_TYPE_REQUISITE_NAME "+"SYSRES_CONST_REFERENCES_ADD_PARAMS_REQUISITE_CODE "+"SYSRES_CONST_REFERENCES_DISPLAY_REQUISITE_REQUISITE_CODE "+"SYSRES_CONST_REMOTE_SERVER_STATUS_WORKING "+"SYSRES_CONST_REMOTE_SERVER_TYPE_MAIN "+"SYSRES_CONST_REMOTE_SERVER_TYPE_SECONDARY "+"SYSRES_CONST_REMOTE_USER_FLAG_VALUE_CODE "+"SYSRES_CONST_REPORT_APP_EDITOR_INTERNAL "+"SYSRES_CONST_REPORT_BASE_REPORT_ID_REQUISITE_CODE "+"SYSRES_CONST_REPORT_BASE_REPORT_REQUISITE_CODE "+"SYSRES_CONST_REPORT_SCRIPT_REQUISITE_CODE "+"SYSRES_CONST_REPORT_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_REPORT_VIEWER_CODE_REQUISITE_CODE "+"SYSRES_CONST_REQ_ALLOW_COMPONENT_DEFAULT_VALUE "+"SYSRES_CONST_REQ_ALLOW_RECORD_DEFAULT_VALUE "+"SYSRES_CONST_REQ_ALLOW_SERVER_COMPONENT_DEFAULT_VALUE "+"SYSRES_CONST_REQ_MODE_AVAILABLE_CODE "+"SYSRES_CONST_REQ_MODE_EDIT_CODE "+"SYSRES_CONST_REQ_MODE_HIDDEN_CODE "+"SYSRES_CONST_REQ_MODE_NOT_AVAILABLE_CODE "+"SYSRES_CONST_REQ_MODE_VIEW_CODE "+"SYSRES_CONST_REQ_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_REQ_SECTION_VALUE "+"SYSRES_CONST_REQ_TYPE_VALUE "+"SYSRES_CONST_REQUISITE_FORMAT_BY_UNIT "+"SYSRES_CONST_REQUISITE_FORMAT_DATE_FULL "+"SYSRES_CONST_REQUISITE_FORMAT_DATE_TIME "+"SYSRES_CONST_REQUISITE_FORMAT_LEFT "+"SYSRES_CONST_REQUISITE_FORMAT_RIGHT "+"SYSRES_CONST_REQUISITE_FORMAT_WITHOUT_UNIT "+"SYSRES_CONST_REQUISITE_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_REQUISITE_SECTION_ACTIONS "+"SYSRES_CONST_REQUISITE_SECTION_BUTTON "+"SYSRES_CONST_REQUISITE_SECTION_BUTTONS "+"SYSRES_CONST_REQUISITE_SECTION_CARD "+"SYSRES_CONST_REQUISITE_SECTION_TABLE "+"SYSRES_CONST_REQUISITE_SECTION_TABLE10 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE11 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE12 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE13 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE14 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE15 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE16 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE17 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE18 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE19 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE2 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE20 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE21 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE22 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE23 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE24 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE3 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE4 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE5 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE6 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE7 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE8 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE9 "+"SYSRES_CONST_REQUISITES_PSEUDOREFERENCE_REQUISITE_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_RIGHT_ALIGNMENT_CODE "+"SYSRES_CONST_ROLES_REFERENCE_CODE "+"SYSRES_CONST_ROUTE_STEP_AFTER_RUS "+"SYSRES_CONST_ROUTE_STEP_AND_CONDITION_RUS "+"SYSRES_CONST_ROUTE_STEP_OR_CONDITION_RUS "+"SYSRES_CONST_ROUTE_TYPE_COMPLEX "+"SYSRES_CONST_ROUTE_TYPE_PARALLEL "+"SYSRES_CONST_ROUTE_TYPE_SERIAL "+"SYSRES_CONST_SBDATASETDESC_NEGATIVE_VALUE "+"SYSRES_CONST_SBDATASETDESC_POSITIVE_VALUE "+"SYSRES_CONST_SBVIEWSDESC_POSITIVE_VALUE "+"SYSRES_CONST_SCRIPT_BLOCK_DESCRIPTION "+"SYSRES_CONST_SEARCH_BY_TEXT_REQUISITE_CODE "+"SYSRES_CONST_SEARCHES_COMPONENT_CONTENT "+"SYSRES_CONST_SEARCHES_CRITERIA_ACTION_NAME "+"SYSRES_CONST_SEARCHES_EDOC_CONTENT "+"SYSRES_CONST_SEARCHES_FOLDER_CONTENT "+"SYSRES_CONST_SEARCHES_JOB_CONTENT "+"SYSRES_CONST_SEARCHES_REFERENCE_CODE "+"SYSRES_CONST_SEARCHES_TASK_CONTENT "+"SYSRES_CONST_SECOND_CHAR "+"SYSRES_CONST_SECTION_REQUISITE_ACTIONS_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_CARD_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_CODE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_1_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_2_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_3_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_4_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_5_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_6_VALUE "+"SYSRES_CONST_SELECT_REFERENCE_MODE_NAME "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE_ONLY_CHILD "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE_WITH_CHILD "+"SYSRES_CONST_SELECT_TYPE_UNSLECTABLE "+"SYSRES_CONST_SERVER_TYPE_MAIN "+"SYSRES_CONST_SERVICE_USER_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_SETTINGS_USER_REQUISITE_CODE "+"SYSRES_CONST_SIGNATURE_AND_ENCODE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_SIGNATURE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_SINGULAR_TITLE_REQUISITE_CODE "+"SYSRES_CONST_SQL_SERVER_AUTHENTIFICATION_FLAG_VALUE_CODE "+"SYSRES_CONST_SQL_SERVER_ENCODE_AUTHENTIFICATION_FLAG_VALUE_CODE "+"SYSRES_CONST_STANDART_ROUTE_REFERENCE_CODE "+"SYSRES_CONST_STANDART_ROUTE_REFERENCE_COMMENT_REQUISITE_CODE "+"SYSRES_CONST_STANDART_ROUTES_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_STATE_REQ_NAME "+"SYSRES_CONST_STATE_REQUISITE_ACTIVE_VALUE "+"SYSRES_CONST_STATE_REQUISITE_CLOSED_VALUE "+"SYSRES_CONST_STATE_REQUISITE_CODE "+"SYSRES_CONST_STATIC_ROLE_TYPE_CODE "+"SYSRES_CONST_STATUS_PLAN_DEFAULT_VALUE "+"SYSRES_CONST_STATUS_VALUE_AUTOCLEANING "+"SYSRES_CONST_STATUS_VALUE_BLUE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_COMPLETE "+"SYSRES_CONST_STATUS_VALUE_GREEN_SQUARE "+"SYSRES_CONST_STATUS_VALUE_ORANGE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_PURPLE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_RED_SQUARE "+"SYSRES_CONST_STATUS_VALUE_SUSPEND "+"SYSRES_CONST_STATUS_VALUE_YELLOW_SQUARE "+"SYSRES_CONST_STDROUTE_SHOW_TO_USERS_REQUISITE_CODE "+"SYSRES_CONST_STORAGE_TYPE_FILE "+"SYSRES_CONST_STORAGE_TYPE_SQL_SERVER "+"SYSRES_CONST_STR_REQUISITE "+"SYSRES_CONST_STRIKEOUT_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_STRING_FORMAT_LEFT_ALIGN_CHAR "+"SYSRES_CONST_STRING_FORMAT_RIGHT_ALIGN_CHAR "+"SYSRES_CONST_STRING_REQUISITE_CODE "+"SYSRES_CONST_STRING_REQUISITE_TYPE "+"SYSRES_CONST_STRING_TYPE_CHAR "+"SYSRES_CONST_SUBSTITUTES_PSEUDOREFERENCE_CODE "+"SYSRES_CONST_SUBTASK_BLOCK_DESCRIPTION "+"SYSRES_CONST_SYSTEM_SETTING_CURRENT_USER_PARAM_VALUE "+"SYSRES_CONST_SYSTEM_SETTING_EMPTY_VALUE_PARAM_VALUE "+"SYSRES_CONST_SYSTEM_VERSION_COMMENT "+"SYSRES_CONST_TASK_ACCESS_TYPE_ALL "+"SYSRES_CONST_TASK_ACCESS_TYPE_ALL_MEMBERS "+"SYSRES_CONST_TASK_ACCESS_TYPE_MANUAL "+"SYSRES_CONST_TASK_ENCODE_TYPE_CERTIFICATION "+"SYSRES_CONST_TASK_ENCODE_TYPE_CERTIFICATION_AND_PASSWORD "+"SYSRES_CONST_TASK_ENCODE_TYPE_NONE "+"SYSRES_CONST_TASK_ENCODE_TYPE_PASSWORD "+"SYSRES_CONST_TASK_ROUTE_ALL_CONDITION "+"SYSRES_CONST_TASK_ROUTE_AND_CONDITION "+"SYSRES_CONST_TASK_ROUTE_OR_CONDITION "+"SYSRES_CONST_TASK_STATE_ABORTED "+"SYSRES_CONST_TASK_STATE_COMPLETE "+"SYSRES_CONST_TASK_STATE_CONTINUED "+"SYSRES_CONST_TASK_STATE_CONTROL "+"SYSRES_CONST_TASK_STATE_INIT "+"SYSRES_CONST_TASK_STATE_WORKING "+"SYSRES_CONST_TASK_TITLE "+"SYSRES_CONST_TASK_TYPES_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_TASK_TYPES_REFERENCE_CODE "+"SYSRES_CONST_TEMPLATES_REFERENCE_CODE "+"SYSRES_CONST_TEST_DATE_REQUISITE_NAME "+"SYSRES_CONST_TEST_DEV_DATABASE_NAME "+"SYSRES_CONST_TEST_DEV_SYSTEM_CODE "+"SYSRES_CONST_TEST_EDMS_DATABASE_NAME "+"SYSRES_CONST_TEST_EDMS_MAIN_CODE "+"SYSRES_CONST_TEST_EDMS_MAIN_DB_NAME "+"SYSRES_CONST_TEST_EDMS_SECOND_CODE "+"SYSRES_CONST_TEST_EDMS_SECOND_DB_NAME "+"SYSRES_CONST_TEST_EDMS_SYSTEM_CODE "+"SYSRES_CONST_TEST_NUMERIC_REQUISITE_NAME "+"SYSRES_CONST_TEXT_REQUISITE "+"SYSRES_CONST_TEXT_REQUISITE_CODE "+"SYSRES_CONST_TEXT_REQUISITE_TYPE "+"SYSRES_CONST_TEXT_TYPE_CHAR "+"SYSRES_CONST_TYPE_CODE_REQUISITE_CODE "+"SYSRES_CONST_TYPE_REQUISITE_CODE "+"SYSRES_CONST_UNDEFINED_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_UNITS_SECTION_ID_REQUISITE_CODE "+"SYSRES_CONST_UNITS_SECTION_REQUISITE_CODE "+"SYSRES_CONST_UNOPERATING_RECORD_FLAG_VALUE_CODE "+"SYSRES_CONST_UNSTORED_DATA_REQUISITE_CODE "+"SYSRES_CONST_UNSTORED_DATA_REQUISITE_NAME "+"SYSRES_CONST_USE_ACCESS_TYPE_CODE "+"SYSRES_CONST_USE_ACCESS_TYPE_NAME "+"SYSRES_CONST_USER_ACCOUNT_TYPE_VALUE_CODE "+"SYSRES_CONST_USER_ADDITIONAL_INFORMATION_REQUISITE_CODE "+"SYSRES_CONST_USER_AND_GROUP_ID_FROM_PSEUDOREFERENCE_REQUISITE_CODE "+"SYSRES_CONST_USER_CATEGORY_NORMAL "+"SYSRES_CONST_USER_CERTIFICATE_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_STATE_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_SUBJECT_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_THUMBPRINT_REQUISITE_CODE "+"SYSRES_CONST_USER_COMMON_CATEGORY "+"SYSRES_CONST_USER_COMMON_CATEGORY_CODE "+"SYSRES_CONST_USER_FULL_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_GROUP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_USER_LOGIN_REQUISITE_CODE "+"SYSRES_CONST_USER_REMOTE_CONTROLLER_REQUISITE_CODE "+"SYSRES_CONST_USER_REMOTE_SYSTEM_REQUISITE_CODE "+"SYSRES_CONST_USER_RIGHTS_T_REQUISITE_CODE "+"SYSRES_CONST_USER_SERVER_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_SERVICE_CATEGORY "+"SYSRES_CONST_USER_SERVICE_CATEGORY_CODE "+"SYSRES_CONST_USER_STATUS_ADMINISTRATOR_CODE "+"SYSRES_CONST_USER_STATUS_ADMINISTRATOR_NAME "+"SYSRES_CONST_USER_STATUS_DEVELOPER_CODE "+"SYSRES_CONST_USER_STATUS_DEVELOPER_NAME "+"SYSRES_CONST_USER_STATUS_DISABLED_CODE "+"SYSRES_CONST_USER_STATUS_DISABLED_NAME "+"SYSRES_CONST_USER_STATUS_SYSTEM_DEVELOPER_CODE "+"SYSRES_CONST_USER_STATUS_USER_CODE "+"SYSRES_CONST_USER_STATUS_USER_NAME "+"SYSRES_CONST_USER_STATUS_USER_NAME_DEPRECATED "+"SYSRES_CONST_USER_TYPE_FIELD_VALUE_USER "+"SYSRES_CONST_USER_TYPE_REQUISITE_CODE "+"SYSRES_CONST_USERS_CONTROLLER_REQUISITE_CODE "+"SYSRES_CONST_USERS_IS_MAIN_SERVER_REQUISITE_CODE "+"SYSRES_CONST_USERS_REFERENCE_CODE "+"SYSRES_CONST_USERS_REGISTRATION_CERTIFICATES_ACTION_NAME "+"SYSRES_CONST_USERS_REQUISITE_CODE "+"SYSRES_CONST_USERS_SYSTEM_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_ACCESS_RIGHTS_TYPR_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_AUTHENTICATION_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_COMPONENT_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_GROUP_REQUISITE_CODE "+"SYSRES_CONST_USERS_VIEW_CERTIFICATES_ACTION_NAME "+"SYSRES_CONST_VIEW_DEFAULT_CODE "+"SYSRES_CONST_VIEW_DEFAULT_NAME "+"SYSRES_CONST_VIEWER_REQUISITE_CODE "+"SYSRES_CONST_WAITING_BLOCK_DESCRIPTION "+"SYSRES_CONST_WIZARD_FORM_LABEL_TEST_STRING  "+"SYSRES_CONST_WIZARD_QUERY_PARAM_HEIGHT_ETALON_STRING "+"SYSRES_CONST_WIZARD_REFERENCE_COMMENT_REQUISITE_CODE "+"SYSRES_CONST_WORK_RULES_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_WORK_TIME_CALENDAR_REFERENCE_CODE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE_CODE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE_CODE_RUS "+"SYSRES_CONST_WORK_WORKFLOW_SOFT_ROUTE_TYPE_VALUE_CODE_RUS "+"SYSRES_CONST_WORKFLOW_ROUTE_TYPR_HARD "+"SYSRES_CONST_WORKFLOW_ROUTE_TYPR_SOFT "+"SYSRES_CONST_XML_ENCODING "+"SYSRES_CONST_XREC_STAT_REQUISITE_CODE "+"SYSRES_CONST_XRECID_FIELD_NAME "+"SYSRES_CONST_YES "+"SYSRES_CONST_YES_NO_2_REQUISITE_CODE "+"SYSRES_CONST_YES_NO_REQUISITE_CODE "+"SYSRES_CONST_YES_NO_T_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_YES_PICK_VALUE "+"SYSRES_CONST_YES_VALUE ";var base_constants="CR FALSE nil NO_VALUE NULL TAB TRUE YES_VALUE ";var base_group_name_constants="ADMINISTRATORS_GROUP_NAME CUSTOMIZERS_GROUP_NAME DEVELOPERS_GROUP_NAME SERVICE_USERS_GROUP_NAME ";var decision_block_properties_constants="DECISION_BLOCK_FIRST_OPERAND_PROPERTY DECISION_BLOCK_NAME_PROPERTY DECISION_BLOCK_OPERATION_PROPERTY "+"DECISION_BLOCK_RESULT_TYPE_PROPERTY DECISION_BLOCK_SECOND_OPERAND_PROPERTY ";var file_extension_constants="ANY_FILE_EXTENTION COMPRESSED_DOCUMENT_EXTENSION EXTENDED_DOCUMENT_EXTENSION "+"SHORT_COMPRESSED_DOCUMENT_EXTENSION SHORT_EXTENDED_DOCUMENT_EXTENSION ";var job_block_properties_constants="JOB_BLOCK_ABORT_DEADLINE_PROPERTY "+"JOB_BLOCK_AFTER_FINISH_EVENT "+"JOB_BLOCK_AFTER_QUERY_PARAMETERS_EVENT "+"JOB_BLOCK_ATTACHMENT_PROPERTY "+"JOB_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"JOB_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"JOB_BLOCK_BEFORE_QUERY_PARAMETERS_EVENT "+"JOB_BLOCK_BEFORE_START_EVENT "+"JOB_BLOCK_CREATED_JOBS_PROPERTY "+"JOB_BLOCK_DEADLINE_PROPERTY "+"JOB_BLOCK_EXECUTION_RESULTS_PROPERTY "+"JOB_BLOCK_IS_PARALLEL_PROPERTY "+"JOB_BLOCK_IS_RELATIVE_ABORT_DEADLINE_PROPERTY "+"JOB_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"JOB_BLOCK_JOB_TEXT_PROPERTY "+"JOB_BLOCK_NAME_PROPERTY "+"JOB_BLOCK_NEED_SIGN_ON_PERFORM_PROPERTY "+"JOB_BLOCK_PERFORMER_PROPERTY "+"JOB_BLOCK_RELATIVE_ABORT_DEADLINE_TYPE_PROPERTY "+"JOB_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"JOB_BLOCK_SUBJECT_PROPERTY ";var language_code_constants="ENGLISH_LANGUAGE_CODE RUSSIAN_LANGUAGE_CODE ";var launching_external_applications_constants="smHidden smMaximized smMinimized smNormal wmNo wmYes ";var link_kind_constants="COMPONENT_TOKEN_LINK_KIND "+"DOCUMENT_LINK_KIND "+"EDOCUMENT_LINK_KIND "+"FOLDER_LINK_KIND "+"JOB_LINK_KIND "+"REFERENCE_LINK_KIND "+"TASK_LINK_KIND ";var lock_type_constants="COMPONENT_TOKEN_LOCK_TYPE EDOCUMENT_VERSION_LOCK_TYPE ";var monitor_block_properties_constants="MONITOR_BLOCK_AFTER_FINISH_EVENT "+"MONITOR_BLOCK_BEFORE_START_EVENT "+"MONITOR_BLOCK_DEADLINE_PROPERTY "+"MONITOR_BLOCK_INTERVAL_PROPERTY "+"MONITOR_BLOCK_INTERVAL_TYPE_PROPERTY "+"MONITOR_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"MONITOR_BLOCK_NAME_PROPERTY "+"MONITOR_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"MONITOR_BLOCK_SEARCH_SCRIPT_PROPERTY ";var notice_block_properties_constants="NOTICE_BLOCK_AFTER_FINISH_EVENT "+"NOTICE_BLOCK_ATTACHMENT_PROPERTY "+"NOTICE_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"NOTICE_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"NOTICE_BLOCK_BEFORE_START_EVENT "+"NOTICE_BLOCK_CREATED_NOTICES_PROPERTY "+"NOTICE_BLOCK_DEADLINE_PROPERTY "+"NOTICE_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"NOTICE_BLOCK_NAME_PROPERTY "+"NOTICE_BLOCK_NOTICE_TEXT_PROPERTY "+"NOTICE_BLOCK_PERFORMER_PROPERTY "+"NOTICE_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"NOTICE_BLOCK_SUBJECT_PROPERTY ";var object_events_constants="dseAfterCancel "+"dseAfterClose "+"dseAfterDelete "+"dseAfterDeleteOutOfTransaction "+"dseAfterInsert "+"dseAfterOpen "+"dseAfterScroll "+"dseAfterUpdate "+"dseAfterUpdateOutOfTransaction "+"dseBeforeCancel "+"dseBeforeClose "+"dseBeforeDelete "+"dseBeforeDetailUpdate "+"dseBeforeInsert "+"dseBeforeOpen "+"dseBeforeUpdate "+"dseOnAnyRequisiteChange "+"dseOnCloseRecord "+"dseOnDeleteError "+"dseOnOpenRecord "+"dseOnPrepareUpdate "+"dseOnUpdateError "+"dseOnUpdateRatifiedRecord "+"dseOnValidDelete "+"dseOnValidUpdate "+"reOnChange "+"reOnChangeValues "+"SELECTION_BEGIN_ROUTE_EVENT "+"SELECTION_END_ROUTE_EVENT ";var object_params_constants="CURRENT_PERIOD_IS_REQUIRED "+"PREVIOUS_CARD_TYPE_NAME "+"SHOW_RECORD_PROPERTIES_FORM ";var other_constants="ACCESS_RIGHTS_SETTING_DIALOG_CODE "+"ADMINISTRATOR_USER_CODE "+"ANALYTIC_REPORT_TYPE "+"asrtHideLocal "+"asrtHideRemote "+"CALCULATED_ROLE_TYPE_CODE "+"COMPONENTS_REFERENCE_DEVELOPER_VIEW_CODE "+"DCTS_TEST_PROTOCOLS_FOLDER_PATH "+"E_EDOC_VERSION_ALREADY_APPROVINGLY_SIGNED "+"E_EDOC_VERSION_ALREADY_APPROVINGLY_SIGNED_BY_USER "+"E_EDOC_VERSION_ALREDY_SIGNED "+"E_EDOC_VERSION_ALREDY_SIGNED_BY_USER "+"EDOC_TYPES_CODE_REQUISITE_FIELD_NAME "+"EDOCUMENTS_ALIAS_NAME "+"FILES_FOLDER_PATH "+"FILTER_OPERANDS_DELIMITER "+"FILTER_OPERATIONS_DELIMITER "+"FORMCARD_NAME "+"FORMLIST_NAME "+"GET_EXTENDED_DOCUMENT_EXTENSION_CREATION_MODE "+"GET_EXTENDED_DOCUMENT_EXTENSION_IMPORT_MODE "+"INTEGRATED_REPORT_TYPE "+"IS_BUILDER_APPLICATION_ROLE "+"IS_BUILDER_APPLICATION_ROLE2 "+"IS_BUILDER_USERS "+"ISBSYSDEV "+"LOG_FOLDER_PATH "+"mbCancel "+"mbNo "+"mbNoToAll "+"mbOK "+"mbYes "+"mbYesToAll "+"MEMORY_DATASET_DESRIPTIONS_FILENAME "+"mrNo "+"mrNoToAll "+"mrYes "+"mrYesToAll "+"MULTIPLE_SELECT_DIALOG_CODE "+"NONOPERATING_RECORD_FLAG_FEMININE "+"NONOPERATING_RECORD_FLAG_MASCULINE "+"OPERATING_RECORD_FLAG_FEMININE "+"OPERATING_RECORD_FLAG_MASCULINE "+"PROFILING_SETTINGS_COMMON_SETTINGS_CODE_VALUE "+"PROGRAM_INITIATED_LOOKUP_ACTION "+"ratDelete "+"ratEdit "+"ratInsert "+"REPORT_TYPE "+"REQUIRED_PICK_VALUES_VARIABLE "+"rmCard "+"rmList "+"SBRTE_PROGID_DEV "+"SBRTE_PROGID_RELEASE "+"STATIC_ROLE_TYPE_CODE "+"SUPPRESS_EMPTY_TEMPLATE_CREATION "+"SYSTEM_USER_CODE "+"UPDATE_DIALOG_DATASET "+"USED_IN_OBJECT_HINT_PARAM "+"USER_INITIATED_LOOKUP_ACTION "+"USER_NAME_FORMAT "+"USER_SELECTION_RESTRICTIONS "+"WORKFLOW_TEST_PROTOCOLS_FOLDER_PATH "+"ELS_SUBTYPE_CONTROL_NAME "+"ELS_FOLDER_KIND_CONTROL_NAME "+"REPEAT_PROCESS_CURRENT_OBJECT_EXCEPTION_NAME ";var privileges_constants="PRIVILEGE_COMPONENT_FULL_ACCESS "+"PRIVILEGE_DEVELOPMENT_EXPORT "+"PRIVILEGE_DEVELOPMENT_IMPORT "+"PRIVILEGE_DOCUMENT_DELETE "+"PRIVILEGE_ESD "+"PRIVILEGE_FOLDER_DELETE "+"PRIVILEGE_MANAGE_ACCESS_RIGHTS "+"PRIVILEGE_MANAGE_REPLICATION "+"PRIVILEGE_MANAGE_SESSION_SERVER "+"PRIVILEGE_OBJECT_FULL_ACCESS "+"PRIVILEGE_OBJECT_VIEW "+"PRIVILEGE_RESERVE_LICENSE "+"PRIVILEGE_SYSTEM_CUSTOMIZE "+"PRIVILEGE_SYSTEM_DEVELOP "+"PRIVILEGE_SYSTEM_INSTALL "+"PRIVILEGE_TASK_DELETE "+"PRIVILEGE_USER_PLUGIN_SETTINGS_CUSTOMIZE "+"PRIVILEGES_PSEUDOREFERENCE_CODE ";var pseudoreference_code_constants="ACCESS_TYPES_PSEUDOREFERENCE_CODE "+"ALL_AVAILABLE_COMPONENTS_PSEUDOREFERENCE_CODE "+"ALL_AVAILABLE_PRIVILEGES_PSEUDOREFERENCE_CODE "+"ALL_REPLICATE_COMPONENTS_PSEUDOREFERENCE_CODE "+"AVAILABLE_DEVELOPERS_COMPONENTS_PSEUDOREFERENCE_CODE "+"COMPONENTS_PSEUDOREFERENCE_CODE "+"FILTRATER_SETTINGS_CONFLICTS_PSEUDOREFERENCE_CODE "+"GROUPS_PSEUDOREFERENCE_CODE "+"RECEIVE_PROTOCOL_PSEUDOREFERENCE_CODE "+"REFERENCE_REQUISITE_PSEUDOREFERENCE_CODE "+"REFERENCE_REQUISITES_PSEUDOREFERENCE_CODE "+"REFTYPES_PSEUDOREFERENCE_CODE "+"REPLICATION_SEANCES_DIARY_PSEUDOREFERENCE_CODE "+"SEND_PROTOCOL_PSEUDOREFERENCE_CODE "+"SUBSTITUTES_PSEUDOREFERENCE_CODE "+"SYSTEM_SETTINGS_PSEUDOREFERENCE_CODE "+"UNITS_PSEUDOREFERENCE_CODE "+"USERS_PSEUDOREFERENCE_CODE "+"VIEWERS_PSEUDOREFERENCE_CODE ";var requisite_ISBCertificateType_values_constants="CERTIFICATE_TYPE_ENCRYPT "+"CERTIFICATE_TYPE_SIGN "+"CERTIFICATE_TYPE_SIGN_AND_ENCRYPT ";var requisite_ISBEDocStorageType_values_constants="STORAGE_TYPE_FILE "+"STORAGE_TYPE_NAS_CIFS "+"STORAGE_TYPE_SAPERION "+"STORAGE_TYPE_SQL_SERVER ";var requisite_compType2_values_constants="COMPTYPE2_REQUISITE_DOCUMENTS_VALUE "+"COMPTYPE2_REQUISITE_TASKS_VALUE "+"COMPTYPE2_REQUISITE_FOLDERS_VALUE "+"COMPTYPE2_REQUISITE_REFERENCES_VALUE ";var requisite_name_constants="SYSREQ_CODE "+"SYSREQ_COMPTYPE2 "+"SYSREQ_CONST_AVAILABLE_FOR_WEB "+"SYSREQ_CONST_COMMON_CODE "+"SYSREQ_CONST_COMMON_VALUE "+"SYSREQ_CONST_FIRM_CODE "+"SYSREQ_CONST_FIRM_STATUS "+"SYSREQ_CONST_FIRM_VALUE "+"SYSREQ_CONST_SERVER_STATUS "+"SYSREQ_CONTENTS "+"SYSREQ_DATE_OPEN "+"SYSREQ_DATE_CLOSE "+"SYSREQ_DESCRIPTION "+"SYSREQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_DOUBLE "+"SYSREQ_EDOC_ACCESS_TYPE "+"SYSREQ_EDOC_AUTHOR "+"SYSREQ_EDOC_CREATED "+"SYSREQ_EDOC_DELEGATE_RIGHTS_REQUISITE_CODE "+"SYSREQ_EDOC_EDITOR "+"SYSREQ_EDOC_ENCODE_TYPE "+"SYSREQ_EDOC_ENCRYPTION_PLUGIN_NAME "+"SYSREQ_EDOC_ENCRYPTION_PLUGIN_VERSION "+"SYSREQ_EDOC_EXPORT_DATE "+"SYSREQ_EDOC_EXPORTER "+"SYSREQ_EDOC_KIND "+"SYSREQ_EDOC_LIFE_STAGE_NAME "+"SYSREQ_EDOC_LOCKED_FOR_SERVER_CODE "+"SYSREQ_EDOC_MODIFIED "+"SYSREQ_EDOC_NAME "+"SYSREQ_EDOC_NOTE "+"SYSREQ_EDOC_QUALIFIED_ID "+"SYSREQ_EDOC_SESSION_KEY "+"SYSREQ_EDOC_SESSION_KEY_ENCRYPTION_PLUGIN_NAME "+"SYSREQ_EDOC_SESSION_KEY_ENCRYPTION_PLUGIN_VERSION "+"SYSREQ_EDOC_SIGNATURE_TYPE "+"SYSREQ_EDOC_SIGNED "+"SYSREQ_EDOC_STORAGE "+"SYSREQ_EDOC_STORAGES_ARCHIVE_STORAGE "+"SYSREQ_EDOC_STORAGES_CHECK_RIGHTS "+"SYSREQ_EDOC_STORAGES_COMPUTER_NAME "+"SYSREQ_EDOC_STORAGES_EDIT_IN_STORAGE "+"SYSREQ_EDOC_STORAGES_EXECUTIVE_STORAGE "+"SYSREQ_EDOC_STORAGES_FUNCTION "+"SYSREQ_EDOC_STORAGES_INITIALIZED "+"SYSREQ_EDOC_STORAGES_LOCAL_PATH "+"SYSREQ_EDOC_STORAGES_SAPERION_DATABASE_NAME "+"SYSREQ_EDOC_STORAGES_SEARCH_BY_TEXT "+"SYSREQ_EDOC_STORAGES_SERVER_NAME "+"SYSREQ_EDOC_STORAGES_SHARED_SOURCE_NAME "+"SYSREQ_EDOC_STORAGES_TYPE "+"SYSREQ_EDOC_TEXT_MODIFIED "+"SYSREQ_EDOC_TYPE_ACT_CODE "+"SYSREQ_EDOC_TYPE_ACT_DESCRIPTION "+"SYSREQ_EDOC_TYPE_ACT_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_ACT_ON_EXECUTE "+"SYSREQ_EDOC_TYPE_ACT_ON_EXECUTE_EXISTS "+"SYSREQ_EDOC_TYPE_ACT_SECTION "+"SYSREQ_EDOC_TYPE_ADD_PARAMS "+"SYSREQ_EDOC_TYPE_COMMENT "+"SYSREQ_EDOC_TYPE_EVENT_TEXT "+"SYSREQ_EDOC_TYPE_NAME_IN_SINGULAR "+"SYSREQ_EDOC_TYPE_NAME_IN_SINGULAR_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_NAME_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_NUMERATION_METHOD "+"SYSREQ_EDOC_TYPE_PSEUDO_REQUISITE_CODE "+"SYSREQ_EDOC_TYPE_REQ_CODE "+"SYSREQ_EDOC_TYPE_REQ_DESCRIPTION "+"SYSREQ_EDOC_TYPE_REQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_REQ_IS_LEADING "+"SYSREQ_EDOC_TYPE_REQ_IS_REQUIRED "+"SYSREQ_EDOC_TYPE_REQ_NUMBER "+"SYSREQ_EDOC_TYPE_REQ_ON_CHANGE "+"SYSREQ_EDOC_TYPE_REQ_ON_CHANGE_EXISTS "+"SYSREQ_EDOC_TYPE_REQ_ON_SELECT "+"SYSREQ_EDOC_TYPE_REQ_ON_SELECT_KIND "+"SYSREQ_EDOC_TYPE_REQ_SECTION "+"SYSREQ_EDOC_TYPE_VIEW_CARD "+"SYSREQ_EDOC_TYPE_VIEW_CODE "+"SYSREQ_EDOC_TYPE_VIEW_COMMENT "+"SYSREQ_EDOC_TYPE_VIEW_IS_MAIN "+"SYSREQ_EDOC_TYPE_VIEW_NAME "+"SYSREQ_EDOC_TYPE_VIEW_NAME_LOCALIZE_ID "+"SYSREQ_EDOC_VERSION_AUTHOR "+"SYSREQ_EDOC_VERSION_CRC "+"SYSREQ_EDOC_VERSION_DATA "+"SYSREQ_EDOC_VERSION_EDITOR "+"SYSREQ_EDOC_VERSION_EXPORT_DATE "+"SYSREQ_EDOC_VERSION_EXPORTER "+"SYSREQ_EDOC_VERSION_HIDDEN "+"SYSREQ_EDOC_VERSION_LIFE_STAGE "+"SYSREQ_EDOC_VERSION_MODIFIED "+"SYSREQ_EDOC_VERSION_NOTE "+"SYSREQ_EDOC_VERSION_SIGNATURE_TYPE "+"SYSREQ_EDOC_VERSION_SIGNED "+"SYSREQ_EDOC_VERSION_SIZE "+"SYSREQ_EDOC_VERSION_SOURCE "+"SYSREQ_EDOC_VERSION_TEXT_MODIFIED "+"SYSREQ_EDOCKIND_DEFAULT_VERSION_STATE_CODE "+"SYSREQ_FOLDER_KIND "+"SYSREQ_FUNC_CATEGORY "+"SYSREQ_FUNC_COMMENT "+"SYSREQ_FUNC_GROUP "+"SYSREQ_FUNC_GROUP_COMMENT "+"SYSREQ_FUNC_GROUP_NUMBER "+"SYSREQ_FUNC_HELP "+"SYSREQ_FUNC_PARAM_DEF_VALUE "+"SYSREQ_FUNC_PARAM_IDENT "+"SYSREQ_FUNC_PARAM_NUMBER "+"SYSREQ_FUNC_PARAM_TYPE "+"SYSREQ_FUNC_TEXT "+"SYSREQ_GROUP_CATEGORY "+"SYSREQ_ID "+"SYSREQ_LAST_UPDATE "+"SYSREQ_LEADER_REFERENCE "+"SYSREQ_LINE_NUMBER "+"SYSREQ_MAIN_RECORD_ID "+"SYSREQ_NAME "+"SYSREQ_NAME_LOCALIZE_ID "+"SYSREQ_NOTE "+"SYSREQ_ORIGINAL_RECORD "+"SYSREQ_OUR_FIRM "+"SYSREQ_PROFILING_SETTINGS_BATCH_LOGING "+"SYSREQ_PROFILING_SETTINGS_BATCH_SIZE "+"SYSREQ_PROFILING_SETTINGS_PROFILING_ENABLED "+"SYSREQ_PROFILING_SETTINGS_SQL_PROFILING_ENABLED "+"SYSREQ_PROFILING_SETTINGS_START_LOGGED "+"SYSREQ_RECORD_STATUS "+"SYSREQ_REF_REQ_FIELD_NAME "+"SYSREQ_REF_REQ_FORMAT "+"SYSREQ_REF_REQ_GENERATED "+"SYSREQ_REF_REQ_LENGTH "+"SYSREQ_REF_REQ_PRECISION "+"SYSREQ_REF_REQ_REFERENCE "+"SYSREQ_REF_REQ_SECTION "+"SYSREQ_REF_REQ_STORED "+"SYSREQ_REF_REQ_TOKENS "+"SYSREQ_REF_REQ_TYPE "+"SYSREQ_REF_REQ_VIEW "+"SYSREQ_REF_TYPE_ACT_CODE "+"SYSREQ_REF_TYPE_ACT_DESCRIPTION "+"SYSREQ_REF_TYPE_ACT_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_REF_TYPE_ACT_ON_EXECUTE "+"SYSREQ_REF_TYPE_ACT_ON_EXECUTE_EXISTS "+"SYSREQ_REF_TYPE_ACT_SECTION "+"SYSREQ_REF_TYPE_ADD_PARAMS "+"SYSREQ_REF_TYPE_COMMENT "+"SYSREQ_REF_TYPE_COMMON_SETTINGS "+"SYSREQ_REF_TYPE_DISPLAY_REQUISITE_NAME "+"SYSREQ_REF_TYPE_EVENT_TEXT "+"SYSREQ_REF_TYPE_MAIN_LEADING_REF "+"SYSREQ_REF_TYPE_NAME_IN_SINGULAR "+"SYSREQ_REF_TYPE_NAME_IN_SINGULAR_LOCALIZE_ID "+"SYSREQ_REF_TYPE_NAME_LOCALIZE_ID "+"SYSREQ_REF_TYPE_NUMERATION_METHOD "+"SYSREQ_REF_TYPE_REQ_CODE "+"SYSREQ_REF_TYPE_REQ_DESCRIPTION "+"SYSREQ_REF_TYPE_REQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_REF_TYPE_REQ_IS_CONTROL "+"SYSREQ_REF_TYPE_REQ_IS_FILTER "+"SYSREQ_REF_TYPE_REQ_IS_LEADING "+"SYSREQ_REF_TYPE_REQ_IS_REQUIRED "+"SYSREQ_REF_TYPE_REQ_NUMBER "+"SYSREQ_REF_TYPE_REQ_ON_CHANGE "+"SYSREQ_REF_TYPE_REQ_ON_CHANGE_EXISTS "+"SYSREQ_REF_TYPE_REQ_ON_SELECT "+"SYSREQ_REF_TYPE_REQ_ON_SELECT_KIND "+"SYSREQ_REF_TYPE_REQ_SECTION "+"SYSREQ_REF_TYPE_VIEW_CARD "+"SYSREQ_REF_TYPE_VIEW_CODE "+"SYSREQ_REF_TYPE_VIEW_COMMENT "+"SYSREQ_REF_TYPE_VIEW_IS_MAIN "+"SYSREQ_REF_TYPE_VIEW_NAME "+"SYSREQ_REF_TYPE_VIEW_NAME_LOCALIZE_ID "+"SYSREQ_REFERENCE_TYPE_ID "+"SYSREQ_STATE "+"SYSREQ_STATЕ "+"SYSREQ_SYSTEM_SETTINGS_VALUE "+"SYSREQ_TYPE "+"SYSREQ_UNIT "+"SYSREQ_UNIT_ID "+"SYSREQ_USER_GROUPS_GROUP_FULL_NAME "+"SYSREQ_USER_GROUPS_GROUP_NAME "+"SYSREQ_USER_GROUPS_GROUP_SERVER_NAME "+"SYSREQ_USERS_ACCESS_RIGHTS "+"SYSREQ_USERS_AUTHENTICATION "+"SYSREQ_USERS_CATEGORY "+"SYSREQ_USERS_COMPONENT "+"SYSREQ_USERS_COMPONENT_USER_IS_PUBLIC "+"SYSREQ_USERS_DOMAIN "+"SYSREQ_USERS_FULL_USER_NAME "+"SYSREQ_USERS_GROUP "+"SYSREQ_USERS_IS_MAIN_SERVER "+"SYSREQ_USERS_LOGIN "+"SYSREQ_USERS_REFERENCE_USER_IS_PUBLIC "+"SYSREQ_USERS_STATUS "+"SYSREQ_USERS_USER_CERTIFICATE "+"SYSREQ_USERS_USER_CERTIFICATE_INFO "+"SYSREQ_USERS_USER_CERTIFICATE_PLUGIN_NAME "+"SYSREQ_USERS_USER_CERTIFICATE_PLUGIN_VERSION "+"SYSREQ_USERS_USER_CERTIFICATE_STATE "+"SYSREQ_USERS_USER_CERTIFICATE_SUBJECT_NAME "+"SYSREQ_USERS_USER_CERTIFICATE_THUMBPRINT "+"SYSREQ_USERS_USER_DEFAULT_CERTIFICATE "+"SYSREQ_USERS_USER_DESCRIPTION "+"SYSREQ_USERS_USER_GLOBAL_NAME "+"SYSREQ_USERS_USER_LOGIN "+"SYSREQ_USERS_USER_MAIN_SERVER "+"SYSREQ_USERS_USER_TYPE "+"SYSREQ_WORK_RULES_FOLDER_ID ";var result_constants="RESULT_VAR_NAME RESULT_VAR_NAME_ENG ";var rule_identification_constants="AUTO_NUMERATION_RULE_ID "+"CANT_CHANGE_ID_REQUISITE_RULE_ID "+"CANT_CHANGE_OURFIRM_REQUISITE_RULE_ID "+"CHECK_CHANGING_REFERENCE_RECORD_USE_RULE_ID "+"CHECK_CODE_REQUISITE_RULE_ID "+"CHECK_DELETING_REFERENCE_RECORD_USE_RULE_ID "+"CHECK_FILTRATER_CHANGES_RULE_ID "+"CHECK_RECORD_INTERVAL_RULE_ID "+"CHECK_REFERENCE_INTERVAL_RULE_ID "+"CHECK_REQUIRED_DATA_FULLNESS_RULE_ID "+"CHECK_REQUIRED_REQUISITES_FULLNESS_RULE_ID "+"MAKE_RECORD_UNRATIFIED_RULE_ID "+"RESTORE_AUTO_NUMERATION_RULE_ID "+"SET_FIRM_CONTEXT_FROM_RECORD_RULE_ID "+"SET_FIRST_RECORD_IN_LIST_FORM_RULE_ID "+"SET_IDSPS_VALUE_RULE_ID "+"SET_NEXT_CODE_VALUE_RULE_ID "+"SET_OURFIRM_BOUNDS_RULE_ID "+"SET_OURFIRM_REQUISITE_RULE_ID ";var script_block_properties_constants="SCRIPT_BLOCK_AFTER_FINISH_EVENT "+"SCRIPT_BLOCK_BEFORE_START_EVENT "+"SCRIPT_BLOCK_EXECUTION_RESULTS_PROPERTY "+"SCRIPT_BLOCK_NAME_PROPERTY "+"SCRIPT_BLOCK_SCRIPT_PROPERTY ";var subtask_block_properties_constants="SUBTASK_BLOCK_ABORT_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_AFTER_FINISH_EVENT "+"SUBTASK_BLOCK_ASSIGN_PARAMS_EVENT "+"SUBTASK_BLOCK_ATTACHMENTS_PROPERTY "+"SUBTASK_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"SUBTASK_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"SUBTASK_BLOCK_BEFORE_START_EVENT "+"SUBTASK_BLOCK_CREATED_TASK_PROPERTY "+"SUBTASK_BLOCK_CREATION_EVENT "+"SUBTASK_BLOCK_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_IMPORTANCE_PROPERTY "+"SUBTASK_BLOCK_INITIATOR_PROPERTY "+"SUBTASK_BLOCK_IS_RELATIVE_ABORT_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_JOBS_TYPE_PROPERTY "+"SUBTASK_BLOCK_NAME_PROPERTY "+"SUBTASK_BLOCK_PARALLEL_ROUTE_PROPERTY "+"SUBTASK_BLOCK_PERFORMERS_PROPERTY "+"SUBTASK_BLOCK_RELATIVE_ABORT_DEADLINE_TYPE_PROPERTY "+"SUBTASK_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"SUBTASK_BLOCK_REQUIRE_SIGN_PROPERTY "+"SUBTASK_BLOCK_STANDARD_ROUTE_PROPERTY "+"SUBTASK_BLOCK_START_EVENT "+"SUBTASK_BLOCK_STEP_CONTROL_PROPERTY "+"SUBTASK_BLOCK_SUBJECT_PROPERTY "+"SUBTASK_BLOCK_TASK_CONTROL_PROPERTY "+"SUBTASK_BLOCK_TEXT_PROPERTY "+"SUBTASK_BLOCK_UNLOCK_ATTACHMENTS_ON_STOP_PROPERTY "+"SUBTASK_BLOCK_USE_STANDARD_ROUTE_PROPERTY "+"SUBTASK_BLOCK_WAIT_FOR_TASK_COMPLETE_PROPERTY ";var system_component_constants="SYSCOMP_CONTROL_JOBS "+"SYSCOMP_FOLDERS "+"SYSCOMP_JOBS "+"SYSCOMP_NOTICES "+"SYSCOMP_TASKS ";var system_dialogs_constants="SYSDLG_CREATE_EDOCUMENT "+"SYSDLG_CREATE_EDOCUMENT_VERSION "+"SYSDLG_CURRENT_PERIOD "+"SYSDLG_EDIT_FUNCTION_HELP "+"SYSDLG_EDOCUMENT_KINDS_FOR_TEMPLATE "+"SYSDLG_EXPORT_MULTIPLE_EDOCUMENTS "+"SYSDLG_EXPORT_SINGLE_EDOCUMENT "+"SYSDLG_IMPORT_EDOCUMENT "+"SYSDLG_MULTIPLE_SELECT "+"SYSDLG_SETUP_ACCESS_RIGHTS "+"SYSDLG_SETUP_DEFAULT_RIGHTS "+"SYSDLG_SETUP_FILTER_CONDITION "+"SYSDLG_SETUP_SIGN_RIGHTS "+"SYSDLG_SETUP_TASK_OBSERVERS "+"SYSDLG_SETUP_TASK_ROUTE "+"SYSDLG_SETUP_USERS_LIST "+"SYSDLG_SIGN_EDOCUMENT "+"SYSDLG_SIGN_MULTIPLE_EDOCUMENTS ";var system_reference_names_constants="SYSREF_ACCESS_RIGHTS_TYPES "+"SYSREF_ADMINISTRATION_HISTORY "+"SYSREF_ALL_AVAILABLE_COMPONENTS "+"SYSREF_ALL_AVAILABLE_PRIVILEGES "+"SYSREF_ALL_REPLICATING_COMPONENTS "+"SYSREF_AVAILABLE_DEVELOPERS_COMPONENTS "+"SYSREF_CALENDAR_EVENTS "+"SYSREF_COMPONENT_TOKEN_HISTORY "+"SYSREF_COMPONENT_TOKENS "+"SYSREF_COMPONENTS "+"SYSREF_CONSTANTS "+"SYSREF_DATA_RECEIVE_PROTOCOL "+"SYSREF_DATA_SEND_PROTOCOL "+"SYSREF_DIALOGS "+"SYSREF_DIALOGS_REQUISITES "+"SYSREF_EDITORS "+"SYSREF_EDOC_CARDS "+"SYSREF_EDOC_TYPES "+"SYSREF_EDOCUMENT_CARD_REQUISITES "+"SYSREF_EDOCUMENT_CARD_TYPES "+"SYSREF_EDOCUMENT_CARD_TYPES_REFERENCE "+"SYSREF_EDOCUMENT_CARDS "+"SYSREF_EDOCUMENT_HISTORY "+"SYSREF_EDOCUMENT_KINDS "+"SYSREF_EDOCUMENT_REQUISITES "+"SYSREF_EDOCUMENT_SIGNATURES "+"SYSREF_EDOCUMENT_TEMPLATES "+"SYSREF_EDOCUMENT_TEXT_STORAGES "+"SYSREF_EDOCUMENT_VIEWS "+"SYSREF_FILTERER_SETUP_CONFLICTS "+"SYSREF_FILTRATER_SETTING_CONFLICTS "+"SYSREF_FOLDER_HISTORY "+"SYSREF_FOLDERS "+"SYSREF_FUNCTION_GROUPS "+"SYSREF_FUNCTION_PARAMS "+"SYSREF_FUNCTIONS "+"SYSREF_JOB_HISTORY "+"SYSREF_LINKS "+"SYSREF_LOCALIZATION_DICTIONARY "+"SYSREF_LOCALIZATION_LANGUAGES "+"SYSREF_MODULES "+"SYSREF_PRIVILEGES "+"SYSREF_RECORD_HISTORY "+"SYSREF_REFERENCE_REQUISITES "+"SYSREF_REFERENCE_TYPE_VIEWS "+"SYSREF_REFERENCE_TYPES "+"SYSREF_REFERENCES "+"SYSREF_REFERENCES_REQUISITES "+"SYSREF_REMOTE_SERVERS "+"SYSREF_REPLICATION_SESSIONS_LOG "+"SYSREF_REPLICATION_SESSIONS_PROTOCOL "+"SYSREF_REPORTS "+"SYSREF_ROLES "+"SYSREF_ROUTE_BLOCK_GROUPS "+"SYSREF_ROUTE_BLOCKS "+"SYSREF_SCRIPTS "+"SYSREF_SEARCHES "+"SYSREF_SERVER_EVENTS "+"SYSREF_SERVER_EVENTS_HISTORY "+"SYSREF_STANDARD_ROUTE_GROUPS "+"SYSREF_STANDARD_ROUTES "+"SYSREF_STATUSES "+"SYSREF_SYSTEM_SETTINGS "+"SYSREF_TASK_HISTORY "+"SYSREF_TASK_KIND_GROUPS "+"SYSREF_TASK_KINDS "+"SYSREF_TASK_RIGHTS "+"SYSREF_TASK_SIGNATURES "+"SYSREF_TASKS "+"SYSREF_UNITS "+"SYSREF_USER_GROUPS "+"SYSREF_USER_GROUPS_REFERENCE "+"SYSREF_USER_SUBSTITUTION "+"SYSREF_USERS "+"SYSREF_USERS_REFERENCE "+"SYSREF_VIEWERS "+"SYSREF_WORKING_TIME_CALENDARS ";var table_name_constants="ACCESS_RIGHTS_TABLE_NAME "+"EDMS_ACCESS_TABLE_NAME "+"EDOC_TYPES_TABLE_NAME ";var test_constants="TEST_DEV_DB_NAME "+"TEST_DEV_SYSTEM_CODE "+"TEST_EDMS_DB_NAME "+"TEST_EDMS_MAIN_CODE "+"TEST_EDMS_MAIN_DB_NAME "+"TEST_EDMS_SECOND_CODE "+"TEST_EDMS_SECOND_DB_NAME "+"TEST_EDMS_SYSTEM_CODE "+"TEST_ISB5_MAIN_CODE "+"TEST_ISB5_SECOND_CODE "+"TEST_SQL_SERVER_2005_NAME "+"TEST_SQL_SERVER_NAME ";var using_the_dialog_windows_constants="ATTENTION_CAPTION "+"cbsCommandLinks "+"cbsDefault "+"CONFIRMATION_CAPTION "+"ERROR_CAPTION "+"INFORMATION_CAPTION "+"mrCancel "+"mrOk ";var using_the_document_constants="EDOC_VERSION_ACTIVE_STAGE_CODE "+"EDOC_VERSION_DESIGN_STAGE_CODE "+"EDOC_VERSION_OBSOLETE_STAGE_CODE ";var using_the_EA_and_encryption_constants="cpDataEnciphermentEnabled "+"cpDigitalSignatureEnabled "+"cpID "+"cpIssuer "+"cpPluginVersion "+"cpSerial "+"cpSubjectName "+"cpSubjSimpleName "+"cpValidFromDate "+"cpValidToDate ";var using_the_ISBL_editor_constants="ISBL_SYNTAX "+"NO_SYNTAX "+"XML_SYNTAX ";var wait_block_properties_constants="WAIT_BLOCK_AFTER_FINISH_EVENT "+"WAIT_BLOCK_BEFORE_START_EVENT "+"WAIT_BLOCK_DEADLINE_PROPERTY "+"WAIT_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"WAIT_BLOCK_NAME_PROPERTY "+"WAIT_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY ";var sysres_common_constants="SYSRES_COMMON "+"SYSRES_CONST "+"SYSRES_MBFUNC "+"SYSRES_SBDATA "+"SYSRES_SBGUI "+"SYSRES_SBINTF "+"SYSRES_SBREFDSC "+"SYSRES_SQLERRORS "+"SYSRES_SYSCOMP ";var CONSTANTS=sysres_constants+base_constants+base_group_name_constants+decision_block_properties_constants+file_extension_constants+job_block_properties_constants+language_code_constants+launching_external_applications_constants+link_kind_constants+lock_type_constants+monitor_block_properties_constants+notice_block_properties_constants+object_events_constants+object_params_constants+other_constants+privileges_constants+pseudoreference_code_constants+requisite_ISBCertificateType_values_constants+requisite_ISBEDocStorageType_values_constants+requisite_compType2_values_constants+requisite_name_constants+result_constants+rule_identification_constants+script_block_properties_constants+subtask_block_properties_constants+system_component_constants+system_dialogs_constants+system_reference_names_constants+table_name_constants+test_constants+using_the_dialog_windows_constants+using_the_document_constants+using_the_EA_and_encryption_constants+using_the_ISBL_editor_constants+wait_block_properties_constants+sysres_common_constants;var TAccountType="atUser atGroup atRole ";var TActionEnabledMode="aemEnabledAlways "+"aemDisabledAlways "+"aemEnabledOnBrowse "+"aemEnabledOnEdit "+"aemDisabledOnBrowseEmpty ";var TAddPosition="apBegin apEnd ";var TAlignment="alLeft alRight ";var TAreaShowMode="asmNever "+"asmNoButCustomize "+"asmAsLastTime "+"asmYesButCustomize "+"asmAlways ";var TCertificateInvalidationReason="cirCommon cirRevoked ";var TCertificateType="ctSignature ctEncode ctSignatureEncode ";var TCheckListBoxItemState="clbUnchecked clbChecked clbGrayed ";var TCloseOnEsc="ceISB ceAlways ceNever ";var TCompType="ctDocument "+"ctReference "+"ctScript "+"ctUnknown "+"ctReport "+"ctDialog "+"ctFunction "+"ctFolder "+"ctEDocument "+"ctTask "+"ctJob "+"ctNotice "+"ctControlJob ";var TConditionFormat="cfInternal cfDisplay ";var TConnectionIntent="ciUnspecified ciWrite ciRead ";var TContentKind="ckFolder "+"ckEDocument "+"ckTask "+"ckJob "+"ckComponentToken "+"ckAny "+"ckReference "+"ckScript "+"ckReport "+"ckDialog ";var TControlType="ctISBLEditor "+"ctBevel "+"ctButton "+"ctCheckListBox "+"ctComboBox "+"ctComboEdit "+"ctGrid "+"ctDBCheckBox "+"ctDBComboBox "+"ctDBEdit "+"ctDBEllipsis "+"ctDBMemo "+"ctDBNavigator "+"ctDBRadioGroup "+"ctDBStatusLabel "+"ctEdit "+"ctGroupBox "+"ctInplaceHint "+"ctMemo "+"ctPanel "+"ctListBox "+"ctRadioButton "+"ctRichEdit "+"ctTabSheet "+"ctWebBrowser "+"ctImage "+"ctHyperLink "+"ctLabel "+"ctDBMultiEllipsis "+"ctRibbon "+"ctRichView "+"ctInnerPanel "+"ctPanelGroup "+"ctBitButton ";var TCriterionContentType="cctDate "+"cctInteger "+"cctNumeric "+"cctPick "+"cctReference "+"cctString "+"cctText ";var TCultureType="cltInternal cltPrimary cltGUI ";var TDataSetEventType="dseBeforeOpen "+"dseAfterOpen "+"dseBeforeClose "+"dseAfterClose "+"dseOnValidDelete "+"dseBeforeDelete "+"dseAfterDelete "+"dseAfterDeleteOutOfTransaction "+"dseOnDeleteError "+"dseBeforeInsert "+"dseAfterInsert "+"dseOnValidUpdate "+"dseBeforeUpdate "+"dseOnUpdateRatifiedRecord "+"dseAfterUpdate "+"dseAfterUpdateOutOfTransaction "+"dseOnUpdateError "+"dseAfterScroll "+"dseOnOpenRecord "+"dseOnCloseRecord "+"dseBeforeCancel "+"dseAfterCancel "+"dseOnUpdateDeadlockError "+"dseBeforeDetailUpdate "+"dseOnPrepareUpdate "+"dseOnAnyRequisiteChange ";var TDataSetState="dssEdit dssInsert dssBrowse dssInActive ";var TDateFormatType="dftDate dftShortDate dftDateTime dftTimeStamp ";var TDateOffsetType="dotDays dotHours dotMinutes dotSeconds ";var TDateTimeKind="dtkndLocal dtkndUTC ";var TDeaAccessRights="arNone arView arEdit arFull ";var TDocumentDefaultAction="ddaView ddaEdit ";var TEditMode="emLock "+"emEdit "+"emSign "+"emExportWithLock "+"emImportWithUnlock "+"emChangeVersionNote "+"emOpenForModify "+"emChangeLifeStage "+"emDelete "+"emCreateVersion "+"emImport "+"emUnlockExportedWithLock "+"emStart "+"emAbort "+"emReInit "+"emMarkAsReaded "+"emMarkAsUnreaded "+"emPerform "+"emAccept "+"emResume "+"emChangeRights "+"emEditRoute "+"emEditObserver "+"emRecoveryFromLocalCopy "+"emChangeWorkAccessType "+"emChangeEncodeTypeToCertificate "+"emChangeEncodeTypeToPassword "+"emChangeEncodeTypeToNone "+"emChangeEncodeTypeToCertificatePassword "+"emChangeStandardRoute "+"emGetText "+"emOpenForView "+"emMoveToStorage "+"emCreateObject "+"emChangeVersionHidden "+"emDeleteVersion "+"emChangeLifeCycleStage "+"emApprovingSign "+"emExport "+"emContinue "+"emLockFromEdit "+"emUnLockForEdit "+"emLockForServer "+"emUnlockFromServer "+"emDelegateAccessRights "+"emReEncode ";var TEditorCloseObservType="ecotFile ecotProcess ";var TEdmsApplicationAction="eaGet eaCopy eaCreate eaCreateStandardRoute ";var TEDocumentLockType="edltAll edltNothing edltQuery ";var TEDocumentStepShowMode="essmText essmCard ";var TEDocumentStepVersionType="esvtLast esvtLastActive esvtSpecified ";var TEDocumentStorageFunction="edsfExecutive edsfArchive ";var TEDocumentStorageType="edstSQLServer edstFile ";var TEDocumentVersionSourceType="edvstNone edvstEDocumentVersionCopy edvstFile edvstTemplate edvstScannedFile ";var TEDocumentVersionState="vsDefault vsDesign vsActive vsObsolete ";var TEncodeType="etNone etCertificate etPassword etCertificatePassword ";var TExceptionCategory="ecException ecWarning ecInformation ";var TExportedSignaturesType="estAll estApprovingOnly ";var TExportedVersionType="evtLast evtLastActive evtQuery ";var TFieldDataType="fdtString "+"fdtNumeric "+"fdtInteger "+"fdtDate "+"fdtText "+"fdtUnknown "+"fdtWideString "+"fdtLargeInteger ";var TFolderType="ftInbox "+"ftOutbox "+"ftFavorites "+"ftCommonFolder "+"ftUserFolder "+"ftComponents "+"ftQuickLaunch "+"ftShortcuts "+"ftSearch ";var TGridRowHeight="grhAuto "+"grhX1 "+"grhX2 "+"grhX3 ";var THyperlinkType="hltText "+"hltRTF "+"hltHTML ";var TImageFileFormat="iffBMP "+"iffJPEG "+"iffMultiPageTIFF "+"iffSinglePageTIFF "+"iffTIFF "+"iffPNG ";var TImageMode="im8bGrayscale "+"im24bRGB "+"im1bMonochrome ";var TImageType="itBMP "+"itJPEG "+"itWMF "+"itPNG ";var TInplaceHintKind="ikhInformation "+"ikhWarning "+"ikhError "+"ikhNoIcon ";var TISBLContext="icUnknown "+"icScript "+"icFunction "+"icIntegratedReport "+"icAnalyticReport "+"icDataSetEventHandler "+"icActionHandler "+"icFormEventHandler "+"icLookUpEventHandler "+"icRequisiteChangeEventHandler "+"icBeforeSearchEventHandler "+"icRoleCalculation "+"icSelectRouteEventHandler "+"icBlockPropertyCalculation "+"icBlockQueryParamsEventHandler "+"icChangeSearchResultEventHandler "+"icBlockEventHandler "+"icSubTaskInitEventHandler "+"icEDocDataSetEventHandler "+"icEDocLookUpEventHandler "+"icEDocActionHandler "+"icEDocFormEventHandler "+"icEDocRequisiteChangeEventHandler "+"icStructuredConversionRule "+"icStructuredConversionEventBefore "+"icStructuredConversionEventAfter "+"icWizardEventHandler "+"icWizardFinishEventHandler "+"icWizardStepEventHandler "+"icWizardStepFinishEventHandler "+"icWizardActionEnableEventHandler "+"icWizardActionExecuteEventHandler "+"icCreateJobsHandler "+"icCreateNoticesHandler "+"icBeforeLookUpEventHandler "+"icAfterLookUpEventHandler "+"icTaskAbortEventHandler "+"icWorkflowBlockActionHandler "+"icDialogDataSetEventHandler "+"icDialogActionHandler "+"icDialogLookUpEventHandler "+"icDialogRequisiteChangeEventHandler "+"icDialogFormEventHandler "+"icDialogValidCloseEventHandler "+"icBlockFormEventHandler "+"icTaskFormEventHandler "+"icReferenceMethod "+"icEDocMethod "+"icDialogMethod "+"icProcessMessageHandler ";var TItemShow="isShow "+"isHide "+"isByUserSettings ";var TJobKind="jkJob "+"jkNotice "+"jkControlJob ";var TJoinType="jtInner "+"jtLeft "+"jtRight "+"jtFull "+"jtCross ";var TLabelPos="lbpAbove "+"lbpBelow "+"lbpLeft "+"lbpRight ";var TLicensingType="eltPerConnection "+"eltPerUser ";var TLifeCycleStageFontColor="sfcUndefined "+"sfcBlack "+"sfcGreen "+"sfcRed "+"sfcBlue "+"sfcOrange "+"sfcLilac ";var TLifeCycleStageFontStyle="sfsItalic "+"sfsStrikeout "+"sfsNormal ";var TLockableDevelopmentComponentType="ldctStandardRoute "+"ldctWizard "+"ldctScript "+"ldctFunction "+"ldctRouteBlock "+"ldctIntegratedReport "+"ldctAnalyticReport "+"ldctReferenceType "+"ldctEDocumentType "+"ldctDialog "+"ldctServerEvents ";var TMaxRecordCountRestrictionType="mrcrtNone "+"mrcrtUser "+"mrcrtMaximal "+"mrcrtCustom ";var TRangeValueType="vtEqual "+"vtGreaterOrEqual "+"vtLessOrEqual "+"vtRange ";var TRelativeDate="rdYesterday "+"rdToday "+"rdTomorrow "+"rdThisWeek "+"rdThisMonth "+"rdThisYear "+"rdNextMonth "+"rdNextWeek "+"rdLastWeek "+"rdLastMonth ";var TReportDestination="rdWindow "+"rdFile "+"rdPrinter ";var TReqDataType="rdtString "+"rdtNumeric "+"rdtInteger "+"rdtDate "+"rdtReference "+"rdtAccount "+"rdtText "+"rdtPick "+"rdtUnknown "+"rdtLargeInteger "+"rdtDocument ";var TRequisiteEventType="reOnChange "+"reOnChangeValues ";var TSBTimeType="ttGlobal "+"ttLocal "+"ttUser "+"ttSystem ";var TSearchShowMode="ssmBrowse "+"ssmSelect "+"ssmMultiSelect "+"ssmBrowseModal ";var TSelectMode="smSelect "+"smLike "+"smCard ";var TSignatureType="stNone "+"stAuthenticating "+"stApproving ";var TSignerContentType="sctString "+"sctStream ";var TStringsSortType="sstAnsiSort "+"sstNaturalSort ";var TStringValueType="svtEqual "+"svtContain ";var TStructuredObjectAttributeType="soatString "+"soatNumeric "+"soatInteger "+"soatDatetime "+"soatReferenceRecord "+"soatText "+"soatPick "+"soatBoolean "+"soatEDocument "+"soatAccount "+"soatIntegerCollection "+"soatNumericCollection "+"soatStringCollection "+"soatPickCollection "+"soatDatetimeCollection "+"soatBooleanCollection "+"soatReferenceRecordCollection "+"soatEDocumentCollection "+"soatAccountCollection "+"soatContents "+"soatUnknown ";var TTaskAbortReason="tarAbortByUser "+"tarAbortByWorkflowException ";var TTextValueType="tvtAllWords "+"tvtExactPhrase "+"tvtAnyWord ";var TUserObjectStatus="usNone "+"usCompleted "+"usRedSquare "+"usBlueSquare "+"usYellowSquare "+"usGreenSquare "+"usOrangeSquare "+"usPurpleSquare "+"usFollowUp ";var TUserType="utUnknown "+"utUser "+"utDeveloper "+"utAdministrator "+"utSystemDeveloper "+"utDisconnected ";var TValuesBuildType="btAnd "+"btDetailAnd "+"btOr "+"btNotOr "+"btOnly ";var TViewMode="vmView "+"vmSelect "+"vmNavigation ";var TViewSelectionMode="vsmSingle "+"vsmMultiple "+"vsmMultipleCheck "+"vsmNoSelection ";var TWizardActionType="wfatPrevious "+"wfatNext "+"wfatCancel "+"wfatFinish ";var TWizardFormElementProperty="wfepUndefined "+"wfepText3 "+"wfepText6 "+"wfepText9 "+"wfepSpinEdit "+"wfepDropDown "+"wfepRadioGroup "+"wfepFlag "+"wfepText12 "+"wfepText15 "+"wfepText18 "+"wfepText21 "+"wfepText24 "+"wfepText27 "+"wfepText30 "+"wfepRadioGroupColumn1 "+"wfepRadioGroupColumn2 "+"wfepRadioGroupColumn3 ";var TWizardFormElementType="wfetQueryParameter "+"wfetText "+"wfetDelimiter "+"wfetLabel ";var TWizardParamType="wptString "+"wptInteger "+"wptNumeric "+"wptBoolean "+"wptDateTime "+"wptPick "+"wptText "+"wptUser "+"wptUserList "+"wptEDocumentInfo "+"wptEDocumentInfoList "+"wptReferenceRecordInfo "+"wptReferenceRecordInfoList "+"wptFolderInfo "+"wptTaskInfo "+"wptContents "+"wptFileName "+"wptDate ";var TWizardStepResult="wsrComplete "+"wsrGoNext "+"wsrGoPrevious "+"wsrCustom "+"wsrCancel "+"wsrGoFinal ";var TWizardStepType="wstForm "+"wstEDocument "+"wstTaskCard "+"wstReferenceRecordCard "+"wstFinal ";var TWorkAccessType="waAll "+"waPerformers "+"waManual ";var TWorkflowBlockType="wsbStart "+"wsbFinish "+"wsbNotice "+"wsbStep "+"wsbDecision "+"wsbWait "+"wsbMonitor "+"wsbScript "+"wsbConnector "+"wsbSubTask "+"wsbLifeCycleStage "+"wsbPause ";var TWorkflowDataType="wdtInteger "+"wdtFloat "+"wdtString "+"wdtPick "+"wdtDateTime "+"wdtBoolean "+"wdtTask "+"wdtJob "+"wdtFolder "+"wdtEDocument "+"wdtReferenceRecord "+"wdtUser "+"wdtGroup "+"wdtRole "+"wdtIntegerCollection "+"wdtFloatCollection "+"wdtStringCollection "+"wdtPickCollection "+"wdtDateTimeCollection "+"wdtBooleanCollection "+"wdtTaskCollection "+"wdtJobCollection "+"wdtFolderCollection "+"wdtEDocumentCollection "+"wdtReferenceRecordCollection "+"wdtUserCollection "+"wdtGroupCollection "+"wdtRoleCollection "+"wdtContents "+"wdtUserList "+"wdtSearchDescription "+"wdtDeadLine "+"wdtPickSet "+"wdtAccountCollection ";var TWorkImportance="wiLow "+"wiNormal "+"wiHigh ";var TWorkRouteType="wrtSoft "+"wrtHard ";var TWorkState="wsInit "+"wsRunning "+"wsDone "+"wsControlled "+"wsAborted "+"wsContinued ";var TWorkTextBuildingMode="wtmFull "+"wtmFromCurrent "+"wtmOnlyCurrent ";var ENUMS=TAccountType+TActionEnabledMode+TAddPosition+TAlignment+TAreaShowMode+TCertificateInvalidationReason+TCertificateType+TCheckListBoxItemState+TCloseOnEsc+TCompType+TConditionFormat+TConnectionIntent+TContentKind+TControlType+TCriterionContentType+TCultureType+TDataSetEventType+TDataSetState+TDateFormatType+TDateOffsetType+TDateTimeKind+TDeaAccessRights+TDocumentDefaultAction+TEditMode+TEditorCloseObservType+TEdmsApplicationAction+TEDocumentLockType+TEDocumentStepShowMode+TEDocumentStepVersionType+TEDocumentStorageFunction+TEDocumentStorageType+TEDocumentVersionSourceType+TEDocumentVersionState+TEncodeType+TExceptionCategory+TExportedSignaturesType+TExportedVersionType+TFieldDataType+TFolderType+TGridRowHeight+THyperlinkType+TImageFileFormat+TImageMode+TImageType+TInplaceHintKind+TISBLContext+TItemShow+TJobKind+TJoinType+TLabelPos+TLicensingType+TLifeCycleStageFontColor+TLifeCycleStageFontStyle+TLockableDevelopmentComponentType+TMaxRecordCountRestrictionType+TRangeValueType+TRelativeDate+TReportDestination+TReqDataType+TRequisiteEventType+TSBTimeType+TSearchShowMode+TSelectMode+TSignatureType+TSignerContentType+TStringsSortType+TStringValueType+TStructuredObjectAttributeType+TTaskAbortReason+TTextValueType+TUserObjectStatus+TUserType+TValuesBuildType+TViewMode+TViewSelectionMode+TWizardActionType+TWizardFormElementProperty+TWizardFormElementType+TWizardParamType+TWizardStepResult+TWizardStepType+TWorkAccessType+TWorkflowBlockType+TWorkflowDataType+TWorkImportance+TWorkRouteType+TWorkState+TWorkTextBuildingMode;var system_functions="AddSubString "+"AdjustLineBreaks "+"AmountInWords "+"Analysis "+"ArrayDimCount "+"ArrayHighBound "+"ArrayLowBound "+"ArrayOf "+"ArrayReDim "+"Assert "+"Assigned "+"BeginOfMonth "+"BeginOfPeriod "+"BuildProfilingOperationAnalysis "+"CallProcedure "+"CanReadFile "+"CArrayElement "+"CDataSetRequisite "+"ChangeDate "+"ChangeReferenceDataset "+"Char "+"CharPos "+"CheckParam "+"CheckParamValue "+"CompareStrings "+"ConstantExists "+"ControlState "+"ConvertDateStr "+"Copy "+"CopyFile "+"CreateArray "+"CreateCachedReference "+"CreateConnection "+"CreateDialog "+"CreateDualListDialog "+"CreateEditor "+"CreateException "+"CreateFile "+"CreateFolderDialog "+"CreateInputDialog "+"CreateLinkFile "+"CreateList "+"CreateLock "+"CreateMemoryDataSet "+"CreateObject "+"CreateOpenDialog "+"CreateProgress "+"CreateQuery "+"CreateReference "+"CreateReport "+"CreateSaveDialog "+"CreateScript "+"CreateSQLPivotFunction "+"CreateStringList "+"CreateTreeListSelectDialog "+"CSelectSQL "+"CSQL "+"CSubString "+"CurrentUserID "+"CurrentUserName "+"CurrentVersion "+"DataSetLocateEx "+"DateDiff "+"DateTimeDiff "+"DateToStr "+"DayOfWeek "+"DeleteFile "+"DirectoryExists "+"DisableCheckAccessRights "+"DisableCheckFullShowingRestriction "+"DisableMassTaskSendingRestrictions "+"DropTable "+"DupeString "+"EditText "+"EnableCheckAccessRights "+"EnableCheckFullShowingRestriction "+"EnableMassTaskSendingRestrictions "+"EndOfMonth "+"EndOfPeriod "+"ExceptionExists "+"ExceptionsOff "+"ExceptionsOn "+"Execute "+"ExecuteProcess "+"Exit "+"ExpandEnvironmentVariables "+"ExtractFileDrive "+"ExtractFileExt "+"ExtractFileName "+"ExtractFilePath "+"ExtractParams "+"FileExists "+"FileSize "+"FindFile "+"FindSubString "+"FirmContext "+"ForceDirectories "+"Format "+"FormatDate "+"FormatNumeric "+"FormatSQLDate "+"FormatString "+"FreeException "+"GetComponent "+"GetComponentLaunchParam "+"GetConstant "+"GetLastException "+"GetReferenceRecord "+"GetRefTypeByRefID "+"GetTableID "+"GetTempFolder "+"IfThen "+"In "+"IndexOf "+"InputDialog "+"InputDialogEx "+"InteractiveMode "+"IsFileLocked "+"IsGraphicFile "+"IsNumeric "+"Length "+"LoadString "+"LoadStringFmt "+"LocalTimeToUTC "+"LowerCase "+"Max "+"MessageBox "+"MessageBoxEx "+"MimeDecodeBinary "+"MimeDecodeString "+"MimeEncodeBinary "+"MimeEncodeString "+"Min "+"MoneyInWords "+"MoveFile "+"NewID "+"Now "+"OpenFile "+"Ord "+"Precision "+"Raise "+"ReadCertificateFromFile "+"ReadFile "+"ReferenceCodeByID "+"ReferenceNumber "+"ReferenceRequisiteMode "+"ReferenceRequisiteValue "+"RegionDateSettings "+"RegionNumberSettings "+"RegionTimeSettings "+"RegRead "+"RegWrite "+"RenameFile "+"Replace "+"Round "+"SelectServerCode "+"SelectSQL "+"ServerDateTime "+"SetConstant "+"SetManagedFolderFieldsState "+"ShowConstantsInputDialog "+"ShowMessage "+"Sleep "+"Split "+"SQL "+"SQL2XLSTAB "+"SQLProfilingSendReport "+"StrToDate "+"SubString "+"SubStringCount "+"SystemSetting "+"Time "+"TimeDiff "+"Today "+"Transliterate "+"Trim "+"UpperCase "+"UserStatus "+"UTCToLocalTime "+"ValidateXML "+"VarIsClear "+"VarIsEmpty "+"VarIsNull "+"WorkTimeDiff "+"WriteFile "+"WriteFileEx "+"WriteObjectHistory "+"Анализ "+"БазаДанных "+"БлокЕсть "+"БлокЕстьРасш "+"БлокИнфо "+"БлокСнять "+"БлокСнятьРасш "+"БлокУстановить "+"Ввод "+"ВводМеню "+"ВедС "+"ВедСпр "+"ВерхняяГраницаМассива "+"ВнешПрогр "+"Восст "+"ВременнаяПапка "+"Время "+"ВыборSQL "+"ВыбратьЗапись "+"ВыделитьСтр "+"Вызвать "+"Выполнить "+"ВыпПрогр "+"ГрафическийФайл "+"ГруппаДополнительно "+"ДатаВремяСерв "+"ДеньНедели "+"ДиалогДаНет "+"ДлинаСтр "+"ДобПодстр "+"ЕПусто "+"ЕслиТо "+"ЕЧисло "+"ЗамПодстр "+"ЗаписьСправочника "+"ЗначПоляСпр "+"ИДТипСпр "+"ИзвлечьДиск "+"ИзвлечьИмяФайла "+"ИзвлечьПуть "+"ИзвлечьРасширение "+"ИзмДат "+"ИзменитьРазмерМассива "+"ИзмеренийМассива "+"ИмяОрг "+"ИмяПоляСпр "+"Индекс "+"ИндикаторЗакрыть "+"ИндикаторОткрыть "+"ИндикаторШаг "+"ИнтерактивныйРежим "+"ИтогТблСпр "+"КодВидВедСпр "+"КодВидСпрПоИД "+"КодПоAnalit "+"КодСимвола "+"КодСпр "+"КолПодстр "+"КолПроп "+"КонМес "+"Конст "+"КонстЕсть "+"КонстЗнач "+"КонТран "+"КопироватьФайл "+"КопияСтр "+"КПериод "+"КСтрТблСпр "+"Макс "+"МаксСтрТблСпр "+"Массив "+"Меню "+"МенюРасш "+"Мин "+"НаборДанныхНайтиРасш "+"НаимВидСпр "+"НаимПоAnalit "+"НаимСпр "+"НастроитьПереводыСтрок "+"НачМес "+"НачТран "+"НижняяГраницаМассива "+"НомерСпр "+"НПериод "+"Окно "+"Окр "+"Окружение "+"ОтлИнфДобавить "+"ОтлИнфУдалить "+"Отчет "+"ОтчетАнал "+"ОтчетИнт "+"ПапкаСуществует "+"Пауза "+"ПВыборSQL "+"ПереименоватьФайл "+"Переменные "+"ПереместитьФайл "+"Подстр "+"ПоискПодстр "+"ПоискСтр "+"ПолучитьИДТаблицы "+"ПользовательДополнительно "+"ПользовательИД "+"ПользовательИмя "+"ПользовательСтатус "+"Прервать "+"ПроверитьПараметр "+"ПроверитьПараметрЗнач "+"ПроверитьУсловие "+"РазбСтр "+"РазнВремя "+"РазнДат "+"РазнДатаВремя "+"РазнРабВремя "+"РегУстВрем "+"РегУстДат "+"РегУстЧсл "+"РедТекст "+"РеестрЗапись "+"РеестрСписокИменПарам "+"РеестрЧтение "+"РеквСпр "+"РеквСпрПр "+"Сегодня "+"Сейчас "+"Сервер "+"СерверПроцессИД "+"СертификатФайлСчитать "+"СжПроб "+"Символ "+"СистемаДиректумКод "+"СистемаИнформация "+"СистемаКод "+"Содержит "+"СоединениеЗакрыть "+"СоединениеОткрыть "+"СоздатьДиалог "+"СоздатьДиалогВыбораИзДвухСписков "+"СоздатьДиалогВыбораПапки "+"СоздатьДиалогОткрытияФайла "+"СоздатьДиалогСохраненияФайла "+"СоздатьЗапрос "+"СоздатьИндикатор "+"СоздатьИсключение "+"СоздатьКэшированныйСправочник "+"СоздатьМассив "+"СоздатьНаборДанных "+"СоздатьОбъект "+"СоздатьОтчет "+"СоздатьПапку "+"СоздатьРедактор "+"СоздатьСоединение "+"СоздатьСписок "+"СоздатьСписокСтрок "+"СоздатьСправочник "+"СоздатьСценарий "+"СоздСпр "+"СостСпр "+"Сохр "+"СохрСпр "+"СписокСистем "+"Спр "+"Справочник "+"СпрБлокЕсть "+"СпрБлокСнять "+"СпрБлокСнятьРасш "+"СпрБлокУстановить "+"СпрИзмНабДан "+"СпрКод "+"СпрНомер "+"СпрОбновить "+"СпрОткрыть "+"СпрОтменить "+"СпрПарам "+"СпрПолеЗнач "+"СпрПолеИмя "+"СпрРекв "+"СпрРеквВведЗн "+"СпрРеквНовые "+"СпрРеквПр "+"СпрРеквПредЗн "+"СпрРеквРежим "+"СпрРеквТипТекст "+"СпрСоздать "+"СпрСост "+"СпрСохранить "+"СпрТблИтог "+"СпрТблСтр "+"СпрТблСтрКол "+"СпрТблСтрМакс "+"СпрТблСтрМин "+"СпрТблСтрПред "+"СпрТблСтрСлед "+"СпрТблСтрСозд "+"СпрТблСтрУд "+"СпрТекПредст "+"СпрУдалить "+"СравнитьСтр "+"СтрВерхРегистр "+"СтрНижнРегистр "+"СтрТблСпр "+"СумПроп "+"Сценарий "+"СценарийПарам "+"ТекВерсия "+"ТекОрг "+"Точн "+"Тран "+"Транслитерация "+"УдалитьТаблицу "+"УдалитьФайл "+"УдСпр "+"УдСтрТблСпр "+"Уст "+"УстановкиКонстант "+"ФайлАтрибутСчитать "+"ФайлАтрибутУстановить "+"ФайлВремя "+"ФайлВремяУстановить "+"ФайлВыбрать "+"ФайлЗанят "+"ФайлЗаписать "+"ФайлИскать "+"ФайлКопировать "+"ФайлМожноЧитать "+"ФайлОткрыть "+"ФайлПереименовать "+"ФайлПерекодировать "+"ФайлПереместить "+"ФайлПросмотреть "+"ФайлРазмер "+"ФайлСоздать "+"ФайлСсылкаСоздать "+"ФайлСуществует "+"ФайлСчитать "+"ФайлУдалить "+"ФмтSQLДат "+"ФмтДат "+"ФмтСтр "+"ФмтЧсл "+"Формат "+"ЦМассивЭлемент "+"ЦНаборДанныхРеквизит "+"ЦПодстр ";var predefined_variables="AltState "+"Application "+"CallType "+"ComponentTokens "+"CreatedJobs "+"CreatedNotices "+"ControlState "+"DialogResult "+"Dialogs "+"EDocuments "+"EDocumentVersionSource "+"Folders "+"GlobalIDs "+"Job "+"Jobs "+"InputValue "+"LookUpReference "+"LookUpRequisiteNames "+"LookUpSearch "+"Object "+"ParentComponent "+"Processes "+"References "+"Requisite "+"ReportName "+"Reports "+"Result "+"Scripts "+"Searches "+"SelectedAttachments "+"SelectedItems "+"SelectMode "+"Sender "+"ServerEvents "+"ServiceFactory "+"ShiftState "+"SubTask "+"SystemDialogs "+"Tasks "+"Wizard "+"Wizards "+"Work "+"ВызовСпособ "+"ИмяОтчета "+"РеквЗнач ";var interfaces="IApplication "+"IAccessRights "+"IAccountRepository "+"IAccountSelectionRestrictions "+"IAction "+"IActionList "+"IAdministrationHistoryDescription "+"IAnchors "+"IApplication "+"IArchiveInfo "+"IAttachment "+"IAttachmentList "+"ICheckListBox "+"ICheckPointedList "+"IColumn "+"IComponent "+"IComponentDescription "+"IComponentToken "+"IComponentTokenFactory "+"IComponentTokenInfo "+"ICompRecordInfo "+"IConnection "+"IContents "+"IControl "+"IControlJob "+"IControlJobInfo "+"IControlList "+"ICrypto "+"ICrypto2 "+"ICustomJob "+"ICustomJobInfo "+"ICustomListBox "+"ICustomObjectWizardStep "+"ICustomWork "+"ICustomWorkInfo "+"IDataSet "+"IDataSetAccessInfo "+"IDataSigner "+"IDateCriterion "+"IDateRequisite "+"IDateRequisiteDescription "+"IDateValue "+"IDeaAccessRights "+"IDeaObjectInfo "+"IDevelopmentComponentLock "+"IDialog "+"IDialogFactory "+"IDialogPickRequisiteItems "+"IDialogsFactory "+"IDICSFactory "+"IDocRequisite "+"IDocumentInfo "+"IDualListDialog "+"IECertificate "+"IECertificateInfo "+"IECertificates "+"IEditControl "+"IEditorForm "+"IEdmsExplorer "+"IEdmsObject "+"IEdmsObjectDescription "+"IEdmsObjectFactory "+"IEdmsObjectInfo "+"IEDocument "+"IEDocumentAccessRights "+"IEDocumentDescription "+"IEDocumentEditor "+"IEDocumentFactory "+"IEDocumentInfo "+"IEDocumentStorage "+"IEDocumentVersion "+"IEDocumentVersionListDialog "+"IEDocumentVersionSource "+"IEDocumentWizardStep "+"IEDocVerSignature "+"IEDocVersionState "+"IEnabledMode "+"IEncodeProvider "+"IEncrypter "+"IEvent "+"IEventList "+"IException "+"IExternalEvents "+"IExternalHandler "+"IFactory "+"IField "+"IFileDialog "+"IFolder "+"IFolderDescription "+"IFolderDialog "+"IFolderFactory "+"IFolderInfo "+"IForEach "+"IForm "+"IFormTitle "+"IFormWizardStep "+"IGlobalIDFactory "+"IGlobalIDInfo "+"IGrid "+"IHasher "+"IHistoryDescription "+"IHyperLinkControl "+"IImageButton "+"IImageControl "+"IInnerPanel "+"IInplaceHint "+"IIntegerCriterion "+"IIntegerList "+"IIntegerRequisite "+"IIntegerValue "+"IISBLEditorForm "+"IJob "+"IJobDescription "+"IJobFactory "+"IJobForm "+"IJobInfo "+"ILabelControl "+"ILargeIntegerCriterion "+"ILargeIntegerRequisite "+"ILargeIntegerValue "+"ILicenseInfo "+"ILifeCycleStage "+"IList "+"IListBox "+"ILocalIDInfo "+"ILocalization "+"ILock "+"IMemoryDataSet "+"IMessagingFactory "+"IMetadataRepository "+"INotice "+"INoticeInfo "+"INumericCriterion "+"INumericRequisite "+"INumericValue "+"IObject "+"IObjectDescription "+"IObjectImporter "+"IObjectInfo "+"IObserver "+"IPanelGroup "+"IPickCriterion "+"IPickProperty "+"IPickRequisite "+"IPickRequisiteDescription "+"IPickRequisiteItem "+"IPickRequisiteItems "+"IPickValue "+"IPrivilege "+"IPrivilegeList "+"IProcess "+"IProcessFactory "+"IProcessMessage "+"IProgress "+"IProperty "+"IPropertyChangeEvent "+"IQuery "+"IReference "+"IReferenceCriterion "+"IReferenceEnabledMode "+"IReferenceFactory "+"IReferenceHistoryDescription "+"IReferenceInfo "+"IReferenceRecordCardWizardStep "+"IReferenceRequisiteDescription "+"IReferencesFactory "+"IReferenceValue "+"IRefRequisite "+"IReport "+"IReportFactory "+"IRequisite "+"IRequisiteDescription "+"IRequisiteDescriptionList "+"IRequisiteFactory "+"IRichEdit "+"IRouteStep "+"IRule "+"IRuleList "+"ISchemeBlock "+"IScript "+"IScriptFactory "+"ISearchCriteria "+"ISearchCriterion "+"ISearchDescription "+"ISearchFactory "+"ISearchFolderInfo "+"ISearchForObjectDescription "+"ISearchResultRestrictions "+"ISecuredContext "+"ISelectDialog "+"IServerEvent "+"IServerEventFactory "+"IServiceDialog "+"IServiceFactory "+"ISignature "+"ISignProvider "+"ISignProvider2 "+"ISignProvider3 "+"ISimpleCriterion "+"IStringCriterion "+"IStringList "+"IStringRequisite "+"IStringRequisiteDescription "+"IStringValue "+"ISystemDialogsFactory "+"ISystemInfo "+"ITabSheet "+"ITask "+"ITaskAbortReasonInfo "+"ITaskCardWizardStep "+"ITaskDescription "+"ITaskFactory "+"ITaskInfo "+"ITaskRoute "+"ITextCriterion "+"ITextRequisite "+"ITextValue "+"ITreeListSelectDialog "+"IUser "+"IUserList "+"IValue "+"IView "+"IWebBrowserControl "+"IWizard "+"IWizardAction "+"IWizardFactory "+"IWizardFormElement "+"IWizardParam "+"IWizardPickParam "+"IWizardReferenceParam "+"IWizardStep "+"IWorkAccessRights "+"IWorkDescription "+"IWorkflowAskableParam "+"IWorkflowAskableParams "+"IWorkflowBlock "+"IWorkflowBlockResult "+"IWorkflowEnabledMode "+"IWorkflowParam "+"IWorkflowPickParam "+"IWorkflowReferenceParam "+"IWorkState "+"IWorkTreeCustomNode "+"IWorkTreeJobNode "+"IWorkTreeTaskNode "+"IXMLEditorForm "+"SBCrypto ";var BUILTIN=CONSTANTS+ENUMS;var CLASS=predefined_variables;var LITERAL="null true false nil ";var NUMBERS={className:"number",begin:hljs.NUMBER_RE,relevance:0};var STRINGS={className:"string",variants:[{begin:'"',end:'"'},{begin:"'",end:"'"}]};var DOCTAGS={className:"doctag",begin:"\\b(?:TODO|DONE|BEGIN|END|STUB|CHG|FIXME|NOTE|BUG|XXX)\\b",relevance:0};var ISBL_LINE_COMMENT_MODE={className:"comment",begin:"//",end:"$",relevance:0,contains:[hljs.PHRASAL_WORDS_MODE,DOCTAGS]};var ISBL_BLOCK_COMMENT_MODE={className:"comment",begin:"/\\*",end:"\\*/",relevance:0,contains:[hljs.PHRASAL_WORDS_MODE,DOCTAGS]};var COMMENTS={variants:[ISBL_LINE_COMMENT_MODE,ISBL_BLOCK_COMMENT_MODE]};var KEYWORDS={keyword:KEYWORD,built_in:BUILTIN,class:CLASS,literal:LITERAL};var METHODS={begin:"\\.\\s*"+hljs.UNDERSCORE_IDENT_RE,keywords:KEYWORDS,relevance:0};var TYPES={className:"type",begin:":[ \\t]*("+interfaces.trim().replace(/\s/g,"|")+")",end:"[ \\t]*=",excludeEnd:true};var VARIABLES={className:"variable",lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,begin:UNDERSCORE_IDENT_RE,relevance:0,contains:[TYPES,METHODS]};var FUNCTION_TITLE=FUNCTION_NAME_IDENT_RE+"\\(";var TITLE_MODE={className:"title",lexemes:UNDERSCORE_IDENT_RE,keywords:{built_in:system_functions},begin:FUNCTION_TITLE,end:"\\(",returnBegin:true,excludeEnd:true};var FUNCTIONS={className:"function",begin:FUNCTION_TITLE,end:"\\)$",returnBegin:true,lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,illegal:"[\\[\\]\\|\\$\\?%,~#@]",contains:[TITLE_MODE,METHODS,VARIABLES,STRINGS,NUMBERS,COMMENTS]};return{aliases:["isbl"],case_insensitive:true,lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,illegal:"\\$|\\?|%|,|;$|~|#|@|</",contains:[FUNCTIONS,TYPES,METHODS,VARIABLES,STRINGS,NUMBERS,COMMENTS]}});hljs.registerLanguage("reasonml",function(hljs){function orReValues(ops){return ops.map(function(op){return op.split("").map(function(char){return"\\"+char}).join("")}).join("|")}var RE_IDENT="~?[a-z$_][0-9a-zA-Z$_]*";var RE_MODULE_IDENT="`?[A-Z$_][0-9a-zA-Z$_]*";var RE_PARAM_TYPEPARAM="'?[a-z$_][0-9a-z$_]*";var RE_PARAM_TYPE="s*:s*[a-z$_][0-9a-z$_]*((s*("+RE_PARAM_TYPEPARAM+"s*(,"+RE_PARAM_TYPEPARAM+")*)?s*))?";var RE_PARAM=RE_IDENT+"("+RE_PARAM_TYPE+")?("+RE_PARAM_TYPE+")?";var RE_OPERATOR="("+orReValues(["||","&&","++","**","+.","*","/","*.","/.","...","|>"])+"|==|===)";var RE_OPERATOR_SPACED="\\s+"+RE_OPERATOR+"\\s+";var KEYWORDS={keyword:"and as asr assert begin class constraint do done downto else end exception external"+"for fun function functor if in include inherit initializer"+"land lazy let lor lsl lsr lxor match method mod module mutable new nonrec"+"object of open or private rec sig struct then to try type val virtual when while with",built_in:"array bool bytes char exn|5 float int int32 int64 list lazy_t|5 nativeint|5 ref string unit ",literal:"true false"};var RE_NUMBER="\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)";var NUMBER_MODE={className:"number",relevance:0,variants:[{begin:RE_NUMBER},{begin:"\\(\\-"+RE_NUMBER+"\\)"}]};var OPERATOR_MODE={className:"operator",relevance:0,begin:RE_OPERATOR};var LIST_CONTENTS_MODES=[{className:"identifier",relevance:0,begin:RE_IDENT},OPERATOR_MODE,NUMBER_MODE];var MODULE_ACCESS_CONTENTS=[hljs.QUOTE_STRING_MODE,OPERATOR_MODE,{className:"module",begin:"\\b"+RE_MODULE_IDENT,returnBegin:true,end:".",contains:[{className:"identifier",begin:RE_MODULE_IDENT,relevance:0}]}];var PARAMS_CONTENTS=[{className:"module",begin:"\\b"+RE_MODULE_IDENT,returnBegin:true,end:".",relevance:0,contains:[{className:"identifier",begin:RE_MODULE_IDENT,relevance:0}]}];var PARAMS_MODE={begin:RE_IDENT,end:"(,|\\n|\\))",relevance:0,contains:[OPERATOR_MODE,{className:"typing",begin:":",end:"(,|\\n)",returnBegin:true,relevance:0,contains:PARAMS_CONTENTS}]};var FUNCTION_BLOCK_MODE={className:"function",relevance:0,keywords:KEYWORDS,variants:[{begin:"\\s(\\(\\.?.*?\\)|"+RE_IDENT+")\\s*=>",end:"\\s*=>",returnBegin:true,relevance:0,contains:[{className:"params",variants:[{begin:RE_IDENT},{begin:RE_PARAM},{begin:/\(\s*\)/}]}]},{begin:"\\s\\(\\.?[^;\\|]*\\)\\s*=>",end:"\\s=>",returnBegin:true,relevance:0,contains:[{className:"params",relevance:0,variants:[PARAMS_MODE]}]},{begin:"\\(\\.\\s"+RE_IDENT+"\\)\\s*=>"}]};MODULE_ACCESS_CONTENTS.push(FUNCTION_BLOCK_MODE);var CONSTRUCTOR_MODE={className:"constructor",begin:RE_MODULE_IDENT+"\\(",end:"\\)",illegal:"\\n",keywords:KEYWORDS,contains:[hljs.QUOTE_STRING_MODE,OPERATOR_MODE,{className:"params",begin:"\\b"+RE_IDENT}]};var PATTERN_MATCH_BLOCK_MODE={className:"pattern-match",begin:"\\|",returnBegin:true,keywords:KEYWORDS,end:"=>",relevance:0,contains:[CONSTRUCTOR_MODE,OPERATOR_MODE,{relevance:0,className:"constructor",begin:RE_MODULE_IDENT}]};var MODULE_ACCESS_MODE={className:"module-access",keywords:KEYWORDS,returnBegin:true,variants:[{begin:"\\b("+RE_MODULE_IDENT+"\\.)+"+RE_IDENT},{begin:"\\b("+RE_MODULE_IDENT+"\\.)+\\(",end:"\\)",returnBegin:true,contains:[FUNCTION_BLOCK_MODE,{begin:"\\(",end:"\\)",skip:true}].concat(MODULE_ACCESS_CONTENTS)},{begin:"\\b("+RE_MODULE_IDENT+"\\.)+{",end:"}"}],contains:MODULE_ACCESS_CONTENTS};PARAMS_CONTENTS.push(MODULE_ACCESS_MODE);return{aliases:["re"],keywords:KEYWORDS,illegal:"(:\\-|:=|\\${|\\+=)",contains:[hljs.COMMENT("/\\*","\\*/",{illegal:"^(\\#,\\/\\/)"}),{className:"character",begin:"'(\\\\[^']+|[^'])'",illegal:"\\n",relevance:0},hljs.QUOTE_STRING_MODE,{className:"literal",begin:"\\(\\)",relevance:0},{className:"literal",begin:"\\[\\|",end:"\\|\\]",relevance:0,contains:LIST_CONTENTS_MODES},{className:"literal",begin:"\\[",end:"\\]",relevance:0,contains:LIST_CONTENTS_MODES},CONSTRUCTOR_MODE,{className:"operator",begin:RE_OPERATOR_SPACED,illegal:"\\-\\->",relevance:0},NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,PATTERN_MATCH_BLOCK_MODE,FUNCTION_BLOCK_MODE,{className:"module-def",begin:"\\bmodule\\s+"+RE_IDENT+"\\s+"+RE_MODULE_IDENT+"\\s+=\\s+{",end:"}",returnBegin:true,keywords:KEYWORDS,relevance:0,contains:[{className:"module",relevance:0,begin:RE_MODULE_IDENT},{begin:"{",end:"}",skip:true}].concat(MODULE_ACCESS_CONTENTS)},MODULE_ACCESS_MODE]}});hljs.registerLanguage("maxima",function(hljs){var KEYWORDS="if then else elseif for thru do while unless step in and or not";var LITERALS="true false unknown inf minf ind und %e %i %pi %phi %gamma";var BUILTIN_FUNCTIONS=" abasep abs absint absolute_real_time acos acosh acot acoth acsc acsch activate"+" addcol add_edge add_edges addmatrices addrow add_vertex add_vertices adjacency_matrix"+" adjoin adjoint af agd airy airy_ai airy_bi airy_dai airy_dbi algsys alg_type"+" alias allroots alphacharp alphanumericp amortization %and annuity_fv"+" annuity_pv antid antidiff AntiDifference append appendfile apply apply1 apply2"+" applyb1 apropos args arit_amortization arithmetic arithsum array arrayapply"+" arrayinfo arraymake arraysetapply ascii asec asech asin asinh askinteger"+" asksign assoc assoc_legendre_p assoc_legendre_q assume assume_external_byte_order"+" asympa at atan atan2 atanh atensimp atom atvalue augcoefmatrix augmented_lagrangian_method"+" av average_degree backtrace bars barsplot barsplot_description base64 base64_decode"+" bashindices batch batchload bc2 bdvac belln benefit_cost bern bernpoly bernstein_approx"+" bernstein_expand bernstein_poly bessel bessel_i bessel_j bessel_k bessel_simplify"+" bessel_y beta beta_incomplete beta_incomplete_generalized beta_incomplete_regularized"+" bezout bfallroots bffac bf_find_root bf_fmin_cobyla bfhzeta bfloat bfloatp"+" bfpsi bfpsi0 bfzeta biconnected_components bimetric binomial bipartition"+" block blockmatrixp bode_gain bode_phase bothcoef box boxplot boxplot_description"+" break bug_report build_info|10 buildq build_sample burn cabs canform canten"+" cardinality carg cartan cartesian_product catch cauchy_matrix cbffac cdf_bernoulli"+" cdf_beta cdf_binomial cdf_cauchy cdf_chi2 cdf_continuous_uniform cdf_discrete_uniform"+" cdf_exp cdf_f cdf_gamma cdf_general_finite_discrete cdf_geometric cdf_gumbel"+" cdf_hypergeometric cdf_laplace cdf_logistic cdf_lognormal cdf_negative_binomial"+" cdf_noncentral_chi2 cdf_noncentral_student_t cdf_normal cdf_pareto cdf_poisson"+" cdf_rank_sum cdf_rayleigh cdf_signed_rank cdf_student_t cdf_weibull cdisplay"+" ceiling central_moment cequal cequalignore cf cfdisrep cfexpand cgeodesic"+" cgreaterp cgreaterpignore changename changevar chaosgame charat charfun charfun2"+" charlist charp charpoly chdir chebyshev_t chebyshev_u checkdiv check_overlaps"+" chinese cholesky christof chromatic_index chromatic_number cint circulant_graph"+" clear_edge_weight clear_rules clear_vertex_label clebsch_gordan clebsch_graph"+" clessp clesspignore close closefile cmetric coeff coefmatrix cograd col collapse"+" collectterms columnop columnspace columnswap columnvector combination combine"+" comp2pui compare compfile compile compile_file complement_graph complete_bipartite_graph"+" complete_graph complex_number_p components compose_functions concan concat"+" conjugate conmetderiv connected_components connect_vertices cons constant"+" constantp constituent constvalue cont2part content continuous_freq contortion"+" contour_plot contract contract_edge contragrad contrib_ode convert coord"+" copy copy_file copy_graph copylist copymatrix cor cos cosh cot coth cov cov1"+" covdiff covect covers crc24sum create_graph create_list csc csch csetup cspline"+" ctaylor ct_coordsys ctransform ctranspose cube_graph cuboctahedron_graph"+" cunlisp cv cycle_digraph cycle_graph cylindrical days360 dblint deactivate"+" declare declare_constvalue declare_dimensions declare_fundamental_dimensions"+" declare_fundamental_units declare_qty declare_translated declare_unit_conversion"+" declare_units declare_weights decsym defcon define define_alt_display define_variable"+" defint defmatch defrule defstruct deftaylor degree_sequence del delete deleten"+" delta demo demoivre denom depends derivdegree derivlist describe desolve"+" determinant dfloat dgauss_a dgauss_b dgeev dgemm dgeqrf dgesv dgesvd diag"+" diagmatrix diag_matrix diagmatrixp diameter diff digitcharp dimacs_export"+" dimacs_import dimension dimensionless dimensions dimensions_as_list direct"+" directory discrete_freq disjoin disjointp disolate disp dispcon dispform"+" dispfun dispJordan display disprule dispterms distrib divide divisors divsum"+" dkummer_m dkummer_u dlange dodecahedron_graph dotproduct dotsimp dpart"+" draw draw2d draw3d drawdf draw_file draw_graph dscalar echelon edge_coloring"+" edge_connectivity edges eigens_by_jacobi eigenvalues eigenvectors eighth"+" einstein eivals eivects elapsed_real_time elapsed_run_time ele2comp ele2polynome"+" ele2pui elem elementp elevation_grid elim elim_allbut eliminate eliminate_using"+" ellipse elliptic_e elliptic_ec elliptic_eu elliptic_f elliptic_kc elliptic_pi"+" ematrix empty_graph emptyp endcons entermatrix entertensor entier equal equalp"+" equiv_classes erf erfc erf_generalized erfi errcatch error errormsg errors"+" euler ev eval_string evenp every evolution evolution2d evundiff example exp"+" expand expandwrt expandwrt_factored expint expintegral_chi expintegral_ci"+" expintegral_e expintegral_e1 expintegral_ei expintegral_e_simplify expintegral_li"+" expintegral_shi expintegral_si explicit explose exponentialize express expt"+" exsec extdiff extract_linear_equations extremal_subset ezgcd %f f90 facsum"+" factcomb factor factorfacsum factorial factorout factorsum facts fast_central_elements"+" fast_linsolve fasttimes featurep fernfale fft fib fibtophi fifth filename_merge"+" file_search file_type fillarray findde find_root find_root_abs find_root_error"+" find_root_rel first fix flatten flength float floatnump floor flower_snark"+" flush flush1deriv flushd flushnd flush_output fmin_cobyla forget fortran"+" fourcos fourexpand fourier fourier_elim fourint fourintcos fourintsin foursimp"+" foursin fourth fposition frame_bracket freeof freshline fresnel_c fresnel_s"+" from_adjacency_matrix frucht_graph full_listify fullmap fullmapl fullratsimp"+" fullratsubst fullsetify funcsolve fundamental_dimensions fundamental_units"+" fundef funmake funp fv g0 g1 gamma gamma_greek gamma_incomplete gamma_incomplete_generalized"+" gamma_incomplete_regularized gauss gauss_a gauss_b gaussprob gcd gcdex gcdivide"+" gcfac gcfactor gd generalized_lambert_w genfact gen_laguerre genmatrix gensym"+" geo_amortization geo_annuity_fv geo_annuity_pv geomap geometric geometric_mean"+" geosum get getcurrentdirectory get_edge_weight getenv get_lu_factors get_output_stream_string"+" get_pixel get_plot_option get_tex_environment get_tex_environment_default"+" get_vertex_label gfactor gfactorsum ggf girth global_variances gn gnuplot_close"+" gnuplot_replot gnuplot_reset gnuplot_restart gnuplot_start go Gosper GosperSum"+" gr2d gr3d gradef gramschmidt graph6_decode graph6_encode graph6_export graph6_import"+" graph_center graph_charpoly graph_eigenvalues graph_flow graph_order graph_periphery"+" graph_product graph_size graph_union great_rhombicosidodecahedron_graph great_rhombicuboctahedron_graph"+" grid_graph grind grobner_basis grotzch_graph hamilton_cycle hamilton_path"+" hankel hankel_1 hankel_2 harmonic harmonic_mean hav heawood_graph hermite"+" hessian hgfred hilbertmap hilbert_matrix hipow histogram histogram_description"+" hodge horner hypergeometric i0 i1 %ibes ic1 ic2 ic_convert ichr1 ichr2 icosahedron_graph"+" icosidodecahedron_graph icurvature ident identfor identity idiff idim idummy"+" ieqn %if ifactors iframes ifs igcdex igeodesic_coords ilt image imagpart"+" imetric implicit implicit_derivative implicit_plot indexed_tensor indices"+" induced_subgraph inferencep inference_result infix info_display init_atensor"+" init_ctensor in_neighbors innerproduct inpart inprod inrt integerp integer_partitions"+" integrate intersect intersection intervalp intopois intosum invariant1 invariant2"+" inverse_fft inverse_jacobi_cd inverse_jacobi_cn inverse_jacobi_cs inverse_jacobi_dc"+" inverse_jacobi_dn inverse_jacobi_ds inverse_jacobi_nc inverse_jacobi_nd inverse_jacobi_ns"+" inverse_jacobi_sc inverse_jacobi_sd inverse_jacobi_sn invert invert_by_adjoint"+" invert_by_lu inv_mod irr is is_biconnected is_bipartite is_connected is_digraph"+" is_edge_in_graph is_graph is_graph_or_digraph ishow is_isomorphic isolate"+" isomorphism is_planar isqrt isreal_p is_sconnected is_tree is_vertex_in_graph"+" items_inference %j j0 j1 jacobi jacobian jacobi_cd jacobi_cn jacobi_cs jacobi_dc"+" jacobi_dn jacobi_ds jacobi_nc jacobi_nd jacobi_ns jacobi_p jacobi_sc jacobi_sd"+" jacobi_sn JF jn join jordan julia julia_set julia_sin %k kdels kdelta kill"+" killcontext kostka kron_delta kronecker_product kummer_m kummer_u kurtosis"+" kurtosis_bernoulli kurtosis_beta kurtosis_binomial kurtosis_chi2 kurtosis_continuous_uniform"+" kurtosis_discrete_uniform kurtosis_exp kurtosis_f kurtosis_gamma kurtosis_general_finite_discrete"+" kurtosis_geometric kurtosis_gumbel kurtosis_hypergeometric kurtosis_laplace"+" kurtosis_logistic kurtosis_lognormal kurtosis_negative_binomial kurtosis_noncentral_chi2"+" kurtosis_noncentral_student_t kurtosis_normal kurtosis_pareto kurtosis_poisson"+" kurtosis_rayleigh kurtosis_student_t kurtosis_weibull label labels lagrange"+" laguerre lambda lambert_w laplace laplacian_matrix last lbfgs lc2kdt lcharp"+" lc_l lcm lc_u ldefint ldisp ldisplay legendre_p legendre_q leinstein length"+" let letrules letsimp levi_civita lfreeof lgtreillis lhs li liediff limit"+" Lindstedt linear linearinterpol linear_program linear_regression line_graph"+" linsolve listarray list_correlations listify list_matrix_entries list_nc_monomials"+" listoftens listofvars listp lmax lmin load loadfile local locate_matrix_entry"+" log logcontract log_gamma lopow lorentz_gauge lowercasep lpart lratsubst"+" lreduce lriemann lsquares_estimates lsquares_estimates_approximate lsquares_estimates_exact"+" lsquares_mse lsquares_residual_mse lsquares_residuals lsum ltreillis lu_backsub"+" lucas lu_factor %m macroexpand macroexpand1 make_array makebox makefact makegamma"+" make_graph make_level_picture makelist makeOrders make_poly_continent make_poly_country"+" make_polygon make_random_state make_rgb_picture makeset make_string_input_stream"+" make_string_output_stream make_transform mandelbrot mandelbrot_set map mapatom"+" maplist matchdeclare matchfix mat_cond mat_fullunblocker mat_function mathml_display"+" mat_norm matrix matrixmap matrixp matrix_size mattrace mat_trace mat_unblocker"+" max max_clique max_degree max_flow maximize_lp max_independent_set max_matching"+" maybe md5sum mean mean_bernoulli mean_beta mean_binomial mean_chi2 mean_continuous_uniform"+" mean_deviation mean_discrete_uniform mean_exp mean_f mean_gamma mean_general_finite_discrete"+" mean_geometric mean_gumbel mean_hypergeometric mean_laplace mean_logistic"+" mean_lognormal mean_negative_binomial mean_noncentral_chi2 mean_noncentral_student_t"+" mean_normal mean_pareto mean_poisson mean_rayleigh mean_student_t mean_weibull"+" median median_deviation member mesh metricexpandall mgf1_sha1 min min_degree"+" min_edge_cut minfactorial minimalPoly minimize_lp minimum_spanning_tree minor"+" minpack_lsquares minpack_solve min_vertex_cover min_vertex_cut mkdir mnewton"+" mod mode_declare mode_identity ModeMatrix moebius mon2schur mono monomial_dimensions"+" multibernstein_poly multi_display_for_texinfo multi_elem multinomial multinomial_coeff"+" multi_orbit multiplot_mode multi_pui multsym multthru mycielski_graph nary"+" natural_unit nc_degree ncexpt ncharpoly negative_picture neighbors new newcontext"+" newdet new_graph newline newton new_variable next_prime nicedummies niceindices"+" ninth nofix nonarray noncentral_moment nonmetricity nonnegintegerp nonscalarp"+" nonzeroandfreeof notequal nounify nptetrad npv nroots nterms ntermst"+" nthroot nullity nullspace num numbered_boundaries numberp number_to_octets"+" num_distinct_partitions numerval numfactor num_partitions nusum nzeta nzetai"+" nzetar octets_to_number octets_to_oid odd_girth oddp ode2 ode_check odelin"+" oid_to_octets op opena opena_binary openr openr_binary openw openw_binary"+" operatorp opsubst optimize %or orbit orbits ordergreat ordergreatp orderless"+" orderlessp orthogonal_complement orthopoly_recur orthopoly_weight outermap"+" out_neighbors outofpois pade parabolic_cylinder_d parametric parametric_surface"+" parg parGosper parse_string parse_timedate part part2cont partfrac partition"+" partition_set partpol path_digraph path_graph pathname_directory pathname_name"+" pathname_type pdf_bernoulli pdf_beta pdf_binomial pdf_cauchy pdf_chi2 pdf_continuous_uniform"+" pdf_discrete_uniform pdf_exp pdf_f pdf_gamma pdf_general_finite_discrete"+" pdf_geometric pdf_gumbel pdf_hypergeometric pdf_laplace pdf_logistic pdf_lognormal"+" pdf_negative_binomial pdf_noncentral_chi2 pdf_noncentral_student_t pdf_normal"+" pdf_pareto pdf_poisson pdf_rank_sum pdf_rayleigh pdf_signed_rank pdf_student_t"+" pdf_weibull pearson_skewness permanent permut permutation permutations petersen_graph"+" petrov pickapart picture_equalp picturep piechart piechart_description planar_embedding"+" playback plog plot2d plot3d plotdf ploteq plsquares pochhammer points poisdiff"+" poisexpt poisint poismap poisplus poissimp poissubst poistimes poistrim polar"+" polarform polartorect polar_to_xy poly_add poly_buchberger poly_buchberger_criterion"+" poly_colon_ideal poly_content polydecomp poly_depends_p poly_elimination_ideal"+" poly_exact_divide poly_expand poly_expt poly_gcd polygon poly_grobner poly_grobner_equal"+" poly_grobner_member poly_grobner_subsetp poly_ideal_intersection poly_ideal_polysaturation"+" poly_ideal_polysaturation1 poly_ideal_saturation poly_ideal_saturation1 poly_lcm"+" poly_minimization polymod poly_multiply polynome2ele polynomialp poly_normal_form"+" poly_normalize poly_normalize_list poly_polysaturation_extension poly_primitive_part"+" poly_pseudo_divide poly_reduced_grobner poly_reduction poly_saturation_extension"+" poly_s_polynomial poly_subtract polytocompanion pop postfix potential power_mod"+" powerseries powerset prefix prev_prime primep primes principal_components"+" print printf printfile print_graph printpois printprops prodrac product properties"+" propvars psi psubst ptriangularize pui pui2comp pui2ele pui2polynome pui_direct"+" puireduc push put pv qput qrange qty quad_control quad_qag quad_qagi quad_qagp"+" quad_qags quad_qawc quad_qawf quad_qawo quad_qaws quadrilateral quantile"+" quantile_bernoulli quantile_beta quantile_binomial quantile_cauchy quantile_chi2"+" quantile_continuous_uniform quantile_discrete_uniform quantile_exp quantile_f"+" quantile_gamma quantile_general_finite_discrete quantile_geometric quantile_gumbel"+" quantile_hypergeometric quantile_laplace quantile_logistic quantile_lognormal"+" quantile_negative_binomial quantile_noncentral_chi2 quantile_noncentral_student_t"+" quantile_normal quantile_pareto quantile_poisson quantile_rayleigh quantile_student_t"+" quantile_weibull quartile_skewness quit qunit quotient racah_v racah_w radcan"+" radius random random_bernoulli random_beta random_binomial random_bipartite_graph"+" random_cauchy random_chi2 random_continuous_uniform random_digraph random_discrete_uniform"+" random_exp random_f random_gamma random_general_finite_discrete random_geometric"+" random_graph random_graph1 random_gumbel random_hypergeometric random_laplace"+" random_logistic random_lognormal random_negative_binomial random_network"+" random_noncentral_chi2 random_noncentral_student_t random_normal random_pareto"+" random_permutation random_poisson random_rayleigh random_regular_graph random_student_t"+" random_tournament random_tree random_weibull range rank rat ratcoef ratdenom"+" ratdiff ratdisrep ratexpand ratinterpol rational rationalize ratnumer ratnump"+" ratp ratsimp ratsubst ratvars ratweight read read_array read_binary_array"+" read_binary_list read_binary_matrix readbyte readchar read_hashed_array readline"+" read_list read_matrix read_nested_list readonly read_xpm real_imagpart_to_conjugate"+" realpart realroots rearray rectangle rectform rectform_log_if_constant recttopolar"+" rediff reduce_consts reduce_order region region_boundaries region_boundaries_plus"+" rem remainder remarray rembox remcomps remcon remcoord remfun remfunction"+" remlet remove remove_constvalue remove_dimensions remove_edge remove_fundamental_dimensions"+" remove_fundamental_units remove_plot_option remove_vertex rempart remrule"+" remsym remvalue rename rename_file reset reset_displays residue resolvante"+" resolvante_alternee1 resolvante_bipartite resolvante_diedrale resolvante_klein"+" resolvante_klein3 resolvante_produit_sym resolvante_unitaire resolvante_vierer"+" rest resultant return reveal reverse revert revert2 rgb2level rhs ricci riemann"+" rinvariant risch rk rmdir rncombine romberg room rootscontract round row"+" rowop rowswap rreduce run_testsuite %s save saving scalarp scaled_bessel_i"+" scaled_bessel_i0 scaled_bessel_i1 scalefactors scanmap scatterplot scatterplot_description"+" scene schur2comp sconcat scopy scsimp scurvature sdowncase sec sech second"+" sequal sequalignore set_alt_display setdifference set_draw_defaults set_edge_weight"+" setelmx setequalp setify setp set_partitions set_plot_option set_prompt set_random_state"+" set_tex_environment set_tex_environment_default setunits setup_autoload set_up_dot_simplifications"+" set_vertex_label seventh sexplode sf sha1sum sha256sum shortest_path shortest_weighted_path"+" show showcomps showratvars sierpinskiale sierpinskimap sign signum similaritytransform"+" simp_inequality simplify_sum simplode simpmetderiv simtran sin sinh sinsert"+" sinvertcase sixth skewness skewness_bernoulli skewness_beta skewness_binomial"+" skewness_chi2 skewness_continuous_uniform skewness_discrete_uniform skewness_exp"+" skewness_f skewness_gamma skewness_general_finite_discrete skewness_geometric"+" skewness_gumbel skewness_hypergeometric skewness_laplace skewness_logistic"+" skewness_lognormal skewness_negative_binomial skewness_noncentral_chi2 skewness_noncentral_student_t"+" skewness_normal skewness_pareto skewness_poisson skewness_rayleigh skewness_student_t"+" skewness_weibull slength smake small_rhombicosidodecahedron_graph small_rhombicuboctahedron_graph"+" smax smin smismatch snowmap snub_cube_graph snub_dodecahedron_graph solve"+" solve_rec solve_rec_rat some somrac sort sparse6_decode sparse6_encode sparse6_export"+" sparse6_import specint spherical spherical_bessel_j spherical_bessel_y spherical_hankel1"+" spherical_hankel2 spherical_harmonic spherical_to_xyz splice split sposition"+" sprint sqfr sqrt sqrtdenest sremove sremovefirst sreverse ssearch ssort sstatus"+" ssubst ssubstfirst staircase standardize standardize_inverse_trig starplot"+" starplot_description status std std1 std_bernoulli std_beta std_binomial"+" std_chi2 std_continuous_uniform std_discrete_uniform std_exp std_f std_gamma"+" std_general_finite_discrete std_geometric std_gumbel std_hypergeometric std_laplace"+" std_logistic std_lognormal std_negative_binomial std_noncentral_chi2 std_noncentral_student_t"+" std_normal std_pareto std_poisson std_rayleigh std_student_t std_weibull"+" stemplot stirling stirling1 stirling2 strim striml strimr string stringout"+" stringp strong_components struve_h struve_l sublis sublist sublist_indices"+" submatrix subsample subset subsetp subst substinpart subst_parallel substpart"+" substring subvar subvarp sum sumcontract summand_to_rec supcase supcontext"+" symbolp symmdifference symmetricp system take_channel take_inference tan"+" tanh taylor taylorinfo taylorp taylor_simplifier taytorat tcl_output tcontract"+" tellrat tellsimp tellsimpafter tentex tenth test_mean test_means_difference"+" test_normality test_proportion test_proportions_difference test_rank_sum"+" test_sign test_signed_rank test_variance test_variance_ratio tex tex1 tex_display"+" texput %th third throw time timedate timer timer_info tldefint tlimit todd_coxeter"+" toeplitz tokens to_lisp topological_sort to_poly to_poly_solve totaldisrep"+" totalfourier totient tpartpol trace tracematrix trace_options transform_sample"+" translate translate_file transpose treefale tree_reduce treillis treinat"+" triangle triangularize trigexpand trigrat trigreduce trigsimp trunc truncate"+" truncated_cube_graph truncated_dodecahedron_graph truncated_icosahedron_graph"+" truncated_tetrahedron_graph tr_warnings_get tube tutte_graph ueivects uforget"+" ultraspherical underlying_graph undiff union unique uniteigenvectors unitp"+" units unit_step unitvector unorder unsum untellrat untimer"+" untrace uppercasep uricci uriemann uvect vandermonde_matrix var var1 var_bernoulli"+" var_beta var_binomial var_chi2 var_continuous_uniform var_discrete_uniform"+" var_exp var_f var_gamma var_general_finite_discrete var_geometric var_gumbel"+" var_hypergeometric var_laplace var_logistic var_lognormal var_negative_binomial"+" var_noncentral_chi2 var_noncentral_student_t var_normal var_pareto var_poisson"+" var_rayleigh var_student_t var_weibull vector vectorpotential vectorsimp"+" verbify vers vertex_coloring vertex_connectivity vertex_degree vertex_distance"+" vertex_eccentricity vertex_in_degree vertex_out_degree vertices vertices_to_cycle"+" vertices_to_path %w weyl wheel_graph wiener_index wigner_3j wigner_6j"+" wigner_9j with_stdout write_binary_data writebyte write_data writefile wronskian"+" xreduce xthru %y Zeilberger zeroequiv zerofor zeromatrix zeromatrixp zeta"+" zgeev zheev zlange zn_add_table zn_carmichael_lambda zn_characteristic_factors"+" zn_determinant zn_factor_generators zn_invert_by_lu zn_log zn_mult_table"+" absboxchar activecontexts adapt_depth additive adim aform algebraic"+" algepsilon algexact aliases allbut all_dotsimp_denoms allocation allsym alphabetic"+" animation antisymmetric arrays askexp assume_pos assume_pos_pred assumescalar"+" asymbol atomgrad atrig1 axes axis_3d axis_bottom axis_left axis_right axis_top"+" azimuth background background_color backsubst berlefact bernstein_explicit"+" besselexpand beta_args_sum_to_integer beta_expand bftorat bftrunc bindtest"+" border boundaries_array box boxchar breakup %c capping cauchysum cbrange"+" cbtics center cflength cframe_flag cnonmet_flag color color_bar color_bar_tics"+" colorbox columns commutative complex cone context contexts contour contour_levels"+" cosnpiflag ctaypov ctaypt ctayswitch ctayvar ct_coords ctorsion_flag ctrgsimp"+" cube current_let_rule_package cylinder data_file_name debugmode decreasing"+" default_let_rule_package delay dependencies derivabbrev derivsubst detout"+" diagmetric diff dim dimensions dispflag display2d|10 display_format_internal"+" distribute_over doallmxops domain domxexpt domxmxops domxnctimes dontfactor"+" doscmxops doscmxplus dot0nscsimp dot0simp dot1simp dotassoc dotconstrules"+" dotdistrib dotexptsimp dotident dotscrules draw_graph_program draw_realpart"+" edge_color edge_coloring edge_partition edge_type edge_width %edispflag"+" elevation %emode endphi endtheta engineering_format_floats enhanced3d %enumer"+" epsilon_lp erfflag erf_representation errormsg error_size error_syms error_type"+" %e_to_numlog eval even evenfun evflag evfun ev_point expandwrt_denom expintexpand"+" expintrep expon expop exptdispflag exptisolate exptsubst facexpand facsum_combine"+" factlim factorflag factorial_expand factors_only fb feature features"+" file_name file_output_append file_search_demo file_search_lisp file_search_maxima|10"+" file_search_tests file_search_usage file_type_lisp file_type_maxima|10 fill_color"+" fill_density filled_func fixed_vertices flipflag float2bf font font_size"+" fortindent fortspaces fpprec fpprintprec functions gamma_expand gammalim"+" gdet genindex gensumnum GGFCFMAX GGFINFINITY globalsolve gnuplot_command"+" gnuplot_curve_styles gnuplot_curve_titles gnuplot_default_term_command gnuplot_dumb_term_command"+" gnuplot_file_args gnuplot_file_name gnuplot_out_file gnuplot_pdf_term_command"+" gnuplot_pm3d gnuplot_png_term_command gnuplot_postamble gnuplot_preamble"+" gnuplot_ps_term_command gnuplot_svg_term_command gnuplot_term gnuplot_view_args"+" Gosper_in_Zeilberger gradefs grid grid2d grind halfangles head_angle head_both"+" head_length head_type height hypergeometric_representation %iargs ibase"+" icc1 icc2 icounter idummyx ieqnprint ifb ifc1 ifc2 ifg ifgi ifr iframe_bracket_form"+" ifri igeowedge_flag ikt1 ikt2 imaginary inchar increasing infeval"+" infinity inflag infolists inm inmc1 inmc2 intanalysis integer integervalued"+" integrate_use_rootsof integration_constant integration_constant_counter interpolate_color"+" intfaclim ip_grid ip_grid_in irrational isolate_wrt_times iterations itr"+" julia_parameter %k1 %k2 keepfloat key key_pos kinvariant kt label label_alignment"+" label_orientation labels lassociative lbfgs_ncorrections lbfgs_nfeval_max"+" leftjust legend letrat let_rule_packages lfg lg lhospitallim limsubst linear"+" linear_solver linechar linel|10 linenum line_type linewidth line_width linsolve_params"+" linsolvewarn lispdisp listarith listconstvars listdummyvars lmxchar load_pathname"+" loadprint logabs logarc logcb logconcoeffp logexpand lognegint logsimp logx"+" logx_secondary logy logy_secondary logz lriem m1pbranch macroexpansion macros"+" mainvar manual_demo maperror mapprint matrix_element_add matrix_element_mult"+" matrix_element_transpose maxapplydepth maxapplyheight maxima_tempdir|10 maxima_userdir|10"+" maxnegex MAX_ORD maxposex maxpsifracdenom maxpsifracnum maxpsinegint maxpsiposint"+" maxtayorder mesh_lines_color method mod_big_prime mode_check_errorp"+" mode_checkp mode_check_warnp mod_test mod_threshold modular_linear_solver"+" modulus multiplicative multiplicities myoptions nary negdistrib negsumdispflag"+" newline newtonepsilon newtonmaxiter nextlayerfactor niceindicespref nm nmc"+" noeval nolabels nonegative_lp noninteger nonscalar noun noundisp nouns np"+" npi nticks ntrig numer numer_pbranch obase odd oddfun opacity opproperties"+" opsubst optimprefix optionset orientation origin orthopoly_returns_intervals"+" outative outchar packagefile palette partswitch pdf_file pfeformat phiresolution"+" %piargs piece pivot_count_sx pivot_max_sx plot_format plot_options plot_realpart"+" png_file pochhammer_max_index points pointsize point_size points_joined point_type"+" poislim poisson poly_coefficient_ring poly_elimination_order polyfactor poly_grobner_algorithm"+" poly_grobner_debug poly_monomial_order poly_primary_elimination_order poly_return_term_list"+" poly_secondary_elimination_order poly_top_reduction_only posfun position"+" powerdisp pred prederror primep_number_of_tests product_use_gamma program"+" programmode promote_float_to_bigfloat prompt proportional_axes props psexpand"+" ps_file radexpand radius radsubstflag rassociative ratalgdenom ratchristof"+" ratdenomdivide rateinstein ratepsilon ratfac rational ratmx ratprint ratriemann"+" ratsimpexpons ratvarswitch ratweights ratweyl ratwtlvl real realonly redraw"+" refcheck resolution restart resultant ric riem rmxchar %rnum_list rombergabs"+" rombergit rombergmin rombergtol rootsconmode rootsepsilon run_viewer same_xy"+" same_xyz savedef savefactors scalar scalarmatrixp scale scale_lp setcheck"+" setcheckbreak setval show_edge_color show_edges show_edge_type show_edge_width"+" show_id show_label showtime show_vertex_color show_vertex_size show_vertex_type"+" show_vertices show_weight simp simplified_output simplify_products simpproduct"+" simpsum sinnpiflag solvedecomposes solveexplicit solvefactors solvenullwarn"+" solveradcan solvetrigwarn space sparse sphere spring_embedding_depth sqrtdispflag"+" stardisp startphi starttheta stats_numer stringdisp structures style sublis_apply_lambda"+" subnumsimp sumexpand sumsplitfact surface surface_hide svg_file symmetric"+" tab taylordepth taylor_logexpand taylor_order_coefficients taylor_truncate_polynomials"+" tensorkill terminal testsuite_files thetaresolution timer_devalue title tlimswitch"+" tr track transcompile transform transform_xy translate_fast_arrays transparent"+" transrun tr_array_as_ref tr_bound_function_applyp tr_file_tty_messagesp tr_float_can_branch_complex"+" tr_function_call_default trigexpandplus trigexpandtimes triginverses trigsign"+" trivial_solutions tr_numer tr_optimize_max_loop tr_semicompile tr_state_vars"+" tr_warn_bad_function_calls tr_warn_fexpr tr_warn_meval tr_warn_mode"+" tr_warn_undeclared tr_warn_undefined_variable tstep ttyoff tube_extremes"+" ufg ug %unitexpand unit_vectors uric uriem use_fast_arrays user_preamble"+" usersetunits values vect_cross verbose vertex_color vertex_coloring vertex_partition"+" vertex_size vertex_type view warnings weyl width windowname windowtitle wired_surface"+" wireframe xaxis xaxis_color xaxis_secondary xaxis_type xaxis_width xlabel"+" xlabel_secondary xlength xrange xrange_secondary xtics xtics_axis xtics_rotate"+" xtics_rotate_secondary xtics_secondary xtics_secondary_axis xu_grid x_voxel"+" xy_file xyplane xy_scale yaxis yaxis_color yaxis_secondary yaxis_type yaxis_width"+" ylabel ylabel_secondary ylength yrange yrange_secondary ytics ytics_axis"+" ytics_rotate ytics_rotate_secondary ytics_secondary ytics_secondary_axis"+" yv_grid y_voxel yx_ratio zaxis zaxis_color zaxis_type zaxis_width zeroa zerob"+" zerobern zeta%pi zlabel zlabel_rotate zlength zmin zn_primroot_limit zn_primroot_pretest";var SYMBOLS="_ __ %|0 %%|0";return{lexemes:"[A-Za-z_%][0-9A-Za-z_%]*",keywords:{keyword:KEYWORDS,literal:LITERALS,built_in:BUILTIN_FUNCTIONS,symbol:SYMBOLS},contains:[{className:"comment",begin:"/\\*",end:"\\*/",contains:["self"]},hljs.QUOTE_STRING_MODE,{className:"number",relevance:0,variants:[{begin:"\\b(\\d+|\\d+\\.|\\.\\d+|\\d+\\.\\d+)[Ee][-+]?\\d+\\b"},{begin:"\\b(\\d+|\\d+\\.|\\.\\d+|\\d+\\.\\d+)[Bb][-+]?\\d+\\b",relevance:10},{begin:"\\b(\\.\\d+|\\d+\\.\\d+)\\b"},{begin:"\\b(\\d+|0[0-9A-Za-z]+)\\.?\\b"}]}],illegal:/@/}});hljs.registerLanguage("1c",function(hljs){var UNDERSCORE_IDENT_RE="[A-Za-zА-Яа-яёЁ_][A-Za-zА-Яа-яёЁ_0-9]+";var v7_keywords="далее ";var v8_keywords="возврат вызватьисключение выполнить для если и из или иначе иначеесли исключение каждого конецесли "+"конецпопытки конеццикла не новый перейти перем по пока попытка прервать продолжить тогда цикл экспорт ";var KEYWORD=v7_keywords+v8_keywords;var v7_meta_keywords="загрузитьизфайла ";var v8_meta_keywords="вебклиент вместо внешнеесоединение клиент конецобласти мобильноеприложениеклиент мобильноеприложениесервер "+"наклиенте наклиентенасервере наклиентенасерверебезконтекста насервере насерверебезконтекста область перед "+"после сервер толстыйклиентобычноеприложение толстыйклиентуправляемоеприложение тонкийклиент ";var METAKEYWORD=v7_meta_keywords+v8_meta_keywords;var v7_system_constants="разделительстраниц разделительстрок символтабуляции ";var v7_global_context_methods="ansitooem oemtoansi ввестивидсубконто ввестиперечисление ввестипериод ввестиплансчетов выбранныйплансчетов "+"датагод датамесяц датачисло заголовоксистемы значениевстроку значениеизстроки каталогиб каталогпользователя "+"кодсимв конгода конецпериодаби конецрассчитанногопериодаби конецстандартногоинтервала конквартала конмесяца "+"коннедели лог лог10 максимальноеколичествосубконто названиеинтерфейса названиенабораправ назначитьвид "+"назначитьсчет найтиссылки началопериодаби началостандартногоинтервала начгода начквартала начмесяца "+"начнедели номерднягода номерднянедели номернеделигода обработкаожидания основнойжурналрасчетов "+"основнойплансчетов основнойязык очиститьокносообщений периодстр получитьвремята получитьдатута "+"получитьдокументта получитьзначенияотбора получитьпозициюта получитьпустоезначение получитьта "+"префиксавтонумерации пропись пустоезначение разм разобратьпозициюдокумента рассчитатьрегистрына "+"рассчитатьрегистрыпо симв создатьобъект статусвозврата стрколичествострок сформироватьпозициюдокумента "+"счетпокоду текущеевремя типзначения типзначениястр установитьтана установитьтапо фиксшаблон шаблон ";var v8_global_context_methods="acos asin atan base64значение base64строка cos exp log log10 pow sin sqrt tan xmlзначение xmlстрока "+"xmlтип xmlтипзнч активноеокно безопасныйрежим безопасныйрежимразделенияданных булево ввестидату ввестизначение "+"ввестистроку ввестичисло возможностьчтенияxml вопрос восстановитьзначение врег выгрузитьжурналрегистрации "+"выполнитьобработкуоповещения выполнитьпроверкуправдоступа вычислить год данныеформывзначение дата день деньгода "+"деньнедели добавитьмесяц заблокироватьданныедляредактирования заблокироватьработупользователя завершитьработусистемы "+"загрузитьвнешнююкомпоненту закрытьсправку записатьjson записатьxml записатьдатуjson записьжурналарегистрации "+"заполнитьзначениясвойств запроситьразрешениепользователя запуститьприложение запуститьсистему зафиксироватьтранзакцию "+"значениевданныеформы значениевстрокувнутр значениевфайл значениезаполнено значениеизстрокивнутр значениеизфайла "+"изxmlтипа импортмоделиxdto имякомпьютера имяпользователя инициализироватьпредопределенныеданные информацияобошибке "+"каталогбиблиотекимобильногоустройства каталогвременныхфайлов каталогдокументов каталогпрограммы кодироватьстроку "+"кодлокализацииинформационнойбазы кодсимвола командасистемы конецгода конецдня конецквартала конецмесяца конецминуты "+"конецнедели конецчаса конфигурациябазыданныхизмененадинамически конфигурацияизменена копироватьданныеформы "+"копироватьфайл краткоепредставлениеошибки лев макс местноевремя месяц мин минута монопольныйрежим найти "+"найтинедопустимыесимволыxml найтиокнопонавигационнойссылке найтипомеченныенаудаление найтипоссылкам найтифайлы "+"началогода началодня началоквартала началомесяца началоминуты началонедели началочаса начатьзапросразрешенияпользователя "+"начатьзапускприложения начатькопированиефайла начатьперемещениефайла начатьподключениевнешнейкомпоненты "+"начатьподключениерасширенияработыскриптографией начатьподключениерасширенияработысфайлами начатьпоискфайлов "+"начатьполучениекаталогавременныхфайлов начатьполучениекаталогадокументов начатьполучениерабочегокаталогаданныхпользователя "+"начатьполучениефайлов начатьпомещениефайла начатьпомещениефайлов начатьсозданиедвоичныхданныхизфайла начатьсозданиекаталога "+"начатьтранзакцию начатьудалениефайлов начатьустановкувнешнейкомпоненты начатьустановкурасширенияработыскриптографией "+"начатьустановкурасширенияработысфайлами неделягода необходимостьзавершениясоединения номерсеансаинформационнойбазы "+"номерсоединенияинформационнойбазы нрег нстр обновитьинтерфейс обновитьнумерациюобъектов обновитьповторноиспользуемыезначения "+"обработкапрерыванияпользователя объединитьфайлы окр описаниеошибки оповестить оповеститьобизменении "+"отключитьобработчикзапросанастроекклиенталицензирования отключитьобработчикожидания отключитьобработчикоповещения "+"открытьзначение открытьиндекссправки открытьсодержаниесправки открытьсправку открытьформу открытьформумодально "+"отменитьтранзакцию очиститьжурналрегистрации очиститьнастройкипользователя очиститьсообщения параметрыдоступа "+"перейтипонавигационнойссылке переместитьфайл подключитьвнешнююкомпоненту "+"подключитьобработчикзапросанастроекклиенталицензирования подключитьобработчикожидания подключитьобработчикоповещения "+"подключитьрасширениеработыскриптографией подключитьрасширениеработысфайлами подробноепредставлениеошибки "+"показатьвводдаты показатьвводзначения показатьвводстроки показатьвводчисла показатьвопрос показатьзначение "+"показатьинформациюобошибке показатьнакарте показатьоповещениепользователя показатьпредупреждение полноеимяпользователя "+"получитьcomобъект получитьxmlтип получитьадреспоместоположению получитьблокировкусеансов получитьвремязавершенияспящегосеанса "+"получитьвремязасыпанияпассивногосеанса получитьвремяожиданияблокировкиданных получитьданныевыбора "+"получитьдополнительныйпараметрклиенталицензирования получитьдопустимыекодылокализации получитьдопустимыечасовыепояса "+"получитьзаголовокклиентскогоприложения получитьзаголовоксистемы получитьзначенияотборажурналарегистрации "+"получитьидентификаторконфигурации получитьизвременногохранилища получитьимявременногофайла "+"получитьимяклиенталицензирования получитьинформациюэкрановклиента получитьиспользованиежурналарегистрации "+"получитьиспользованиесобытияжурналарегистрации получитькраткийзаголовокприложения получитьмакетоформления "+"получитьмаскувсефайлы получитьмаскувсефайлыклиента получитьмаскувсефайлысервера получитьместоположениепоадресу "+"получитьминимальнуюдлинупаролейпользователей получитьнавигационнуюссылку получитьнавигационнуюссылкуинформационнойбазы "+"получитьобновлениеконфигурациибазыданных получитьобновлениепредопределенныхданныхинформационнойбазы получитьобщиймакет "+"получитьобщуюформу получитьокна получитьоперативнуюотметкувремени получитьотключениебезопасногорежима "+"получитьпараметрыфункциональныхопцийинтерфейса получитьполноеимяпредопределенногозначения "+"получитьпредставлениянавигационныхссылок получитьпроверкусложностипаролейпользователей получитьразделительпути "+"получитьразделительпутиклиента получитьразделительпутисервера получитьсеансыинформационнойбазы "+"получитьскоростьклиентскогосоединения получитьсоединенияинформационнойбазы получитьсообщенияпользователю "+"получитьсоответствиеобъектаиформы получитьсоставстандартногоинтерфейсаodata получитьструктурухранениябазыданных "+"получитьтекущийсеансинформационнойбазы получитьфайл получитьфайлы получитьформу получитьфункциональнуюопцию "+"получитьфункциональнуюопциюинтерфейса получитьчасовойпоясинформационнойбазы пользователиос поместитьвовременноехранилище "+"поместитьфайл поместитьфайлы прав праводоступа предопределенноезначение представлениекодалокализации представлениепериода "+"представлениеправа представлениеприложения представлениесобытияжурналарегистрации представлениечасовогопояса предупреждение "+"прекратитьработусистемы привилегированныйрежим продолжитьвызов прочитатьjson прочитатьxml прочитатьдатуjson пустаястрока "+"рабочийкаталогданныхпользователя разблокироватьданныедляредактирования разделитьфайл разорватьсоединениесвнешнимисточникомданных "+"раскодироватьстроку рольдоступна секунда сигнал символ скопироватьжурналрегистрации смещениелетнеговремени "+"смещениестандартноговремени соединитьбуферыдвоичныхданных создатькаталог создатьфабрикуxdto сокрл сокрлп сокрп сообщить "+"состояние сохранитьзначение сохранитьнастройкипользователя сред стрдлина стрзаканчиваетсяна стрзаменить стрнайти стрначинаетсяс "+"строка строкасоединенияинформационнойбазы стрполучитьстроку стрразделить стрсоединить стрсравнить стрчисловхождений "+"стрчислострок стршаблон текущаядата текущаядатасеанса текущаяуниверсальнаядата текущаяуниверсальнаядатавмиллисекундах "+"текущийвариантинтерфейсаклиентскогоприложения текущийвариантосновногошрифтаклиентскогоприложения текущийкодлокализации "+"текущийрежимзапуска текущийязык текущийязыксистемы тип типзнч транзакцияактивна трег удалитьданныеинформационнойбазы "+"удалитьизвременногохранилища удалитьобъекты удалитьфайлы универсальноевремя установитьбезопасныйрежим "+"установитьбезопасныйрежимразделенияданных установитьблокировкусеансов установитьвнешнююкомпоненту "+"установитьвремязавершенияспящегосеанса установитьвремязасыпанияпассивногосеанса установитьвремяожиданияблокировкиданных "+"установитьзаголовокклиентскогоприложения установитьзаголовоксистемы установитьиспользованиежурналарегистрации "+"установитьиспользованиесобытияжурналарегистрации установитькраткийзаголовокприложения "+"установитьминимальнуюдлинупаролейпользователей установитьмонопольныйрежим установитьнастройкиклиенталицензирования "+"установитьобновлениепредопределенныхданныхинформационнойбазы установитьотключениебезопасногорежима "+"установитьпараметрыфункциональныхопцийинтерфейса установитьпривилегированныйрежим "+"установитьпроверкусложностипаролейпользователей установитьрасширениеработыскриптографией "+"установитьрасширениеработысфайлами установитьсоединениесвнешнимисточникомданных установитьсоответствиеобъектаиформы "+"установитьсоставстандартногоинтерфейсаodata установитьчасовойпоясинформационнойбазы установитьчасовойпояссеанса "+"формат цел час часовойпояс часовойпояссеанса число числопрописью этоадресвременногохранилища ";var v8_global_context_property="wsссылки библиотекакартинок библиотекамакетовоформлениякомпоновкиданных библиотекастилей бизнеспроцессы "+"внешниеисточникиданных внешниеобработки внешниеотчеты встроенныепокупки главныйинтерфейс главныйстиль "+"документы доставляемыеуведомления журналыдокументов задачи информацияобинтернетсоединении использованиерабочейдаты "+"историяработыпользователя константы критерииотбора метаданные обработки отображениерекламы отправкадоставляемыхуведомлений "+"отчеты панельзадачос параметрзапуска параметрысеанса перечисления планывидоврасчета планывидовхарактеристик "+"планыобмена планысчетов полнотекстовыйпоиск пользователиинформационнойбазы последовательности проверкавстроенныхпокупок "+"рабочаядата расширенияконфигурации регистрыбухгалтерии регистрынакопления регистрырасчета регистрысведений "+"регламентныезадания сериализаторxdto справочники средствагеопозиционирования средствакриптографии средствамультимедиа "+"средстваотображениярекламы средствапочты средствателефонии фабрикаxdto файловыепотоки фоновыезадания хранилищанастроек "+"хранилищевариантовотчетов хранилищенастроекданныхформ хранилищеобщихнастроек хранилищепользовательскихнастроекдинамическихсписков "+"хранилищепользовательскихнастроекотчетов хранилищесистемныхнастроек ";var BUILTIN=v7_system_constants+v7_global_context_methods+v8_global_context_methods+v8_global_context_property;var v8_system_sets_of_values="webцвета windowsцвета windowsшрифты библиотекакартинок рамкистиля символы цветастиля шрифтыстиля ";var v8_system_enums_interface="автоматическоесохранениеданныхформывнастройках автонумерациявформе автораздвижениесерий "+"анимациядиаграммы вариантвыравниванияэлементовизаголовков вариантуправлениявысотойтаблицы "+"вертикальнаяпрокруткаформы вертикальноеположение вертикальноеположениеэлемента видгруппыформы "+"виддекорацииформы виддополненияэлементаформы видизмененияданных видкнопкиформы видпереключателя "+"видподписейкдиаграмме видполяформы видфлажка влияниеразмеранапузырекдиаграммы горизонтальноеположение "+"горизонтальноеположениеэлемента группировкаколонок группировкаподчиненныхэлементовформы "+"группыиэлементы действиеперетаскивания дополнительныйрежимотображения допустимыедействияперетаскивания "+"интервалмеждуэлементамиформы использованиевывода использованиеполосыпрокрутки "+"используемоезначениеточкибиржевойдиаграммы историявыборапривводе источникзначенийоситочекдиаграммы "+"источникзначенияразмерапузырькадиаграммы категориягруппыкоманд максимумсерий начальноеотображениедерева "+"начальноеотображениесписка обновлениетекстаредактирования ориентациядендрограммы ориентациядиаграммы "+"ориентацияметокдиаграммы ориентацияметоксводнойдиаграммы ориентацияэлементаформы отображениевдиаграмме "+"отображениевлегендедиаграммы отображениегруппыкнопок отображениезаголовкашкалыдиаграммы "+"отображениезначенийсводнойдиаграммы отображениезначенияизмерительнойдиаграммы "+"отображениеинтерваладиаграммыганта отображениекнопки отображениекнопкивыбора отображениеобсужденийформы "+"отображениеобычнойгруппы отображениеотрицательныхзначенийпузырьковойдиаграммы отображениепанелипоиска "+"отображениеподсказки отображениепредупрежденияприредактировании отображениеразметкиполосырегулирования "+"отображениестраницформы отображениетаблицы отображениетекстазначениядиаграммыганта "+"отображениеуправленияобычнойгруппы отображениефигурыкнопки палитрацветовдиаграммы поведениеобычнойгруппы "+"поддержкамасштабадендрограммы поддержкамасштабадиаграммыганта поддержкамасштабасводнойдиаграммы "+"поисквтаблицепривводе положениезаголовкаэлементаформы положениекартинкикнопкиформы "+"положениекартинкиэлементаграфическойсхемы положениекоманднойпанелиформы положениекоманднойпанелиэлементаформы "+"положениеопорнойточкиотрисовки положениеподписейкдиаграмме положениеподписейшкалызначенийизмерительнойдиаграммы "+"положениесостоянияпросмотра положениестрокипоиска положениетекстасоединительнойлинии положениеуправленияпоиском "+"положениешкалывремени порядокотображенияточекгоризонтальнойгистограммы порядоксерийвлегендедиаграммы "+"размеркартинки расположениезаголовкашкалыдиаграммы растягиваниеповертикалидиаграммыганта "+"режимавтоотображениясостояния режимвводастроктаблицы режимвыборанезаполненного режимвыделениядаты "+"режимвыделениястрокитаблицы режимвыделениятаблицы режимизмененияразмера режимизменениясвязанногозначения "+"режимиспользованиядиалогапечати режимиспользованияпараметракоманды режиммасштабированияпросмотра "+"режимосновногоокнаклиентскогоприложения режимоткрытияокнаформы режимотображениявыделения "+"режимотображениягеографическойсхемы режимотображениязначенийсерии режимотрисовкисеткиграфическойсхемы "+"режимполупрозрачностидиаграммы режимпробеловдиаграммы режимразмещениянастранице режимредактированияколонки "+"режимсглаживаниядиаграммы режимсглаживанияиндикатора режимсписказадач сквозноевыравнивание "+"сохранениеданныхформывнастройках способзаполнениятекстазаголовкашкалыдиаграммы "+"способопределенияограничивающегозначениядиаграммы стандартнаягруппакоманд стандартноеоформление "+"статусоповещенияпользователя стильстрелки типаппроксимациилиниитрендадиаграммы типдиаграммы "+"типединицышкалывремени типимпортасерийслоягеографическойсхемы типлиниигеографическойсхемы типлиниидиаграммы "+"типмаркерагеографическойсхемы типмаркерадиаграммы типобластиоформления "+"типорганизацииисточникаданныхгеографическойсхемы типотображениясериислоягеографическойсхемы "+"типотображенияточечногообъектагеографическойсхемы типотображенияшкалыэлементалегендыгеографическойсхемы "+"типпоискаобъектовгеографическойсхемы типпроекциигеографическойсхемы типразмещенияизмерений "+"типразмещенияреквизитовизмерений типрамкиэлементауправления типсводнойдиаграммы "+"типсвязидиаграммыганта типсоединениязначенийпосериямдиаграммы типсоединенияточекдиаграммы "+"типсоединительнойлинии типстороныэлементаграфическойсхемы типформыотчета типшкалырадарнойдиаграммы "+"факторлиниитрендадиаграммы фигуракнопки фигурыграфическойсхемы фиксациявтаблице форматдняшкалывремени "+"форматкартинки ширинаподчиненныхэлементовформы ";var v8_system_enums_objects_properties="виддвижениябухгалтерии виддвижениянакопления видпериодарегистрарасчета видсчета видточкимаршрутабизнеспроцесса "+"использованиеагрегатарегистранакопления использованиегруппиэлементов использованиережимапроведения "+"использованиесреза периодичностьагрегатарегистранакопления режимавтовремя режимзаписидокумента режимпроведениядокумента ";var v8_system_enums_exchange_plans="авторегистрацияизменений допустимыйномерсообщения отправкаэлементаданных получениеэлементаданных ";var v8_system_enums_tabular_document="использованиерасшифровкитабличногодокумента ориентациястраницы положениеитоговколоноксводнойтаблицы "+"положениеитоговстроксводнойтаблицы положениетекстаотносительнокартинки расположениезаголовкагруппировкитабличногодокумента "+"способчтениязначенийтабличногодокумента типдвустороннейпечати типзаполненияобластитабличногодокумента "+"типкурсоровтабличногодокумента типлиниирисункатабличногодокумента типлинииячейкитабличногодокумента "+"типнаправленияпереходатабличногодокумента типотображениявыделениятабличногодокумента типотображениялинийсводнойтаблицы "+"типразмещениятекстатабличногодокумента типрисункатабличногодокумента типсмещениятабличногодокумента "+"типузоратабличногодокумента типфайлатабличногодокумента точностьпечати чередованиерасположениястраниц ";var v8_system_enums_sheduler="отображениевремениэлементовпланировщика ";var v8_system_enums_formatted_document="типфайлаформатированногодокумента ";var v8_system_enums_query="обходрезультатазапроса типзаписизапроса ";var v8_system_enums_report_builder="видзаполнениярасшифровкипостроителяотчета типдобавленияпредставлений типизмеренияпостроителяотчета типразмещенияитогов ";var v8_system_enums_files="доступкфайлу режимдиалогавыборафайла режимоткрытияфайла ";var v8_system_enums_query_builder="типизмеренияпостроителязапроса ";var v8_system_enums_data_analysis="видданныханализа методкластеризации типединицыинтервалавременианализаданных типзаполнениятаблицырезультатаанализаданных "+"типиспользованиячисловыхзначенийанализаданных типисточникаданныхпоискаассоциаций типколонкианализаданныхдереворешений "+"типколонкианализаданныхкластеризация типколонкианализаданныхобщаястатистика типколонкианализаданныхпоискассоциаций "+"типколонкианализаданныхпоискпоследовательностей типколонкимоделипрогноза типмерырасстоянияанализаданных "+"типотсеченияправилассоциации типполяанализаданных типстандартизациианализаданных типупорядочиванияправилассоциациианализаданных "+"типупорядочиванияшаблоновпоследовательностейанализаданных типупрощениядереварешений ";var v8_system_enums_xml_json_xs_dom_xdto_ws="wsнаправлениепараметра вариантxpathxs вариантзаписидатыjson вариантпростоготипаxs видгруппымоделиxs видфасетаxdto "+"действиепостроителяdom завершенностьпростоготипаxs завершенностьсоставноготипаxs завершенностьсхемыxs запрещенныеподстановкиxs "+"исключениягруппподстановкиxs категорияиспользованияатрибутаxs категорияограниченияидентичностиxs категорияограниченияпространствименxs "+"методнаследованияxs модельсодержимогоxs назначениетипаxml недопустимыеподстановкиxs обработкапробельныхсимволовxs обработкасодержимогоxs "+"ограничениезначенияxs параметрыотбораузловdom переносстрокjson позициявдокументеdom пробельныесимволыxml типатрибутаxml типзначенияjson "+"типканоническогоxml типкомпонентыxs типпроверкиxml типрезультатаdomxpath типузлаdom типузлаxml формаxml формапредставленияxs "+"форматдатыjson экранированиесимволовjson ";var v8_system_enums_data_composition_system="видсравнениякомпоновкиданных действиеобработкирасшифровкикомпоновкиданных направлениесортировкикомпоновкиданных "+"расположениевложенныхэлементоврезультатакомпоновкиданных расположениеитоговкомпоновкиданных расположениегруппировкикомпоновкиданных "+"расположениеполейгруппировкикомпоновкиданных расположениеполякомпоновкиданных расположениереквизитовкомпоновкиданных "+"расположениересурсовкомпоновкиданных типбухгалтерскогоостаткакомпоновкиданных типвыводатекстакомпоновкиданных "+"типгруппировкикомпоновкиданных типгруппыэлементовотборакомпоновкиданных типдополненияпериодакомпоновкиданных "+"типзаголовкаполейкомпоновкиданных типмакетагруппировкикомпоновкиданных типмакетаобластикомпоновкиданных типостаткакомпоновкиданных "+"типпериодакомпоновкиданных типразмещениятекстакомпоновкиданных типсвязинаборовданныхкомпоновкиданных типэлементарезультатакомпоновкиданных "+"расположениелегендыдиаграммыкомпоновкиданных типпримененияотборакомпоновкиданных режимотображенияэлементанастройкикомпоновкиданных "+"режимотображениянастроеккомпоновкиданных состояниеэлементанастройкикомпоновкиданных способвосстановлениянастроеккомпоновкиданных "+"режимкомпоновкирезультата использованиепараметракомпоновкиданных автопозицияресурсовкомпоновкиданных "+"вариантиспользованиягруппировкикомпоновкиданных расположениересурсоввдиаграммекомпоновкиданных фиксациякомпоновкиданных "+"использованиеусловногооформлениякомпоновкиданных ";var v8_system_enums_email="важностьинтернетпочтовогосообщения обработкатекстаинтернетпочтовогосообщения способкодированияинтернетпочтовоговложения "+"способкодированиянеasciiсимволовинтернетпочтовогосообщения типтекстапочтовогосообщения протоколинтернетпочты "+"статусразборапочтовогосообщения ";var v8_system_enums_logbook="режимтранзакциизаписижурналарегистрации статустранзакциизаписижурналарегистрации уровеньжурналарегистрации ";var v8_system_enums_cryptography="расположениехранилищасертификатовкриптографии режимвключениясертификатовкриптографии режимпроверкисертификатакриптографии "+"типхранилищасертификатовкриптографии ";var v8_system_enums_zip="кодировкаименфайловвzipфайле методсжатияzip методшифрованияzip режимвосстановленияпутейфайловzip режимобработкиподкаталоговzip "+"режимсохраненияпутейzip уровеньсжатияzip ";var v8_system_enums_other="звуковоеоповещение направлениепереходакстроке позициявпотоке порядокбайтов режимблокировкиданных режимуправленияблокировкойданных "+"сервисвстроенныхпокупок состояниефоновогозадания типподписчикадоставляемыхуведомлений уровеньиспользованиязащищенногосоединенияftp ";var v8_system_enums_request_schema="направлениепорядкасхемызапроса типдополненияпериодамисхемызапроса типконтрольнойточкисхемызапроса типобъединениясхемызапроса "+"типпараметрадоступнойтаблицысхемызапроса типсоединениясхемызапроса ";var v8_system_enums_properties_of_metadata_objects="httpметод автоиспользованиеобщегореквизита автопрефиксномеразадачи вариантвстроенногоязыка видиерархии видрегистранакопления "+"видтаблицывнешнегоисточникаданных записьдвиженийприпроведении заполнениепоследовательностей индексирование "+"использованиебазыпланавидоврасчета использованиебыстроговыбора использованиеобщегореквизита использованиеподчинения "+"использованиеполнотекстовогопоиска использованиеразделяемыхданныхобщегореквизита использованиереквизита "+"назначениеиспользованияприложения назначениерасширенияконфигурации направлениепередачи обновлениепредопределенныхданных "+"оперативноепроведение основноепредставлениевидарасчета основноепредставлениевидахарактеристики основноепредставлениезадачи "+"основноепредставлениепланаобмена основноепредставлениесправочника основноепредставлениесчета перемещениеграницыприпроведении "+"периодичностьномерабизнеспроцесса периодичностьномерадокумента периодичностьрегистрарасчета периодичностьрегистрасведений "+"повторноеиспользованиевозвращаемыхзначений полнотекстовыйпоискпривводепостроке принадлежностьобъекта проведение "+"разделениеаутентификацииобщегореквизита разделениеданныхобщегореквизита разделениерасширенийконфигурацииобщегореквизита "+"режимавтонумерацииобъектов режимзаписирегистра режимиспользованиямодальности "+"режимиспользованиясинхронныхвызововрасширенийплатформыивнешнихкомпонент режимповторногоиспользованиясеансов "+"режимполученияданныхвыборапривводепостроке режимсовместимости режимсовместимостиинтерфейса "+"режимуправленияблокировкойданныхпоумолчанию сериикодовпланавидовхарактеристик сериикодовпланасчетов "+"сериикодовсправочника созданиепривводе способвыбора способпоискастрокипривводепостроке способредактирования "+"типданныхтаблицывнешнегоисточникаданных типкодапланавидоврасчета типкодасправочника типмакета типномерабизнеспроцесса "+"типномерадокумента типномеразадачи типформы удалениедвижений ";var v8_system_enums_differents="важностьпроблемыприменениярасширенияконфигурации вариантинтерфейсаклиентскогоприложения вариантмасштабаформклиентскогоприложения "+"вариантосновногошрифтаклиентскогоприложения вариантстандартногопериода вариантстандартнойдатыначала видграницы видкартинки "+"видотображенияполнотекстовогопоиска видрамки видсравнения видцвета видчисловогозначения видшрифта допустимаядлина допустимыйзнак "+"использованиеbyteordermark использованиеметаданныхполнотекстовогопоиска источникрасширенийконфигурации клавиша кодвозвратадиалога "+"кодировкаxbase кодировкатекста направлениепоиска направлениесортировки обновлениепредопределенныхданных обновлениеприизмененииданных "+"отображениепанелиразделов проверказаполнения режимдиалогавопрос режимзапускаклиентскогоприложения режимокругления режимоткрытияформприложения "+"режимполнотекстовогопоиска скоростьклиентскогосоединения состояниевнешнегоисточникаданных состояниеобновленияконфигурациибазыданных "+"способвыборасертификатаwindows способкодированиястроки статуссообщения типвнешнейкомпоненты типплатформы типповеденияклавишиenter "+"типэлементаинформацииовыполненииобновленияконфигурациибазыданных уровеньизоляциитранзакций хешфункция частидаты";var CLASS=v8_system_sets_of_values+v8_system_enums_interface+v8_system_enums_objects_properties+v8_system_enums_exchange_plans+v8_system_enums_tabular_document+v8_system_enums_sheduler+v8_system_enums_formatted_document+v8_system_enums_query+v8_system_enums_report_builder+v8_system_enums_files+v8_system_enums_query_builder+v8_system_enums_data_analysis+v8_system_enums_xml_json_xs_dom_xdto_ws+v8_system_enums_data_composition_system+v8_system_enums_email+v8_system_enums_logbook+v8_system_enums_cryptography+v8_system_enums_zip+v8_system_enums_other+v8_system_enums_request_schema+v8_system_enums_properties_of_metadata_objects+v8_system_enums_differents;var v8_shared_object="comобъект ftpсоединение httpзапрос httpсервисответ httpсоединение wsопределения wsпрокси xbase анализданных аннотацияxs "+"блокировкаданных буфердвоичныхданных включениеxs выражениекомпоновкиданных генераторслучайныхчисел географическаясхема "+"географическиекоординаты графическаясхема группамоделиxs данныерасшифровкикомпоновкиданных двоичныеданные дендрограмма "+"диаграмма диаграммаганта диалогвыборафайла диалогвыборацвета диалогвыборашрифта диалограсписаниярегламентногозадания "+"диалогредактированиястандартногопериода диапазон документdom документhtml документацияxs доставляемоеуведомление "+"записьdom записьfastinfoset записьhtml записьjson записьxml записьzipфайла записьданных записьтекста записьузловdom "+"запрос защищенноесоединениеopenssl значенияполейрасшифровкикомпоновкиданных извлечениетекста импортxs интернетпочта "+"интернетпочтовоесообщение интернетпочтовыйпрофиль интернетпрокси интернетсоединение информациядляприложенияxs "+"использованиеатрибутаxs использованиесобытияжурналарегистрации источникдоступныхнастроеккомпоновкиданных "+"итераторузловdom картинка квалификаторыдаты квалификаторыдвоичныхданных квалификаторыстроки квалификаторычисла "+"компоновщикмакетакомпоновкиданных компоновщикнастроеккомпоновкиданных конструктормакетаоформлениякомпоновкиданных "+"конструкторнастроеккомпоновкиданных конструкторформатнойстроки линия макеткомпоновкиданных макетобластикомпоновкиданных "+"макетоформлениякомпоновкиданных маскаxs менеджеркриптографии наборсхемxml настройкикомпоновкиданных настройкисериализацииjson "+"обработкакартинок обработкарасшифровкикомпоновкиданных обходдереваdom объявлениеатрибутаxs объявлениенотацииxs "+"объявлениеэлементаxs описаниеиспользованиясобытиядоступжурналарегистрации "+"описаниеиспользованиясобытияотказвдоступежурналарегистрации описаниеобработкирасшифровкикомпоновкиданных "+"описаниепередаваемогофайла описаниетипов определениегруппыатрибутовxs определениегруппымоделиxs "+"определениеограниченияидентичностиxs определениепростоготипаxs определениесоставноготипаxs определениетипадокументаdom "+"определенияxpathxs отборкомпоновкиданных пакетотображаемыхдокументов параметрвыбора параметркомпоновкиданных "+"параметрызаписиjson параметрызаписиxml параметрычтенияxml переопределениеxs планировщик полеанализаданных "+"полекомпоновкиданных построительdom построительзапроса построительотчета построительотчетаанализаданных "+"построительсхемxml поток потоквпамяти почта почтовоесообщение преобразованиеxsl преобразованиекканоническомуxml "+"процессорвыводарезультатакомпоновкиданныхвколлекциюзначений процессорвыводарезультатакомпоновкиданныхвтабличныйдокумент "+"процессоркомпоновкиданных разыменовательпространствименdom рамка расписаниерегламентногозадания расширенноеимяxml "+"результатчтенияданных своднаядиаграмма связьпараметравыбора связьпотипу связьпотипукомпоновкиданных сериализаторxdto "+"сертификатклиентаwindows сертификатклиентафайл сертификаткриптографии сертификатыудостоверяющихцентровwindows "+"сертификатыудостоверяющихцентровфайл сжатиеданных системнаяинформация сообщениепользователю сочетаниеклавиш "+"сравнениезначений стандартнаядатаначала стандартныйпериод схемаxml схемакомпоновкиданных табличныйдокумент "+"текстовыйдокумент тестируемоеприложение типданныхxml уникальныйидентификатор фабрикаxdto файл файловыйпоток "+"фасетдлиныxs фасетколичестваразрядовдробнойчастиxs фасетмаксимальноговключающегозначенияxs "+"фасетмаксимальногоисключающегозначенияxs фасетмаксимальнойдлиныxs фасетминимальноговключающегозначенияxs "+"фасетминимальногоисключающегозначенияxs фасетминимальнойдлиныxs фасетобразцаxs фасетобщегоколичестваразрядовxs "+"фасетперечисленияxs фасетпробельныхсимволовxs фильтрузловdom форматированнаястрока форматированныйдокумент "+"фрагментxs хешированиеданных хранилищезначения цвет чтениеfastinfoset чтениеhtml чтениеjson чтениеxml чтениеzipфайла "+"чтениеданных чтениетекста чтениеузловdom шрифт элементрезультатакомпоновкиданных ";var v8_universal_collection="comsafearray деревозначений массив соответствие списокзначений структура таблицазначений фиксированнаяструктура "+"фиксированноесоответствие фиксированныймассив ";var TYPE=v8_shared_object+v8_universal_collection;var LITERAL="null истина ложь неопределено";var NUMBERS=hljs.inherit(hljs.NUMBER_MODE);var STRINGS={className:"string",begin:'"|\\|',end:'"|$',contains:[{begin:'""'}]};var DATE={begin:"'",end:"'",excludeBegin:true,excludeEnd:true,contains:[{className:"number",begin:"\\d{4}([\\.\\\\/:-]?\\d{2}){0,5}"}]};var COMMENTS=hljs.inherit(hljs.C_LINE_COMMENT_MODE);var META={className:"meta",lexemes:UNDERSCORE_IDENT_RE,begin:"#|&",end:"$",keywords:{"meta-keyword":KEYWORD+METAKEYWORD},contains:[COMMENTS]};var SYMBOL={className:"symbol",begin:"~",end:";|:",excludeEnd:true};var FUNCTION={className:"function",lexemes:UNDERSCORE_IDENT_RE,variants:[{begin:"процедура|функция",end:"\\)",keywords:"процедура функция"},{begin:"конецпроцедуры|конецфункции",keywords:"конецпроцедуры конецфункции"}],contains:[{begin:"\\(",end:"\\)",endsParent:true,contains:[{className:"params",lexemes:UNDERSCORE_IDENT_RE,begin:UNDERSCORE_IDENT_RE,end:",",excludeEnd:true,endsWithParent:true,keywords:{keyword:"знач",literal:LITERAL},contains:[NUMBERS,STRINGS,DATE]},COMMENTS]},hljs.inherit(hljs.TITLE_MODE,{begin:UNDERSCORE_IDENT_RE})]};return{case_insensitive:true,lexemes:UNDERSCORE_IDENT_RE,keywords:{keyword:KEYWORD,built_in:BUILTIN,class:CLASS,type:TYPE,literal:LITERAL},contains:[META,FUNCTION,COMMENTS,SYMBOL,NUMBERS,STRINGS,DATE]}});hljs.registerLanguage("asciidoc",function(hljs){return{aliases:["adoc"],contains:[hljs.COMMENT("^/{4,}\\n","\\n/{4,}$",{relevance:10}),hljs.COMMENT("^//","$",{relevance:0}),{className:"title",begin:"^\\.\\w.*$"},{begin:"^[=\\*]{4,}\\n",end:"\\n^[=\\*]{4,}$",relevance:10},{className:"section",relevance:10,variants:[{begin:"^(={1,5}) .+?( \\1)?$"},{begin:"^[^\\[\\]\\n]+?\\n[=\\-~\\^\\+]{2,}$"}]},{className:"meta",begin:"^:.+?:",end:"\\s",excludeEnd:true,relevance:10},{className:"meta",begin:"^\\[.+?\\]$",relevance:0},{className:"quote",begin:"^_{4,}\\n",end:"\\n_{4,}$",relevance:10},{className:"code",begin:"^[\\-\\.]{4,}\\n",end:"\\n[\\-\\.]{4,}$",relevance:10},{begin:"^\\+{4,}\\n",end:"\\n\\+{4,}$",contains:[{begin:"<",end:">",subLanguage:"xml",relevance:0}],relevance:10},{className:"bullet",begin:"^(\\*+|\\-+|\\.+|[^\\n]+?::)\\s+"},{className:"symbol",begin:"^(NOTE|TIP|IMPORTANT|WARNING|CAUTION):\\s+",relevance:10},{className:"strong",begin:"\\B\\*(?![\\*\\s])",end:"(\\n{2}|\\*)",contains:[{begin:"\\\\*\\w",relevance:0}]},{className:"emphasis",begin:"\\B'(?!['\\s])",end:"(\\n{2}|')",contains:[{begin:"\\\\'\\w",relevance:0}],relevance:0},{className:"emphasis",begin:"_(?![_\\s])",end:"(\\n{2}|_)",relevance:0},{className:"string",variants:[{begin:"``.+?''"},{begin:"`.+?'"}]},{className:"code",begin:"(`.+?`|\\+.+?\\+)",relevance:0},{className:"code",begin:"^[ \\t]",end:"$",relevance:0},{begin:"^'{3,}[ \\t]*$",relevance:10},{begin:"(link:)?(http|https|ftp|file|irc|image:?):\\S+\\[.*?\\]",returnBegin:true,contains:[{begin:"(link|image:?):",relevance:0},{className:"link",begin:"\\w",end:"[^\\[]+",relevance:0},{className:"string",begin:"\\[",end:"\\]",excludeBegin:true,excludeEnd:true,relevance:0}],relevance:10}]}});hljs.registerLanguage("flix",function(hljs){var CHAR={className:"string",begin:/'(.|\\[xXuU][a-zA-Z0-9]+)'/};var STRING={className:"string",variants:[{begin:'"',end:'"'}]};var NAME={className:"title",begin:/[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/};var METHOD={className:"function",beginKeywords:"def",end:/[:={\[(\n;]/,excludeEnd:true,contains:[NAME]};return{keywords:{literal:"true false",keyword:"case class def else enum if impl import in lat rel index let match namespace switch type yield with"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,CHAR,STRING,METHOD,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("python",function(hljs){var KEYWORDS={keyword:"and elif is global as in if from raise for except finally print import pass return "+"exec else break not with class assert yield try while continue del or def lambda "+"async await nonlocal|10",built_in:"Ellipsis NotImplemented",literal:"False None True"};var PROMPT={className:"meta",begin:/^(>>>|\.\.\.) /};var SUBST={className:"subst",begin:/\{/,end:/\}/,keywords:KEYWORDS,illegal:/#/};var LITERAL_BRACKET={begin:/\{\{/,relevance:0};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:/(u|b)?r?'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT],relevance:10},{begin:/(u|b)?r?"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT],relevance:10},{begin:/(fr|rf|f)'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT,LITERAL_BRACKET,SUBST]},{begin:/(fr|rf|f)"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT,LITERAL_BRACKET,SUBST]},{begin:/(u|r|ur)'/,end:/'/,relevance:10},{begin:/(u|r|ur)"/,end:/"/,relevance:10},{begin:/(b|br)'/,end:/'/},{begin:/(b|br)"/,end:/"/},{begin:/(fr|rf|f)'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE,LITERAL_BRACKET,SUBST]},{begin:/(fr|rf|f)"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,LITERAL_BRACKET,SUBST]},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};var NUMBER={className:"number",relevance:0,variants:[{begin:hljs.BINARY_NUMBER_RE+"[lLjJ]?"},{begin:"\\b(0o[0-7]+)[lLjJ]?"},{begin:hljs.C_NUMBER_RE+"[lLjJ]?"}]};var PARAMS={className:"params",begin:/\(/,end:/\)/,contains:["self",PROMPT,NUMBER,STRING,hljs.HASH_COMMENT_MODE]};SUBST.contains=[STRING,NUMBER,PROMPT];return{aliases:["py","gyp","ipython"],keywords:KEYWORDS,illegal:/(<\/|->|\?)|=>/,contains:[PROMPT,NUMBER,{beginKeywords:"if",relevance:0},STRING,hljs.HASH_COMMENT_MODE,{variants:[{className:"function",beginKeywords:"def"},{className:"class",beginKeywords:"class"}],end:/:/,illegal:/[${=;\n,]/,contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS,{begin:/->/,endsWithParent:true,keywords:"None"}]},{className:"meta",begin:/^[\t ]*@/,end:/$/},{begin:/\b(print|exec)\(/}]}});hljs.registerLanguage("leaf",function(hljs){return{contains:[{className:"function",begin:"#+"+"[A-Za-z_0-9]*"+"\\(",end:" {",returnBegin:true,excludeEnd:true,contains:[{className:"keyword",begin:"#+"},{className:"title",begin:"[A-Za-z_][A-Za-z_0-9]*"},{className:"params",begin:"\\(",end:"\\)",endsParent:true,contains:[{className:"string",begin:'"',end:'"'},{className:"variable",begin:"[A-Za-z_][A-Za-z_0-9]*"}]}]}]}});hljs.registerLanguage("groovy",function(hljs){return{keywords:{literal:"true false null",keyword:"byte short char int long boolean float double void "+"def as in assert trait "+"super this abstract static volatile transient public private protected synchronized final "+"class interface enum if else for while switch case break default continue "+"throw throws try catch finally implements extends new import package return instanceof"},contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",begin:'"""',end:'"""'},{className:"string",begin:"'''",end:"'''"},{className:"string",begin:"\\$/",end:"/\\$",relevance:10},hljs.APOS_STRING_MODE,{className:"regexp",begin:/~?\/[^\/\n]+\//,contains:[hljs.BACKSLASH_ESCAPE]},hljs.QUOTE_STRING_MODE,{className:"meta",begin:"^#!/usr/bin/env",end:"$",illegal:"\n"},hljs.BINARY_NUMBER_MODE,{className:"class",beginKeywords:"class interface trait enum",end:"{",illegal:":",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"},{className:"string",begin:/[^\?]{0}[A-Za-z0-9_$]+ *:/},{begin:/\?/,end:/\:/},{className:"symbol",begin:"^\\s*[A-Za-z0-9_$]+:",relevance:0}],illegal:/#|<\//}});hljs.registerLanguage("armasm",function(hljs){return{case_insensitive:true,aliases:["arm"],lexemes:"\\.?"+hljs.IDENT_RE,keywords:{meta:".2byte .4byte .align .ascii .asciz .balign .byte .code .data .else .end .endif .endm .endr .equ .err .exitm .extern .global .hword .if .ifdef .ifndef .include .irp .long .macro .rept .req .section .set .skip .space .text .word .arm .thumb .code16 .code32 .force_thumb .thumb_func .ltorg "+"ALIAS ALIGN ARM AREA ASSERT ATTR CN CODE CODE16 CODE32 COMMON CP DATA DCB DCD DCDU DCDO DCFD DCFDU DCI DCQ DCQU DCW DCWU DN ELIF ELSE END ENDFUNC ENDIF ENDP ENTRY EQU EXPORT EXPORTAS EXTERN FIELD FILL FUNCTION GBLA GBLL GBLS GET GLOBAL IF IMPORT INCBIN INCLUDE INFO KEEP LCLA LCLL LCLS LTORG MACRO MAP MEND MEXIT NOFP OPT PRESERVE8 PROC QN READONLY RELOC REQUIRE REQUIRE8 RLIST FN ROUT SETA SETL SETS SN SPACE SUBT THUMB THUMBX TTL WHILE WEND ",built_in:"r0 r1 r2 r3 r4 r5 r6 r7 r8 r9 r10 r11 r12 r13 r14 r15 "+"pc lr sp ip sl sb fp "+"a1 a2 a3 a4 v1 v2 v3 v4 v5 v6 v7 v8 f0 f1 f2 f3 f4 f5 f6 f7 "+"p0 p1 p2 p3 p4 p5 p6 p7 p8 p9 p10 p11 p12 p13 p14 p15 "+"c0 c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c11 c12 c13 c14 c15 "+"q0 q1 q2 q3 q4 q5 q6 q7 q8 q9 q10 q11 q12 q13 q14 q15 "+"cpsr_c cpsr_x cpsr_s cpsr_f cpsr_cx cpsr_cxs cpsr_xs cpsr_xsf cpsr_sf cpsr_cxsf "+"spsr_c spsr_x spsr_s spsr_f spsr_cx spsr_cxs spsr_xs spsr_xsf spsr_sf spsr_cxsf "+"s0 s1 s2 s3 s4 s5 s6 s7 s8 s9 s10 s11 s12 s13 s14 s15 "+"s16 s17 s18 s19 s20 s21 s22 s23 s24 s25 s26 s27 s28 s29 s30 s31 "+"d0 d1 d2 d3 d4 d5 d6 d7 d8 d9 d10 d11 d12 d13 d14 d15 "+"d16 d17 d18 d19 d20 d21 d22 d23 d24 d25 d26 d27 d28 d29 d30 d31 "+"{PC} {VAR} {TRUE} {FALSE} {OPT} {CONFIG} {ENDIAN} {CODESIZE} {CPU} {FPU} {ARCHITECTURE} {PCSTOREOFFSET} {ARMASM_VERSION} {INTER} {ROPI} {RWPI} {SWST} {NOSWST} . @"},contains:[{className:"keyword",begin:"\\b("+"adc|"+"(qd?|sh?|u[qh]?)?add(8|16)?|usada?8|(q|sh?|u[qh]?)?(as|sa)x|"+"and|adrl?|sbc|rs[bc]|asr|b[lx]?|blx|bxj|cbn?z|tb[bh]|bic|"+"bfc|bfi|[su]bfx|bkpt|cdp2?|clz|clrex|cmp|cmn|cpsi[ed]|cps|"+"setend|dbg|dmb|dsb|eor|isb|it[te]{0,3}|lsl|lsr|ror|rrx|"+"ldm(([id][ab])|f[ds])?|ldr((s|ex)?[bhd])?|movt?|mvn|mra|mar|"+"mul|[us]mull|smul[bwt][bt]|smu[as]d|smmul|smmla|"+"mla|umlaal|smlal?([wbt][bt]|d)|mls|smlsl?[ds]|smc|svc|sev|"+"mia([bt]{2}|ph)?|mrr?c2?|mcrr2?|mrs|msr|orr|orn|pkh(tb|bt)|rbit|"+"rev(16|sh)?|sel|[su]sat(16)?|nop|pop|push|rfe([id][ab])?|"+"stm([id][ab])?|str(ex)?[bhd]?|(qd?)?sub|(sh?|q|u[qh]?)?sub(8|16)|"+"[su]xt(a?h|a?b(16)?)|srs([id][ab])?|swpb?|swi|smi|tst|teq|"+"wfe|wfi|yield"+")"+"(eq|ne|cs|cc|mi|pl|vs|vc|hi|ls|ge|lt|gt|le|al|hs|lo)?"+"[sptrx]?",end:"\\s"},hljs.COMMENT("[;@]","$",{relevance:0}),hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",relevance:0},{className:"title",begin:"\\|",end:"\\|",illegal:"\\n",relevance:0},{className:"number",variants:[{begin:"[#$=]?0x[0-9a-f]+"},{begin:"[#$=]?0b[01]+"},{begin:"[#$=]\\d+"},{begin:"\\b\\d+"}],relevance:0},{className:"symbol",variants:[{begin:"^[a-z_\\.\\$][a-z0-9_\\.\\$]+"},{begin:"^\\s*[a-z_\\.\\$][a-z0-9_\\.\\$]+:"},{begin:"[=#]\\w+"}],relevance:0}]}});hljs.registerLanguage("glsl",function(hljs){return{keywords:{keyword:"break continue discard do else for if return while switch case default "+"attribute binding buffer ccw centroid centroid varying coherent column_major const cw "+"depth_any depth_greater depth_less depth_unchanged early_fragment_tests equal_spacing "+"flat fractional_even_spacing fractional_odd_spacing highp in index inout invariant "+"invocations isolines layout line_strip lines lines_adjacency local_size_x local_size_y "+"local_size_z location lowp max_vertices mediump noperspective offset origin_upper_left "+"out packed patch pixel_center_integer point_mode points precise precision quads r11f_g11f_b10f "+"r16 r16_snorm r16f r16i r16ui r32f r32i r32ui r8 r8_snorm r8i r8ui readonly restrict "+"rg16 rg16_snorm rg16f rg16i rg16ui rg32f rg32i rg32ui rg8 rg8_snorm rg8i rg8ui rgb10_a2 "+"rgb10_a2ui rgba16 rgba16_snorm rgba16f rgba16i rgba16ui rgba32f rgba32i rgba32ui rgba8 "+"rgba8_snorm rgba8i rgba8ui row_major sample shared smooth std140 std430 stream triangle_strip "+"triangles triangles_adjacency uniform varying vertices volatile writeonly",type:"atomic_uint bool bvec2 bvec3 bvec4 dmat2 dmat2x2 dmat2x3 dmat2x4 dmat3 dmat3x2 dmat3x3 "+"dmat3x4 dmat4 dmat4x2 dmat4x3 dmat4x4 double dvec2 dvec3 dvec4 float iimage1D iimage1DArray "+"iimage2D iimage2DArray iimage2DMS iimage2DMSArray iimage2DRect iimage3D iimageBuffer"+"iimageCube iimageCubeArray image1D image1DArray image2D image2DArray image2DMS image2DMSArray "+"image2DRect image3D imageBuffer imageCube imageCubeArray int isampler1D isampler1DArray "+"isampler2D isampler2DArray isampler2DMS isampler2DMSArray isampler2DRect isampler3D "+"isamplerBuffer isamplerCube isamplerCubeArray ivec2 ivec3 ivec4 mat2 mat2x2 mat2x3 "+"mat2x4 mat3 mat3x2 mat3x3 mat3x4 mat4 mat4x2 mat4x3 mat4x4 sampler1D sampler1DArray "+"sampler1DArrayShadow sampler1DShadow sampler2D sampler2DArray sampler2DArrayShadow "+"sampler2DMS sampler2DMSArray sampler2DRect sampler2DRectShadow sampler2DShadow sampler3D "+"samplerBuffer samplerCube samplerCubeArray samplerCubeArrayShadow samplerCubeShadow "+"image1D uimage1DArray uimage2D uimage2DArray uimage2DMS uimage2DMSArray uimage2DRect "+"uimage3D uimageBuffer uimageCube uimageCubeArray uint usampler1D usampler1DArray "+"usampler2D usampler2DArray usampler2DMS usampler2DMSArray usampler2DRect usampler3D "+"samplerBuffer usamplerCube usamplerCubeArray uvec2 uvec3 uvec4 vec2 vec3 vec4 void",built_in:"gl_MaxAtomicCounterBindings gl_MaxAtomicCounterBufferSize gl_MaxClipDistances gl_MaxClipPlanes "+"gl_MaxCombinedAtomicCounterBuffers gl_MaxCombinedAtomicCounters gl_MaxCombinedImageUniforms "+"gl_MaxCombinedImageUnitsAndFragmentOutputs gl_MaxCombinedTextureImageUnits gl_MaxComputeAtomicCounterBuffers "+"gl_MaxComputeAtomicCounters gl_MaxComputeImageUniforms gl_MaxComputeTextureImageUnits "+"gl_MaxComputeUniformComponents gl_MaxComputeWorkGroupCount gl_MaxComputeWorkGroupSize "+"gl_MaxDrawBuffers gl_MaxFragmentAtomicCounterBuffers gl_MaxFragmentAtomicCounters "+"gl_MaxFragmentImageUniforms gl_MaxFragmentInputComponents gl_MaxFragmentInputVectors "+"gl_MaxFragmentUniformComponents gl_MaxFragmentUniformVectors gl_MaxGeometryAtomicCounterBuffers "+"gl_MaxGeometryAtomicCounters gl_MaxGeometryImageUniforms gl_MaxGeometryInputComponents "+"gl_MaxGeometryOutputComponents gl_MaxGeometryOutputVertices gl_MaxGeometryTextureImageUnits "+"gl_MaxGeometryTotalOutputComponents gl_MaxGeometryUniformComponents gl_MaxGeometryVaryingComponents "+"gl_MaxImageSamples gl_MaxImageUnits gl_MaxLights gl_MaxPatchVertices gl_MaxProgramTexelOffset "+"gl_MaxTessControlAtomicCounterBuffers gl_MaxTessControlAtomicCounters gl_MaxTessControlImageUniforms "+"gl_MaxTessControlInputComponents gl_MaxTessControlOutputComponents gl_MaxTessControlTextureImageUnits "+"gl_MaxTessControlTotalOutputComponents gl_MaxTessControlUniformComponents "+"gl_MaxTessEvaluationAtomicCounterBuffers gl_MaxTessEvaluationAtomicCounters "+"gl_MaxTessEvaluationImageUniforms gl_MaxTessEvaluationInputComponents gl_MaxTessEvaluationOutputComponents "+"gl_MaxTessEvaluationTextureImageUnits gl_MaxTessEvaluationUniformComponents "+"gl_MaxTessGenLevel gl_MaxTessPatchComponents gl_MaxTextureCoords gl_MaxTextureImageUnits "+"gl_MaxTextureUnits gl_MaxVaryingComponents gl_MaxVaryingFloats gl_MaxVaryingVectors "+"gl_MaxVertexAtomicCounterBuffers gl_MaxVertexAtomicCounters gl_MaxVertexAttribs gl_MaxVertexImageUniforms "+"gl_MaxVertexOutputComponents gl_MaxVertexOutputVectors gl_MaxVertexTextureImageUnits "+"gl_MaxVertexUniformComponents gl_MaxVertexUniformVectors gl_MaxViewports gl_MinProgramTexelOffset "+"gl_BackColor gl_BackLightModelProduct gl_BackLightProduct gl_BackMaterial "+"gl_BackSecondaryColor gl_ClipDistance gl_ClipPlane gl_ClipVertex gl_Color "+"gl_DepthRange gl_EyePlaneQ gl_EyePlaneR gl_EyePlaneS gl_EyePlaneT gl_Fog gl_FogCoord "+"gl_FogFragCoord gl_FragColor gl_FragCoord gl_FragData gl_FragDepth gl_FrontColor "+"gl_FrontFacing gl_FrontLightModelProduct gl_FrontLightProduct gl_FrontMaterial "+"gl_FrontSecondaryColor gl_GlobalInvocationID gl_InstanceID gl_InvocationID gl_Layer gl_LightModel "+"gl_LightSource gl_LocalInvocationID gl_LocalInvocationIndex gl_ModelViewMatrix "+"gl_ModelViewMatrixInverse gl_ModelViewMatrixInverseTranspose gl_ModelViewMatrixTranspose "+"gl_ModelViewProjectionMatrix gl_ModelViewProjectionMatrixInverse gl_ModelViewProjectionMatrixInverseTranspose "+"gl_ModelViewProjectionMatrixTranspose gl_MultiTexCoord0 gl_MultiTexCoord1 gl_MultiTexCoord2 "+"gl_MultiTexCoord3 gl_MultiTexCoord4 gl_MultiTexCoord5 gl_MultiTexCoord6 gl_MultiTexCoord7 "+"gl_Normal gl_NormalMatrix gl_NormalScale gl_NumSamples gl_NumWorkGroups gl_ObjectPlaneQ "+"gl_ObjectPlaneR gl_ObjectPlaneS gl_ObjectPlaneT gl_PatchVerticesIn gl_Point gl_PointCoord "+"gl_PointSize gl_Position gl_PrimitiveID gl_PrimitiveIDIn gl_ProjectionMatrix gl_ProjectionMatrixInverse "+"gl_ProjectionMatrixInverseTranspose gl_ProjectionMatrixTranspose gl_SampleID gl_SampleMask "+"gl_SampleMaskIn gl_SamplePosition gl_SecondaryColor gl_TessCoord gl_TessLevelInner gl_TessLevelOuter "+"gl_TexCoord gl_TextureEnvColor gl_TextureMatrix gl_TextureMatrixInverse gl_TextureMatrixInverseTranspose "+"gl_TextureMatrixTranspose gl_Vertex gl_VertexID gl_ViewportIndex gl_WorkGroupID gl_WorkGroupSize gl_in gl_out "+"EmitStreamVertex EmitVertex EndPrimitive EndStreamPrimitive abs acos acosh all any asin "+"asinh atan atanh atomicAdd atomicAnd atomicCompSwap atomicCounter atomicCounterDecrement "+"atomicCounterIncrement atomicExchange atomicMax atomicMin atomicOr atomicXor barrier "+"bitCount bitfieldExtract bitfieldInsert bitfieldReverse ceil clamp cos cosh cross "+"dFdx dFdy degrees determinant distance dot equal exp exp2 faceforward findLSB findMSB "+"floatBitsToInt floatBitsToUint floor fma fract frexp ftransform fwidth greaterThan "+"greaterThanEqual groupMemoryBarrier imageAtomicAdd imageAtomicAnd imageAtomicCompSwap "+"imageAtomicExchange imageAtomicMax imageAtomicMin imageAtomicOr imageAtomicXor imageLoad "+"imageSize imageStore imulExtended intBitsToFloat interpolateAtCentroid interpolateAtOffset "+"interpolateAtSample inverse inversesqrt isinf isnan ldexp length lessThan lessThanEqual log "+"log2 matrixCompMult max memoryBarrier memoryBarrierAtomicCounter memoryBarrierBuffer "+"memoryBarrierImage memoryBarrierShared min mix mod modf noise1 noise2 noise3 noise4 "+"normalize not notEqual outerProduct packDouble2x32 packHalf2x16 packSnorm2x16 packSnorm4x8 "+"packUnorm2x16 packUnorm4x8 pow radians reflect refract round roundEven shadow1D shadow1DLod "+"shadow1DProj shadow1DProjLod shadow2D shadow2DLod shadow2DProj shadow2DProjLod sign sin sinh "+"smoothstep sqrt step tan tanh texelFetch texelFetchOffset texture texture1D texture1DLod "+"texture1DProj texture1DProjLod texture2D texture2DLod texture2DProj texture2DProjLod "+"texture3D texture3DLod texture3DProj texture3DProjLod textureCube textureCubeLod "+"textureGather textureGatherOffset textureGatherOffsets textureGrad textureGradOffset "+"textureLod textureLodOffset textureOffset textureProj textureProjGrad textureProjGradOffset "+"textureProjLod textureProjLodOffset textureProjOffset textureQueryLevels textureQueryLod "+"textureSize transpose trunc uaddCarry uintBitsToFloat umulExtended unpackDouble2x32 "+"unpackHalf2x16 unpackSnorm2x16 unpackSnorm4x8 unpackUnorm2x16 unpackUnorm4x8 usubBorrow",literal:"true false"},illegal:'"',contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"}]}});hljs.registerLanguage("q",function(hljs){var Q_KEYWORDS={keyword:"do while select delete by update from",literal:"0b 1b",built_in:"neg not null string reciprocal floor ceiling signum mod xbar xlog and or each scan over prior mmu lsq inv md5 ltime gtime count first var dev med cov cor all any rand sums prds mins maxs fills deltas ratios avgs differ prev next rank reverse iasc idesc asc desc msum mcount mavg mdev xrank mmin mmax xprev rotate distinct group where flip type key til get value attr cut set upsert raze union inter except cross sv vs sublist enlist read0 read1 hopen hclose hdel hsym hcount peach system ltrim rtrim trim lower upper ssr view tables views cols xcols keys xkey xcol xasc xdesc fkeys meta lj aj aj0 ij pj asof uj ww wj wj1 fby xgroup ungroup ej save load rsave rload show csv parse eval min max avg wavg wsum sin cos tan sum",type:"`float `double int `timestamp `timespan `datetime `time `boolean `symbol `char `byte `short `long `real `month `date `minute `second `guid"};return{aliases:["k","kdb"],keywords:Q_KEYWORDS,lexemes:/(`?)[A-Za-z0-9_]+\b/,contains:[hljs.C_LINE_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("css",function(hljs){var FUNCTION_LIKE={begin:/[\w-]+\(/,returnBegin:true,contains:[{className:"built_in",begin:/[\w-]+/},{begin:/\(/,end:/\)/,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE]}]};var ATTRIBUTE={className:"attribute",begin:/\S/,end:":",excludeEnd:true,starts:{endsWithParent:true,excludeEnd:true,contains:[FUNCTION_LIKE,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"number",begin:"#[0-9A-Fa-f]+"},{className:"meta",begin:"!important"}]}};var AT_IDENTIFIER="@[a-z-]+";var AT_MODIFIERS="and or not only";var MEDIA_TYPES="all print screen speech";var AT_PROPERTY_RE=/@\-?\w[\w]*(\-\w+)*/;var IDENT_RE="[a-zA-Z-][a-zA-Z0-9_-]*";var RULE={begin:/(?:[A-Z\_\.\-]+|--[a-zA-Z0-9_-]+)\s*:/,returnBegin:true,end:";",endsWithParent:true,contains:[ATTRIBUTE]};return{case_insensitive:true,illegal:/[=\/|'\$]/,contains:[hljs.C_BLOCK_COMMENT_MODE,{className:"selector-id",begin:/#[A-Za-z0-9_-]+/},{className:"selector-class",begin:/\.[A-Za-z0-9_-]+/},{className:"selector-attr",begin:/\[/,end:/\]/,illegal:"$",contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]},{className:"selector-pseudo",begin:/:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/},{begin:"@(page|font-face)",lexemes:AT_IDENTIFIER,keywords:"@page @font-face"},{begin:"@",end:"[{;]",illegal:/:/,returnBegin:true,contains:[{className:"keyword",begin:AT_PROPERTY_RE},{begin:/\s/,endsWithParent:true,excludeEnd:true,relevance:0,keywords:AT_MODIFIERS,contains:[{begin:/[a-z-]+:/,className:"attribute"},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE]}]},{className:"selector-tag",begin:IDENT_RE,relevance:0},{begin:"{",end:"}",illegal:/\S/,contains:[hljs.C_BLOCK_COMMENT_MODE,RULE]}]}});hljs.registerLanguage("ini",function(hljs){var NUMBERS={className:"number",relevance:0,variants:[{begin:/([\+\-]+)?[\d]+_[\d_]+/},{begin:hljs.NUMBER_RE}]};var COMMENTS=hljs.COMMENT();COMMENTS.variants=[{begin:/;/,end:/$/},{begin:/#/,end:/$/}];var VARIABLES={className:"variable",variants:[{begin:/\$[\w\d"][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var LITERALS={className:"literal",begin:/\bon|off|true|false|yes|no\b/};var STRINGS={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:"'''",end:"'''",relevance:10},{begin:'"""',end:'"""',relevance:10},{begin:'"',end:'"'},{begin:"'",end:"'"}]};var ARRAY={begin:/\[/,end:/\]/,contains:[COMMENTS,LITERALS,VARIABLES,STRINGS,NUMBERS,"self"],relevance:0};return{aliases:["toml"],case_insensitive:true,illegal:/\S/,contains:[COMMENTS,{className:"section",begin:/\[+/,end:/\]+/},{begin:/^[a-z0-9\[\]_\.-]+(?=\s*=\s*)/,className:"attr",starts:{end:/$/,contains:[COMMENTS,ARRAY,LITERALS,VARIABLES,STRINGS,NUMBERS]}}]}});hljs.registerLanguage("elm",function(hljs){var COMMENT={variants:[hljs.COMMENT("--","$"),hljs.COMMENT("{-","-}",{contains:["self"]})]};var CONSTRUCTOR={className:"type",begin:"\\b[A-Z][\\w']*",relevance:0};var LIST={begin:"\\(",end:"\\)",illegal:'"',contains:[{className:"type",begin:"\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?"},COMMENT]};var RECORD={begin:"{",end:"}",contains:LIST.contains};var CHARACTER={className:"string",begin:"'\\\\?.",end:"'",illegal:"."};return{keywords:"let in if then else case of where module import exposing "+"type alias as infix infixl infixr port effect command subscription",contains:[{beginKeywords:"port effect module",end:"exposing",keywords:"port effect module where command subscription exposing",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"import",end:"$",keywords:"import as exposing",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"type",end:"$",keywords:"type alias",contains:[CONSTRUCTOR,LIST,RECORD,COMMENT]},{beginKeywords:"infix infixl infixr",end:"$",contains:[hljs.C_NUMBER_MODE,COMMENT]},{begin:"port",end:"$",keywords:"port",contains:[COMMENT]},CHARACTER,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,CONSTRUCTOR,hljs.inherit(hljs.TITLE_MODE,{begin:"^[_a-z][\\w']*"}),COMMENT,{begin:"->|<-"}],illegal:/;/}});hljs.registerLanguage("ldif",function(hljs){return{contains:[{className:"attribute",begin:"^dn",end:": ",excludeEnd:true,starts:{end:"$",relevance:0},relevance:10},{className:"attribute",begin:"^\\w",end:": ",excludeEnd:true,starts:{end:"$",relevance:0}},{className:"literal",begin:"^-",end:"$"},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("makefile",function(hljs){var VARIABLE={className:"variable",variants:[{begin:"\\$\\("+hljs.UNDERSCORE_IDENT_RE+"\\)",contains:[hljs.BACKSLASH_ESCAPE]},{begin:/\$[@%<?\^\+\*]/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VARIABLE]};var FUNC={className:"variable",begin:/\$\([\w-]+\s/,end:/\)/,keywords:{built_in:"subst patsubst strip findstring filter filter-out sort "+"word wordlist firstword lastword dir notdir suffix basename "+"addsuffix addprefix join wildcard realpath abspath error warning "+"shell origin flavor foreach if or and call eval file value"},contains:[VARIABLE]};var ASSIGNMENT={begin:"^"+hljs.UNDERSCORE_IDENT_RE+"\\s*(?=[:+?]?=)"};var META={className:"meta",begin:/^\.PHONY:/,end:/$/,keywords:{"meta-keyword":".PHONY"},lexemes:/[\.\w]+/};var TARGET={className:"section",begin:/^[^\s]+:/,end:/$/,contains:[VARIABLE]};return{aliases:["mk","mak"],keywords:"define endef undefine ifdef ifndef ifeq ifneq else endif "+"include -include sinclude override export unexport private vpath",lexemes:/[\w-]+/,contains:[hljs.HASH_COMMENT_MODE,VARIABLE,QUOTE_STRING,FUNC,ASSIGNMENT,META,TARGET]}});hljs.registerLanguage("coffeescript",function(hljs){var KEYWORDS={keyword:"in if for while finally new do return else break catch instanceof throw try this "+"switch continue typeof delete debugger super yield import export from as default await "+"then unless until loop of by when and or is isnt not",literal:"true false null undefined "+"yes no on off",built_in:"npm require console print module global window document"};var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.BINARY_NUMBER_MODE,hljs.inherit(hljs.C_NUMBER_MODE,{starts:{end:"(\\s*/)?",relevance:0}}),{className:"string",variants:[{begin:/'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]},{className:"regexp",variants:[{begin:"///",end:"///",contains:[SUBST,hljs.HASH_COMMENT_MODE]},{begin:"//[gim]{0,3}(?=\\W)",relevance:0},{begin:/\/(?![ *]).*?(?![\\]).\/[gim]{0,3}(?=\W)/}]},{begin:"@"+JS_IDENT_RE},{subLanguage:"javascript",excludeBegin:true,excludeEnd:true,variants:[{begin:"```",end:"```"},{begin:"`",end:"`"}]}];SUBST.contains=EXPRESSIONS;var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var PARAMS_RE="(\\(.*\\))?\\s*\\B[-=]>";var PARAMS={className:"params",begin:"\\([^\\(]",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};return{aliases:["coffee","cson","iced"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("###","###"),hljs.HASH_COMMENT_MODE,{className:"function",begin:"^\\s*"+JS_IDENT_RE+"\\s*=\\s*"+PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[TITLE,PARAMS]},{begin:/[:\(,=]\s*/,relevance:0,contains:[{className:"function",begin:PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[PARAMS]}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("abnf",function(hljs){var regexes={ruleDeclaration:"^[a-zA-Z][a-zA-Z0-9-]*",unexpectedChars:"[!@#$^&',?+~`|:]"};var keywords=["ALPHA","BIT","CHAR","CR","CRLF","CTL","DIGIT","DQUOTE","HEXDIG","HTAB","LF","LWSP","OCTET","SP","VCHAR","WSP"];var commentMode=hljs.COMMENT(";","$");var terminalBinaryMode={className:"symbol",begin:/%b[0-1]+(-[0-1]+|(\.[0-1]+)+){0,1}/};var terminalDecimalMode={className:"symbol",begin:/%d[0-9]+(-[0-9]+|(\.[0-9]+)+){0,1}/};var terminalHexadecimalMode={className:"symbol",begin:/%x[0-9A-F]+(-[0-9A-F]+|(\.[0-9A-F]+)+){0,1}/};var caseSensitivityIndicatorMode={className:"symbol",begin:/%[si]/};var ruleDeclarationMode={className:"attribute",begin:regexes.ruleDeclaration+"(?=\\s*=)"};return{illegal:regexes.unexpectedChars,keywords:keywords.join(" "),contains:[ruleDeclarationMode,commentMode,terminalBinaryMode,terminalDecimalMode,terminalHexadecimalMode,caseSensitivityIndicatorMode,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("kotlin",function(hljs){var KEYWORDS={keyword:"abstract as val var vararg get set class object open private protected public noinline "+"crossinline dynamic final enum if else do while for when throw try catch finally "+"import package is in fun override companion reified inline lateinit init "+"interface annotation data sealed internal infix operator out by constructor super "+"tailrec where const inner suspend typealias external expect actual "+"trait volatile transient native default",built_in:"Byte Short Char Int Long Boolean Float Double Void Unit Nothing",literal:"true false null"};var KEYWORDS_WITH_LABEL={className:"keyword",begin:/\b(break|continue|return|this)\b/,starts:{contains:[{className:"symbol",begin:/@\w+/}]}};var LABEL={className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"@"};var SUBST={className:"subst",begin:"\\${",end:"}",contains:[hljs.C_NUMBER_MODE]};var VARIABLE={className:"variable",begin:"\\$"+hljs.UNDERSCORE_IDENT_RE};var STRING={className:"string",variants:[{begin:'"""',end:'"""(?=[^"])',contains:[VARIABLE,SUBST]},{begin:"'",end:"'",illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"',end:'"',illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE,VARIABLE,SUBST]}]};SUBST.contains.push(STRING);var ANNOTATION_USE_SITE={className:"meta",begin:"@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*"+hljs.UNDERSCORE_IDENT_RE+")?"};var ANNOTATION={className:"meta",begin:"@"+hljs.UNDERSCORE_IDENT_RE,contains:[{begin:/\(/,end:/\)/,contains:[hljs.inherit(STRING,{className:"meta-string"})]}]};var KOTLIN_NUMBER_RE="\\b"+"("+"0[bB]([01]+[01_]+[01]+|[01]+)"+"|"+"0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)"+"|"+"("+"([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?"+"|"+"\\.([\\d]+[\\d_]+[\\d]+|[\\d]+)"+")"+"([eE][-+]?\\d+)?"+")"+"[lLfF]?";var KOTLIN_NUMBER_MODE={className:"number",begin:KOTLIN_NUMBER_RE,relevance:0};var KOTLIN_NESTED_COMMENT=hljs.COMMENT("/\\*","\\*/",{contains:[hljs.C_BLOCK_COMMENT_MODE]});var KOTLIN_PAREN_TYPE={variants:[{className:"type",begin:hljs.UNDERSCORE_IDENT_RE},{begin:/\(/,end:/\)/,contains:[]}]};var KOTLIN_PAREN_TYPE2=KOTLIN_PAREN_TYPE;KOTLIN_PAREN_TYPE2.variants[1].contains=[KOTLIN_PAREN_TYPE];KOTLIN_PAREN_TYPE.variants[1].contains=[KOTLIN_PAREN_TYPE2];return{aliases:["kt"],keywords:KEYWORDS,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT,KEYWORDS_WITH_LABEL,LABEL,ANNOTATION_USE_SITE,ANNOTATION,{className:"function",beginKeywords:"fun",end:"[(]|$",returnBegin:true,excludeEnd:true,keywords:KEYWORDS,illegal:/fun\s+(<.*>)?[^\s\(]+(\s+[^\s\(]+)\s*=/,relevance:5,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"type",begin:/</,end:/>/,keywords:"reified",relevance:0},{className:"params",begin:/\(/,end:/\)/,endsParent:true,keywords:KEYWORDS,relevance:0,contains:[{begin:/:/,end:/[=,\/]/,endsWithParent:true,contains:[KOTLIN_PAREN_TYPE,hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT],relevance:0},hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT,ANNOTATION_USE_SITE,ANNOTATION,STRING,hljs.C_NUMBER_MODE]},KOTLIN_NESTED_COMMENT]},{className:"class",beginKeywords:"class interface trait",end:/[:\{(]|$/,excludeEnd:true,illegal:"extends implements",contains:[{beginKeywords:"public protected internal private constructor"},hljs.UNDERSCORE_TITLE_MODE,{className:"type",begin:/</,end:/>/,excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:/[,:]\s*/,end:/[<\(,]|$/,excludeBegin:true,returnEnd:true},ANNOTATION_USE_SITE,ANNOTATION]},STRING,{className:"meta",begin:"^#!/usr/bin/env",end:"$",illegal:"\n"},KOTLIN_NUMBER_MODE]}});hljs.registerLanguage("dsconfig",function(hljs){var QUOTED_PROPERTY={className:"string",begin:/"/,end:/"/};var APOS_PROPERTY={className:"string",begin:/'/,end:/'/};var UNQUOTED_PROPERTY={className:"string",begin:"[\\w-?]+:\\w+",end:"\\W",relevance:0};var VALUELESS_PROPERTY={className:"string",begin:"\\w+-?\\w+",end:"\\W",relevance:0};return{keywords:"dsconfig",contains:[{className:"keyword",begin:"^dsconfig",end:"\\s",excludeEnd:true,relevance:10},{className:"built_in",begin:"(list|create|get|set|delete)-(\\w+)",end:"\\s",excludeEnd:true,illegal:"!@#$%^&*()",relevance:10},{className:"built_in",begin:"--(\\w+)",end:"\\s",excludeEnd:true},QUOTED_PROPERTY,APOS_PROPERTY,UNQUOTED_PROPERTY,VALUELESS_PROPERTY,hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("http",function(hljs){var VERSION="HTTP/[0-9\\.]+";return{aliases:["https"],illegal:"\\S",contains:[{begin:"^"+VERSION,end:"$",contains:[{className:"number",begin:"\\b\\d{3}\\b"}]},{begin:"^[A-Z]+ (.*?) "+VERSION+"$",returnBegin:true,end:"$",contains:[{className:"string",begin:" ",end:" ",excludeBegin:true,excludeEnd:true},{begin:VERSION},{className:"keyword",begin:"[A-Z]+"}]},{className:"attribute",begin:"^\\w",end:": ",excludeEnd:true,illegal:"\\n|\\s|=",starts:{end:"$",relevance:0}},{begin:"\\n\\n",starts:{subLanguage:[],endsWithParent:true}}]}});hljs.registerLanguage("pony",function(hljs){var KEYWORDS={keyword:"actor addressof and as be break class compile_error compile_intrinsic "+"consume continue delegate digestof do else elseif embed end error "+"for fun if ifdef in interface is isnt lambda let match new not object "+"or primitive recover repeat return struct then trait try type until "+"use var where while with xor",meta:"iso val tag trn box ref",literal:"this false true"};var TRIPLE_QUOTE_STRING_MODE={className:"string",begin:'"""',end:'"""',relevance:10};var QUOTE_STRING_MODE={className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE]};var SINGLE_QUOTE_CHAR_MODE={className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE],relevance:0};var TYPE_NAME={className:"type",begin:"\\b_?[A-Z][\\w]*",relevance:0};var PRIMED_NAME={begin:hljs.IDENT_RE+"'",relevance:0};var NUMBER_MODE={className:"number",begin:"(-?)(\\b0[xX][a-fA-F0-9]+|\\b0[bB][01]+|(\\b\\d+(_\\d+)?(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",relevance:0};return{keywords:KEYWORDS,contains:[TYPE_NAME,TRIPLE_QUOTE_STRING_MODE,QUOTE_STRING_MODE,SINGLE_QUOTE_CHAR_MODE,PRIMED_NAME,NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("cpp",function(hljs){function optional(s){return"(?:"+s+")?"}var DECLTYPE_AUTO_RE="decltype\\(auto\\)";var NAMESPACE_RE="[a-zA-Z_]\\w*::";var TEMPLATE_ARGUMENT_RE="<.*?>";var FUNCTION_TYPE_RE="("+DECLTYPE_AUTO_RE+"|"+optional(NAMESPACE_RE)+"[a-zA-Z_]\\w*"+optional(TEMPLATE_ARGUMENT_RE)+")";var CPP_PRIMITIVE_TYPES={className:"keyword",begin:"\\b[a-z\\d_]*_t\\b"};var CHARACTER_ESCAPES="\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)";var STRINGS={className:"string",variants:[{begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"(u8?|U|L)?'("+CHARACTER_ESCAPES+"|.)",end:"'",illegal:"."},{begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\((?:.|\n)*?\)\1"/}]};var NUMBERS={className:"number",variants:[{begin:"\\b(0b[01']+)"},{begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],relevance:0};var PREPROCESSOR={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"if else elif endif define undef warning error line "+"pragma _Pragma ifdef ifndef include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:/<.*?>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};var TITLE_MODE={className:"title",begin:optional(NAMESPACE_RE)+hljs.IDENT_RE,relevance:0};var FUNCTION_TITLE=optional(NAMESPACE_RE)+hljs.IDENT_RE+"\\s*\\(";var CPP_KEYWORDS={keyword:"int float while private char char8_t char16_t char32_t catch import module export virtual operator sizeof "+"dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace "+"unsigned long volatile static protected bool template mutable if public friend "+"do goto auto void enum else break extern using asm case typeid wchar_t"+"short reinterpret_cast|10 default double register explicit signed typename try this "+"switch continue inline delete alignas alignof constexpr consteval constinit decltype "+"concept co_await co_return co_yield requires "+"noexcept static_assert thread_local restrict final override "+"atomic_bool atomic_char atomic_schar "+"atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong "+"atomic_ullong new throw return "+"and and_eq bitand bitor compl not not_eq or or_eq xor xor_eq",built_in:"std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream "+"auto_ptr deque list queue stack vector map set bitset multiset multimap unordered_set "+"unordered_map unordered_multiset unordered_multimap array shared_ptr abort terminate abs acos "+"asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp "+"fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper "+"isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow "+"printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp "+"strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan "+"vfprintf vprintf vsprintf endl initializer_list unique_ptr _Bool complex _Complex imaginary _Imaginary",literal:"true false nullptr NULL"};var EXPRESSION_CONTAINS=[CPP_PRIMITIVE_TYPES,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS];var EXPRESSION_CONTEXT={variants:[{begin:/=/,end:/;/},{begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}],keywords:CPP_KEYWORDS,contains:EXPRESSION_CONTAINS.concat([{begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,contains:EXPRESSION_CONTAINS.concat(["self"]),relevance:0}]),relevance:0};var FUNCTION_DECLARATION={className:"function",begin:"("+FUNCTION_TYPE_RE+"[\\*&\\s]+)+"+FUNCTION_TITLE,returnBegin:true,end:/[{;=]/,excludeEnd:true,keywords:CPP_KEYWORDS,illegal:/[^\w\s\*&:<>]/,contains:[{begin:DECLTYPE_AUTO_RE,keywords:CPP_KEYWORDS,relevance:0},{begin:FUNCTION_TITLE,returnBegin:true,contains:[TITLE_MODE],relevance:0},{className:"params",begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,relevance:0,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRINGS,NUMBERS,CPP_PRIMITIVE_TYPES,{begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,relevance:0,contains:["self",hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRINGS,NUMBERS,CPP_PRIMITIVE_TYPES]}]},CPP_PRIMITIVE_TYPES,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,PREPROCESSOR]};return{aliases:["c","cc","h","c++","h++","hpp","hh","hxx","cxx"],keywords:CPP_KEYWORDS,illegal:"</",contains:[].concat(EXPRESSION_CONTEXT,FUNCTION_DECLARATION,EXPRESSION_CONTAINS,[PREPROCESSOR,{begin:"\\b(deque|list|queue|stack|vector|map|set|bitset|multiset|multimap|unordered_map|unordered_set|unordered_multiset|unordered_multimap|array)\\s*<",end:">",keywords:CPP_KEYWORDS,contains:["self",CPP_PRIMITIVE_TYPES]},{begin:hljs.IDENT_RE+"::",keywords:CPP_KEYWORDS},{className:"class",beginKeywords:"class struct",end:/[{;:]/,contains:[{begin:/</,end:/>/,contains:["self"]},hljs.TITLE_MODE]}]),exports:{preprocessor:PREPROCESSOR,strings:STRINGS,keywords:CPP_KEYWORDS}}});hljs.registerLanguage("arduino",function(hljs){var ARDUINO_KW={keyword:"boolean byte word String",built_in:"setup loop"+"KeyboardController MouseController SoftwareSerial "+"EthernetServer EthernetClient LiquidCrystal "+"RobotControl GSMVoiceCall EthernetUDP EsploraTFT "+"HttpClient RobotMotor WiFiClient GSMScanner "+"FileSystem Scheduler GSMServer YunClient YunServer "+"IPAddress GSMClient GSMModem Keyboard Ethernet "+"Console GSMBand Esplora Stepper Process "+"WiFiUDP GSM_SMS Mailbox USBHost Firmata PImage "+"Client Server GSMPIN FileIO Bridge Serial "+"EEPROM Stream Mouse Audio Servo File Task "+"GPRS WiFi Wire TFT GSM SPI SD "+"runShellCommandAsynchronously analogWriteResolution "+"retrieveCallingNumber printFirmwareVersion "+"analogReadResolution sendDigitalPortPair "+"noListenOnLocalhost readJoystickButton setFirmwareVersion "+"readJoystickSwitch scrollDisplayRight getVoiceCallStatus "+"scrollDisplayLeft writeMicroseconds delayMicroseconds "+"beginTransmission getSignalStrength runAsynchronously "+"getAsynchronously listenOnLocalhost getCurrentCarrier "+"readAccelerometer messageAvailable sendDigitalPorts "+"lineFollowConfig countryNameWrite runShellCommand "+"readStringUntil rewindDirectory readTemperature "+"setClockDivider readLightSensor endTransmission "+"analogReference detachInterrupt countryNameRead "+"attachInterrupt encryptionType readBytesUntil "+"robotNameWrite readMicrophone robotNameRead cityNameWrite "+"userNameWrite readJoystickY readJoystickX mouseReleased "+"openNextFile scanNetworks noInterrupts digitalWrite "+"beginSpeaker mousePressed isActionDone mouseDragged "+"displayLogos noAutoscroll addParameter remoteNumber "+"getModifiers keyboardRead userNameRead waitContinue "+"processInput parseCommand printVersion readNetworks "+"writeMessage blinkVersion cityNameRead readMessage "+"setDataMode parsePacket isListening setBitOrder "+"beginPacket isDirectory motorsWrite drawCompass "+"digitalRead clearScreen serialEvent rightToLeft "+"setTextSize leftToRight requestFrom keyReleased "+"compassRead analogWrite interrupts WiFiServer "+"disconnect playMelody parseFloat autoscroll "+"getPINUsed setPINUsed setTimeout sendAnalog "+"readSlider analogRead beginWrite createChar "+"motorsStop keyPressed tempoWrite readButton "+"subnetMask debugPrint macAddress writeGreen "+"randomSeed attachGPRS readString sendString "+"remotePort releaseAll mouseMoved background "+"getXChange getYChange answerCall getResult "+"voiceCall endPacket constrain getSocket writeJSON "+"getButton available connected findUntil readBytes "+"exitValue readGreen writeBlue startLoop IPAddress "+"isPressed sendSysex pauseMode gatewayIP setCursor "+"getOemKey tuneWrite noDisplay loadImage switchPIN "+"onRequest onReceive changePIN playFile noBuffer "+"parseInt overflow checkPIN knobRead beginTFT "+"bitClear updateIR bitWrite position writeRGB "+"highByte writeRed setSpeed readBlue noStroke "+"remoteIP transfer shutdown hangCall beginSMS "+"endWrite attached maintain noCursor checkReg "+"checkPUK shiftOut isValid shiftIn pulseIn "+"connect println localIP pinMode getIMEI "+"display noBlink process getBand running beginSD "+"drawBMP lowByte setBand release bitRead prepare "+"pointTo readRed setMode noFill remove listen "+"stroke detach attach noTone exists buffer "+"height bitSet circle config cursor random "+"IRread setDNS endSMS getKey micros "+"millis begin print write ready flush width "+"isPIN blink clear press mkdir rmdir close "+"point yield image BSSID click delay "+"read text move peek beep rect line open "+"seek fill size turn stop home find "+"step tone sqrt RSSI SSID "+"end bit tan cos sin pow map abs max "+"min get run put",literal:"DIGITAL_MESSAGE FIRMATA_STRING ANALOG_MESSAGE "+"REPORT_DIGITAL REPORT_ANALOG INPUT_PULLUP "+"SET_PIN_MODE INTERNAL2V56 SYSTEM_RESET LED_BUILTIN "+"INTERNAL1V1 SYSEX_START INTERNAL EXTERNAL "+"DEFAULT OUTPUT INPUT HIGH LOW"};var ARDUINO=hljs.requireLanguage("cpp").rawDefinition();var kws=ARDUINO.keywords;kws.keyword+=" "+ARDUINO_KW.keyword;kws.literal+=" "+ARDUINO_KW.literal;kws.built_in+=" "+ARDUINO_KW.built_in;return ARDUINO});hljs.registerLanguage("ocaml",function(hljs){return{aliases:["ml"],keywords:{keyword:"and as assert asr begin class constraint do done downto else end "+"exception external for fun function functor if in include "+"inherit! inherit initializer land lazy let lor lsl lsr lxor match method!|10 method "+"mod module mutable new object of open! open or private rec sig struct "+"then to try type val! val virtual when while with "+"parser value",built_in:"array bool bytes char exn|5 float int int32 int64 list lazy_t|5 nativeint|5 string unit "+"in_channel out_channel ref",literal:"true false"},illegal:/\/\/|>>/,lexemes:"[a-z_]\\w*!?",contains:[{className:"literal",begin:"\\[(\\|\\|)?\\]|\\(\\)",relevance:0},hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),{className:"symbol",begin:"'[A-Za-z_](?!')[\\w']*"},{className:"type",begin:"`[A-Z][\\w']*"},{className:"type",begin:"\\b[A-Z][\\w']*",relevance:0},{begin:"[a-z_]\\w*'[\\w']*",relevance:0},hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"number",begin:"\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)",relevance:0},{begin:/[-=]>/}]}});hljs.registerLanguage("oxygene",function(hljs){var OXYGENE_KEYWORDS="abstract add and array as asc aspect assembly async begin break block by case class concat const copy constructor continue "+"create default delegate desc distinct div do downto dynamic each else empty end ensure enum equals event except exit extension external false "+"final finalize finalizer finally flags for forward from function future global group has if implementation implements implies in index inherited "+"inline interface into invariants is iterator join locked locking loop matching method mod module namespace nested new nil not notify nullable of "+"old on operator or order out override parallel params partial pinned private procedure property protected public queryable raise read readonly "+"record reintroduce remove repeat require result reverse sealed select self sequence set shl shr skip static step soft take then to true try tuple "+"type union unit unsafe until uses using var virtual raises volatile where while with write xor yield await mapped deprecated stdcall cdecl pascal "+"register safecall overload library platform reference packed strict published autoreleasepool selector strong weak unretained";var CURLY_COMMENT=hljs.COMMENT("{","}",{relevance:0});var PAREN_COMMENT=hljs.COMMENT("\\(\\*","\\*\\)",{relevance:10});var STRING={className:"string",begin:"'",end:"'",contains:[{begin:"''"}]};var CHAR_STRING={className:"string",begin:"(#\\d+)+"};var FUNCTION={className:"function",beginKeywords:"function constructor destructor procedure method",end:"[:;]",keywords:"function constructor|10 destructor|10 procedure|10 method|10",contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",keywords:OXYGENE_KEYWORDS,contains:[STRING,CHAR_STRING]},CURLY_COMMENT,PAREN_COMMENT]};return{case_insensitive:true,lexemes:/\.?\w+/,keywords:OXYGENE_KEYWORDS,illegal:'("|\\$[G-Zg-z]|\\/\\*|</|=>|->)',contains:[CURLY_COMMENT,PAREN_COMMENT,hljs.C_LINE_COMMENT_MODE,STRING,CHAR_STRING,hljs.NUMBER_MODE,FUNCTION,{className:"class",begin:"=\\bclass\\b",end:"end;",keywords:OXYGENE_KEYWORDS,contains:[STRING,CHAR_STRING,CURLY_COMMENT,PAREN_COMMENT,hljs.C_LINE_COMMENT_MODE,FUNCTION]}]}});hljs.registerLanguage("scilab",function(hljs){var COMMON_CONTAINS=[hljs.C_NUMBER_MODE,{className:"string",begin:"'|\"",end:"'|\"",contains:[hljs.BACKSLASH_ESCAPE,{begin:"''"}]}];return{aliases:["sci"],lexemes:/%?\w+/,keywords:{keyword:"abort break case clear catch continue do elseif else endfunction end for function "+"global if pause return resume select try then while",literal:"%f %F %t %T %pi %eps %inf %nan %e %i %z %s",built_in:"abs and acos asin atan ceil cd chdir clearglobal cosh cos cumprod deff disp error "+"exec execstr exists exp eye gettext floor fprintf fread fsolve imag isdef isempty "+"isinfisnan isvector lasterror length load linspace list listfiles log10 log2 log "+"max min msprintf mclose mopen ones or pathconvert poly printf prod pwd rand real "+"round sinh sin size gsort sprintf sqrt strcat strcmps tring sum system tanh tan "+"type typename warning zeros matrix"},illegal:'("|#|/\\*|\\s+/\\w+)',contains:[{className:"function",beginKeywords:"function",end:"$",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)"}]},{begin:"[a-zA-Z_][a-zA-Z_0-9]*('+[\\.']*|[\\.']+)",end:"",relevance:0},{begin:"\\[",end:"\\]'*[\\.']*",relevance:0,contains:COMMON_CONTAINS},hljs.COMMENT("//","$")].concat(COMMON_CONTAINS)}});hljs.registerLanguage("jboss-cli",function(hljs){var PARAM={begin:/[\w-]+ *=/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/[\w-]+/}]};var PARAMSBLOCK={className:"params",begin:/\(/,end:/\)/,contains:[PARAM],relevance:0};var OPERATION={className:"function",begin:/:[\w\-.]+/,relevance:0};var PATH={className:"string",begin:/\B(([\/.])[\w\-.\/=]+)+/};var COMMAND_PARAMS={className:"params",begin:/--[\w\-=\/]+/};return{aliases:["wildfly-cli"],lexemes:"[a-z-]+",keywords:{keyword:"alias batch cd clear command connect connection-factory connection-info data-source deploy "+"deployment-info deployment-overlay echo echo-dmr help history if jdbc-driver-info jms-queue|20 jms-topic|20 ls "+"patch pwd quit read-attribute read-operation reload rollout-plan run-batch set shutdown try unalias "+"undeploy unset version xa-data-source",literal:"true false"},contains:[hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,COMMAND_PARAMS,OPERATION,PATH,PARAMSBLOCK]}});hljs.registerLanguage("angelscript",function(hljs){var builtInTypeMode={className:"built_in",begin:"\\b(void|bool|int|int8|int16|int32|int64|uint|uint8|uint16|uint32|uint64|string|ref|array|double|float|auto|dictionary)"};var objectHandleMode={className:"symbol",begin:"[a-zA-Z0-9_]+@"};var genericMode={className:"keyword",begin:"<",end:">",contains:[builtInTypeMode,objectHandleMode]};builtInTypeMode.contains=[genericMode];objectHandleMode.contains=[genericMode];return{aliases:["asc"],keywords:"for in|0 break continue while do|0 return if else case switch namespace is cast "+"or and xor not get|0 in inout|10 out override set|0 private public const default|0 "+"final shared external mixin|10 enum typedef funcdef this super import from interface "+"abstract|0 try catch protected explicit property",illegal:"(^using\\s+[A-Za-z0-9_\\.]+;$|\\bfunctions*[^\\(])",contains:[{className:"string",begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"""',end:'"""'},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{beginKeywords:"interface namespace",end:"{",illegal:"[;.\\-]",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+"}]},{beginKeywords:"class",end:"{",illegal:"[;.\\-]",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+",contains:[{begin:"[:,]\\s*",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+"}]}]}]},builtInTypeMode,objectHandleMode,{className:"literal",begin:"\\b(null|true|false)"},{className:"number",begin:"(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?f?|\\.\\d+f?)([eE][-+]?\\d+f?)?)"}]}});hljs.registerLanguage("r",function(hljs){var IDENT_RE="([a-zA-Z]|\\.[a-zA-Z.])[a-zA-Z0-9._]*";return{contains:[hljs.HASH_COMMENT_MODE,{begin:IDENT_RE,lexemes:IDENT_RE,keywords:{keyword:"function if in break next repeat else for return switch while try tryCatch "+"stop warning require library attach detach source setMethod setGeneric "+"setGroupGeneric setClass ...",literal:"NULL NA TRUE FALSE T F Inf NaN NA_integer_|10 NA_real_|10 NA_character_|10 "+"NA_complex_|10"},relevance:0},{className:"number",begin:"0[xX][0-9a-fA-F]+[Li]?\\b",relevance:0},{className:"number",begin:"\\d+(?:[eE][+\\-]?\\d*)?L\\b",relevance:0},{className:"number",begin:"\\d+\\.(?!\\d)(?:i\\b)?",relevance:0},{className:"number",begin:"\\d+(?:\\.\\d*)?(?:[eE][+\\-]?\\d*)?i?\\b",relevance:0},{className:"number",begin:"\\.\\d+(?:[eE][+\\-]?\\d*)?i?\\b",relevance:0},{begin:"`",end:"`",relevance:0},{className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:'"',end:'"'},{begin:"'",end:"'"}]}]}});hljs.registerLanguage("json",function(hljs){var LITERALS={literal:"true false null"};var ALLOWED_COMMENTS=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE];var TYPES=[hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE];var VALUE_CONTAINER={end:",",endsWithParent:true,excludeEnd:true,contains:TYPES,keywords:LITERALS};var OBJECT={begin:"{",end:"}",contains:[{className:"attr",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE],illegal:"\\n"},hljs.inherit(VALUE_CONTAINER,{begin:/:/})].concat(ALLOWED_COMMENTS),illegal:"\\S"};var ARRAY={begin:"\\[",end:"\\]",contains:[hljs.inherit(VALUE_CONTAINER)],illegal:"\\S"};TYPES.push(OBJECT,ARRAY);ALLOWED_COMMENTS.forEach(function(rule){TYPES.push(rule)});return{contains:TYPES,keywords:LITERALS,illegal:"\\S"}});hljs.registerLanguage("haskell",function(hljs){var COMMENT={variants:[hljs.COMMENT("--","$"),hljs.COMMENT("{-","-}",{contains:["self"]})]};var PRAGMA={className:"meta",begin:"{-#",end:"#-}"};var PREPROCESSOR={className:"meta",begin:"^#",end:"$"};var CONSTRUCTOR={className:"type",begin:"\\b[A-Z][\\w']*",relevance:0};var LIST={begin:"\\(",end:"\\)",illegal:'"',contains:[PRAGMA,PREPROCESSOR,{className:"type",begin:"\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?"},hljs.inherit(hljs.TITLE_MODE,{begin:"[_a-z][\\w']*"}),COMMENT]};var RECORD={begin:"{",end:"}",contains:LIST.contains};return{aliases:["hs"],keywords:"let in if then else case of where do module import hiding "+"qualified type data newtype deriving class instance as default "+"infix infixl infixr foreign export ccall stdcall cplusplus "+"jvm dotnet safe unsafe family forall mdo proc rec",contains:[{beginKeywords:"module",end:"where",keywords:"module where",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"\\bimport\\b",end:"$",keywords:"import qualified as hiding",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{className:"class",begin:"^(\\s*)?(class|instance)\\b",end:"where",keywords:"class family instance where",contains:[CONSTRUCTOR,LIST,COMMENT]},{className:"class",begin:"\\b(data|(new)?type)\\b",end:"$",keywords:"data family type newtype deriving",contains:[PRAGMA,CONSTRUCTOR,LIST,RECORD,COMMENT]},{beginKeywords:"default",end:"$",contains:[CONSTRUCTOR,LIST,COMMENT]},{beginKeywords:"infix infixl infixr",end:"$",contains:[hljs.C_NUMBER_MODE,COMMENT]},{begin:"\\bforeign\\b",end:"$",keywords:"foreign import export ccall stdcall cplusplus jvm "+"dotnet safe unsafe",contains:[CONSTRUCTOR,hljs.QUOTE_STRING_MODE,COMMENT]},{className:"meta",begin:"#!\\/usr\\/bin\\/env runhaskell",end:"$"},PRAGMA,PREPROCESSOR,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,CONSTRUCTOR,hljs.inherit(hljs.TITLE_MODE,{begin:"^[_a-z][\\w']*"}),COMMENT,{begin:"->|<-"}]}});hljs.registerLanguage("taggerscript",function(hljs){var COMMENT={className:"comment",begin:/\$noop\(/,end:/\)/,contains:[{begin:/\(/,end:/\)/,contains:["self",{begin:/\\./}]}],relevance:10};var FUNCTION={className:"keyword",begin:/\$(?!noop)[a-zA-Z][_a-zA-Z0-9]*/,end:/\(/,excludeEnd:true};var VARIABLE={className:"variable",begin:/%[_a-zA-Z0-9:]*/,end:"%"};var ESCAPE_SEQUENCE={className:"symbol",begin:/\\./};return{contains:[COMMENT,FUNCTION,VARIABLE,ESCAPE_SEQUENCE]}});hljs.registerLanguage("xl",function(hljs){var BUILTIN_MODULES="ObjectLoader Animate MovieCredits Slides Filters Shading Materials LensFlare Mapping VLCAudioVideo "+"StereoDecoder PointCloud NetworkAccess RemoteControl RegExp ChromaKey Snowfall NodeJS Speech Charts";var XL_KEYWORDS={keyword:"if then else do while until for loop import with is as where when by data constant "+"integer real text name boolean symbol infix prefix postfix block tree",literal:"true false nil",built_in:"in mod rem and or xor not abs sign floor ceil sqrt sin cos tan asin "+"acos atan exp expm1 log log2 log10 log1p pi at text_length text_range "+"text_find text_replace contains page slide basic_slide title_slide "+"title subtitle fade_in fade_out fade_at clear_color color line_color "+"line_width texture_wrap texture_transform texture scale_?x scale_?y "+"scale_?z? translate_?x translate_?y translate_?z? rotate_?x rotate_?y "+"rotate_?z? rectangle circle ellipse sphere path line_to move_to "+"quad_to curve_to theme background contents locally time mouse_?x "+"mouse_?y mouse_buttons "+BUILTIN_MODULES};var DOUBLE_QUOTE_TEXT={className:"string",begin:'"',end:'"',illegal:"\\n"};var SINGLE_QUOTE_TEXT={className:"string",begin:"'",end:"'",illegal:"\\n"};var LONG_TEXT={className:"string",begin:"<<",end:">>"};var BASED_NUMBER={className:"number",begin:"[0-9]+#[0-9A-Z_]+(\\.[0-9-A-Z_]+)?#?([Ee][+-]?[0-9]+)?"};var IMPORT={beginKeywords:"import",end:"$",keywords:XL_KEYWORDS,contains:[DOUBLE_QUOTE_TEXT]};var FUNCTION_DEFINITION={className:"function",begin:/[a-z][^\n]*->/,returnBegin:true,end:/->/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,keywords:XL_KEYWORDS}})]};return{aliases:["tao"],lexemes:/[a-zA-Z][a-zA-Z0-9_?]*/,keywords:XL_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,DOUBLE_QUOTE_TEXT,SINGLE_QUOTE_TEXT,LONG_TEXT,FUNCTION_DEFINITION,IMPORT,BASED_NUMBER,hljs.NUMBER_MODE]}});hljs.registerLanguage("axapta",function(hljs){return{keywords:"false int abstract private char boolean static null if for true "+"while long throw finally protected final return void enum else "+"break new catch byte super case short default double public try this switch "+"continue reverse firstfast firstonly forupdate nofetch sum avg minof maxof count "+"order group by asc desc index hint like dispaly edit client server ttsbegin "+"ttscommit str real date container anytype common div mod",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:":",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]}]}});hljs.registerLanguage("purebasic",function(hljs){var STRINGS={className:"string",begin:'(~)?"',end:'"',illegal:"\\n"};var CONSTANTS={className:"symbol",begin:"#[a-zA-Z_]\\w*\\$?"};return{aliases:["pb","pbi"],keywords:"Align And Array As Break CallDebugger Case CompilerCase CompilerDefault "+"CompilerElse CompilerElseIf CompilerEndIf CompilerEndSelect CompilerError "+"CompilerIf CompilerSelect CompilerWarning Continue Data DataSection Debug "+"DebugLevel Declare DeclareC DeclareCDLL DeclareDLL DeclareModule Default "+"Define Dim DisableASM DisableDebugger DisableExplicit Else ElseIf EnableASM "+"EnableDebugger EnableExplicit End EndDataSection EndDeclareModule EndEnumeration "+"EndIf EndImport EndInterface EndMacro EndModule EndProcedure EndSelect "+"EndStructure EndStructureUnion EndWith Enumeration EnumerationBinary Extends "+"FakeReturn For ForEach ForEver Global Gosub Goto If Import ImportC "+"IncludeBinary IncludeFile IncludePath Interface List Macro MacroExpandedCount "+"Map Module NewList NewMap Next Not Or Procedure ProcedureC "+"ProcedureCDLL ProcedureDLL ProcedureReturn Protected Prototype PrototypeC ReDim "+"Read Repeat Restore Return Runtime Select Shared Static Step Structure "+"StructureUnion Swap Threaded To UndefineMacro Until Until  UnuseModule "+"UseModule Wend While With XIncludeFile XOr",contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"function",begin:"\\b(Procedure|Declare)(C|CDLL|DLL)?\\b",end:"\\(",excludeEnd:true,returnBegin:true,contains:[{className:"keyword",begin:"(Procedure|Declare)(C|CDLL|DLL)?",excludeEnd:true},{className:"type",begin:"\\.\\w*"},hljs.UNDERSCORE_TITLE_MODE]},STRINGS,CONSTANTS]}});hljs.registerLanguage("zephir",function(hljs){var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:'b"',end:'"'},{begin:"b'",end:"'"},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]};var NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]};return{aliases:["zep"],case_insensitive:true,keywords:"and include_once list abstract global private echo interface as static endswitch "+"array null if endwhile or const for endforeach self var let while isset public "+"protected exit foreach throw elseif include __FILE__ empty require_once do xor "+"return parent clone use __CLASS__ __LINE__ else break print eval new "+"catch __METHOD__ case exception default die require __FUNCTION__ "+"enddeclare final try switch continue endfor endif declare unset true false "+"trait goto instanceof insteadof __DIR__ __NAMESPACE__ "+"yield finally int uint long ulong char uchar double float bool boolean string"+"likely unlikely",contains:[hljs.C_LINE_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.COMMENT("__halt_compiler.+?;",false,{endsWithParent:true,keywords:"__halt_compiler",lexemes:hljs.UNDERSCORE_IDENT_RE}),{className:"string",begin:"<<<['\"]?\\w+['\"]?$",end:"^\\w+;",contains:[hljs.BACKSLASH_ESCAPE]},{begin:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{className:"function",beginKeywords:"function",end:/[;{]/,excludeEnd:true,illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:["self",hljs.C_BLOCK_COMMENT_MODE,STRING,NUMBER]}]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:/[:\(\$"]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"namespace",end:";",illegal:/[\.']/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"use",end:";",contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"=>"},STRING,NUMBER]}});hljs.registerLanguage("coq",function(hljs){return{keywords:{keyword:"_|0 as at cofix else end exists exists2 fix for forall fun if IF in let "+"match mod Prop return Set then Type using where with "+"Abort About Add Admit Admitted All Arguments Assumptions Axiom Back BackTo "+"Backtrack Bind Blacklist Canonical Cd Check Class Classes Close Coercion "+"Coercions CoFixpoint CoInductive Collection Combined Compute Conjecture "+"Conjectures Constant constr Constraint Constructors Context Corollary "+"CreateHintDb Cut Declare Defined Definition Delimit Dependencies Dependent"+"Derive Drop eauto End Equality Eval Example Existential Existentials "+"Existing Export exporting Extern Extract Extraction Fact Field Fields File "+"Fixpoint Focus for From Function Functional Generalizable Global Goal Grab "+"Grammar Graph Guarded Heap Hint HintDb Hints Hypotheses Hypothesis ident "+"Identity If Immediate Implicit Import Include Inductive Infix Info Initial "+"Inline Inspect Instance Instances Intro Intros Inversion Inversion_clear "+"Language Left Lemma Let Libraries Library Load LoadPath Local Locate Ltac ML "+"Mode Module Modules Monomorphic Morphism Next NoInline Notation Obligation "+"Obligations Opaque Open Optimize Options Parameter Parameters Parametric "+"Path Paths pattern Polymorphic Preterm Print Printing Program Projections "+"Proof Proposition Pwd Qed Quit Rec Record Recursive Redirect Relation Remark "+"Remove Require Reserved Reset Resolve Restart Rewrite Right Ring Rings Save "+"Scheme Scope Scopes Script Search SearchAbout SearchHead SearchPattern "+"SearchRewrite Section Separate Set Setoid Show Solve Sorted Step Strategies "+"Strategy Structure SubClass Table Tables Tactic Term Test Theorem Time "+"Timeout Transparent Type Typeclasses Types Undelimit Undo Unfocus Unfocused "+"Unfold Universe Universes Unset Unshelve using Variable Variables Variant "+"Verbose Visibility where with",built_in:"abstract absurd admit after apply as assert assumption at auto autorewrite "+"autounfold before bottom btauto by case case_eq cbn cbv change "+"classical_left classical_right clear clearbody cofix compare compute "+"congruence constr_eq constructor contradict contradiction cut cutrewrite "+"cycle decide decompose dependent destruct destruction dintuition "+"discriminate discrR do double dtauto eapply eassumption eauto ecase "+"econstructor edestruct ediscriminate eelim eexact eexists einduction "+"einjection eleft elim elimtype enough equality erewrite eright "+"esimplify_eq esplit evar exact exactly_once exfalso exists f_equal fail "+"field field_simplify field_simplify_eq first firstorder fix fold fourier "+"functional generalize generalizing gfail give_up has_evar hnf idtac in "+"induction injection instantiate intro intro_pattern intros intuition "+"inversion inversion_clear is_evar is_var lapply lazy left lia lra move "+"native_compute nia nsatz omega once pattern pose progress proof psatz quote "+"record red refine reflexivity remember rename repeat replace revert "+"revgoals rewrite rewrite_strat right ring ring_simplify rtauto set "+"setoid_reflexivity setoid_replace setoid_rewrite setoid_symmetry "+"setoid_transitivity shelve shelve_unifiable simpl simple simplify_eq solve "+"specialize split split_Rabs split_Rmult stepl stepr subst sum swap "+"symmetry tactic tauto time timeout top transitivity trivial try tryif "+"unfold unify until using vm_compute with"},contains:[hljs.QUOTE_STRING_MODE,hljs.COMMENT("\\(\\*","\\*\\)"),hljs.C_NUMBER_MODE,{className:"type",excludeBegin:true,begin:"\\|\\s*",end:"\\w+"},{begin:/[-=]>/}]}});hljs.registerLanguage("clojure",function(hljs){var keywords={"builtin-name":"def defonce cond apply if-not if-let if not not= = < > <= >= == + / * - rem "+"quot neg? pos? delay? symbol? keyword? true? false? integer? empty? coll? list? "+"set? ifn? fn? associative? sequential? sorted? counted? reversible? number? decimal? "+"class? distinct? isa? float? rational? reduced? ratio? odd? even? char? seq? vector? "+"string? map? nil? contains? zero? instance? not-every? not-any? libspec? -> ->> .. . "+"inc compare do dotimes mapcat take remove take-while drop letfn drop-last take-last "+"drop-while while intern condp case reduced cycle split-at split-with repeat replicate "+"iterate range merge zipmap declare line-seq sort comparator sort-by dorun doall nthnext "+"nthrest partition eval doseq await await-for let agent atom send send-off release-pending-sends "+"add-watch mapv filterv remove-watch agent-error restart-agent set-error-handler error-handler "+"set-error-mode! error-mode shutdown-agents quote var fn loop recur throw try monitor-enter "+"monitor-exit defmacro defn defn- macroexpand macroexpand-1 for dosync and or "+"when when-not when-let comp juxt partial sequence memoize constantly complement identity assert "+"peek pop doto proxy defstruct first rest cons defprotocol cast coll deftype defrecord last butlast "+"sigs reify second ffirst fnext nfirst nnext defmulti defmethod meta with-meta ns in-ns create-ns import "+"refer keys select-keys vals key val rseq name namespace promise into transient persistent! conj! "+"assoc! dissoc! pop! disj! use class type num float double short byte boolean bigint biginteger "+"bigdec print-method print-dup throw-if printf format load compile get-in update-in pr pr-on newline "+"flush read slurp read-line subvec with-open memfn time re-find re-groups rand-int rand mod locking "+"assert-valid-fdecl alias resolve ref deref refset swap! reset! set-validator! compare-and-set! alter-meta! "+"reset-meta! commute get-validator alter ref-set ref-history-count ref-min-history ref-max-history ensure sync io! "+"new next conj set! to-array future future-call into-array aset gen-class reduce map filter find empty "+"hash-map hash-set sorted-map sorted-map-by sorted-set sorted-set-by vec vector seq flatten reverse assoc dissoc list "+"disj get union difference intersection extend extend-type extend-protocol int nth delay count concat chunk chunk-buffer "+"chunk-append chunk-first chunk-rest max min dec unchecked-inc-int unchecked-inc unchecked-dec-inc unchecked-dec unchecked-negate "+"unchecked-add-int unchecked-add unchecked-subtract-int unchecked-subtract chunk-next chunk-cons chunked-seq? prn vary-meta "+"lazy-seq spread list* str find-keyword keyword symbol gensym force rationalize"};var SYMBOLSTART="a-zA-Z_\\-!.?+*=<>&#'";var SYMBOL_RE="["+SYMBOLSTART+"]["+SYMBOLSTART+"0-9/;:]*";var SIMPLE_NUMBER_RE="[-+]?\\d+(\\.\\d+)?";var SYMBOL={begin:SYMBOL_RE,relevance:0};var NUMBER={className:"number",begin:SIMPLE_NUMBER_RE,relevance:0};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var LITERAL={className:"literal",begin:/\b(true|false|nil)\b/};var COLLECTION={begin:"[\\[\\{]",end:"[\\]\\}]"};var HINT={className:"comment",begin:"\\^"+SYMBOL_RE};var HINT_COL=hljs.COMMENT("\\^\\{","\\}");var KEY={className:"symbol",begin:"[:]{1,2}"+SYMBOL_RE};var LIST={begin:"\\(",end:"\\)"};var BODY={endsWithParent:true,relevance:0};var NAME={keywords:keywords,lexemes:SYMBOL_RE,className:"name",begin:SYMBOL_RE,starts:BODY};var DEFAULT_CONTAINS=[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL,SYMBOL];LIST.contains=[hljs.COMMENT("comment",""),NAME,BODY];BODY.contains=DEFAULT_CONTAINS;COLLECTION.contains=DEFAULT_CONTAINS;HINT_COL.contains=[COLLECTION];return{aliases:["clj"],illegal:/\S/,contains:[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL]}});hljs.registerLanguage("clojure-repl",function(hljs){return{contains:[{className:"meta",begin:/^([\w.-]+|\s*#_)?=>/,starts:{end:/$/,subLanguage:"clojure"}}]}});hljs.registerLanguage("java",function(hljs){var JAVA_IDENT_RE="[À-ʸa-zA-Z_$][À-ʸa-zA-Z_$0-9]*";var GENERIC_IDENT_RE=JAVA_IDENT_RE+"(<"+JAVA_IDENT_RE+"(\\s*,\\s*"+JAVA_IDENT_RE+")*>)?";var KEYWORDS="false synchronized int abstract float private char boolean var static null if const "+"for true while long strictfp finally protected import native final void "+"enum else break transient catch instanceof byte super volatile case assert short "+"package default double public try this switch continue throws protected public private "+"module requires exports do";var JAVA_NUMBER_RE="\\b"+"("+"0[bB]([01]+[01_]+[01]+|[01]+)"+"|"+"0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)"+"|"+"("+"([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?"+"|"+"\\.([\\d]+[\\d_]+[\\d]+|[\\d]+)"+")"+"([eE][-+]?\\d+)?"+")"+"[lLfF]?";var JAVA_NUMBER_MODE={className:"number",begin:JAVA_NUMBER_RE,relevance:0};return{aliases:["jsp"],keywords:KEYWORDS,illegal:/<\/|#/,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"class",beginKeywords:"class interface",end:/[{;=]/,excludeEnd:true,keywords:"class interface",illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"new throw return else",relevance:0},{className:"function",begin:"("+GENERIC_IDENT_RE+"\\s+)+"+hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,end:/[{;=]/,excludeEnd:true,keywords:KEYWORDS,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,relevance:0,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},JAVA_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("csp",function(hljs){return{case_insensitive:false,lexemes:"[a-zA-Z][a-zA-Z0-9_-]*",keywords:{keyword:"base-uri child-src connect-src default-src font-src form-action"+" frame-ancestors frame-src img-src media-src object-src plugin-types"+" report-uri sandbox script-src style-src"},contains:[{className:"string",begin:"'",end:"'"},{className:"attribute",begin:"^Content",end:":",excludeEnd:true}]}});hljs.registerLanguage("ruby",function(hljs){var RUBY_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?";var RUBY_KEYWORDS={keyword:"and then defined module in return redo if BEGIN retry end for self when "+"next until do begin unless END rescue else break undef not super class case "+"require yield alias while ensure elsif or include attr_reader attr_writer attr_accessor",literal:"true false nil"};var YARDOCTAG={className:"doctag",begin:"@[A-Za-z]+"};var IRB_OBJECT={begin:"#<",end:">"};var COMMENT_MODES=[hljs.COMMENT("#","$",{contains:[YARDOCTAG]}),hljs.COMMENT("^\\=begin","^\\=end",{contains:[YARDOCTAG],relevance:10}),hljs.COMMENT("^__END__","\\n$")];var SUBST={className:"subst",begin:"#\\{",end:"}",keywords:RUBY_KEYWORDS};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{begin:"%[qQwWx]?\\(",end:"\\)"},{begin:"%[qQwWx]?\\[",end:"\\]"},{begin:"%[qQwWx]?{",end:"}"},{begin:"%[qQwWx]?<",end:">"},{begin:"%[qQwWx]?/",end:"/"},{begin:"%[qQwWx]?%",end:"%"},{begin:"%[qQwWx]?-",end:"-"},{begin:"%[qQwWx]?\\|",end:"\\|"},{begin:/\B\?(\\\d{1,3}|\\x[A-Fa-f0-9]{1,2}|\\u[A-Fa-f0-9]{4}|\\?\S)\b/},{begin:/<<[-~]?'?(\w+)(?:.|\n)*?\n\s*\1\b/,returnBegin:true,contains:[{begin:/<<[-~]?'?/},{begin:/\w+/,endSameAsBegin:true,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]}]};var PARAMS={className:"params",begin:"\\(",end:"\\)",endsParent:true,keywords:RUBY_KEYWORDS};var RUBY_DEFAULT_CONTAINS=[STRING,IRB_OBJECT,{className:"class",beginKeywords:"class module",end:"$|;",illegal:/=/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?"}),{begin:"<\\s*",contains:[{begin:"("+hljs.IDENT_RE+"::)?"+hljs.IDENT_RE}]}].concat(COMMENT_MODES)},{className:"function",beginKeywords:"def",end:"$|;",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:RUBY_METHOD_RE}),PARAMS].concat(COMMENT_MODES)},{begin:hljs.IDENT_RE+"::"},{className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"(\\!|\\?)?:",relevance:0},{className:"symbol",begin:":(?!\\s)",contains:[STRING,{begin:RUBY_METHOD_RE}],relevance:0},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},{begin:"(\\$\\W)|((\\$|\\@\\@?)(\\w+))"},{className:"params",begin:/\|/,end:/\|/,keywords:RUBY_KEYWORDS},{begin:"("+hljs.RE_STARTERS_RE+"|unless)\\s*",keywords:"unless",contains:[IRB_OBJECT,{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],illegal:/\n/,variants:[{begin:"/",end:"/[a-z]*"},{begin:"%r{",end:"}[a-z]*"},{begin:"%r\\(",end:"\\)[a-z]*"},{begin:"%r!",end:"![a-z]*"},{begin:"%r\\[",end:"\\][a-z]*"}]}].concat(COMMENT_MODES),relevance:0}].concat(COMMENT_MODES);SUBST.contains=RUBY_DEFAULT_CONTAINS;PARAMS.contains=RUBY_DEFAULT_CONTAINS;var SIMPLE_PROMPT="[>?]>";var DEFAULT_PROMPT="[\\w#]+\\(\\w+\\):\\d+:\\d+>";var RVM_PROMPT="(\\w+-)?\\d+\\.\\d+\\.\\d(p\\d+)?[^>]+>";var IRB_DEFAULT=[{begin:/^\s*=>/,starts:{end:"$",contains:RUBY_DEFAULT_CONTAINS}},{className:"meta",begin:"^("+SIMPLE_PROMPT+"|"+DEFAULT_PROMPT+"|"+RVM_PROMPT+")",starts:{end:"$",contains:RUBY_DEFAULT_CONTAINS}}];return{aliases:["rb","gemspec","podspec","thor","irb"],keywords:RUBY_KEYWORDS,illegal:/\/\*/,contains:COMMENT_MODES.concat(IRB_DEFAULT).concat(RUBY_DEFAULT_CONTAINS)}});hljs.registerLanguage("cs",function(hljs){var KEYWORDS={keyword:"abstract as base bool break byte case catch char checked const continue decimal "+"default delegate do double enum event explicit extern finally fixed float "+"for foreach goto if implicit in int interface internal is lock long "+"object operator out override params private protected public readonly ref sbyte "+"sealed short sizeof stackalloc static string struct switch this try typeof "+"uint ulong unchecked unsafe ushort using virtual void volatile while "+"add alias ascending async await by descending dynamic equals from get global group into join "+"let nameof on orderby partial remove select set value var when where yield",literal:"null false true"};var NUMBERS={className:"number",variants:[{begin:"\\b(0b[01']+)"},{begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],relevance:0};var VERBATIM_STRING={className:"string",begin:'@"',end:'"',contains:[{begin:'""'}]};var VERBATIM_STRING_NO_LF=hljs.inherit(VERBATIM_STRING,{illegal:/\n/});var SUBST={className:"subst",begin:"{",end:"}",keywords:KEYWORDS};var SUBST_NO_LF=hljs.inherit(SUBST,{illegal:/\n/});var INTERPOLATED_STRING={className:"string",begin:/\$"/,end:'"',illegal:/\n/,contains:[{begin:"{{"},{begin:"}}"},hljs.BACKSLASH_ESCAPE,SUBST_NO_LF]};var INTERPOLATED_VERBATIM_STRING={className:"string",begin:/\$@"/,end:'"',contains:[{begin:"{{"},{begin:"}}"},{begin:'""'},SUBST]};var INTERPOLATED_VERBATIM_STRING_NO_LF=hljs.inherit(INTERPOLATED_VERBATIM_STRING,{illegal:/\n/,contains:[{begin:"{{"},{begin:"}}"},{begin:'""'},SUBST_NO_LF]});SUBST.contains=[INTERPOLATED_VERBATIM_STRING,INTERPOLATED_STRING,VERBATIM_STRING,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,NUMBERS,hljs.C_BLOCK_COMMENT_MODE];SUBST_NO_LF.contains=[INTERPOLATED_VERBATIM_STRING_NO_LF,INTERPOLATED_STRING,VERBATIM_STRING_NO_LF,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,NUMBERS,hljs.inherit(hljs.C_BLOCK_COMMENT_MODE,{illegal:/\n/})];var STRING={variants:[INTERPOLATED_VERBATIM_STRING,INTERPOLATED_STRING,VERBATIM_STRING,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};var TYPE_IDENT_RE=hljs.IDENT_RE+"(<"+hljs.IDENT_RE+"(\\s*,\\s*"+hljs.IDENT_RE+")*>)?(\\[\\])?";return{aliases:["csharp","c#"],keywords:KEYWORDS,illegal:/::/,contains:[hljs.COMMENT("///","$",{returnBegin:true,contains:[{className:"doctag",variants:[{begin:"///",relevance:0},{begin:"\x3c!--|--\x3e"},{begin:"</?",end:">"}]}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elif endif define undef warning error line region endregion pragma checksum"}},STRING,NUMBERS,{beginKeywords:"class interface",end:/[{;=]/,illegal:/[^\s:,]/,contains:[hljs.TITLE_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{beginKeywords:"namespace",end:/[{;=]/,illegal:/[^\s:]/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"[a-zA-Z](\\.?\\w)*"}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"meta",begin:"^\\s*\\[",excludeBegin:true,end:"\\]",excludeEnd:true,contains:[{className:"meta-string",begin:/"/,end:/"/}]},{beginKeywords:"new return throw await else",relevance:0},{className:"function",begin:"("+TYPE_IDENT_RE+"\\s+)+"+hljs.IDENT_RE+"\\s*\\(",returnBegin:true,end:/\s*[{;=]/,excludeEnd:true,keywords:KEYWORDS,contains:[{begin:hljs.IDENT_RE+"\\s*\\(",returnBegin:true,contains:[hljs.TITLE_MODE],relevance:0},{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,relevance:0,contains:[STRING,NUMBERS,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}]}});hljs.registerLanguage("x86asm",function(hljs){return{case_insensitive:true,lexemes:"[.%]?"+hljs.IDENT_RE,keywords:{keyword:"lock rep repe repz repne repnz xaquire xrelease bnd nobnd "+"aaa aad aam aas adc add and arpl bb0_reset bb1_reset bound bsf bsr bswap bt btc btr bts call cbw cdq cdqe clc cld cli clts cmc cmp cmpsb cmpsd cmpsq cmpsw cmpxchg cmpxchg486 cmpxchg8b cmpxchg16b cpuid cpu_read cpu_write cqo cwd cwde daa das dec div dmint emms enter equ f2xm1 fabs fadd faddp fbld fbstp fchs fclex fcmovb fcmovbe fcmove fcmovnb fcmovnbe fcmovne fcmovnu fcmovu fcom fcomi fcomip fcomp fcompp fcos fdecstp fdisi fdiv fdivp fdivr fdivrp femms feni ffree ffreep fiadd ficom ficomp fidiv fidivr fild fimul fincstp finit fist fistp fisttp fisub fisubr fld fld1 fldcw fldenv fldl2e fldl2t fldlg2 fldln2 fldpi fldz fmul fmulp fnclex fndisi fneni fninit fnop fnsave fnstcw fnstenv fnstsw fpatan fprem fprem1 fptan frndint frstor fsave fscale fsetpm fsin fsincos fsqrt fst fstcw fstenv fstp fstsw fsub fsubp fsubr fsubrp ftst fucom fucomi fucomip fucomp fucompp fxam fxch fxtract fyl2x fyl2xp1 hlt ibts icebp idiv imul in inc incbin insb insd insw int int01 int1 int03 int3 into invd invpcid invlpg invlpga iret iretd iretq iretw jcxz jecxz jrcxz jmp jmpe lahf lar lds lea leave les lfence lfs lgdt lgs lidt lldt lmsw loadall loadall286 lodsb lodsd lodsq lodsw loop loope loopne loopnz loopz lsl lss ltr mfence monitor mov movd movq movsb movsd movsq movsw movsx movsxd movzx mul mwait neg nop not or out outsb outsd outsw packssdw packsswb packuswb paddb paddd paddsb paddsiw paddsw paddusb paddusw paddw pand pandn pause paveb pavgusb pcmpeqb pcmpeqd pcmpeqw pcmpgtb pcmpgtd pcmpgtw pdistib pf2id pfacc pfadd pfcmpeq pfcmpge pfcmpgt pfmax pfmin pfmul pfrcp pfrcpit1 pfrcpit2 pfrsqit1 pfrsqrt pfsub pfsubr pi2fd pmachriw pmaddwd pmagw pmulhriw pmulhrwa pmulhrwc pmulhw pmullw pmvgezb pmvlzb pmvnzb pmvzb pop popa popad popaw popf popfd popfq popfw por prefetch prefetchw pslld psllq psllw psrad psraw psrld psrlq psrlw psubb psubd psubsb psubsiw psubsw psubusb psubusw psubw punpckhbw punpckhdq punpckhwd punpcklbw punpckldq punpcklwd push pusha pushad pushaw pushf pushfd pushfq pushfw pxor rcl rcr rdshr rdmsr rdpmc rdtsc rdtscp ret retf retn rol ror rdm rsdc rsldt rsm rsts sahf sal salc sar sbb scasb scasd scasq scasw sfence sgdt shl shld shr shrd sidt sldt skinit smi smint smintold smsw stc std sti stosb stosd stosq stosw str sub svdc svldt svts swapgs syscall sysenter sysexit sysret test ud0 ud1 ud2b ud2 ud2a umov verr verw fwait wbinvd wrshr wrmsr xadd xbts xchg xlatb xlat xor cmove cmovz cmovne cmovnz cmova cmovnbe cmovae cmovnb cmovb cmovnae cmovbe cmovna cmovg cmovnle cmovge cmovnl cmovl cmovnge cmovle cmovng cmovc cmovnc cmovo cmovno cmovs cmovns cmovp cmovpe cmovnp cmovpo je jz jne jnz ja jnbe jae jnb jb jnae jbe jna jg jnle jge jnl jl jnge jle jng jc jnc jo jno js jns jpo jnp jpe jp sete setz setne setnz seta setnbe setae setnb setnc setb setnae setcset setbe setna setg setnle setge setnl setl setnge setle setng sets setns seto setno setpe setp setpo setnp addps addss andnps andps cmpeqps cmpeqss cmpleps cmpless cmpltps cmpltss cmpneqps cmpneqss cmpnleps cmpnless cmpnltps cmpnltss cmpordps cmpordss cmpunordps cmpunordss cmpps cmpss comiss cvtpi2ps cvtps2pi cvtsi2ss cvtss2si cvttps2pi cvttss2si divps divss ldmxcsr maxps maxss minps minss movaps movhps movlhps movlps movhlps movmskps movntps movss movups mulps mulss orps rcpps rcpss rsqrtps rsqrtss shufps sqrtps sqrtss stmxcsr subps subss ucomiss unpckhps unpcklps xorps fxrstor fxrstor64 fxsave fxsave64 xgetbv xsetbv xsave xsave64 xsaveopt xsaveopt64 xrstor xrstor64 prefetchnta prefetcht0 prefetcht1 prefetcht2 maskmovq movntq pavgb pavgw pextrw pinsrw pmaxsw pmaxub pminsw pminub pmovmskb pmulhuw psadbw pshufw pf2iw pfnacc pfpnacc pi2fw pswapd maskmovdqu clflush movntdq movnti movntpd movdqa movdqu movdq2q movq2dq paddq pmuludq pshufd pshufhw pshuflw pslldq psrldq psubq punpckhqdq punpcklqdq addpd addsd andnpd andpd cmpeqpd cmpeqsd cmplepd cmplesd cmpltpd cmpltsd cmpneqpd cmpneqsd cmpnlepd cmpnlesd cmpnltpd cmpnltsd cmpordpd cmpordsd cmpunordpd cmpunordsd cmppd comisd cvtdq2pd cvtdq2ps cvtpd2dq cvtpd2pi cvtpd2ps cvtpi2pd cvtps2dq cvtps2pd cvtsd2si cvtsd2ss cvtsi2sd cvtss2sd cvttpd2pi cvttpd2dq cvttps2dq cvttsd2si divpd divsd maxpd maxsd minpd minsd movapd movhpd movlpd movmskpd movupd mulpd mulsd orpd shufpd sqrtpd sqrtsd subpd subsd ucomisd unpckhpd unpcklpd xorpd addsubpd addsubps haddpd haddps hsubpd hsubps lddqu movddup movshdup movsldup clgi stgi vmcall vmclear vmfunc vmlaunch vmload vmmcall vmptrld vmptrst vmread vmresume vmrun vmsave vmwrite vmxoff vmxon invept invvpid pabsb pabsw pabsd palignr phaddw phaddd phaddsw phsubw phsubd phsubsw pmaddubsw pmulhrsw pshufb psignb psignw psignd extrq insertq movntsd movntss lzcnt blendpd blendps blendvpd blendvps dppd dpps extractps insertps movntdqa mpsadbw packusdw pblendvb pblendw pcmpeqq pextrb pextrd pextrq phminposuw pinsrb pinsrd pinsrq pmaxsb pmaxsd pmaxud pmaxuw pminsb pminsd pminud pminuw pmovsxbw pmovsxbd pmovsxbq pmovsxwd pmovsxwq pmovsxdq pmovzxbw pmovzxbd pmovzxbq pmovzxwd pmovzxwq pmovzxdq pmuldq pmulld ptest roundpd roundps roundsd roundss crc32 pcmpestri pcmpestrm pcmpistri pcmpistrm pcmpgtq popcnt getsec pfrcpv pfrsqrtv movbe aesenc aesenclast aesdec aesdeclast aesimc aeskeygenassist vaesenc vaesenclast vaesdec vaesdeclast vaesimc vaeskeygenassist vaddpd vaddps vaddsd vaddss vaddsubpd vaddsubps vandpd vandps vandnpd vandnps vblendpd vblendps vblendvpd vblendvps vbroadcastss vbroadcastsd vbroadcastf128 vcmpeq_ospd vcmpeqpd vcmplt_ospd vcmpltpd vcmple_ospd vcmplepd vcmpunord_qpd vcmpunordpd vcmpneq_uqpd vcmpneqpd vcmpnlt_uspd vcmpnltpd vcmpnle_uspd vcmpnlepd vcmpord_qpd vcmpordpd vcmpeq_uqpd vcmpnge_uspd vcmpngepd vcmpngt_uspd vcmpngtpd vcmpfalse_oqpd vcmpfalsepd vcmpneq_oqpd vcmpge_ospd vcmpgepd vcmpgt_ospd vcmpgtpd vcmptrue_uqpd vcmptruepd vcmplt_oqpd vcmple_oqpd vcmpunord_spd vcmpneq_uspd vcmpnlt_uqpd vcmpnle_uqpd vcmpord_spd vcmpeq_uspd vcmpnge_uqpd vcmpngt_uqpd vcmpfalse_ospd vcmpneq_ospd vcmpge_oqpd vcmpgt_oqpd vcmptrue_uspd vcmppd vcmpeq_osps vcmpeqps vcmplt_osps vcmpltps vcmple_osps vcmpleps vcmpunord_qps vcmpunordps vcmpneq_uqps vcmpneqps vcmpnlt_usps vcmpnltps vcmpnle_usps vcmpnleps vcmpord_qps vcmpordps vcmpeq_uqps vcmpnge_usps vcmpngeps vcmpngt_usps vcmpngtps vcmpfalse_oqps vcmpfalseps vcmpneq_oqps vcmpge_osps vcmpgeps vcmpgt_osps vcmpgtps vcmptrue_uqps vcmptrueps vcmplt_oqps vcmple_oqps vcmpunord_sps vcmpneq_usps vcmpnlt_uqps vcmpnle_uqps vcmpord_sps vcmpeq_usps vcmpnge_uqps vcmpngt_uqps vcmpfalse_osps vcmpneq_osps vcmpge_oqps vcmpgt_oqps vcmptrue_usps vcmpps vcmpeq_ossd vcmpeqsd vcmplt_ossd vcmpltsd vcmple_ossd vcmplesd vcmpunord_qsd vcmpunordsd vcmpneq_uqsd vcmpneqsd vcmpnlt_ussd vcmpnltsd vcmpnle_ussd vcmpnlesd vcmpord_qsd vcmpordsd vcmpeq_uqsd vcmpnge_ussd vcmpngesd vcmpngt_ussd vcmpngtsd vcmpfalse_oqsd vcmpfalsesd vcmpneq_oqsd vcmpge_ossd vcmpgesd vcmpgt_ossd vcmpgtsd vcmptrue_uqsd vcmptruesd vcmplt_oqsd vcmple_oqsd vcmpunord_ssd vcmpneq_ussd vcmpnlt_uqsd vcmpnle_uqsd vcmpord_ssd vcmpeq_ussd vcmpnge_uqsd vcmpngt_uqsd vcmpfalse_ossd vcmpneq_ossd vcmpge_oqsd vcmpgt_oqsd vcmptrue_ussd vcmpsd vcmpeq_osss vcmpeqss vcmplt_osss vcmpltss vcmple_osss vcmpless vcmpunord_qss vcmpunordss vcmpneq_uqss vcmpneqss vcmpnlt_usss vcmpnltss vcmpnle_usss vcmpnless vcmpord_qss vcmpordss vcmpeq_uqss vcmpnge_usss vcmpngess vcmpngt_usss vcmpngtss vcmpfalse_oqss vcmpfalsess vcmpneq_oqss vcmpge_osss vcmpgess vcmpgt_osss vcmpgtss vcmptrue_uqss vcmptruess vcmplt_oqss vcmple_oqss vcmpunord_sss vcmpneq_usss vcmpnlt_uqss vcmpnle_uqss vcmpord_sss vcmpeq_usss vcmpnge_uqss vcmpngt_uqss vcmpfalse_osss vcmpneq_osss vcmpge_oqss vcmpgt_oqss vcmptrue_usss vcmpss vcomisd vcomiss vcvtdq2pd vcvtdq2ps vcvtpd2dq vcvtpd2ps vcvtps2dq vcvtps2pd vcvtsd2si vcvtsd2ss vcvtsi2sd vcvtsi2ss vcvtss2sd vcvtss2si vcvttpd2dq vcvttps2dq vcvttsd2si vcvttss2si vdivpd vdivps vdivsd vdivss vdppd vdpps vextractf128 vextractps vhaddpd vhaddps vhsubpd vhsubps vinsertf128 vinsertps vlddqu vldqqu vldmxcsr vmaskmovdqu vmaskmovps vmaskmovpd vmaxpd vmaxps vmaxsd vmaxss vminpd vminps vminsd vminss vmovapd vmovaps vmovd vmovq vmovddup vmovdqa vmovqqa vmovdqu vmovqqu vmovhlps vmovhpd vmovhps vmovlhps vmovlpd vmovlps vmovmskpd vmovmskps vmovntdq vmovntqq vmovntdqa vmovntpd vmovntps vmovsd vmovshdup vmovsldup vmovss vmovupd vmovups vmpsadbw vmulpd vmulps vmulsd vmulss vorpd vorps vpabsb vpabsw vpabsd vpacksswb vpackssdw vpackuswb vpackusdw vpaddb vpaddw vpaddd vpaddq vpaddsb vpaddsw vpaddusb vpaddusw vpalignr vpand vpandn vpavgb vpavgw vpblendvb vpblendw vpcmpestri vpcmpestrm vpcmpistri vpcmpistrm vpcmpeqb vpcmpeqw vpcmpeqd vpcmpeqq vpcmpgtb vpcmpgtw vpcmpgtd vpcmpgtq vpermilpd vpermilps vperm2f128 vpextrb vpextrw vpextrd vpextrq vphaddw vphaddd vphaddsw vphminposuw vphsubw vphsubd vphsubsw vpinsrb vpinsrw vpinsrd vpinsrq vpmaddwd vpmaddubsw vpmaxsb vpmaxsw vpmaxsd vpmaxub vpmaxuw vpmaxud vpminsb vpminsw vpminsd vpminub vpminuw vpminud vpmovmskb vpmovsxbw vpmovsxbd vpmovsxbq vpmovsxwd vpmovsxwq vpmovsxdq vpmovzxbw vpmovzxbd vpmovzxbq vpmovzxwd vpmovzxwq vpmovzxdq vpmulhuw vpmulhrsw vpmulhw vpmullw vpmulld vpmuludq vpmuldq vpor vpsadbw vpshufb vpshufd vpshufhw vpshuflw vpsignb vpsignw vpsignd vpslldq vpsrldq vpsllw vpslld vpsllq vpsraw vpsrad vpsrlw vpsrld vpsrlq vptest vpsubb vpsubw vpsubd vpsubq vpsubsb vpsubsw vpsubusb vpsubusw vpunpckhbw vpunpckhwd vpunpckhdq vpunpckhqdq vpunpcklbw vpunpcklwd vpunpckldq vpunpcklqdq vpxor vrcpps vrcpss vrsqrtps vrsqrtss vroundpd vroundps vroundsd vroundss vshufpd vshufps vsqrtpd vsqrtps vsqrtsd vsqrtss vstmxcsr vsubpd vsubps vsubsd vsubss vtestps vtestpd vucomisd vucomiss vunpckhpd vunpckhps vunpcklpd vunpcklps vxorpd vxorps vzeroall vzeroupper pclmullqlqdq pclmulhqlqdq pclmullqhqdq pclmulhqhqdq pclmulqdq vpclmullqlqdq vpclmulhqlqdq vpclmullqhqdq vpclmulhqhqdq vpclmulqdq vfmadd132ps vfmadd132pd vfmadd312ps vfmadd312pd vfmadd213ps vfmadd213pd vfmadd123ps vfmadd123pd vfmadd231ps vfmadd231pd vfmadd321ps vfmadd321pd vfmaddsub132ps vfmaddsub132pd vfmaddsub312ps vfmaddsub312pd vfmaddsub213ps vfmaddsub213pd vfmaddsub123ps vfmaddsub123pd vfmaddsub231ps vfmaddsub231pd vfmaddsub321ps vfmaddsub321pd vfmsub132ps vfmsub132pd vfmsub312ps vfmsub312pd vfmsub213ps vfmsub213pd vfmsub123ps vfmsub123pd vfmsub231ps vfmsub231pd vfmsub321ps vfmsub321pd vfmsubadd132ps vfmsubadd132pd vfmsubadd312ps vfmsubadd312pd vfmsubadd213ps vfmsubadd213pd vfmsubadd123ps vfmsubadd123pd vfmsubadd231ps vfmsubadd231pd vfmsubadd321ps vfmsubadd321pd vfnmadd132ps vfnmadd132pd vfnmadd312ps vfnmadd312pd vfnmadd213ps vfnmadd213pd vfnmadd123ps vfnmadd123pd vfnmadd231ps vfnmadd231pd vfnmadd321ps vfnmadd321pd vfnmsub132ps vfnmsub132pd vfnmsub312ps vfnmsub312pd vfnmsub213ps vfnmsub213pd vfnmsub123ps vfnmsub123pd vfnmsub231ps vfnmsub231pd vfnmsub321ps vfnmsub321pd vfmadd132ss vfmadd132sd vfmadd312ss vfmadd312sd vfmadd213ss vfmadd213sd vfmadd123ss vfmadd123sd vfmadd231ss vfmadd231sd vfmadd321ss vfmadd321sd vfmsub132ss vfmsub132sd vfmsub312ss vfmsub312sd vfmsub213ss vfmsub213sd vfmsub123ss vfmsub123sd vfmsub231ss vfmsub231sd vfmsub321ss vfmsub321sd vfnmadd132ss vfnmadd132sd vfnmadd312ss vfnmadd312sd vfnmadd213ss vfnmadd213sd vfnmadd123ss vfnmadd123sd vfnmadd231ss vfnmadd231sd vfnmadd321ss vfnmadd321sd vfnmsub132ss vfnmsub132sd vfnmsub312ss vfnmsub312sd vfnmsub213ss vfnmsub213sd vfnmsub123ss vfnmsub123sd vfnmsub231ss vfnmsub231sd vfnmsub321ss vfnmsub321sd rdfsbase rdgsbase rdrand wrfsbase wrgsbase vcvtph2ps vcvtps2ph adcx adox rdseed clac stac xstore xcryptecb xcryptcbc xcryptctr xcryptcfb xcryptofb montmul xsha1 xsha256 llwpcb slwpcb lwpval lwpins vfmaddpd vfmaddps vfmaddsd vfmaddss vfmaddsubpd vfmaddsubps vfmsubaddpd vfmsubaddps vfmsubpd vfmsubps vfmsubsd vfmsubss vfnmaddpd vfnmaddps vfnmaddsd vfnmaddss vfnmsubpd vfnmsubps vfnmsubsd vfnmsubss vfrczpd vfrczps vfrczsd vfrczss vpcmov vpcomb vpcomd vpcomq vpcomub vpcomud vpcomuq vpcomuw vpcomw vphaddbd vphaddbq vphaddbw vphadddq vphaddubd vphaddubq vphaddubw vphaddudq vphadduwd vphadduwq vphaddwd vphaddwq vphsubbw vphsubdq vphsubwd vpmacsdd vpmacsdqh vpmacsdql vpmacssdd vpmacssdqh vpmacssdql vpmacsswd vpmacssww vpmacswd vpmacsww vpmadcsswd vpmadcswd vpperm vprotb vprotd vprotq vprotw vpshab vpshad vpshaq vpshaw vpshlb vpshld vpshlq vpshlw vbroadcasti128 vpblendd vpbroadcastb vpbroadcastw vpbroadcastd vpbroadcastq vpermd vpermpd vpermps vpermq vperm2i128 vextracti128 vinserti128 vpmaskmovd vpmaskmovq vpsllvd vpsllvq vpsravd vpsrlvd vpsrlvq vgatherdpd vgatherqpd vgatherdps vgatherqps vpgatherdd vpgatherqd vpgatherdq vpgatherqq xabort xbegin xend xtest andn bextr blci blcic blsi blsic blcfill blsfill blcmsk blsmsk blsr blcs bzhi mulx pdep pext rorx sarx shlx shrx tzcnt tzmsk t1mskc valignd valignq vblendmpd vblendmps vbroadcastf32x4 vbroadcastf64x4 vbroadcasti32x4 vbroadcasti64x4 vcompresspd vcompressps vcvtpd2udq vcvtps2udq vcvtsd2usi vcvtss2usi vcvttpd2udq vcvttps2udq vcvttsd2usi vcvttss2usi vcvtudq2pd vcvtudq2ps vcvtusi2sd vcvtusi2ss vexpandpd vexpandps vextractf32x4 vextractf64x4 vextracti32x4 vextracti64x4 vfixupimmpd vfixupimmps vfixupimmsd vfixupimmss vgetexppd vgetexpps vgetexpsd vgetexpss vgetmantpd vgetmantps vgetmantsd vgetmantss vinsertf32x4 vinsertf64x4 vinserti32x4 vinserti64x4 vmovdqa32 vmovdqa64 vmovdqu32 vmovdqu64 vpabsq vpandd vpandnd vpandnq vpandq vpblendmd vpblendmq vpcmpltd vpcmpled vpcmpneqd vpcmpnltd vpcmpnled vpcmpd vpcmpltq vpcmpleq vpcmpneqq vpcmpnltq vpcmpnleq vpcmpq vpcmpequd vpcmpltud vpcmpleud vpcmpnequd vpcmpnltud vpcmpnleud vpcmpud vpcmpequq vpcmpltuq vpcmpleuq vpcmpnequq vpcmpnltuq vpcmpnleuq vpcmpuq vpcompressd vpcompressq vpermi2d vpermi2pd vpermi2ps vpermi2q vpermt2d vpermt2pd vpermt2ps vpermt2q vpexpandd vpexpandq vpmaxsq vpmaxuq vpminsq vpminuq vpmovdb vpmovdw vpmovqb vpmovqd vpmovqw vpmovsdb vpmovsdw vpmovsqb vpmovsqd vpmovsqw vpmovusdb vpmovusdw vpmovusqb vpmovusqd vpmovusqw vpord vporq vprold vprolq vprolvd vprolvq vprord vprorq vprorvd vprorvq vpscatterdd vpscatterdq vpscatterqd vpscatterqq vpsraq vpsravq vpternlogd vpternlogq vptestmd vptestmq vptestnmd vptestnmq vpxord vpxorq vrcp14pd vrcp14ps vrcp14sd vrcp14ss vrndscalepd vrndscaleps vrndscalesd vrndscaless vrsqrt14pd vrsqrt14ps vrsqrt14sd vrsqrt14ss vscalefpd vscalefps vscalefsd vscalefss vscatterdpd vscatterdps vscatterqpd vscatterqps vshuff32x4 vshuff64x2 vshufi32x4 vshufi64x2 kandnw kandw kmovw knotw kortestw korw kshiftlw kshiftrw kunpckbw kxnorw kxorw vpbroadcastmb2q vpbroadcastmw2d vpconflictd vpconflictq vplzcntd vplzcntq vexp2pd vexp2ps vrcp28pd vrcp28ps vrcp28sd vrcp28ss vrsqrt28pd vrsqrt28ps vrsqrt28sd vrsqrt28ss vgatherpf0dpd vgatherpf0dps vgatherpf0qpd vgatherpf0qps vgatherpf1dpd vgatherpf1dps vgatherpf1qpd vgatherpf1qps vscatterpf0dpd vscatterpf0dps vscatterpf0qpd vscatterpf0qps vscatterpf1dpd vscatterpf1dps vscatterpf1qpd vscatterpf1qps prefetchwt1 bndmk bndcl bndcu bndcn bndmov bndldx bndstx sha1rnds4 sha1nexte sha1msg1 sha1msg2 sha256rnds2 sha256msg1 sha256msg2 hint_nop0 hint_nop1 hint_nop2 hint_nop3 hint_nop4 hint_nop5 hint_nop6 hint_nop7 hint_nop8 hint_nop9 hint_nop10 hint_nop11 hint_nop12 hint_nop13 hint_nop14 hint_nop15 hint_nop16 hint_nop17 hint_nop18 hint_nop19 hint_nop20 hint_nop21 hint_nop22 hint_nop23 hint_nop24 hint_nop25 hint_nop26 hint_nop27 hint_nop28 hint_nop29 hint_nop30 hint_nop31 hint_nop32 hint_nop33 hint_nop34 hint_nop35 hint_nop36 hint_nop37 hint_nop38 hint_nop39 hint_nop40 hint_nop41 hint_nop42 hint_nop43 hint_nop44 hint_nop45 hint_nop46 hint_nop47 hint_nop48 hint_nop49 hint_nop50 hint_nop51 hint_nop52 hint_nop53 hint_nop54 hint_nop55 hint_nop56 hint_nop57 hint_nop58 hint_nop59 hint_nop60 hint_nop61 hint_nop62 hint_nop63",built_in:"ip eip rip "+"al ah bl bh cl ch dl dh sil dil bpl spl r8b r9b r10b r11b r12b r13b r14b r15b "+"ax bx cx dx si di bp sp r8w r9w r10w r11w r12w r13w r14w r15w "+"eax ebx ecx edx esi edi ebp esp eip r8d r9d r10d r11d r12d r13d r14d r15d "+"rax rbx rcx rdx rsi rdi rbp rsp r8 r9 r10 r11 r12 r13 r14 r15 "+"cs ds es fs gs ss "+"st st0 st1 st2 st3 st4 st5 st6 st7 "+"mm0 mm1 mm2 mm3 mm4 mm5 mm6 mm7 "+"xmm0  xmm1  xmm2  xmm3  xmm4  xmm5  xmm6  xmm7  xmm8  xmm9 xmm10  xmm11 xmm12 xmm13 xmm14 xmm15 "+"xmm16 xmm17 xmm18 xmm19 xmm20 xmm21 xmm22 xmm23 xmm24 xmm25 xmm26 xmm27 xmm28 xmm29 xmm30 xmm31 "+"ymm0  ymm1  ymm2  ymm3  ymm4  ymm5  ymm6  ymm7  ymm8  ymm9 ymm10  ymm11 ymm12 ymm13 ymm14 ymm15 "+"ymm16 ymm17 ymm18 ymm19 ymm20 ymm21 ymm22 ymm23 ymm24 ymm25 ymm26 ymm27 ymm28 ymm29 ymm30 ymm31 "+"zmm0  zmm1  zmm2  zmm3  zmm4  zmm5  zmm6  zmm7  zmm8  zmm9 zmm10  zmm11 zmm12 zmm13 zmm14 zmm15 "+"zmm16 zmm17 zmm18 zmm19 zmm20 zmm21 zmm22 zmm23 zmm24 zmm25 zmm26 zmm27 zmm28 zmm29 zmm30 zmm31 "+"k0 k1 k2 k3 k4 k5 k6 k7 "+"bnd0 bnd1 bnd2 bnd3 "+"cr0 cr1 cr2 cr3 cr4 cr8 dr0 dr1 dr2 dr3 dr8 tr3 tr4 tr5 tr6 tr7 "+"r0 r1 r2 r3 r4 r5 r6 r7 r0b r1b r2b r3b r4b r5b r6b r7b "+"r0w r1w r2w r3w r4w r5w r6w r7w r0d r1d r2d r3d r4d r5d r6d r7d "+"r0h r1h r2h r3h "+"r0l r1l r2l r3l r4l r5l r6l r7l r8l r9l r10l r11l r12l r13l r14l r15l "+"db dw dd dq dt ddq do dy dz "+"resb resw resd resq rest resdq reso resy resz "+"incbin equ times "+"byte word dword qword nosplit rel abs seg wrt strict near far a32 ptr",meta:"%define %xdefine %+ %undef %defstr %deftok %assign %strcat %strlen %substr %rotate %elif %else %endif "+"%if %ifmacro %ifctx %ifidn %ifidni %ifid %ifnum %ifstr %iftoken %ifempty %ifenv %error %warning %fatal %rep "+"%endrep %include %push %pop %repl %pathsearch %depend %use %arg %stacksize %local %line %comment %endcomment "+".nolist "+"__FILE__ __LINE__ __SECT__  __BITS__ __OUTPUT_FORMAT__ __DATE__ __TIME__ __DATE_NUM__ __TIME_NUM__ "+"__UTC_DATE__ __UTC_TIME__ __UTC_DATE_NUM__ __UTC_TIME_NUM__  __PASS__ struc endstruc istruc at iend "+"align alignb sectalign daz nodaz up down zero default option assume public "+"bits use16 use32 use64 default section segment absolute extern global common cpu float "+"__utf16__ __utf16le__ __utf16be__ __utf32__ __utf32le__ __utf32be__ "+"__float8__ __float16__ __float32__ __float64__ __float80m__ __float80e__ __float128l__ __float128h__ "+"__Infinity__ __QNaN__ __SNaN__ Inf NaN QNaN SNaN float8 float16 float32 float64 float80m float80e "+"float128l float128h __FLOAT_DAZ__ __FLOAT_ROUND__ __FLOAT__"},contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"number",variants:[{begin:"\\b(?:([0-9][0-9_]*)?\\.[0-9_]*(?:[eE][+-]?[0-9_]+)?|"+"(0[Xx])?[0-9][0-9_]*\\.?[0-9_]*(?:[pP](?:[+-]?[0-9_]+)?)?)\\b",relevance:0},{begin:"\\$[0-9][0-9A-Fa-f]*",relevance:0},{begin:"\\b(?:[0-9A-Fa-f][0-9A-Fa-f_]*[Hh]|[0-9][0-9_]*[DdTt]?|[0-7][0-7_]*[QqOo]|[0-1][0-1_]*[BbYy])\\b"},{begin:"\\b(?:0[Xx][0-9A-Fa-f_]+|0[DdTt][0-9_]+|0[QqOo][0-7_]+|0[BbYy][0-1_]+)\\b"}]},hljs.QUOTE_STRING_MODE,{className:"string",variants:[{begin:"'",end:"[^\\\\]'"},{begin:"`",end:"[^\\\\]`"}],relevance:0},{className:"symbol",variants:[{begin:"^\\s*[A-Za-z._?][A-Za-z0-9_$#@~.?]*(:|\\s+label)"},{begin:"^\\s*%%[A-Za-z0-9_$#@~.?]*:"}],relevance:0},{className:"subst",begin:"%[0-9]+",relevance:0},{className:"subst",begin:"%!S+",relevance:0},{className:"meta",begin:/^\s*\.[\w_-]+/}]}});hljs.registerLanguage("mipsasm",function(hljs){return{case_insensitive:true,aliases:["mips"],lexemes:"\\.?"+hljs.IDENT_RE,keywords:{meta:".2byte .4byte .align .ascii .asciz .balign .byte .code .data .else .end .endif .endm .endr .equ .err .exitm .extern .global .hword .if .ifdef .ifndef .include .irp .long .macro .rept .req .section .set .skip .space .text .word .ltorg ",built_in:"$0 $1 $2 $3 $4 $5 $6 $7 $8 $9 $10 $11 $12 $13 $14 $15 "+"$16 $17 $18 $19 $20 $21 $22 $23 $24 $25 $26 $27 $28 $29 $30 $31 "+"zero at v0 v1 a0 a1 a2 a3 a4 a5 a6 a7 "+"t0 t1 t2 t3 t4 t5 t6 t7 t8 t9 s0 s1 s2 s3 s4 s5 s6 s7 s8 "+"k0 k1 gp sp fp ra "+"$f0 $f1 $f2 $f2 $f4 $f5 $f6 $f7 $f8 $f9 $f10 $f11 $f12 $f13 $f14 $f15 "+"$f16 $f17 $f18 $f19 $f20 $f21 $f22 $f23 $f24 $f25 $f26 $f27 $f28 $f29 $f30 $f31 "+"Context Random EntryLo0 EntryLo1 Context PageMask Wired EntryHi "+"HWREna BadVAddr Count Compare SR IntCtl SRSCtl SRSMap Cause EPC PRId "+"EBase Config Config1 Config2 Config3 LLAddr Debug DEPC DESAVE CacheErr "+"ECC ErrorEPC TagLo DataLo TagHi DataHi WatchLo WatchHi PerfCtl PerfCnt "},contains:[{className:"keyword",begin:"\\b("+"addi?u?|andi?|b(al)?|beql?|bgez(al)?l?|bgtzl?|blezl?|bltz(al)?l?|"+"bnel?|cl[oz]|divu?|ext|ins|j(al)?|jalr(.hb)?|jr(.hb)?|lbu?|lhu?|"+"ll|lui|lw[lr]?|maddu?|mfhi|mflo|movn|movz|move|msubu?|mthi|mtlo|mul|"+"multu?|nop|nor|ori?|rotrv?|sb|sc|se[bh]|sh|sllv?|slti?u?|srav?|"+"srlv?|subu?|sw[lr]?|xori?|wsbh|"+"abs.[sd]|add.[sd]|alnv.ps|bc1[ft]l?|"+"c.(s?f|un|u?eq|[ou]lt|[ou]le|ngle?|seq|l[et]|ng[et]).[sd]|"+"(ceil|floor|round|trunc).[lw].[sd]|cfc1|cvt.d.[lsw]|"+"cvt.l.[dsw]|cvt.ps.s|cvt.s.[dlw]|cvt.s.p[lu]|cvt.w.[dls]|"+"div.[ds]|ldx?c1|luxc1|lwx?c1|madd.[sd]|mfc1|mov[fntz]?.[ds]|"+"msub.[sd]|mth?c1|mul.[ds]|neg.[ds]|nmadd.[ds]|nmsub.[ds]|"+"p[lu][lu].ps|recip.fmt|r?sqrt.[ds]|sdx?c1|sub.[ds]|suxc1|"+"swx?c1|"+"break|cache|d?eret|[de]i|ehb|mfc0|mtc0|pause|prefx?|rdhwr|"+"rdpgpr|sdbbp|ssnop|synci?|syscall|teqi?|tgei?u?|tlb(p|r|w[ir])|"+"tlti?u?|tnei?|wait|wrpgpr"+")",end:"\\s"},hljs.COMMENT("[;#](?!s*$)","$"),hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",relevance:0},{className:"title",begin:"\\|",end:"\\|",illegal:"\\n",relevance:0},{className:"number",variants:[{begin:"0x[0-9a-f]+"},{begin:"\\b-?\\d+"}],relevance:0},{className:"symbol",variants:[{begin:"^\\s*[a-z_\\.\\$][a-z0-9_\\.\\$]+:"},{begin:"^\\s*[0-9]+:"},{begin:"[0-9]+[bf]"}],relevance:0}],illegal:"/"}});hljs.registerLanguage("gams",function(hljs){var KEYWORDS={keyword:"abort acronym acronyms alias all and assign binary card diag display "+"else eq file files for free ge gt if integer le loop lt maximizing "+"minimizing model models ne negative no not option options or ord "+"positive prod put putpage puttl repeat sameas semicont semiint smax "+"smin solve sos1 sos2 sum system table then until using while xor yes",literal:"eps inf na","built-in":"abs arccos arcsin arctan arctan2 Beta betaReg binomial ceil centropy "+"cos cosh cvPower div div0 eDist entropy errorf execSeed exp fact "+"floor frac gamma gammaReg log logBeta logGamma log10 log2 mapVal max "+"min mod ncpCM ncpF ncpVUpow ncpVUsin normal pi poly power "+"randBinomial randLinear randTriangle round rPower sigmoid sign "+"signPower sin sinh slexp sllog10 slrec sqexp sqlog10 sqr sqrec sqrt "+"tan tanh trunc uniform uniformInt vcPower bool_and bool_eqv bool_imp "+"bool_not bool_or bool_xor ifThen rel_eq rel_ge rel_gt rel_le rel_lt "+"rel_ne gday gdow ghour gleap gmillisec gminute gmonth gsecond gyear "+"jdate jnow jstart jtime errorLevel execError gamsRelease gamsVersion "+"handleCollect handleDelete handleStatus handleSubmit heapFree "+"heapLimit heapSize jobHandle jobKill jobStatus jobTerminate "+"licenseLevel licenseStatus maxExecError sleep timeClose timeComp "+"timeElapsed timeExec timeStart"};var PARAMS={className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true};var SYMBOLS={className:"symbol",variants:[{begin:/\=[lgenxc]=/},{begin:/\$/}]};var QSTR={className:"comment",variants:[{begin:"'",end:"'"},{begin:'"',end:'"'}],illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};var ASSIGNMENT={begin:"/",end:"/",keywords:KEYWORDS,contains:[QSTR,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE]};var DESCTEXT={begin:/[a-z][a-z0-9_]*(\([a-z0-9_, ]*\))?[ \t]+/,excludeBegin:true,end:"$",endsWithParent:true,contains:[QSTR,ASSIGNMENT,{className:"comment",begin:/([ ]*[a-z0-9&#*=?@>\\<:\-,()$\[\]_.{}!+%^]+)+/,relevance:0}]};return{aliases:["gms"],case_insensitive:true,keywords:KEYWORDS,contains:[hljs.COMMENT(/^\$ontext/,/^\$offtext/),{className:"meta",begin:"^\\$[a-z0-9]+",end:"$",returnBegin:true,contains:[{className:"meta-keyword",begin:"^\\$[a-z0-9]+"}]},hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{beginKeywords:"set sets parameter parameters variable variables "+"scalar scalars equation equations",end:";",contains:[hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,ASSIGNMENT,DESCTEXT]},{beginKeywords:"table",end:";",returnBegin:true,contains:[{beginKeywords:"table",end:"$",contains:[DESCTEXT]},hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE]},{className:"function",begin:/^[a-z][a-z0-9_,\-+' ()$]+\.{2}/,returnBegin:true,contains:[{className:"title",begin:/^[a-z0-9_]+/},PARAMS,SYMBOLS]},hljs.C_NUMBER_MODE,SYMBOLS]}});hljs.registerLanguage("nimrod",function(hljs){return{aliases:["nim"],keywords:{keyword:"addr and as asm bind block break case cast const continue converter "+"discard distinct div do elif else end enum except export finally "+"for from generic if import in include interface is isnot iterator "+"let macro method mixin mod nil not notin object of or out proc ptr "+"raise ref return shl shr static template try tuple type using var "+"when while with without xor yield",literal:"shared guarded stdin stdout stderr result true false",built_in:"int int8 int16 int32 int64 uint uint8 uint16 uint32 uint64 float "+"float32 float64 bool char string cstring pointer expr stmt void "+"auto any range array openarray varargs seq set clong culong cchar "+"cschar cshort cint csize clonglong cfloat cdouble clongdouble "+"cuchar cushort cuint culonglong cstringarray semistatic"},contains:[{className:"meta",begin:/{\./,end:/\.}/,relevance:10},{className:"string",begin:/[a-zA-Z]\w*"/,end:/"/,contains:[{begin:/""/}]},{className:"string",begin:/([a-zA-Z]\w*)?"""/,end:/"""/},hljs.QUOTE_STRING_MODE,{className:"type",begin:/\b[A-Z]\w+\b/,relevance:0},{className:"number",relevance:0,variants:[{begin:/\b(0[xX][0-9a-fA-F][_0-9a-fA-F]*)('?[iIuU](8|16|32|64))?/},{begin:/\b(0o[0-7][_0-7]*)('?[iIuUfF](8|16|32|64))?/},{begin:/\b(0(b|B)[01][_01]*)('?[iIuUfF](8|16|32|64))?/},{begin:/\b(\d[_\d]*)('?[iIuUfF](8|16|32|64))?/}]},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("dts",function(hljs){var STRINGS={className:"string",variants:[hljs.inherit(hljs.QUOTE_STRING_MODE,{begin:'((u8?|U)|L)?"'}),{begin:'(u8?|U)?R"',end:'"',contains:[hljs.BACKSLASH_ESCAPE]},{begin:"'\\\\?.",end:"'",illegal:"."}]};var NUMBERS={className:"number",variants:[{begin:"\\b(\\d+(\\.\\d*)?|\\.\\d+)(u|U|l|L|ul|UL|f|F)"},{begin:hljs.C_NUMBER_RE}],relevance:0};var PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elif endif define undef ifdef ifndef"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",end:"$",keywords:{"meta-keyword":"include"},contains:[hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:"<",end:">",illegal:"\\n"}]},STRINGS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};var DTS_REFERENCE={className:"variable",begin:"\\&[a-z\\d_]*\\b"};var DTS_KEYWORD={className:"meta-keyword",begin:"/[a-z][a-z\\d-]*/"};var DTS_LABEL={className:"symbol",begin:"^\\s*[a-zA-Z_][a-zA-Z\\d_]*:"};var DTS_CELL_PROPERTY={className:"params",begin:"<",end:">",contains:[NUMBERS,DTS_REFERENCE]};var DTS_NODE={className:"class",begin:/[a-zA-Z_][a-zA-Z\d_@]*\s{/,end:/[{;=]/,returnBegin:true,excludeEnd:true};var DTS_ROOT_NODE={className:"class",begin:"/\\s*{",end:"};",relevance:10,contains:[DTS_REFERENCE,DTS_KEYWORD,DTS_LABEL,DTS_NODE,DTS_CELL_PROPERTY,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS]};return{keywords:"",contains:[DTS_ROOT_NODE,DTS_REFERENCE,DTS_KEYWORD,DTS_LABEL,DTS_NODE,DTS_CELL_PROPERTY,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS,PREPROCESSOR,{begin:hljs.IDENT_RE+"::",keywords:""}]}});hljs.registerLanguage("step21",function(hljs){var STEP21_IDENT_RE="[A-Z_][A-Z0-9_.]*";var STEP21_KEYWORDS={keyword:"HEADER ENDSEC DATA"};var STEP21_START={className:"meta",begin:"ISO-10303-21;",relevance:10};var STEP21_CLOSE={className:"meta",begin:"END-ISO-10303-21;",relevance:10};return{aliases:["p21","step","stp"],case_insensitive:true,lexemes:STEP21_IDENT_RE,keywords:STEP21_KEYWORDS,contains:[STEP21_START,STEP21_CLOSE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT("/\\*\\*!","\\*/"),hljs.C_NUMBER_MODE,hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"string",begin:"'",end:"'"},{className:"symbol",variants:[{begin:"#",end:"\\d+",illegal:"\\W"}]}]}});hljs.registerLanguage("sml",function(hljs){return{aliases:["ml"],keywords:{keyword:"abstype and andalso as case datatype do else end eqtype "+"exception fn fun functor handle if in include infix infixr "+"let local nonfix of op open orelse raise rec sharing sig "+"signature struct structure then type val with withtype where while",built_in:"array bool char exn int list option order real ref string substring vector unit word",literal:"true false NONE SOME LESS EQUAL GREATER nil"},illegal:/\/\/|>>/,lexemes:"[a-z_]\\w*!?",contains:[{className:"literal",begin:/\[(\|\|)?\]|\(\)/,relevance:0},hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),{className:"symbol",begin:"'[A-Za-z_](?!')[\\w']*"},{className:"type",begin:"`[A-Z][\\w']*"},{className:"type",begin:"\\b[A-Z][\\w']*",relevance:0},{begin:"[a-z_]\\w*'[\\w']*"},hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"number",begin:"\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)",relevance:0},{begin:/[-=]>/}]}});hljs.registerLanguage("fsharp",function(hljs){var TYPEPARAM={begin:"<",end:">",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/'[a-zA-Z0-9_]+/})]};return{aliases:["fs"],keywords:"abstract and as assert base begin class default delegate do done "+"downcast downto elif else end exception extern false finally for "+"fun function global if in inherit inline interface internal lazy let "+"match member module mutable namespace new null of open or "+"override private public rec return sig static struct then to "+"true try type upcast use val void when while with yield",illegal:/\/\*/,contains:[{className:"keyword",begin:/\b(yield|return|let|do)!/},{className:"string",begin:'@"',end:'"',contains:[{begin:'""'}]},{className:"string",begin:'"""',end:'"""'},hljs.COMMENT("\\(\\*","\\*\\)"),{className:"class",beginKeywords:"type",end:"\\(|=|$",excludeEnd:true,contains:[hljs.UNDERSCORE_TITLE_MODE,TYPEPARAM]},{className:"meta",begin:"\\[<",end:">\\]",relevance:10},{className:"symbol",begin:"\\B('[A-Za-z])\\b",contains:[hljs.BACKSLASH_ESCAPE]},hljs.C_LINE_COMMENT_MODE,hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),hljs.C_NUMBER_MODE]}});hljs.registerLanguage("julia",function(hljs){var KEYWORDS={keyword:"in isa where "+"baremodule begin break catch ccall const continue do else elseif end export false finally for function "+"global if import importall let local macro module quote return true try using while "+"type immutable abstract bitstype typealias ",literal:"true false "+"ARGS C_NULL DevNull ENDIAN_BOM ENV I Inf Inf16 Inf32 Inf64 InsertionSort JULIA_HOME LOAD_PATH MergeSort "+"NaN NaN16 NaN32 NaN64 PROGRAM_FILE QuickSort RoundDown RoundFromZero RoundNearest RoundNearestTiesAway "+"RoundNearestTiesUp RoundToZero RoundUp STDERR STDIN STDOUT VERSION catalan e|0 eu|0 eulergamma golden im "+"nothing pi γ π φ ",built_in:"ANY AbstractArray AbstractChannel AbstractFloat AbstractMatrix AbstractRNG AbstractSerializer AbstractSet "+"AbstractSparseArray AbstractSparseMatrix AbstractSparseVector AbstractString AbstractUnitRange AbstractVecOrMat "+"AbstractVector Any ArgumentError Array AssertionError Associative Base64DecodePipe Base64EncodePipe Bidiagonal "+"BigFloat BigInt BitArray BitMatrix BitVector Bool BoundsError BufferStream CachingPool CapturedException "+"CartesianIndex CartesianRange Cchar Cdouble Cfloat Channel Char Cint Cintmax_t Clong Clonglong ClusterManager "+"Cmd CodeInfo Colon Complex Complex128 Complex32 Complex64 CompositeException Condition ConjArray ConjMatrix "+"ConjVector Cptrdiff_t Cshort Csize_t Cssize_t Cstring Cuchar Cuint Cuintmax_t Culong Culonglong Cushort Cwchar_t "+"Cwstring DataType Date DateFormat DateTime DenseArray DenseMatrix DenseVecOrMat DenseVector Diagonal Dict "+"DimensionMismatch Dims DirectIndexString Display DivideError DomainError EOFError EachLine Enum Enumerate "+"ErrorException Exception ExponentialBackOff Expr Factorization FileMonitor Float16 Float32 Float64 Function "+"Future GlobalRef GotoNode HTML Hermitian IO IOBuffer IOContext IOStream IPAddr IPv4 IPv6 IndexCartesian IndexLinear "+"IndexStyle InexactError InitError Int Int128 Int16 Int32 Int64 Int8 IntSet Integer InterruptException "+"InvalidStateException Irrational KeyError LabelNode LinSpace LineNumberNode LoadError LowerTriangular MIME Matrix "+"MersenneTwister Method MethodError MethodTable Module NTuple NewvarNode NullException Nullable Number ObjectIdDict "+"OrdinalRange OutOfMemoryError OverflowError Pair ParseError PartialQuickSort PermutedDimsArray Pipe "+"PollingFileWatcher ProcessExitedException Ptr QuoteNode RandomDevice Range RangeIndex Rational RawFD "+"ReadOnlyMemoryError Real ReentrantLock Ref Regex RegexMatch RemoteChannel RemoteException RevString RoundingMode "+"RowVector SSAValue SegmentationFault SerializationState Set SharedArray SharedMatrix SharedVector Signed "+"SimpleVector Slot SlotNumber SparseMatrixCSC SparseVector StackFrame StackOverflowError StackTrace StepRange "+"StepRangeLen StridedArray StridedMatrix StridedVecOrMat StridedVector String SubArray SubString SymTridiagonal "+"Symbol Symmetric SystemError TCPSocket Task Text TextDisplay Timer Tridiagonal Tuple Type TypeError TypeMapEntry "+"TypeMapLevel TypeName TypeVar TypedSlot UDPSocket UInt UInt128 UInt16 UInt32 UInt64 UInt8 UndefRefError UndefVarError "+"UnicodeError UniformScaling Union UnionAll UnitRange Unsigned UpperTriangular Val Vararg VecElement VecOrMat Vector "+"VersionNumber Void WeakKeyDict WeakRef WorkerConfig WorkerPool "};var VARIABLE_NAME_RE="[A-Za-z_\\u00A1-\\uFFFF][A-Za-z_0-9\\u00A1-\\uFFFF]*";var DEFAULT={lexemes:VARIABLE_NAME_RE,keywords:KEYWORDS,illegal:/<\//};var NUMBER={className:"number",begin:/(\b0x[\d_]*(\.[\d_]*)?|0x\.\d[\d_]*)p[-+]?\d+|\b0[box][a-fA-F0-9][a-fA-F0-9_]*|(\b\d[\d_]*(\.[\d_]*)?|\.\d[\d_]*)([eEfF][-+]?\d+)?/,relevance:0};var CHAR={className:"string",begin:/'(.|\\[xXuU][a-zA-Z0-9]+)'/};var INTERPOLATION={className:"subst",begin:/\$\(/,end:/\)/,keywords:KEYWORDS};var INTERPOLATED_VARIABLE={className:"variable",begin:"\\$"+VARIABLE_NAME_RE};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,INTERPOLATION,INTERPOLATED_VARIABLE],variants:[{begin:/\w*"""/,end:/"""\w*/,relevance:10},{begin:/\w*"/,end:/"\w*/}]};var COMMAND={className:"string",contains:[hljs.BACKSLASH_ESCAPE,INTERPOLATION,INTERPOLATED_VARIABLE],begin:"`",end:"`"};var MACROCALL={className:"meta",begin:"@"+VARIABLE_NAME_RE};var COMMENT={className:"comment",variants:[{begin:"#=",end:"=#",relevance:10},{begin:"#",end:"$"}]};DEFAULT.contains=[NUMBER,CHAR,STRING,COMMAND,MACROCALL,COMMENT,hljs.HASH_COMMENT_MODE,{className:"keyword",begin:"\\b(((abstract|primitive)\\s+)type|(mutable\\s+)?struct)\\b"},{begin:/<:/}];INTERPOLATION.contains=DEFAULT.contains;return DEFAULT});hljs.registerLanguage("julia-repl",function(hljs){return{contains:[{className:"meta",begin:/^julia>/,relevance:10,starts:{end:/^(?![ ]{6})/,subLanguage:"julia"},aliases:["jldoctest"]}]}});hljs.registerLanguage("fortran",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var F_KEYWORDS={literal:".False. .True.",keyword:"kind do while private call intrinsic where elsewhere "+"type endtype endmodule endselect endinterface end enddo endif if forall endforall only contains default return stop then block endblock "+"public subroutine|10 function program .and. .or. .not. .le. .eq. .ge. .gt. .lt. "+"goto save else use module select case "+"access blank direct exist file fmt form formatted iostat name named nextrec number opened rec recl sequential status unformatted unit "+"continue format pause cycle exit "+"c_null_char c_alert c_backspace c_form_feed flush wait decimal round iomsg "+"synchronous nopass non_overridable pass protected volatile abstract extends import "+"non_intrinsic value deferred generic final enumerator class associate bind enum "+"c_int c_short c_long c_long_long c_signed_char c_size_t c_int8_t c_int16_t c_int32_t c_int64_t c_int_least8_t c_int_least16_t "+"c_int_least32_t c_int_least64_t c_int_fast8_t c_int_fast16_t c_int_fast32_t c_int_fast64_t c_intmax_t C_intptr_t c_float c_double "+"c_long_double c_float_complex c_double_complex c_long_double_complex c_bool c_char c_null_ptr c_null_funptr "+"c_new_line c_carriage_return c_horizontal_tab c_vertical_tab iso_c_binding c_loc c_funloc c_associated  c_f_pointer "+"c_ptr c_funptr iso_fortran_env character_storage_size error_unit file_storage_size input_unit iostat_end iostat_eor "+"numeric_storage_size output_unit c_f_procpointer ieee_arithmetic ieee_support_underflow_control "+"ieee_get_underflow_mode ieee_set_underflow_mode newunit contiguous recursive "+"pad position action delim readwrite eor advance nml interface procedure namelist include sequence elemental pure "+"integer real character complex logical dimension allocatable|10 parameter "+"external implicit|10 none double precision assign intent optional pointer "+"target in out common equivalence data",built_in:"alog alog10 amax0 amax1 amin0 amin1 amod cabs ccos cexp clog csin csqrt dabs dacos dasin datan datan2 dcos dcosh ddim dexp dint "+"dlog dlog10 dmax1 dmin1 dmod dnint dsign dsin dsinh dsqrt dtan dtanh float iabs idim idint idnint ifix isign max0 max1 min0 min1 sngl "+"algama cdabs cdcos cdexp cdlog cdsin cdsqrt cqabs cqcos cqexp cqlog cqsin cqsqrt dcmplx dconjg derf derfc dfloat dgamma dimag dlgama "+"iqint qabs qacos qasin qatan qatan2 qcmplx qconjg qcos qcosh qdim qerf qerfc qexp qgamma qimag qlgama qlog qlog10 qmax1 qmin1 qmod "+"qnint qsign qsin qsinh qsqrt qtan qtanh abs acos aimag aint anint asin atan atan2 char cmplx conjg cos cosh exp ichar index int log "+"log10 max min nint sign sin sinh sqrt tan tanh print write dim lge lgt lle llt mod nullify allocate deallocate "+"adjustl adjustr all allocated any associated bit_size btest ceiling count cshift date_and_time digits dot_product "+"eoshift epsilon exponent floor fraction huge iand ibclr ibits ibset ieor ior ishft ishftc lbound len_trim matmul "+"maxexponent maxloc maxval merge minexponent minloc minval modulo mvbits nearest pack present product "+"radix random_number random_seed range repeat reshape rrspacing scale scan selected_int_kind selected_real_kind "+"set_exponent shape size spacing spread sum system_clock tiny transpose trim ubound unpack verify achar iachar transfer "+"dble entry dprod cpu_time command_argument_count get_command get_command_argument get_environment_variable is_iostat_end "+"ieee_arithmetic ieee_support_underflow_control ieee_get_underflow_mode ieee_set_underflow_mode "+"is_iostat_eor move_alloc new_line selected_char_kind same_type_as extends_type_of"+"acosh asinh atanh bessel_j0 bessel_j1 bessel_jn bessel_y0 bessel_y1 bessel_yn erf erfc erfc_scaled gamma log_gamma hypot norm2 "+"atomic_define atomic_ref execute_command_line leadz trailz storage_size merge_bits "+"bge bgt ble blt dshiftl dshiftr findloc iall iany iparity image_index lcobound ucobound maskl maskr "+"num_images parity popcnt poppar shifta shiftl shiftr this_image"};return{case_insensitive:true,aliases:["f90","f95"],keywords:F_KEYWORDS,illegal:/\/\*/,contains:[hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"string",relevance:0}),{className:"function",beginKeywords:"subroutine function program",illegal:"[${=\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]},hljs.COMMENT("!","$",{relevance:0}),{className:"number",begin:"(?=\\b|\\+|\\-|\\.)(?=\\.\\d|\\d)(?:\\d+)?(?:\\.?\\d*)(?:[de][+-]?\\d+)?\\b\\.?",relevance:0}]}});hljs.registerLanguage("basic",function(hljs){return{case_insensitive:true,illegal:"^.",lexemes:"[a-zA-Z][a-zA-Z0-9_$%!#]*",keywords:{keyword:"ABS ASC AND ATN AUTO|0 BEEP BLOAD|10 BSAVE|10 CALL CALLS CDBL CHAIN CHDIR CHR$|10 CINT CIRCLE "+"CLEAR CLOSE CLS COLOR COM COMMON CONT COS CSNG CSRLIN CVD CVI CVS DATA DATE$ "+"DEFDBL DEFINT DEFSNG DEFSTR DEF|0 SEG USR DELETE DIM DRAW EDIT END ENVIRON ENVIRON$ "+"EOF EQV ERASE ERDEV ERDEV$ ERL ERR ERROR EXP FIELD FILES FIX FOR|0 FRE GET GOSUB|10 GOTO "+"HEX$ IF THEN ELSE|0 INKEY$ INP INPUT INPUT# INPUT$ INSTR IMP INT IOCTL IOCTL$ KEY ON "+"OFF LIST KILL LEFT$ LEN LET LINE LLIST LOAD LOC LOCATE LOF LOG LPRINT USING LSET "+"MERGE MID$ MKDIR MKD$ MKI$ MKS$ MOD NAME NEW NEXT NOISE NOT OCT$ ON OR PEN PLAY STRIG OPEN OPTION "+"BASE OUT PAINT PALETTE PCOPY PEEK PMAP POINT POKE POS PRINT PRINT] PSET PRESET "+"PUT RANDOMIZE READ REM RENUM RESET|0 RESTORE RESUME RETURN|0 RIGHT$ RMDIR RND RSET "+"RUN SAVE SCREEN SGN SHELL SIN SOUND SPACE$ SPC SQR STEP STICK STOP STR$ STRING$ SWAP "+"SYSTEM TAB TAN TIME$ TIMER TROFF TRON TO USR VAL VARPTR VARPTR$ VIEW WAIT WHILE "+"WEND WIDTH WINDOW WRITE XOR"},contains:[hljs.QUOTE_STRING_MODE,hljs.COMMENT("REM","$",{relevance:10}),hljs.COMMENT("'","$",{relevance:0}),{className:"symbol",begin:"^[0-9]+ ",relevance:10},{className:"number",begin:"\\b([0-9]+[0-9edED.]*[#!]?)",relevance:0},{className:"number",begin:"(&[hH][0-9a-fA-F]{1,4})"},{className:"number",begin:"(&[oO][0-7]{1,6})"}]}});hljs.registerLanguage("ada",function(hljs){var INTEGER_RE="\\d(_|\\d)*";var EXPONENT_RE="[eE][-+]?"+INTEGER_RE;var DECIMAL_LITERAL_RE=INTEGER_RE+"(\\."+INTEGER_RE+")?"+"("+EXPONENT_RE+")?";var BASED_INTEGER_RE="\\w+";var BASED_LITERAL_RE=INTEGER_RE+"#"+BASED_INTEGER_RE+"(\\."+BASED_INTEGER_RE+")?"+"#"+"("+EXPONENT_RE+")?";var NUMBER_RE="\\b("+BASED_LITERAL_RE+"|"+DECIMAL_LITERAL_RE+")";var ID_REGEX="[A-Za-z](_?[A-Za-z0-9.])*";var BAD_CHARS="[]{}%#'\"";var COMMENTS=hljs.COMMENT("--","$");var VAR_DECLS={begin:"\\s+:\\s+",end:"\\s*(:=|;|\\)|=>|$)",illegal:BAD_CHARS,contains:[{beginKeywords:"loop for declare others",endsParent:true},{className:"keyword",beginKeywords:"not null constant access function procedure in out aliased exception"},{className:"type",begin:ID_REGEX,endsParent:true,relevance:0}]};return{case_insensitive:true,keywords:{keyword:"abort else new return abs elsif not reverse abstract end "+"accept entry select access exception of separate aliased exit or some "+"all others subtype and for out synchronized array function overriding "+"at tagged generic package task begin goto pragma terminate "+"body private then if procedure type case in protected constant interface "+"is raise use declare range delay limited record when delta loop rem while "+"digits renames with do mod requeue xor",literal:"True False"},contains:[COMMENTS,{className:"string",begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{className:"string",begin:/'.'/},{className:"number",begin:NUMBER_RE,relevance:0},{className:"symbol",begin:"'"+ID_REGEX},{className:"title",begin:"(\\bwith\\s+)?(\\bprivate\\s+)?\\bpackage\\s+(\\bbody\\s+)?",end:"(is|$)",keywords:"package body",excludeBegin:true,excludeEnd:true,illegal:BAD_CHARS},{begin:"(\\b(with|overriding)\\s+)?\\b(function|procedure)\\s+",end:"(\\bis|\\bwith|\\brenames|\\)\\s*;)",keywords:"overriding function procedure with is renames return",returnBegin:true,contains:[COMMENTS,{className:"title",begin:"(\\bwith\\s+)?\\b(function|procedure)\\s+",end:"(\\(|\\s+|$)",excludeBegin:true,excludeEnd:true,illegal:BAD_CHARS},VAR_DECLS,{className:"type",begin:"\\breturn\\s+",end:"(\\s+|;|$)",keywords:"return",excludeBegin:true,excludeEnd:true,endsParent:true,illegal:BAD_CHARS}]},{className:"type",begin:"\\b(sub)?type\\s+",end:"\\s+",keywords:"type",excludeBegin:true,illegal:BAD_CHARS},VAR_DECLS]}});hljs.registerLanguage("puppet",function(hljs){var PUPPET_KEYWORDS={keyword:"and case default else elsif false if in import enherits node or true undef unless main settings $string ",literal:"alias audit before loglevel noop require subscribe tag "+"owner ensure group mode name|0 changes context force incl lens load_path onlyif provider returns root show_diff type_check "+"en_address ip_address realname command environment hour monute month monthday special target weekday "+"creates cwd ogoutput refresh refreshonly tries try_sleep umask backup checksum content ctime force ignore "+"links mtime purge recurse recurselimit replace selinux_ignore_defaults selrange selrole seltype seluser source "+"souirce_permissions sourceselect validate_cmd validate_replacement allowdupe attribute_membership auth_membership forcelocal gid "+"ia_load_module members system host_aliases ip allowed_trunk_vlans description device_url duplex encapsulation etherchannel "+"native_vlan speed principals allow_root auth_class auth_type authenticate_user k_of_n mechanisms rule session_owner shared options "+"device fstype enable hasrestart directory present absent link atboot blockdevice device dump pass remounts poller_tag use "+"message withpath adminfile allow_virtual allowcdrom category configfiles flavor install_options instance package_settings platform "+"responsefile status uninstall_options vendor unless_system_user unless_uid binary control flags hasstatus manifest pattern restart running "+"start stop allowdupe auths expiry gid groups home iterations key_membership keys managehome membership password password_max_age "+"password_min_age profile_membership profiles project purge_ssh_keys role_membership roles salt shell uid baseurl cost descr enabled "+"enablegroups exclude failovermethod gpgcheck gpgkey http_caching include includepkgs keepalive metadata_expire metalink mirrorlist "+"priority protect proxy proxy_password proxy_username repo_gpgcheck s3_enabled skip_if_unavailable sslcacert sslclientcert sslclientkey "+"sslverify mounted",built_in:"architecture augeasversion blockdevices boardmanufacturer boardproductname boardserialnumber cfkey dhcp_servers "+"domain ec2_ ec2_userdata facterversion filesystems ldom fqdn gid hardwareisa hardwaremodel hostname id|0 interfaces "+"ipaddress ipaddress_ ipaddress6 ipaddress6_ iphostnumber is_virtual kernel kernelmajversion kernelrelease kernelversion "+"kernelrelease kernelversion lsbdistcodename lsbdistdescription lsbdistid lsbdistrelease lsbmajdistrelease lsbminordistrelease "+"lsbrelease macaddress macaddress_ macosx_buildversion macosx_productname macosx_productversion macosx_productverson_major "+"macosx_productversion_minor manufacturer memoryfree memorysize netmask metmask_ network_ operatingsystem operatingsystemmajrelease "+"operatingsystemrelease osfamily partitions path physicalprocessorcount processor processorcount productname ps puppetversion "+"rubysitedir rubyversion selinux selinux_config_mode selinux_config_policy selinux_current_mode selinux_current_mode selinux_enforced "+"selinux_policyversion serialnumber sp_ sshdsakey sshecdsakey sshrsakey swapencrypted swapfree swapsize timezone type uniqueid uptime "+"uptime_days uptime_hours uptime_seconds uuid virtual vlans xendomains zfs_version zonenae zones zpool_version"};var COMMENT=hljs.COMMENT("#","$");var IDENT_RE="([A-Za-z_]|::)(\\w|::)*";var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE});var VARIABLE={className:"variable",begin:"\\$"+IDENT_RE};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,VARIABLE],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/}]};return{aliases:["pp"],contains:[COMMENT,VARIABLE,STRING,{beginKeywords:"class",end:"\\{|;",illegal:/=/,contains:[TITLE,COMMENT]},{beginKeywords:"define",end:/\{/,contains:[{className:"section",begin:hljs.IDENT_RE,endsParent:true}]},{begin:hljs.IDENT_RE+"\\s+\\{",returnBegin:true,end:/\S/,contains:[{className:"keyword",begin:hljs.IDENT_RE},{begin:/\{/,end:/\}/,keywords:PUPPET_KEYWORDS,relevance:0,contains:[STRING,COMMENT,{begin:"[a-zA-Z_]+\\s*=>",returnBegin:true,end:"=>",contains:[{className:"attr",begin:hljs.IDENT_RE}]},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},VARIABLE]}],relevance:0}]}});hljs.registerLanguage("vim",function(hljs){return{lexemes:/[!#@\w]+/,keywords:{keyword:"N|0 P|0 X|0 a|0 ab abc abo al am an|0 ar arga argd arge argdo argg argl argu as au aug aun b|0 bN ba bad bd be bel bf bl bm bn bo bp br brea breaka breakd breakl bro bufdo buffers bun bw c|0 cN cNf ca cabc caddb cad caddf cal cat cb cc ccl cd ce cex cf cfir cgetb cgete cg changes chd che checkt cl cla clo cm cmapc cme cn cnew cnf cno cnorea cnoreme co col colo com comc comp con conf cope "+"cp cpf cq cr cs cst cu cuna cunme cw delm deb debugg delc delf dif diffg diffo diffp diffpu diffs diffthis dig di dl dell dj dli do doautoa dp dr ds dsp e|0 ea ec echoe echoh echom echon el elsei em en endfo endf endt endw ene ex exe exi exu f|0 files filet fin fina fini fir fix fo foldc foldd folddoc foldo for fu go gr grepa gu gv ha helpf helpg helpt hi hid his ia iabc if ij il im imapc "+"ime ino inorea inoreme int is isp iu iuna iunme j|0 ju k|0 keepa kee keepj lN lNf l|0 lad laddb laddf la lan lat lb lc lch lcl lcs le lefta let lex lf lfir lgetb lgete lg lgr lgrepa lh ll lla lli lmak lm lmapc lne lnew lnf ln loadk lo loc lockv lol lope lp lpf lr ls lt lu lua luad luaf lv lvimgrepa lw m|0 ma mak map mapc marks mat me menut mes mk mks mksp mkv mkvie mod mz mzf nbc nb nbs new nm nmapc nme nn nnoreme noa no noh norea noreme norm nu nun nunme ol o|0 om omapc ome on ono onoreme opt ou ounme ow p|0 "+"profd prof pro promptr pc ped pe perld po popu pp pre prev ps pt ptN ptf ptj ptl ptn ptp ptr pts pu pw py3 python3 py3d py3f py pyd pyf quita qa rec red redi redr redraws reg res ret retu rew ri rightb rub rubyd rubyf rund ru rv sN san sa sal sav sb sbN sba sbf sbl sbm sbn sbp sbr scrip scripte scs se setf setg setl sf sfir sh sim sig sil sl sla sm smap smapc sme sn sni sno snor snoreme sor "+"so spelld spe spelli spellr spellu spellw sp spr sre st sta startg startr star stopi stj sts sun sunm sunme sus sv sw sy synti sync tN tabN tabc tabdo tabe tabf tabfir tabl tabm tabnew "+"tabn tabo tabp tabr tabs tab ta tags tc tcld tclf te tf th tj tl tm tn to tp tr try ts tu u|0 undoj undol una unh unl unlo unm unme uns up ve verb vert vim vimgrepa vi viu vie vm vmapc vme vne vn vnoreme vs vu vunme windo w|0 wN wa wh wi winc winp wn wp wq wqa ws wu wv x|0 xa xmapc xm xme xn xnoreme xu xunme y|0 z|0 ~ "+"Next Print append abbreviate abclear aboveleft all amenu anoremenu args argadd argdelete argedit argglobal arglocal argument ascii autocmd augroup aunmenu buffer bNext ball badd bdelete behave belowright bfirst blast bmodified bnext botright bprevious brewind break breakadd breakdel breaklist browse bunload "+"bwipeout change cNext cNfile cabbrev cabclear caddbuffer caddexpr caddfile call catch cbuffer cclose center cexpr cfile cfirst cgetbuffer cgetexpr cgetfile chdir checkpath checktime clist clast close cmap cmapclear cmenu cnext cnewer cnfile cnoremap cnoreabbrev cnoremenu copy colder colorscheme command comclear compiler continue confirm copen cprevious cpfile cquit crewind cscope cstag cunmap "+"cunabbrev cunmenu cwindow delete delmarks debug debuggreedy delcommand delfunction diffupdate diffget diffoff diffpatch diffput diffsplit digraphs display deletel djump dlist doautocmd doautoall deletep drop dsearch dsplit edit earlier echo echoerr echohl echomsg else elseif emenu endif endfor "+"endfunction endtry endwhile enew execute exit exusage file filetype find finally finish first fixdel fold foldclose folddoopen folddoclosed foldopen function global goto grep grepadd gui gvim hardcopy help helpfind helpgrep helptags highlight hide history insert iabbrev iabclear ijump ilist imap "+"imapclear imenu inoremap inoreabbrev inoremenu intro isearch isplit iunmap iunabbrev iunmenu join jumps keepalt keepmarks keepjumps lNext lNfile list laddexpr laddbuffer laddfile last language later lbuffer lcd lchdir lclose lcscope left leftabove lexpr lfile lfirst lgetbuffer lgetexpr lgetfile lgrep lgrepadd lhelpgrep llast llist lmake lmap lmapclear lnext lnewer lnfile lnoremap loadkeymap loadview "+"lockmarks lockvar lolder lopen lprevious lpfile lrewind ltag lunmap luado luafile lvimgrep lvimgrepadd lwindow move mark make mapclear match menu menutranslate messages mkexrc mksession mkspell mkvimrc mkview mode mzscheme mzfile nbclose nbkey nbsart next nmap nmapclear nmenu nnoremap "+"nnoremenu noautocmd noremap nohlsearch noreabbrev noremenu normal number nunmap nunmenu oldfiles open omap omapclear omenu only onoremap onoremenu options ounmap ounmenu ownsyntax print profdel profile promptfind promptrepl pclose pedit perl perldo pop popup ppop preserve previous psearch ptag ptNext "+"ptfirst ptjump ptlast ptnext ptprevious ptrewind ptselect put pwd py3do py3file python pydo pyfile quit quitall qall read recover redo redir redraw redrawstatus registers resize retab return rewind right rightbelow ruby rubydo rubyfile rundo runtime rviminfo substitute sNext sandbox sargument sall saveas sbuffer sbNext sball sbfirst sblast sbmodified sbnext sbprevious sbrewind scriptnames scriptencoding "+"scscope set setfiletype setglobal setlocal sfind sfirst shell simalt sign silent sleep slast smagic smapclear smenu snext sniff snomagic snoremap snoremenu sort source spelldump spellgood spellinfo spellrepall spellundo spellwrong split sprevious srewind stop stag startgreplace startreplace "+"startinsert stopinsert stjump stselect sunhide sunmap sunmenu suspend sview swapname syntax syntime syncbind tNext tabNext tabclose tabedit tabfind tabfirst tablast tabmove tabnext tabonly tabprevious tabrewind tag tcl tcldo tclfile tearoff tfirst throw tjump tlast tmenu tnext topleft tprevious "+"trewind tselect tunmenu undo undojoin undolist unabbreviate unhide unlet unlockvar unmap unmenu unsilent update vglobal version verbose vertical vimgrep vimgrepadd visual viusage view vmap vmapclear vmenu vnew "+"vnoremap vnoremenu vsplit vunmap vunmenu write wNext wall while winsize wincmd winpos wnext wprevious wqall wsverb wundo wviminfo xit xall xmapclear xmap xmenu xnoremap xnoremenu xunmap xunmenu yank",built_in:"synIDtrans atan2 range matcharg did_filetype asin feedkeys xor argv "+"complete_check add getwinposx getqflist getwinposy screencol "+"clearmatches empty extend getcmdpos mzeval garbagecollect setreg "+"ceil sqrt diff_hlID inputsecret get getfperm getpid filewritable "+"shiftwidth max sinh isdirectory synID system inputrestore winline "+"atan visualmode inputlist tabpagewinnr round getregtype mapcheck "+"hasmapto histdel argidx findfile sha256 exists toupper getcmdline "+"taglist string getmatches bufnr strftime winwidth bufexists "+"strtrans tabpagebuflist setcmdpos remote_read printf setloclist "+"getpos getline bufwinnr float2nr len getcmdtype diff_filler luaeval "+"resolve libcallnr foldclosedend reverse filter has_key bufname "+"str2float strlen setline getcharmod setbufvar index searchpos "+"shellescape undofile foldclosed setqflist buflisted strchars str2nr "+"virtcol floor remove undotree remote_expr winheight gettabwinvar "+"reltime cursor tabpagenr finddir localtime acos getloclist search "+"tanh matchend rename gettabvar strdisplaywidth type abs py3eval "+"setwinvar tolower wildmenumode log10 spellsuggest bufloaded "+"synconcealed nextnonblank server2client complete settabwinvar "+"executable input wincol setmatches getftype hlID inputsave "+"searchpair or screenrow line settabvar histadd deepcopy strpart "+"remote_peek and eval getftime submatch screenchar winsaveview "+"matchadd mkdir screenattr getfontname libcall reltimestr getfsize "+"winnr invert pow getbufline byte2line soundfold repeat fnameescape "+"tagfiles sin strwidth spellbadword trunc maparg log lispindent "+"hostname setpos globpath remote_foreground getchar synIDattr "+"fnamemodify cscope_connection stridx winbufnr indent min "+"complete_add nr2char searchpairpos inputdialog values matchlist "+"items hlexists strridx browsedir expand fmod pathshorten line2byte "+"argc count getwinvar glob foldtextresult getreg foreground cosh "+"matchdelete has char2nr simplify histget searchdecl iconv "+"winrestcmd pumvisible writefile foldlevel haslocaldir keys cos "+"matchstr foldtext histnr tan tempname getcwd byteidx getbufvar "+"islocked escape eventhandler remote_send serverlist winrestview "+"synstack pyeval prevnonblank readfile cindent filereadable changenr "+"exp"},illegal:/;/,contains:[hljs.NUMBER_MODE,{className:"string",begin:"'",end:"'",illegal:"\\n"},{className:"string",begin:/"(\\"|\n\\|[^"\n])*"/},hljs.COMMENT('"',"$"),{className:"variable",begin:/[bwtglsav]:[\w\d_]*/},{className:"function",beginKeywords:"function function!",end:"$",relevance:0,contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)"}]},{className:"symbol",begin:/<[\w-]+>/}]}});hljs.registerLanguage("dos",function(hljs){var COMMENT=hljs.COMMENT(/^\s*@?rem\b/,/$/,{relevance:10});var LABEL={className:"symbol",begin:"^\\s*[A-Za-z._?][A-Za-z0-9_$#@~.?]*(:|\\s+label)",relevance:0};return{aliases:["bat","cmd"],case_insensitive:true,illegal:/\/\*/,keywords:{keyword:"if else goto for in do call exit not exist errorlevel defined "+"equ neq lss leq gtr geq",built_in:"prn nul lpt3 lpt2 lpt1 con com4 com3 com2 com1 aux "+"shift cd dir echo setlocal endlocal set pause copy "+"append assoc at attrib break cacls cd chcp chdir chkdsk chkntfs cls cmd color "+"comp compact convert date dir diskcomp diskcopy doskey erase fs "+"find findstr format ftype graftabl help keyb label md mkdir mode more move path "+"pause print popd pushd promt rd recover rem rename replace restore rmdir shift"+"sort start subst time title tree type ver verify vol "+"ping net ipconfig taskkill xcopy ren del"},contains:[{className:"variable",begin:/%%[^ ]|%[^ ]+?%|![^ ]+?!/},{className:"function",begin:LABEL.begin,end:"goto:eof",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),COMMENT]},{className:"number",begin:"\\b\\d+",relevance:0},COMMENT]}});hljs.registerLanguage("ceylon",function(hljs){var KEYWORDS="assembly module package import alias class interface object given value "+"assign void function new of extends satisfies abstracts in out return "+"break continue throw assert dynamic if else switch case for while try "+"catch finally then let this outer super is exists nonempty";var DECLARATION_MODIFIERS="shared abstract formal default actual variable late native deprecated"+"final sealed annotation suppressWarnings small";var DOCUMENTATION="doc by license see throws tagged";var SUBST={className:"subst",excludeBegin:true,excludeEnd:true,begin:/``/,end:/``/,keywords:KEYWORDS,relevance:10};var EXPRESSIONS=[{className:"string",begin:'"""',end:'"""',relevance:10},{className:"string",begin:'"',end:'"',contains:[SUBST]},{className:"string",begin:"'",end:"'"},{className:"number",begin:"#[0-9a-fA-F_]+|\\$[01_]+|[0-9_]+(?:\\.[0-9_](?:[eE][+-]?\\d+)?)?[kMGTPmunpf]?",relevance:0}];SUBST.contains=EXPRESSIONS;return{keywords:{keyword:KEYWORDS+" "+DECLARATION_MODIFIERS,meta:DOCUMENTATION},illegal:"\\$[^01]|#[^0-9a-fA-F]",contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:["self"]}),{className:"meta",begin:'@[a-z]\\w*(?:\\:"[^"]*")?'}].concat(EXPRESSIONS)}});hljs.registerLanguage("nsis",function(hljs){var CONSTANTS={className:"variable",begin:/\$(ADMINTOOLS|APPDATA|CDBURN_AREA|CMDLINE|COMMONFILES32|COMMONFILES64|COMMONFILES|COOKIES|DESKTOP|DOCUMENTS|EXEDIR|EXEFILE|EXEPATH|FAVORITES|FONTS|HISTORY|HWNDPARENT|INSTDIR|INTERNET_CACHE|LANGUAGE|LOCALAPPDATA|MUSIC|NETHOOD|OUTDIR|PICTURES|PLUGINSDIR|PRINTHOOD|PROFILE|PROGRAMFILES32|PROGRAMFILES64|PROGRAMFILES|QUICKLAUNCH|RECENT|RESOURCES_LOCALIZED|RESOURCES|SENDTO|SMPROGRAMS|SMSTARTUP|STARTMENU|SYSDIR|TEMP|TEMPLATES|VIDEOS|WINDIR)/};var DEFINES={className:"variable",begin:/\$+{[\w\.:-]+}/};var VARIABLES={className:"variable",begin:/\$+\w+/,illegal:/\(\){}/};var LANGUAGES={className:"variable",begin:/\$+\([\w\^\.:-]+\)/};var PARAMETERS={className:"params",begin:"(ARCHIVE|FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_NORMAL|FILE_ATTRIBUTE_OFFLINE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_TEMPORARY|HKCR|HKCU|HKDD|HKEY_CLASSES_ROOT|HKEY_CURRENT_CONFIG|HKEY_CURRENT_USER|HKEY_DYN_DATA|HKEY_LOCAL_MACHINE|HKEY_PERFORMANCE_DATA|HKEY_USERS|HKLM|HKPD|HKU|IDABORT|IDCANCEL|IDIGNORE|IDNO|IDOK|IDRETRY|IDYES|MB_ABORTRETRYIGNORE|MB_DEFBUTTON1|MB_DEFBUTTON2|MB_DEFBUTTON3|MB_DEFBUTTON4|MB_ICONEXCLAMATION|MB_ICONINFORMATION|MB_ICONQUESTION|MB_ICONSTOP|MB_OK|MB_OKCANCEL|MB_RETRYCANCEL|MB_RIGHT|MB_RTLREADING|MB_SETFOREGROUND|MB_TOPMOST|MB_USERICON|MB_YESNO|NORMAL|OFFLINE|READONLY|SHCTX|SHELL_CONTEXT|SYSTEM|TEMPORARY)"};var COMPILER={className:"keyword",begin:/\!(addincludedir|addplugindir|appendfile|cd|define|delfile|echo|else|endif|error|execute|finalize|getdllversion|gettlbversion|if|ifdef|ifmacrodef|ifmacrondef|ifndef|include|insertmacro|macro|macroend|makensis|packhdr|searchparse|searchreplace|system|tempfile|undef|verbose|warning)/};var METACHARS={className:"meta",begin:/\$(\\[nrt]|\$)/};var PLUGINS={className:"class",begin:/\w+\:\:\w+/};var STRING={className:"string",variants:[{begin:'"',end:'"'},{begin:"'",end:"'"},{begin:"`",end:"`"}],illegal:/\n/,contains:[METACHARS,CONSTANTS,DEFINES,VARIABLES,LANGUAGES]};return{case_insensitive:false,keywords:{keyword:"Abort AddBrandingImage AddSize AllowRootDirInstall AllowSkipFiles AutoCloseWindow BGFont BGGradient BrandingText BringToFront Call CallInstDLL Caption ChangeUI CheckBitmap ClearErrors CompletedText ComponentText CopyFiles CRCCheck CreateDirectory CreateFont CreateShortCut Delete DeleteINISec DeleteINIStr DeleteRegKey DeleteRegValue DetailPrint DetailsButtonText DirText DirVar DirVerify EnableWindow EnumRegKey EnumRegValue Exch Exec ExecShell ExecShellWait ExecWait ExpandEnvStrings File FileBufSize FileClose FileErrorText FileOpen FileRead FileReadByte FileReadUTF16LE FileReadWord FileSeek FileWrite FileWriteByte FileWriteUTF16LE FileWriteWord FindClose FindFirst FindNext FindWindow FlushINI FunctionEnd GetCurInstType GetCurrentAddress GetDlgItem GetDLLVersion GetDLLVersionLocal GetErrorLevel GetFileTime GetFileTimeLocal GetFullPathName GetFunctionAddress GetInstDirError GetLabelAddress GetTempFileName Goto HideWindow Icon IfAbort IfErrors IfFileExists IfRebootFlag IfSilent InitPluginsDir InstallButtonText InstallColors InstallDir InstallDirRegKey InstProgressFlags InstType InstTypeGetText InstTypeSetText Int64Cmp Int64CmpU Int64Fmt IntCmp IntCmpU IntFmt IntOp IntPtrCmp IntPtrCmpU IntPtrOp IsWindow LangString LicenseBkColor LicenseData LicenseForceSelection LicenseLangString LicenseText LoadLanguageFile LockWindow LogSet LogText ManifestDPIAware ManifestSupportedOS MessageBox MiscButtonText Name Nop OutFile Page PageCallbacks PageExEnd Pop Push Quit ReadEnvStr ReadINIStr ReadRegDWORD ReadRegStr Reboot RegDLL Rename RequestExecutionLevel ReserveFile Return RMDir SearchPath SectionEnd SectionGetFlags SectionGetInstTypes SectionGetSize SectionGetText SectionGroupEnd SectionIn SectionSetFlags SectionSetInstTypes SectionSetSize SectionSetText SendMessage SetAutoClose SetBrandingImage SetCompress SetCompressor SetCompressorDictSize SetCtlColors SetCurInstType SetDatablockOptimize SetDateSave SetDetailsPrint SetDetailsView SetErrorLevel SetErrors SetFileAttributes SetFont SetOutPath SetOverwrite SetRebootFlag SetRegView SetShellVarContext SetSilent ShowInstDetails ShowUninstDetails ShowWindow SilentInstall SilentUnInstall Sleep SpaceTexts StrCmp StrCmpS StrCpy StrLen SubCaption Unicode UninstallButtonText UninstallCaption UninstallIcon UninstallSubCaption UninstallText UninstPage UnRegDLL Var VIAddVersionKey VIFileVersion VIProductVersion WindowIcon WriteINIStr WriteRegBin WriteRegDWORD WriteRegExpandStr WriteRegMultiStr WriteRegNone WriteRegStr WriteUninstaller XPStyle",literal:"admin all auto both bottom bzip2 colored components current custom directory false force hide highest ifdiff ifnewer instfiles lastused leave left license listonly lzma nevershow none normal notset off on open print right show silent silentlog smooth textonly top true try un.components un.custom un.directory un.instfiles un.license uninstConfirm user Win10 Win7 Win8 WinVista zlib"},contains:[hljs.HASH_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(";","$",{relevance:0}),{className:"function",beginKeywords:"Function PageEx Section SectionGroup",end:"$"},STRING,COMPILER,DEFINES,VARIABLES,LANGUAGES,PARAMETERS,PLUGINS,hljs.NUMBER_MODE]}});hljs.registerLanguage("go",function(hljs){var GO_KEYWORDS={keyword:"break default func interface select case map struct chan else goto package switch "+"const fallthrough if range type continue for import return var go defer "+"bool byte complex64 complex128 float32 float64 int8 int16 int32 int64 string uint8 "+"uint16 uint32 uint64 int uint uintptr rune",literal:"true false iota nil",built_in:"append cap close complex copy imag len make new panic print println real recover delete"};return{aliases:["golang"],keywords:GO_KEYWORDS,illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",variants:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{begin:"`",end:"`"}]},{className:"number",variants:[{begin:hljs.C_NUMBER_RE+"[i]",relevance:1},hljs.C_NUMBER_MODE]},{begin:/:=/},{className:"function",beginKeywords:"func",end:"\\s*(\\{|$)",excludeEnd:true,contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:GO_KEYWORDS,illegal:/["']/}]}]}});hljs.registerLanguage("nix",function(hljs){var NIX_KEYWORDS={keyword:"rec with let in inherit assert if else then",literal:"true false or and null",built_in:"import abort baseNameOf dirOf isNull builtins map removeAttrs throw "+"toString derivation"};var ANTIQUOTE={className:"subst",begin:/\$\{/,end:/}/,keywords:NIX_KEYWORDS};var ATTRS={begin:/[a-zA-Z0-9-_]+(\s*=)/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/\S+/}]};var STRING={className:"string",contains:[ANTIQUOTE],variants:[{begin:"''",end:"''"},{begin:'"',end:'"'}]};var EXPRESSIONS=[hljs.NUMBER_MODE,hljs.HASH_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING,ATTRS];ANTIQUOTE.contains=EXPRESSIONS;return{aliases:["nixos"],keywords:NIX_KEYWORDS,contains:EXPRESSIONS}});hljs.registerLanguage("mel",function(hljs){return{keywords:"int float string vector matrix if else switch case default while do for in break "+"continue global proc return about abs addAttr addAttributeEditorNodeHelp addDynamic "+"addNewShelfTab addPP addPanelCategory addPrefixToName advanceToNextDrivenKey "+"affectedNet affects aimConstraint air alias aliasAttr align alignCtx alignCurve "+"alignSurface allViewFit ambientLight angle angleBetween animCone animCurveEditor "+"animDisplay animView annotate appendStringArray applicationName applyAttrPreset "+"applyTake arcLenDimContext arcLengthDimension arclen arrayMapper art3dPaintCtx "+"artAttrCtx artAttrPaintVertexCtx artAttrSkinPaintCtx artAttrTool artBuildPaintMenu "+"artFluidAttrCtx artPuttyCtx artSelectCtx artSetPaintCtx artUserPaintCtx assignCommand "+"assignInputDevice assignViewportFactories attachCurve attachDeviceAttr attachSurface "+"attrColorSliderGrp attrCompatibility attrControlGrp attrEnumOptionMenu "+"attrEnumOptionMenuGrp attrFieldGrp attrFieldSliderGrp attrNavigationControlGrp "+"attrPresetEditWin attributeExists attributeInfo attributeMenu attributeQuery "+"autoKeyframe autoPlace bakeClip bakeFluidShading bakePartialHistory bakeResults "+"bakeSimulation basename basenameEx batchRender bessel bevel bevelPlus binMembership "+"bindSkin blend2 blendShape blendShapeEditor blendShapePanel blendTwoAttr blindDataType "+"boneLattice boundary boxDollyCtx boxZoomCtx bufferCurve buildBookmarkMenu "+"buildKeyframeMenu button buttonManip CBG cacheFile cacheFileCombine cacheFileMerge "+"cacheFileTrack camera cameraView canCreateManip canvas capitalizeString catch "+"catchQuiet ceil changeSubdivComponentDisplayLevel changeSubdivRegion channelBox "+"character characterMap characterOutlineEditor characterize chdir checkBox checkBoxGrp "+"checkDefaultRenderGlobals choice circle circularFillet clamp clear clearCache clip "+"clipEditor clipEditorCurrentTimeCtx clipSchedule clipSchedulerOutliner clipTrimBefore "+"closeCurve closeSurface cluster cmdFileOutput cmdScrollFieldExecuter "+"cmdScrollFieldReporter cmdShell coarsenSubdivSelectionList collision color "+"colorAtPoint colorEditor colorIndex colorIndexSliderGrp colorSliderButtonGrp "+"colorSliderGrp columnLayout commandEcho commandLine commandPort compactHairSystem "+"componentEditor compositingInterop computePolysetVolume condition cone confirmDialog "+"connectAttr connectControl connectDynamic connectJoint connectionInfo constrain "+"constrainValue constructionHistory container containsMultibyte contextInfo control "+"convertFromOldLayers convertIffToPsd convertLightmap convertSolidTx convertTessellation "+"convertUnit copyArray copyFlexor copyKey copySkinWeights cos cpButton cpCache "+"cpClothSet cpCollision cpConstraint cpConvClothToMesh cpForces cpGetSolverAttr cpPanel "+"cpProperty cpRigidCollisionFilter cpSeam cpSetEdit cpSetSolverAttr cpSolver "+"cpSolverTypes cpTool cpUpdateClothUVs createDisplayLayer createDrawCtx createEditor "+"createLayeredPsdFile createMotionField createNewShelf createNode createRenderLayer "+"createSubdivRegion cross crossProduct ctxAbort ctxCompletion ctxEditMode ctxTraverse "+"currentCtx currentTime currentTimeCtx currentUnit curve curveAddPtCtx "+"curveCVCtx curveEPCtx curveEditorCtx curveIntersect curveMoveEPCtx curveOnSurface "+"curveSketchCtx cutKey cycleCheck cylinder dagPose date defaultLightListCheckBox "+"defaultNavigation defineDataServer defineVirtualDevice deformer deg_to_rad delete "+"deleteAttr deleteShadingGroupsAndMaterials deleteShelfTab deleteUI deleteUnusedBrushes "+"delrandstr detachCurve detachDeviceAttr detachSurface deviceEditor devicePanel dgInfo "+"dgdirty dgeval dgtimer dimWhen directKeyCtx directionalLight dirmap dirname disable "+"disconnectAttr disconnectJoint diskCache displacementToPoly displayAffected "+"displayColor displayCull displayLevelOfDetail displayPref displayRGBColor "+"displaySmoothness displayStats displayString displaySurface distanceDimContext "+"distanceDimension doBlur dolly dollyCtx dopeSheetEditor dot dotProduct "+"doubleProfileBirailSurface drag dragAttrContext draggerContext dropoffLocator "+"duplicate duplicateCurve duplicateSurface dynCache dynControl dynExport dynExpression "+"dynGlobals dynPaintEditor dynParticleCtx dynPref dynRelEdPanel dynRelEditor "+"dynamicLoad editAttrLimits editDisplayLayerGlobals editDisplayLayerMembers "+"editRenderLayerAdjustment editRenderLayerGlobals editRenderLayerMembers editor "+"editorTemplate effector emit emitter enableDevice encodeString endString endsWith env "+"equivalent equivalentTol erf error eval evalDeferred evalEcho event "+"exactWorldBoundingBox exclusiveLightCheckBox exec executeForEachObject exists exp "+"expression expressionEditorListen extendCurve extendSurface extrude fcheck fclose feof "+"fflush fgetline fgetword file fileBrowserDialog fileDialog fileExtension fileInfo "+"filetest filletCurve filter filterCurve filterExpand filterStudioImport "+"findAllIntersections findAnimCurves findKeyframe findMenuItem findRelatedSkinCluster "+"finder firstParentOf fitBspline flexor floatEq floatField floatFieldGrp floatScrollBar "+"floatSlider floatSlider2 floatSliderButtonGrp floatSliderGrp floor flow fluidCacheInfo "+"fluidEmitter fluidVoxelInfo flushUndo fmod fontDialog fopen formLayout format fprint "+"frameLayout fread freeFormFillet frewind fromNativePath fwrite gamma gauss "+"geometryConstraint getApplicationVersionAsFloat getAttr getClassification "+"getDefaultBrush getFileList getFluidAttr getInputDeviceRange getMayaPanelTypes "+"getModifiers getPanel getParticleAttr getPluginResource getenv getpid glRender "+"glRenderEditor globalStitch gmatch goal gotoBindPose grabColor gradientControl "+"gradientControlNoAttr graphDollyCtx graphSelectContext graphTrackCtx gravity grid "+"gridLayout group groupObjectsByName HfAddAttractorToAS HfAssignAS HfBuildEqualMap "+"HfBuildFurFiles HfBuildFurImages HfCancelAFR HfConnectASToHF HfCreateAttractor "+"HfDeleteAS HfEditAS HfPerformCreateAS HfRemoveAttractorFromAS HfSelectAttached "+"HfSelectAttractors HfUnAssignAS hardenPointCurve hardware hardwareRenderPanel "+"headsUpDisplay headsUpMessage help helpLine hermite hide hilite hitTest hotBox hotkey "+"hotkeyCheck hsv_to_rgb hudButton hudSlider hudSliderButton hwReflectionMap hwRender "+"hwRenderLoad hyperGraph hyperPanel hyperShade hypot iconTextButton iconTextCheckBox "+"iconTextRadioButton iconTextRadioCollection iconTextScrollList iconTextStaticLabel "+"ikHandle ikHandleCtx ikHandleDisplayScale ikSolver ikSplineHandleCtx ikSystem "+"ikSystemInfo ikfkDisplayMethod illustratorCurves image imfPlugins inheritTransform "+"insertJoint insertJointCtx insertKeyCtx insertKnotCurve insertKnotSurface instance "+"instanceable instancer intField intFieldGrp intScrollBar intSlider intSliderGrp "+"interToUI internalVar intersect iprEngine isAnimCurve isConnected isDirty isParentOf "+"isSameObject isTrue isValidObjectName isValidString isValidUiName isolateSelect "+"itemFilter itemFilterAttr itemFilterRender itemFilterType joint jointCluster jointCtx "+"jointDisplayScale jointLattice keyTangent keyframe keyframeOutliner "+"keyframeRegionCurrentTimeCtx keyframeRegionDirectKeyCtx keyframeRegionDollyCtx "+"keyframeRegionInsertKeyCtx keyframeRegionMoveKeyCtx keyframeRegionScaleKeyCtx "+"keyframeRegionSelectKeyCtx keyframeRegionSetKeyCtx keyframeRegionTrackCtx "+"keyframeStats lassoContext lattice latticeDeformKeyCtx launch launchImageEditor "+"layerButton layeredShaderPort layeredTexturePort layout layoutDialog lightList "+"lightListEditor lightListPanel lightlink lineIntersection linearPrecision linstep "+"listAnimatable listAttr listCameras listConnections listDeviceAttachments listHistory "+"listInputDeviceAxes listInputDeviceButtons listInputDevices listMenuAnnotation "+"listNodeTypes listPanelCategories listRelatives listSets listTransforms "+"listUnselected listerEditor loadFluid loadNewShelf loadPlugin "+"loadPluginLanguageResources loadPrefObjects localizedPanelLabel lockNode loft log "+"longNameOf lookThru ls lsThroughFilter lsType lsUI Mayatomr mag makeIdentity makeLive "+"makePaintable makeRoll makeSingleSurface makeTubeOn makebot manipMoveContext "+"manipMoveLimitsCtx manipOptions manipRotateContext manipRotateLimitsCtx "+"manipScaleContext manipScaleLimitsCtx marker match max memory menu menuBarLayout "+"menuEditor menuItem menuItemToShelf menuSet menuSetPref messageLine min minimizeApp "+"mirrorJoint modelCurrentTimeCtx modelEditor modelPanel mouse movIn movOut move "+"moveIKtoFK moveKeyCtx moveVertexAlongDirection multiProfileBirailSurface mute "+"nParticle nameCommand nameField namespace namespaceInfo newPanelItems newton nodeCast "+"nodeIconButton nodeOutliner nodePreset nodeType noise nonLinear normalConstraint "+"normalize nurbsBoolean nurbsCopyUVSet nurbsCube nurbsEditUV nurbsPlane nurbsSelect "+"nurbsSquare nurbsToPoly nurbsToPolygonsPref nurbsToSubdiv nurbsToSubdivPref "+"nurbsUVSet nurbsViewDirectionVector objExists objectCenter objectLayer objectType "+"objectTypeUI obsoleteProc oceanNurbsPreviewPlane offsetCurve offsetCurveOnSurface "+"offsetSurface openGLExtension openMayaPref optionMenu optionMenuGrp optionVar orbit "+"orbitCtx orientConstraint outlinerEditor outlinerPanel overrideModifier "+"paintEffectsDisplay pairBlend palettePort paneLayout panel panelConfiguration "+"panelHistory paramDimContext paramDimension paramLocator parent parentConstraint "+"particle particleExists particleInstancer particleRenderInfo partition pasteKey "+"pathAnimation pause pclose percent performanceOptions pfxstrokes pickWalk picture "+"pixelMove planarSrf plane play playbackOptions playblast plugAttr plugNode pluginInfo "+"pluginResourceUtil pointConstraint pointCurveConstraint pointLight pointMatrixMult "+"pointOnCurve pointOnSurface pointPosition poleVectorConstraint polyAppend "+"polyAppendFacetCtx polyAppendVertex polyAutoProjection polyAverageNormal "+"polyAverageVertex polyBevel polyBlendColor polyBlindData polyBoolOp polyBridgeEdge "+"polyCacheMonitor polyCheck polyChipOff polyClipboard polyCloseBorder polyCollapseEdge "+"polyCollapseFacet polyColorBlindData polyColorDel polyColorPerVertex polyColorSet "+"polyCompare polyCone polyCopyUV polyCrease polyCreaseCtx polyCreateFacet "+"polyCreateFacetCtx polyCube polyCut polyCutCtx polyCylinder polyCylindricalProjection "+"polyDelEdge polyDelFacet polyDelVertex polyDuplicateAndConnect polyDuplicateEdge "+"polyEditUV polyEditUVShell polyEvaluate polyExtrudeEdge polyExtrudeFacet "+"polyExtrudeVertex polyFlipEdge polyFlipUV polyForceUV polyGeoSampler polyHelix "+"polyInfo polyInstallAction polyLayoutUV polyListComponentConversion polyMapCut "+"polyMapDel polyMapSew polyMapSewMove polyMergeEdge polyMergeEdgeCtx polyMergeFacet "+"polyMergeFacetCtx polyMergeUV polyMergeVertex polyMirrorFace polyMoveEdge "+"polyMoveFacet polyMoveFacetUV polyMoveUV polyMoveVertex polyNormal polyNormalPerVertex "+"polyNormalizeUV polyOptUvs polyOptions polyOutput polyPipe polyPlanarProjection "+"polyPlane polyPlatonicSolid polyPoke polyPrimitive polyPrism polyProjection "+"polyPyramid polyQuad polyQueryBlindData polyReduce polySelect polySelectConstraint "+"polySelectConstraintMonitor polySelectCtx polySelectEditCtx polySeparate "+"polySetToFaceNormal polySewEdge polyShortestPathCtx polySmooth polySoftEdge "+"polySphere polySphericalProjection polySplit polySplitCtx polySplitEdge polySplitRing "+"polySplitVertex polyStraightenUVBorder polySubdivideEdge polySubdivideFacet "+"polyToSubdiv polyTorus polyTransfer polyTriangulate polyUVSet polyUnite polyWedgeFace "+"popen popupMenu pose pow preloadRefEd print progressBar progressWindow projFileViewer "+"projectCurve projectTangent projectionContext projectionManip promptDialog propModCtx "+"propMove psdChannelOutliner psdEditTextureFile psdExport psdTextureFile putenv pwd "+"python querySubdiv quit rad_to_deg radial radioButton radioButtonGrp radioCollection "+"radioMenuItemCollection rampColorPort rand randomizeFollicles randstate rangeControl "+"readTake rebuildCurve rebuildSurface recordAttr recordDevice redo reference "+"referenceEdit referenceQuery refineSubdivSelectionList refresh refreshAE "+"registerPluginResource rehash reloadImage removeJoint removeMultiInstance "+"removePanelCategory rename renameAttr renameSelectionList renameUI render "+"renderGlobalsNode renderInfo renderLayerButton renderLayerParent "+"renderLayerPostProcess renderLayerUnparent renderManip renderPartition "+"renderQualityNode renderSettings renderThumbnailUpdate renderWindowEditor "+"renderWindowSelectContext renderer reorder reorderDeformers requires reroot "+"resampleFluid resetAE resetPfxToPolyCamera resetTool resolutionNode retarget "+"reverseCurve reverseSurface revolve rgb_to_hsv rigidBody rigidSolver roll rollCtx "+"rootOf rot rotate rotationInterpolation roundConstantRadius rowColumnLayout rowLayout "+"runTimeCommand runup sampleImage saveAllShelves saveAttrPreset saveFluid saveImage "+"saveInitialState saveMenu savePrefObjects savePrefs saveShelf saveToolSettings scale "+"scaleBrushBrightness scaleComponents scaleConstraint scaleKey scaleKeyCtx sceneEditor "+"sceneUIReplacement scmh scriptCtx scriptEditorInfo scriptJob scriptNode scriptTable "+"scriptToShelf scriptedPanel scriptedPanelType scrollField scrollLayout sculpt "+"searchPathArray seed selLoadSettings select selectContext selectCurveCV selectKey "+"selectKeyCtx selectKeyframeRegionCtx selectMode selectPref selectPriority selectType "+"selectedNodes selectionConnection separator setAttr setAttrEnumResource "+"setAttrMapping setAttrNiceNameResource setConstraintRestPosition "+"setDefaultShadingGroup setDrivenKeyframe setDynamic setEditCtx setEditor setFluidAttr "+"setFocus setInfinity setInputDeviceMapping setKeyCtx setKeyPath setKeyframe "+"setKeyframeBlendshapeTargetWts setMenuMode setNodeNiceNameResource setNodeTypeFlag "+"setParent setParticleAttr setPfxToPolyCamera setPluginResource setProject "+"setStampDensity setStartupMessage setState setToolTo setUITemplate setXformManip sets "+"shadingConnection shadingGeometryRelCtx shadingLightRelCtx shadingNetworkCompare "+"shadingNode shapeCompare shelfButton shelfLayout shelfTabLayout shellField "+"shortNameOf showHelp showHidden showManipCtx showSelectionInTitle "+"showShadingGroupAttrEditor showWindow sign simplify sin singleProfileBirailSurface "+"size sizeBytes skinCluster skinPercent smoothCurve smoothTangentSurface smoothstep "+"snap2to2 snapKey snapMode snapTogetherCtx snapshot soft softMod softModCtx sort sound "+"soundControl source spaceLocator sphere sphrand spotLight spotLightPreviewPort "+"spreadSheetEditor spring sqrt squareSurface srtContext stackTrace startString "+"startsWith stitchAndExplodeShell stitchSurface stitchSurfacePoints strcmp "+"stringArrayCatenate stringArrayContains stringArrayCount stringArrayInsertAtIndex "+"stringArrayIntersector stringArrayRemove stringArrayRemoveAtIndex "+"stringArrayRemoveDuplicates stringArrayRemoveExact stringArrayToString "+"stringToStringArray strip stripPrefixFromName stroke subdAutoProjection "+"subdCleanTopology subdCollapse subdDuplicateAndConnect subdEditUV "+"subdListComponentConversion subdMapCut subdMapSewMove subdMatchTopology subdMirror "+"subdToBlind subdToPoly subdTransferUVsToCache subdiv subdivCrease "+"subdivDisplaySmoothness substitute substituteAllString substituteGeometry substring "+"surface surfaceSampler surfaceShaderList swatchDisplayPort switchTable symbolButton "+"symbolCheckBox sysFile system tabLayout tan tangentConstraint texLatticeDeformContext "+"texManipContext texMoveContext texMoveUVShellContext texRotateContext texScaleContext "+"texSelectContext texSelectShortestPathCtx texSmudgeUVContext texWinToolCtx text "+"textCurves textField textFieldButtonGrp textFieldGrp textManip textScrollList "+"textToShelf textureDisplacePlane textureHairColor texturePlacementContext "+"textureWindow threadCount threePointArcCtx timeControl timePort timerX toNativePath "+"toggle toggleAxis toggleWindowVisibility tokenize tokenizeList tolerance tolower "+"toolButton toolCollection toolDropped toolHasOptions toolPropertyWindow torus toupper "+"trace track trackCtx transferAttributes transformCompare transformLimits translator "+"trim trunc truncateFluidCache truncateHairCache tumble tumbleCtx turbulence "+"twoPointArcCtx uiRes uiTemplate unassignInputDevice undo undoInfo ungroup uniform unit "+"unloadPlugin untangleUV untitledFileName untrim upAxis updateAE userCtx uvLink "+"uvSnapshot validateShelfName vectorize view2dToolCtx viewCamera viewClipPlane "+"viewFit viewHeadOn viewLookAt viewManip viewPlace viewSet visor volumeAxis vortex "+"waitCursor warning webBrowser webBrowserPrefs whatIs window windowPref wire "+"wireContext workspace wrinkle wrinkleContext writeTake xbmLangPathList xform",illegal:"</",contains:[hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"[\\$\\%\\@](\\^\\w\\b|#\\w+|[^\\s\\w{]|{\\w+}|\\w+)"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("autoit",function(hljs){var KEYWORDS="ByRef Case Const ContinueCase ContinueLoop "+"Default Dim Do Else ElseIf EndFunc EndIf EndSelect "+"EndSwitch EndWith Enum Exit ExitLoop For Func "+"Global If In Local Next ReDim Return Select Static "+"Step Switch Then To Until Volatile WEnd While With",LITERAL="True False And Null Not Or",BUILT_IN="Abs ACos AdlibRegister AdlibUnRegister Asc AscW ASin Assign ATan AutoItSetOption AutoItWinGetTitle AutoItWinSetTitle Beep Binary BinaryLen BinaryMid BinaryToString BitAND BitNOT BitOR BitRotate BitShift BitXOR BlockInput Break Call CDTray Ceiling Chr ChrW ClipGet ClipPut ConsoleRead ConsoleWrite ConsoleWriteError ControlClick ControlCommand ControlDisable ControlEnable ControlFocus ControlGetFocus ControlGetHandle ControlGetPos ControlGetText ControlHide ControlListView ControlMove ControlSend ControlSetText ControlShow ControlTreeView Cos Dec DirCopy DirCreate DirGetSize DirMove DirRemove DllCall DllCallAddress DllCallbackFree DllCallbackGetPtr DllCallbackRegister DllClose DllOpen DllStructCreate DllStructGetData DllStructGetPtr DllStructGetSize DllStructSetData DriveGetDrive DriveGetFileSystem DriveGetLabel DriveGetSerial DriveGetType DriveMapAdd DriveMapDel DriveMapGet DriveSetLabel DriveSpaceFree DriveSpaceTotal DriveStatus EnvGet EnvSet EnvUpdate Eval Execute Exp FileChangeDir FileClose FileCopy FileCreateNTFSLink FileCreateShortcut FileDelete FileExists FileFindFirstFile FileFindNextFile FileFlush FileGetAttrib FileGetEncoding FileGetLongName FileGetPos FileGetShortcut FileGetShortName FileGetSize FileGetTime FileGetVersion FileInstall FileMove FileOpen FileOpenDialog FileRead FileReadLine FileReadToArray FileRecycle FileRecycleEmpty FileSaveDialog FileSelectFolder FileSetAttrib FileSetEnd FileSetPos FileSetTime FileWrite FileWriteLine Floor FtpSetProxy FuncName GUICreate GUICtrlCreateAvi GUICtrlCreateButton GUICtrlCreateCheckbox GUICtrlCreateCombo GUICtrlCreateContextMenu GUICtrlCreateDate GUICtrlCreateDummy GUICtrlCreateEdit GUICtrlCreateGraphic GUICtrlCreateGroup GUICtrlCreateIcon GUICtrlCreateInput GUICtrlCreateLabel GUICtrlCreateList GUICtrlCreateListView GUICtrlCreateListViewItem GUICtrlCreateMenu GUICtrlCreateMenuItem GUICtrlCreateMonthCal GUICtrlCreateObj GUICtrlCreatePic GUICtrlCreateProgress GUICtrlCreateRadio GUICtrlCreateSlider GUICtrlCreateTab GUICtrlCreateTabItem GUICtrlCreateTreeView GUICtrlCreateTreeViewItem GUICtrlCreateUpdown GUICtrlDelete GUICtrlGetHandle GUICtrlGetState GUICtrlRead GUICtrlRecvMsg GUICtrlRegisterListViewSort GUICtrlSendMsg GUICtrlSendToDummy GUICtrlSetBkColor GUICtrlSetColor GUICtrlSetCursor GUICtrlSetData GUICtrlSetDefBkColor GUICtrlSetDefColor GUICtrlSetFont GUICtrlSetGraphic GUICtrlSetImage GUICtrlSetLimit GUICtrlSetOnEvent GUICtrlSetPos GUICtrlSetResizing GUICtrlSetState GUICtrlSetStyle GUICtrlSetTip GUIDelete GUIGetCursorInfo GUIGetMsg GUIGetStyle GUIRegisterMsg GUISetAccelerators GUISetBkColor GUISetCoord GUISetCursor GUISetFont GUISetHelp GUISetIcon GUISetOnEvent GUISetState GUISetStyle GUIStartGroup GUISwitch Hex HotKeySet HttpSetProxy HttpSetUserAgent HWnd InetClose InetGet InetGetInfo InetGetSize InetRead IniDelete IniRead IniReadSection IniReadSectionNames IniRenameSection IniWrite IniWriteSection InputBox Int IsAdmin IsArray IsBinary IsBool IsDeclared IsDllStruct IsFloat IsFunc IsHWnd IsInt IsKeyword IsNumber IsObj IsPtr IsString Log MemGetStats Mod MouseClick MouseClickDrag MouseDown MouseGetCursor MouseGetPos MouseMove MouseUp MouseWheel MsgBox Number ObjCreate ObjCreateInterface ObjEvent ObjGet ObjName OnAutoItExitRegister OnAutoItExitUnRegister Ping PixelChecksum PixelGetColor PixelSearch ProcessClose ProcessExists ProcessGetStats ProcessList ProcessSetPriority ProcessWait ProcessWaitClose ProgressOff ProgressOn ProgressSet Ptr Random RegDelete RegEnumKey RegEnumVal RegRead RegWrite Round Run RunAs RunAsWait RunWait Send SendKeepActive SetError SetExtended ShellExecute ShellExecuteWait Shutdown Sin Sleep SoundPlay SoundSetWaveVolume SplashImageOn SplashOff SplashTextOn Sqrt SRandom StatusbarGetText StderrRead StdinWrite StdioClose StdoutRead String StringAddCR StringCompare StringFormat StringFromASCIIArray StringInStr StringIsAlNum StringIsAlpha StringIsASCII StringIsDigit StringIsFloat StringIsInt StringIsLower StringIsSpace StringIsUpper StringIsXDigit StringLeft StringLen StringLower StringMid StringRegExp StringRegExpReplace StringReplace StringReverse StringRight StringSplit StringStripCR StringStripWS StringToASCIIArray StringToBinary StringTrimLeft StringTrimRight StringUpper Tan TCPAccept TCPCloseSocket TCPConnect TCPListen TCPNameToIP TCPRecv TCPSend TCPShutdown, UDPShutdown TCPStartup, UDPStartup TimerDiff TimerInit ToolTip TrayCreateItem TrayCreateMenu TrayGetMsg TrayItemDelete TrayItemGetHandle TrayItemGetState TrayItemGetText TrayItemSetOnEvent TrayItemSetState TrayItemSetText TraySetClick TraySetIcon TraySetOnEvent TraySetPauseIcon TraySetState TraySetToolTip TrayTip UBound UDPBind UDPCloseSocket UDPOpen UDPRecv UDPSend VarGetType WinActivate WinActive WinClose WinExists WinFlash WinGetCaretPos WinGetClassList WinGetClientSize WinGetHandle WinGetPos WinGetProcess WinGetState WinGetText WinGetTitle WinKill WinList WinMenuSelectItem WinMinimizeAll WinMinimizeAllUndo WinMove WinSetOnTop WinSetState WinSetTitle WinSetTrans WinWait",COMMENT={variants:[hljs.COMMENT(";","$",{relevance:0}),hljs.COMMENT("#cs","#ce"),hljs.COMMENT("#comments-start","#comments-end")]},VARIABLE={begin:"\\$[A-z0-9_]+"},STRING={className:"string",variants:[{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]},NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]},PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"comments include include-once NoTrayIcon OnAutoItStartRegister pragma compile RequireAdmin"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",keywords:{"meta-keyword":"include"},end:"$",contains:[STRING,{className:"meta-string",variants:[{begin:"<",end:">"},{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]}]},STRING,COMMENT]},CONSTANT={className:"symbol",begin:"@[A-z0-9_]+"},FUNCTION={className:"function",beginKeywords:"Func",end:"$",illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:[VARIABLE,STRING,NUMBER]}]};return{case_insensitive:true,illegal:/\/\*/,keywords:{keyword:KEYWORDS,built_in:BUILT_IN,literal:LITERAL},contains:[COMMENT,VARIABLE,STRING,NUMBER,PREPROCESSOR,CONSTANT,FUNCTION]}});hljs.registerLanguage("less",function(hljs){var IDENT_RE="[\\w-]+";var INTERP_IDENT_RE="("+IDENT_RE+"|@{"+IDENT_RE+"})";var RULES=[],VALUE=[];var STRING_MODE=function(c){return{className:"string",begin:"~?"+c+".*?"+c}};var IDENT_MODE=function(name,begin,relevance){return{className:name,begin:begin,relevance:relevance}};var PARENS_MODE={begin:"\\(",end:"\\)",contains:VALUE,relevance:0};VALUE.push(hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING_MODE("'"),STRING_MODE('"'),hljs.CSS_NUMBER_MODE,{begin:"(url|data-uri)\\(",starts:{className:"string",end:"[\\)\\n]",excludeEnd:true}},IDENT_MODE("number","#[0-9A-Fa-f]+\\b"),PARENS_MODE,IDENT_MODE("variable","@@?"+IDENT_RE,10),IDENT_MODE("variable","@{"+IDENT_RE+"}"),IDENT_MODE("built_in","~?`[^`]*?`"),{className:"attribute",begin:IDENT_RE+"\\s*:",end:":",returnBegin:true,excludeEnd:true},{className:"meta",begin:"!important"});var VALUE_WITH_RULESETS=VALUE.concat({begin:"{",end:"}",contains:RULES});var MIXIN_GUARD_MODE={beginKeywords:"when",endsWithParent:true,contains:[{beginKeywords:"and not"}].concat(VALUE)};var RULE_MODE={begin:INTERP_IDENT_RE+"\\s*:",returnBegin:true,end:"[;}]",relevance:0,contains:[{className:"attribute",begin:INTERP_IDENT_RE,end:":",excludeEnd:true,starts:{endsWithParent:true,illegal:"[<=$]",relevance:0,contains:VALUE}}]};var AT_RULE_MODE={className:"keyword",begin:"@(import|media|charset|font-face|(-[a-z]+-)?keyframes|supports|document|namespace|page|viewport|host)\\b",starts:{end:"[;{}]",returnEnd:true,contains:VALUE,relevance:0}};var VAR_RULE_MODE={className:"variable",variants:[{begin:"@"+IDENT_RE+"\\s*:",relevance:15},{begin:"@"+IDENT_RE}],starts:{end:"[;}]",returnEnd:true,contains:VALUE_WITH_RULESETS}};var SELECTOR_MODE={variants:[{begin:"[\\.#:&\\[>]",end:"[;{}]"},{begin:INTERP_IDENT_RE,end:"{"}],returnBegin:true,returnEnd:true,illegal:"[<='$\"]",relevance:0,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,MIXIN_GUARD_MODE,IDENT_MODE("keyword","all\\b"),IDENT_MODE("variable","@{"+IDENT_RE+"}"),IDENT_MODE("selector-tag",INTERP_IDENT_RE+"%?",0),IDENT_MODE("selector-id","#"+INTERP_IDENT_RE),IDENT_MODE("selector-class","\\."+INTERP_IDENT_RE,0),IDENT_MODE("selector-tag","&",0),{className:"selector-attr",begin:"\\[",end:"\\]"},{className:"selector-pseudo",begin:/:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/},{begin:"\\(",end:"\\)",contains:VALUE_WITH_RULESETS},{begin:"!important"}]};RULES.push(hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_RULE_MODE,VAR_RULE_MODE,RULE_MODE,SELECTOR_MODE);return{case_insensitive:true,illegal:"[=>'/<($\"]",contains:RULES}});hljs.registerLanguage("tp",function(hljs){var TPID={className:"number",begin:"[1-9][0-9]*",relevance:0};var TPLABEL={className:"symbol",begin:":[^\\]]+"};var TPDATA={className:"built_in",begin:"(AR|P|PAYLOAD|PR|R|SR|RSR|LBL|VR|UALM|MESSAGE|UTOOL|UFRAME|TIMER|"+"TIMER_OVERFLOW|JOINT_MAX_SPEED|RESUME_PROG|DIAG_REC)\\[",end:"\\]",contains:["self",TPID,TPLABEL]};var TPIO={className:"built_in",begin:"(AI|AO|DI|DO|F|RI|RO|UI|UO|GI|GO|SI|SO)\\[",end:"\\]",contains:["self",TPID,hljs.QUOTE_STRING_MODE,TPLABEL]};return{keywords:{keyword:"ABORT ACC ADJUST AND AP_LD BREAK CALL CNT COL CONDITION CONFIG DA DB "+"DIV DETECT ELSE END ENDFOR ERR_NUM ERROR_PROG FINE FOR GP GUARD INC "+"IF JMP LINEAR_MAX_SPEED LOCK MOD MONITOR OFFSET Offset OR OVERRIDE "+"PAUSE PREG PTH RT_LD RUN SELECT SKIP Skip TA TB TO TOOL_OFFSET "+"Tool_Offset UF UT UFRAME_NUM UTOOL_NUM UNLOCK WAIT X Y Z W P R STRLEN "+"SUBSTR FINDSTR VOFFSET PROG ATTR MN POS",literal:"ON OFF max_speed LPOS JPOS ENABLE DISABLE START STOP RESET"},contains:[TPDATA,TPIO,{className:"keyword",begin:"/(PROG|ATTR|MN|POS|END)\\b"},{className:"keyword",begin:"(CALL|RUN|POINT_LOGIC|LBL)\\b"},{className:"keyword",begin:"\\b(ACC|CNT|Skip|Offset|PSPD|RT_LD|AP_LD|Tool_Offset)"},{className:"number",begin:"\\d+(sec|msec|mm/sec|cm/min|inch/min|deg/sec|mm|in|cm)?\\b",relevance:0},hljs.COMMENT("//","[;$]"),hljs.COMMENT("!","[;$]"),hljs.COMMENT("--eg:","$"),hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"'"},hljs.C_NUMBER_MODE,{className:"variable",begin:"\\$[A-Za-z0-9_]+"}]}});hljs.registerLanguage("cos",function cos(hljs){var STRINGS={className:"string",variants:[{begin:'"',end:'"',contains:[{begin:'""',relevance:0}]}]};var NUMBERS={className:"number",begin:"\\b(\\d+(\\.\\d*)?|\\.\\d+)",relevance:0};var COS_KEYWORDS="property parameter class classmethod clientmethod extends as break "+"catch close continue do d|0 else elseif for goto halt hang h|0 if job "+"j|0 kill k|0 lock l|0 merge new open quit q|0 read r|0 return set s|0 "+"tcommit throw trollback try tstart use view while write w|0 xecute x|0 "+"zkill znspace zn ztrap zwrite zw zzdump zzwrite print zbreak zinsert "+"zload zprint zremove zsave zzprint mv mvcall mvcrt mvdim mvprint zquit "+"zsync ascii";return{case_insensitive:true,aliases:["cos","cls"],keywords:COS_KEYWORDS,contains:[NUMBERS,STRINGS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"comment",begin:/;/,end:"$",relevance:0},{className:"built_in",begin:/(?:\$\$?|\.\.)\^?[a-zA-Z]+/},{className:"built_in",begin:/\$\$\$[a-zA-Z]+/},{className:"built_in",begin:/%[a-z]+(?:\.[a-z]+)*/},{className:"symbol",begin:/\^%?[a-zA-Z][\w]*/},{className:"keyword",begin:/##class|##super|#define|#dim/},{begin:/&sql\(/,end:/\)/,excludeBegin:true,excludeEnd:true,subLanguage:"sql"},{begin:/&(js|jscript|javascript)</,end:/>/,excludeBegin:true,excludeEnd:true,subLanguage:"javascript"},{begin:/&html<\s*</,end:/>\s*>/,subLanguage:"xml"}]}});hljs.registerLanguage("erlang-repl",function(hljs){return{keywords:{built_in:"spawn spawn_link self",keyword:"after and andalso|10 band begin bnot bor bsl bsr bxor case catch cond div end fun if "+"let not of or orelse|10 query receive rem try when xor"},contains:[{className:"meta",begin:"^[0-9]+> ",relevance:10},hljs.COMMENT("%","$"),{className:"number",begin:"\\b(\\d+#[a-fA-F0-9]+|\\d+(\\.\\d+)?([eE][-+]?\\d+)?)",relevance:0},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\?(::)?([A-Z]\\w*(::)?)+"},{begin:"->"},{begin:"ok"},{begin:"!"},{begin:"(\\b[a-z'][a-zA-Z0-9_']*:[a-z'][a-zA-Z0-9_']*)|(\\b[a-z'][a-zA-Z0-9_']*)",relevance:0},{begin:"[A-Z][a-zA-Z0-9_']*",relevance:0}]}});hljs.registerLanguage("vbscript",function(hljs){return{aliases:["vbs"],case_insensitive:true,keywords:{keyword:"call class const dim do loop erase execute executeglobal exit for each next function "+"if then else on error option explicit new private property let get public randomize "+"redim rem select case set stop sub while wend with end to elseif is or xor and not "+"class_initialize class_terminate default preserve in me byval byref step resume goto",built_in:"lcase month vartype instrrev ubound setlocale getobject rgb getref string "+"weekdayname rnd dateadd monthname now day minute isarray cbool round formatcurrency "+"conversions csng timevalue second year space abs clng timeserial fixs len asc "+"isempty maths dateserial atn timer isobject filter weekday datevalue ccur isdate "+"instr datediff formatdatetime replace isnull right sgn array snumeric log cdbl hex "+"chr lbound msgbox ucase getlocale cos cdate cbyte rtrim join hour oct typename trim "+"strcomp int createobject loadpicture tan formatnumber mid scriptenginebuildversion "+"scriptengine split scriptengineminorversion cint sin datepart ltrim sqr "+"scriptenginemajorversion time derived eval date formatpercent exp inputbox left ascw "+"chrw regexp server response request cstr err",literal:"true false null nothing empty"},illegal:"//",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[{begin:'""'}]}),hljs.COMMENT(/'/,/$/,{relevance:0}),hljs.C_NUMBER_MODE]}});hljs.registerLanguage("vbscript-html",function(hljs){return{subLanguage:"xml",contains:[{begin:"<%",end:"%>",subLanguage:"vbscript"}]}});hljs.registerLanguage("twig",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var FUNCTION_NAMES="attribute block constant cycle date dump include "+"max min parent random range source template_from_string";var FUNCTIONS={beginKeywords:FUNCTION_NAMES,keywords:{name:FUNCTION_NAMES},relevance:0,contains:[PARAMS]};var FILTER={begin:/\|[A-Za-z_]+:?/,keywords:"abs batch capitalize column convert_encoding date date_modify default "+"escape filter first format inky_to_html inline_css join json_encode keys last "+"length lower map markdown merge nl2br number_format raw reduce replace "+"reverse round slice sort spaceless split striptags title trim upper url_encode",contains:[FUNCTIONS]};var TAGS="apply autoescape block deprecated do embed extends filter flush for from "+"if import include macro sandbox set use verbatim with";TAGS=TAGS+" "+TAGS.split(" ").map(function(t){return"end"+t}).join(" ");return{aliases:["craftcms"],case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT(/\{#/,/#}/),{className:"template-tag",begin:/\{%/,end:/%}/,contains:[{className:"name",begin:/\w+/,keywords:TAGS,starts:{endsWithParent:true,contains:[FILTER,FUNCTIONS],relevance:0}}]},{className:"template-variable",begin:/\{\{/,end:/}}/,contains:["self",FILTER,FUNCTIONS]}]}});hljs.registerLanguage("cmake",function(hljs){return{aliases:["cmake.in"],case_insensitive:true,keywords:{keyword:"break cmake_host_system_information cmake_minimum_required cmake_parse_arguments "+"cmake_policy configure_file continue elseif else endforeach endfunction endif endmacro "+"endwhile execute_process file find_file find_library find_package find_path "+"find_program foreach function get_cmake_property get_directory_property "+"get_filename_component get_property if include include_guard list macro "+"mark_as_advanced math message option return separate_arguments "+"set_directory_properties set_property set site_name string unset variable_watch while "+"add_compile_definitions add_compile_options add_custom_command add_custom_target "+"add_definitions add_dependencies add_executable add_library add_link_options "+"add_subdirectory add_test aux_source_directory build_command create_test_sourcelist "+"define_property enable_language enable_testing export fltk_wrap_ui "+"get_source_file_property get_target_property get_test_property include_directories "+"include_external_msproject include_regular_expression install link_directories "+"link_libraries load_cache project qt_wrap_cpp qt_wrap_ui remove_definitions "+"set_source_files_properties set_target_properties set_tests_properties source_group "+"target_compile_definitions target_compile_features target_compile_options "+"target_include_directories target_link_directories target_link_libraries "+"target_link_options target_sources try_compile try_run "+"ctest_build ctest_configure ctest_coverage ctest_empty_binary_directory ctest_memcheck "+"ctest_read_custom_files ctest_run_script ctest_sleep ctest_start ctest_submit "+"ctest_test ctest_update ctest_upload "+"build_name exec_program export_library_dependencies install_files install_programs "+"install_targets load_command make_directory output_required_files remove "+"subdir_depends subdirs use_mangled_mesa utility_source variable_requires write_file "+"qt5_use_modules qt5_use_package qt5_wrap_cpp "+"on off true false and or not command policy target test exists is_newer_than "+"is_directory is_symlink is_absolute matches less greater equal less_equal "+"greater_equal strless strgreater strequal strless_equal strgreater_equal version_less "+"version_greater version_equal version_less_equal version_greater_equal in_list defined"},contains:[{className:"variable",begin:"\\${",end:"}"},hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("handlebars",function(hljs){var BUILT_INS={"builtin-name":"each in with if else unless bindattr action collection debugger log outlet template unbound view yield lookup"};var IDENTIFIER_PLAIN_OR_QUOTED={begin:/".*?"|'.*?'|\[.*?\]|\w+/};var EXPRESSION_OR_HELPER_CALL=hljs.inherit(IDENTIFIER_PLAIN_OR_QUOTED,{keywords:BUILT_INS,starts:{endsWithParent:true,relevance:0,contains:[hljs.inherit(IDENTIFIER_PLAIN_OR_QUOTED,{relevance:0})]}});var BLOCK_MUSTACHE_CONTENTS=hljs.inherit(EXPRESSION_OR_HELPER_CALL,{className:"name"});var BASIC_MUSTACHE_CONTENTS=hljs.inherit(EXPRESSION_OR_HELPER_CALL,{relevance:0});var ESCAPE_MUSTACHE_WITH_PRECEEDING_BACKSLASH={begin:/\\\{\{/,skip:true};var PREVENT_ESCAPE_WITH_ANOTHER_PRECEEDING_BACKSLASH={begin:/\\\\(?=\{\{)/,skip:true};return{aliases:["hbs","html.hbs","html.handlebars"],case_insensitive:true,subLanguage:"xml",contains:[ESCAPE_MUSTACHE_WITH_PRECEEDING_BACKSLASH,PREVENT_ESCAPE_WITH_ANOTHER_PRECEEDING_BACKSLASH,hljs.COMMENT(/\{\{!--/,/--\}\}/),hljs.COMMENT(/\{\{!/,/\}\}/),{className:"template-tag",begin:/\{\{\{\{(?!\/)/,end:/\}\}\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS],starts:{end:/\{\{\{\{\//,returnEnd:true,subLanguage:"xml"}},{className:"template-tag",begin:/\{\{\{\{\//,end:/\}\}\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS]},{className:"template-tag",begin:/\{\{[#\/]/,end:/\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS]},{className:"template-variable",begin:/\{\{\{/,end:/\}\}\}/,keywords:BUILT_INS,contains:[BASIC_MUSTACHE_CONTENTS]},{className:"template-variable",begin:/\{\{/,end:/\}\}/,keywords:BUILT_INS,contains:[BASIC_MUSTACHE_CONTENTS]}]}});hljs.registerLanguage("delphi",function(hljs){var KEYWORDS="exports register file shl array record property for mod while set ally label uses raise not "+"stored class safecall var interface or private static exit index inherited to else stdcall "+"override shr asm far resourcestring finalization packed virtual out and protected library do "+"xorwrite goto near function end div overload object unit begin string on inline repeat until "+"destructor write message program with read initialization except default nil if case cdecl in "+"downto threadvar of try pascal const external constructor type public then implementation "+"finally published procedure absolute reintroduce operator as is abstract alias assembler "+"bitpacked break continue cppdecl cvar enumerator experimental platform deprecated "+"unimplemented dynamic export far16 forward generic helper implements interrupt iochecks "+"local name nodefault noreturn nostackframe oldfpccall otherwise saveregisters softfloat "+"specialize strict unaligned varargs ";var COMMENT_MODES=[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/\{/,/\}/,{relevance:0}),hljs.COMMENT(/\(\*/,/\*\)/,{relevance:10})];var DIRECTIVE={className:"meta",variants:[{begin:/\{\$/,end:/\}/},{begin:/\(\*\$/,end:/\*\)/}]};var STRING={className:"string",begin:/'/,end:/'/,contains:[{begin:/''/}]};var CHAR_STRING={className:"string",begin:/(#\d+)+/};var CLASS={begin:hljs.IDENT_RE+"\\s*=\\s*class\\s*\\(",returnBegin:true,contains:[hljs.TITLE_MODE]};var FUNCTION={className:"function",beginKeywords:"function constructor destructor procedure",end:/[:;]/,keywords:"function constructor|10 destructor|10 procedure|10",contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:[STRING,CHAR_STRING,DIRECTIVE].concat(COMMENT_MODES)},DIRECTIVE].concat(COMMENT_MODES)};return{aliases:["dpr","dfm","pas","pascal","freepascal","lazarus","lpr","lfm"],case_insensitive:true,keywords:KEYWORDS,illegal:/"|\$[G-Zg-z]|\/\*|<\/|\|/,contains:[STRING,CHAR_STRING,hljs.NUMBER_MODE,CLASS,FUNCTION,DIRECTIVE].concat(COMMENT_MODES)}});hljs.registerLanguage("javascript",function(hljs){var FRAGMENT={begin:"<>",end:"</>"};var XML_TAG={begin:/<[A-Za-z0-9\\._:-]+/,end:/\/[A-Za-z0-9\\._:-]+>|\/>/};var IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var KEYWORDS={keyword:"in of if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const export super debugger as async await static "+"import from as",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect "+"Promise"};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)n?"},{begin:"\\b(0[oO][0-7]+)n?"},{begin:hljs.C_NUMBER_RE+"n?"}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var HTML_TEMPLATE={begin:"html`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"xml"}};var CSS_TEMPLATE={begin:"css`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"css"}};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];var PARAMS_CONTAINS=SUBST.contains.concat([hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE]);return{aliases:["js","jsx","mjs","cjs"],keywords:KEYWORDS,contains:[{className:"meta",relevance:10,begin:/^\s*['"]use (strict|asm)['"]/},{className:"meta",begin:/^#!/,end:/$/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag",begin:"@[A-Za-z]+",contains:[{className:"type",begin:"\\{",end:"\\}",relevance:0},{className:"variable",begin:IDENT_RE+"(?=\\s*(-)|$)",endsParent:true,relevance:0},{begin:/(?=[^\n])\s/,relevance:0}]}]}),hljs.C_BLOCK_COMMENT_MODE,NUMBER,{begin:/[{,\n]\s*/,relevance:0,contains:[{begin:IDENT_RE+"\\s*:",returnBegin:true,relevance:0,contains:[{className:"attr",begin:IDENT_RE,relevance:0}]}]},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:PARAMS_CONTAINS}]}]},{className:"",begin:/\s/,end:/\s*/,skip:true},{variants:[{begin:FRAGMENT.begin,end:FRAGMENT.end},{begin:XML_TAG.begin,end:XML_TAG.end}],subLanguage:"xml",contains:[{begin:XML_TAG.begin,end:XML_TAG.end,skip:true,contains:["self"]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:PARAMS_CONTAINS}],illegal:/\[|%/},{begin:/\$[(.]/},hljs.METHOD_GUARD,{className:"class",beginKeywords:"class",end:/[{;=]/,excludeEnd:true,illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"constructor get set",end:/\{/,excludeEnd:true}],illegal:/#(?!!)/}});hljs.registerLanguage("rust",function(hljs){var NUM_SUFFIX="([ui](8|16|32|64|128|size)|f(32|64))?";var KEYWORDS="abstract as async await become box break const continue crate do dyn "+"else enum extern false final fn for if impl in let loop macro match mod "+"move mut override priv pub ref return self Self static struct super "+"trait true try type typeof unsafe unsized use virtual where while yield";var BUILTINS="drop "+"i8 i16 i32 i64 i128 isize "+"u8 u16 u32 u64 u128 usize "+"f32 f64 "+"str char bool "+"Box Option Result String Vec "+"Copy Send Sized Sync Drop Fn FnMut FnOnce ToOwned Clone Debug "+"PartialEq PartialOrd Eq Ord AsRef AsMut Into From Default Iterator "+"Extend IntoIterator DoubleEndedIterator ExactSizeIterator "+"SliceConcatExt ToString "+"assert! assert_eq! bitflags! bytes! cfg! col! concat! concat_idents! "+"debug_assert! debug_assert_eq! env! panic! file! format! format_args! "+"include_bin! include_str! line! local_data_key! module_path! "+"option_env! print! println! select! stringify! try! unimplemented! "+"unreachable! vec! write! writeln! macro_rules! assert_ne! debug_assert_ne!";return{aliases:["rs"],keywords:{keyword:KEYWORDS,literal:"true false Some None Ok Err",built_in:BUILTINS},lexemes:hljs.IDENT_RE+"!?",illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:["self"]}),hljs.inherit(hljs.QUOTE_STRING_MODE,{begin:/b?"/,illegal:null}),{className:"string",variants:[{begin:/r(#*)"(.|\n)*?"\1(?!#)/},{begin:/b?'\\?(x\w{2}|u\w{4}|U\w{8}|.)'/}]},{className:"symbol",begin:/'[a-zA-Z_][a-zA-Z0-9_]*/},{className:"number",variants:[{begin:"\\b0b([01_]+)"+NUM_SUFFIX},{begin:"\\b0o([0-7_]+)"+NUM_SUFFIX},{begin:"\\b0x([A-Fa-f0-9_]+)"+NUM_SUFFIX},{begin:"\\b(\\d[\\d_]*(\\.[0-9_]+)?([eE][+-]?[0-9_]+)?)"+NUM_SUFFIX}],relevance:0},{className:"function",beginKeywords:"fn",end:"(\\(|<)",excludeEnd:true,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"meta",begin:"#\\!?\\[",end:"\\]",contains:[{className:"meta-string",begin:/"/,end:/"/}]},{className:"class",beginKeywords:"type",end:";",contains:[hljs.inherit(hljs.UNDERSCORE_TITLE_MODE,{endsParent:true})],illegal:"\\S"},{className:"class",beginKeywords:"trait enum struct union",end:"{",contains:[hljs.inherit(hljs.UNDERSCORE_TITLE_MODE,{endsParent:true})],illegal:"[\\w\\d]"},{begin:hljs.IDENT_RE+"::",keywords:{built_in:BUILTINS}},{begin:"->"}]}});hljs.registerLanguage("diff",function(hljs){return{aliases:["patch"],contains:[{className:"meta",relevance:10,variants:[{begin:/^@@ +\-\d+,\d+ +\+\d+,\d+ +@@$/},{begin:/^\*\*\* +\d+,\d+ +\*\*\*\*$/},{begin:/^\-\-\- +\d+,\d+ +\-\-\-\-$/}]},{className:"comment",variants:[{begin:/Index: /,end:/$/},{begin:/={3,}/,end:/$/},{begin:/^\-{3}/,end:/$/},{begin:/^\*{3} /,end:/$/},{begin:/^\+{3}/,end:/$/},{begin:/^\*{15}$/}]},{className:"addition",begin:"^\\+",end:"$"},{className:"deletion",begin:"^\\-",end:"$"},{className:"addition",begin:"^\\!",end:"$"}]}});hljs.registerLanguage("thrift",function(hljs){var BUILT_IN_TYPES="bool byte i16 i32 i64 double string binary";return{keywords:{keyword:"namespace const typedef struct enum service exception void oneway set list map required optional",built_in:BUILT_IN_TYPES,literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"class",beginKeywords:"struct enum service exception",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{begin:"\\b(set|list|map)\\s*<",end:">",keywords:BUILT_IN_TYPES,contains:["self"]}]}});hljs.registerLanguage("accesslog",function(hljs){var HTTP_VERBS=["GET","POST","HEAD","PUT","DELETE","CONNECT","OPTIONS","PATCH","TRACE"];return{contains:[{className:"number",begin:"^\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b",relevance:5},{className:"number",begin:"\\b\\d+\\b",relevance:0},{className:"string",begin:'"('+HTTP_VERBS.join("|")+")",end:'"',keywords:HTTP_VERBS.join(" "),illegal:"\\n",relevance:5,contains:[{begin:"HTTP/[12]\\.\\d",relevance:5}]},{className:"string",begin:/\[\d[^\]\n]{8,}\]/,illegal:"\\n",relevance:1},{className:"string",begin:/\[/,end:/\]/,illegal:"\\n",relevance:0},{className:"string",begin:'"Mozilla/\\d\\.\\d \\(',end:'"',illegal:"\\n",relevance:3},{className:"string",begin:'"',end:'"',illegal:"\\n",relevance:0}]}});hljs.registerLanguage("tex",function(hljs){var COMMAND={className:"tag",begin:/\\/,relevance:0,contains:[{className:"name",variants:[{begin:/[a-zA-Z\u0430-\u044f\u0410-\u042f]+[*]?/},{begin:/[^a-zA-Z\u0430-\u044f\u0410-\u042f0-9]/}],starts:{endsWithParent:true,relevance:0,contains:[{className:"string",variants:[{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/}]},{begin:/\s*=\s*/,endsWithParent:true,relevance:0,contains:[{className:"number",begin:/-?\d*\.?\d+(pt|pc|mm|cm|in|dd|cc|ex|em)?/}]}]}}]};return{contains:[COMMAND,{className:"formula",contains:[COMMAND],relevance:0,variants:[{begin:/\$\$/,end:/\$\$/},{begin:/\$/,end:/\$/}]},hljs.COMMENT("%","$",{relevance:0})]}});hljs.registerLanguage("livescript",function(hljs){var KEYWORDS={keyword:"in if for while finally new do return else break catch instanceof throw try this "+"switch continue typeof delete debugger case default function var with "+"then unless until loop of by when and or is isnt not it that otherwise from to til fallthrough super "+"case default function var void const let enum export import native list map "+"__hasProp __extends __slice __bind __indexOf",literal:"true false null undefined "+"yes no on off it that void",built_in:"npm require console print module global window document"};var JS_IDENT_RE="[A-Za-z$_](?:-[0-9A-Za-z$_]|[0-9A-Za-z$_])*";var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var SUBST_SIMPLE={className:"subst",begin:/#[A-Za-z$_]/,end:/(?:\-[0-9A-Za-z$_]|[0-9A-Za-z$_])*/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.BINARY_NUMBER_MODE,{className:"number",begin:"(\\b0[xX][a-fA-F0-9_]+)|(\\b\\d(\\d|_\\d)*(\\.(\\d(\\d|_\\d)*)?)?(_*[eE]([-+]\\d(_\\d|\\d)*)?)?[_a-z]*)",relevance:0,starts:{end:"(\\s*/)?",relevance:0}},{className:"string",variants:[{begin:/'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,SUBST,SUBST_SIMPLE]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST,SUBST_SIMPLE]},{begin:/\\/,end:/(\s|$)/,excludeEnd:true}]},{className:"regexp",variants:[{begin:"//",end:"//[gim]*",contains:[SUBST,hljs.HASH_COMMENT_MODE]},{begin:/\/(?![ *])(\\.|[^\\\n])*?\/[gim]*(?=\W)/}]},{begin:"@"+JS_IDENT_RE},{begin:"``",end:"``",excludeBegin:true,excludeEnd:true,subLanguage:"javascript"}];SUBST.contains=EXPRESSIONS;var PARAMS={className:"params",begin:"\\(",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};var SYMBOLS={begin:"(#=>|=>|\\|>>|-?->|\\!->)"};return{aliases:["ls"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("\\/\\*","\\*\\/"),hljs.HASH_COMMENT_MODE,SYMBOLS,{className:"function",contains:[TITLE,PARAMS],returnBegin:true,variants:[{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?(\\(.*\\))?\\s*\\B\\->\\*?",end:"\\->\\*?"},{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?!?(\\(.*\\))?\\s*\\B[-~]{1,2}>\\*?",end:"[-~]{1,2}>\\*?"},{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?(\\(.*\\))?\\s*\\B!?[-~]{1,2}>\\*?",end:"!?[-~]{1,2}>\\*?"}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("apache",function(hljs){var NUMBER={className:"number",begin:"[\\$%]\\d+"};return{aliases:["apacheconf"],case_insensitive:true,contains:[hljs.HASH_COMMENT_MODE,{className:"section",begin:"</?",end:">"},{className:"attribute",begin:/\w+/,relevance:0,keywords:{nomarkup:"order deny allow setenv rewriterule rewriteengine rewritecond documentroot "+"sethandler errordocument loadmodule options header listen serverroot "+"servername"},starts:{end:/$/,relevance:0,keywords:{literal:"on off all"},contains:[{className:"meta",begin:"\\s\\[",end:"\\]$"},{className:"variable",begin:"[\\$%]\\{",end:"\\}",contains:["self",NUMBER]},NUMBER,hljs.QUOTE_STRING_MODE]}}],illegal:/\S/}});hljs.registerLanguage("typescript",function(hljs){var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var KEYWORDS={keyword:"in if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const class public private protected get set super "+"static implements enum export import declare type namespace abstract "+"as from extends async await",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document any number boolean string void Promise"};var DECORATOR={className:"meta",begin:"@"+JS_IDENT_RE};var ARGS={begin:"\\(",end:/\)/,keywords:KEYWORDS,contains:["self",hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.NUMBER_MODE]};var PARAMS={className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,DECORATOR,ARGS]};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)n?"},{begin:"\\b(0[oO][0-7]+)n?"},{begin:hljs.C_NUMBER_RE+"n?"}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var HTML_TEMPLATE={begin:"html`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"xml"}};var CSS_TEMPLATE={begin:"css`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"css"}};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];return{aliases:["ts"],keywords:KEYWORDS,contains:[{className:"meta",begin:/^\s*['"]use strict['"]/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBER,{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+hljs.IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:hljs.IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:["self",hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/[\{;]/,excludeEnd:true,keywords:KEYWORDS,contains:["self",hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE}),PARAMS],illegal:/%/,relevance:0},{beginKeywords:"constructor",end:/[\{;]/,excludeEnd:true,contains:["self",PARAMS]},{begin:/module\./,keywords:{built_in:"module"},relevance:0},{beginKeywords:"module",end:/\{/,excludeEnd:true},{beginKeywords:"interface",end:/\{/,excludeEnd:true,keywords:"interface extends"},{begin:/\$[(.]/},{begin:"\\."+hljs.IDENT_RE,relevance:0},DECORATOR,ARGS]}});hljs.registerLanguage("roboconf",function(hljs){var IDENTIFIER="[a-zA-Z-_][^\\n{]+\\{";var PROPERTY={className:"attribute",begin:/[a-zA-Z-_]+/,end:/\s*:/,excludeEnd:true,starts:{end:";",relevance:0,contains:[{className:"variable",begin:/\.[a-zA-Z-_]+/},{className:"keyword",begin:/\(optional\)/}]}};return{aliases:["graph","instances"],case_insensitive:true,keywords:"import",contains:[{begin:"^facet "+IDENTIFIER,end:"}",keywords:"facet",contains:[PROPERTY,hljs.HASH_COMMENT_MODE]},{begin:"^\\s*instance of "+IDENTIFIER,end:"}",keywords:"name count channels instance-data instance-state instance of",illegal:/\S/,contains:["self",PROPERTY,hljs.HASH_COMMENT_MODE]},{begin:"^"+IDENTIFIER,end:"}",contains:[PROPERTY,hljs.HASH_COMMENT_MODE]},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("htmlbars",function(hljs){var BUILT_INS="action collection component concat debugger each each-in else get hash if input link-to loc log mut outlet partial query-params render textarea unbound unless with yield view";var ATTR_ASSIGNMENT={illegal:/\}\}/,begin:/[a-zA-Z0-9_]+=/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/[a-zA-Z0-9_]+/}]};var SUB_EXPR={illegal:/\}\}/,begin:/\)/,end:/\)/,contains:[{begin:/[a-zA-Z\.\-]+/,keywords:{built_in:BUILT_INS},starts:{endsWithParent:true,relevance:0,contains:[hljs.QUOTE_STRING_MODE]}}]};var TAG_INNARDS={endsWithParent:true,relevance:0,keywords:{keyword:"as",built_in:BUILT_INS},contains:[hljs.QUOTE_STRING_MODE,ATTR_ASSIGNMENT,hljs.NUMBER_MODE]};return{case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT("{{!(--)?","(--)?}}"),{className:"template-tag",begin:/\{\{[#\/]/,end:/\}\}/,contains:[{className:"name",begin:/[a-zA-Z\.\-]+/,keywords:{"builtin-name":BUILT_INS},starts:TAG_INNARDS}]},{className:"template-variable",begin:/\{\{[a-zA-Z][a-zA-Z\-]+/,end:/\}\}/,keywords:{keyword:"as",built_in:BUILT_INS},contains:[hljs.QUOTE_STRING_MODE]}]}});hljs.registerLanguage("routeros",function(hljs){var STATEMENTS="foreach do while for if from to step else on-error and or not in";var GLOBAL_COMMANDS="global local beep delay put len typeof pick log time set find environment terminal error execute parse resolve toarray tobool toid toip toip6 tonum tostr totime";var COMMON_COMMANDS="add remove enable disable set get print export edit find run debug error info warning";var LITERALS="true false yes no nothing nil null";var OBJECTS="traffic-flow traffic-generator firewall scheduler aaa accounting address-list address align area bandwidth-server bfd bgp bridge client clock community config connection console customer default dhcp-client dhcp-server discovery dns e-mail ethernet filter firewall firmware gps graphing group hardware health hotspot identity igmp-proxy incoming instance interface ip ipsec ipv6 irq l2tp-server lcd ldp logging mac-server mac-winbox mangle manual mirror mme mpls nat nd neighbor network note ntp ospf ospf-v3 ovpn-server page peer pim ping policy pool port ppp pppoe-client pptp-server prefix profile proposal proxy queue radius resource rip ripng route routing screen script security-profiles server service service-port settings shares smb sms sniffer snmp snooper socks sstp-server system tool tracking type upgrade upnp user-manager users user vlan secret vrrp watchdog web-access wireless pptp pppoe lan wan layer7-protocol lease simple raw";var VAR_PREFIX="global local set for foreach";var VAR={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VAR,{className:"variable",begin:/\$\(/,end:/\)/,contains:[hljs.BACKSLASH_ESCAPE]}]};var APOS_STRING={className:"string",begin:/'/,end:/'/};var IPADDR="((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]).){3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\\b";var IPADDR_wBITMASK=IPADDR+"/(3[0-2]|[1-2][0-9]|\\d)";return{aliases:["routeros","mikrotik"],case_insensitive:true,lexemes:/:?[\w-]+/,keywords:{literal:LITERALS,keyword:STATEMENTS+" :"+STATEMENTS.split(" ").join(" :")+" :"+GLOBAL_COMMANDS.split(" ").join(" :")},contains:[{variants:[{begin:/^@/,end:/$/},{begin:/\/\*/,end:/\*\//},{begin:/%%/,end:/$/},{begin:/^'/,end:/$/},{begin:/^\s*\/[\w-]+=/,end:/$/},{begin:/\/\//,end:/$/},{begin:/^\[\</,end:/\>\]$/},{begin:/<\//,end:/>/},{begin:/^facet /,end:/\}/},{begin:"^1\\.\\.(\\d+)$",end:/$/}],illegal:/./},hljs.COMMENT("^#","$"),QUOTE_STRING,APOS_STRING,VAR,{begin:/[\w-]+\=([^\s\{\}\[\]\(\)]+)/,relevance:0,returnBegin:true,contains:[{className:"attribute",begin:/[^=]+/},{begin:/=/,endsWithParent:true,relevance:0,contains:[QUOTE_STRING,APOS_STRING,VAR,{className:"literal",begin:"\\b("+LITERALS.split(" ").join("|")+")\\b"},{begin:/("[^"]*"|[^\s\{\}\[\]]+)/}]}]},{className:"number",begin:/\*[0-9a-fA-F]+/},{begin:"\\b("+COMMON_COMMANDS.split(" ").join("|")+")([\\s[(]|])",returnBegin:true,contains:[{className:"builtin-name",begin:/\w+/}]},{className:"built_in",variants:[{begin:"(\\.\\./|/|\\s)(("+OBJECTS.split(" ").join("|")+");?\\s)+",relevance:10},{begin:/\.\./}]}]}});hljs.registerLanguage("mercury",function(hljs){var KEYWORDS={keyword:"module use_module import_module include_module end_module initialise "+"mutable initialize finalize finalise interface implementation pred "+"mode func type inst solver any_pred any_func is semidet det nondet "+"multi erroneous failure cc_nondet cc_multi typeclass instance where "+"pragma promise external trace atomic or_else require_complete_switch "+"require_det require_semidet require_multi require_nondet "+"require_cc_multi require_cc_nondet require_erroneous require_failure",meta:"inline no_inline type_spec source_file fact_table obsolete memo "+"loop_check minimal_model terminates does_not_terminate "+"check_termination promise_equivalent_clauses "+"foreign_proc foreign_decl foreign_code foreign_type "+"foreign_import_module foreign_export_enum foreign_export "+"foreign_enum may_call_mercury will_not_call_mercury thread_safe "+"not_thread_safe maybe_thread_safe promise_pure promise_semipure "+"tabled_for_io local untrailed trailed attach_to_io_state "+"can_pass_as_mercury_type stable will_not_throw_exception "+"may_modify_trail will_not_modify_trail may_duplicate "+"may_not_duplicate affects_liveness does_not_affect_liveness "+"doesnt_affect_liveness no_sharing unknown_sharing sharing",built_in:"some all not if then else true fail false try catch catch_any "+"semidet_true semidet_false semidet_fail impure_true impure semipure"};var COMMENT=hljs.COMMENT("%","$");var NUMCODE={className:"number",begin:"0'.\\|0[box][0-9a-fA-F]*"};var ATOM=hljs.inherit(hljs.APOS_STRING_MODE,{relevance:0});var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{relevance:0});var STRING_FMT={className:"subst",begin:"\\\\[abfnrtv]\\|\\\\x[0-9a-fA-F]*\\\\\\|%[-+# *.0-9]*[dioxXucsfeEgGp]",relevance:0};STRING.contains=STRING.contains.slice();STRING.contains.push(STRING_FMT);var IMPLICATION={className:"built_in",variants:[{begin:"<=>"},{begin:"<=",relevance:0},{begin:"=>",relevance:0},{begin:"/\\\\"},{begin:"\\\\/"}]};var HEAD_BODY_CONJUNCTION={className:"built_in",variants:[{begin:":-\\|--\x3e"},{begin:"=",relevance:0}]};return{aliases:["m","moo"],keywords:KEYWORDS,contains:[IMPLICATION,HEAD_BODY_CONJUNCTION,COMMENT,hljs.C_BLOCK_COMMENT_MODE,NUMCODE,hljs.NUMBER_MODE,ATOM,STRING,{begin:/:-/},{begin:/\.$/}]}});hljs.registerLanguage("inform7",function(hljs){var START_BRACKET="\\[";var END_BRACKET="\\]";return{aliases:["i7"],case_insensitive:true,keywords:{keyword:"thing room person man woman animal container "+"supporter backdrop door "+"scenery open closed locked inside gender "+"is are say understand "+"kind of rule"},contains:[{className:"string",begin:'"',end:'"',relevance:0,contains:[{className:"subst",begin:START_BRACKET,end:END_BRACKET}]},{className:"section",begin:/^(Volume|Book|Part|Chapter|Section|Table)\b/,end:"$"},{begin:/^(Check|Carry out|Report|Instead of|To|Rule|When|Before|After)\b/,end:":",contains:[{begin:"\\(This",end:"\\)"}]},{className:"comment",begin:START_BRACKET,end:END_BRACKET,contains:["self"]}]}});hljs.registerLanguage("lua",function(hljs){var OPENING_LONG_BRACKET="\\[=*\\[";var CLOSING_LONG_BRACKET="\\]=*\\]";var LONG_BRACKETS={begin:OPENING_LONG_BRACKET,end:CLOSING_LONG_BRACKET,contains:["self"]};var COMMENTS=[hljs.COMMENT("--(?!"+OPENING_LONG_BRACKET+")","$"),hljs.COMMENT("--"+OPENING_LONG_BRACKET,CLOSING_LONG_BRACKET,{contains:[LONG_BRACKETS],relevance:10})];return{lexemes:hljs.UNDERSCORE_IDENT_RE,keywords:{literal:"true false nil",keyword:"and break do else elseif end for goto if in local not or repeat return then until while",built_in:"_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len "+"__gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert "+"collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstring"+"module next pairs pcall print rawequal rawget rawset require select setfenv"+"setmetatable tonumber tostring type unpack xpcall arg self"+"coroutine resume yield status wrap create running debug getupvalue "+"debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv "+"io lines write close flush open output type read stderr stdin input stdout popen tmpfile "+"math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan "+"os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall "+"string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower "+"table setn insert getn foreachi maxn foreach concat sort remove"},contains:COMMENTS.concat([{className:"function",beginKeywords:"function",end:"\\)",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),{className:"params",begin:"\\(",endsWithParent:true,contains:COMMENTS}].concat(COMMENTS)},hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:OPENING_LONG_BRACKET,end:CLOSING_LONG_BRACKET,contains:[LONG_BRACKETS],relevance:5}])}});hljs.registerLanguage("crmsh",function(hljs){var RESOURCES="primitive rsc_template";var COMMANDS="group clone ms master location colocation order fencing_topology "+"rsc_ticket acl_target acl_group user role "+"tag xml";var PROPERTY_SETS="property rsc_defaults op_defaults";var KEYWORDS="params meta operations op rule attributes utilization";var OPERATORS="read write deny defined not_defined in_range date spec in "+"ref reference attribute type xpath version and or lt gt tag "+"lte gte eq ne \\";var TYPES="number string";var LITERALS="Master Started Slave Stopped start promote demote stop monitor true false";return{aliases:["crm","pcmk"],case_insensitive:true,keywords:{keyword:KEYWORDS+" "+OPERATORS+" "+TYPES,literal:LITERALS},contains:[hljs.HASH_COMMENT_MODE,{beginKeywords:"node",starts:{end:"\\s*([\\w_-]+:)?",starts:{className:"title",end:"\\s*[\\$\\w_][\\w_-]*"}}},{beginKeywords:RESOURCES,starts:{className:"title",end:"\\s*[\\$\\w_][\\w_-]*",starts:{end:"\\s*@?[\\w_][\\w_\\.:-]*"}}},{begin:"\\b("+COMMANDS.split(" ").join("|")+")\\s+",keywords:COMMANDS,starts:{className:"title",end:"[\\$\\w_][\\w_-]*"}},{beginKeywords:PROPERTY_SETS,starts:{className:"title",end:"\\s*([\\w_-]+:)?"}},hljs.QUOTE_STRING_MODE,{className:"meta",begin:"(ocf|systemd|service|lsb):[\\w_:-]+",relevance:0},{className:"number",begin:"\\b\\d+(\\.\\d+)?(ms|s|h|m)?",relevance:0},{className:"literal",begin:"[-]?(infinity|inf)",relevance:0},{className:"attr",begin:/([A-Za-z\$_\#][\w_-]+)=/,relevance:0},{className:"tag",begin:"</?",end:"/?>",relevance:0}]}});hljs.registerLanguage("scheme",function(hljs){var SCHEME_IDENT_RE="[^\\(\\)\\[\\]\\{\\}\",'`;#|\\\\\\s]+";var SCHEME_SIMPLE_NUMBER_RE="(\\-|\\+)?\\d+([./]\\d+)?";var SCHEME_COMPLEX_NUMBER_RE=SCHEME_SIMPLE_NUMBER_RE+"[+\\-]"+SCHEME_SIMPLE_NUMBER_RE+"i";var BUILTINS={"builtin-name":"case-lambda call/cc class define-class exit-handler field import "+"inherit init-field interface let*-values let-values let/ec mixin "+"opt-lambda override protect provide public rename require "+"require-for-syntax syntax syntax-case syntax-error unit/sig unless "+"when with-syntax and begin call-with-current-continuation "+"call-with-input-file call-with-output-file case cond define "+"define-syntax delay do dynamic-wind else for-each if lambda let let* "+"let-syntax letrec letrec-syntax map or syntax-rules ' * + , ,@ - ... / "+"; < <= = => > >= ` abs acos angle append apply asin assoc assq assv atan "+"boolean? caar cadr call-with-input-file call-with-output-file "+"call-with-values car cdddar cddddr cdr ceiling char->integer "+"char-alphabetic? char-ci<=? char-ci<? char-ci=? char-ci>=? char-ci>? "+"char-downcase char-lower-case? char-numeric? char-ready? char-upcase "+"char-upper-case? char-whitespace? char<=? char<? char=? char>=? char>? "+"char? close-input-port close-output-port complex? cons cos "+"current-input-port current-output-port denominator display eof-object? "+"eq? equal? eqv? eval even? exact->inexact exact? exp expt floor "+"force gcd imag-part inexact->exact inexact? input-port? integer->char "+"integer? interaction-environment lcm length list list->string "+"list->vector list-ref list-tail list? load log magnitude make-polar "+"make-rectangular make-string make-vector max member memq memv min "+"modulo negative? newline not null-environment null? number->string "+"number? numerator odd? open-input-file open-output-file output-port? "+"pair? peek-char port? positive? procedure? quasiquote quote quotient "+"rational? rationalize read read-char real-part real? remainder reverse "+"round scheme-report-environment set! set-car! set-cdr! sin sqrt string "+"string->list string->number string->symbol string-append string-ci<=? "+"string-ci<? string-ci=? string-ci>=? string-ci>? string-copy "+"string-fill! string-length string-ref string-set! string<=? string<? "+"string=? string>=? string>? string? substring symbol->string symbol? "+"tan transcript-off transcript-on truncate values vector "+"vector->list vector-fill! vector-length vector-ref vector-set! "+"with-input-from-file with-output-to-file write write-char zero?"};var SHEBANG={className:"meta",begin:"^#!",end:"$"};var LITERAL={className:"literal",begin:"(#t|#f|#\\\\"+SCHEME_IDENT_RE+"|#\\\\.)"};var NUMBER={className:"number",variants:[{begin:SCHEME_SIMPLE_NUMBER_RE,relevance:0},{begin:SCHEME_COMPLEX_NUMBER_RE,relevance:0},{begin:"#b[0-1]+(/[0-1]+)?"},{begin:"#o[0-7]+(/[0-7]+)?"},{begin:"#x[0-9a-f]+(/[0-9a-f]+)?"}]};var STRING=hljs.QUOTE_STRING_MODE;var REGULAR_EXPRESSION={className:"regexp",begin:'#[pr]x"',end:'[^\\\\]"'};var COMMENT_MODES=[hljs.COMMENT(";","$",{relevance:0}),hljs.COMMENT("#\\|","\\|#")];var IDENT={begin:SCHEME_IDENT_RE,relevance:0};var QUOTED_IDENT={className:"symbol",begin:"'"+SCHEME_IDENT_RE};var BODY={endsWithParent:true,relevance:0};var QUOTED_LIST={variants:[{begin:/'/},{begin:"`"}],contains:[{begin:"\\(",end:"\\)",contains:["self",LITERAL,STRING,NUMBER,IDENT,QUOTED_IDENT]}]};var NAME={className:"name",begin:SCHEME_IDENT_RE,lexemes:SCHEME_IDENT_RE,keywords:BUILTINS};var LAMBDA={begin:/lambda/,endsWithParent:true,returnBegin:true,contains:[NAME,{begin:/\(/,end:/\)/,endsParent:true,contains:[IDENT]}]};var LIST={variants:[{begin:"\\(",end:"\\)"},{begin:"\\[",end:"\\]"}],contains:[LAMBDA,NAME,BODY]};BODY.contains=[LITERAL,NUMBER,STRING,IDENT,QUOTED_IDENT,QUOTED_LIST,LIST].concat(COMMENT_MODES);return{illegal:/\S/,contains:[SHEBANG,NUMBER,STRING,QUOTED_IDENT,QUOTED_LIST,LIST].concat(COMMENT_MODES)}});hljs.registerLanguage("bash",function(hljs){var VAR={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VAR,{className:"variable",begin:/\$\(/,end:/\)/,contains:[hljs.BACKSLASH_ESCAPE]}]};var ESCAPED_QUOTE={className:"",begin:/\\"/};var APOS_STRING={className:"string",begin:/'/,end:/'/};return{aliases:["sh","zsh"],lexemes:/\b-?[a-z\._]+\b/,keywords:{keyword:"if then else elif fi for while in do done case esac function",literal:"true false",built_in:"break cd continue eval exec exit export getopts hash pwd readonly return shift test times "+"trap umask unset "+"alias bind builtin caller command declare echo enable help let local logout mapfile printf "+"read readarray source type typeset ulimit unalias "+"set shopt "+"autoload bg bindkey bye cap chdir clone comparguments compcall compctl compdescribe compfiles "+"compgroups compquote comptags comptry compvalues dirs disable disown echotc echoti emulate "+"fc fg float functions getcap getln history integer jobs kill limit log noglob popd print "+"pushd pushln rehash sched setcap setopt stat suspend ttyctl unfunction unhash unlimit "+"unsetopt vared wait whence where which zcompile zformat zftp zle zmodload zparseopts zprof "+"zpty zregexparse zsocket zstyle ztcp",_:"-ne -eq -lt -gt -f -d -e -s -l -a"},contains:[{className:"meta",begin:/^#![^\n]+sh\s*$/,relevance:10},{className:"function",begin:/\w[\w\d_]*\s*\(\s*\)\s*\{/,returnBegin:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/\w[\w\d_]*/})],relevance:0},hljs.HASH_COMMENT_MODE,QUOTE_STRING,ESCAPED_QUOTE,APOS_STRING,VAR]}});hljs.registerLanguage("dockerfile",function(hljs){return{aliases:["docker"],case_insensitive:true,keywords:"from maintainer expose env arg user onbuild stopsignal",contains:[hljs.HASH_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,{beginKeywords:"run cmd entrypoint volume add copy workdir label healthcheck shell",starts:{end:/[^\\]$/,subLanguage:"bash"}}],illegal:"</"}});hljs.registerLanguage("properties",function(hljs){var WS0="[ \\t\\f]*";var WS1="[ \\t\\f]+";var DELIM="("+WS0+"[:=]"+WS0+"|"+WS1+")";var KEY_ALPHANUM="([^\\\\\\W:= \\t\\f\\n]|\\\\.)+";var KEY_OTHER="([^\\\\:= \\t\\f\\n]|\\\\.)+";var DELIM_AND_VALUE={end:DELIM,relevance:0,starts:{className:"string",end:/$/,relevance:0,contains:[{begin:"\\\\\\n"}]}};return{case_insensitive:true,illegal:/\S/,contains:[hljs.COMMENT("^\\s*[!#]","$"),{begin:KEY_ALPHANUM+DELIM,returnBegin:true,contains:[{className:"attr",begin:KEY_ALPHANUM,endsParent:true,relevance:0}],starts:DELIM_AND_VALUE},{begin:KEY_OTHER+DELIM,returnBegin:true,relevance:0,contains:[{className:"meta",begin:KEY_OTHER,endsParent:true,relevance:0}],starts:DELIM_AND_VALUE},{className:"attr",relevance:0,begin:KEY_OTHER+WS0+"$"}]}});hljs.registerLanguage("dns",function(hljs){return{aliases:["bind","zone"],keywords:{keyword:"IN A AAAA AFSDB APL CAA CDNSKEY CDS CERT CNAME DHCID DLV DNAME DNSKEY DS HIP IPSECKEY KEY KX "+"LOC MX NAPTR NS NSEC NSEC3 NSEC3PARAM PTR RRSIG RP SIG SOA SRV SSHFP TA TKEY TLSA TSIG TXT"},contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"meta",begin:/^\$(TTL|GENERATE|INCLUDE|ORIGIN)\b/},{className:"number",begin:"((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:)))\\b"},{className:"number",begin:"((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]).){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\\b"},hljs.inherit(hljs.NUMBER_MODE,{begin:/\b\d+[dhwm]?/})]}});hljs.registerLanguage("monkey",function(hljs){var NUMBER={className:"number",relevance:0,variants:[{begin:"[$][a-fA-F0-9]+"},hljs.NUMBER_MODE]};return{case_insensitive:true,keywords:{keyword:"public private property continue exit extern new try catch "+"eachin not abstract final select case default const local global field "+"end if then else elseif endif while wend repeat until forever for "+"to step next return module inline throw import",built_in:"DebugLog DebugStop Error Print ACos ACosr ASin ASinr ATan ATan2 ATan2r ATanr Abs Abs Ceil "+"Clamp Clamp Cos Cosr Exp Floor Log Max Max Min Min Pow Sgn Sgn Sin Sinr Sqrt Tan Tanr Seed PI HALFPI TWOPI",literal:"true false null and or shl shr mod"},illegal:/\/\*/,contains:[hljs.COMMENT("#rem","#end"),hljs.COMMENT("'","$",{relevance:0}),{className:"function",beginKeywords:"function method",end:"[(=:]|$",illegal:/\n/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"class",beginKeywords:"class interface",end:"$",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{className:"built_in",begin:"\\b(self|super)\\b"},{className:"meta",begin:"\\s*#",end:"$",keywords:{"meta-keyword":"if else elseif endif end then"}},{className:"meta",begin:"^\\s*strict\\b"},{beginKeywords:"alias",end:"=",contains:[hljs.UNDERSCORE_TITLE_MODE]},hljs.QUOTE_STRING_MODE,NUMBER]}});hljs.registerLanguage("golo",function(hljs){return{keywords:{keyword:"println readln print import module function local return let var "+"while for foreach times in case when match with break continue "+"augment augmentation each find filter reduce "+"if then else otherwise try catch finally raise throw orIfNull "+"DynamicObject|10 DynamicVariable struct Observable map set vector list array",literal:"true false null"},contains:[hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("livecodeserver",function(hljs){var VARIABLE={className:"variable",variants:[{begin:"\\b([gtps][A-Z]{1}[a-zA-Z0-9]*)(\\[.+\\])?(?:\\s*?)"},{begin:"\\$_[A-Z]+"}],relevance:0};var COMMENT_MODES=[hljs.C_BLOCK_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.COMMENT("--","$"),hljs.COMMENT("[^:]//","$")];var TITLE1=hljs.inherit(hljs.TITLE_MODE,{variants:[{begin:"\\b_*rig[A-Z]+[A-Za-z0-9_\\-]*"},{begin:"\\b_[a-z0-9\\-]+"}]});var TITLE2=hljs.inherit(hljs.TITLE_MODE,{begin:"\\b([A-Za-z0-9_\\-]+)\\b"});return{case_insensitive:false,keywords:{keyword:"$_COOKIE $_FILES $_GET $_GET_BINARY $_GET_RAW $_POST $_POST_BINARY $_POST_RAW $_SESSION $_SERVER "+"codepoint codepoints segment segments codeunit codeunits sentence sentences trueWord trueWords paragraph "+"after byte bytes english the until http forever descending using line real8 with seventh "+"for stdout finally element word words fourth before black ninth sixth characters chars stderr "+"uInt1 uInt1s uInt2 uInt2s stdin string lines relative rel any fifth items from middle mid "+"at else of catch then third it file milliseconds seconds second secs sec int1 int1s int4 "+"int4s internet int2 int2s normal text item last long detailed effective uInt4 uInt4s repeat "+"end repeat URL in try into switch to words https token binfile each tenth as ticks tick "+"system real4 by dateItems without char character ascending eighth whole dateTime numeric short "+"first ftp integer abbreviated abbr abbrev private case while if "+"div mod wrap and or bitAnd bitNot bitOr bitXor among not in a an within "+"contains ends with begins the keys of keys",literal:"SIX TEN FORMFEED NINE ZERO NONE SPACE FOUR FALSE COLON CRLF PI COMMA ENDOFFILE EOF EIGHT FIVE "+"QUOTE EMPTY ONE TRUE RETURN CR LINEFEED RIGHT BACKSLASH NULL SEVEN TAB THREE TWO "+"six ten formfeed nine zero none space four false colon crlf pi comma endoffile eof eight five "+"quote empty one true return cr linefeed right backslash null seven tab three two "+"RIVERSION RISTATE FILE_READ_MODE FILE_WRITE_MODE FILE_WRITE_MODE DIR_WRITE_MODE FILE_READ_UMASK "+"FILE_WRITE_UMASK DIR_READ_UMASK DIR_WRITE_UMASK",built_in:"put abs acos aliasReference annuity arrayDecode arrayEncode asin atan atan2 average avg avgDev base64Decode "+"base64Encode baseConvert binaryDecode binaryEncode byteOffset byteToNum cachedURL cachedURLs charToNum "+"cipherNames codepointOffset codepointProperty codepointToNum codeunitOffset commandNames compound compress "+"constantNames cos date dateFormat decompress difference directories "+"diskSpace DNSServers exp exp1 exp2 exp10 extents files flushEvents folders format functionNames geometricMean global "+"globals hasMemory harmonicMean hostAddress hostAddressToName hostName hostNameToAddress isNumber ISOToMac itemOffset "+"keys len length libURLErrorData libUrlFormData libURLftpCommand libURLLastHTTPHeaders libURLLastRHHeaders "+"libUrlMultipartFormAddPart libUrlMultipartFormData libURLVersion lineOffset ln ln1 localNames log log2 log10 "+"longFilePath lower macToISO matchChunk matchText matrixMultiply max md5Digest median merge messageAuthenticationCode messageDigest millisec "+"millisecs millisecond milliseconds min monthNames nativeCharToNum normalizeText num number numToByte numToChar "+"numToCodepoint numToNativeChar offset open openfiles openProcesses openProcessIDs openSockets "+"paragraphOffset paramCount param params peerAddress pendingMessages platform popStdDev populationStandardDeviation "+"populationVariance popVariance processID random randomBytes replaceText result revCreateXMLTree revCreateXMLTreeFromFile "+"revCurrentRecord revCurrentRecordIsFirst revCurrentRecordIsLast revDatabaseColumnCount revDatabaseColumnIsNull "+"revDatabaseColumnLengths revDatabaseColumnNames revDatabaseColumnNamed revDatabaseColumnNumbered "+"revDatabaseColumnTypes revDatabaseConnectResult revDatabaseCursors revDatabaseID revDatabaseTableNames "+"revDatabaseType revDataFromQuery revdb_closeCursor revdb_columnbynumber revdb_columncount revdb_columnisnull "+"revdb_columnlengths revdb_columnnames revdb_columntypes revdb_commit revdb_connect revdb_connections "+"revdb_connectionerr revdb_currentrecord revdb_cursorconnection revdb_cursorerr revdb_cursors revdb_dbtype "+"revdb_disconnect revdb_execute revdb_iseof revdb_isbof revdb_movefirst revdb_movelast revdb_movenext "+"revdb_moveprev revdb_query revdb_querylist revdb_recordcount revdb_rollback revdb_tablenames "+"revGetDatabaseDriverPath revNumberOfRecords revOpenDatabase revOpenDatabases revQueryDatabase "+"revQueryDatabaseBlob revQueryResult revQueryIsAtStart revQueryIsAtEnd revUnixFromMacPath revXMLAttribute "+"revXMLAttributes revXMLAttributeValues revXMLChildContents revXMLChildNames revXMLCreateTreeFromFileWithNamespaces "+"revXMLCreateTreeWithNamespaces revXMLDataFromXPathQuery revXMLEvaluateXPath revXMLFirstChild revXMLMatchingNode "+"revXMLNextSibling revXMLNodeContents revXMLNumberOfChildren revXMLParent revXMLPreviousSibling "+"revXMLRootNode revXMLRPC_CreateRequest revXMLRPC_Documents revXMLRPC_Error "+"revXMLRPC_GetHost revXMLRPC_GetMethod revXMLRPC_GetParam revXMLText revXMLRPC_Execute "+"revXMLRPC_GetParamCount revXMLRPC_GetParamNode revXMLRPC_GetParamType revXMLRPC_GetPath revXMLRPC_GetPort "+"revXMLRPC_GetProtocol revXMLRPC_GetRequest revXMLRPC_GetResponse revXMLRPC_GetSocket revXMLTree "+"revXMLTrees revXMLValidateDTD revZipDescribeItem revZipEnumerateItems revZipOpenArchives round sampVariance "+"sec secs seconds sentenceOffset sha1Digest shell shortFilePath sin specialFolderPath sqrt standardDeviation statRound "+"stdDev sum sysError systemVersion tan tempName textDecode textEncode tick ticks time to tokenOffset toLower toUpper "+"transpose truewordOffset trunc uniDecode uniEncode upper URLDecode URLEncode URLStatus uuid value variableNames "+"variance version waitDepth weekdayNames wordOffset xsltApplyStylesheet xsltApplyStylesheetFromFile xsltLoadStylesheet "+"xsltLoadStylesheetFromFile add breakpoint cancel clear local variable file word line folder directory URL close socket process "+"combine constant convert create new alias folder directory decrypt delete variable word line folder "+"directory URL dispatch divide do encrypt filter get include intersect kill libURLDownloadToFile "+"libURLFollowHttpRedirects libURLftpUpload libURLftpUploadFile libURLresetAll libUrlSetAuthCallback libURLSetDriver "+"libURLSetCustomHTTPHeaders libUrlSetExpect100 libURLSetFTPListCommand libURLSetFTPMode libURLSetFTPStopTime "+"libURLSetStatusCallback load extension loadedExtensions multiply socket prepare process post seek rel relative read from process rename "+"replace require resetAll resolve revAddXMLNode revAppendXML revCloseCursor revCloseDatabase revCommitDatabase "+"revCopyFile revCopyFolder revCopyXMLNode revDeleteFolder revDeleteXMLNode revDeleteAllXMLTrees "+"revDeleteXMLTree revExecuteSQL revGoURL revInsertXMLNode revMoveFolder revMoveToFirstRecord revMoveToLastRecord "+"revMoveToNextRecord revMoveToPreviousRecord revMoveToRecord revMoveXMLNode revPutIntoXMLNode revRollBackDatabase "+"revSetDatabaseDriverPath revSetXMLAttribute revXMLRPC_AddParam revXMLRPC_DeleteAllDocuments revXMLAddDTD "+"revXMLRPC_Free revXMLRPC_FreeAll revXMLRPC_DeleteDocument revXMLRPC_DeleteParam revXMLRPC_SetHost "+"revXMLRPC_SetMethod revXMLRPC_SetPort revXMLRPC_SetProtocol revXMLRPC_SetSocket revZipAddItemWithData "+"revZipAddItemWithFile revZipAddUncompressedItemWithData revZipAddUncompressedItemWithFile revZipCancel "+"revZipCloseArchive revZipDeleteItem revZipExtractItemToFile revZipExtractItemToVariable revZipSetProgressCallback "+"revZipRenameItem revZipReplaceItemWithData revZipReplaceItemWithFile revZipOpenArchive send set sort split start stop "+"subtract symmetric union unload vectorDotProduct wait write"},contains:[VARIABLE,{className:"keyword",begin:"\\bend\\sif\\b"},{className:"function",beginKeywords:"function",end:"$",contains:[VARIABLE,TITLE2,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1]},{className:"function",begin:"\\bend\\s+",end:"$",keywords:"end",contains:[TITLE2,TITLE1],relevance:0},{beginKeywords:"command on",end:"$",contains:[VARIABLE,TITLE2,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1]},{className:"meta",variants:[{begin:"<\\?(rev|lc|livecode)",relevance:10},{begin:"<\\?"},{begin:"\\?>"}]},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1].concat(COMMENT_MODES),illegal:";$|^\\[|^=|&|{"}});hljs.registerLanguage("haml",function(hljs){return{case_insensitive:true,contains:[{className:"meta",begin:"^!!!( (5|1\\.1|Strict|Frameset|Basic|Mobile|RDFa|XML\\b.*))?$",relevance:10},hljs.COMMENT("^\\s*(!=#|=#|-#|/).*$",false,{relevance:0}),{begin:"^\\s*(-|=|!=)(?!#)",starts:{end:"\\n",subLanguage:"ruby"}},{className:"tag",begin:"^\\s*%",contains:[{className:"selector-tag",begin:"\\w+"},{className:"selector-id",begin:"#[\\w-]+"},{className:"selector-class",begin:"\\.[\\w-]+"},{begin:"{\\s*",end:"\\s*}",contains:[{begin:":\\w+\\s*=>",end:",\\s+",returnBegin:true,endsWithParent:true,contains:[{className:"attr",begin:":\\w+"},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\w+",relevance:0}]}]},{begin:"\\(\\s*",end:"\\s*\\)",excludeEnd:true,contains:[{begin:"\\w+\\s*=",end:"\\s+",returnBegin:true,endsWithParent:true,contains:[{className:"attr",begin:"\\w+",relevance:0},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\w+",relevance:0}]}]}]},{begin:"^\\s*[=~]\\s*"},{begin:"#{",starts:{end:"}",subLanguage:"ruby"}}]}});hljs.registerLanguage("gcode",function(hljs){var GCODE_IDENT_RE="[A-Z_][A-Z0-9_.]*";var GCODE_CLOSE_RE="\\%";var GCODE_KEYWORDS="IF DO WHILE ENDWHILE CALL ENDIF SUB ENDSUB GOTO REPEAT ENDREPEAT "+"EQ LT GT NE GE LE OR XOR";var GCODE_START={className:"meta",begin:"([O])([0-9]+)"};var GCODE_CODE=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(/\(/,/\)/),hljs.inherit(hljs.C_NUMBER_MODE,{begin:"([-+]?([0-9]*\\.?[0-9]+\\.?))|"+hljs.C_NUMBER_RE}),hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"name",begin:"([G])([0-9]+\\.?[0-9]?)"},{className:"name",begin:"([M])([0-9]+\\.?[0-9]?)"},{className:"attr",begin:"(VC|VS|#)",end:"(\\d+)"},{className:"attr",begin:"(VZOFX|VZOFY|VZOFZ)"},{className:"built_in",begin:"(ATAN|ABS|ACOS|ASIN|SIN|COS|EXP|FIX|FUP|ROUND|LN|TAN)(\\[)",end:"([-+]?([0-9]*\\.?[0-9]+\\.?))(\\])"},{className:"symbol",variants:[{begin:"N",end:"\\d+",illegal:"\\W"}]}];return{aliases:["nc"],case_insensitive:true,lexemes:GCODE_IDENT_RE,keywords:GCODE_KEYWORDS,contains:[{className:"meta",begin:GCODE_CLOSE_RE},GCODE_START].concat(GCODE_CODE)}});hljs.registerLanguage("erlang",function(hljs){var BASIC_ATOM_RE="[a-z'][a-zA-Z0-9_']*";var FUNCTION_NAME_RE="("+BASIC_ATOM_RE+":"+BASIC_ATOM_RE+"|"+BASIC_ATOM_RE+")";var ERLANG_RESERVED={keyword:"after and andalso|10 band begin bnot bor bsl bzr bxor case catch cond div end fun if "+"let not of orelse|10 query receive rem try when xor",literal:"false true"};var COMMENT=hljs.COMMENT("%","$");var NUMBER={className:"number",begin:"\\b(\\d+#[a-fA-F0-9]+|\\d+(\\.\\d+)?([eE][-+]?\\d+)?)",relevance:0};var NAMED_FUN={begin:"fun\\s+"+BASIC_ATOM_RE+"/\\d+"};var FUNCTION_CALL={begin:FUNCTION_NAME_RE+"\\(",end:"\\)",returnBegin:true,relevance:0,contains:[{begin:FUNCTION_NAME_RE,relevance:0},{begin:"\\(",end:"\\)",endsWithParent:true,returnEnd:true,relevance:0}]};var TUPLE={begin:"{",end:"}",relevance:0};var VAR1={begin:"\\b_([A-Z][A-Za-z0-9_]*)?",relevance:0};var VAR2={begin:"[A-Z][a-zA-Z0-9_]*",relevance:0};var RECORD_ACCESS={begin:"#"+hljs.UNDERSCORE_IDENT_RE,relevance:0,returnBegin:true,contains:[{begin:"#"+hljs.UNDERSCORE_IDENT_RE,relevance:0},{begin:"{",end:"}",relevance:0}]};var BLOCK_STATEMENTS={beginKeywords:"fun receive if try case",end:"end",keywords:ERLANG_RESERVED};BLOCK_STATEMENTS.contains=[COMMENT,NAMED_FUN,hljs.inherit(hljs.APOS_STRING_MODE,{className:""}),BLOCK_STATEMENTS,FUNCTION_CALL,hljs.QUOTE_STRING_MODE,NUMBER,TUPLE,VAR1,VAR2,RECORD_ACCESS];var BASIC_MODES=[COMMENT,NAMED_FUN,BLOCK_STATEMENTS,FUNCTION_CALL,hljs.QUOTE_STRING_MODE,NUMBER,TUPLE,VAR1,VAR2,RECORD_ACCESS];FUNCTION_CALL.contains[1].contains=BASIC_MODES;TUPLE.contains=BASIC_MODES;RECORD_ACCESS.contains[1].contains=BASIC_MODES;var PARAMS={className:"params",begin:"\\(",end:"\\)",contains:BASIC_MODES};return{aliases:["erl"],keywords:ERLANG_RESERVED,illegal:"(</|\\*=|\\+=|-=|/\\*|\\*/|\\(\\*|\\*\\))",contains:[{className:"function",begin:"^"+BASIC_ATOM_RE+"\\s*\\(",end:"->",returnBegin:true,illegal:"\\(|#|//|/\\*|\\\\|:|;",contains:[PARAMS,hljs.inherit(hljs.TITLE_MODE,{begin:BASIC_ATOM_RE})],starts:{end:";|\\.",keywords:ERLANG_RESERVED,contains:BASIC_MODES}},COMMENT,{begin:"^-",end:"\\.",relevance:0,excludeEnd:true,returnBegin:true,lexemes:"-"+hljs.IDENT_RE,keywords:"-module -record -undef -export -ifdef -ifndef -author -copyright -doc -vsn "+"-import -include -include_lib -compile -define -else -endif -file -behaviour "+"-behavior -spec",contains:[PARAMS]},NUMBER,hljs.QUOTE_STRING_MODE,RECORD_ACCESS,VAR1,VAR2,TUPLE,{begin:/\.$/}]}});hljs.registerLanguage("dust",function(hljs){var EXPRESSION_KEYWORDS="if eq ne lt lte gt gte select default math sep";return{aliases:["dst"],case_insensitive:true,subLanguage:"xml",contains:[{className:"template-tag",begin:/\{[#\/]/,end:/\}/,illegal:/;/,contains:[{className:"name",begin:/[a-zA-Z\.-]+/,starts:{endsWithParent:true,relevance:0,contains:[hljs.QUOTE_STRING_MODE]}}]},{className:"template-variable",begin:/\{/,end:/\}/,illegal:/;/,keywords:EXPRESSION_KEYWORDS}]}});hljs.registerLanguage("mizar",function(hljs){return{keywords:"environ vocabularies notations constructors definitions "+"registrations theorems schemes requirements begin end definition "+"registration cluster existence pred func defpred deffunc theorem "+"proof let take assume then thus hence ex for st holds consider "+"reconsider such that and in provided of as from be being by means "+"equals implies iff redefine define now not or attr is mode "+"suppose per cases set thesis contradiction scheme reserve struct "+"correctness compatibility coherence symmetry assymetry "+"reflexivity irreflexivity connectedness uniqueness commutativity "+"idempotence involutiveness projectivity",contains:[hljs.COMMENT("::","$")]}});hljs.registerLanguage("arcade",function(hljs){var IDENT_RE="[A-Za-z_][0-9A-Za-z_]*";var KEYWORDS={keyword:"if for while var new function do return void else break",literal:"BackSlash DoubleQuote false ForwardSlash Infinity NaN NewLine null PI SingleQuote Tab TextFormatting true undefined",built_in:"Abs Acos Angle Attachments Area AreaGeodetic Asin Atan Atan2 Average Bearing Boolean Buffer BufferGeodetic "+"Ceil Centroid Clip Console Constrain Contains Cos Count Crosses Cut Date DateAdd "+"DateDiff Day Decode DefaultValue Dictionary Difference Disjoint Distance DistanceGeodetic Distinct "+"DomainCode DomainName Equals Exp Extent Feature FeatureSet FeatureSetByAssociation FeatureSetById FeatureSetByPortalItem "+"FeatureSetByRelationshipName FeatureSetByTitle FeatureSetByUrl Filter First Floor Geometry GroupBy Guid HasKey Hour IIf IndexOf "+"Intersection Intersects IsEmpty IsNan IsSelfIntersecting Length LengthGeodetic Log Max Mean Millisecond Min Minute Month "+"MultiPartToSinglePart Multipoint NextSequenceValue Now Number OrderBy Overlaps Point Polygon "+"Polyline Portal Pow Random Relate Reverse RingIsClockWise Round Second SetGeometry Sin Sort Sqrt Stdev Sum "+"SymmetricDifference Tan Text Timestamp Today ToLocal Top Touches ToUTC TrackCurrentTime "+"TrackGeometryWindow TrackIndex TrackStartTime TrackWindow TypeOf Union UrlEncode Variance "+"Weekday When Within Year "};var EXPRESSIONS;var SYMBOL={className:"symbol",begin:"\\$[datastore|feature|layer|map|measure|sourcefeature|sourcelayer|targetfeature|targetlayer|value|view]+"};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)"},{begin:"\\b(0[oO][0-7]+)"},{begin:hljs.C_NUMBER_RE}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];var PARAMS_CONTAINS=SUBST.contains.concat([hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE]);return{aliases:["arcade"],keywords:KEYWORDS,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,SYMBOL,NUMBER,{begin:/[{,]\s*/,relevance:0,contains:[{begin:IDENT_RE+"\\s*:",returnBegin:true,relevance:0,contains:[{className:"attr",begin:IDENT_RE,relevance:0}]}]},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(return)\\b)\\s*",keywords:"return",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:PARAMS_CONTAINS}]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:PARAMS_CONTAINS}],illegal:/\[|%/},{begin:/\$[(.]/}],illegal:/#(?!!)/}});hljs.registerLanguage("php",function(hljs){var VARIABLE={begin:"\\$+[a-zA-Z_-ÿ][a-zA-Z0-9_-ÿ]*"};var PREPROCESSOR={className:"meta",begin:/<\?(php)?|\?>/};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,PREPROCESSOR],variants:[{begin:'b"',end:'"'},{begin:"b'",end:"'"},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]};var NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]};return{aliases:["php","php3","php4","php5","php6","php7"],case_insensitive:true,keywords:"and include_once list abstract global private echo interface as static endswitch "+"array null if endwhile or const for endforeach self var while isset public "+"protected exit foreach throw elseif include __FILE__ empty require_once do xor "+"return parent clone use __CLASS__ __LINE__ else break print eval new "+"catch __METHOD__ case exception default die require __FUNCTION__ "+"enddeclare final try switch continue endfor endif declare unset true false "+"trait goto instanceof insteadof __DIR__ __NAMESPACE__ "+"yield finally",contains:[hljs.HASH_COMMENT_MODE,hljs.COMMENT("//","$",{contains:[PREPROCESSOR]}),hljs.COMMENT("/\\*","\\*/",{contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.COMMENT("__halt_compiler.+?;",false,{endsWithParent:true,keywords:"__halt_compiler",lexemes:hljs.UNDERSCORE_IDENT_RE}),{className:"string",begin:/<<<['"]?\w+['"]?$/,end:/^\w+;?$/,contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",variants:[{begin:/\$\w+/},{begin:/\{\$/,end:/\}/}]}]},PREPROCESSOR,{className:"keyword",begin:/\$this\b/},VARIABLE,{begin:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{className:"function",beginKeywords:"function",end:/[;{]/,excludeEnd:true,illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:["self",VARIABLE,hljs.C_BLOCK_COMMENT_MODE,STRING,NUMBER]}]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:/[:\(\$"]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"namespace",end:";",illegal:/[\.']/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"use",end:";",contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"=>"},STRING,NUMBER]}});hljs.registerLanguage("haxe",function(hljs){var IDENT_RE="[a-zA-Z_$][a-zA-Z0-9_$]*";var IDENT_FUNC_RETURN_TYPE_RE="([*]|[a-zA-Z_$][a-zA-Z0-9_$]*)";var HAXE_BASIC_TYPES="Int Float String Bool Dynamic Void Array ";return{aliases:["hx"],keywords:{keyword:"break case cast catch continue default do dynamic else enum extern "+"for function here if import in inline never new override package private get set "+"public return static super switch this throw trace try typedef untyped using var while "+HAXE_BASIC_TYPES,built_in:"trace this",literal:"true false null _"},contains:[{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",begin:"\\$\\{",end:"\\}"},{className:"subst",begin:"\\$",end:"\\W}"}]},hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"@:",end:"$"},{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elseif end error"}},{className:"type",begin:":[ \t]*",end:"[^A-Za-z0-9_ \t\\->]",excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:":[ \t]*",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"type",begin:"new *",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"class",beginKeywords:"enum",end:"\\{",contains:[hljs.TITLE_MODE]},{className:"class",beginKeywords:"abstract",end:"[\\{$]",contains:[{className:"type",begin:"\\(",end:"\\)",excludeBegin:true,excludeEnd:true},{className:"type",begin:"from +",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"type",begin:"to +",end:"\\W",excludeBegin:true,excludeEnd:true},hljs.TITLE_MODE],keywords:{keyword:"abstract from to"}},{className:"class",begin:"\\b(class|interface) +",end:"[\\{$]",excludeEnd:true,keywords:"class interface",contains:[{className:"keyword",begin:"\\b(extends|implements) +",keywords:"extends implements",contains:[{className:"type",begin:hljs.IDENT_RE,relevance:0}]},hljs.TITLE_MODE]},{className:"function",beginKeywords:"function",end:"\\(",excludeEnd:true,illegal:"\\S",contains:[hljs.TITLE_MODE]}],illegal:/<\//}});hljs.registerLanguage("autohotkey",function(hljs){var BACKTICK_ESCAPE={begin:"`[\\s\\S]"};return{case_insensitive:true,aliases:["ahk"],keywords:{keyword:"Break Continue Critical Exit ExitApp Gosub Goto New OnExit Pause return SetBatchLines SetTimer Suspend Thread Throw Until ahk_id ahk_class ahk_pid ahk_exe ahk_group",literal:"true false NOT AND OR",built_in:"ComSpec Clipboard ClipboardAll ErrorLevel"},contains:[BACKTICK_ESCAPE,hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[BACKTICK_ESCAPE]}),hljs.COMMENT(";","$",{relevance:0}),hljs.C_BLOCK_COMMENT_MODE,{className:"number",begin:hljs.NUMBER_RE,relevance:0},{className:"variable",begin:"%[a-zA-Z0-9#_$@]+%"},{className:"built_in",begin:"^\\s*\\w+\\s*(,|%)"},{className:"title",variants:[{begin:'^[^\\n";]+::(?!=)'},{begin:'^[^\\n";]+:(?!=)',relevance:0}]},{className:"meta",begin:"^\\s*#\\w+",end:"$",relevance:0},{className:"built_in",begin:"A_[a-zA-Z0-9]+"},{begin:",\\s*,"}]}});hljs.registerLanguage("protobuf",function(hljs){return{keywords:{keyword:"package import option optional required repeated group oneof",built_in:"double float int32 int64 uint32 uint64 sint32 sint64 "+"fixed32 fixed64 sfixed32 sfixed64 bool string bytes",literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,{className:"class",beginKeywords:"message enum service",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{className:"function",beginKeywords:"rpc",end:/;/,excludeEnd:true,keywords:"rpc returns"},{begin:/^\s*[A-Z_]+/,end:/\s*=/,excludeEnd:true}]}});hljs.registerLanguage("capnproto",function(hljs){return{aliases:["capnp"],keywords:{keyword:"struct enum interface union group import using const annotation extends in of on as with from fixed",built_in:"Void Bool Int8 Int16 Int32 Int64 UInt8 UInt16 UInt32 UInt64 Float32 Float64 "+"Text Data AnyPointer AnyStruct Capability List",literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.HASH_COMMENT_MODE,{className:"meta",begin:/@0x[\w\d]{16};/,illegal:/\n/},{className:"symbol",begin:/@\d+\b/},{className:"class",beginKeywords:"struct enum",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{className:"class",beginKeywords:"interface",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]}]}});hljs.registerLanguage("scala",function(hljs){var ANNOTATION={className:"meta",begin:"@[A-Za-z]+"};var SUBST={className:"subst",variants:[{begin:"\\$[A-Za-z0-9_]+"},{begin:"\\${",end:"}"}]};var STRING={className:"string",variants:[{begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"""',end:'"""',relevance:10},{begin:'[a-z]+"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST]},{className:"string",begin:'[a-z]+"""',end:'"""',contains:[SUBST],relevance:10}]};var SYMBOL={className:"symbol",begin:"'\\w[\\w\\d_]*(?!')"};var TYPE={className:"type",begin:"\\b[A-Z][A-Za-z0-9_]*",relevance:0};var NAME={className:"title",begin:/[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/,relevance:0};var CLASS={className:"class",beginKeywords:"class object trait type",end:/[:={\[\n;]/,excludeEnd:true,contains:[{beginKeywords:"extends with",relevance:10},{begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[TYPE]},{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[TYPE]},NAME]};var METHOD={className:"function",beginKeywords:"def",end:/[:={\[(\n;]/,excludeEnd:true,contains:[NAME]};return{keywords:{literal:"true false null",keyword:"type yield lazy override def with val var sealed abstract private trait object if forSome for while throw finally protected extends import final return else break new catch super class case package default try this match continue throws implicit"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING,SYMBOL,TYPE,METHOD,CLASS,hljs.C_NUMBER_MODE,ANNOTATION]}});hljs.registerLanguage("django",function(hljs){var FILTER={begin:/\|[A-Za-z]+:?/,keywords:{name:"truncatewords removetags linebreaksbr yesno get_digit timesince random striptags "+"filesizeformat escape linebreaks length_is ljust rjust cut urlize fix_ampersands "+"title floatformat capfirst pprint divisibleby add make_list unordered_list urlencode "+"timeuntil urlizetrunc wordcount stringformat linenumbers slice date dictsort "+"dictsortreversed default_if_none pluralize lower join center default "+"truncatewords_html upper length phone2numeric wordwrap time addslashes slugify first "+"escapejs force_escape iriencode last safe safeseq truncatechars localize unlocalize "+"localtime utc timezone"},contains:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE]};return{aliases:["jinja"],case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT(/\{%\s*comment\s*%}/,/\{%\s*endcomment\s*%}/),hljs.COMMENT(/\{#/,/#}/),{className:"template-tag",begin:/\{%/,end:/%}/,contains:[{className:"name",begin:/\w+/,keywords:{name:"comment endcomment load templatetag ifchanged endifchanged if endif firstof for "+"endfor ifnotequal endifnotequal widthratio extends include spaceless "+"endspaceless regroup ifequal endifequal ssi now with cycle url filter "+"endfilter debug block endblock else autoescape endautoescape csrf_token empty elif "+"endwith static trans blocktrans endblocktrans get_static_prefix get_media_prefix "+"plural get_current_language language get_available_languages "+"get_current_language_bidi get_language_info get_language_info_list localize "+"endlocalize localtime endlocaltime timezone endtimezone get_current_timezone "+"verbatim"},starts:{endsWithParent:true,keywords:"in by as",contains:[FILTER],relevance:0}}]},{className:"template-variable",begin:/\{\{/,end:/}}/,contains:[FILTER]}]}});hljs.registerLanguage("shell",function(hljs){return{aliases:["console"],contains:[{className:"meta",begin:"^\\s{0,3}[/\\w\\d\\[\\]()@-]*[>%$#]",starts:{end:"$",subLanguage:"bash"}}]}});hljs.registerLanguage("powershell",function(hljs){var TYPES=["string","char","byte","int","long","bool","decimal","single","double","DateTime","xml","array","hashtable","void"];var VALID_VERBS="Add|Clear|Close|Copy|Enter|Exit|Find|Format|Get|Hide|Join|Lock|"+"Move|New|Open|Optimize|Pop|Push|Redo|Remove|Rename|Reset|Resize|"+"Search|Select|Set|Show|Skip|Split|Step|Switch|Undo|Unlock|"+"Watch|Backup|Checkpoint|Compare|Compress|Convert|ConvertFrom|"+"ConvertTo|Dismount|Edit|Expand|Export|Group|Import|Initialize|"+"Limit|Merge|New|Out|Publish|Restore|Save|Sync|Unpublish|Update|"+"Approve|Assert|Complete|Confirm|Deny|Disable|Enable|Install|Invoke|Register|"+"Request|Restart|Resume|Start|Stop|Submit|Suspend|Uninstall|"+"Unregister|Wait|Debug|Measure|Ping|Repair|Resolve|Test|Trace|Connect|"+"Disconnect|Read|Receive|Send|Write|Block|Grant|Protect|Revoke|Unblock|"+"Unprotect|Use|ForEach|Sort|Tee|Where";var COMPARISON_OPERATORS="-and|-as|-band|-bnot|-bor|-bxor|-casesensitive|-ccontains|-ceq|-cge|-cgt|"+"-cle|-clike|-clt|-cmatch|-cne|-cnotcontains|-cnotlike|-cnotmatch|-contains|"+"-creplace|-csplit|-eq|-exact|-f|-file|-ge|-gt|-icontains|-ieq|-ige|-igt|"+"-ile|-ilike|-ilt|-imatch|-in|-ine|-inotcontains|-inotlike|-inotmatch|"+"-ireplace|-is|-isnot|-isplit|-join|-le|-like|-lt|-match|-ne|-not|"+"-notcontains|-notin|-notlike|-notmatch|-or|-regex|-replace|-shl|-shr|"+"-split|-wildcard|-xor";var KEYWORDS={keyword:"if else foreach return do while until elseif begin for trap data dynamicparam "+"end break throw param continue finally in switch exit filter try process catch "+"hidden static parameter"};var TITLE_NAME_RE=/\w[\w\d]*((-)[\w\d]+)*/;var BACKTICK_ESCAPE={begin:"`[\\s\\S]",relevance:0};var VAR={className:"variable",variants:[{begin:/\$\B/},{className:"keyword",begin:/\$this/},{begin:/\$[\w\d][\w\d_:]*/}]};var LITERAL={className:"literal",begin:/\$(null|true|false)\b/};var QUOTE_STRING={className:"string",variants:[{begin:/"/,end:/"/},{begin:/@"/,end:/^"@/}],contains:[BACKTICK_ESCAPE,VAR,{className:"variable",begin:/\$[A-z]/,end:/[^A-z]/}]};var APOS_STRING={className:"string",variants:[{begin:/'/,end:/'/},{begin:/@'/,end:/^'@/}]};var PS_HELPTAGS={className:"doctag",variants:[{begin:/\.(synopsis|description|example|inputs|outputs|notes|link|component|role|functionality)/},{begin:/\.(parameter|forwardhelptargetname|forwardhelpcategory|remotehelprunspace|externalhelp)\s+\S+/}]};var PS_COMMENT=hljs.inherit(hljs.COMMENT(null,null),{variants:[{begin:/#/,end:/$/},{begin:/<#/,end:/#>/}],contains:[PS_HELPTAGS]});var CMDLETS={className:"built_in",variants:[{begin:"(".concat(VALID_VERBS,")+(-)[\\w\\d]+")}]};var PS_CLASS={className:"class",beginKeywords:"class enum",end:/\s*[{]/,excludeEnd:true,relevance:0,contains:[hljs.TITLE_MODE]};var PS_FUNCTION={className:"function",begin:/function\s+/,end:/\s*\{|$/,excludeEnd:true,returnBegin:true,relevance:0,contains:[{begin:"function",relevance:0,className:"keyword"},{className:"title",begin:TITLE_NAME_RE,relevance:0},{begin:/\(/,end:/\)/,className:"params",relevance:0,contains:[VAR]}]};var PS_USING={begin:/using\s/,end:/$/,returnBegin:true,contains:[QUOTE_STRING,APOS_STRING,{className:"keyword",begin:/(using|assembly|command|module|namespace|type)/}]};var PS_ARGUMENTS={variants:[{className:"operator",begin:"(".concat(COMPARISON_OPERATORS,")\\b")},{className:"literal",begin:/(-)[\w\d]+/,relevance:0}]};var STATIC_MEMBER={className:"selector-tag",begin:/::\w+\b/,end:/$/,returnBegin:true,contains:[{className:"attribute",begin:/\w+/,endsParent:true}]};var HASH_SIGNS={className:"selector-tag",begin:/\@\B/,relevance:0};var PS_NEW_OBJECT_TYPE={className:"built_in",begin:/New-Object\s+\w/,end:/$/,returnBegin:true,contains:[{begin:/New-Object\s+/,relevance:0},{className:"meta",begin:/([\w\.])+/,endsParent:true}]};var PS_METHODS={className:"function",begin:/\[.*\]\s*[\w]+[ ]??\(/,end:/$/,returnBegin:true,relevance:0,contains:[{className:"keyword",begin:"(".concat(KEYWORDS.keyword.toString().replace(/\s/g,"|"),")\\b"),endsParent:true,relevance:0},hljs.inherit(hljs.TITLE_MODE,{endsParent:true})]};var GENTLEMANS_SET=[PS_METHODS,PS_COMMENT,BACKTICK_ESCAPE,hljs.NUMBER_MODE,QUOTE_STRING,APOS_STRING,CMDLETS,VAR,LITERAL,HASH_SIGNS];var PS_TYPE={begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[].concat("self",GENTLEMANS_SET,{begin:"("+TYPES.join("|")+")",className:"built_in",relevance:0},{className:"type",begin:/[\.\w\d]+/,relevance:0})};PS_METHODS.contains.unshift(PS_TYPE);return{aliases:["ps","ps1"],lexemes:/-?[A-z\.\-]+/,case_insensitive:true,keywords:KEYWORDS,contains:GENTLEMANS_SET.concat(PS_CLASS,PS_FUNCTION,PS_USING,PS_ARGUMENTS,PS_TYPE)}});hljs.registerLanguage("swift",function(hljs){var SWIFT_KEYWORDS={keyword:"#available #colorLiteral #column #else #elseif #endif #file "+"#fileLiteral #function #if #imageLiteral #line #selector #sourceLocation "+"_ __COLUMN__ __FILE__ __FUNCTION__ __LINE__ Any as as! as? associatedtype "+"associativity break case catch class continue convenience default defer deinit didSet do "+"dynamic dynamicType else enum extension fallthrough false fileprivate final for func "+"get guard if import in indirect infix init inout internal is lazy left let "+"mutating nil none nonmutating open operator optional override postfix precedence "+"prefix private protocol Protocol public repeat required rethrows return "+"right self Self set static struct subscript super switch throw throws true "+"try try! try? Type typealias unowned var weak where while willSet",literal:"true false nil",built_in:"abs advance alignof alignofValue anyGenerator assert assertionFailure "+"bridgeFromObjectiveC bridgeFromObjectiveCUnconditional bridgeToObjectiveC "+"bridgeToObjectiveCUnconditional c contains count countElements countLeadingZeros "+"debugPrint debugPrintln distance dropFirst dropLast dump encodeBitsAsWords "+"enumerate equal fatalError filter find getBridgedObjectiveCType getVaList "+"indices insertionSort isBridgedToObjectiveC isBridgedVerbatimToObjectiveC "+"isUniquelyReferenced isUniquelyReferencedNonObjC join lazy lexicographicalCompare "+"map max maxElement min minElement numericCast overlaps partition posix "+"precondition preconditionFailure print println quickSort readLine reduce reflect "+"reinterpretCast reverse roundUpToAlignment sizeof sizeofValue sort split "+"startsWith stride strideof strideofValue swap toString transcode "+"underestimateCount unsafeAddressOf unsafeBitCast unsafeDowncast unsafeUnwrap "+"unsafeReflect withExtendedLifetime withObjectAtPlusZero withUnsafePointer "+"withUnsafePointerToObject withUnsafeMutablePointer withUnsafeMutablePointers "+"withUnsafePointer withUnsafePointers withVaList zip"};var TYPE={className:"type",begin:"\\b[A-Z][\\wÀ-ʸ']*",relevance:0};var OPTIONAL_USING_TYPE={className:"type",begin:"\\b[A-Z][\\wÀ-ʸ']*[!?]"};var BLOCK_COMMENT=hljs.COMMENT("/\\*","\\*/",{contains:["self"]});var SUBST={className:"subst",begin:/\\\(/,end:"\\)",keywords:SWIFT_KEYWORDS,contains:[]};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"""/,end:/"""/},{begin:/"/,end:/"/}]};var NUMBERS={className:"number",begin:"\\b([\\d_]+(\\.[\\deE_]+)?|0x[a-fA-F0-9_]+(\\.[a-fA-F0-9p_]+)?|0b[01_]+|0o[0-7_]+)\\b",relevance:0};SUBST.contains=[NUMBERS];return{keywords:SWIFT_KEYWORDS,contains:[STRING,hljs.C_LINE_COMMENT_MODE,BLOCK_COMMENT,OPTIONAL_USING_TYPE,TYPE,NUMBERS,{className:"function",beginKeywords:"func",end:"{",excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][0-9A-Za-z$_]*/}),{begin:/</,end:/>/},{className:"params",begin:/\(/,end:/\)/,endsParent:true,keywords:SWIFT_KEYWORDS,contains:["self",NUMBERS,STRING,hljs.C_BLOCK_COMMENT_MODE,{begin:":"}],illegal:/["']/}],illegal:/\[|%/},{className:"class",beginKeywords:"struct protocol class extension enum",keywords:SWIFT_KEYWORDS,end:"\\{",excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/})]},{className:"meta",begin:"(@discardableResult|@warn_unused_result|@exported|@lazy|@noescape|"+"@NSCopying|@NSManaged|@objc|@objcMembers|@convention|@required|"+"@noreturn|@IBAction|@IBDesignable|@IBInspectable|@IBOutlet|"+"@infix|@prefix|@postfix|@autoclosure|@testable|@available|"+"@nonobjc|@NSApplicationMain|@UIApplicationMain|@dynamicMemberLookup|"+"@propertyWrapper)"},{beginKeywords:"import",end:/$/,contains:[hljs.C_LINE_COMMENT_MODE,BLOCK_COMMENT]}]}});hljs.registerLanguage("sas",function(hljs){var SAS_KEYWORDS=""+"do if then else end until while "+""+"abort array attrib by call cards cards4 catname continue "+"datalines datalines4 delete delim delimiter display dm drop "+"endsas error file filename footnote format goto in infile "+"informat input keep label leave length libname link list "+"lostcard merge missing modify options output out page put "+"redirect remove rename replace retain return select set skip "+"startsas stop title update waitsas where window x systask "+""+"add and alter as cascade check create delete describe "+"distinct drop foreign from group having index insert into in "+"key like message modify msgtype not null on or order primary "+"references reset restrict select set table unique update "+"validate view where";var SAS_FUN=""+"abs|addr|airy|arcos|arsin|atan|attrc|attrn|band|"+"betainv|blshift|bnot|bor|brshift|bxor|byte|cdf|ceil|"+"cexist|cinv|close|cnonct|collate|compbl|compound|"+"compress|cos|cosh|css|curobs|cv|daccdb|daccdbsl|"+"daccsl|daccsyd|dacctab|dairy|date|datejul|datepart|"+"datetime|day|dclose|depdb|depdbsl|depdbsl|depsl|"+"depsl|depsyd|depsyd|deptab|deptab|dequote|dhms|dif|"+"digamma|dim|dinfo|dnum|dopen|doptname|doptnum|dread|"+"dropnote|dsname|erf|erfc|exist|exp|fappend|fclose|"+"fcol|fdelete|fetch|fetchobs|fexist|fget|fileexist|"+"filename|fileref|finfo|finv|fipname|fipnamel|"+"fipstate|floor|fnonct|fnote|fopen|foptname|foptnum|"+"fpoint|fpos|fput|fread|frewind|frlen|fsep|fuzz|"+"fwrite|gaminv|gamma|getoption|getvarc|getvarn|hbound|"+"hms|hosthelp|hour|ibessel|index|indexc|indexw|input|"+"inputc|inputn|int|intck|intnx|intrr|irr|jbessel|"+"juldate|kurtosis|lag|lbound|left|length|lgamma|"+"libname|libref|log|log10|log2|logpdf|logpmf|logsdf|"+"lowcase|max|mdy|mean|min|minute|mod|month|mopen|"+"mort|n|netpv|nmiss|normal|note|npv|open|ordinal|"+"pathname|pdf|peek|peekc|pmf|point|poisson|poke|"+"probbeta|probbnml|probchi|probf|probgam|probhypr|"+"probit|probnegb|probnorm|probt|put|putc|putn|qtr|"+"quote|ranbin|rancau|ranexp|rangam|range|rank|rannor|"+"ranpoi|rantbl|rantri|ranuni|repeat|resolve|reverse|"+"rewind|right|round|saving|scan|sdf|second|sign|"+"sin|sinh|skewness|soundex|spedis|sqrt|std|stderr|"+"stfips|stname|stnamel|substr|sum|symget|sysget|"+"sysmsg|sysprod|sysrc|system|tan|tanh|time|timepart|"+"tinv|tnonct|today|translate|tranwrd|trigamma|"+"trim|trimn|trunc|uniform|upcase|uss|var|varfmt|"+"varinfmt|varlabel|varlen|varname|varnum|varray|"+"varrayx|vartype|verify|vformat|vformatd|vformatdx|"+"vformatn|vformatnx|vformatw|vformatwx|vformatx|"+"vinarray|vinarrayx|vinformat|vinformatd|vinformatdx|"+"vinformatn|vinformatnx|vinformatw|vinformatwx|"+"vinformatx|vlabel|vlabelx|vlength|vlengthx|vname|"+"vnamex|vtype|vtypex|weekday|year|yyq|zipfips|zipname|"+"zipnamel|zipstate";var SAS_MACRO_FUN="bquote|nrbquote|cmpres|qcmpres|compstor|"+"datatyp|display|do|else|end|eval|global|goto|"+"if|index|input|keydef|label|left|length|let|"+"local|lowcase|macro|mend|nrbquote|nrquote|"+"nrstr|put|qcmpres|qleft|qlowcase|qscan|"+"qsubstr|qsysfunc|qtrim|quote|qupcase|scan|str|"+"substr|superq|syscall|sysevalf|sysexec|sysfunc|"+"sysget|syslput|sysprod|sysrc|sysrput|then|to|"+"trim|unquote|until|upcase|verify|while|window";return{aliases:["sas","SAS"],case_insensitive:true,keywords:{literal:"null missing _all_ _automatic_ _character_ _infile_ "+"_n_ _name_ _null_ _numeric_ _user_ _webout_",meta:SAS_KEYWORDS},contains:[{className:"keyword",begin:/^\s*(proc [\w\d_]+|data|run|quit)[\s\;]/},{className:"variable",begin:/\&[a-zA-Z_\&][a-zA-Z0-9_]*\.?/},{className:"emphasis",begin:/^\s*datalines|cards.*;/,end:/^\s*;\s*$/},{className:"built_in",begin:"%("+SAS_MACRO_FUN+")"},{className:"name",begin:/%[a-zA-Z_][a-zA-Z_0-9]*/},{className:"meta",begin:"[^%]("+SAS_FUN+")[(]"},{className:"string",variants:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]},hljs.COMMENT("\\*",";"),hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("gradle",function(hljs){return{case_insensitive:true,keywords:{keyword:"task project allprojects subprojects artifacts buildscript configurations "+"dependencies repositories sourceSets description delete from into include "+"exclude source classpath destinationDir includes options sourceCompatibility "+"targetCompatibility group flatDir doLast doFirst flatten todir fromdir ant "+"def abstract break case catch continue default do else extends final finally "+"for if implements instanceof native new private protected public return static "+"switch synchronized throw throws transient try volatile while strictfp package "+"import false null super this true antlrtask checkstyle codenarc copy boolean "+"byte char class double float int interface long short void compile runTime "+"file fileTree abs any append asList asWritable call collect compareTo count "+"div dump each eachByte eachFile eachLine every find findAll flatten getAt "+"getErr getIn getOut getText grep immutable inject inspect intersect invokeMethods "+"isCase join leftShift minus multiply newInputStream newOutputStream newPrintWriter "+"newReader newWriter next plus pop power previous print println push putAt read "+"readBytes readLines reverse reverseEach round size sort splitEachLine step subMap "+"times toInteger toList tokenize upto waitForOrKill withPrintWriter withReader "+"withStream withWriter withWriterAppend write writeLine"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.REGEXP_MODE]}});hljs.registerLanguage("stylus",function(hljs){var VARIABLE={className:"variable",begin:"\\$"+hljs.IDENT_RE};var HEX_COLOR={className:"number",begin:"#([a-fA-F0-9]{6}|[a-fA-F0-9]{3})"};var AT_KEYWORDS=["charset","css","debug","extend","font-face","for","import","include","media","mixin","page","warn","while"];var PSEUDO_SELECTORS=["after","before","first-letter","first-line","active","first-child","focus","hover","lang","link","visited"];var TAGS=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"];var LOOKAHEAD_TAG_END="(?=[\\.\\s\\n\\[\\:,])";var ATTRIBUTES=["align-content","align-items","align-self","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","auto","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","clip-path","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","font","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-stretch","font-style","font-variant","font-variant-ligatures","font-weight","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inherit","initial","justify-content","left","letter-spacing","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marks","mask","max-height","max-width","min-height","min-width","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","perspective","perspective-origin","pointer-events","position","quotes","resize","right","tab-size","table-layout","text-align","text-align-last","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-indent","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","white-space","widows","width","word-break","word-spacing","word-wrap","z-index"];var ILLEGAL=["\\?","(\\bReturn\\b)","(\\bEnd\\b)","(\\bend\\b)","(\\bdef\\b)",";","#\\s","\\*\\s","===\\s","\\|","%"];return{aliases:["styl"],case_insensitive:false,keywords:"if else for in",illegal:"("+ILLEGAL.join("|")+")",contains:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,HEX_COLOR,{begin:"\\.[a-zA-Z][a-zA-Z0-9_-]*"+LOOKAHEAD_TAG_END,className:"selector-class"},{begin:"\\#[a-zA-Z][a-zA-Z0-9_-]*"+LOOKAHEAD_TAG_END,className:"selector-id"},{begin:"\\b("+TAGS.join("|")+")"+LOOKAHEAD_TAG_END,className:"selector-tag"},{begin:"&?:?:\\b("+PSEUDO_SELECTORS.join("|")+")"+LOOKAHEAD_TAG_END},{begin:"@("+AT_KEYWORDS.join("|")+")\\b"},VARIABLE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,{className:"function",begin:"^[a-zA-Z][a-zA-Z0-9_-]*\\(.*\\)",illegal:"[\\n]",returnBegin:true,contains:[{className:"title",begin:"\\b[a-zA-Z][a-zA-Z0-9_-]*"},{className:"params",begin:/\(/,end:/\)/,contains:[HEX_COLOR,VARIABLE,hljs.APOS_STRING_MODE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,hljs.QUOTE_STRING_MODE]}]},{className:"attribute",begin:"\\b("+ATTRIBUTES.reverse().join("|")+")\\b",starts:{end:/;|$/,contains:[HEX_COLOR,VARIABLE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE],illegal:/\./,relevance:0}}]}});hljs.registerLanguage("profile",function(hljs){return{contains:[hljs.C_NUMBER_MODE,{begin:"[a-zA-Z_][\\da-zA-Z_]+\\.[\\da-zA-Z_]{1,3}",end:":",excludeEnd:true},{begin:"(ncalls|tottime|cumtime)",end:"$",keywords:"ncalls tottime|10 cumtime|10 filename",relevance:10},{begin:"function calls",end:"$",contains:[hljs.C_NUMBER_MODE],relevance:10},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"\\(",end:"\\)$",excludeBegin:true,excludeEnd:true,relevance:0}]}});hljs.registerLanguage("sqf",function(hljs){var VARIABLE={className:"variable",begin:/\b_+[a-zA-Z_]\w*/};var FUNCTION={className:"title",begin:/[a-zA-Z][a-zA-Z0-9]+_fnc_\w*/};var STRINGS={className:"string",variants:[{begin:'"',end:'"',contains:[{begin:'""',relevance:0}]},{begin:"'",end:"'",contains:[{begin:"''",relevance:0}]}]};var PREPROCESSOR={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"define undef ifdef ifndef else endif include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:/<[^\n>]*>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};return{aliases:["sqf"],case_insensitive:true,keywords:{keyword:"case catch default do else exit exitWith for forEach from if "+"private switch then throw to try waitUntil while with",built_in:"abs accTime acos action actionIDs actionKeys actionKeysImages actionKeysNames "+"actionKeysNamesArray actionName actionParams activateAddons activatedAddons activateKey "+"add3DENConnection add3DENEventHandler add3DENLayer addAction addBackpack addBackpackCargo "+"addBackpackCargoGlobal addBackpackGlobal addCamShake addCuratorAddons addCuratorCameraArea "+"addCuratorEditableObjects addCuratorEditingArea addCuratorPoints addEditorObject addEventHandler "+"addForce addGoggles addGroupIcon addHandgunItem addHeadgear addItem addItemCargo "+"addItemCargoGlobal addItemPool addItemToBackpack addItemToUniform addItemToVest addLiveStats "+"addMagazine addMagazineAmmoCargo addMagazineCargo addMagazineCargoGlobal addMagazineGlobal "+"addMagazinePool addMagazines addMagazineTurret addMenu addMenuItem addMissionEventHandler "+"addMPEventHandler addMusicEventHandler addOwnedMine addPlayerScores addPrimaryWeaponItem "+"addPublicVariableEventHandler addRating addResources addScore addScoreSide addSecondaryWeaponItem "+"addSwitchableUnit addTeamMember addToRemainsCollector addTorque addUniform addVehicle addVest "+"addWaypoint addWeapon addWeaponCargo addWeaponCargoGlobal addWeaponGlobal addWeaponItem "+"addWeaponPool addWeaponTurret admin agent agents AGLToASL aimedAtTarget aimPos airDensityRTD "+"airplaneThrottle airportSide AISFinishHeal alive all3DENEntities allAirports allControls "+"allCurators allCutLayers allDead allDeadMen allDisplays allGroups allMapMarkers allMines "+"allMissionObjects allow3DMode allowCrewInImmobile allowCuratorLogicIgnoreAreas allowDamage "+"allowDammage allowFileOperations allowFleeing allowGetIn allowSprint allPlayers allSimpleObjects "+"allSites allTurrets allUnits allUnitsUAV allVariables ammo ammoOnPylon and animate animateBay "+"animateDoor animatePylon animateSource animationNames animationPhase animationSourcePhase "+"animationState append apply armoryPoints arrayIntersect asin ASLToAGL ASLToATL assert "+"assignAsCargo assignAsCargoIndex assignAsCommander assignAsDriver assignAsGunner assignAsTurret "+"assignCurator assignedCargo assignedCommander assignedDriver assignedGunner assignedItems "+"assignedTarget assignedTeam assignedVehicle assignedVehicleRole assignItem assignTeam "+"assignToAirport atan atan2 atg ATLToASL attachedObject attachedObjects attachedTo attachObject "+"attachTo attackEnabled backpack backpackCargo backpackContainer backpackItems backpackMagazines "+"backpackSpaceFor behaviour benchmark binocular boundingBox boundingBoxReal boundingCenter "+"breakOut breakTo briefingName buildingExit buildingPos buttonAction buttonSetAction cadetMode "+"call callExtension camCommand camCommit camCommitPrepared camCommitted camConstuctionSetParams "+"camCreate camDestroy cameraEffect cameraEffectEnableHUD cameraInterest cameraOn cameraView "+"campaignConfigFile camPreload camPreloaded camPrepareBank camPrepareDir camPrepareDive "+"camPrepareFocus camPrepareFov camPrepareFovRange camPreparePos camPrepareRelPos camPrepareTarget "+"camSetBank camSetDir camSetDive camSetFocus camSetFov camSetFovRange camSetPos camSetRelPos "+"camSetTarget camTarget camUseNVG canAdd canAddItemToBackpack canAddItemToUniform canAddItemToVest "+"cancelSimpleTaskDestination canFire canMove canSlingLoad canStand canSuspend "+"canTriggerDynamicSimulation canUnloadInCombat canVehicleCargo captive captiveNum cbChecked "+"cbSetChecked ceil channelEnabled cheatsEnabled checkAIFeature checkVisibility className "+"clearAllItemsFromBackpack clearBackpackCargo clearBackpackCargoGlobal clearGroupIcons "+"clearItemCargo clearItemCargoGlobal clearItemPool clearMagazineCargo clearMagazineCargoGlobal "+"clearMagazinePool clearOverlay clearRadio clearWeaponCargo clearWeaponCargoGlobal clearWeaponPool "+"clientOwner closeDialog closeDisplay closeOverlay collapseObjectTree collect3DENHistory "+"collectiveRTD combatMode commandArtilleryFire commandChat commander commandFire commandFollow "+"commandFSM commandGetOut commandingMenu commandMove commandRadio commandStop "+"commandSuppressiveFire commandTarget commandWatch comment commitOverlay compile compileFinal "+"completedFSM composeText configClasses configFile configHierarchy configName configProperties "+"configSourceAddonList configSourceMod configSourceModList confirmSensorTarget "+"connectTerminalToUAV controlsGroupCtrl copyFromClipboard copyToClipboard copyWaypoints cos count "+"countEnemy countFriendly countSide countType countUnknown create3DENComposition create3DENEntity "+"createAgent createCenter createDialog createDiaryLink createDiaryRecord createDiarySubject "+"createDisplay createGearDialog createGroup createGuardedPoint createLocation createMarker "+"createMarkerLocal createMenu createMine createMissionDisplay createMPCampaignDisplay "+"createSimpleObject createSimpleTask createSite createSoundSource createTask createTeam "+"createTrigger createUnit createVehicle createVehicleCrew createVehicleLocal crew ctAddHeader "+"ctAddRow ctClear ctCurSel ctData ctFindHeaderRows ctFindRowHeader ctHeaderControls ctHeaderCount "+"ctRemoveHeaders ctRemoveRows ctrlActivate ctrlAddEventHandler ctrlAngle ctrlAutoScrollDelay "+"ctrlAutoScrollRewind ctrlAutoScrollSpeed ctrlChecked ctrlClassName ctrlCommit ctrlCommitted "+"ctrlCreate ctrlDelete ctrlEnable ctrlEnabled ctrlFade ctrlHTMLLoaded ctrlIDC ctrlIDD "+"ctrlMapAnimAdd ctrlMapAnimClear ctrlMapAnimCommit ctrlMapAnimDone ctrlMapCursor ctrlMapMouseOver "+"ctrlMapScale ctrlMapScreenToWorld ctrlMapWorldToScreen ctrlModel ctrlModelDirAndUp ctrlModelScale "+"ctrlParent ctrlParentControlsGroup ctrlPosition ctrlRemoveAllEventHandlers ctrlRemoveEventHandler "+"ctrlScale ctrlSetActiveColor ctrlSetAngle ctrlSetAutoScrollDelay ctrlSetAutoScrollRewind "+"ctrlSetAutoScrollSpeed ctrlSetBackgroundColor ctrlSetChecked ctrlSetEventHandler ctrlSetFade "+"ctrlSetFocus ctrlSetFont ctrlSetFontH1 ctrlSetFontH1B ctrlSetFontH2 ctrlSetFontH2B ctrlSetFontH3 "+"ctrlSetFontH3B ctrlSetFontH4 ctrlSetFontH4B ctrlSetFontH5 ctrlSetFontH5B ctrlSetFontH6 "+"ctrlSetFontH6B ctrlSetFontHeight ctrlSetFontHeightH1 ctrlSetFontHeightH2 ctrlSetFontHeightH3 "+"ctrlSetFontHeightH4 ctrlSetFontHeightH5 ctrlSetFontHeightH6 ctrlSetFontHeightSecondary "+"ctrlSetFontP ctrlSetFontPB ctrlSetFontSecondary ctrlSetForegroundColor ctrlSetModel "+"ctrlSetModelDirAndUp ctrlSetModelScale ctrlSetPixelPrecision ctrlSetPosition ctrlSetScale "+"ctrlSetStructuredText ctrlSetText ctrlSetTextColor ctrlSetTooltip ctrlSetTooltipColorBox "+"ctrlSetTooltipColorShade ctrlSetTooltipColorText ctrlShow ctrlShown ctrlText ctrlTextHeight "+"ctrlTextWidth ctrlType ctrlVisible ctRowControls ctRowCount ctSetCurSel ctSetData "+"ctSetHeaderTemplate ctSetRowTemplate ctSetValue ctValue curatorAddons curatorCamera "+"curatorCameraArea curatorCameraAreaCeiling curatorCoef curatorEditableObjects curatorEditingArea "+"curatorEditingAreaType curatorMouseOver curatorPoints curatorRegisteredObjects curatorSelected "+"curatorWaypointCost current3DENOperation currentChannel currentCommand currentMagazine "+"currentMagazineDetail currentMagazineDetailTurret currentMagazineTurret currentMuzzle "+"currentNamespace currentTask currentTasks currentThrowable currentVisionMode currentWaypoint "+"currentWeapon currentWeaponMode currentWeaponTurret currentZeroing cursorObject cursorTarget "+"customChat customRadio cutFadeOut cutObj cutRsc cutText damage date dateToNumber daytime "+"deActivateKey debriefingText debugFSM debugLog deg delete3DENEntities deleteAt deleteCenter "+"deleteCollection deleteEditorObject deleteGroup deleteGroupWhenEmpty deleteIdentity "+"deleteLocation deleteMarker deleteMarkerLocal deleteRange deleteResources deleteSite deleteStatus "+"deleteTeam deleteVehicle deleteVehicleCrew deleteWaypoint detach detectedMines "+"diag_activeMissionFSMs diag_activeScripts diag_activeSQFScripts diag_activeSQSScripts "+"diag_captureFrame diag_captureFrameToFile diag_captureSlowFrame diag_codePerformance "+"diag_drawMode diag_enable diag_enabled diag_fps diag_fpsMin diag_frameNo diag_lightNewLoad "+"diag_list diag_log diag_logSlowFrame diag_mergeConfigFile diag_recordTurretLimits "+"diag_setLightNew diag_tickTime diag_toggle dialog diarySubjectExists didJIP didJIPOwner "+"difficulty difficultyEnabled difficultyEnabledRTD difficultyOption direction directSay disableAI "+"disableCollisionWith disableConversation disableDebriefingStats disableMapIndicators "+"disableNVGEquipment disableRemoteSensors disableSerialization disableTIEquipment "+"disableUAVConnectability disableUserInput displayAddEventHandler displayCtrl displayParent "+"displayRemoveAllEventHandlers displayRemoveEventHandler displaySetEventHandler dissolveTeam "+"distance distance2D distanceSqr distributionRegion do3DENAction doArtilleryFire doFire doFollow "+"doFSM doGetOut doMove doorPhase doStop doSuppressiveFire doTarget doWatch drawArrow drawEllipse "+"drawIcon drawIcon3D drawLine drawLine3D drawLink drawLocation drawPolygon drawRectangle "+"drawTriangle driver drop dynamicSimulationDistance dynamicSimulationDistanceCoef "+"dynamicSimulationEnabled dynamicSimulationSystemEnabled echo edit3DENMissionAttributes editObject "+"editorSetEventHandler effectiveCommander emptyPositions enableAI enableAIFeature "+"enableAimPrecision enableAttack enableAudioFeature enableAutoStartUpRTD enableAutoTrimRTD "+"enableCamShake enableCaustics enableChannel enableCollisionWith enableCopilot "+"enableDebriefingStats enableDiagLegend enableDynamicSimulation enableDynamicSimulationSystem "+"enableEndDialog enableEngineArtillery enableEnvironment enableFatigue enableGunLights "+"enableInfoPanelComponent enableIRLasers enableMimics enablePersonTurret enableRadio enableReload "+"enableRopeAttach enableSatNormalOnDetail enableSaving enableSentences enableSimulation "+"enableSimulationGlobal enableStamina enableTeamSwitch enableTraffic enableUAVConnectability "+"enableUAVWaypoints enableVehicleCargo enableVehicleSensor enableWeaponDisassembly "+"endLoadingScreen endMission engineOn enginesIsOnRTD enginesRpmRTD enginesTorqueRTD entities "+"environmentEnabled estimatedEndServerTime estimatedTimeLeft evalObjectArgument everyBackpack "+"everyContainer exec execEditorScript execFSM execVM exp expectedDestination exportJIPMessages "+"eyeDirection eyePos face faction fadeMusic fadeRadio fadeSound fadeSpeech failMission "+"fillWeaponsFromPool find findCover findDisplay findEditorObject findEmptyPosition "+"findEmptyPositionReady findIf findNearestEnemy finishMissionInit finite fire fireAtTarget "+"firstBackpack flag flagAnimationPhase flagOwner flagSide flagTexture fleeing floor flyInHeight "+"flyInHeightASL fog fogForecast fogParams forceAddUniform forcedMap forceEnd forceFlagTexture "+"forceFollowRoad forceMap forceRespawn forceSpeed forceWalk forceWeaponFire forceWeatherChange "+"forEachMember forEachMemberAgent forEachMemberTeam forgetTarget format formation "+"formationDirection formationLeader formationMembers formationPosition formationTask formatText "+"formLeader freeLook fromEditor fuel fullCrew gearIDCAmmoCount gearSlotAmmoCount gearSlotData "+"get3DENActionState get3DENAttribute get3DENCamera get3DENConnections get3DENEntity "+"get3DENEntityID get3DENGrid get3DENIconsVisible get3DENLayerEntities get3DENLinesVisible "+"get3DENMissionAttribute get3DENMouseOver get3DENSelected getAimingCoef getAllEnvSoundControllers "+"getAllHitPointsDamage getAllOwnedMines getAllSoundControllers getAmmoCargo getAnimAimPrecision "+"getAnimSpeedCoef getArray getArtilleryAmmo getArtilleryComputerSettings getArtilleryETA "+"getAssignedCuratorLogic getAssignedCuratorUnit getBackpackCargo getBleedingRemaining "+"getBurningValue getCameraViewDirection getCargoIndex getCenterOfMass getClientState "+"getClientStateNumber getCompatiblePylonMagazines getConnectedUAV getContainerMaxLoad "+"getCursorObjectParams getCustomAimCoef getDammage getDescription getDir getDirVisual "+"getDLCAssetsUsage getDLCAssetsUsageByName getDLCs getEditorCamera getEditorMode "+"getEditorObjectScope getElevationOffset getEnvSoundController getFatigue getForcedFlagTexture "+"getFriend getFSMVariable getFuelCargo getGroupIcon getGroupIconParams getGroupIcons getHideFrom "+"getHit getHitIndex getHitPointDamage getItemCargo getMagazineCargo getMarkerColor getMarkerPos "+"getMarkerSize getMarkerType getMass getMissionConfig getMissionConfigValue getMissionDLCs "+"getMissionLayerEntities getModelInfo getMousePosition getMusicPlayedTime getNumber "+"getObjectArgument getObjectChildren getObjectDLC getObjectMaterials getObjectProxy "+"getObjectTextures getObjectType getObjectViewDistance getOxygenRemaining getPersonUsedDLCs "+"getPilotCameraDirection getPilotCameraPosition getPilotCameraRotation getPilotCameraTarget "+"getPlateNumber getPlayerChannel getPlayerScores getPlayerUID getPos getPosASL getPosASLVisual "+"getPosASLW getPosATL getPosATLVisual getPosVisual getPosWorld getPylonMagazines getRelDir "+"getRelPos getRemoteSensorsDisabled getRepairCargo getResolution getShadowDistance getShotParents "+"getSlingLoad getSoundController getSoundControllerResult getSpeed getStamina getStatValue "+"getSuppression getTerrainGrid getTerrainHeightASL getText getTotalDLCUsageTime getUnitLoadout "+"getUnitTrait getUserMFDText getUserMFDvalue getVariable getVehicleCargo getWeaponCargo "+"getWeaponSway getWingsOrientationRTD getWingsPositionRTD getWPPos glanceAt globalChat globalRadio "+"goggles goto group groupChat groupFromNetId groupIconSelectable groupIconsVisible groupId "+"groupOwner groupRadio groupSelectedUnits groupSelectUnit gunner gusts halt handgunItems "+"handgunMagazine handgunWeapon handsHit hasInterface hasPilotCamera hasWeapon hcAllGroups "+"hcGroupParams hcLeader hcRemoveAllGroups hcRemoveGroup hcSelected hcSelectGroup hcSetGroup "+"hcShowBar hcShownBar headgear hideBody hideObject hideObjectGlobal hideSelection hint hintC "+"hintCadet hintSilent hmd hostMission htmlLoad HUDMovementLevels humidity image importAllGroups "+"importance in inArea inAreaArray incapacitatedState inflame inflamed infoPanel "+"infoPanelComponentEnabled infoPanelComponents infoPanels inGameUISetEventHandler inheritsFrom "+"initAmbientLife inPolygon inputAction inRangeOfArtillery insertEditorObject intersect is3DEN "+"is3DENMultiplayer isAbleToBreathe isAgent isArray isAutoHoverOn isAutonomous isAutotest "+"isBleeding isBurning isClass isCollisionLightOn isCopilotEnabled isDamageAllowed isDedicated "+"isDLCAvailable isEngineOn isEqualTo isEqualType isEqualTypeAll isEqualTypeAny isEqualTypeArray "+"isEqualTypeParams isFilePatchingEnabled isFlashlightOn isFlatEmpty isForcedWalk isFormationLeader "+"isGroupDeletedWhenEmpty isHidden isInRemainsCollector isInstructorFigureEnabled isIRLaserOn "+"isKeyActive isKindOf isLaserOn isLightOn isLocalized isManualFire isMarkedForCollection "+"isMultiplayer isMultiplayerSolo isNil isNull isNumber isObjectHidden isObjectRTD isOnRoad "+"isPipEnabled isPlayer isRealTime isRemoteExecuted isRemoteExecutedJIP isServer isShowing3DIcons "+"isSimpleObject isSprintAllowed isStaminaEnabled isSteamMission isStreamFriendlyUIEnabled isText "+"isTouchingGround isTurnedOut isTutHintsEnabled isUAVConnectable isUAVConnected isUIContext "+"isUniformAllowed isVehicleCargo isVehicleRadarOn isVehicleSensorEnabled isWalking "+"isWeaponDeployed isWeaponRested itemCargo items itemsWithMagazines join joinAs joinAsSilent "+"joinSilent joinString kbAddDatabase kbAddDatabaseTargets kbAddTopic kbHasTopic kbReact "+"kbRemoveTopic kbTell kbWasSaid keyImage keyName knowsAbout land landAt landResult language "+"laserTarget lbAdd lbClear lbColor lbColorRight lbCurSel lbData lbDelete lbIsSelected lbPicture "+"lbPictureRight lbSelection lbSetColor lbSetColorRight lbSetCurSel lbSetData lbSetPicture "+"lbSetPictureColor lbSetPictureColorDisabled lbSetPictureColorSelected lbSetPictureRight "+"lbSetPictureRightColor lbSetPictureRightColorDisabled lbSetPictureRightColorSelected "+"lbSetSelectColor lbSetSelectColorRight lbSetSelected lbSetText lbSetTextRight lbSetTooltip "+"lbSetValue lbSize lbSort lbSortByValue lbText lbTextRight lbValue leader leaderboardDeInit "+"leaderboardGetRows leaderboardInit leaderboardRequestRowsFriends leaderboardsRequestUploadScore "+"leaderboardsRequestUploadScoreKeepBest leaderboardState leaveVehicle libraryCredits "+"libraryDisclaimers lifeState lightAttachObject lightDetachObject lightIsOn lightnings limitSpeed "+"linearConversion lineIntersects lineIntersectsObjs lineIntersectsSurfaces lineIntersectsWith "+"linkItem list listObjects listRemoteTargets listVehicleSensors ln lnbAddArray lnbAddColumn "+"lnbAddRow lnbClear lnbColor lnbCurSelRow lnbData lnbDeleteColumn lnbDeleteRow "+"lnbGetColumnsPosition lnbPicture lnbSetColor lnbSetColumnsPos lnbSetCurSelRow lnbSetData "+"lnbSetPicture lnbSetText lnbSetValue lnbSize lnbSort lnbSortByValue lnbText lnbValue load loadAbs "+"loadBackpack loadFile loadGame loadIdentity loadMagazine loadOverlay loadStatus loadUniform "+"loadVest local localize locationPosition lock lockCameraTo lockCargo lockDriver locked "+"lockedCargo lockedDriver lockedTurret lockIdentity lockTurret lockWP log logEntities logNetwork "+"logNetworkTerminate lookAt lookAtPos magazineCargo magazines magazinesAllTurrets magazinesAmmo "+"magazinesAmmoCargo magazinesAmmoFull magazinesDetail magazinesDetailBackpack "+"magazinesDetailUniform magazinesDetailVest magazinesTurret magazineTurretAmmo mapAnimAdd "+"mapAnimClear mapAnimCommit mapAnimDone mapCenterOnCamera mapGridPosition markAsFinishedOnSteam "+"markerAlpha markerBrush markerColor markerDir markerPos markerShape markerSize markerText "+"markerType max members menuAction menuAdd menuChecked menuClear menuCollapse menuData menuDelete "+"menuEnable menuEnabled menuExpand menuHover menuPicture menuSetAction menuSetCheck menuSetData "+"menuSetPicture menuSetValue menuShortcut menuShortcutText menuSize menuSort menuText menuURL "+"menuValue min mineActive mineDetectedBy missionConfigFile missionDifficulty missionName "+"missionNamespace missionStart missionVersion mod modelToWorld modelToWorldVisual "+"modelToWorldVisualWorld modelToWorldWorld modParams moonIntensity moonPhase morale move "+"move3DENCamera moveInAny moveInCargo moveInCommander moveInDriver moveInGunner moveInTurret "+"moveObjectToEnd moveOut moveTime moveTo moveToCompleted moveToFailed musicVolume name nameSound "+"nearEntities nearestBuilding nearestLocation nearestLocations nearestLocationWithDubbing "+"nearestObject nearestObjects nearestTerrainObjects nearObjects nearObjectsReady nearRoads "+"nearSupplies nearTargets needReload netId netObjNull newOverlay nextMenuItemIndex "+"nextWeatherChange nMenuItems not numberOfEnginesRTD numberToDate objectCurators objectFromNetId "+"objectParent objStatus onBriefingGroup onBriefingNotes onBriefingPlan onBriefingTeamSwitch "+"onCommandModeChanged onDoubleClick onEachFrame onGroupIconClick onGroupIconOverEnter "+"onGroupIconOverLeave onHCGroupSelectionChanged onMapSingleClick onPlayerConnected "+"onPlayerDisconnected onPreloadFinished onPreloadStarted onShowNewObject onTeamSwitch "+"openCuratorInterface openDLCPage openMap openSteamApp openYoutubeVideo or orderGetIn overcast "+"overcastForecast owner param params parseNumber parseSimpleArray parseText parsingNamespace "+"particlesQuality pickWeaponPool pitch pixelGrid pixelGridBase pixelGridNoUIScale pixelH pixelW "+"playableSlotsNumber playableUnits playAction playActionNow player playerRespawnTime playerSide "+"playersNumber playGesture playMission playMove playMoveNow playMusic playScriptedMission "+"playSound playSound3D position positionCameraToWorld posScreenToWorld posWorldToScreen "+"ppEffectAdjust ppEffectCommit ppEffectCommitted ppEffectCreate ppEffectDestroy ppEffectEnable "+"ppEffectEnabled ppEffectForceInNVG precision preloadCamera preloadObject preloadSound "+"preloadTitleObj preloadTitleRsc preprocessFile preprocessFileLineNumbers primaryWeapon "+"primaryWeaponItems primaryWeaponMagazine priority processDiaryLink productVersion profileName "+"profileNamespace profileNameSteam progressLoadingScreen progressPosition progressSetPosition "+"publicVariable publicVariableClient publicVariableServer pushBack pushBackUnique putWeaponPool "+"queryItemsPool queryMagazinePool queryWeaponPool rad radioChannelAdd radioChannelCreate "+"radioChannelRemove radioChannelSetCallSign radioChannelSetLabel radioVolume rain rainbow random "+"rank rankId rating rectangular registeredTasks registerTask reload reloadEnabled remoteControl "+"remoteExec remoteExecCall remoteExecutedOwner remove3DENConnection remove3DENEventHandler "+"remove3DENLayer removeAction removeAll3DENEventHandlers removeAllActions removeAllAssignedItems "+"removeAllContainers removeAllCuratorAddons removeAllCuratorCameraAreas "+"removeAllCuratorEditingAreas removeAllEventHandlers removeAllHandgunItems removeAllItems "+"removeAllItemsWithMagazines removeAllMissionEventHandlers removeAllMPEventHandlers "+"removeAllMusicEventHandlers removeAllOwnedMines removeAllPrimaryWeaponItems removeAllWeapons "+"removeBackpack removeBackpackGlobal removeCuratorAddons removeCuratorCameraArea "+"removeCuratorEditableObjects removeCuratorEditingArea removeDrawIcon removeDrawLinks "+"removeEventHandler removeFromRemainsCollector removeGoggles removeGroupIcon removeHandgunItem "+"removeHeadgear removeItem removeItemFromBackpack removeItemFromUniform removeItemFromVest "+"removeItems removeMagazine removeMagazineGlobal removeMagazines removeMagazinesTurret "+"removeMagazineTurret removeMenuItem removeMissionEventHandler removeMPEventHandler "+"removeMusicEventHandler removeOwnedMine removePrimaryWeaponItem removeSecondaryWeaponItem "+"removeSimpleTask removeSwitchableUnit removeTeamMember removeUniform removeVest removeWeapon "+"removeWeaponAttachmentCargo removeWeaponCargo removeWeaponGlobal removeWeaponTurret "+"reportRemoteTarget requiredVersion resetCamShake resetSubgroupDirection resize resources "+"respawnVehicle restartEditorCamera reveal revealMine reverse reversedMouseY roadAt "+"roadsConnectedTo roleDescription ropeAttachedObjects ropeAttachedTo ropeAttachEnabled "+"ropeAttachTo ropeCreate ropeCut ropeDestroy ropeDetach ropeEndPosition ropeLength ropes "+"ropeUnwind ropeUnwound rotorsForcesRTD rotorsRpmRTD round runInitScript safeZoneH safeZoneW "+"safeZoneWAbs safeZoneX safeZoneXAbs safeZoneY save3DENInventory saveGame saveIdentity "+"saveJoysticks saveOverlay saveProfileNamespace saveStatus saveVar savingEnabled say say2D say3D "+"scopeName score scoreSide screenshot screenToWorld scriptDone scriptName scudState "+"secondaryWeapon secondaryWeaponItems secondaryWeaponMagazine select selectBestPlaces "+"selectDiarySubject selectedEditorObjects selectEditorObject selectionNames selectionPosition "+"selectLeader selectMax selectMin selectNoPlayer selectPlayer selectRandom selectRandomWeighted "+"selectWeapon selectWeaponTurret sendAUMessage sendSimpleCommand sendTask sendTaskResult "+"sendUDPMessage serverCommand serverCommandAvailable serverCommandExecutable serverName serverTime "+"set set3DENAttribute set3DENAttributes set3DENGrid set3DENIconsVisible set3DENLayer "+"set3DENLinesVisible set3DENLogicType set3DENMissionAttribute set3DENMissionAttributes "+"set3DENModelsVisible set3DENObjectType set3DENSelected setAccTime setActualCollectiveRTD "+"setAirplaneThrottle setAirportSide setAmmo setAmmoCargo setAmmoOnPylon setAnimSpeedCoef "+"setAperture setApertureNew setArmoryPoints setAttributes setAutonomous setBehaviour "+"setBleedingRemaining setBrakesRTD setCameraInterest setCamShakeDefParams setCamShakeParams "+"setCamUseTI setCaptive setCenterOfMass setCollisionLight setCombatMode setCompassOscillation "+"setConvoySeparation setCuratorCameraAreaCeiling setCuratorCoef setCuratorEditingAreaType "+"setCuratorWaypointCost setCurrentChannel setCurrentTask setCurrentWaypoint setCustomAimCoef "+"setCustomWeightRTD setDamage setDammage setDate setDebriefingText setDefaultCamera setDestination "+"setDetailMapBlendPars setDir setDirection setDrawIcon setDriveOnPath setDropInterval "+"setDynamicSimulationDistance setDynamicSimulationDistanceCoef setEditorMode setEditorObjectScope "+"setEffectCondition setEngineRPMRTD setFace setFaceAnimation setFatigue setFeatureType "+"setFlagAnimationPhase setFlagOwner setFlagSide setFlagTexture setFog setFormation "+"setFormationTask setFormDir setFriend setFromEditor setFSMVariable setFuel setFuelCargo "+"setGroupIcon setGroupIconParams setGroupIconsSelectable setGroupIconsVisible setGroupId "+"setGroupIdGlobal setGroupOwner setGusts setHideBehind setHit setHitIndex setHitPointDamage "+"setHorizonParallaxCoef setHUDMovementLevels setIdentity setImportance setInfoPanel setLeader "+"setLightAmbient setLightAttenuation setLightBrightness setLightColor setLightDayLight "+"setLightFlareMaxDistance setLightFlareSize setLightIntensity setLightnings setLightUseFlare "+"setLocalWindParams setMagazineTurretAmmo setMarkerAlpha setMarkerAlphaLocal setMarkerBrush "+"setMarkerBrushLocal setMarkerColor setMarkerColorLocal setMarkerDir setMarkerDirLocal "+"setMarkerPos setMarkerPosLocal setMarkerShape setMarkerShapeLocal setMarkerSize "+"setMarkerSizeLocal setMarkerText setMarkerTextLocal setMarkerType setMarkerTypeLocal setMass "+"setMimic setMousePosition setMusicEffect setMusicEventHandler setName setNameSound "+"setObjectArguments setObjectMaterial setObjectMaterialGlobal setObjectProxy setObjectTexture "+"setObjectTextureGlobal setObjectViewDistance setOvercast setOwner setOxygenRemaining "+"setParticleCircle setParticleClass setParticleFire setParticleParams setParticleRandom "+"setPilotCameraDirection setPilotCameraRotation setPilotCameraTarget setPilotLight setPiPEffect "+"setPitch setPlateNumber setPlayable setPlayerRespawnTime setPos setPosASL setPosASL2 setPosASLW "+"setPosATL setPosition setPosWorld setPylonLoadOut setPylonsPriority setRadioMsg setRain "+"setRainbow setRandomLip setRank setRectangular setRepairCargo setRotorBrakeRTD setShadowDistance "+"setShotParents setSide setSimpleTaskAlwaysVisible setSimpleTaskCustomData "+"setSimpleTaskDescription setSimpleTaskDestination setSimpleTaskTarget setSimpleTaskType "+"setSimulWeatherLayers setSize setSkill setSlingLoad setSoundEffect setSpeaker setSpeech "+"setSpeedMode setStamina setStaminaScheme setStatValue setSuppression setSystemOfUnits "+"setTargetAge setTaskMarkerOffset setTaskResult setTaskState setTerrainGrid setText "+"setTimeMultiplier setTitleEffect setTrafficDensity setTrafficDistance setTrafficGap "+"setTrafficSpeed setTriggerActivation setTriggerArea setTriggerStatements setTriggerText "+"setTriggerTimeout setTriggerType setType setUnconscious setUnitAbility setUnitLoadout setUnitPos "+"setUnitPosWeak setUnitRank setUnitRecoilCoefficient setUnitTrait setUnloadInCombat "+"setUserActionText setUserMFDText setUserMFDvalue setVariable setVectorDir setVectorDirAndUp "+"setVectorUp setVehicleAmmo setVehicleAmmoDef setVehicleArmor setVehicleCargo setVehicleId "+"setVehicleLock setVehiclePosition setVehicleRadar setVehicleReceiveRemoteTargets "+"setVehicleReportOwnPosition setVehicleReportRemoteTargets setVehicleTIPars setVehicleVarName "+"setVelocity setVelocityModelSpace setVelocityTransformation setViewDistance "+"setVisibleIfTreeCollapsed setWantedRPMRTD setWaves setWaypointBehaviour setWaypointCombatMode "+"setWaypointCompletionRadius setWaypointDescription setWaypointForceBehaviour setWaypointFormation "+"setWaypointHousePosition setWaypointLoiterRadius setWaypointLoiterType setWaypointName "+"setWaypointPosition setWaypointScript setWaypointSpeed setWaypointStatements setWaypointTimeout "+"setWaypointType setWaypointVisible setWeaponReloadingTime setWind setWindDir setWindForce "+"setWindStr setWingForceScaleRTD setWPPos show3DIcons showChat showCinemaBorder showCommandingMenu "+"showCompass showCuratorCompass showGPS showHUD showLegend showMap shownArtilleryComputer "+"shownChat shownCompass shownCuratorCompass showNewEditorObject shownGPS shownHUD shownMap "+"shownPad shownRadio shownScoretable shownUAVFeed shownWarrant shownWatch showPad showRadio "+"showScoretable showSubtitles showUAVFeed showWarrant showWatch showWaypoint showWaypoints side "+"sideChat sideEnemy sideFriendly sideRadio simpleTasks simulationEnabled simulCloudDensity "+"simulCloudOcclusion simulInClouds simulWeatherSync sin size sizeOf skill skillFinal skipTime "+"sleep sliderPosition sliderRange sliderSetPosition sliderSetRange sliderSetSpeed sliderSpeed "+"slingLoadAssistantShown soldierMagazines someAmmo sort soundVolume spawn speaker speed speedMode "+"splitString sqrt squadParams stance startLoadingScreen step stop stopEngineRTD stopped str "+"sunOrMoon supportInfo suppressFor surfaceIsWater surfaceNormal surfaceType swimInDepth "+"switchableUnits switchAction switchCamera switchGesture switchLight switchMove "+"synchronizedObjects synchronizedTriggers synchronizedWaypoints synchronizeObjectsAdd "+"synchronizeObjectsRemove synchronizeTrigger synchronizeWaypoint systemChat systemOfUnits tan "+"targetKnowledge targets targetsAggregate targetsQuery taskAlwaysVisible taskChildren "+"taskCompleted taskCustomData taskDescription taskDestination taskHint taskMarkerOffset taskParent "+"taskResult taskState taskType teamMember teamName teams teamSwitch teamSwitchEnabled teamType "+"terminate terrainIntersect terrainIntersectASL terrainIntersectAtASL text textLog textLogFormat "+"tg time timeMultiplier titleCut titleFadeOut titleObj titleRsc titleText toArray toFixed toLower "+"toString toUpper triggerActivated triggerActivation triggerArea triggerAttachedVehicle "+"triggerAttachObject triggerAttachVehicle triggerDynamicSimulation triggerStatements triggerText "+"triggerTimeout triggerTimeoutCurrent triggerType turretLocal turretOwner turretUnit tvAdd tvClear "+"tvCollapse tvCollapseAll tvCount tvCurSel tvData tvDelete tvExpand tvExpandAll tvPicture "+"tvSetColor tvSetCurSel tvSetData tvSetPicture tvSetPictureColor tvSetPictureColorDisabled "+"tvSetPictureColorSelected tvSetPictureRight tvSetPictureRightColor tvSetPictureRightColorDisabled "+"tvSetPictureRightColorSelected tvSetText tvSetTooltip tvSetValue tvSort tvSortByValue tvText "+"tvTooltip tvValue type typeName typeOf UAVControl uiNamespace uiSleep unassignCurator "+"unassignItem unassignTeam unassignVehicle underwater uniform uniformContainer uniformItems "+"uniformMagazines unitAddons unitAimPosition unitAimPositionVisual unitBackpack unitIsUAV unitPos "+"unitReady unitRecoilCoefficient units unitsBelowHeight unlinkItem unlockAchievement "+"unregisterTask updateDrawIcon updateMenuItem updateObjectTree useAISteeringComponent "+"useAudioTimeForMoves userInputDisabled vectorAdd vectorCos vectorCrossProduct vectorDiff "+"vectorDir vectorDirVisual vectorDistance vectorDistanceSqr vectorDotProduct vectorFromTo "+"vectorMagnitude vectorMagnitudeSqr vectorModelToWorld vectorModelToWorldVisual vectorMultiply "+"vectorNormalized vectorUp vectorUpVisual vectorWorldToModel vectorWorldToModelVisual vehicle "+"vehicleCargoEnabled vehicleChat vehicleRadio vehicleReceiveRemoteTargets vehicleReportOwnPosition "+"vehicleReportRemoteTargets vehicles vehicleVarName velocity velocityModelSpace verifySignature "+"vest vestContainer vestItems vestMagazines viewDistance visibleCompass visibleGPS visibleMap "+"visiblePosition visiblePositionASL visibleScoretable visibleWatch waves waypointAttachedObject "+"waypointAttachedVehicle waypointAttachObject waypointAttachVehicle waypointBehaviour "+"waypointCombatMode waypointCompletionRadius waypointDescription waypointForceBehaviour "+"waypointFormation waypointHousePosition waypointLoiterRadius waypointLoiterType waypointName "+"waypointPosition waypoints waypointScript waypointsEnabledUAV waypointShow waypointSpeed "+"waypointStatements waypointTimeout waypointTimeoutCurrent waypointType waypointVisible "+"weaponAccessories weaponAccessoriesCargo weaponCargo weaponDirection weaponInertia weaponLowered "+"weapons weaponsItems weaponsItemsCargo weaponState weaponsTurret weightRTD WFSideText wind ",literal:"blufor civilian configNull controlNull displayNull east endl false grpNull independent lineBreak "+"locationNull nil objNull opfor pi resistance scriptNull sideAmbientLife sideEmpty sideLogic "+"sideUnknown taskNull teamMemberNull true west"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.NUMBER_MODE,VARIABLE,FUNCTION,STRINGS,PREPROCESSOR],illegal:/#|^\$ /}});hljs.registerLanguage("qml",function(hljs){var KEYWORDS={keyword:"in of on if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const export super debugger as async await import",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect "+"Behavior bool color coordinate date double enumeration font geocircle georectangle "+"geoshape int list matrix4x4 parent point quaternion real rect "+"size string url variant vector2d vector3d vector4d"+"Promise"};var QML_IDENT_RE="[a-zA-Z_][a-zA-Z0-9\\._]*";var PROPERTY={className:"keyword",begin:"\\bproperty\\b",starts:{className:"string",end:"(:|=|;|,|//|/\\*|$)",returnEnd:true}};var SIGNAL={className:"keyword",begin:"\\bsignal\\b",starts:{className:"string",end:"(\\(|:|=|;|,|//|/\\*|$)",returnEnd:true}};var ID_ID={className:"attribute",begin:"\\bid\\s*:",starts:{className:"string",end:QML_IDENT_RE,returnEnd:false}};var QML_ATTRIBUTE={begin:QML_IDENT_RE+"\\s*:",returnBegin:true,contains:[{className:"attribute",begin:QML_IDENT_RE,end:"\\s*:",excludeEnd:true,relevance:0}],relevance:0};var QML_OBJECT={begin:QML_IDENT_RE+"\\s*{",end:"{",returnBegin:true,relevance:0,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:QML_IDENT_RE})]};return{aliases:["qt"],case_insensitive:false,keywords:KEYWORDS,contains:[{className:"meta",begin:/^\s*['"]use (strict|asm)['"]/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",begin:"\\$\\{",end:"\\}"}]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"number",variants:[{begin:"\\b(0[bB][01]+)"},{begin:"\\b(0[oO][0-7]+)"},{begin:hljs.C_NUMBER_RE}],relevance:0},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{begin:/</,end:/>\s*[);\]]/,relevance:0,subLanguage:"xml"}],relevance:0},SIGNAL,PROPERTY,{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][0-9A-Za-z$_]*/}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}],illegal:/\[|%/},{begin:"\\."+hljs.IDENT_RE,relevance:0},ID_ID,QML_ATTRIBUTE,QML_OBJECT],illegal:/#/}});hljs.registerLanguage("erb",function(hljs){return{subLanguage:"xml",contains:[hljs.COMMENT("<%#","%>"),{begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:true,excludeEnd:true}]}});hljs.registerLanguage("openscad",function(hljs){var SPECIAL_VARS={className:"keyword",begin:"\\$(f[asn]|t|vp[rtd]|children)"},LITERALS={className:"literal",begin:"false|true|PI|undef"},NUMBERS={className:"number",begin:"\\b\\d+(\\.\\d+)?(e-?\\d+)?",relevance:0},STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),PREPRO={className:"meta",keywords:{"meta-keyword":"include use"},begin:"include|use <",end:">"},PARAMS={className:"params",begin:"\\(",end:"\\)",contains:["self",NUMBERS,STRING,SPECIAL_VARS,LITERALS]},MODIFIERS={begin:"[*!#%]",relevance:0},FUNCTIONS={className:"function",beginKeywords:"module function",end:"\\=|\\{",contains:[PARAMS,hljs.UNDERSCORE_TITLE_MODE]};return{aliases:["scad"],keywords:{keyword:"function module include use for intersection_for if else \\%",literal:"false true PI undef",built_in:"circle square polygon text sphere cube cylinder polyhedron translate rotate scale resize mirror multmatrix color offset hull minkowski union difference intersection abs sign sin cos tan acos asin atan atan2 floor round ceil ln log pow sqrt exp rands min max concat lookup str chr search version version_num norm cross parent_module echo import import_dxf dxf_linear_extrude linear_extrude rotate_extrude surface projection render children dxf_cross dxf_dim let assign"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,PREPRO,STRING,SPECIAL_VARS,MODIFIERS,FUNCTIONS]}});hljs.registerLanguage("elixir",function(hljs){var ELIXIR_IDENT_RE="[a-zA-Z_][a-zA-Z0-9_.]*(\\!|\\?)?";var ELIXIR_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?";var ELIXIR_KEYWORDS="and false then defined module in return redo retry end for true self when "+"next until do begin unless nil break not case cond alias while ensure or "+"include use alias fn quote require import with|0";var SUBST={className:"subst",begin:"#\\{",end:"}",lexemes:ELIXIR_IDENT_RE,keywords:ELIXIR_KEYWORDS};var SIGIL_DELIMITERS="[/|([{<\"']";var LOWERCASE_SIGIL={className:"string",begin:"~[a-z]"+"(?="+SIGIL_DELIMITERS+")",contains:[{endsParent:true,contains:[{contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"/,end:/"/},{begin:/'/,end:/'/},{begin:/\//,end:/\//},{begin:/\|/,end:/\|/},{begin:/\(/,end:/\)/},{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/},{begin:/</,end:/>/}]}]}]};var UPCASE_SIGIL={className:"string",begin:"~[A-Z]"+"(?="+SIGIL_DELIMITERS+")",contains:[{begin:/"/,end:/"/},{begin:/'/,end:/'/},{begin:/\//,end:/\//},{begin:/\|/,end:/\|/},{begin:/\(/,end:/\)/},{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/},{begin:/\</,end:/\>/}]};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"""/,end:/"""/},{begin:/'''/,end:/'''/},{begin:/~S"""/,end:/"""/,contains:[]},{begin:/~S"/,end:/"/,contains:[]},{begin:/~S'''/,end:/'''/,contains:[]},{begin:/~S'/,end:/'/,contains:[]},{begin:/'/,end:/'/},{begin:/"/,end:/"/}]};var FUNCTION={className:"function",beginKeywords:"def defp defmacro",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:ELIXIR_IDENT_RE,endsParent:true})]};var CLASS=hljs.inherit(FUNCTION,{className:"class",beginKeywords:"defimpl defmodule defprotocol defrecord",end:/\bdo\b|$|;/});var ELIXIR_DEFAULT_CONTAINS=[STRING,UPCASE_SIGIL,LOWERCASE_SIGIL,hljs.HASH_COMMENT_MODE,CLASS,FUNCTION,{begin:"::"},{className:"symbol",begin:":(?![\\s:])",contains:[STRING,{begin:ELIXIR_METHOD_RE}],relevance:0},{className:"symbol",begin:ELIXIR_IDENT_RE+":(?!:)",relevance:0},{className:"number",begin:"(\\b0o[0-7_]+)|(\\b0b[01_]+)|(\\b0x[0-9a-fA-F_]+)|(-?\\b[1-9][0-9_]*(.[0-9_]+([eE][-+]?[0-9]+)?)?)",relevance:0},{className:"variable",begin:"(\\$\\W)|((\\$|\\@\\@?)(\\w+))"},{begin:"->"},{begin:"("+hljs.RE_STARTERS_RE+")\\s*",contains:[hljs.HASH_COMMENT_MODE,{className:"regexp",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"/",end:"/[a-z]*"},{begin:"%r\\[",end:"\\][a-z]*"}]}],relevance:0}];SUBST.contains=ELIXIR_DEFAULT_CONTAINS;return{lexemes:ELIXIR_IDENT_RE,keywords:ELIXIR_KEYWORDS,contains:ELIXIR_DEFAULT_CONTAINS}});hljs.registerLanguage("vhdl",function(hljs){var INTEGER_RE="\\d(_|\\d)*";var EXPONENT_RE="[eE][-+]?"+INTEGER_RE;var DECIMAL_LITERAL_RE=INTEGER_RE+"(\\."+INTEGER_RE+")?"+"("+EXPONENT_RE+")?";var BASED_INTEGER_RE="\\w+";var BASED_LITERAL_RE=INTEGER_RE+"#"+BASED_INTEGER_RE+"(\\."+BASED_INTEGER_RE+")?"+"#"+"("+EXPONENT_RE+")?";var NUMBER_RE="\\b("+BASED_LITERAL_RE+"|"+DECIMAL_LITERAL_RE+")";return{case_insensitive:true,keywords:{keyword:"abs access after alias all and architecture array assert assume assume_guarantee attribute "+"begin block body buffer bus case component configuration constant context cover disconnect "+"downto default else elsif end entity exit fairness file for force function generate "+"generic group guarded if impure in inertial inout is label library linkage literal "+"loop map mod nand new next nor not null of on open or others out package parameter port "+"postponed procedure process property protected pure range record register reject "+"release rem report restrict restrict_guarantee return rol ror select sequence "+"severity shared signal sla sll sra srl strong subtype then to transport type "+"unaffected units until use variable view vmode vprop vunit wait when while with xnor xor",built_in:"boolean bit character "+"integer time delay_length natural positive "+"string bit_vector file_open_kind file_open_status "+"std_logic std_logic_vector unsigned signed boolean_vector integer_vector "+"std_ulogic std_ulogic_vector unresolved_unsigned u_unsigned unresolved_signed u_signed "+"real_vector time_vector",literal:"false true note warning error failure "+"line text side width"},illegal:"{",contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT("--","$"),hljs.QUOTE_STRING_MODE,{className:"number",begin:NUMBER_RE,relevance:0},{className:"string",begin:"'(U|X|0|1|Z|W|L|H|-)'",contains:[hljs.BACKSLASH_ESCAPE]},{className:"symbol",begin:"'[A-Za-z](_?[A-Za-z0-9])*",contains:[hljs.BACKSLASH_ESCAPE]}]}});hljs.registerLanguage("aspectj",function(hljs){var KEYWORDS="false synchronized int abstract float private char boolean static null if const "+"for true while long throw strictfp finally protected import native final return void "+"enum else extends implements break transient new catch instanceof byte super volatile case "+"assert short package default double public try this switch continue throws privileged "+"aspectOf adviceexecution proceed cflowbelow cflow initialization preinitialization "+"staticinitialization withincode target within execution getWithinTypeName handler "+"thisJoinPoint thisJoinPointStaticPart thisEnclosingJoinPointStaticPart declare parents "+"warning error soft precedence thisAspectInstance";var SHORTKEYS="get set args call";return{keywords:KEYWORDS,illegal:/<\/|#/,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"class",beginKeywords:"aspect",end:/[{;=]/,excludeEnd:true,illegal:/[:;"\[\]]/,contains:[{beginKeywords:"extends implements pertypewithin perthis pertarget percflowbelow percflow issingleton"},hljs.UNDERSCORE_TITLE_MODE,{begin:/\([^\)]*/,end:/[)]+/,keywords:KEYWORDS+" "+SHORTKEYS,excludeEnd:false}]},{className:"class",beginKeywords:"class interface",end:/[{;=]/,excludeEnd:true,relevance:0,keywords:"class interface",illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"pointcut after before around throwing returning",end:/[)]/,excludeEnd:false,illegal:/["\[\]]/,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,contains:[hljs.UNDERSCORE_TITLE_MODE]}]},{begin:/[:]/,returnBegin:true,end:/[{;]/,relevance:0,excludeEnd:false,keywords:KEYWORDS,illegal:/["\[\]]/,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",keywords:KEYWORDS+" "+SHORTKEYS,relevance:0},hljs.QUOTE_STRING_MODE]},{beginKeywords:"new throw",relevance:0},{className:"function",begin:/\w+ +\w+(\.)?\w+\s*\([^\)]*\)\s*((throws)[\w\s,]+)?[\{;]/,returnBegin:true,end:/[{;=]/,keywords:KEYWORDS,excludeEnd:true,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/,relevance:0,keywords:KEYWORDS,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("plaintext",function(hljs){return{disableAutodetect:true}});hljs.registerLanguage("xquery",function(hljs){var KEYWORDS="module schema namespace boundary-space preserve no-preserve strip default collation base-uri ordering context decimal-format decimal-separator copy-namespaces empty-sequence except exponent-separator external grouping-separator inherit no-inherit lax minus-sign per-mille percent schema-attribute schema-element strict unordered zero-digit "+"declare import option function validate variable "+"for at in let where order group by return if then else "+"tumbling sliding window start when only end previous next stable "+"ascending descending allowing empty greatest least some every satisfies switch case typeswitch try catch "+"and or to union intersect instance of treat as castable cast map array "+"delete insert into replace value rename copy modify update";var TYPE="item document-node node attribute document element comment namespace namespace-node processing-instruction text construction "+"xs:anyAtomicType xs:untypedAtomic xs:duration xs:time xs:decimal xs:float xs:double xs:gYearMonth xs:gYear xs:gMonthDay xs:gMonth xs:gDay xs:boolean xs:base64Binary xs:hexBinary xs:anyURI xs:QName xs:NOTATION xs:dateTime xs:dateTimeStamp xs:date xs:string xs:normalizedString xs:token xs:language xs:NMTOKEN xs:Name xs:NCName xs:ID xs:IDREF xs:ENTITY xs:integer xs:nonPositiveInteger xs:negativeInteger xs:long xs:int xs:short xs:byte xs:nonNegativeInteger xs:unisignedLong xs:unsignedInt xs:unsignedShort xs:unsignedByte xs:positiveInteger xs:yearMonthDuration xs:dayTimeDuration";var LITERAL="eq ne lt le gt ge is "+"self:: child:: descendant:: descendant-or-self:: attribute:: following:: following-sibling:: parent:: ancestor:: ancestor-or-self:: preceding:: preceding-sibling:: "+"NaN";var BUILT_IN={className:"built_in",variants:[{begin:/\barray\:/,end:/(?:append|filter|flatten|fold\-(?:left|right)|for-each(?:\-pair)?|get|head|insert\-before|join|put|remove|reverse|size|sort|subarray|tail)\b/},{begin:/\bmap\:/,end:/(?:contains|entry|find|for\-each|get|keys|merge|put|remove|size)\b/},{begin:/\bmath\:/,end:/(?:a(?:cos|sin|tan[2]?)|cos|exp(?:10)?|log(?:10)?|pi|pow|sin|sqrt|tan)\b/},{begin:/\bop\:/,end:/\(/,excludeEnd:true},{begin:/\bfn\:/,end:/\(/,excludeEnd:true},{begin:/[^<\/\$\:'"-]\b(?:abs|accumulator\-(?:after|before)|adjust\-(?:date(?:Time)?|time)\-to\-timezone|analyze\-string|apply|available\-(?:environment\-variables|system\-properties)|avg|base\-uri|boolean|ceiling|codepoints?\-(?:equal|to\-string)|collation\-key|collection|compare|concat|contains(?:\-token)?|copy\-of|count|current(?:\-)?(?:date(?:Time)?|time|group(?:ing\-key)?|output\-uri|merge\-(?:group|key))?data|dateTime|days?\-from\-(?:date(?:Time)?|duration)|deep\-equal|default\-(?:collation|language)|distinct\-values|document(?:\-uri)?|doc(?:\-available)?|element\-(?:available|with\-id)|empty|encode\-for\-uri|ends\-with|environment\-variable|error|escape\-html\-uri|exactly\-one|exists|false|filter|floor|fold\-(?:left|right)|for\-each(?:\-pair)?|format\-(?:date(?:Time)?|time|integer|number)|function\-(?:arity|available|lookup|name)|generate\-id|has\-children|head|hours\-from\-(?:dateTime|duration|time)|id(?:ref)?|implicit\-timezone|in\-scope\-prefixes|index\-of|innermost|insert\-before|iri\-to\-uri|json\-(?:doc|to\-xml)|key|lang|last|load\-xquery\-module|local\-name(?:\-from\-QName)?|(?:lower|upper)\-case|matches|max|minutes\-from\-(?:dateTime|duration|time)|min|months?\-from\-(?:date(?:Time)?|duration)|name(?:space\-uri\-?(?:for\-prefix|from\-QName)?)?|nilled|node\-name|normalize\-(?:space|unicode)|not|number|one\-or\-more|outermost|parse\-(?:ietf\-date|json)|path|position|(?:prefix\-from\-)?QName|random\-number\-generator|regex\-group|remove|replace|resolve\-(?:QName|uri)|reverse|root|round(?:\-half\-to\-even)?|seconds\-from\-(?:dateTime|duration|time)|snapshot|sort|starts\-with|static\-base\-uri|stream\-available|string\-?(?:join|length|to\-codepoints)?|subsequence|substring\-?(?:after|before)?|sum|system\-property|tail|timezone\-from\-(?:date(?:Time)?|time)|tokenize|trace|trans(?:form|late)|true|type\-available|unordered|unparsed\-(?:entity|text)?\-?(?:public\-id|uri|available|lines)?|uri\-collection|xml\-to\-json|years?\-from\-(?:date(?:Time)?|duration)|zero\-or\-one)\b/},{begin:/\blocal\:/,end:/\(/,excludeEnd:true},{begin:/\bzip\:/,end:/(?:zip\-file|(?:xml|html|text|binary)\-entry| (?:update\-)?entries)\b/},{begin:/\b(?:util|db|functx|app|xdmp|xmldb)\:/,end:/\(/,excludeEnd:true}]};var TITLE={className:"title",begin:/\bxquery version "[13]\.[01]"\s?(?:encoding ".+")?/,end:/;/};var VAR={className:"variable",begin:/[\$][\w-:]+/};var NUMBER={className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0};var STRING={className:"string",variants:[{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]};var ANNOTATION={className:"meta",begin:/%[\w-:]+/};var COMMENT={className:"comment",begin:"\\(:",end:":\\)",relevance:10,contains:[{className:"doctag",begin:"@\\w+"}]};var COMPUTED={beginKeywords:"element attribute comment document processing-instruction",end:"{",excludeEnd:true};var DIRECT={begin:/<([\w\._:\-]+)((\s*.*)=('|").*('|"))?>/,end:/(\/[\w\._:\-]+>)/,subLanguage:"xml",contains:[{begin:"{",end:"}",subLanguage:"xquery"},"self"]};var CONTAINS=[VAR,BUILT_IN,STRING,NUMBER,COMMENT,ANNOTATION,TITLE,COMPUTED,DIRECT];var METHOD={begin:"{",end:"}",contains:CONTAINS};return{aliases:["xpath","xq"],case_insensitive:false,lexemes:/[a-zA-Z\$][a-zA-Z0-9_:\-]*/,illegal:/(proc)|(abstract)|(extends)|(until)|(#)/,keywords:{keyword:KEYWORDS,type:TYPE,literal:LITERAL},contains:CONTAINS}});hljs.registerLanguage("irpf90",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var F_KEYWORDS={literal:".False. .True.",keyword:"kind do while private call intrinsic where elsewhere "+"type endtype endmodule endselect endinterface end enddo endif if forall endforall only contains default return stop then "+"public subroutine|10 function program .and. .or. .not. .le. .eq. .ge. .gt. .lt. "+"goto save else use module select case "+"access blank direct exist file fmt form formatted iostat name named nextrec number opened rec recl sequential status unformatted unit "+"continue format pause cycle exit "+"c_null_char c_alert c_backspace c_form_feed flush wait decimal round iomsg "+"synchronous nopass non_overridable pass protected volatile abstract extends import "+"non_intrinsic value deferred generic final enumerator class associate bind enum "+"c_int c_short c_long c_long_long c_signed_char c_size_t c_int8_t c_int16_t c_int32_t c_int64_t c_int_least8_t c_int_least16_t "+"c_int_least32_t c_int_least64_t c_int_fast8_t c_int_fast16_t c_int_fast32_t c_int_fast64_t c_intmax_t C_intptr_t c_float c_double "+"c_long_double c_float_complex c_double_complex c_long_double_complex c_bool c_char c_null_ptr c_null_funptr "+"c_new_line c_carriage_return c_horizontal_tab c_vertical_tab iso_c_binding c_loc c_funloc c_associated  c_f_pointer "+"c_ptr c_funptr iso_fortran_env character_storage_size error_unit file_storage_size input_unit iostat_end iostat_eor "+"numeric_storage_size output_unit c_f_procpointer ieee_arithmetic ieee_support_underflow_control "+"ieee_get_underflow_mode ieee_set_underflow_mode newunit contiguous recursive "+"pad position action delim readwrite eor advance nml interface procedure namelist include sequence elemental pure "+"integer real character complex logical dimension allocatable|10 parameter "+"external implicit|10 none double precision assign intent optional pointer "+"target in out common equivalence data "+"begin_provider &begin_provider end_provider begin_shell end_shell begin_template end_template subst assert touch "+"soft_touch provide no_dep free irp_if irp_else irp_endif irp_write irp_read",built_in:"alog alog10 amax0 amax1 amin0 amin1 amod cabs ccos cexp clog csin csqrt dabs dacos dasin datan datan2 dcos dcosh ddim dexp dint "+"dlog dlog10 dmax1 dmin1 dmod dnint dsign dsin dsinh dsqrt dtan dtanh float iabs idim idint idnint ifix isign max0 max1 min0 min1 sngl "+"algama cdabs cdcos cdexp cdlog cdsin cdsqrt cqabs cqcos cqexp cqlog cqsin cqsqrt dcmplx dconjg derf derfc dfloat dgamma dimag dlgama "+"iqint qabs qacos qasin qatan qatan2 qcmplx qconjg qcos qcosh qdim qerf qerfc qexp qgamma qimag qlgama qlog qlog10 qmax1 qmin1 qmod "+"qnint qsign qsin qsinh qsqrt qtan qtanh abs acos aimag aint anint asin atan atan2 char cmplx conjg cos cosh exp ichar index int log "+"log10 max min nint sign sin sinh sqrt tan tanh print write dim lge lgt lle llt mod nullify allocate deallocate "+"adjustl adjustr all allocated any associated bit_size btest ceiling count cshift date_and_time digits dot_product "+"eoshift epsilon exponent floor fraction huge iand ibclr ibits ibset ieor ior ishft ishftc lbound len_trim matmul "+"maxexponent maxloc maxval merge minexponent minloc minval modulo mvbits nearest pack present product "+"radix random_number random_seed range repeat reshape rrspacing scale scan selected_int_kind selected_real_kind "+"set_exponent shape size spacing spread sum system_clock tiny transpose trim ubound unpack verify achar iachar transfer "+"dble entry dprod cpu_time command_argument_count get_command get_command_argument get_environment_variable is_iostat_end "+"ieee_arithmetic ieee_support_underflow_control ieee_get_underflow_mode ieee_set_underflow_mode "+"is_iostat_eor move_alloc new_line selected_char_kind same_type_as extends_type_of"+"acosh asinh atanh bessel_j0 bessel_j1 bessel_jn bessel_y0 bessel_y1 bessel_yn erf erfc erfc_scaled gamma log_gamma hypot norm2 "+"atomic_define atomic_ref execute_command_line leadz trailz storage_size merge_bits "+"bge bgt ble blt dshiftl dshiftr findloc iall iany iparity image_index lcobound ucobound maskl maskr "+"num_images parity popcnt poppar shifta shiftl shiftr this_image "+"IRP_ALIGN irp_here"};return{case_insensitive:true,keywords:F_KEYWORDS,illegal:/\/\*/,contains:[hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"string",relevance:0}),{className:"function",beginKeywords:"subroutine function program",illegal:"[${=\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]},hljs.COMMENT("!","$",{relevance:0}),hljs.COMMENT("begin_doc","end_doc",{relevance:10}),{className:"number",begin:"(?=\\b|\\+|\\-|\\.)(?=\\.\\d|\\d)(?:\\d+)?(?:\\.?\\d*)(?:[de][+-]?\\d+)?\\b\\.?",relevance:0}]}});hljs.registerLanguage("cal",function(hljs){var KEYWORDS="div mod in and or not xor asserterror begin case do downto else end exit for if of repeat then to "+"until while with var";var LITERALS="false true";var COMMENT_MODES=[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/\{/,/\}/,{relevance:0}),hljs.COMMENT(/\(\*/,/\*\)/,{relevance:10})];var STRING={className:"string",begin:/'/,end:/'/,contains:[{begin:/''/}]};var CHAR_STRING={className:"string",begin:/(#\d+)+/};var DATE={className:"number",begin:"\\b\\d+(\\.\\d+)?(DT|D|T)",relevance:0};var DBL_QUOTED_VARIABLE={className:"string",begin:'"',end:'"'};var PROCEDURE={className:"function",beginKeywords:"procedure",end:/[:;]/,keywords:"procedure|10",contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:[STRING,CHAR_STRING]}].concat(COMMENT_MODES)};var OBJECT={className:"class",begin:"OBJECT (Table|Form|Report|Dataport|Codeunit|XMLport|MenuSuite|Page|Query) (\\d+) ([^\\r\\n]+)",returnBegin:true,contains:[hljs.TITLE_MODE,PROCEDURE]};return{case_insensitive:true,keywords:{keyword:KEYWORDS,literal:LITERALS},illegal:/\/\*/,contains:[STRING,CHAR_STRING,DATE,DBL_QUOTED_VARIABLE,hljs.NUMBER_MODE,OBJECT,PROCEDURE]}});hljs.registerLanguage("gherkin",function(hljs){return{aliases:["feature"],keywords:"Feature Background Ability Business Need Scenario Scenarios Scenario Outline Scenario Template Examples Given And Then But When",contains:[{className:"symbol",begin:"\\*",relevance:0},{className:"meta",begin:"@[^@\\s]+"},{begin:"\\|",end:"\\|\\w*$",contains:[{className:"string",begin:"[^|]+"}]},{className:"variable",begin:"<",end:">"},hljs.HASH_COMMENT_MODE,{className:"string",begin:'"""',end:'"""'},hljs.QUOTE_STRING_MODE]}});hljs.registerLanguage("subunit",function(hljs){var DETAILS={className:"string",begin:"\\[\n(multipart)?",end:"\\]\n"};var TIME={className:"string",begin:"\\d{4}-\\d{2}-\\d{2}(\\s+)\\d{2}:\\d{2}:\\d{2}.\\d+Z"};var PROGRESSVALUE={className:"string",begin:"(\\+|-)\\d+"};var KEYWORDS={className:"keyword",relevance:10,variants:[{begin:"^(test|testing|success|successful|failure|error|skip|xfail|uxsuccess)(:?)\\s+(test)?"},{begin:"^progress(:?)(\\s+)?(pop|push)?"},{begin:"^tags:"},{begin:"^time:"}]};return{case_insensitive:true,contains:[DETAILS,TIME,PROGRESSVALUE,KEYWORDS]}});hljs.registerLanguage("stata",function(hljs){return{aliases:["do","ado"],case_insensitive:true,keywords:"if else in foreach for forv forva forval forvalu forvalue forvalues by bys bysort xi quietly qui capture about ac ac_7 acprplot acprplot_7 adjust ado adopath adoupdate alpha ameans an ano anov anova anova_estat anova_terms anovadef aorder ap app appe appen append arch arch_dr arch_estat arch_p archlm areg areg_p args arima arima_dr arima_estat arima_p as asmprobit asmprobit_estat asmprobit_lf asmprobit_mfx__dlg asmprobit_p ass asse asser assert avplot avplot_7 avplots avplots_7 bcskew0 bgodfrey bias binreg bip0_lf biplot bipp_lf bipr_lf bipr_p biprobit bitest bitesti bitowt blogit bmemsize boot bootsamp bootstrap bootstrap_8 boxco_l boxco_p boxcox boxcox_6 boxcox_p bprobit br break brier bro brow brows browse brr brrstat bs bs_7 bsampl_w bsample bsample_7 bsqreg bstat bstat_7 bstat_8 bstrap bstrap_7 bubble bubbleplot ca ca_estat ca_p cabiplot camat canon canon_8 canon_8_p canon_estat canon_p cap caprojection capt captu captur capture cat cc cchart cchart_7 cci cd censobs_table centile cf char chdir checkdlgfiles checkestimationsample checkhlpfiles checksum chelp ci cii cl class classutil clear cli clis clist clo clog clog_lf clog_p clogi clogi_sw clogit clogit_lf clogit_p clogitp clogl_sw cloglog clonevar clslistarray cluster cluster_measures cluster_stop cluster_tree cluster_tree_8 clustermat cmdlog cnr cnre cnreg cnreg_p cnreg_sw cnsreg codebook collaps4 collapse colormult_nb colormult_nw compare compress conf confi confir confirm conren cons const constr constra constrai constrain constraint continue contract copy copyright copysource cor corc corr corr2data corr_anti corr_kmo corr_smc corre correl correla correlat correlate corrgram cou coun count cox cox_p cox_sw coxbase coxhaz coxvar cprplot cprplot_7 crc cret cretu cretur creturn cross cs cscript cscript_log csi ct ct_is ctset ctst_5 ctst_st cttost cumsp cumsp_7 cumul cusum cusum_7 cutil d|0 datasig datasign datasigna datasignat datasignatu datasignatur datasignature datetof db dbeta de dec deco decod decode deff des desc descr descri describ describe destring dfbeta dfgls dfuller di di_g dir dirstats dis discard disp disp_res disp_s displ displa display distinct do doe doed doedi doedit dotplot dotplot_7 dprobit drawnorm drop ds ds_util dstdize duplicates durbina dwstat dydx e|0 ed edi edit egen eivreg emdef en enc enco encod encode eq erase ereg ereg_lf ereg_p ereg_sw ereghet ereghet_glf ereghet_glf_sh ereghet_gp ereghet_ilf ereghet_ilf_sh ereghet_ip eret eretu eretur ereturn err erro error esize est est_cfexist est_cfname est_clickable est_expand est_hold est_table est_unhold est_unholdok estat estat_default estat_summ estat_vce_only esti estimates etodow etof etomdy ex exi exit expand expandcl fac fact facto factor factor_estat factor_p factor_pca_rotated factor_rotate factormat fcast fcast_compute fcast_graph fdades fdadesc fdadescr fdadescri fdadescrib fdadescribe fdasav fdasave fdause fh_st file open file read file close file filefilter fillin find_hlp_file findfile findit findit_7 fit fl fli flis flist for5_0 forest forestplot form forma format fpredict frac_154 frac_adj frac_chk frac_cox frac_ddp frac_dis frac_dv frac_in frac_mun frac_pp frac_pq frac_pv frac_wgt frac_xo fracgen fracplot fracplot_7 fracpoly fracpred fron_ex fron_hn fron_p fron_tn fron_tn2 frontier ftodate ftoe ftomdy ftowdate funnel funnelplot g|0 gamhet_glf gamhet_gp gamhet_ilf gamhet_ip gamma gamma_d2 gamma_p gamma_sw gammahet gdi_hexagon gdi_spokes ge gen gene gener genera generat generate genrank genstd genvmean gettoken gl gladder gladder_7 glim_l01 glim_l02 glim_l03 glim_l04 glim_l05 glim_l06 glim_l07 glim_l08 glim_l09 glim_l10 glim_l11 glim_l12 glim_lf glim_mu glim_nw1 glim_nw2 glim_nw3 glim_p glim_v1 glim_v2 glim_v3 glim_v4 glim_v5 glim_v6 glim_v7 glm glm_6 glm_p glm_sw glmpred glo glob globa global glogit glogit_8 glogit_p gmeans gnbre_lf gnbreg gnbreg_5 gnbreg_p gomp_lf gompe_sw gomper_p gompertz gompertzhet gomphet_glf gomphet_glf_sh gomphet_gp gomphet_ilf gomphet_ilf_sh gomphet_ip gphdot gphpen gphprint gprefs gprobi_p gprobit gprobit_8 gr gr7 gr_copy gr_current gr_db gr_describe gr_dir gr_draw gr_draw_replay gr_drop gr_edit gr_editviewopts gr_example gr_example2 gr_export gr_print gr_qscheme gr_query gr_read gr_rename gr_replay gr_save gr_set gr_setscheme gr_table gr_undo gr_use graph graph7 grebar greigen greigen_7 greigen_8 grmeanby grmeanby_7 gs_fileinfo gs_filetype gs_graphinfo gs_stat gsort gwood h|0 hadimvo hareg hausman haver he heck_d2 heckma_p heckman heckp_lf heckpr_p heckprob hel help hereg hetpr_lf hetpr_p hetprob hettest hexdump hilite hist hist_7 histogram hlogit hlu hmeans hotel hotelling hprobit hreg hsearch icd9 icd9_ff icd9p iis impute imtest inbase include inf infi infil infile infix inp inpu input ins insheet insp inspe inspec inspect integ inten intreg intreg_7 intreg_p intrg2_ll intrg_ll intrg_ll2 ipolate iqreg ir irf irf_create irfm iri is_svy is_svysum isid istdize ivprob_1_lf ivprob_lf ivprobit ivprobit_p ivreg ivreg_footnote ivtob_1_lf ivtob_lf ivtobit ivtobit_p jackknife jacknife jknife jknife_6 jknife_8 jkstat joinby kalarma1 kap kap_3 kapmeier kappa kapwgt kdensity kdensity_7 keep ksm ksmirnov ktau kwallis l|0 la lab labbe labbeplot labe label labelbook ladder levels levelsof leverage lfit lfit_p li lincom line linktest lis list lloghet_glf lloghet_glf_sh lloghet_gp lloghet_ilf lloghet_ilf_sh lloghet_ip llogi_sw llogis_p llogist llogistic llogistichet lnorm_lf lnorm_sw lnorma_p lnormal lnormalhet lnormhet_glf lnormhet_glf_sh lnormhet_gp lnormhet_ilf lnormhet_ilf_sh lnormhet_ip lnskew0 loadingplot loc loca local log logi logis_lf logistic logistic_p logit logit_estat logit_p loglogs logrank loneway lookfor lookup lowess lowess_7 lpredict lrecomp lroc lroc_7 lrtest ls lsens lsens_7 lsens_x lstat ltable ltable_7 ltriang lv lvr2plot lvr2plot_7 m|0 ma mac macr macro makecns man manova manova_estat manova_p manovatest mantel mark markin markout marksample mat mat_capp mat_order mat_put_rr mat_rapp mata mata_clear mata_describe mata_drop mata_matdescribe mata_matsave mata_matuse mata_memory mata_mlib mata_mosave mata_rename mata_which matalabel matcproc matlist matname matr matri matrix matrix_input__dlg matstrik mcc mcci md0_ md1_ md1debug_ md2_ md2debug_ mds mds_estat mds_p mdsconfig mdslong mdsmat mdsshepard mdytoe mdytof me_derd mean means median memory memsize menl meqparse mer merg merge meta mfp mfx mhelp mhodds minbound mixed_ll mixed_ll_reparm mkassert mkdir mkmat mkspline ml ml_5 ml_adjs ml_bhhhs ml_c_d ml_check ml_clear ml_cnt ml_debug ml_defd ml_e0 ml_e0_bfgs ml_e0_cycle ml_e0_dfp ml_e0i ml_e1 ml_e1_bfgs ml_e1_bhhh ml_e1_cycle ml_e1_dfp ml_e2 ml_e2_cycle ml_ebfg0 ml_ebfr0 ml_ebfr1 ml_ebh0q ml_ebhh0 ml_ebhr0 ml_ebr0i ml_ecr0i ml_edfp0 ml_edfr0 ml_edfr1 ml_edr0i ml_eds ml_eer0i ml_egr0i ml_elf ml_elf_bfgs ml_elf_bhhh ml_elf_cycle ml_elf_dfp ml_elfi ml_elfs ml_enr0i ml_enrr0 ml_erdu0 ml_erdu0_bfgs ml_erdu0_bhhh ml_erdu0_bhhhq ml_erdu0_cycle ml_erdu0_dfp ml_erdu0_nrbfgs ml_exde ml_footnote ml_geqnr ml_grad0 ml_graph ml_hbhhh ml_hd0 ml_hold ml_init ml_inv ml_log ml_max ml_mlout ml_mlout_8 ml_model ml_nb0 ml_opt ml_p ml_plot ml_query ml_rdgrd ml_repor ml_s_e ml_score ml_searc ml_technique ml_unhold mleval mlf_ mlmatbysum mlmatsum mlog mlogi mlogit mlogit_footnote mlogit_p mlopts mlsum mlvecsum mnl0_ mor more mov move mprobit mprobit_lf mprobit_p mrdu0_ mrdu1_ mvdecode mvencode mvreg mvreg_estat n|0 nbreg nbreg_al nbreg_lf nbreg_p nbreg_sw nestreg net newey newey_7 newey_p news nl nl_7 nl_9 nl_9_p nl_p nl_p_7 nlcom nlcom_p nlexp2 nlexp2_7 nlexp2a nlexp2a_7 nlexp3 nlexp3_7 nlgom3 nlgom3_7 nlgom4 nlgom4_7 nlinit nllog3 nllog3_7 nllog4 nllog4_7 nlog_rd nlogit nlogit_p nlogitgen nlogittree nlpred no nobreak noi nois noisi noisil noisily note notes notes_dlg nptrend numlabel numlist odbc old_ver olo olog ologi ologi_sw ologit ologit_p ologitp on one onew onewa oneway op_colnm op_comp op_diff op_inv op_str opr opro oprob oprob_sw oprobi oprobi_p oprobit oprobitp opts_exclusive order orthog orthpoly ou out outf outfi outfil outfile outs outsh outshe outshee outsheet ovtest pac pac_7 palette parse parse_dissim pause pca pca_8 pca_display pca_estat pca_p pca_rotate pcamat pchart pchart_7 pchi pchi_7 pcorr pctile pentium pergram pergram_7 permute permute_8 personal peto_st pkcollapse pkcross pkequiv pkexamine pkexamine_7 pkshape pksumm pksumm_7 pl plo plot plugin pnorm pnorm_7 poisgof poiss_lf poiss_sw poisso_p poisson poisson_estat post postclose postfile postutil pperron pr prais prais_e prais_e2 prais_p predict predictnl preserve print pro prob probi probit probit_estat probit_p proc_time procoverlay procrustes procrustes_estat procrustes_p profiler prog progr progra program prop proportion prtest prtesti pwcorr pwd q\\s qby qbys qchi qchi_7 qladder qladder_7 qnorm qnorm_7 qqplot qqplot_7 qreg qreg_c qreg_p qreg_sw qu quadchk quantile quantile_7 que quer query range ranksum ratio rchart rchart_7 rcof recast reclink recode reg reg3 reg3_p regdw regr regre regre_p2 regres regres_p regress regress_estat regriv_p remap ren rena renam rename renpfix repeat replace report reshape restore ret retu retur return rm rmdir robvar roccomp roccomp_7 roccomp_8 rocf_lf rocfit rocfit_8 rocgold rocplot rocplot_7 roctab roctab_7 rolling rologit rologit_p rot rota rotat rotate rotatemat rreg rreg_p ru run runtest rvfplot rvfplot_7 rvpplot rvpplot_7 sa safesum sample sampsi sav save savedresults saveold sc sca scal scala scalar scatter scm_mine sco scob_lf scob_p scobi_sw scobit scor score scoreplot scoreplot_help scree screeplot screeplot_help sdtest sdtesti se search separate seperate serrbar serrbar_7 serset set set_defaults sfrancia sh she shel shell shewhart shewhart_7 signestimationsample signrank signtest simul simul_7 simulate simulate_8 sktest sleep slogit slogit_d2 slogit_p smooth snapspan so sor sort spearman spikeplot spikeplot_7 spikeplt spline_x split sqreg sqreg_p sret sretu sretur sreturn ssc st st_ct st_hc st_hcd st_hcd_sh st_is st_issys st_note st_promo st_set st_show st_smpl st_subid stack statsby statsby_8 stbase stci stci_7 stcox stcox_estat stcox_fr stcox_fr_ll stcox_p stcox_sw stcoxkm stcoxkm_7 stcstat stcurv stcurve stcurve_7 stdes stem stepwise stereg stfill stgen stir stjoin stmc stmh stphplot stphplot_7 stphtest stphtest_7 stptime strate strate_7 streg streg_sw streset sts sts_7 stset stsplit stsum sttocc sttoct stvary stweib su suest suest_8 sum summ summa summar summari summariz summarize sunflower sureg survcurv survsum svar svar_p svmat svy svy_disp svy_dreg svy_est svy_est_7 svy_estat svy_get svy_gnbreg_p svy_head svy_header svy_heckman_p svy_heckprob_p svy_intreg_p svy_ivreg_p svy_logistic_p svy_logit_p svy_mlogit_p svy_nbreg_p svy_ologit_p svy_oprobit_p svy_poisson_p svy_probit_p svy_regress_p svy_sub svy_sub_7 svy_x svy_x_7 svy_x_p svydes svydes_8 svygen svygnbreg svyheckman svyheckprob svyintreg svyintreg_7 svyintrg svyivreg svylc svylog_p svylogit svymarkout svymarkout_8 svymean svymlog svymlogit svynbreg svyolog svyologit svyoprob svyoprobit svyopts svypois svypois_7 svypoisson svyprobit svyprobt svyprop svyprop_7 svyratio svyreg svyreg_p svyregress svyset svyset_7 svyset_8 svytab svytab_7 svytest svytotal sw sw_8 swcnreg swcox swereg swilk swlogis swlogit swologit swoprbt swpois swprobit swqreg swtobit swweib symmetry symmi symplot symplot_7 syntax sysdescribe sysdir sysuse szroeter ta tab tab1 tab2 tab_or tabd tabdi tabdis tabdisp tabi table tabodds tabodds_7 tabstat tabu tabul tabula tabulat tabulate te tempfile tempname tempvar tes test testnl testparm teststd tetrachoric time_it timer tis tob tobi tobit tobit_p tobit_sw token tokeni tokeniz tokenize tostring total translate translator transmap treat_ll treatr_p treatreg trim trimfill trnb_cons trnb_mean trpoiss_d2 trunc_ll truncr_p truncreg tsappend tset tsfill tsline tsline_ex tsreport tsrevar tsrline tsset tssmooth tsunab ttest ttesti tut_chk tut_wait tutorial tw tware_st two twoway twoway__fpfit_serset twoway__function_gen twoway__histogram_gen twoway__ipoint_serset twoway__ipoints_serset twoway__kdensity_gen twoway__lfit_serset twoway__normgen_gen twoway__pci_serset twoway__qfit_serset twoway__scatteri_serset twoway__sunflower_gen twoway_ksm_serset ty typ type typeof u|0 unab unabbrev unabcmd update us use uselabel var var_mkcompanion var_p varbasic varfcast vargranger varirf varirf_add varirf_cgraph varirf_create varirf_ctable varirf_describe varirf_dir varirf_drop varirf_erase varirf_graph varirf_ograph varirf_rename varirf_set varirf_table varlist varlmar varnorm varsoc varstable varstable_w varstable_w2 varwle vce vec vec_fevd vec_mkphi vec_p vec_p_w vecirf_create veclmar veclmar_w vecnorm vecnorm_w vecrank vecstable verinst vers versi versio version view viewsource vif vwls wdatetof webdescribe webseek webuse weib1_lf weib2_lf weib_lf weib_lf0 weibhet_glf weibhet_glf_sh weibhet_glfa weibhet_glfa_sh weibhet_gp weibhet_ilf weibhet_ilf_sh weibhet_ilfa weibhet_ilfa_sh weibhet_ip weibu_sw weibul_p weibull weibull_c weibull_s weibullhet wh whelp whi which whil while wilc_st wilcoxon win wind windo window winexec wntestb wntestb_7 wntestq xchart xchart_7 xcorr xcorr_7 xi xi_6 xmlsav xmlsave xmluse xpose xsh xshe xshel xshell xt_iis xt_tis xtab_p xtabond xtbin_p xtclog xtcloglog xtcloglog_8 xtcloglog_d2 xtcloglog_pa_p xtcloglog_re_p xtcnt_p xtcorr xtdata xtdes xtfront_p xtfrontier xtgee xtgee_elink xtgee_estat xtgee_makeivar xtgee_p xtgee_plink xtgls xtgls_p xthaus xthausman xtht_p xthtaylor xtile xtint_p xtintreg xtintreg_8 xtintreg_d2 xtintreg_p xtivp_1 xtivp_2 xtivreg xtline xtline_ex xtlogit xtlogit_8 xtlogit_d2 xtlogit_fe_p xtlogit_pa_p xtlogit_re_p xtmixed xtmixed_estat xtmixed_p xtnb_fe xtnb_lf xtnbreg xtnbreg_pa_p xtnbreg_refe_p xtpcse xtpcse_p xtpois xtpoisson xtpoisson_d2 xtpoisson_pa_p xtpoisson_refe_p xtpred xtprobit xtprobit_8 xtprobit_d2 xtprobit_re_p xtps_fe xtps_lf xtps_ren xtps_ren_8 xtrar_p xtrc xtrc_p xtrchh xtrefe_p xtreg xtreg_be xtreg_fe xtreg_ml xtreg_pa_p xtreg_re xtregar xtrere_p xtset xtsf_ll xtsf_llti xtsum xttab xttest0 xttobit xttobit_8 xttobit_p xttrans yx yxview__barlike_draw yxview_area_draw yxview_bar_draw yxview_dot_draw yxview_dropline_draw yxview_function_draw yxview_iarrow_draw yxview_ilabels_draw yxview_normal_draw yxview_pcarrow_draw yxview_pcbarrow_draw yxview_pccapsym_draw yxview_pcscatter_draw yxview_pcspike_draw yxview_rarea_draw yxview_rbar_draw yxview_rbarm_draw yxview_rcap_draw yxview_rcapsym_draw yxview_rconnected_draw yxview_rline_draw yxview_rscatter_draw yxview_rspike_draw yxview_spike_draw yxview_sunflower_draw zap_s zinb zinb_llf zinb_plf zip zip_llf zip_p zip_plf zt_ct_5 zt_hc_5 zt_hcd_5 zt_is_5 zt_iss_5 zt_sho_5 zt_smp_5 ztbase_5 ztcox_5 ztdes_5 ztereg_5 ztfill_5 ztgen_5 ztir_5 ztjoin_5 ztnb ztnb_p ztp ztp_p zts_5 ztset_5 ztspli_5 ztsum_5 zttoct_5 ztvary_5 ztweib_5",contains:[{className:"symbol",begin:/`[a-zA-Z0-9_]+'/},{className:"variable",begin:/\$\{?[a-zA-Z0-9_]+\}?/},{className:"string",variants:[{begin:'`"[^\r\n]*?"\''},{begin:'"[^\r\n"]*"'}]},{className:"built_in",variants:[{begin:"\\b(abs|acos|asin|atan|atan2|atanh|ceil|cloglog|comb|cos|digamma|exp|floor|invcloglog|invlogit|ln|lnfact|lnfactorial|lngamma|log|log10|max|min|mod|reldif|round|sign|sin|sqrt|sum|tan|tanh|trigamma|trunc|betaden|Binomial|binorm|binormal|chi2|chi2tail|dgammapda|dgammapdada|dgammapdadx|dgammapdx|dgammapdxdx|F|Fden|Ftail|gammaden|gammap|ibeta|invbinomial|invchi2|invchi2tail|invF|invFtail|invgammap|invibeta|invnchi2|invnFtail|invnibeta|invnorm|invnormal|invttail|nbetaden|nchi2|nFden|nFtail|nibeta|norm|normal|normalden|normd|npnchi2|tden|ttail|uniform|abbrev|char|index|indexnot|length|lower|ltrim|match|plural|proper|real|regexm|regexr|regexs|reverse|rtrim|string|strlen|strlower|strltrim|strmatch|strofreal|strpos|strproper|strreverse|strrtrim|strtrim|strupper|subinstr|subinword|substr|trim|upper|word|wordcount|_caller|autocode|byteorder|chop|clip|cond|e|epsdouble|epsfloat|group|inlist|inrange|irecode|matrix|maxbyte|maxdouble|maxfloat|maxint|maxlong|mi|minbyte|mindouble|minfloat|minint|minlong|missing|r|recode|replay|return|s|scalar|d|date|day|dow|doy|halfyear|mdy|month|quarter|week|year|d|daily|dofd|dofh|dofm|dofq|dofw|dofy|h|halfyearly|hofd|m|mofd|monthly|q|qofd|quarterly|tin|twithin|w|weekly|wofd|y|yearly|yh|ym|yofd|yq|yw|cholesky|colnumb|colsof|corr|det|diag|diag0cnt|el|get|hadamard|I|inv|invsym|issym|issymmetric|J|matmissing|matuniform|mreldif|nullmat|rownumb|rowsof|sweep|syminv|trace|vec|vecdiag)(?=\\()"}]},hljs.COMMENT("^[ \t]*\\*.*$",false),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("scss",function(hljs){var AT_IDENTIFIER="@[a-z-]+";var AT_MODIFIERS="and or not only";var IDENT_RE="[a-zA-Z-][a-zA-Z0-9_-]*";var VARIABLE={className:"variable",begin:"(\\$"+IDENT_RE+")\\b"};var HEXCOLOR={className:"number",begin:"#[0-9A-Fa-f]+"};var DEF_INTERNALS={className:"attribute",begin:"[A-Z\\_\\.\\-]+",end:":",excludeEnd:true,illegal:"[^\\s]",starts:{endsWithParent:true,excludeEnd:true,contains:[HEXCOLOR,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"!important"}]}};return{case_insensitive:true,illegal:"[=/|']",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"selector-id",begin:"\\#[A-Za-z0-9_-]+",relevance:0},{className:"selector-class",begin:"\\.[A-Za-z0-9_-]+",relevance:0},{className:"selector-attr",begin:"\\[",end:"\\]",illegal:"$"},{className:"selector-tag",begin:"\\b(a|abbr|acronym|address|area|article|aside|audio|b|base|big|blockquote|body|br|button|canvas|caption|cite|code|col|colgroup|command|datalist|dd|del|details|dfn|div|dl|dt|em|embed|fieldset|figcaption|figure|footer|form|frame|frameset|(h[1-6])|head|header|hgroup|hr|html|i|iframe|img|input|ins|kbd|keygen|label|legend|li|link|map|mark|meta|meter|nav|noframes|noscript|object|ol|optgroup|option|output|p|param|pre|progress|q|rp|rt|ruby|samp|script|section|select|small|span|strike|strong|style|sub|sup|table|tbody|td|textarea|tfoot|th|thead|time|title|tr|tt|ul|var|video)\\b",relevance:0},{className:"selector-pseudo",begin:":(visited|valid|root|right|required|read-write|read-only|out-range|optional|only-of-type|only-child|nth-of-type|nth-last-of-type|nth-last-child|nth-child|not|link|left|last-of-type|last-child|lang|invalid|indeterminate|in-range|hover|focus|first-of-type|first-line|first-letter|first-child|first|enabled|empty|disabled|default|checked|before|after|active)"},{className:"selector-pseudo",begin:"::(after|before|choices|first-letter|first-line|repeat-index|repeat-item|selection|value)"},VARIABLE,{className:"attribute",begin:"\\b(src|z-index|word-wrap|word-spacing|word-break|width|widows|white-space|visibility|vertical-align|unicode-bidi|transition-timing-function|transition-property|transition-duration|transition-delay|transition|transform-style|transform-origin|transform|top|text-underline-position|text-transform|text-shadow|text-rendering|text-overflow|text-indent|text-decoration-style|text-decoration-line|text-decoration-color|text-decoration|text-align-last|text-align|tab-size|table-layout|right|resize|quotes|position|pointer-events|perspective-origin|perspective|page-break-inside|page-break-before|page-break-after|padding-top|padding-right|padding-left|padding-bottom|padding|overflow-y|overflow-x|overflow-wrap|overflow|outline-width|outline-style|outline-offset|outline-color|outline|orphans|order|opacity|object-position|object-fit|normal|none|nav-up|nav-right|nav-left|nav-index|nav-down|min-width|min-height|max-width|max-height|mask|marks|margin-top|margin-right|margin-left|margin-bottom|margin|list-style-type|list-style-position|list-style-image|list-style|line-height|letter-spacing|left|justify-content|initial|inherit|ime-mode|image-orientation|image-resolution|image-rendering|icon|hyphens|height|font-weight|font-variant-ligatures|font-variant|font-style|font-stretch|font-size-adjust|font-size|font-language-override|font-kerning|font-feature-settings|font-family|font|float|flex-wrap|flex-shrink|flex-grow|flex-flow|flex-direction|flex-basis|flex|filter|empty-cells|display|direction|cursor|counter-reset|counter-increment|content|column-width|column-span|column-rule-width|column-rule-style|column-rule-color|column-rule|column-gap|column-fill|column-count|columns|color|clip-path|clip|clear|caption-side|break-inside|break-before|break-after|box-sizing|box-shadow|box-decoration-break|bottom|border-width|border-top-width|border-top-style|border-top-right-radius|border-top-left-radius|border-top-color|border-top|border-style|border-spacing|border-right-width|border-right-style|border-right-color|border-right|border-radius|border-left-width|border-left-style|border-left-color|border-left|border-image-width|border-image-source|border-image-slice|border-image-repeat|border-image-outset|border-image|border-color|border-collapse|border-bottom-width|border-bottom-style|border-bottom-right-radius|border-bottom-left-radius|border-bottom-color|border-bottom|border|background-size|background-repeat|background-position|background-origin|background-image|background-color|background-clip|background-attachment|background-blend-mode|background|backface-visibility|auto|animation-timing-function|animation-play-state|animation-name|animation-iteration-count|animation-fill-mode|animation-duration|animation-direction|animation-delay|animation|align-self|align-items|align-content)\\b",illegal:"[^\\s]"},{begin:"\\b(whitespace|wait|w-resize|visible|vertical-text|vertical-ideographic|uppercase|upper-roman|upper-alpha|underline|transparent|top|thin|thick|text|text-top|text-bottom|tb-rl|table-header-group|table-footer-group|sw-resize|super|strict|static|square|solid|small-caps|separate|se-resize|scroll|s-resize|rtl|row-resize|ridge|right|repeat|repeat-y|repeat-x|relative|progress|pointer|overline|outside|outset|oblique|nowrap|not-allowed|normal|none|nw-resize|no-repeat|no-drop|newspaper|ne-resize|n-resize|move|middle|medium|ltr|lr-tb|lowercase|lower-roman|lower-alpha|loose|list-item|line|line-through|line-edge|lighter|left|keep-all|justify|italic|inter-word|inter-ideograph|inside|inset|inline|inline-block|inherit|inactive|ideograph-space|ideograph-parenthesis|ideograph-numeric|ideograph-alpha|horizontal|hidden|help|hand|groove|fixed|ellipsis|e-resize|double|dotted|distribute|distribute-space|distribute-letter|distribute-all-lines|disc|disabled|default|decimal|dashed|crosshair|collapse|col-resize|circle|char|center|capitalize|break-word|break-all|bottom|both|bolder|bold|block|bidi-override|below|baseline|auto|always|all-scroll|absolute|table|table-cell)\\b"},{begin:":",end:";",contains:[VARIABLE,HEXCOLOR,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"meta",begin:"!important"}]},{begin:"@(page|font-face)",lexemes:AT_IDENTIFIER,keywords:"@page @font-face"},{begin:"@",end:"[{;]",returnBegin:true,keywords:AT_MODIFIERS,contains:[{begin:AT_IDENTIFIER,className:"keyword"},VARIABLE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,HEXCOLOR,hljs.CSS_NUMBER_MODE]}]}});hljs.registerLanguage("tcl",function(hljs){return{aliases:["tk"],keywords:"after append apply array auto_execok auto_import auto_load auto_mkindex "+"auto_mkindex_old auto_qualify auto_reset bgerror binary break catch cd chan clock "+"close concat continue dde dict encoding eof error eval exec exit expr fblocked "+"fconfigure fcopy file fileevent filename flush for foreach format gets glob global "+"history http if incr info interp join lappend|10 lassign|10 lindex|10 linsert|10 list "+"llength|10 load lrange|10 lrepeat|10 lreplace|10 lreverse|10 lsearch|10 lset|10 lsort|10 "+"mathfunc mathop memory msgcat namespace open package parray pid pkg::create pkg_mkIndex "+"platform platform::shell proc puts pwd read refchan regexp registry regsub|10 rename "+"return safe scan seek set socket source split string subst switch tcl_endOfWord "+"tcl_findLibrary tcl_startOfNextWord tcl_startOfPreviousWord tcl_wordBreakAfter "+"tcl_wordBreakBefore tcltest tclvars tell time tm trace unknown unload unset update "+"uplevel upvar variable vwait while",contains:[hljs.COMMENT(";[ \\t]*#","$"),hljs.COMMENT("^[ \\t]*#","$"),{beginKeywords:"proc",end:"[\\{]",excludeEnd:true,contains:[{className:"title",begin:"[ \\t\\n\\r]+(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*",end:"[ \\t\\n\\r]",endsWithParent:true,excludeEnd:true}]},{excludeEnd:true,variants:[{begin:"\\$(\\{)?(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*\\(([a-zA-Z0-9_])*\\)",end:"[^a-zA-Z0-9_\\}\\$]"},{begin:"\\$(\\{)?(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*",end:"(\\))?[^a-zA-Z0-9_\\}\\$]"}]},{className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]},{className:"number",variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]}]}});hljs.registerLanguage("pgsql",function(hljs){var COMMENT_MODE=hljs.COMMENT("--","$");var UNQUOTED_IDENT="[a-zA-Z_][a-zA-Z_0-9$]*";var DOLLAR_STRING="\\$([a-zA-Z_]?|[a-zA-Z_][a-zA-Z_0-9]*)\\$";var LABEL="<<\\s*"+UNQUOTED_IDENT+"\\s*>>";var SQL_KW="ABORT ALTER ANALYZE BEGIN CALL CHECKPOINT|10 CLOSE CLUSTER COMMENT COMMIT COPY CREATE DEALLOCATE DECLARE "+"DELETE DISCARD DO DROP END EXECUTE EXPLAIN FETCH GRANT IMPORT INSERT LISTEN LOAD LOCK MOVE NOTIFY "+"PREPARE REASSIGN|10 REFRESH REINDEX RELEASE RESET REVOKE ROLLBACK SAVEPOINT SECURITY SELECT SET SHOW "+"START TRUNCATE UNLISTEN|10 UPDATE VACUUM|10 VALUES "+"AGGREGATE COLLATION CONVERSION|10 DATABASE DEFAULT PRIVILEGES DOMAIN TRIGGER EXTENSION FOREIGN "+"WRAPPER|10 TABLE FUNCTION GROUP LANGUAGE LARGE OBJECT MATERIALIZED VIEW OPERATOR CLASS "+"FAMILY POLICY PUBLICATION|10 ROLE RULE SCHEMA SEQUENCE SERVER STATISTICS SUBSCRIPTION SYSTEM "+"TABLESPACE CONFIGURATION DICTIONARY PARSER TEMPLATE TYPE USER MAPPING PREPARED ACCESS "+"METHOD CAST AS TRANSFORM TRANSACTION OWNED TO INTO SESSION AUTHORIZATION "+"INDEX PROCEDURE ASSERTION "+"ALL ANALYSE AND ANY ARRAY ASC ASYMMETRIC|10 BOTH CASE CHECK "+"COLLATE COLUMN CONCURRENTLY|10 CONSTRAINT CROSS "+"DEFERRABLE RANGE "+"DESC DISTINCT ELSE EXCEPT FOR FREEZE|10 FROM FULL HAVING "+"ILIKE IN INITIALLY INNER INTERSECT IS ISNULL JOIN LATERAL LEADING LIKE LIMIT "+"NATURAL NOT NOTNULL NULL OFFSET ON ONLY OR ORDER OUTER OVERLAPS PLACING PRIMARY "+"REFERENCES RETURNING SIMILAR SOME SYMMETRIC TABLESAMPLE THEN "+"TRAILING UNION UNIQUE USING VARIADIC|10 VERBOSE WHEN WHERE WINDOW WITH "+"BY RETURNS INOUT OUT SETOF|10 IF STRICT CURRENT CONTINUE OWNER LOCATION OVER PARTITION WITHIN "+"BETWEEN ESCAPE EXTERNAL INVOKER DEFINER WORK RENAME VERSION CONNECTION CONNECT "+"TABLES TEMP TEMPORARY FUNCTIONS SEQUENCES TYPES SCHEMAS OPTION CASCADE RESTRICT ADD ADMIN "+"EXISTS VALID VALIDATE ENABLE DISABLE REPLICA|10 ALWAYS PASSING COLUMNS PATH "+"REF VALUE OVERRIDING IMMUTABLE STABLE VOLATILE BEFORE AFTER EACH ROW PROCEDURAL "+"ROUTINE NO HANDLER VALIDATOR OPTIONS STORAGE OIDS|10 WITHOUT INHERIT DEPENDS CALLED "+"INPUT LEAKPROOF|10 COST ROWS NOWAIT SEARCH UNTIL ENCRYPTED|10 PASSWORD CONFLICT|10 "+"INSTEAD INHERITS CHARACTERISTICS WRITE CURSOR ALSO STATEMENT SHARE EXCLUSIVE INLINE "+"ISOLATION REPEATABLE READ COMMITTED SERIALIZABLE UNCOMMITTED LOCAL GLOBAL SQL PROCEDURES "+"RECURSIVE SNAPSHOT ROLLUP CUBE TRUSTED|10 INCLUDE FOLLOWING PRECEDING UNBOUNDED RANGE GROUPS "+"UNENCRYPTED|10 SYSID FORMAT DELIMITER HEADER QUOTE ENCODING FILTER OFF "+"FORCE_QUOTE FORCE_NOT_NULL FORCE_NULL COSTS BUFFERS TIMING SUMMARY DISABLE_PAGE_SKIPPING "+"RESTART CYCLE GENERATED IDENTITY DEFERRED IMMEDIATE LEVEL LOGGED UNLOGGED "+"OF NOTHING NONE EXCLUDE ATTRIBUTE "+"USAGE ROUTINES "+"TRUE FALSE NAN INFINITY ";var ROLE_ATTRS="SUPERUSER NOSUPERUSER CREATEDB NOCREATEDB CREATEROLE NOCREATEROLE INHERIT NOINHERIT "+"LOGIN NOLOGIN REPLICATION NOREPLICATION BYPASSRLS NOBYPASSRLS ";var PLPGSQL_KW="ALIAS BEGIN CONSTANT DECLARE END EXCEPTION RETURN PERFORM|10 RAISE GET DIAGNOSTICS "+"STACKED|10 FOREACH LOOP ELSIF EXIT WHILE REVERSE SLICE DEBUG LOG INFO NOTICE WARNING ASSERT "+"OPEN ";var TYPES="BIGINT INT8 BIGSERIAL SERIAL8 BIT VARYING VARBIT BOOLEAN BOOL BOX BYTEA CHARACTER CHAR VARCHAR "+"CIDR CIRCLE DATE DOUBLE PRECISION FLOAT8 FLOAT INET INTEGER INT INT4 INTERVAL JSON JSONB LINE LSEG|10 "+"MACADDR MACADDR8 MONEY NUMERIC DEC DECIMAL PATH POINT POLYGON REAL FLOAT4 SMALLINT INT2 "+"SMALLSERIAL|10 SERIAL2|10 SERIAL|10 SERIAL4|10 TEXT TIME ZONE TIMETZ|10 TIMESTAMP TIMESTAMPTZ|10 TSQUERY|10 TSVECTOR|10 "+"TXID_SNAPSHOT|10 UUID XML NATIONAL NCHAR "+"INT4RANGE|10 INT8RANGE|10 NUMRANGE|10 TSRANGE|10 TSTZRANGE|10 DATERANGE|10 "+"ANYELEMENT ANYARRAY ANYNONARRAY ANYENUM ANYRANGE CSTRING INTERNAL "+"RECORD PG_DDL_COMMAND VOID UNKNOWN OPAQUE REFCURSOR "+"NAME "+"OID REGPROC|10 REGPROCEDURE|10 REGOPER|10 REGOPERATOR|10 REGCLASS|10 REGTYPE|10 REGROLE|10 "+"REGNAMESPACE|10 REGCONFIG|10 REGDICTIONARY|10 ";"HSTORE|10 LO LTREE|10 ";var TYPES_RE=TYPES.trim().split(" ").map(function(val){return val.split("|")[0]}).join("|");var SQL_BI="CURRENT_TIME CURRENT_TIMESTAMP CURRENT_USER CURRENT_CATALOG|10 CURRENT_DATE LOCALTIME LOCALTIMESTAMP "+"CURRENT_ROLE|10 CURRENT_SCHEMA|10 SESSION_USER PUBLIC ";var PLPGSQL_BI="FOUND NEW OLD TG_NAME|10 TG_WHEN|10 TG_LEVEL|10 TG_OP|10 TG_RELID|10 TG_RELNAME|10 "+"TG_TABLE_NAME|10 TG_TABLE_SCHEMA|10 TG_NARGS|10 TG_ARGV|10 TG_EVENT|10 TG_TAG|10 "+"ROW_COUNT RESULT_OID|10 PG_CONTEXT|10 RETURNED_SQLSTATE COLUMN_NAME CONSTRAINT_NAME "+"PG_DATATYPE_NAME|10 MESSAGE_TEXT TABLE_NAME SCHEMA_NAME PG_EXCEPTION_DETAIL|10 "+"PG_EXCEPTION_HINT|10 PG_EXCEPTION_CONTEXT|10 ";var PLPGSQL_EXCEPTIONS="SQLSTATE SQLERRM|10 "+"SUCCESSFUL_COMPLETION WARNING DYNAMIC_RESULT_SETS_RETURNED IMPLICIT_ZERO_BIT_PADDING "+"NULL_VALUE_ELIMINATED_IN_SET_FUNCTION PRIVILEGE_NOT_GRANTED PRIVILEGE_NOT_REVOKED "+"STRING_DATA_RIGHT_TRUNCATION DEPRECATED_FEATURE NO_DATA NO_ADDITIONAL_DYNAMIC_RESULT_SETS_RETURNED "+"SQL_STATEMENT_NOT_YET_COMPLETE CONNECTION_EXCEPTION CONNECTION_DOES_NOT_EXIST CONNECTION_FAILURE "+"SQLCLIENT_UNABLE_TO_ESTABLISH_SQLCONNECTION SQLSERVER_REJECTED_ESTABLISHMENT_OF_SQLCONNECTION "+"TRANSACTION_RESOLUTION_UNKNOWN PROTOCOL_VIOLATION TRIGGERED_ACTION_EXCEPTION FEATURE_NOT_SUPPORTED "+"INVALID_TRANSACTION_INITIATION LOCATOR_EXCEPTION INVALID_LOCATOR_SPECIFICATION INVALID_GRANTOR "+"INVALID_GRANT_OPERATION INVALID_ROLE_SPECIFICATION DIAGNOSTICS_EXCEPTION "+"STACKED_DIAGNOSTICS_ACCESSED_WITHOUT_ACTIVE_HANDLER CASE_NOT_FOUND CARDINALITY_VIOLATION "+"DATA_EXCEPTION ARRAY_SUBSCRIPT_ERROR CHARACTER_NOT_IN_REPERTOIRE DATETIME_FIELD_OVERFLOW "+"DIVISION_BY_ZERO ERROR_IN_ASSIGNMENT ESCAPE_CHARACTER_CONFLICT INDICATOR_OVERFLOW "+"INTERVAL_FIELD_OVERFLOW INVALID_ARGUMENT_FOR_LOGARITHM INVALID_ARGUMENT_FOR_NTILE_FUNCTION "+"INVALID_ARGUMENT_FOR_NTH_VALUE_FUNCTION INVALID_ARGUMENT_FOR_POWER_FUNCTION "+"INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION INVALID_CHARACTER_VALUE_FOR_CAST "+"INVALID_DATETIME_FORMAT INVALID_ESCAPE_CHARACTER INVALID_ESCAPE_OCTET INVALID_ESCAPE_SEQUENCE "+"NONSTANDARD_USE_OF_ESCAPE_CHARACTER INVALID_INDICATOR_PARAMETER_VALUE INVALID_PARAMETER_VALUE "+"INVALID_REGULAR_EXPRESSION INVALID_ROW_COUNT_IN_LIMIT_CLAUSE "+"INVALID_ROW_COUNT_IN_RESULT_OFFSET_CLAUSE INVALID_TABLESAMPLE_ARGUMENT INVALID_TABLESAMPLE_REPEAT "+"INVALID_TIME_ZONE_DISPLACEMENT_VALUE INVALID_USE_OF_ESCAPE_CHARACTER MOST_SPECIFIC_TYPE_MISMATCH "+"NULL_VALUE_NOT_ALLOWED NULL_VALUE_NO_INDICATOR_PARAMETER NUMERIC_VALUE_OUT_OF_RANGE "+"SEQUENCE_GENERATOR_LIMIT_EXCEEDED STRING_DATA_LENGTH_MISMATCH STRING_DATA_RIGHT_TRUNCATION "+"SUBSTRING_ERROR TRIM_ERROR UNTERMINATED_C_STRING ZERO_LENGTH_CHARACTER_STRING "+"FLOATING_POINT_EXCEPTION INVALID_TEXT_REPRESENTATION INVALID_BINARY_REPRESENTATION "+"BAD_COPY_FILE_FORMAT UNTRANSLATABLE_CHARACTER NOT_AN_XML_DOCUMENT INVALID_XML_DOCUMENT "+"INVALID_XML_CONTENT INVALID_XML_COMMENT INVALID_XML_PROCESSING_INSTRUCTION "+"INTEGRITY_CONSTRAINT_VIOLATION RESTRICT_VIOLATION NOT_NULL_VIOLATION FOREIGN_KEY_VIOLATION "+"UNIQUE_VIOLATION CHECK_VIOLATION EXCLUSION_VIOLATION INVALID_CURSOR_STATE "+"INVALID_TRANSACTION_STATE ACTIVE_SQL_TRANSACTION BRANCH_TRANSACTION_ALREADY_ACTIVE "+"HELD_CURSOR_REQUIRES_SAME_ISOLATION_LEVEL INAPPROPRIATE_ACCESS_MODE_FOR_BRANCH_TRANSACTION "+"INAPPROPRIATE_ISOLATION_LEVEL_FOR_BRANCH_TRANSACTION "+"NO_ACTIVE_SQL_TRANSACTION_FOR_BRANCH_TRANSACTION READ_ONLY_SQL_TRANSACTION "+"SCHEMA_AND_DATA_STATEMENT_MIXING_NOT_SUPPORTED NO_ACTIVE_SQL_TRANSACTION "+"IN_FAILED_SQL_TRANSACTION IDLE_IN_TRANSACTION_SESSION_TIMEOUT INVALID_SQL_STATEMENT_NAME "+"TRIGGERED_DATA_CHANGE_VIOLATION INVALID_AUTHORIZATION_SPECIFICATION INVALID_PASSWORD "+"DEPENDENT_PRIVILEGE_DESCRIPTORS_STILL_EXIST DEPENDENT_OBJECTS_STILL_EXIST "+"INVALID_TRANSACTION_TERMINATION SQL_ROUTINE_EXCEPTION FUNCTION_EXECUTED_NO_RETURN_STATEMENT "+"MODIFYING_SQL_DATA_NOT_PERMITTED PROHIBITED_SQL_STATEMENT_ATTEMPTED "+"READING_SQL_DATA_NOT_PERMITTED INVALID_CURSOR_NAME EXTERNAL_ROUTINE_EXCEPTION "+"CONTAINING_SQL_NOT_PERMITTED MODIFYING_SQL_DATA_NOT_PERMITTED "+"PROHIBITED_SQL_STATEMENT_ATTEMPTED READING_SQL_DATA_NOT_PERMITTED "+"EXTERNAL_ROUTINE_INVOCATION_EXCEPTION INVALID_SQLSTATE_RETURNED NULL_VALUE_NOT_ALLOWED "+"TRIGGER_PROTOCOL_VIOLATED SRF_PROTOCOL_VIOLATED EVENT_TRIGGER_PROTOCOL_VIOLATED "+"SAVEPOINT_EXCEPTION INVALID_SAVEPOINT_SPECIFICATION INVALID_CATALOG_NAME "+"INVALID_SCHEMA_NAME TRANSACTION_ROLLBACK TRANSACTION_INTEGRITY_CONSTRAINT_VIOLATION "+"SERIALIZATION_FAILURE STATEMENT_COMPLETION_UNKNOWN DEADLOCK_DETECTED "+"SYNTAX_ERROR_OR_ACCESS_RULE_VIOLATION SYNTAX_ERROR INSUFFICIENT_PRIVILEGE CANNOT_COERCE "+"GROUPING_ERROR WINDOWING_ERROR INVALID_RECURSION INVALID_FOREIGN_KEY INVALID_NAME "+"NAME_TOO_LONG RESERVED_NAME DATATYPE_MISMATCH INDETERMINATE_DATATYPE COLLATION_MISMATCH "+"INDETERMINATE_COLLATION WRONG_OBJECT_TYPE GENERATED_ALWAYS UNDEFINED_COLUMN "+"UNDEFINED_FUNCTION UNDEFINED_TABLE UNDEFINED_PARAMETER UNDEFINED_OBJECT "+"DUPLICATE_COLUMN DUPLICATE_CURSOR DUPLICATE_DATABASE DUPLICATE_FUNCTION "+"DUPLICATE_PREPARED_STATEMENT DUPLICATE_SCHEMA DUPLICATE_TABLE DUPLICATE_ALIAS "+"DUPLICATE_OBJECT AMBIGUOUS_COLUMN AMBIGUOUS_FUNCTION AMBIGUOUS_PARAMETER AMBIGUOUS_ALIAS "+"INVALID_COLUMN_REFERENCE INVALID_COLUMN_DEFINITION INVALID_CURSOR_DEFINITION "+"INVALID_DATABASE_DEFINITION INVALID_FUNCTION_DEFINITION "+"INVALID_PREPARED_STATEMENT_DEFINITION INVALID_SCHEMA_DEFINITION INVALID_TABLE_DEFINITION "+"INVALID_OBJECT_DEFINITION WITH_CHECK_OPTION_VIOLATION INSUFFICIENT_RESOURCES DISK_FULL "+"OUT_OF_MEMORY TOO_MANY_CONNECTIONS CONFIGURATION_LIMIT_EXCEEDED PROGRAM_LIMIT_EXCEEDED "+"STATEMENT_TOO_COMPLEX TOO_MANY_COLUMNS TOO_MANY_ARGUMENTS OBJECT_NOT_IN_PREREQUISITE_STATE "+"OBJECT_IN_USE CANT_CHANGE_RUNTIME_PARAM LOCK_NOT_AVAILABLE OPERATOR_INTERVENTION "+"QUERY_CANCELED ADMIN_SHUTDOWN CRASH_SHUTDOWN CANNOT_CONNECT_NOW DATABASE_DROPPED "+"SYSTEM_ERROR IO_ERROR UNDEFINED_FILE DUPLICATE_FILE SNAPSHOT_TOO_OLD CONFIG_FILE_ERROR "+"LOCK_FILE_EXISTS FDW_ERROR FDW_COLUMN_NAME_NOT_FOUND FDW_DYNAMIC_PARAMETER_VALUE_NEEDED "+"FDW_FUNCTION_SEQUENCE_ERROR FDW_INCONSISTENT_DESCRIPTOR_INFORMATION "+"FDW_INVALID_ATTRIBUTE_VALUE FDW_INVALID_COLUMN_NAME FDW_INVALID_COLUMN_NUMBER "+"FDW_INVALID_DATA_TYPE FDW_INVALID_DATA_TYPE_DESCRIPTORS "+"FDW_INVALID_DESCRIPTOR_FIELD_IDENTIFIER FDW_INVALID_HANDLE FDW_INVALID_OPTION_INDEX "+"FDW_INVALID_OPTION_NAME FDW_INVALID_STRING_LENGTH_OR_BUFFER_LENGTH "+"FDW_INVALID_STRING_FORMAT FDW_INVALID_USE_OF_NULL_POINTER FDW_TOO_MANY_HANDLES "+"FDW_OUT_OF_MEMORY FDW_NO_SCHEMAS FDW_OPTION_NAME_NOT_FOUND FDW_REPLY_HANDLE "+"FDW_SCHEMA_NOT_FOUND FDW_TABLE_NOT_FOUND FDW_UNABLE_TO_CREATE_EXECUTION "+"FDW_UNABLE_TO_CREATE_REPLY FDW_UNABLE_TO_ESTABLISH_CONNECTION PLPGSQL_ERROR "+"RAISE_EXCEPTION NO_DATA_FOUND TOO_MANY_ROWS ASSERT_FAILURE INTERNAL_ERROR DATA_CORRUPTED "+"INDEX_CORRUPTED ";var FUNCTIONS="ARRAY_AGG AVG BIT_AND BIT_OR BOOL_AND BOOL_OR COUNT EVERY JSON_AGG JSONB_AGG JSON_OBJECT_AGG "+"JSONB_OBJECT_AGG MAX MIN MODE STRING_AGG SUM XMLAGG "+"CORR COVAR_POP COVAR_SAMP REGR_AVGX REGR_AVGY REGR_COUNT REGR_INTERCEPT REGR_R2 REGR_SLOPE "+"REGR_SXX REGR_SXY REGR_SYY STDDEV STDDEV_POP STDDEV_SAMP VARIANCE VAR_POP VAR_SAMP "+"PERCENTILE_CONT PERCENTILE_DISC "+"ROW_NUMBER RANK DENSE_RANK PERCENT_RANK CUME_DIST NTILE LAG LEAD FIRST_VALUE LAST_VALUE NTH_VALUE "+"NUM_NONNULLS NUM_NULLS "+"ABS CBRT CEIL CEILING DEGREES DIV EXP FLOOR LN LOG MOD PI POWER RADIANS ROUND SCALE SIGN SQRT "+"TRUNC WIDTH_BUCKET "+"RANDOM SETSEED "+"ACOS ACOSD ASIN ASIND ATAN ATAND ATAN2 ATAN2D COS COSD COT COTD SIN SIND TAN TAND "+"BIT_LENGTH CHAR_LENGTH CHARACTER_LENGTH LOWER OCTET_LENGTH OVERLAY POSITION SUBSTRING TREAT TRIM UPPER "+"ASCII BTRIM CHR CONCAT CONCAT_WS CONVERT CONVERT_FROM CONVERT_TO DECODE ENCODE INITCAP"+"LEFT LENGTH LPAD LTRIM MD5 PARSE_IDENT PG_CLIENT_ENCODING QUOTE_IDENT|10 QUOTE_LITERAL|10 "+"QUOTE_NULLABLE|10 REGEXP_MATCH REGEXP_MATCHES REGEXP_REPLACE REGEXP_SPLIT_TO_ARRAY "+"REGEXP_SPLIT_TO_TABLE REPEAT REPLACE REVERSE RIGHT RPAD RTRIM SPLIT_PART STRPOS SUBSTR "+"TO_ASCII TO_HEX TRANSLATE "+"OCTET_LENGTH GET_BIT GET_BYTE SET_BIT SET_BYTE "+"TO_CHAR TO_DATE TO_NUMBER TO_TIMESTAMP "+"AGE CLOCK_TIMESTAMP|10 DATE_PART DATE_TRUNC ISFINITE JUSTIFY_DAYS JUSTIFY_HOURS JUSTIFY_INTERVAL "+"MAKE_DATE MAKE_INTERVAL|10 MAKE_TIME MAKE_TIMESTAMP|10 MAKE_TIMESTAMPTZ|10 NOW STATEMENT_TIMESTAMP|10 "+"TIMEOFDAY TRANSACTION_TIMESTAMP|10 "+"ENUM_FIRST ENUM_LAST ENUM_RANGE "+"AREA CENTER DIAMETER HEIGHT ISCLOSED ISOPEN NPOINTS PCLOSE POPEN RADIUS WIDTH "+"BOX BOUND_BOX CIRCLE LINE LSEG PATH POLYGON "+"ABBREV BROADCAST HOST HOSTMASK MASKLEN NETMASK NETWORK SET_MASKLEN TEXT INET_SAME_FAMILY"+"INET_MERGE MACADDR8_SET7BIT "+"ARRAY_TO_TSVECTOR GET_CURRENT_TS_CONFIG NUMNODE PLAINTO_TSQUERY PHRASETO_TSQUERY WEBSEARCH_TO_TSQUERY "+"QUERYTREE SETWEIGHT STRIP TO_TSQUERY TO_TSVECTOR JSON_TO_TSVECTOR JSONB_TO_TSVECTOR TS_DELETE "+"TS_FILTER TS_HEADLINE TS_RANK TS_RANK_CD TS_REWRITE TSQUERY_PHRASE TSVECTOR_TO_ARRAY "+"TSVECTOR_UPDATE_TRIGGER TSVECTOR_UPDATE_TRIGGER_COLUMN "+"XMLCOMMENT XMLCONCAT XMLELEMENT XMLFOREST XMLPI XMLROOT "+"XMLEXISTS XML_IS_WELL_FORMED XML_IS_WELL_FORMED_DOCUMENT XML_IS_WELL_FORMED_CONTENT "+"XPATH XPATH_EXISTS XMLTABLE XMLNAMESPACES "+"TABLE_TO_XML TABLE_TO_XMLSCHEMA TABLE_TO_XML_AND_XMLSCHEMA "+"QUERY_TO_XML QUERY_TO_XMLSCHEMA QUERY_TO_XML_AND_XMLSCHEMA "+"CURSOR_TO_XML CURSOR_TO_XMLSCHEMA "+"SCHEMA_TO_XML SCHEMA_TO_XMLSCHEMA SCHEMA_TO_XML_AND_XMLSCHEMA "+"DATABASE_TO_XML DATABASE_TO_XMLSCHEMA DATABASE_TO_XML_AND_XMLSCHEMA "+"XMLATTRIBUTES "+"TO_JSON TO_JSONB ARRAY_TO_JSON ROW_TO_JSON JSON_BUILD_ARRAY JSONB_BUILD_ARRAY JSON_BUILD_OBJECT "+"JSONB_BUILD_OBJECT JSON_OBJECT JSONB_OBJECT JSON_ARRAY_LENGTH JSONB_ARRAY_LENGTH JSON_EACH "+"JSONB_EACH JSON_EACH_TEXT JSONB_EACH_TEXT JSON_EXTRACT_PATH JSONB_EXTRACT_PATH "+"JSON_OBJECT_KEYS JSONB_OBJECT_KEYS JSON_POPULATE_RECORD JSONB_POPULATE_RECORD JSON_POPULATE_RECORDSET "+"JSONB_POPULATE_RECORDSET JSON_ARRAY_ELEMENTS JSONB_ARRAY_ELEMENTS JSON_ARRAY_ELEMENTS_TEXT "+"JSONB_ARRAY_ELEMENTS_TEXT JSON_TYPEOF JSONB_TYPEOF JSON_TO_RECORD JSONB_TO_RECORD JSON_TO_RECORDSET "+"JSONB_TO_RECORDSET JSON_STRIP_NULLS JSONB_STRIP_NULLS JSONB_SET JSONB_INSERT JSONB_PRETTY "+"CURRVAL LASTVAL NEXTVAL SETVAL "+"COALESCE NULLIF GREATEST LEAST "+"ARRAY_APPEND ARRAY_CAT ARRAY_NDIMS ARRAY_DIMS ARRAY_FILL ARRAY_LENGTH ARRAY_LOWER ARRAY_POSITION "+"ARRAY_POSITIONS ARRAY_PREPEND ARRAY_REMOVE ARRAY_REPLACE ARRAY_TO_STRING ARRAY_UPPER CARDINALITY "+"STRING_TO_ARRAY UNNEST "+"ISEMPTY LOWER_INC UPPER_INC LOWER_INF UPPER_INF RANGE_MERGE "+"GENERATE_SERIES GENERATE_SUBSCRIPTS "+"CURRENT_DATABASE CURRENT_QUERY CURRENT_SCHEMA|10 CURRENT_SCHEMAS|10 INET_CLIENT_ADDR INET_CLIENT_PORT "+"INET_SERVER_ADDR INET_SERVER_PORT ROW_SECURITY_ACTIVE FORMAT_TYPE "+"TO_REGCLASS TO_REGPROC TO_REGPROCEDURE TO_REGOPER TO_REGOPERATOR TO_REGTYPE TO_REGNAMESPACE TO_REGROLE "+"COL_DESCRIPTION OBJ_DESCRIPTION SHOBJ_DESCRIPTION "+"TXID_CURRENT TXID_CURRENT_IF_ASSIGNED TXID_CURRENT_SNAPSHOT TXID_SNAPSHOT_XIP TXID_SNAPSHOT_XMAX "+"TXID_SNAPSHOT_XMIN TXID_VISIBLE_IN_SNAPSHOT TXID_STATUS "+"CURRENT_SETTING SET_CONFIG BRIN_SUMMARIZE_NEW_VALUES BRIN_SUMMARIZE_RANGE BRIN_DESUMMARIZE_RANGE "+"GIN_CLEAN_PENDING_LIST "+"SUPPRESS_REDUNDANT_UPDATES_TRIGGER "+"LO_FROM_BYTEA LO_PUT LO_GET LO_CREAT LO_CREATE LO_UNLINK LO_IMPORT LO_EXPORT LOREAD LOWRITE "+"GROUPING CAST ";var FUNCTIONS_RE=FUNCTIONS.trim().split(" ").map(function(val){return val.split("|")[0]}).join("|");return{aliases:["postgres","postgresql"],case_insensitive:true,keywords:{keyword:SQL_KW+PLPGSQL_KW+ROLE_ATTRS,built_in:SQL_BI+PLPGSQL_BI+PLPGSQL_EXCEPTIONS},illegal:/:==|\W\s*\(\*|(^|\s)\$[a-z]|{{|[a-z]:\s*$|\.\.\.|TO:|DO:/,contains:[{className:"keyword",variants:[{begin:/\bTEXT\s*SEARCH\b/},{begin:/\b(PRIMARY|FOREIGN|FOR(\s+NO)?)\s+KEY\b/},{begin:/\bPARALLEL\s+(UNSAFE|RESTRICTED|SAFE)\b/},{begin:/\bSTORAGE\s+(PLAIN|EXTERNAL|EXTENDED|MAIN)\b/},{begin:/\bMATCH\s+(FULL|PARTIAL|SIMPLE)\b/},{begin:/\bNULLS\s+(FIRST|LAST)\b/},{begin:/\bEVENT\s+TRIGGER\b/},{begin:/\b(MAPPING|OR)\s+REPLACE\b/},{begin:/\b(FROM|TO)\s+(PROGRAM|STDIN|STDOUT)\b/},{begin:/\b(SHARE|EXCLUSIVE)\s+MODE\b/},{begin:/\b(LEFT|RIGHT)\s+(OUTER\s+)?JOIN\b/},{begin:/\b(FETCH|MOVE)\s+(NEXT|PRIOR|FIRST|LAST|ABSOLUTE|RELATIVE|FORWARD|BACKWARD)\b/},{begin:/\bPRESERVE\s+ROWS\b/},{begin:/\bDISCARD\s+PLANS\b/},{begin:/\bREFERENCING\s+(OLD|NEW)\b/},{begin:/\bSKIP\s+LOCKED\b/},{begin:/\bGROUPING\s+SETS\b/},{begin:/\b(BINARY|INSENSITIVE|SCROLL|NO\s+SCROLL)\s+(CURSOR|FOR)\b/},{begin:/\b(WITH|WITHOUT)\s+HOLD\b/},{begin:/\bWITH\s+(CASCADED|LOCAL)\s+CHECK\s+OPTION\b/},{begin:/\bEXCLUDE\s+(TIES|NO\s+OTHERS)\b/},{begin:/\bFORMAT\s+(TEXT|XML|JSON|YAML)\b/},{begin:/\bSET\s+((SESSION|LOCAL)\s+)?NAMES\b/},{begin:/\bIS\s+(NOT\s+)?UNKNOWN\b/},{begin:/\bSECURITY\s+LABEL\b/},{begin:/\bSTANDALONE\s+(YES|NO|NO\s+VALUE)\b/},{begin:/\bWITH\s+(NO\s+)?DATA\b/},{begin:/\b(FOREIGN|SET)\s+DATA\b/},{begin:/\bSET\s+(CATALOG|CONSTRAINTS)\b/},{begin:/\b(WITH|FOR)\s+ORDINALITY\b/},{begin:/\bIS\s+(NOT\s+)?DOCUMENT\b/},{begin:/\bXML\s+OPTION\s+(DOCUMENT|CONTENT)\b/},{begin:/\b(STRIP|PRESERVE)\s+WHITESPACE\b/},{begin:/\bNO\s+(ACTION|MAXVALUE|MINVALUE)\b/},{begin:/\bPARTITION\s+BY\s+(RANGE|LIST|HASH)\b/},{begin:/\bAT\s+TIME\s+ZONE\b/},{begin:/\bGRANTED\s+BY\b/},{begin:/\bRETURN\s+(QUERY|NEXT)\b/},{begin:/\b(ATTACH|DETACH)\s+PARTITION\b/},{begin:/\bFORCE\s+ROW\s+LEVEL\s+SECURITY\b/},{begin:/\b(INCLUDING|EXCLUDING)\s+(COMMENTS|CONSTRAINTS|DEFAULTS|IDENTITY|INDEXES|STATISTICS|STORAGE|ALL)\b/},{begin:/\bAS\s+(ASSIGNMENT|IMPLICIT|PERMISSIVE|RESTRICTIVE|ENUM|RANGE)\b/}]},{begin:/\b(FORMAT|FAMILY|VERSION)\s*\(/},{begin:/\bINCLUDE\s*\(/,keywords:"INCLUDE"},{begin:/\bRANGE(?!\s*(BETWEEN|UNBOUNDED|CURRENT|[-0-9]+))/},{begin:/\b(VERSION|OWNER|TEMPLATE|TABLESPACE|CONNECTION\s+LIMIT|PROCEDURE|RESTRICT|JOIN|PARSER|COPY|START|END|COLLATION|INPUT|ANALYZE|STORAGE|LIKE|DEFAULT|DELIMITER|ENCODING|COLUMN|CONSTRAINT|TABLE|SCHEMA)\s*=/},{begin:/\b(PG_\w+?|HAS_[A-Z_]+_PRIVILEGE)\b/,relevance:10},{begin:/\bEXTRACT\s*\(/,end:/\bFROM\b/,returnEnd:true,keywords:{type:"CENTURY DAY DECADE DOW DOY EPOCH HOUR ISODOW ISOYEAR MICROSECONDS "+"MILLENNIUM MILLISECONDS MINUTE MONTH QUARTER SECOND TIMEZONE TIMEZONE_HOUR "+"TIMEZONE_MINUTE WEEK YEAR"}},{begin:/\b(XMLELEMENT|XMLPI)\s*\(\s*NAME/,keywords:{keyword:"NAME"}},{begin:/\b(XMLPARSE|XMLSERIALIZE)\s*\(\s*(DOCUMENT|CONTENT)/,keywords:{keyword:"DOCUMENT CONTENT"}},{beginKeywords:"CACHE INCREMENT MAXVALUE MINVALUE",end:hljs.C_NUMBER_RE,returnEnd:true,keywords:"BY CACHE INCREMENT MAXVALUE MINVALUE"},{className:"type",begin:/\b(WITH|WITHOUT)\s+TIME\s+ZONE\b/},{className:"type",begin:/\bINTERVAL\s+(YEAR|MONTH|DAY|HOUR|MINUTE|SECOND)(\s+TO\s+(MONTH|HOUR|MINUTE|SECOND))?\b/},{begin:/\bRETURNS\s+(LANGUAGE_HANDLER|TRIGGER|EVENT_TRIGGER|FDW_HANDLER|INDEX_AM_HANDLER|TSM_HANDLER)\b/,keywords:{keyword:"RETURNS",type:"LANGUAGE_HANDLER TRIGGER EVENT_TRIGGER FDW_HANDLER INDEX_AM_HANDLER TSM_HANDLER"}},{begin:"\\b("+FUNCTIONS_RE+")\\s*\\("},{begin:"\\.("+TYPES_RE+")\\b"},{begin:"\\b("+TYPES_RE+")\\s+PATH\\b",keywords:{keyword:"PATH",type:TYPES.replace("PATH ","")}},{className:"type",begin:"\\b("+TYPES_RE+")\\b"},{className:"string",begin:"'",end:"'",contains:[{begin:"''"}]},{className:"string",begin:"(e|E|u&|U&)'",end:"'",contains:[{begin:"\\\\."}],relevance:10},{begin:DOLLAR_STRING,endSameAsBegin:true,contains:[{subLanguage:["pgsql","perl","python","tcl","r","lua","java","php","ruby","bash","scheme","xml","json"],endsWithParent:true}]},{begin:'"',end:'"',contains:[{begin:'""'}]},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,{className:"meta",variants:[{begin:"%(ROW)?TYPE",relevance:10},{begin:"\\$\\d+"},{begin:"^#\\w",end:"$"}]},{className:"symbol",begin:LABEL,relevance:10}]}});hljs.registerLanguage("brainfuck",function(hljs){var LITERAL={className:"literal",begin:"[\\+\\-]",relevance:0};return{aliases:["bf"],contains:[hljs.COMMENT("[^\\[\\]\\.,\\+\\-<> \r\n]","[\\[\\]\\.,\\+\\-<> \r\n]",{returnEnd:true,relevance:0}),{className:"title",begin:"[\\[\\]]",relevance:0},{className:"string",begin:"[\\.,]",relevance:0},{begin:/(?:\+\+|\-\-)/,contains:[LITERAL]},LITERAL]}});hljs.registerLanguage("rsl",function(hljs){return{keywords:{keyword:"float color point normal vector matrix while for if do return else break extern continue",built_in:"abs acos ambient area asin atan atmosphere attribute calculatenormal ceil cellnoise "+"clamp comp concat cos degrees depth Deriv diffuse distance Du Dv environment exp "+"faceforward filterstep floor format fresnel incident length lightsource log match "+"max min mod noise normalize ntransform opposite option phong pnoise pow printf "+"ptlined radians random reflect refract renderinfo round setcomp setxcomp setycomp "+"setzcomp shadow sign sin smoothstep specular specularbrdf spline sqrt step tan "+"texture textureinfo trace transform vtransform xcomp ycomp zcomp"},illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"},{className:"class",beginKeywords:"surface displacement light volume imager",end:"\\("},{beginKeywords:"illuminate illuminance gather",end:"\\("}]}});hljs.registerLanguage("crystal",function(hljs){var INT_SUFFIX="(_*[ui](8|16|32|64|128))?";var FLOAT_SUFFIX="(_*f(32|64))?";var CRYSTAL_IDENT_RE="[a-zA-Z_]\\w*[!?=]?";var CRYSTAL_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|[=!]~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~|]|//|//=|&[-+*]=?|&\\*\\*|\\[\\][=?]?";var CRYSTAL_PATH_RE="[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?";var CRYSTAL_KEYWORDS={keyword:"abstract alias annotation as as? asm begin break case class def do else elsif end ensure enum extend for fun if "+"include instance_sizeof is_a? lib macro module next nil? of out pointerof private protected rescue responds_to? "+"return require select self sizeof struct super then type typeof union uninitialized unless until verbatim when while with yield "+"__DIR__ __END_LINE__ __FILE__ __LINE__",literal:"false nil true"};var SUBST={className:"subst",begin:"#{",end:"}",keywords:CRYSTAL_KEYWORDS};var EXPANSION={className:"template-variable",variants:[{begin:"\\{\\{",end:"\\}\\}"},{begin:"\\{%",end:"%\\}"}],keywords:CRYSTAL_KEYWORDS};function recursiveParen(begin,end){var contains=[{begin:begin,end:end}];contains[0].contains=contains;return contains}var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{begin:"%[Qwi]?\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%[Qwi]?\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%[Qwi]?{",end:"}",contains:recursiveParen("{","}")},{begin:"%[Qwi]?<",end:">",contains:recursiveParen("<",">")},{begin:"%[Qwi]?\\|",end:"\\|"},{begin:/<<-\w+$/,end:/^\s*\w+$/}],relevance:0};var Q_STRING={className:"string",variants:[{begin:"%q\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%q\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%q{",end:"}",contains:recursiveParen("{","}")},{begin:"%q<",end:">",contains:recursiveParen("<",">")},{begin:"%q\\|",end:"\\|"},{begin:/<<-'\w+'$/,end:/^\s*\w+$/}],relevance:0};var REGEXP={begin:"(?!%})("+hljs.RE_STARTERS_RE+"|\\n|\\b(case|if|select|unless|until|when|while)\\b)\\s*",keywords:"case if select unless until when while",contains:[{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"//[a-z]*",relevance:0},{begin:"/(?!\\/)",end:"/[a-z]*"}]}],relevance:0};var REGEXP2={className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"%r\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%r\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%r{",end:"}",contains:recursiveParen("{","}")},{begin:"%r<",end:">",contains:recursiveParen("<",">")},{begin:"%r\\|",end:"\\|"}],relevance:0};var ATTRIBUTE={className:"meta",begin:"@\\[",end:"\\]",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"})]};var CRYSTAL_DEFAULT_CONTAINS=[EXPANSION,STRING,Q_STRING,REGEXP2,REGEXP,ATTRIBUTE,hljs.HASH_COMMENT_MODE,{className:"class",beginKeywords:"class module struct",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE}),{begin:"<"}]},{className:"class",beginKeywords:"lib enum union",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE})],relevance:10},{beginKeywords:"annotation",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE})],relevance:10},{className:"function",beginKeywords:"def",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_METHOD_RE,endsParent:true})]},{className:"function",beginKeywords:"fun macro",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_METHOD_RE,endsParent:true})],relevance:5},{className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"(\\!|\\?)?:",relevance:0},{className:"symbol",begin:":",contains:[STRING,{begin:CRYSTAL_METHOD_RE}],relevance:0},{className:"number",variants:[{begin:"\\b0b([01_]+)"+INT_SUFFIX},{begin:"\\b0o([0-7_]+)"+INT_SUFFIX},{begin:"\\b0x([A-Fa-f0-9_]+)"+INT_SUFFIX},{begin:"\\b([1-9][0-9_]*[0-9]|[0-9])(\\.[0-9][0-9_]*)?([eE]_*[-+]?[0-9_]*)?"+FLOAT_SUFFIX+"(?!_)"},{begin:"\\b([1-9][0-9_]*|0)"+INT_SUFFIX}],relevance:0}];SUBST.contains=CRYSTAL_DEFAULT_CONTAINS;EXPANSION.contains=CRYSTAL_DEFAULT_CONTAINS.slice(1);return{aliases:["cr"],lexemes:CRYSTAL_IDENT_RE,keywords:CRYSTAL_KEYWORDS,contains:CRYSTAL_DEFAULT_CONTAINS}});hljs.registerLanguage("smalltalk",function(hljs){var VAR_IDENT_RE="[a-z][a-zA-Z0-9_]*";var CHAR={className:"string",begin:"\\$.{1}"};var SYMBOL={className:"symbol",begin:"#"+hljs.UNDERSCORE_IDENT_RE};return{aliases:["st"],keywords:"self super nil true false thisContext",contains:[hljs.COMMENT('"','"'),hljs.APOS_STRING_MODE,{className:"type",begin:"\\b[A-Z][A-Za-z0-9_]*",relevance:0},{begin:VAR_IDENT_RE+":",relevance:0},hljs.C_NUMBER_MODE,SYMBOL,CHAR,{begin:"\\|[ ]*"+VAR_IDENT_RE+"([ ]+"+VAR_IDENT_RE+")*[ ]*\\|",returnBegin:true,end:/\|/,illegal:/\S/,contains:[{begin:"(\\|[ ]*)?"+VAR_IDENT_RE}]},{begin:"\\#\\(",end:"\\)",contains:[hljs.APOS_STRING_MODE,CHAR,hljs.C_NUMBER_MODE,SYMBOL]}]}});hljs.registerLanguage("vbnet",function(hljs){return{aliases:["vb"],case_insensitive:true,keywords:{keyword:"addhandler addressof alias and andalso aggregate ansi as async assembly auto await binary by byref byval "+"call case catch class compare const continue custom declare default delegate dim distinct do "+"each equals else elseif end enum erase error event exit explicit finally for friend from function "+"get global goto group handles if implements imports in inherits interface into is isfalse isnot istrue iterator "+"join key let lib like loop me mid mod module mustinherit mustoverride mybase myclass "+"nameof namespace narrowing new next not notinheritable notoverridable "+"of off on operator option optional or order orelse overloads overridable overrides "+"paramarray partial preserve private property protected public "+"raiseevent readonly redim rem removehandler resume return "+"select set shadows shared skip static step stop structure strict sub synclock "+"take text then throw to try unicode until using when where while widening with withevents writeonly xor yield",built_in:"boolean byte cbool cbyte cchar cdate cdec cdbl char cint clng cobj csbyte cshort csng cstr ctype "+"date decimal directcast double gettype getxmlnamespace iif integer long object "+"sbyte short single string trycast typeof uinteger ulong ushort",literal:"true false nothing"},illegal:"//|{|}|endif|gosub|variant|wend|^\\$ ",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[{begin:'""'}]}),hljs.COMMENT("'","$",{returnBegin:true,contains:[{className:"doctag",begin:"'''|\x3c!--|--\x3e",contains:[hljs.PHRASAL_WORDS_MODE]},{className:"doctag",begin:"</?",end:">",contains:[hljs.PHRASAL_WORDS_MODE]}]}),hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elseif end region externalsource"}}]}});hljs.registerLanguage("hy",function(hljs){var keywords={"builtin-name":"!= % %= & &= * ** **= *= *map "+"+ += , --build-class-- --import-- -= . / // //= "+"/= < << <<= <= = > >= >> >>= "+"@ @= ^ ^= abs accumulate all and any ap-compose "+"ap-dotimes ap-each ap-each-while ap-filter ap-first ap-if ap-last ap-map ap-map-when ap-pipe "+"ap-reduce ap-reject apply as-> ascii assert assoc bin break butlast "+"callable calling-module-name car case cdr chain chr coll? combinations compile "+"compress cond cons cons? continue count curry cut cycle dec "+"def default-method defclass defmacro defmacro-alias defmacro/g! defmain defmethod defmulti defn "+"defn-alias defnc defnr defreader defseq del delattr delete-route dict-comp dir "+"disassemble dispatch-reader-macro distinct divmod do doto drop drop-last drop-while empty? "+"end-sequence eval eval-and-compile eval-when-compile even? every? except exec filter first "+"flatten float? fn fnc fnr for for* format fraction genexpr "+"gensym get getattr global globals group-by hasattr hash hex id "+"identity if if* if-not if-python2 import in inc input instance? "+"integer integer-char? integer? interleave interpose is is-coll is-cons is-empty is-even "+"is-every is-float is-instance is-integer is-integer-char is-iterable is-iterator is-keyword is-neg is-none "+"is-not is-numeric is-odd is-pos is-string is-symbol is-zero isinstance islice issubclass "+"iter iterable? iterate iterator? keyword keyword? lambda last len let "+"lif lif-not list* list-comp locals loop macro-error macroexpand macroexpand-1 macroexpand-all "+"map max merge-with method-decorator min multi-decorator multicombinations name neg? next "+"none? nonlocal not not-in not? nth numeric? oct odd? open "+"or ord partition permutations pos? post-route postwalk pow prewalk print "+"product profile/calls profile/cpu put-route quasiquote quote raise range read read-str "+"recursive-replace reduce remove repeat repeatedly repr require rest round route "+"route-with-methods rwm second seq set-comp setattr setv some sorted string "+"string? sum switch symbol? take take-nth take-while tee try unless "+"unquote unquote-splicing vars walk when while with with* with-decorator with-gensyms "+"xi xor yield yield-from zero? zip zip-longest | |= ~"};var SYMBOLSTART="a-zA-Z_\\-!.?+*=<>&#'";var SYMBOL_RE="["+SYMBOLSTART+"]["+SYMBOLSTART+"0-9/;:]*";var SIMPLE_NUMBER_RE="[-+]?\\d+(\\.\\d+)?";var SHEBANG={className:"meta",begin:"^#!",end:"$"};var SYMBOL={begin:SYMBOL_RE,relevance:0};var NUMBER={className:"number",begin:SIMPLE_NUMBER_RE,relevance:0};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var LITERAL={className:"literal",begin:/\b([Tt]rue|[Ff]alse|nil|None)\b/};var COLLECTION={begin:"[\\[\\{]",end:"[\\]\\}]"};var HINT={className:"comment",begin:"\\^"+SYMBOL_RE};var HINT_COL=hljs.COMMENT("\\^\\{","\\}");var KEY={className:"symbol",begin:"[:]{1,2}"+SYMBOL_RE};var LIST={begin:"\\(",end:"\\)"};var BODY={endsWithParent:true,relevance:0};var NAME={keywords:keywords,lexemes:SYMBOL_RE,className:"name",begin:SYMBOL_RE,starts:BODY};var DEFAULT_CONTAINS=[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL,SYMBOL];LIST.contains=[hljs.COMMENT("comment",""),NAME,BODY];BODY.contains=DEFAULT_CONTAINS;COLLECTION.contains=DEFAULT_CONTAINS;return{aliases:["hylang"],illegal:/\S/,contains:[SHEBANG,LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL]}});hljs.registerLanguage("objectivec",function(hljs){var API_CLASS={className:"built_in",begin:"\\b(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)\\w+"};var OBJC_KEYWORDS={keyword:"int float while char export sizeof typedef const struct for union "+"unsigned long volatile static bool mutable if do return goto void "+"enum else break extern asm case short default double register explicit "+"signed typename this switch continue wchar_t inline readonly assign "+"readwrite self @synchronized id typeof "+"nonatomic super unichar IBOutlet IBAction strong weak copy "+"in out inout bycopy byref oneway __strong __weak __block __autoreleasing "+"@private @protected @public @try @property @end @throw @catch @finally "+"@autoreleasepool @synthesize @dynamic @selector @optional @required "+"@encode @package @import @defs @compatibility_alias "+"__bridge __bridge_transfer __bridge_retained __bridge_retain "+"__covariant __contravariant __kindof "+"_Nonnull _Nullable _Null_unspecified "+"__FUNCTION__ __PRETTY_FUNCTION__ __attribute__ "+"getter setter retain unsafe_unretained "+"nonnull nullable null_unspecified null_resettable class instancetype "+"NS_DESIGNATED_INITIALIZER NS_UNAVAILABLE NS_REQUIRES_SUPER "+"NS_RETURNS_INNER_POINTER NS_INLINE NS_AVAILABLE NS_DEPRECATED "+"NS_ENUM NS_OPTIONS NS_SWIFT_UNAVAILABLE "+"NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END "+"NS_REFINED_FOR_SWIFT NS_SWIFT_NAME NS_SWIFT_NOTHROW "+"NS_DURING NS_HANDLER NS_ENDHANDLER NS_VALUERETURN NS_VOIDRETURN",literal:"false true FALSE TRUE nil YES NO NULL",built_in:"BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once"};var LEXEMES=/[a-zA-Z@][a-zA-Z0-9_]*/;var CLASS_KEYWORDS="@interface @class @protocol @implementation";return{aliases:["mm","objc","obj-c"],keywords:OBJC_KEYWORDS,lexemes:LEXEMES,illegal:"</",contains:[API_CLASS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"string",variants:[{begin:'@"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]}]},{className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"if else elif endif define undef warning error line "+"pragma ifdef ifndef include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"}),{className:"meta-string",begin:/<.*?>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"class",begin:"("+CLASS_KEYWORDS.split(" ").join("|")+")\\b",end:"({|$)",excludeEnd:true,keywords:CLASS_KEYWORDS,lexemes:LEXEMES,contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"\\."+hljs.UNDERSCORE_IDENT_RE,relevance:0}]}});hljs.registerLanguage("stan",function(hljs){var BLOCKS=["functions","model","data","parameters","quantities","transformed","generated"];var STATEMENTS=["for","in","if","else","while","break","continue","return"];var SPECIAL_FUNCTIONS=["print","reject","increment_log_prob|10","integrate_ode|10","integrate_ode_rk45|10","integrate_ode_bdf|10","algebra_solver"];var VAR_TYPES=["int","real","vector","ordered","positive_ordered","simplex","unit_vector","row_vector","matrix","cholesky_factor_corr|10","cholesky_factor_cov|10","corr_matrix|10","cov_matrix|10","void"];var FUNCTIONS=["Phi","Phi_approx","abs","acos","acosh","algebra_solver","append_array","append_col","append_row","asin","asinh","atan","atan2","atanh","bernoulli_cdf","bernoulli_lccdf","bernoulli_lcdf","bernoulli_logit_lpmf","bernoulli_logit_rng","bernoulli_lpmf","bernoulli_rng","bessel_first_kind","bessel_second_kind","beta_binomial_cdf","beta_binomial_lccdf","beta_binomial_lcdf","beta_binomial_lpmf","beta_binomial_rng","beta_cdf","beta_lccdf","beta_lcdf","beta_lpdf","beta_rng","binary_log_loss","binomial_cdf","binomial_coefficient_log","binomial_lccdf","binomial_lcdf","binomial_logit_lpmf","binomial_lpmf","binomial_rng","block","categorical_logit_lpmf","categorical_logit_rng","categorical_lpmf","categorical_rng","cauchy_cdf","cauchy_lccdf","cauchy_lcdf","cauchy_lpdf","cauchy_rng","cbrt","ceil","chi_square_cdf","chi_square_lccdf","chi_square_lcdf","chi_square_lpdf","chi_square_rng","cholesky_decompose","choose","col","cols","columns_dot_product","columns_dot_self","cos","cosh","cov_exp_quad","crossprod","csr_extract_u","csr_extract_v","csr_extract_w","csr_matrix_times_vector","csr_to_dense_matrix","cumulative_sum","determinant","diag_matrix","diag_post_multiply","diag_pre_multiply","diagonal","digamma","dims","dirichlet_lpdf","dirichlet_rng","distance","dot_product","dot_self","double_exponential_cdf","double_exponential_lccdf","double_exponential_lcdf","double_exponential_lpdf","double_exponential_rng","e","eigenvalues_sym","eigenvectors_sym","erf","erfc","exp","exp2","exp_mod_normal_cdf","exp_mod_normal_lccdf","exp_mod_normal_lcdf","exp_mod_normal_lpdf","exp_mod_normal_rng","expm1","exponential_cdf","exponential_lccdf","exponential_lcdf","exponential_lpdf","exponential_rng","fabs","falling_factorial","fdim","floor","fma","fmax","fmin","fmod","frechet_cdf","frechet_lccdf","frechet_lcdf","frechet_lpdf","frechet_rng","gamma_cdf","gamma_lccdf","gamma_lcdf","gamma_lpdf","gamma_p","gamma_q","gamma_rng","gaussian_dlm_obs_lpdf","get_lp","gumbel_cdf","gumbel_lccdf","gumbel_lcdf","gumbel_lpdf","gumbel_rng","head","hypergeometric_lpmf","hypergeometric_rng","hypot","inc_beta","int_step","integrate_ode","integrate_ode_bdf","integrate_ode_rk45","inv","inv_Phi","inv_chi_square_cdf","inv_chi_square_lccdf","inv_chi_square_lcdf","inv_chi_square_lpdf","inv_chi_square_rng","inv_cloglog","inv_gamma_cdf","inv_gamma_lccdf","inv_gamma_lcdf","inv_gamma_lpdf","inv_gamma_rng","inv_logit","inv_sqrt","inv_square","inv_wishart_lpdf","inv_wishart_rng","inverse","inverse_spd","is_inf","is_nan","lbeta","lchoose","lgamma","lkj_corr_cholesky_lpdf","lkj_corr_cholesky_rng","lkj_corr_lpdf","lkj_corr_rng","lmgamma","lmultiply","log","log10","log1m","log1m_exp","log1m_inv_logit","log1p","log1p_exp","log2","log_determinant","log_diff_exp","log_falling_factorial","log_inv_logit","log_mix","log_rising_factorial","log_softmax","log_sum_exp","logistic_cdf","logistic_lccdf","logistic_lcdf","logistic_lpdf","logistic_rng","logit","lognormal_cdf","lognormal_lccdf","lognormal_lcdf","lognormal_lpdf","lognormal_rng","machine_precision","matrix_exp","max","mdivide_left_spd","mdivide_left_tri_low","mdivide_right_spd","mdivide_right_tri_low","mean","min","modified_bessel_first_kind","modified_bessel_second_kind","multi_gp_cholesky_lpdf","multi_gp_lpdf","multi_normal_cholesky_lpdf","multi_normal_cholesky_rng","multi_normal_lpdf","multi_normal_prec_lpdf","multi_normal_rng","multi_student_t_lpdf","multi_student_t_rng","multinomial_lpmf","multinomial_rng","multiply_log","multiply_lower_tri_self_transpose","neg_binomial_2_cdf","neg_binomial_2_lccdf","neg_binomial_2_lcdf","neg_binomial_2_log_lpmf","neg_binomial_2_log_rng","neg_binomial_2_lpmf","neg_binomial_2_rng","neg_binomial_cdf","neg_binomial_lccdf","neg_binomial_lcdf","neg_binomial_lpmf","neg_binomial_rng","negative_infinity","normal_cdf","normal_lccdf","normal_lcdf","normal_lpdf","normal_rng","not_a_number","num_elements","ordered_logistic_lpmf","ordered_logistic_rng","owens_t","pareto_cdf","pareto_lccdf","pareto_lcdf","pareto_lpdf","pareto_rng","pareto_type_2_cdf","pareto_type_2_lccdf","pareto_type_2_lcdf","pareto_type_2_lpdf","pareto_type_2_rng","pi","poisson_cdf","poisson_lccdf","poisson_lcdf","poisson_log_lpmf","poisson_log_rng","poisson_lpmf","poisson_rng","positive_infinity","pow","print","prod","qr_Q","qr_R","quad_form","quad_form_diag","quad_form_sym","rank","rayleigh_cdf","rayleigh_lccdf","rayleigh_lcdf","rayleigh_lpdf","rayleigh_rng","reject","rep_array","rep_matrix","rep_row_vector","rep_vector","rising_factorial","round","row","rows","rows_dot_product","rows_dot_self","scaled_inv_chi_square_cdf","scaled_inv_chi_square_lccdf","scaled_inv_chi_square_lcdf","scaled_inv_chi_square_lpdf","scaled_inv_chi_square_rng","sd","segment","sin","singular_values","sinh","size","skew_normal_cdf","skew_normal_lccdf","skew_normal_lcdf","skew_normal_lpdf","skew_normal_rng","softmax","sort_asc","sort_desc","sort_indices_asc","sort_indices_desc","sqrt","sqrt2","square","squared_distance","step","student_t_cdf","student_t_lccdf","student_t_lcdf","student_t_lpdf","student_t_rng","sub_col","sub_row","sum","tail","tan","tanh","target","tcrossprod","tgamma","to_array_1d","to_array_2d","to_matrix","to_row_vector","to_vector","trace","trace_gen_quad_form","trace_quad_form","trigamma","trunc","uniform_cdf","uniform_lccdf","uniform_lcdf","uniform_lpdf","uniform_rng","variance","von_mises_lpdf","von_mises_rng","weibull_cdf","weibull_lccdf","weibull_lcdf","weibull_lpdf","weibull_rng","wiener_lpdf","wishart_lpdf","wishart_rng"];var DISTRIBUTIONS=["bernoulli","bernoulli_logit","beta","beta_binomial","binomial","binomial_logit","categorical","categorical_logit","cauchy","chi_square","dirichlet","double_exponential","exp_mod_normal","exponential","frechet","gamma","gaussian_dlm_obs","gumbel","hypergeometric","inv_chi_square","inv_gamma","inv_wishart","lkj_corr","lkj_corr_cholesky","logistic","lognormal","multi_gp","multi_gp_cholesky","multi_normal","multi_normal_cholesky","multi_normal_prec","multi_student_t","multinomial","neg_binomial","neg_binomial_2","neg_binomial_2_log","normal","ordered_logistic","pareto","pareto_type_2","poisson","poisson_log","rayleigh","scaled_inv_chi_square","skew_normal","student_t","uniform","von_mises","weibull","wiener","wishart"];return{aliases:["stanfuncs"],keywords:{title:BLOCKS.join(" "),keyword:STATEMENTS.concat(VAR_TYPES).concat(SPECIAL_FUNCTIONS).join(" "),built_in:FUNCTIONS.join(" ")},lexemes:hljs.IDENT_RE,contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/#/,/$/,{relevance:0,keywords:{"meta-keyword":"include"}}),hljs.COMMENT(/\/\*/,/\*\//,{relevance:0,contains:[{className:"doctag",begin:/@(return|param)/}]}),{begin:/<\s*lower\s*=/,keywords:"lower"},{begin:/[<,]*upper\s*=/,keywords:"upper"},{className:"keyword",begin:/\btarget\s*\+=/,relevance:10},{begin:"~\\s*("+hljs.IDENT_RE+")\\s*\\(",keywords:DISTRIBUTIONS.join(" ")},{className:"number",variants:[{begin:/\b\d+(?:\.\d*)?(?:[eE][+-]?\d+)?/},{begin:/\.\d+(?:[eE][+-]?\d+)?\b/}],relevance:0},{className:"string",begin:'"',end:'"',relevance:0}]}});hljs.registerLanguage("lasso",function(hljs){var LASSO_IDENT_RE="[a-zA-Z_][\\w.]*";var LASSO_ANGLE_RE="<\\?(lasso(script)?|=)";var LASSO_CLOSE_RE="\\]|\\?>";var LASSO_KEYWORDS={literal:"true false none minimal full all void and or not "+"bw nbw ew new cn ncn lt lte gt gte eq neq rx nrx ft",built_in:"array date decimal duration integer map pair string tag xml null "+"boolean bytes keyword list locale queue set stack staticarray "+"local var variable global data self inherited currentcapture givenblock",keyword:"cache database_names database_schemanames database_tablenames "+"define_tag define_type email_batch encode_set html_comment handle "+"handle_error header if inline iterate ljax_target link "+"link_currentaction link_currentgroup link_currentrecord link_detail "+"link_firstgroup link_firstrecord link_lastgroup link_lastrecord "+"link_nextgroup link_nextrecord link_prevgroup link_prevrecord log "+"loop namespace_using output_none portal private protect records "+"referer referrer repeating resultset rows search_args "+"search_arguments select sort_args sort_arguments thread_atomic "+"value_list while abort case else fail_if fail_ifnot fail if_empty "+"if_false if_null if_true loop_abort loop_continue loop_count params "+"params_up return return_value run_children soap_definetag "+"soap_lastrequest soap_lastresponse tag_name ascending average by "+"define descending do equals frozen group handle_failure import in "+"into join let match max min on order parent protected provide public "+"require returnhome skip split_thread sum take thread to trait type "+"where with yield yieldhome"};var HTML_COMMENT=hljs.COMMENT("\x3c!--","--\x3e",{relevance:0});var LASSO_NOPROCESS={className:"meta",begin:"\\[noprocess\\]",starts:{end:"\\[/noprocess\\]",returnEnd:true,contains:[HTML_COMMENT]}};var LASSO_START={className:"meta",begin:"\\[/noprocess|"+LASSO_ANGLE_RE};var LASSO_DATAMEMBER={className:"symbol",begin:"'"+LASSO_IDENT_RE+"'"};var LASSO_CODE=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.inherit(hljs.C_NUMBER_MODE,{begin:hljs.C_NUMBER_RE+"|(-?infinity|NaN)\\b"}),hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"string",begin:"`",end:"`"},{variants:[{begin:"[#$]"+LASSO_IDENT_RE},{begin:"#",end:"\\d+",illegal:"\\W"}]},{className:"type",begin:"::\\s*",end:LASSO_IDENT_RE,illegal:"\\W"},{className:"params",variants:[{begin:"-(?!infinity)"+LASSO_IDENT_RE,relevance:0},{begin:"(\\.\\.\\.)"}]},{begin:/(->|\.)\s*/,relevance:0,contains:[LASSO_DATAMEMBER]},{className:"class",beginKeywords:"define",returnEnd:true,end:"\\(|=>",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:LASSO_IDENT_RE+"(=(?!>))?|[-+*/%](?!>)"})]}];return{aliases:["ls","lassoscript"],case_insensitive:true,lexemes:LASSO_IDENT_RE+"|&[lg]t;",keywords:LASSO_KEYWORDS,contains:[{className:"meta",begin:LASSO_CLOSE_RE,relevance:0,starts:{end:"\\[|"+LASSO_ANGLE_RE,returnEnd:true,relevance:0,contains:[HTML_COMMENT]}},LASSO_NOPROCESS,LASSO_START,{className:"meta",begin:"\\[no_square_brackets",starts:{end:"\\[/no_square_brackets\\]",lexemes:LASSO_IDENT_RE+"|&[lg]t;",keywords:LASSO_KEYWORDS,contains:[{className:"meta",begin:LASSO_CLOSE_RE,relevance:0,starts:{end:"\\[noprocess\\]|"+LASSO_ANGLE_RE,returnEnd:true,contains:[HTML_COMMENT]}},LASSO_NOPROCESS,LASSO_START].concat(LASSO_CODE)}},{className:"meta",begin:"\\[",relevance:0},{className:"meta",begin:"^#!",end:"lasso9$",relevance:10}].concat(LASSO_CODE)}});hljs.registerLanguage("lisp",function(hljs){var LISP_IDENT_RE="[a-zA-Z_\\-\\+\\*\\/\\<\\=\\>\\&\\#][a-zA-Z0-9_\\-\\+\\*\\/\\<\\=\\>\\&\\#!]*";var MEC_RE="\\|[^]*?\\|";var LISP_SIMPLE_NUMBER_RE="(\\-|\\+)?\\d+(\\.\\d+|\\/\\d+)?((d|e|f|l|s|D|E|F|L|S)(\\+|\\-)?\\d+)?";var SHEBANG={className:"meta",begin:"^#!",end:"$"};var LITERAL={className:"literal",begin:"\\b(t{1}|nil)\\b"};var NUMBER={className:"number",variants:[{begin:LISP_SIMPLE_NUMBER_RE,relevance:0},{begin:"#(b|B)[0-1]+(/[0-1]+)?"},{begin:"#(o|O)[0-7]+(/[0-7]+)?"},{begin:"#(x|X)[0-9a-fA-F]+(/[0-9a-fA-F]+)?"},{begin:"#(c|C)\\("+LISP_SIMPLE_NUMBER_RE+" +"+LISP_SIMPLE_NUMBER_RE,end:"\\)"}]};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var VARIABLE={begin:"\\*",end:"\\*"};var KEYWORD={className:"symbol",begin:"[:&]"+LISP_IDENT_RE};var IDENT={begin:LISP_IDENT_RE,relevance:0};var MEC={begin:MEC_RE};var QUOTED_LIST={begin:"\\(",end:"\\)",contains:["self",LITERAL,STRING,NUMBER,IDENT]};var QUOTED={contains:[NUMBER,STRING,VARIABLE,KEYWORD,QUOTED_LIST,IDENT],variants:[{begin:"['`]\\(",end:"\\)"},{begin:"\\(quote ",end:"\\)",keywords:{name:"quote"}},{begin:"'"+MEC_RE}]};var QUOTED_ATOM={variants:[{begin:"'"+LISP_IDENT_RE},{begin:"#'"+LISP_IDENT_RE+"(::"+LISP_IDENT_RE+")*"}]};var LIST={begin:"\\(\\s*",end:"\\)"};var BODY={endsWithParent:true,relevance:0};LIST.contains=[{className:"name",variants:[{begin:LISP_IDENT_RE},{begin:MEC_RE}]},BODY];BODY.contains=[QUOTED,QUOTED_ATOM,LIST,LITERAL,NUMBER,STRING,COMMENT,VARIABLE,KEYWORD,MEC,IDENT];return{illegal:/\S/,contains:[NUMBER,SHEBANG,LITERAL,STRING,COMMENT,QUOTED,QUOTED_ATOM,LIST,IDENT]}});hljs.registerLanguage("mojolicious",function(hljs){return{subLanguage:"xml",contains:[{className:"meta",begin:"^__(END|DATA)__$"},{begin:"^\\s*%{1,2}={0,2}",end:"$",subLanguage:"perl"},{begin:"<%{1,2}={0,2}",end:"={0,1}%>",subLanguage:"perl",excludeBegin:true,excludeEnd:true}]}});hljs.registerLanguage("processing",function(hljs){return{keywords:{keyword:"BufferedReader PVector PFont PImage PGraphics HashMap boolean byte char color "+"double float int long String Array FloatDict FloatList IntDict IntList JSONArray JSONObject "+"Object StringDict StringList Table TableRow XML "+"false synchronized int abstract float private char boolean static null if const "+"for true while long throw strictfp finally protected import native final return void "+"enum else break transient new catch instanceof byte super volatile case assert short "+"package default double public try this switch continue throws protected public private",literal:"P2D P3D HALF_PI PI QUARTER_PI TAU TWO_PI",title:"setup draw",built_in:"displayHeight displayWidth mouseY mouseX mousePressed pmouseX pmouseY key "+"keyCode pixels focused frameCount frameRate height width "+"size createGraphics beginDraw createShape loadShape PShape arc ellipse line point "+"quad rect triangle bezier bezierDetail bezierPoint bezierTangent curve curveDetail curvePoint "+"curveTangent curveTightness shape shapeMode beginContour beginShape bezierVertex curveVertex "+"endContour endShape quadraticVertex vertex ellipseMode noSmooth rectMode smooth strokeCap "+"strokeJoin strokeWeight mouseClicked mouseDragged mouseMoved mousePressed mouseReleased "+"mouseWheel keyPressed keyPressedkeyReleased keyTyped print println save saveFrame day hour "+"millis minute month second year background clear colorMode fill noFill noStroke stroke alpha "+"blue brightness color green hue lerpColor red saturation modelX modelY modelZ screenX screenY "+"screenZ ambient emissive shininess specular add createImage beginCamera camera endCamera frustum "+"ortho perspective printCamera printProjection cursor frameRate noCursor exit loop noLoop popStyle "+"pushStyle redraw binary boolean byte char float hex int str unbinary unhex join match matchAll nf "+"nfc nfp nfs split splitTokens trim append arrayCopy concat expand reverse shorten sort splice subset "+"box sphere sphereDetail createInput createReader loadBytes loadJSONArray loadJSONObject loadStrings "+"loadTable loadXML open parseXML saveTable selectFolder selectInput beginRaw beginRecord createOutput "+"createWriter endRaw endRecord PrintWritersaveBytes saveJSONArray saveJSONObject saveStream saveStrings "+"saveXML selectOutput popMatrix printMatrix pushMatrix resetMatrix rotate rotateX rotateY rotateZ scale "+"shearX shearY translate ambientLight directionalLight lightFalloff lights lightSpecular noLights normal "+"pointLight spotLight image imageMode loadImage noTint requestImage tint texture textureMode textureWrap "+"blend copy filter get loadPixels set updatePixels blendMode loadShader PShaderresetShader shader createFont "+"loadFont text textFont textAlign textLeading textMode textSize textWidth textAscent textDescent abs ceil "+"constrain dist exp floor lerp log mag map max min norm pow round sq sqrt acos asin atan atan2 cos degrees "+"radians sin tan noise noiseDetail noiseSeed random randomGaussian randomSeed"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("clean",function(hljs){return{aliases:["clean","icl","dcl"],keywords:{keyword:"if let in with where case of class instance otherwise "+"implementation definition system module from import qualified as "+"special code inline foreign export ccall stdcall generic derive "+"infix infixl infixr",built_in:"Int Real Char Bool",literal:"True False"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{begin:"->|<-[|:]?|#!?|>>=|\\{\\||\\|\\}|:==|=:|<>"}]}});hljs.registerLanguage("d",function(hljs){var D_KEYWORDS={keyword:"abstract alias align asm assert auto body break byte case cast catch class "+"const continue debug default delete deprecated do else enum export extern final "+"finally for foreach foreach_reverse|10 goto if immutable import in inout int "+"interface invariant is lazy macro mixin module new nothrow out override package "+"pragma private protected public pure ref return scope shared static struct "+"super switch synchronized template this throw try typedef typeid typeof union "+"unittest version void volatile while with __FILE__ __LINE__ __gshared|10 "+"__thread __traits __DATE__ __EOF__ __TIME__ __TIMESTAMP__ __VENDOR__ __VERSION__",built_in:"bool cdouble cent cfloat char creal dchar delegate double dstring float function "+"idouble ifloat ireal long real short string ubyte ucent uint ulong ushort wchar "+"wstring",literal:"false null true"};var decimal_integer_re="(0|[1-9][\\d_]*)",decimal_integer_nosus_re="(0|[1-9][\\d_]*|\\d[\\d_]*|[\\d_]+?\\d)",binary_integer_re="0[bB][01_]+",hexadecimal_digits_re="([\\da-fA-F][\\da-fA-F_]*|_[\\da-fA-F][\\da-fA-F_]*)",hexadecimal_integer_re="0[xX]"+hexadecimal_digits_re,decimal_exponent_re="([eE][+-]?"+decimal_integer_nosus_re+")",decimal_float_re="("+decimal_integer_nosus_re+"(\\.\\d*|"+decimal_exponent_re+")|"+"\\d+\\."+decimal_integer_nosus_re+decimal_integer_nosus_re+"|"+"\\."+decimal_integer_re+decimal_exponent_re+"?"+")",hexadecimal_float_re="(0[xX]("+hexadecimal_digits_re+"\\."+hexadecimal_digits_re+"|"+"\\.?"+hexadecimal_digits_re+")[pP][+-]?"+decimal_integer_nosus_re+")",integer_re="("+decimal_integer_re+"|"+binary_integer_re+"|"+hexadecimal_integer_re+")",float_re="("+hexadecimal_float_re+"|"+decimal_float_re+")";var escape_sequence_re="\\\\("+"['\"\\?\\\\abfnrtv]|"+"u[\\dA-Fa-f]{4}|"+"[0-7]{1,3}|"+"x[\\dA-Fa-f]{2}|"+"U[\\dA-Fa-f]{8}"+")|"+"&[a-zA-Z\\d]{2,};";var D_INTEGER_MODE={className:"number",begin:"\\b"+integer_re+"(L|u|U|Lu|LU|uL|UL)?",relevance:0};var D_FLOAT_MODE={className:"number",begin:"\\b("+float_re+"([fF]|L|i|[fF]i|Li)?|"+integer_re+"(i|[fF]i|Li)"+")",relevance:0};var D_CHARACTER_MODE={className:"string",begin:"'("+escape_sequence_re+"|.)",end:"'",illegal:"."};var D_ESCAPE_SEQUENCE={begin:escape_sequence_re,relevance:0};var D_STRING_MODE={className:"string",begin:'"',contains:[D_ESCAPE_SEQUENCE],end:'"[cwd]?'};var D_WYSIWYG_DELIMITED_STRING_MODE={className:"string",begin:'[rq]"',end:'"[cwd]?',relevance:5};var D_ALTERNATE_WYSIWYG_STRING_MODE={className:"string",begin:"`",end:"`[cwd]?"};var D_HEX_STRING_MODE={className:"string",begin:'x"[\\da-fA-F\\s\\n\\r]*"[cwd]?',relevance:10};var D_TOKEN_STRING_MODE={className:"string",begin:'q"\\{',end:'\\}"'};var D_HASHBANG_MODE={className:"meta",begin:"^#!",end:"$",relevance:5};var D_SPECIAL_TOKEN_SEQUENCE_MODE={className:"meta",begin:"#(line)",end:"$",relevance:5};var D_ATTRIBUTE_MODE={className:"keyword",begin:"@[a-zA-Z_][a-zA-Z_\\d]*"};var D_NESTING_COMMENT_MODE=hljs.COMMENT("\\/\\+","\\+\\/",{contains:["self"],relevance:10});return{lexemes:hljs.UNDERSCORE_IDENT_RE,keywords:D_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,D_NESTING_COMMENT_MODE,D_HEX_STRING_MODE,D_STRING_MODE,D_WYSIWYG_DELIMITED_STRING_MODE,D_ALTERNATE_WYSIWYG_STRING_MODE,D_TOKEN_STRING_MODE,D_FLOAT_MODE,D_INTEGER_MODE,D_CHARACTER_MODE,D_HASHBANG_MODE,D_SPECIAL_TOKEN_SEQUENCE_MODE,D_ATTRIBUTE_MODE]}});hljs.registerLanguage("excel",function(hljs){return{aliases:["xlsx","xls"],case_insensitive:true,lexemes:/[a-zA-Z][\w\.]*/,keywords:{built_in:"ABS ACCRINT ACCRINTM ACOS ACOSH ACOT ACOTH AGGREGATE ADDRESS AMORDEGRC AMORLINC AND ARABIC AREAS ASC ASIN ASINH ATAN ATAN2 ATANH AVEDEV AVERAGE AVERAGEA AVERAGEIF AVERAGEIFS BAHTTEXT BASE BESSELI BESSELJ BESSELK BESSELY BETADIST BETA.DIST BETAINV BETA.INV BIN2DEC BIN2HEX BIN2OCT BINOMDIST BINOM.DIST BINOM.DIST.RANGE BINOM.INV BITAND BITLSHIFT BITOR BITRSHIFT BITXOR CALL CEILING CEILING.MATH CEILING.PRECISE CELL CHAR CHIDIST CHIINV CHITEST CHISQ.DIST CHISQ.DIST.RT CHISQ.INV CHISQ.INV.RT CHISQ.TEST CHOOSE CLEAN CODE COLUMN COLUMNS COMBIN COMBINA COMPLEX CONCAT CONCATENATE CONFIDENCE CONFIDENCE.NORM CONFIDENCE.T CONVERT CORREL COS COSH COT COTH COUNT COUNTA COUNTBLANK COUNTIF COUNTIFS COUPDAYBS COUPDAYS COUPDAYSNC COUPNCD COUPNUM COUPPCD COVAR COVARIANCE.P COVARIANCE.S CRITBINOM CSC CSCH CUBEKPIMEMBER CUBEMEMBER CUBEMEMBERPROPERTY CUBERANKEDMEMBER CUBESET CUBESETCOUNT CUBEVALUE CUMIPMT CUMPRINC DATE DATEDIF DATEVALUE DAVERAGE DAY DAYS DAYS360 DB DBCS DCOUNT DCOUNTA DDB DEC2BIN DEC2HEX DEC2OCT DECIMAL DEGREES DELTA DEVSQ DGET DISC DMAX DMIN DOLLAR DOLLARDE DOLLARFR DPRODUCT DSTDEV DSTDEVP DSUM DURATION DVAR DVARP EDATE EFFECT ENCODEURL EOMONTH ERF ERF.PRECISE ERFC ERFC.PRECISE ERROR.TYPE EUROCONVERT EVEN EXACT EXP EXPON.DIST EXPONDIST FACT FACTDOUBLE FALSE|0 F.DIST FDIST F.DIST.RT FILTERXML FIND FINDB F.INV F.INV.RT FINV FISHER FISHERINV FIXED FLOOR FLOOR.MATH FLOOR.PRECISE FORECAST FORECAST.ETS FORECAST.ETS.CONFINT FORECAST.ETS.SEASONALITY FORECAST.ETS.STAT FORECAST.LINEAR FORMULATEXT FREQUENCY F.TEST FTEST FV FVSCHEDULE GAMMA GAMMA.DIST GAMMADIST GAMMA.INV GAMMAINV GAMMALN GAMMALN.PRECISE GAUSS GCD GEOMEAN GESTEP GETPIVOTDATA GROWTH HARMEAN HEX2BIN HEX2DEC HEX2OCT HLOOKUP HOUR HYPERLINK HYPGEOM.DIST HYPGEOMDIST IF IFERROR IFNA IFS IMABS IMAGINARY IMARGUMENT IMCONJUGATE IMCOS IMCOSH IMCOT IMCSC IMCSCH IMDIV IMEXP IMLN IMLOG10 IMLOG2 IMPOWER IMPRODUCT IMREAL IMSEC IMSECH IMSIN IMSINH IMSQRT IMSUB IMSUM IMTAN INDEX INDIRECT INFO INT INTERCEPT INTRATE IPMT IRR ISBLANK ISERR ISERROR ISEVEN ISFORMULA ISLOGICAL ISNA ISNONTEXT ISNUMBER ISODD ISREF ISTEXT ISO.CEILING ISOWEEKNUM ISPMT JIS KURT LARGE LCM LEFT LEFTB LEN LENB LINEST LN LOG LOG10 LOGEST LOGINV LOGNORM.DIST LOGNORMDIST LOGNORM.INV LOOKUP LOWER MATCH MAX MAXA MAXIFS MDETERM MDURATION MEDIAN MID MIDBs MIN MINIFS MINA MINUTE MINVERSE MIRR MMULT MOD MODE MODE.MULT MODE.SNGL MONTH MROUND MULTINOMIAL MUNIT N NA NEGBINOM.DIST NEGBINOMDIST NETWORKDAYS NETWORKDAYS.INTL NOMINAL NORM.DIST NORMDIST NORMINV NORM.INV NORM.S.DIST NORMSDIST NORM.S.INV NORMSINV NOT NOW NPER NPV NUMBERVALUE OCT2BIN OCT2DEC OCT2HEX ODD ODDFPRICE ODDFYIELD ODDLPRICE ODDLYIELD OFFSET OR PDURATION PEARSON PERCENTILE.EXC PERCENTILE.INC PERCENTILE PERCENTRANK.EXC PERCENTRANK.INC PERCENTRANK PERMUT PERMUTATIONA PHI PHONETIC PI PMT POISSON.DIST POISSON POWER PPMT PRICE PRICEDISC PRICEMAT PROB PRODUCT PROPER PV QUARTILE QUARTILE.EXC QUARTILE.INC QUOTIENT RADIANS RAND RANDBETWEEN RANK.AVG RANK.EQ RANK RATE RECEIVED REGISTER.ID REPLACE REPLACEB REPT RIGHT RIGHTB ROMAN ROUND ROUNDDOWN ROUNDUP ROW ROWS RRI RSQ RTD SEARCH SEARCHB SEC SECH SECOND SERIESSUM SHEET SHEETS SIGN SIN SINH SKEW SKEW.P SLN SLOPE SMALL SQL.REQUEST SQRT SQRTPI STANDARDIZE STDEV STDEV.P STDEV.S STDEVA STDEVP STDEVPA STEYX SUBSTITUTE SUBTOTAL SUM SUMIF SUMIFS SUMPRODUCT SUMSQ SUMX2MY2 SUMX2PY2 SUMXMY2 SWITCH SYD T TAN TANH TBILLEQ TBILLPRICE TBILLYIELD T.DIST T.DIST.2T T.DIST.RT TDIST TEXT TEXTJOIN TIME TIMEVALUE T.INV T.INV.2T TINV TODAY TRANSPOSE TREND TRIM TRIMMEAN TRUE|0 TRUNC T.TEST TTEST TYPE UNICHAR UNICODE UPPER VALUE VAR VAR.P VAR.S VARA VARP VARPA VDB VLOOKUP WEBSERVICE WEEKDAY WEEKNUM WEIBULL WEIBULL.DIST WORKDAY WORKDAY.INTL XIRR XNPV XOR YEAR YEARFRAC YIELD YIELDDISC YIELDMAT Z.TEST ZTEST"},contains:[{begin:/^=/,end:/[^=]/,returnEnd:true,illegal:/=/,relevance:10},{className:"symbol",begin:/\b[A-Z]{1,2}\d+\b/,end:/[^\d]/,excludeEnd:true,relevance:0},{className:"symbol",begin:/[A-Z]{0,2}\d*:[A-Z]{0,2}\d*/,relevance:0},hljs.BACKSLASH_ESCAPE,hljs.QUOTE_STRING_MODE,{className:"number",begin:hljs.NUMBER_RE+"(%)?",relevance:0},hljs.COMMENT(/\bN\(/,/\)/,{excludeBegin:true,excludeEnd:true,illegal:/\n/})]}});hljs.registerLanguage("moonscript",function(hljs){var KEYWORDS={keyword:"if then not for in while do return else elseif break continue switch and or "+"unless when class extends super local import export from using",literal:"true false nil",built_in:"_G _VERSION assert collectgarbage dofile error getfenv getmetatable ipairs load "+"loadfile loadstring module next pairs pcall print rawequal rawget rawset require "+"select setfenv setmetatable tonumber tostring type unpack xpcall coroutine debug "+"io math os package string table"};var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.inherit(hljs.C_NUMBER_MODE,{starts:{end:"(\\s*/)?",relevance:0}}),{className:"string",variants:[{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]},{className:"built_in",begin:"@__"+hljs.IDENT_RE},{begin:"@"+hljs.IDENT_RE},{begin:hljs.IDENT_RE+"\\\\"+hljs.IDENT_RE}];SUBST.contains=EXPRESSIONS;var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var PARAMS_RE="(\\(.*\\))?\\s*\\B[-=]>";var PARAMS={className:"params",begin:"\\([^\\(]",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};return{aliases:["moon"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("--","$"),{className:"function",begin:"^\\s*"+JS_IDENT_RE+"\\s*=\\s*"+PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[TITLE,PARAMS]},{begin:/[\(,:=]\s*/,relevance:0,contains:[{className:"function",begin:PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[PARAMS]}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{className:"name",begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("vala",function(hljs){return{keywords:{keyword:"char uchar unichar int uint long ulong short ushort int8 int16 int32 int64 uint8 "+"uint16 uint32 uint64 float double bool struct enum string void "+"weak unowned owned "+"async signal static abstract interface override virtual delegate "+"if while do for foreach else switch case break default return try catch "+"public private protected internal "+"using new this get set const stdout stdin stderr var",built_in:"DBus GLib CCode Gee Object Gtk Posix",literal:"false true null"},contains:[{className:"class",beginKeywords:"class interface namespace",end:"{",excludeEnd:true,illegal:"[^,:\\n\\s\\.]",contains:[hljs.UNDERSCORE_TITLE_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",begin:'"""',end:'"""',relevance:5},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"^#",end:"$",relevance:2}]}});hljs.registerLanguage("ruleslanguage",function(hljs){return{keywords:{keyword:"BILL_PERIOD BILL_START BILL_STOP RS_EFFECTIVE_START RS_EFFECTIVE_STOP RS_JURIS_CODE RS_OPCO_CODE "+"INTDADDATTRIBUTE|5 INTDADDVMSG|5 INTDBLOCKOP|5 INTDBLOCKOPNA|5 INTDCLOSE|5 INTDCOUNT|5 "+"INTDCOUNTSTATUSCODE|5 INTDCREATEMASK|5 INTDCREATEDAYMASK|5 INTDCREATEFACTORMASK|5 "+"INTDCREATEHANDLE|5 INTDCREATEOVERRIDEDAYMASK|5 INTDCREATEOVERRIDEMASK|5 "+"INTDCREATESTATUSCODEMASK|5 INTDCREATETOUPERIOD|5 INTDDELETE|5 INTDDIPTEST|5 INTDEXPORT|5 "+"INTDGETERRORCODE|5 INTDGETERRORMESSAGE|5 INTDISEQUAL|5 INTDJOIN|5 INTDLOAD|5 INTDLOADACTUALCUT|5 "+"INTDLOADDATES|5 INTDLOADHIST|5 INTDLOADLIST|5 INTDLOADLISTDATES|5 INTDLOADLISTENERGY|5 "+"INTDLOADLISTHIST|5 INTDLOADRELATEDCHANNEL|5 INTDLOADSP|5 INTDLOADSTAGING|5 INTDLOADUOM|5 "+"INTDLOADUOMDATES|5 INTDLOADUOMHIST|5 INTDLOADVERSION|5 INTDOPEN|5 INTDREADFIRST|5 INTDREADNEXT|5 "+"INTDRECCOUNT|5 INTDRELEASE|5 INTDREPLACE|5 INTDROLLAVG|5 INTDROLLPEAK|5 INTDSCALAROP|5 INTDSCALE|5 "+"INTDSETATTRIBUTE|5 INTDSETDSTPARTICIPANT|5 INTDSETSTRING|5 INTDSETVALUE|5 INTDSETVALUESTATUS|5 "+"INTDSHIFTSTARTTIME|5 INTDSMOOTH|5 INTDSORT|5 INTDSPIKETEST|5 INTDSUBSET|5 INTDTOU|5 "+"INTDTOURELEASE|5 INTDTOUVALUE|5 INTDUPDATESTATS|5 INTDVALUE|5 STDEV INTDDELETEEX|5 "+"INTDLOADEXACTUAL|5 INTDLOADEXCUT|5 INTDLOADEXDATES|5 INTDLOADEX|5 INTDLOADEXRELATEDCHANNEL|5 "+"INTDSAVEEX|5 MVLOAD|5 MVLOADACCT|5 MVLOADACCTDATES|5 MVLOADACCTHIST|5 MVLOADDATES|5 MVLOADHIST|5 "+"MVLOADLIST|5 MVLOADLISTDATES|5 MVLOADLISTHIST|5 IF FOR NEXT DONE SELECT END CALL ABORT CLEAR CHANNEL FACTOR LIST NUMBER "+"OVERRIDE SET WEEK DISTRIBUTIONNODE ELSE WHEN THEN OTHERWISE IENUM CSV INCLUDE LEAVE RIDER SAVE DELETE "+"NOVALUE SECTION WARN SAVE_UPDATE DETERMINANT LABEL REPORT REVENUE EACH "+"IN FROM TOTAL CHARGE BLOCK AND OR CSV_FILE RATE_CODE AUXILIARY_DEMAND "+"UIDACCOUNT RS BILL_PERIOD_SELECT HOURS_PER_MONTH INTD_ERROR_STOP SEASON_SCHEDULE_NAME "+"ACCOUNTFACTOR ARRAYUPPERBOUND CALLSTOREDPROC GETADOCONNECTION GETCONNECT GETDATASOURCE "+"GETQUALIFIER GETUSERID HASVALUE LISTCOUNT LISTOP LISTUPDATE LISTVALUE PRORATEFACTOR RSPRORATE "+"SETBINPATH SETDBMONITOR WQ_OPEN BILLINGHOURS DATE DATEFROMFLOAT DATETIMEFROMSTRING "+"DATETIMETOSTRING DATETOFLOAT DAY DAYDIFF DAYNAME DBDATETIME HOUR MINUTE MONTH MONTHDIFF "+"MONTHHOURS MONTHNAME ROUNDDATE SAMEWEEKDAYLASTYEAR SECOND WEEKDAY WEEKDIFF YEAR YEARDAY "+"YEARSTR COMPSUM HISTCOUNT HISTMAX HISTMIN HISTMINNZ HISTVALUE MAXNRANGE MAXRANGE MINRANGE "+"COMPIKVA COMPKVA COMPKVARFROMKQKW COMPLF IDATTR FLAG LF2KW LF2KWH MAXKW POWERFACTOR "+"READING2USAGE AVGSEASON MAXSEASON MONTHLYMERGE SEASONVALUE SUMSEASON ACCTREADDATES "+"ACCTTABLELOAD CONFIGADD CONFIGGET CREATEOBJECT CREATEREPORT EMAILCLIENT EXPBLKMDMUSAGE "+"EXPMDMUSAGE EXPORT_USAGE FACTORINEFFECT GETUSERSPECIFIEDSTOP INEFFECT ISHOLIDAY RUNRATE "+"SAVE_PROFILE SETREPORTTITLE USEREXIT WATFORRUNRATE TO TABLE ACOS ASIN ATAN ATAN2 BITAND CEIL "+"COS COSECANT COSH COTANGENT DIVQUOT DIVREM EXP FABS FLOOR FMOD FREPM FREXPN LOG LOG10 MAX MAXN "+"MIN MINNZ MODF POW ROUND ROUND2VALUE ROUNDINT SECANT SIN SINH SQROOT TAN TANH FLOAT2STRING "+"FLOAT2STRINGNC INSTR LEFT LEN LTRIM MID RIGHT RTRIM STRING STRINGNC TOLOWER TOUPPER TRIM "+"NUMDAYS READ_DATE STAGING",built_in:"IDENTIFIER OPTIONS XML_ELEMENT XML_OP XML_ELEMENT_OF DOMDOCCREATE DOMDOCLOADFILE DOMDOCLOADXML "+"DOMDOCSAVEFILE DOMDOCGETROOT DOMDOCADDPI DOMNODEGETNAME DOMNODEGETTYPE DOMNODEGETVALUE DOMNODEGETCHILDCT "+"DOMNODEGETFIRSTCHILD DOMNODEGETSIBLING DOMNODECREATECHILDELEMENT DOMNODESETATTRIBUTE "+"DOMNODEGETCHILDELEMENTCT DOMNODEGETFIRSTCHILDELEMENT DOMNODEGETSIBLINGELEMENT DOMNODEGETATTRIBUTECT "+"DOMNODEGETATTRIBUTEI DOMNODEGETATTRIBUTEBYNAME DOMNODEGETBYNAME"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"literal",variants:[{begin:"#\\s+[a-zA-Z\\ \\.]*",relevance:0},{begin:"#[a-zA-Z\\ \\.]+"}]}]}});hljs.registerLanguage("verilog",function(hljs){var SV_KEYWORDS={keyword:"accept_on alias always always_comb always_ff always_latch and assert assign "+"assume automatic before begin bind bins binsof bit break buf|0 bufif0 bufif1 "+"byte case casex casez cell chandle checker class clocking cmos config const "+"constraint context continue cover covergroup coverpoint cross deassign default "+"defparam design disable dist do edge else end endcase endchecker endclass "+"endclocking endconfig endfunction endgenerate endgroup endinterface endmodule "+"endpackage endprimitive endprogram endproperty endspecify endsequence endtable "+"endtask enum event eventually expect export extends extern final first_match for "+"force foreach forever fork forkjoin function generate|5 genvar global highz0 highz1 "+"if iff ifnone ignore_bins illegal_bins implements implies import incdir include "+"initial inout input inside instance int integer interconnect interface intersect "+"join join_any join_none large let liblist library local localparam logic longint "+"macromodule matches medium modport module nand negedge nettype new nexttime nmos "+"nor noshowcancelled not notif0 notif1 or output package packed parameter pmos "+"posedge primitive priority program property protected pull0 pull1 pulldown pullup "+"pulsestyle_ondetect pulsestyle_onevent pure rand randc randcase randsequence rcmos "+"real realtime ref reg reject_on release repeat restrict return rnmos rpmos rtran "+"rtranif0 rtranif1 s_always s_eventually s_nexttime s_until s_until_with scalared "+"sequence shortint shortreal showcancelled signed small soft solve specify specparam "+"static string strong strong0 strong1 struct super supply0 supply1 sync_accept_on "+"sync_reject_on table tagged task this throughout time timeprecision timeunit tran "+"tranif0 tranif1 tri tri0 tri1 triand trior trireg type typedef union unique unique0 "+"unsigned until until_with untyped use uwire var vectored virtual void wait wait_order "+"wand weak weak0 weak1 while wildcard wire with within wor xnor xor",literal:"null",built_in:"$finish $stop $exit $fatal $error $warning $info $realtime $time $printtimescale "+"$bitstoreal $bitstoshortreal $itor $signed $cast $bits $stime $timeformat "+"$realtobits $shortrealtobits $rtoi $unsigned $asserton $assertkill $assertpasson "+"$assertfailon $assertnonvacuouson $assertoff $assertcontrol $assertpassoff "+"$assertfailoff $assertvacuousoff $isunbounded $sampled $fell $changed $past_gclk "+"$fell_gclk $changed_gclk $rising_gclk $steady_gclk $coverage_control "+"$coverage_get $coverage_save $set_coverage_db_name $rose $stable $past "+"$rose_gclk $stable_gclk $future_gclk $falling_gclk $changing_gclk $display "+"$coverage_get_max $coverage_merge $get_coverage $load_coverage_db $typename "+"$unpacked_dimensions $left $low $increment $clog2 $ln $log10 $exp $sqrt $pow "+"$floor $ceil $sin $cos $tan $countbits $onehot $isunknown $fatal $warning "+"$dimensions $right $high $size $asin $acos $atan $atan2 $hypot $sinh $cosh "+"$tanh $asinh $acosh $atanh $countones $onehot0 $error $info $random "+"$dist_chi_square $dist_erlang $dist_exponential $dist_normal $dist_poisson "+"$dist_t $dist_uniform $q_initialize $q_remove $q_exam $async$and$array "+"$async$nand$array $async$or$array $async$nor$array $sync$and$array "+"$sync$nand$array $sync$or$array $sync$nor$array $q_add $q_full $psprintf "+"$async$and$plane $async$nand$plane $async$or$plane $async$nor$plane "+"$sync$and$plane $sync$nand$plane $sync$or$plane $sync$nor$plane $system "+"$display $displayb $displayh $displayo $strobe $strobeb $strobeh $strobeo "+"$write $readmemb $readmemh $writememh $value$plusargs "+"$dumpvars $dumpon $dumplimit $dumpports $dumpportson $dumpportslimit "+"$writeb $writeh $writeo $monitor $monitorb $monitorh $monitoro $writememb "+"$dumpfile $dumpoff $dumpall $dumpflush $dumpportsoff $dumpportsall "+"$dumpportsflush $fclose $fdisplay $fdisplayb $fdisplayh $fdisplayo "+"$fstrobe $fstrobeb $fstrobeh $fstrobeo $swrite $swriteb $swriteh "+"$swriteo $fscanf $fread $fseek $fflush $feof $fopen $fwrite $fwriteb "+"$fwriteh $fwriteo $fmonitor $fmonitorb $fmonitorh $fmonitoro $sformat "+"$sformatf $fgetc $ungetc $fgets $sscanf $rewind $ftell $ferror"};return{aliases:["v","sv","svh"],case_insensitive:false,keywords:SV_KEYWORDS,lexemes:/[\w\$]+/,contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"number",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:"\\b((\\d+'(b|h|o|d|B|H|O|D))[0-9xzXZa-fA-F_]+)"},{begin:"\\B(('(b|h|o|d|B|H|O|D))[0-9xzXZa-fA-F_]+)"},{begin:"\\b([0-9_])+",relevance:0}]},{className:"variable",variants:[{begin:"#\\((?!parameter).+\\)"},{begin:"\\.\\w+",relevance:0}]},{className:"meta",begin:"`",end:"$",keywords:{"meta-keyword":"define __FILE__ "+"__LINE__ begin_keywords celldefine default_nettype define "+"else elsif end_keywords endcelldefine endif ifdef ifndef "+"include line nounconnected_drive pragma resetall timescale "+"unconnected_drive undef undefineall"},relevance:0}]}});hljs.registerLanguage("mathematica",function(hljs){return{aliases:["mma","wl"],lexemes:"(\\$|\\b)"+hljs.IDENT_RE+"\\b",keywords:"AASTriangle AbelianGroup Abort AbortKernels AbortProtect AbortScheduledTask Above Abs AbsArg AbsArgPlot Absolute AbsoluteCorrelation AbsoluteCorrelationFunction AbsoluteCurrentValue AbsoluteDashing AbsoluteFileName AbsoluteOptions AbsolutePointSize AbsoluteThickness AbsoluteTime AbsoluteTiming AcceptanceThreshold AccountingForm Accumulate Accuracy AccuracyGoal ActionDelay ActionMenu ActionMenuBox ActionMenuBoxOptions Activate Active ActiveClassification ActiveClassificationObject ActiveItem ActivePrediction ActivePredictionObject ActiveStyle AcyclicGraphQ AddOnHelpPath AddSides AddTo AddToSearchIndex AddUsers AdjacencyGraph AdjacencyList AdjacencyMatrix AdjustmentBox AdjustmentBoxOptions AdjustTimeSeriesForecast AdministrativeDivisionData AffineHalfSpace AffineSpace AffineStateSpaceModel AffineTransform After AggregatedEntityClass AggregationLayer AircraftData AirportData AirPressureData AirTemperatureData AiryAi AiryAiPrime AiryAiZero AiryBi AiryBiPrime AiryBiZero AlgebraicIntegerQ AlgebraicNumber AlgebraicNumberDenominator AlgebraicNumberNorm AlgebraicNumberPolynomial AlgebraicNumberTrace AlgebraicRules AlgebraicRulesData Algebraics AlgebraicUnitQ Alignment AlignmentMarker AlignmentPoint All AllowAdultContent AllowedCloudExtraParameters AllowedCloudParameterExtensions AllowedDimensions AllowedFrequencyRange AllowedHeads AllowGroupClose AllowIncomplete AllowInlineCells AllowKernelInitialization AllowLooseGrammar AllowReverseGroupClose AllowScriptLevelChange AllTrue Alphabet AlphabeticOrder AlphabeticSort AlphaChannel AlternateImage AlternatingFactorial AlternatingGroup AlternativeHypothesis Alternatives AltitudeMethod AmbientLight AmbiguityFunction AmbiguityList Analytic AnatomyData AnatomyForm AnatomyPlot3D AnatomySkinStyle AnatomyStyling AnchoredSearch And AndersonDarlingTest AngerJ AngleBisector AngleBracket AnglePath AnglePath3D AngleVector AngularGauge Animate AnimationCycleOffset AnimationCycleRepetitions AnimationDirection AnimationDisplayTime AnimationRate AnimationRepetitions AnimationRunning AnimationRunTime AnimationTimeIndex Animator AnimatorBox AnimatorBoxOptions AnimatorElements Annotate Annotation AnnotationDelete AnnotationNames AnnotationRules AnnotationValue Annuity AnnuityDue Annulus AnomalyDetection AnomalyDetectorFunction Anonymous Antialiasing AntihermitianMatrixQ Antisymmetric AntisymmetricMatrixQ Antonyms AnyOrder AnySubset AnyTrue Apart ApartSquareFree APIFunction Appearance AppearanceElements AppearanceRules AppellF1 Append AppendCheck AppendLayer AppendTo ApplicationIdentificationKey Apply ApplySides ArcCos ArcCosh ArcCot ArcCoth ArcCsc ArcCsch ArcCurvature ARCHProcess ArcLength ArcSec ArcSech ArcSin ArcSinDistribution ArcSinh ArcTan ArcTanh Area Arg ArgMax ArgMin ArgumentCountQ ARIMAProcess ArithmeticGeometricMean ARMAProcess Around AroundReplace ARProcess Array ArrayComponents ArrayDepth ArrayFilter ArrayFlatten ArrayMesh ArrayPad ArrayPlot ArrayQ ArrayResample ArrayReshape ArrayRules Arrays Arrow Arrow3DBox ArrowBox Arrowheads ASATriangle Ask AskAppend AskConfirm AskDisplay AskedQ AskedValue AskFunction AskState AskTemplateDisplay AspectRatio AspectRatioFixed Assert AssociateTo Association AssociationFormat AssociationMap AssociationQ AssociationThread AssumeDeterministic Assuming Assumptions AstronomicalData AsymptoticDSolveValue AsymptoticEqual AsymptoticEquivalent AsymptoticGreater AsymptoticGreaterEqual AsymptoticIntegrate AsymptoticLess AsymptoticLessEqual AsymptoticOutputTracker AsymptoticRSolveValue AsymptoticSolve AsymptoticSum Asynchronous AsynchronousTaskObject AsynchronousTasks Atom AtomCoordinates AtomCount AtomDiagramCoordinates AtomList AtomQ AttentionLayer Attributes Audio AudioAmplify AudioAnnotate AudioAnnotationLookup AudioBlockMap AudioCapture AudioChannelAssignment AudioChannelCombine AudioChannelMix AudioChannels AudioChannelSeparate AudioData AudioDelay AudioDelete AudioDevice AudioDistance AudioFade AudioFrequencyShift AudioGenerator AudioIdentify AudioInputDevice AudioInsert AudioIntervals AudioJoin AudioLabel AudioLength AudioLocalMeasurements AudioLooping AudioLoudness AudioMeasurements AudioNormalize AudioOutputDevice AudioOverlay AudioPad AudioPan AudioPartition AudioPause AudioPitchShift AudioPlay AudioPlot AudioQ AudioRecord AudioReplace AudioResample AudioReverb AudioSampleRate AudioSpectralMap AudioSpectralTransformation AudioSplit AudioStop AudioStream AudioStreams AudioTimeStretch AudioTrim AudioType AugmentedPolyhedron AugmentedSymmetricPolynomial Authenticate Authentication AuthenticationDialog AutoAction Autocomplete AutocompletionFunction AutoCopy AutocorrelationTest AutoDelete AutoEvaluateEvents AutoGeneratedPackage AutoIndent AutoIndentSpacings AutoItalicWords AutoloadPath AutoMatch Automatic AutomaticImageSize AutoMultiplicationSymbol AutoNumberFormatting AutoOpenNotebooks AutoOpenPalettes AutoQuoteCharacters AutoRefreshed AutoRemove AutorunSequencing AutoScaling AutoScroll AutoSpacing AutoStyleOptions AutoStyleWords AutoSubmitting Axes AxesEdge AxesLabel AxesOrigin AxesStyle AxiomaticTheory Axis"+"BabyMonsterGroupB Back Background BackgroundAppearance BackgroundTasksSettings Backslash Backsubstitution Backward Ball Band BandpassFilter BandstopFilter BarabasiAlbertGraphDistribution BarChart BarChart3D BarcodeImage BarcodeRecognize BaringhausHenzeTest BarLegend BarlowProschanImportance BarnesG BarOrigin BarSpacing BartlettHannWindow BartlettWindow BaseDecode BaseEncode BaseForm Baseline BaselinePosition BaseStyle BasicRecurrentLayer BatchNormalizationLayer BatchSize BatesDistribution BattleLemarieWavelet BayesianMaximization BayesianMaximizationObject BayesianMinimization BayesianMinimizationObject Because BeckmannDistribution Beep Before Begin BeginDialogPacket BeginFrontEndInteractionPacket BeginPackage BellB BellY Below BenfordDistribution BeniniDistribution BenktanderGibratDistribution BenktanderWeibullDistribution BernoulliB BernoulliDistribution BernoulliGraphDistribution BernoulliProcess BernsteinBasis BesselFilterModel BesselI BesselJ BesselJZero BesselK BesselY BesselYZero Beta BetaBinomialDistribution BetaDistribution BetaNegativeBinomialDistribution BetaPrimeDistribution BetaRegularized Between BetweennessCentrality BeveledPolyhedron BezierCurve BezierCurve3DBox BezierCurve3DBoxOptions BezierCurveBox BezierCurveBoxOptions BezierFunction BilateralFilter Binarize BinaryDeserialize BinaryDistance BinaryFormat BinaryImageQ BinaryRead BinaryReadList BinarySerialize BinaryWrite BinCounts BinLists Binomial BinomialDistribution BinomialProcess BinormalDistribution BiorthogonalSplineWavelet BipartiteGraphQ BiquadraticFilterModel BirnbaumImportance BirnbaumSaundersDistribution BitAnd BitClear BitGet BitLength BitNot BitOr BitSet BitShiftLeft BitShiftRight BitXor BiweightLocation BiweightMidvariance Black BlackmanHarrisWindow BlackmanNuttallWindow BlackmanWindow Blank BlankForm BlankNullSequence BlankSequence Blend Block BlockchainAddressData BlockchainBase BlockchainBlockData BlockchainContractValue BlockchainData BlockchainGet BlockchainKeyEncode BlockchainPut BlockchainTokenData BlockchainTransaction BlockchainTransactionData BlockchainTransactionSign BlockchainTransactionSubmit BlockMap BlockRandom BlomqvistBeta BlomqvistBetaTest Blue Blur BodePlot BohmanWindow Bold Bond BondCount BondList BondQ Bookmarks Boole BooleanConsecutiveFunction BooleanConvert BooleanCountingFunction BooleanFunction BooleanGraph BooleanMaxterms BooleanMinimize BooleanMinterms BooleanQ BooleanRegion Booleans BooleanStrings BooleanTable BooleanVariables BorderDimensions BorelTannerDistribution Bottom BottomHatTransform BoundaryDiscretizeGraphics BoundaryDiscretizeRegion BoundaryMesh BoundaryMeshRegion BoundaryMeshRegionQ BoundaryStyle BoundedRegionQ BoundingRegion Bounds Box BoxBaselineShift BoxData BoxDimensions Boxed Boxes BoxForm BoxFormFormatTypes BoxFrame BoxID BoxMargins BoxMatrix BoxObject BoxRatios BoxRotation BoxRotationPoint BoxStyle BoxWhiskerChart Bra BracketingBar BraKet BrayCurtisDistance BreadthFirstScan Break BridgeData BrightnessEqualize BroadcastStationData Brown BrownForsytheTest BrownianBridgeProcess BrowserCategory BSplineBasis BSplineCurve BSplineCurve3DBox BSplineCurve3DBoxOptions BSplineCurveBox BSplineCurveBoxOptions BSplineFunction BSplineSurface BSplineSurface3DBox BSplineSurface3DBoxOptions BubbleChart BubbleChart3D BubbleScale BubbleSizes BuildingData BulletGauge BusinessDayQ ButterflyGraph ButterworthFilterModel Button ButtonBar ButtonBox ButtonBoxOptions ButtonCell ButtonContents ButtonData ButtonEvaluator ButtonExpandable ButtonFrame ButtonFunction ButtonMargins ButtonMinHeight ButtonNote ButtonNotebook ButtonSource ButtonStyle ButtonStyleMenuListing Byte ByteArray ByteArrayFormat ByteArrayQ ByteArrayToString ByteCount ByteOrdering"+"C CachedValue CacheGraphics CachePersistence CalendarConvert CalendarData CalendarType Callout CalloutMarker CalloutStyle CallPacket CanberraDistance Cancel CancelButton CandlestickChart CanonicalGraph CanonicalizePolygon CanonicalizePolyhedron CanonicalName CanonicalWarpingCorrespondence CanonicalWarpingDistance CantorMesh CantorStaircase Cap CapForm CapitalDifferentialD Capitalize CapsuleShape CaptureRunning CardinalBSplineBasis CarlemanLinearize CarmichaelLambda CaseOrdering Cases CaseSensitive Cashflow Casoratian Catalan CatalanNumber Catch Catenate CatenateLayer CauchyDistribution CauchyWindow CayleyGraph CDF CDFDeploy CDFInformation CDFWavelet Ceiling CelestialSystem Cell CellAutoOverwrite CellBaseline CellBoundingBox CellBracketOptions CellChangeTimes CellContents CellContext CellDingbat CellDynamicExpression CellEditDuplicate CellElementsBoundingBox CellElementSpacings CellEpilog CellEvaluationDuplicate CellEvaluationFunction CellEvaluationLanguage CellEventActions CellFrame CellFrameColor CellFrameLabelMargins CellFrameLabels CellFrameMargins CellGroup CellGroupData CellGrouping CellGroupingRules CellHorizontalScrolling CellID CellLabel CellLabelAutoDelete CellLabelMargins CellLabelPositioning CellLabelStyle CellLabelTemplate CellMargins CellObject CellOpen CellPrint CellProlog Cells CellSize CellStyle CellTags CellularAutomaton CensoredDistribution Censoring Center CenterArray CenterDot CentralFeature CentralMoment CentralMomentGeneratingFunction Cepstrogram CepstrogramArray CepstrumArray CForm ChampernowneNumber ChangeOptions ChannelBase ChannelBrokerAction ChannelDatabin ChannelHistoryLength ChannelListen ChannelListener ChannelListeners ChannelListenerWait ChannelObject ChannelPreSendFunction ChannelReceiverFunction ChannelSend ChannelSubscribers ChanVeseBinarize Character CharacterCounts CharacterEncoding CharacterEncodingsPath CharacteristicFunction CharacteristicPolynomial CharacterName CharacterRange Characters ChartBaseStyle ChartElementData ChartElementDataFunction ChartElementFunction ChartElements ChartLabels ChartLayout ChartLegends ChartStyle Chebyshev1FilterModel Chebyshev2FilterModel ChebyshevDistance ChebyshevT ChebyshevU Check CheckAbort CheckAll Checkbox CheckboxBar CheckboxBox CheckboxBoxOptions ChemicalData ChessboardDistance ChiDistribution ChineseRemainder ChiSquareDistribution ChoiceButtons ChoiceDialog CholeskyDecomposition Chop ChromaticityPlot ChromaticityPlot3D ChromaticPolynomial Circle CircleBox CircleDot CircleMinus CirclePlus CirclePoints CircleThrough CircleTimes CirculantGraph CircularOrthogonalMatrixDistribution CircularQuaternionMatrixDistribution CircularRealMatrixDistribution CircularSymplecticMatrixDistribution CircularUnitaryMatrixDistribution Circumsphere CityData ClassifierFunction ClassifierInformation ClassifierMeasurements ClassifierMeasurementsObject Classify ClassPriors Clear ClearAll ClearAttributes ClearCookies ClearPermissions ClearSystemCache ClebschGordan ClickPane Clip ClipboardNotebook ClipFill ClippingStyle ClipPlanes ClipPlanesStyle ClipRange Clock ClockGauge ClockwiseContourIntegral Close Closed CloseKernels ClosenessCentrality Closing ClosingAutoSave ClosingEvent CloudAccountData CloudBase CloudConnect CloudDeploy CloudDirectory CloudDisconnect CloudEvaluate CloudExport CloudExpression CloudExpressions CloudFunction CloudGet CloudImport CloudLoggingData CloudObject CloudObjectInformation CloudObjectInformationData CloudObjectNameFormat CloudObjects CloudObjectURLType CloudPublish CloudPut CloudRenderingMethod CloudSave CloudShare CloudSubmit CloudSymbol CloudUnshare ClusterClassify ClusterDissimilarityFunction ClusteringComponents ClusteringTree CMYKColor Coarse CodeAssistOptions Coefficient CoefficientArrays CoefficientDomain CoefficientList CoefficientRules CoifletWavelet Collect Colon ColonForm ColorBalance ColorCombine ColorConvert ColorCoverage ColorData ColorDataFunction ColorDetect ColorDistance ColorFunction ColorFunctionScaling Colorize ColorNegate ColorOutput ColorProfileData ColorQ ColorQuantize ColorReplace ColorRules ColorSelectorSettings ColorSeparate ColorSetter ColorSetterBox ColorSetterBoxOptions ColorSlider ColorsNear ColorSpace ColorToneMapping Column ColumnAlignments ColumnBackgrounds ColumnForm ColumnLines ColumnsEqual ColumnSpacings ColumnWidths CombinedEntityClass CombinerFunction CometData CommonDefaultFormatTypes Commonest CommonestFilter CommonName CommonUnits CommunityBoundaryStyle CommunityGraphPlot CommunityLabels CommunityRegionStyle CompanyData CompatibleUnitQ CompilationOptions CompilationTarget Compile Compiled CompiledCodeFunction CompiledFunction CompilerOptions Complement CompleteGraph CompleteGraphQ CompleteKaryTree CompletionsListPacket Complex Complexes ComplexExpand ComplexInfinity ComplexityFunction ComplexListPlot ComplexPlot ComplexPlot3D ComponentMeasurements ComponentwiseContextMenu Compose ComposeList ComposeSeries CompositeQ Composition CompoundElement CompoundExpression CompoundPoissonDistribution CompoundPoissonProcess CompoundRenewalProcess Compress CompressedData ComputeUncertainty Condition ConditionalExpression Conditioned Cone ConeBox ConfidenceLevel ConfidenceRange ConfidenceTransform ConfigurationPath ConformAudio ConformImages Congruent ConicHullRegion ConicHullRegion3DBox ConicHullRegionBox ConicOptimization Conjugate ConjugateTranspose Conjunction Connect ConnectedComponents ConnectedGraphComponents ConnectedGraphQ ConnectedMeshComponents ConnectedMoleculeComponents ConnectedMoleculeQ ConnectionSettings ConnectLibraryCallbackFunction ConnectSystemModelComponents ConnesWindow ConoverTest ConsoleMessage ConsoleMessagePacket ConsolePrint Constant ConstantArray ConstantArrayLayer ConstantImage ConstantPlusLayer ConstantRegionQ Constants ConstantTimesLayer ConstellationData ConstrainedMax ConstrainedMin Construct Containing ContainsAll ContainsAny ContainsExactly ContainsNone ContainsOnly ContentFieldOptions ContentLocationFunction ContentObject ContentPadding ContentsBoundingBox ContentSelectable ContentSize Context ContextMenu Contexts ContextToFileName Continuation Continue ContinuedFraction ContinuedFractionK ContinuousAction ContinuousMarkovProcess ContinuousTask ContinuousTimeModelQ ContinuousWaveletData ContinuousWaveletTransform ContourDetect ContourGraphics ContourIntegral ContourLabels ContourLines ContourPlot ContourPlot3D Contours ContourShading ContourSmoothing ContourStyle ContraharmonicMean ContrastiveLossLayer Control ControlActive ControlAlignment ControlGroupContentsBox ControllabilityGramian ControllabilityMatrix ControllableDecomposition ControllableModelQ ControllerDuration ControllerInformation ControllerInformationData ControllerLinking ControllerManipulate ControllerMethod ControllerPath ControllerState ControlPlacement ControlsRendering ControlType Convergents ConversionOptions ConversionRules ConvertToBitmapPacket ConvertToPostScript ConvertToPostScriptPacket ConvexHullMesh ConvexPolygonQ ConvexPolyhedronQ ConvolutionLayer Convolve ConwayGroupCo1 ConwayGroupCo2 ConwayGroupCo3 CookieFunction Cookies CoordinateBoundingBox CoordinateBoundingBoxArray CoordinateBounds CoordinateBoundsArray CoordinateChartData CoordinatesToolOptions CoordinateTransform CoordinateTransformData CoprimeQ Coproduct CopulaDistribution Copyable CopyDatabin CopyDirectory CopyFile CopyTag CopyToClipboard CornerFilter CornerNeighbors Correlation CorrelationDistance CorrelationFunction CorrelationTest Cos Cosh CoshIntegral CosineDistance CosineWindow CosIntegral Cot Coth Count CountDistinct CountDistinctBy CounterAssignments CounterBox CounterBoxOptions CounterClockwiseContourIntegral CounterEvaluator CounterFunction CounterIncrements CounterStyle CounterStyleMenuListing CountRoots CountryData Counts CountsBy Covariance CovarianceEstimatorFunction CovarianceFunction CoxianDistribution CoxIngersollRossProcess CoxModel CoxModelFit CramerVonMisesTest CreateArchive CreateCellID CreateChannel CreateCloudExpression CreateDatabin CreateDataSystemModel CreateDialog CreateDirectory CreateDocument CreateFile CreateIntermediateDirectories CreateManagedLibraryExpression CreateNotebook CreatePalette CreatePalettePacket CreatePermissionsGroup CreateScheduledTask CreateSearchIndex CreateSystemModel CreateTemporary CreateUUID CreateWindow CriterionFunction CriticalityFailureImportance CriticalitySuccessImportance CriticalSection Cross CrossEntropyLossLayer CrossingCount CrossingDetect CrossingPolygon CrossMatrix Csc Csch CTCLossLayer Cube CubeRoot Cubics Cuboid CuboidBox Cumulant CumulantGeneratingFunction Cup CupCap Curl CurlyDoubleQuote CurlyQuote CurrencyConvert CurrentDate CurrentImage CurrentlySpeakingPacket CurrentNotebookImage CurrentScreenImage CurrentValue Curry CurvatureFlowFilter CurveClosed Cyan CycleGraph CycleIndexPolynomial Cycles CyclicGroup Cyclotomic Cylinder CylinderBox CylindricalDecomposition"+"D DagumDistribution DamData DamerauLevenshteinDistance DampingFactor Darker Dashed Dashing DatabaseConnect DatabaseDisconnect DatabaseReference Databin DatabinAdd DatabinRemove Databins DatabinUpload DataCompression DataDistribution DataRange DataReversed Dataset Date DateBounds Dated DateDelimiters DateDifference DatedUnit DateFormat DateFunction DateHistogram DateList DateListLogPlot DateListPlot DateListStepPlot DateObject DateObjectQ DateOverlapsQ DatePattern DatePlus DateRange DateReduction DateString DateTicksFormat DateValue DateWithinQ DaubechiesWavelet DavisDistribution DawsonF DayCount DayCountConvention DayHemisphere DaylightQ DayMatchQ DayName DayNightTerminator DayPlus DayRange DayRound DeBruijnGraph DeBruijnSequence Debug DebugTag Decapitalize Decimal DecimalForm DeclareKnownSymbols DeclarePackage Decompose DeconvolutionLayer Decrement Decrypt DecryptFile DedekindEta DeepSpaceProbeData Default DefaultAxesStyle DefaultBaseStyle DefaultBoxStyle DefaultButton DefaultColor DefaultControlPlacement DefaultDuplicateCellStyle DefaultDuration DefaultElement DefaultFaceGridsStyle DefaultFieldHintStyle DefaultFont DefaultFontProperties DefaultFormatType DefaultFormatTypeForStyle DefaultFrameStyle DefaultFrameTicksStyle DefaultGridLinesStyle DefaultInlineFormatType DefaultInputFormatType DefaultLabelStyle DefaultMenuStyle DefaultNaturalLanguage DefaultNewCellStyle DefaultNewInlineCellStyle DefaultNotebook DefaultOptions DefaultOutputFormatType DefaultPrintPrecision DefaultStyle DefaultStyleDefinitions DefaultTextFormatType DefaultTextInlineFormatType DefaultTicksStyle DefaultTooltipStyle DefaultValue DefaultValues Defer DefineExternal DefineInputStreamMethod DefineOutputStreamMethod DefineResourceFunction Definition Degree DegreeCentrality DegreeGraphDistribution DegreeLexicographic DegreeReverseLexicographic DEigensystem DEigenvalues Deinitialization Del DelaunayMesh Delayed Deletable Delete DeleteAnomalies DeleteBorderComponents DeleteCases DeleteChannel DeleteCloudExpression DeleteContents DeleteDirectory DeleteDuplicates DeleteDuplicatesBy DeleteFile DeleteMissing DeleteObject DeletePermissionsKey DeleteSearchIndex DeleteSmallComponents DeleteStopwords DeleteWithContents DeletionWarning DelimitedArray DelimitedSequence Delimiter DelimiterFlashTime DelimiterMatching Delimiters DeliveryFunction Dendrogram Denominator DensityGraphics DensityHistogram DensityPlot DensityPlot3D DependentVariables Deploy Deployed Depth DepthFirstScan Derivative DerivativeFilter DerivedKey DescriptorStateSpace DesignMatrix DestroyAfterEvaluation Det DeviceClose DeviceConfigure DeviceExecute DeviceExecuteAsynchronous DeviceObject DeviceOpen DeviceOpenQ DeviceRead DeviceReadBuffer DeviceReadLatest DeviceReadList DeviceReadTimeSeries Devices DeviceStreams DeviceWrite DeviceWriteBuffer DGaussianWavelet DiacriticalPositioning Diagonal DiagonalizableMatrixQ DiagonalMatrix DiagonalMatrixQ Dialog DialogIndent DialogInput DialogLevel DialogNotebook DialogProlog DialogReturn DialogSymbols Diamond DiamondMatrix DiceDissimilarity DictionaryLookup DictionaryWordQ DifferenceDelta DifferenceOrder DifferenceQuotient DifferenceRoot DifferenceRootReduce Differences DifferentialD DifferentialRoot DifferentialRootReduce DifferentiatorFilter DigitalSignature DigitBlock DigitBlockMinimum DigitCharacter DigitCount DigitQ DihedralAngle DihedralGroup Dilation DimensionalCombinations DimensionalMeshComponents DimensionReduce DimensionReducerFunction DimensionReduction Dimensions DiracComb DiracDelta DirectedEdge DirectedEdges DirectedGraph DirectedGraphQ DirectedInfinity Direction Directive Directory DirectoryName DirectoryQ DirectoryStack DirichletBeta DirichletCharacter DirichletCondition DirichletConvolve DirichletDistribution DirichletEta DirichletL DirichletLambda DirichletTransform DirichletWindow DisableConsolePrintPacket DisableFormatting DiscreteChirpZTransform DiscreteConvolve DiscreteDelta DiscreteHadamardTransform DiscreteIndicator DiscreteLimit DiscreteLQEstimatorGains DiscreteLQRegulatorGains DiscreteLyapunovSolve DiscreteMarkovProcess DiscreteMaxLimit DiscreteMinLimit DiscretePlot DiscretePlot3D DiscreteRatio DiscreteRiccatiSolve DiscreteShift DiscreteTimeModelQ DiscreteUniformDistribution DiscreteVariables DiscreteWaveletData DiscreteWaveletPacketTransform DiscreteWaveletTransform DiscretizeGraphics DiscretizeRegion Discriminant DisjointQ Disjunction Disk DiskBox DiskMatrix DiskSegment Dispatch DispatchQ DispersionEstimatorFunction Display DisplayAllSteps DisplayEndPacket DisplayFlushImagePacket DisplayForm DisplayFunction DisplayPacket DisplayRules DisplaySetSizePacket DisplayString DisplayTemporary DisplayWith DisplayWithRef DisplayWithVariable DistanceFunction DistanceMatrix DistanceTransform Distribute Distributed DistributedContexts DistributeDefinitions DistributionChart DistributionDomain DistributionFitTest DistributionParameterAssumptions DistributionParameterQ Dithering Div Divergence Divide DivideBy Dividers DivideSides Divisible Divisors DivisorSigma DivisorSum DMSList DMSString Do DockedCells DocumentGenerator DocumentGeneratorInformation DocumentGeneratorInformationData DocumentGenerators DocumentNotebook DocumentWeightingRules Dodecahedron DomainRegistrationInformation DominantColors DOSTextFormat Dot DotDashed DotEqual DotLayer DotPlusLayer Dotted DoubleBracketingBar DoubleContourIntegral DoubleDownArrow DoubleLeftArrow DoubleLeftRightArrow DoubleLeftTee DoubleLongLeftArrow DoubleLongLeftRightArrow DoubleLongRightArrow DoubleRightArrow DoubleRightTee DoubleUpArrow DoubleUpDownArrow DoubleVerticalBar DoublyInfinite Down DownArrow DownArrowBar DownArrowUpArrow DownLeftRightVector DownLeftTeeVector DownLeftVector DownLeftVectorBar DownRightTeeVector DownRightVector DownRightVectorBar Downsample DownTee DownTeeArrow DownValues DragAndDrop DrawEdges DrawFrontFaces DrawHighlighted Drop DropoutLayer DSolve DSolveValue Dt DualLinearProgramming DualPolyhedron DualSystemsModel DumpGet DumpSave DuplicateFreeQ Duration Dynamic DynamicBox DynamicBoxOptions DynamicEvaluationTimeout DynamicGeoGraphics DynamicImage DynamicLocation DynamicModule DynamicModuleBox DynamicModuleBoxOptions DynamicModuleParent DynamicModuleValues DynamicName DynamicNamespace DynamicReference DynamicSetting DynamicUpdating DynamicWrapper DynamicWrapperBox DynamicWrapperBoxOptions"+"E EarthImpactData EarthquakeData EccentricityCentrality Echo EchoFunction EclipseType EdgeAdd EdgeBetweennessCentrality EdgeCapacity EdgeCapForm EdgeColor EdgeConnectivity EdgeContract EdgeCost EdgeCount EdgeCoverQ EdgeCycleMatrix EdgeDashing EdgeDelete EdgeDetect EdgeForm EdgeIndex EdgeJoinForm EdgeLabeling EdgeLabels EdgeLabelStyle EdgeList EdgeOpacity EdgeQ EdgeRenderingFunction EdgeRules EdgeShapeFunction EdgeStyle EdgeThickness EdgeWeight EdgeWeightedGraphQ Editable EditButtonSettings EditCellTagsSettings EditDistance EffectiveInterest Eigensystem Eigenvalues EigenvectorCentrality Eigenvectors Element ElementData ElementwiseLayer ElidedForms Eliminate EliminationOrder Ellipsoid EllipticE EllipticExp EllipticExpPrime EllipticF EllipticFilterModel EllipticK EllipticLog EllipticNomeQ EllipticPi EllipticReducedHalfPeriods EllipticTheta EllipticThetaPrime EmbedCode EmbeddedHTML EmbeddedService EmbeddingLayer EmbeddingObject EmitSound EmphasizeSyntaxErrors EmpiricalDistribution Empty EmptyGraphQ EmptyRegion EnableConsolePrintPacket Enabled Encode Encrypt EncryptedObject EncryptFile End EndAdd EndDialogPacket EndFrontEndInteractionPacket EndOfBuffer EndOfFile EndOfLine EndOfString EndPackage EngineEnvironment EngineeringForm Enter EnterExpressionPacket EnterTextPacket Entity EntityClass EntityClassList EntityCopies EntityFunction EntityGroup EntityInstance EntityList EntityPrefetch EntityProperties EntityProperty EntityPropertyClass EntityRegister EntityStore EntityStores EntityTypeName EntityUnregister EntityValue Entropy EntropyFilter Environment Epilog EpilogFunction Equal EqualColumns EqualRows EqualTilde EqualTo EquatedTo Equilibrium EquirippleFilterKernel Equivalent Erf Erfc Erfi ErlangB ErlangC ErlangDistribution Erosion ErrorBox ErrorBoxOptions ErrorNorm ErrorPacket ErrorsDialogSettings EscapeRadius EstimatedBackground EstimatedDistribution EstimatedProcess EstimatorGains EstimatorRegulator EuclideanDistance EulerAngles EulerCharacteristic EulerE EulerGamma EulerianGraphQ EulerMatrix EulerPhi Evaluatable Evaluate Evaluated EvaluatePacket EvaluateScheduledTask EvaluationBox EvaluationCell EvaluationCompletionAction EvaluationData EvaluationElements EvaluationEnvironment EvaluationMode EvaluationMonitor EvaluationNotebook EvaluationObject EvaluationOrder Evaluator EvaluatorNames EvenQ EventData EventEvaluator EventHandler EventHandlerTag EventLabels EventSeries ExactBlackmanWindow ExactNumberQ ExactRootIsolation ExampleData Except ExcludedForms ExcludedLines ExcludedPhysicalQuantities ExcludePods Exclusions ExclusionsStyle Exists Exit ExitDialog ExoplanetData Exp Expand ExpandAll ExpandDenominator ExpandFileName ExpandNumerator Expectation ExpectationE ExpectedValue ExpGammaDistribution ExpIntegralE ExpIntegralEi ExpirationDate Exponent ExponentFunction ExponentialDistribution ExponentialFamily ExponentialGeneratingFunction ExponentialMovingAverage ExponentialPowerDistribution ExponentPosition ExponentStep Export ExportAutoReplacements ExportByteArray ExportForm ExportPacket ExportString Expression ExpressionCell ExpressionPacket ExpressionUUID ExpToTrig ExtendedEntityClass ExtendedGCD Extension ExtentElementFunction ExtentMarkers ExtentSize ExternalBundle ExternalCall ExternalDataCharacterEncoding ExternalEvaluate ExternalFunction ExternalFunctionName ExternalObject ExternalOptions ExternalSessionObject ExternalSessions ExternalTypeSignature ExternalValue Extract ExtractArchive ExtractLayer ExtremeValueDistribution"+"FaceForm FaceGrids FaceGridsStyle FacialFeatures Factor FactorComplete Factorial Factorial2 FactorialMoment FactorialMomentGeneratingFunction FactorialPower FactorInteger FactorList FactorSquareFree FactorSquareFreeList FactorTerms FactorTermsList Fail Failure FailureAction FailureDistribution FailureQ False FareySequence FARIMAProcess FeatureDistance FeatureExtract FeatureExtraction FeatureExtractor FeatureExtractorFunction FeatureNames FeatureNearest FeatureSpacePlot FeatureSpacePlot3D FeatureTypes FEDisableConsolePrintPacket FeedbackLinearize FeedbackSector FeedbackSectorStyle FeedbackType FEEnableConsolePrintPacket FetalGrowthData Fibonacci Fibonorial FieldCompletionFunction FieldHint FieldHintStyle FieldMasked FieldSize File FileBaseName FileByteCount FileConvert FileDate FileExistsQ FileExtension FileFormat FileHandler FileHash FileInformation FileName FileNameDepth FileNameDialogSettings FileNameDrop FileNameForms FileNameJoin FileNames FileNameSetter FileNameSplit FileNameTake FilePrint FileSize FileSystemMap FileSystemScan FileTemplate FileTemplateApply FileType FilledCurve FilledCurveBox FilledCurveBoxOptions Filling FillingStyle FillingTransform FilteredEntityClass FilterRules FinancialBond FinancialData FinancialDerivative FinancialIndicator Find FindAnomalies FindArgMax FindArgMin FindChannels FindClique FindClusters FindCookies FindCurvePath FindCycle FindDevices FindDistribution FindDistributionParameters FindDivisions FindEdgeCover FindEdgeCut FindEdgeIndependentPaths FindEquationalProof FindEulerianCycle FindExternalEvaluators FindFaces FindFile FindFit FindFormula FindFundamentalCycles FindGeneratingFunction FindGeoLocation FindGeometricConjectures FindGeometricTransform FindGraphCommunities FindGraphIsomorphism FindGraphPartition FindHamiltonianCycle FindHamiltonianPath FindHiddenMarkovStates FindIndependentEdgeSet FindIndependentVertexSet FindInstance FindIntegerNullVector FindKClan FindKClique FindKClub FindKPlex FindLibrary FindLinearRecurrence FindList FindMatchingColor FindMaximum FindMaximumFlow FindMaxValue FindMeshDefects FindMinimum FindMinimumCostFlow FindMinimumCut FindMinValue FindMoleculeSubstructure FindPath FindPeaks FindPermutation FindPostmanTour FindProcessParameters FindRepeat FindRoot FindSequenceFunction FindSettings FindShortestPath FindShortestTour FindSpanningTree FindSystemModelEquilibrium FindTextualAnswer FindThreshold FindTransientRepeat FindVertexCover FindVertexCut FindVertexIndependentPaths Fine FinishDynamic FiniteAbelianGroupCount FiniteGroupCount FiniteGroupData First FirstCase FirstPassageTimeDistribution FirstPosition FischerGroupFi22 FischerGroupFi23 FischerGroupFi24Prime FisherHypergeometricDistribution FisherRatioTest FisherZDistribution Fit FitAll FitRegularization FittedModel FixedOrder FixedPoint FixedPointList FlashSelection Flat Flatten FlattenAt FlattenLayer FlatTopWindow FlipView Floor FlowPolynomial FlushPrintOutputPacket Fold FoldList FoldPair FoldPairList FollowRedirects Font FontColor FontFamily FontForm FontName FontOpacity FontPostScriptName FontProperties FontReencoding FontSize FontSlant FontSubstitutions FontTracking FontVariations FontWeight For ForAll Format FormatRules FormatType FormatTypeAutoConvert FormatValues FormBox FormBoxOptions FormControl FormFunction FormLayoutFunction FormObject FormPage FormTheme FormulaData FormulaLookup FortranForm Forward ForwardBackward Fourier FourierCoefficient FourierCosCoefficient FourierCosSeries FourierCosTransform FourierDCT FourierDCTFilter FourierDCTMatrix FourierDST FourierDSTMatrix FourierMatrix FourierParameters FourierSequenceTransform FourierSeries FourierSinCoefficient FourierSinSeries FourierSinTransform FourierTransform FourierTrigSeries FractionalBrownianMotionProcess FractionalGaussianNoiseProcess FractionalPart FractionBox FractionBoxOptions FractionLine Frame FrameBox FrameBoxOptions Framed FrameInset FrameLabel Frameless FrameMargins FrameRate FrameStyle FrameTicks FrameTicksStyle FRatioDistribution FrechetDistribution FreeQ FrenetSerretSystem FrequencySamplingFilterKernel FresnelC FresnelF FresnelG FresnelS Friday FrobeniusNumber FrobeniusSolve FromAbsoluteTime FromCharacterCode FromCoefficientRules FromContinuedFraction FromDate FromDigits FromDMS FromEntity FromJulianDate FromLetterNumber FromPolarCoordinates FromRomanNumeral FromSphericalCoordinates FromUnixTime Front FrontEndDynamicExpression FrontEndEventActions FrontEndExecute FrontEndObject FrontEndResource FrontEndResourceString FrontEndStackSize FrontEndToken FrontEndTokenExecute FrontEndValueCache FrontEndVersion FrontFaceColor FrontFaceOpacity Full FullAxes FullDefinition FullForm FullGraphics FullInformationOutputRegulator FullOptions FullRegion FullSimplify Function FunctionCompile FunctionCompileExport FunctionCompileExportByteArray FunctionCompileExportLibrary FunctionCompileExportString FunctionDomain FunctionExpand FunctionInterpolation FunctionPeriod FunctionRange FunctionSpace FussellVeselyImportance"+"GaborFilter GaborMatrix GaborWavelet GainMargins GainPhaseMargins GalaxyData GalleryView Gamma GammaDistribution GammaRegularized GapPenalty GARCHProcess GatedRecurrentLayer Gather GatherBy GaugeFaceElementFunction GaugeFaceStyle GaugeFrameElementFunction GaugeFrameSize GaugeFrameStyle GaugeLabels GaugeMarkers GaugeStyle GaussianFilter GaussianIntegers GaussianMatrix GaussianOrthogonalMatrixDistribution GaussianSymplecticMatrixDistribution GaussianUnitaryMatrixDistribution GaussianWindow GCD GegenbauerC General GeneralizedLinearModelFit GenerateAsymmetricKeyPair GenerateConditions GeneratedCell GeneratedDocumentBinding GenerateDerivedKey GenerateDigitalSignature GenerateDocument GeneratedParameters GeneratedQuantityMagnitudes GenerateHTTPResponse GenerateSecuredAuthenticationKey GenerateSymmetricKey GeneratingFunction GeneratorDescription GeneratorHistoryLength GeneratorOutputType Generic GenericCylindricalDecomposition GenomeData GenomeLookup GeoAntipode GeoArea GeoArraySize GeoBackground GeoBoundingBox GeoBounds GeoBoundsRegion GeoBubbleChart GeoCenter GeoCircle GeodesicClosing GeodesicDilation GeodesicErosion GeodesicOpening GeoDestination GeodesyData GeoDirection GeoDisk GeoDisplacement GeoDistance GeoDistanceList GeoElevationData GeoEntities GeoGraphics GeogravityModelData GeoGridDirectionDifference GeoGridLines GeoGridLinesStyle GeoGridPosition GeoGridRange GeoGridRangePadding GeoGridUnitArea GeoGridUnitDistance GeoGridVector GeoGroup GeoHemisphere GeoHemisphereBoundary GeoHistogram GeoIdentify GeoImage GeoLabels GeoLength GeoListPlot GeoLocation GeologicalPeriodData GeomagneticModelData GeoMarker GeometricAssertion GeometricBrownianMotionProcess GeometricDistribution GeometricMean GeometricMeanFilter GeometricScene GeometricTransformation GeometricTransformation3DBox GeometricTransformation3DBoxOptions GeometricTransformationBox GeometricTransformationBoxOptions GeoModel GeoNearest GeoPath GeoPosition GeoPositionENU GeoPositionXYZ GeoProjection GeoProjectionData GeoRange GeoRangePadding GeoRegionValuePlot GeoResolution GeoScaleBar GeoServer GeoSmoothHistogram GeoStreamPlot GeoStyling GeoStylingImageFunction GeoVariant GeoVector GeoVectorENU GeoVectorPlot GeoVectorXYZ GeoVisibleRegion GeoVisibleRegionBoundary GeoWithinQ GeoZoomLevel GestureHandler GestureHandlerTag Get GetBoundingBoxSizePacket GetContext GetEnvironment GetFileName GetFrontEndOptionsDataPacket GetLinebreakInformationPacket GetMenusPacket GetPageBreakInformationPacket Glaisher GlobalClusteringCoefficient GlobalPreferences GlobalSession Glow GoldenAngle GoldenRatio GompertzMakehamDistribution GoodmanKruskalGamma GoodmanKruskalGammaTest Goto Grad Gradient GradientFilter GradientOrientationFilter GrammarApply GrammarRules GrammarToken Graph Graph3D GraphAssortativity GraphAutomorphismGroup GraphCenter GraphComplement GraphData GraphDensity GraphDiameter GraphDifference GraphDisjointUnion GraphDistance GraphDistanceMatrix GraphElementData GraphEmbedding GraphHighlight GraphHighlightStyle GraphHub Graphics Graphics3D Graphics3DBox Graphics3DBoxOptions GraphicsArray GraphicsBaseline GraphicsBox GraphicsBoxOptions GraphicsColor GraphicsColumn GraphicsComplex GraphicsComplex3DBox GraphicsComplex3DBoxOptions GraphicsComplexBox GraphicsComplexBoxOptions GraphicsContents GraphicsData GraphicsGrid GraphicsGridBox GraphicsGroup GraphicsGroup3DBox GraphicsGroup3DBoxOptions GraphicsGroupBox GraphicsGroupBoxOptions GraphicsGrouping GraphicsHighlightColor GraphicsRow GraphicsSpacing GraphicsStyle GraphIntersection GraphLayout GraphLinkEfficiency GraphPeriphery GraphPlot GraphPlot3D GraphPower GraphPropertyDistribution GraphQ GraphRadius GraphReciprocity GraphRoot GraphStyle GraphUnion Gray GrayLevel Greater GreaterEqual GreaterEqualLess GreaterEqualThan GreaterFullEqual GreaterGreater GreaterLess GreaterSlantEqual GreaterThan GreaterTilde Green GreenFunction Grid GridBaseline GridBox GridBoxAlignment GridBoxBackground GridBoxDividers GridBoxFrame GridBoxItemSize GridBoxItemStyle GridBoxOptions GridBoxSpacings GridCreationSettings GridDefaultElement GridElementStyleOptions GridFrame GridFrameMargins GridGraph GridLines GridLinesStyle GroebnerBasis GroupActionBase GroupBy GroupCentralizer GroupElementFromWord GroupElementPosition GroupElementQ GroupElements GroupElementToWord GroupGenerators Groupings GroupMultiplicationTable GroupOrbits GroupOrder GroupPageBreakWithin GroupSetwiseStabilizer GroupStabilizer GroupStabilizerChain GroupTogetherGrouping GroupTogetherNestedGrouping GrowCutComponents Gudermannian GuidedFilter GumbelDistribution"+"HaarWavelet HadamardMatrix HalfLine HalfNormalDistribution HalfPlane HalfSpace HamiltonianGraphQ HammingDistance HammingWindow HandlerFunctions HandlerFunctionsKeys HankelH1 HankelH2 HankelMatrix HankelTransform HannPoissonWindow HannWindow HaradaNortonGroupHN HararyGraph HarmonicMean HarmonicMeanFilter HarmonicNumber Hash Haversine HazardFunction Head HeadCompose HeaderLines Heads HeavisideLambda HeavisidePi HeavisideTheta HeldGroupHe HeldPart HelpBrowserLookup HelpBrowserNotebook HelpBrowserSettings Here HermiteDecomposition HermiteH HermitianMatrixQ HessenbergDecomposition Hessian HexadecimalCharacter Hexahedron HexahedronBox HexahedronBoxOptions HiddenMarkovProcess HiddenSurface Highlighted HighlightGraph HighlightImage HighlightMesh HighpassFilter HigmanSimsGroupHS HilbertCurve HilbertFilter HilbertMatrix Histogram Histogram3D HistogramDistribution HistogramList HistogramTransform HistogramTransformInterpolation HistoricalPeriodData HitMissTransform HITSCentrality HjorthDistribution HodgeDual HoeffdingD HoeffdingDTest Hold HoldAll HoldAllComplete HoldComplete HoldFirst HoldForm HoldPattern HoldRest HolidayCalendar HomeDirectory HomePage Horizontal HorizontalForm HorizontalGauge HorizontalScrollPosition HornerForm HostLookup HotellingTSquareDistribution HoytDistribution HTMLSave HTTPErrorResponse HTTPRedirect HTTPRequest HTTPRequestData HTTPResponse Hue HumanGrowthData HumpDownHump HumpEqual HurwitzLerchPhi HurwitzZeta HyperbolicDistribution HypercubeGraph HyperexponentialDistribution Hyperfactorial Hypergeometric0F1 Hypergeometric0F1Regularized Hypergeometric1F1 Hypergeometric1F1Regularized Hypergeometric2F1 Hypergeometric2F1Regularized HypergeometricDistribution HypergeometricPFQ HypergeometricPFQRegularized HypergeometricU Hyperlink HyperlinkCreationSettings Hyperplane Hyphenation HyphenationOptions HypoexponentialDistribution HypothesisTestData"+"I IconData Iconize IconizedObject IconRules Icosahedron Identity IdentityMatrix If IgnoreCase IgnoreDiacritics IgnorePunctuation IgnoreSpellCheck IgnoringInactive Im Image Image3D Image3DProjection Image3DSlices ImageAccumulate ImageAdd ImageAdjust ImageAlign ImageApply ImageApplyIndexed ImageAspectRatio ImageAssemble ImageAugmentationLayer ImageBoundingBoxes ImageCache ImageCacheValid ImageCapture ImageCaptureFunction ImageCases ImageChannels ImageClip ImageCollage ImageColorSpace ImageCompose ImageContainsQ ImageContents ImageConvolve ImageCooccurrence ImageCorners ImageCorrelate ImageCorrespondingPoints ImageCrop ImageData ImageDeconvolve ImageDemosaic ImageDifference ImageDimensions ImageDisplacements ImageDistance ImageEffect ImageExposureCombine ImageFeatureTrack ImageFileApply ImageFileFilter ImageFileScan ImageFilter ImageFocusCombine ImageForestingComponents ImageFormattingWidth ImageForwardTransformation ImageGraphics ImageHistogram ImageIdentify ImageInstanceQ ImageKeypoints ImageLevels ImageLines ImageMargins ImageMarker ImageMarkers ImageMeasurements ImageMesh ImageMultiply ImageOffset ImagePad ImagePadding ImagePartition ImagePeriodogram ImagePerspectiveTransformation ImagePosition ImagePreviewFunction ImagePyramid ImagePyramidApply ImageQ ImageRangeCache ImageRecolor ImageReflect ImageRegion ImageResize ImageResolution ImageRestyle ImageRotate ImageRotated ImageSaliencyFilter ImageScaled ImageScan ImageSize ImageSizeAction ImageSizeCache ImageSizeMultipliers ImageSizeRaw ImageSubtract ImageTake ImageTransformation ImageTrim ImageType ImageValue ImageValuePositions ImagingDevice ImplicitRegion Implies Import ImportAutoReplacements ImportByteArray ImportOptions ImportString ImprovementImportance In Inactivate Inactive IncidenceGraph IncidenceList IncidenceMatrix IncludeAromaticBonds IncludeConstantBasis IncludeDefinitions IncludeDirectories IncludeFileExtension IncludeGeneratorTasks IncludeHydrogens IncludeInflections IncludeMetaInformation IncludePods IncludeQuantities IncludeRelatedTables IncludeSingularTerm IncludeWindowTimes Increment IndefiniteMatrixQ Indent IndentingNewlineSpacings IndentMaxFraction IndependenceTest IndependentEdgeSetQ IndependentPhysicalQuantity IndependentUnit IndependentUnitDimension IndependentVertexSetQ Indeterminate IndeterminateThreshold IndexCreationOptions Indexed IndexGraph IndexTag Inequality InexactNumberQ InexactNumbers InfiniteLine InfinitePlane Infinity Infix InflationAdjust InflationMethod Information InformationData InformationDataGrid Inherited InheritScope InhomogeneousPoissonProcess InitialEvaluationHistory Initialization InitializationCell InitializationCellEvaluation InitializationCellWarning InitializationObjects InitializationValue Initialize InitialSeeding InlineCounterAssignments InlineCounterIncrements InlineRules Inner InnerPolygon InnerPolyhedron Inpaint Input InputAliases InputAssumptions InputAutoReplacements InputField InputFieldBox InputFieldBoxOptions InputForm InputGrouping InputNamePacket InputNotebook InputPacket InputSettings InputStream InputString InputStringPacket InputToBoxFormPacket Insert InsertionFunction InsertionPointObject InsertLinebreaks InsertResults Inset Inset3DBox Inset3DBoxOptions InsetBox InsetBoxOptions Insphere Install InstallService InstanceNormalizationLayer InString Integer IntegerDigits IntegerExponent IntegerLength IntegerName IntegerPart IntegerPartitions IntegerQ IntegerReverse Integers IntegerString Integral Integrate Interactive InteractiveTradingChart Interlaced Interleaving InternallyBalancedDecomposition InterpolatingFunction InterpolatingPolynomial Interpolation InterpolationOrder InterpolationPoints InterpolationPrecision Interpretation InterpretationBox InterpretationBoxOptions InterpretationFunction Interpreter InterpretTemplate InterquartileRange Interrupt InterruptSettings IntersectingQ Intersection Interval IntervalIntersection IntervalMarkers IntervalMarkersStyle IntervalMemberQ IntervalSlider IntervalUnion Into Inverse InverseBetaRegularized InverseCDF InverseChiSquareDistribution InverseContinuousWaveletTransform InverseDistanceTransform InverseEllipticNomeQ InverseErf InverseErfc InverseFourier InverseFourierCosTransform InverseFourierSequenceTransform InverseFourierSinTransform InverseFourierTransform InverseFunction InverseFunctions InverseGammaDistribution InverseGammaRegularized InverseGaussianDistribution InverseGudermannian InverseHankelTransform InverseHaversine InverseImagePyramid InverseJacobiCD InverseJacobiCN InverseJacobiCS InverseJacobiDC InverseJacobiDN InverseJacobiDS InverseJacobiNC InverseJacobiND InverseJacobiNS InverseJacobiSC InverseJacobiSD InverseJacobiSN InverseLaplaceTransform InverseMellinTransform InversePermutation InverseRadon InverseRadonTransform InverseSeries InverseShortTimeFourier InverseSpectrogram InverseSurvivalFunction InverseTransformedRegion InverseWaveletTransform InverseWeierstrassP InverseWishartMatrixDistribution InverseZTransform Invisible InvisibleApplication InvisibleTimes IPAddress IrreduciblePolynomialQ IslandData IsolatingInterval IsomorphicGraphQ IsotopeData Italic Item ItemAspectRatio ItemBox ItemBoxOptions ItemSize ItemStyle ItoProcess"+"JaccardDissimilarity JacobiAmplitude Jacobian JacobiCD JacobiCN JacobiCS JacobiDC JacobiDN JacobiDS JacobiNC JacobiND JacobiNS JacobiP JacobiSC JacobiSD JacobiSN JacobiSymbol JacobiZeta JankoGroupJ1 JankoGroupJ2 JankoGroupJ3 JankoGroupJ4 JarqueBeraALMTest JohnsonDistribution Join JoinAcross Joined JoinedCurve JoinedCurveBox JoinedCurveBoxOptions JoinForm JordanDecomposition JordanModelDecomposition JulianDate JuliaSetBoettcher JuliaSetIterationCount JuliaSetPlot JuliaSetPoints"+"K KagiChart KaiserBesselWindow KaiserWindow KalmanEstimator KalmanFilter KarhunenLoeveDecomposition KaryTree KatzCentrality KCoreComponents KDistribution KEdgeConnectedComponents KEdgeConnectedGraphQ KelvinBei KelvinBer KelvinKei KelvinKer KendallTau KendallTauTest KernelExecute KernelFunction KernelMixtureDistribution Kernels Ket Key KeyCollisionFunction KeyComplement KeyDrop KeyDropFrom KeyExistsQ KeyFreeQ KeyIntersection KeyMap KeyMemberQ KeypointStrength Keys KeySelect KeySort KeySortBy KeyTake KeyUnion KeyValueMap KeyValuePattern Khinchin KillProcess KirchhoffGraph KirchhoffMatrix KleinInvariantJ KnapsackSolve KnightTourGraph KnotData KnownUnitQ KochCurve KolmogorovSmirnovTest KroneckerDelta KroneckerModelDecomposition KroneckerProduct KroneckerSymbol KuiperTest KumaraswamyDistribution Kurtosis KuwaharaFilter KVertexConnectedComponents KVertexConnectedGraphQ"+"LABColor Label Labeled LabeledSlider LabelingFunction LabelingSize LabelStyle LabelVisibility LaguerreL LakeData LambdaComponents LambertW LaminaData LanczosWindow LandauDistribution Language LanguageCategory LanguageData LanguageIdentify LanguageOptions LaplaceDistribution LaplaceTransform Laplacian LaplacianFilter LaplacianGaussianFilter Large Larger Last Latitude LatitudeLongitude LatticeData LatticeReduce Launch LaunchKernels LayeredGraphPlot LayerSizeFunction LayoutInformation LCHColor LCM LeaderSize LeafCount LeapYearQ LearnDistribution LearnedDistribution LearningRate LearningRateMultipliers LeastSquares LeastSquaresFilterKernel Left LeftArrow LeftArrowBar LeftArrowRightArrow LeftDownTeeVector LeftDownVector LeftDownVectorBar LeftRightArrow LeftRightVector LeftTee LeftTeeArrow LeftTeeVector LeftTriangle LeftTriangleBar LeftTriangleEqual LeftUpDownVector LeftUpTeeVector LeftUpVector LeftUpVectorBar LeftVector LeftVectorBar LegendAppearance Legended LegendFunction LegendLabel LegendLayout LegendMargins LegendMarkers LegendMarkerSize LegendreP LegendreQ LegendreType Length LengthWhile LerchPhi Less LessEqual LessEqualGreater LessEqualThan LessFullEqual LessGreater LessLess LessSlantEqual LessThan LessTilde LetterCharacter LetterCounts LetterNumber LetterQ Level LeveneTest LeviCivitaTensor LevyDistribution Lexicographic LibraryDataType LibraryFunction LibraryFunctionError LibraryFunctionInformation LibraryFunctionLoad LibraryFunctionUnload LibraryLoad LibraryUnload LicenseID LiftingFilterData LiftingWaveletTransform LightBlue LightBrown LightCyan Lighter LightGray LightGreen Lighting LightingAngle LightMagenta LightOrange LightPink LightPurple LightRed LightSources LightYellow Likelihood Limit LimitsPositioning LimitsPositioningTokens LindleyDistribution Line Line3DBox Line3DBoxOptions LinearFilter LinearFractionalOptimization LinearFractionalTransform LinearGradientImage LinearizingTransformationData LinearLayer LinearModelFit LinearOffsetFunction LinearOptimization LinearProgramming LinearRecurrence LinearSolve LinearSolveFunction LineBox LineBoxOptions LineBreak LinebreakAdjustments LineBreakChart LinebreakSemicolonWeighting LineBreakWithin LineColor LineGraph LineIndent LineIndentMaxFraction LineIntegralConvolutionPlot LineIntegralConvolutionScale LineLegend LineOpacity LineSpacing LineWrapParts LinkActivate LinkClose LinkConnect LinkConnectedQ LinkCreate LinkError LinkFlush LinkFunction LinkHost LinkInterrupt LinkLaunch LinkMode LinkObject LinkOpen LinkOptions LinkPatterns LinkProtocol LinkRankCentrality LinkRead LinkReadHeld LinkReadyQ Links LinkService LinkWrite LinkWriteHeld LiouvilleLambda List Listable ListAnimate ListContourPlot ListContourPlot3D ListConvolve ListCorrelate ListCurvePathPlot ListDeconvolve ListDensityPlot ListDensityPlot3D Listen ListFormat ListFourierSequenceTransform ListInterpolation ListLineIntegralConvolutionPlot ListLinePlot ListLogLinearPlot ListLogLogPlot ListLogPlot ListPicker ListPickerBox ListPickerBoxBackground ListPickerBoxOptions ListPlay ListPlot ListPlot3D ListPointPlot3D ListPolarPlot ListQ ListSliceContourPlot3D ListSliceDensityPlot3D ListSliceVectorPlot3D ListStepPlot ListStreamDensityPlot ListStreamPlot ListSurfacePlot3D ListVectorDensityPlot ListVectorPlot ListVectorPlot3D ListZTransform Literal LiteralSearch LocalAdaptiveBinarize LocalCache LocalClusteringCoefficient LocalizeDefinitions LocalizeVariables LocalObject LocalObjects LocalResponseNormalizationLayer LocalSubmit LocalSymbol LocalTime LocalTimeZone LocationEquivalenceTest LocationTest Locator LocatorAutoCreate LocatorBox LocatorBoxOptions LocatorCentering LocatorPane LocatorPaneBox LocatorPaneBoxOptions LocatorRegion Locked Log Log10 Log2 LogBarnesG LogGamma LogGammaDistribution LogicalExpand LogIntegral LogisticDistribution LogisticSigmoid LogitModelFit LogLikelihood LogLinearPlot LogLogisticDistribution LogLogPlot LogMultinormalDistribution LogNormalDistribution LogPlot LogRankTest LogSeriesDistribution LongEqual Longest LongestCommonSequence LongestCommonSequencePositions LongestCommonSubsequence LongestCommonSubsequencePositions LongestMatch LongestOrderedSequence LongForm Longitude LongLeftArrow LongLeftRightArrow LongRightArrow LongShortTermMemoryLayer Lookup Loopback LoopFreeGraphQ LossFunction LowerCaseQ LowerLeftArrow LowerRightArrow LowerTriangularize LowerTriangularMatrixQ LowpassFilter LQEstimatorGains LQGRegulator LQOutputRegulatorGains LQRegulatorGains LUBackSubstitution LucasL LuccioSamiComponents LUDecomposition LunarEclipse LUVColor LyapunovSolve LyonsGroupLy"+"MachineID MachineName MachineNumberQ MachinePrecision MacintoshSystemPageSetup Magenta Magnification Magnify MailAddressValidation MailExecute MailFolder MailItem MailReceiverFunction MailResponseFunction MailSearch MailServerConnect MailServerConnection MailSettings MainSolve MaintainDynamicCaches Majority MakeBoxes MakeExpression MakeRules ManagedLibraryExpressionID ManagedLibraryExpressionQ MandelbrotSetBoettcher MandelbrotSetDistance MandelbrotSetIterationCount MandelbrotSetMemberQ MandelbrotSetPlot MangoldtLambda ManhattanDistance Manipulate Manipulator MannedSpaceMissionData MannWhitneyTest MantissaExponent Manual Map MapAll MapAt MapIndexed MAProcess MapThread MarchenkoPasturDistribution MarcumQ MardiaCombinedTest MardiaKurtosisTest MardiaSkewnessTest MarginalDistribution MarkovProcessProperties Masking MatchingDissimilarity MatchLocalNameQ MatchLocalNames MatchQ Material MathematicalFunctionData MathematicaNotation MathieuC MathieuCharacteristicA MathieuCharacteristicB MathieuCharacteristicExponent MathieuCPrime MathieuGroupM11 MathieuGroupM12 MathieuGroupM22 MathieuGroupM23 MathieuGroupM24 MathieuS MathieuSPrime MathMLForm MathMLText Matrices MatrixExp MatrixForm MatrixFunction MatrixLog MatrixNormalDistribution MatrixPlot MatrixPower MatrixPropertyDistribution MatrixQ MatrixRank MatrixTDistribution Max MaxBend MaxCellMeasure MaxColorDistance MaxDetect MaxDuration MaxExtraBandwidths MaxExtraConditions MaxFeatureDisplacement MaxFeatures MaxFilter MaximalBy Maximize MaxItems MaxIterations MaxLimit MaxMemoryUsed MaxMixtureKernels MaxOverlapFraction MaxPlotPoints MaxPoints MaxRecursion MaxStableDistribution MaxStepFraction MaxSteps MaxStepSize MaxTrainingRounds MaxValue MaxwellDistribution MaxWordGap McLaughlinGroupMcL Mean MeanAbsoluteLossLayer MeanAround MeanClusteringCoefficient MeanDegreeConnectivity MeanDeviation MeanFilter MeanGraphDistance MeanNeighborDegree MeanShift MeanShiftFilter MeanSquaredLossLayer Median MedianDeviation MedianFilter MedicalTestData Medium MeijerG MeijerGReduce MeixnerDistribution MellinConvolve MellinTransform MemberQ MemoryAvailable MemoryConstrained MemoryConstraint MemoryInUse MengerMesh Menu MenuAppearance MenuCommandKey MenuEvaluator MenuItem MenuList MenuPacket MenuSortingValue MenuStyle MenuView Merge MergeDifferences MergingFunction MersennePrimeExponent MersennePrimeExponentQ Mesh MeshCellCentroid MeshCellCount MeshCellHighlight MeshCellIndex MeshCellLabel MeshCellMarker MeshCellMeasure MeshCellQuality MeshCells MeshCellShapeFunction MeshCellStyle MeshCoordinates MeshFunctions MeshPrimitives MeshQualityGoal MeshRange MeshRefinementFunction MeshRegion MeshRegionQ MeshShading MeshStyle Message MessageDialog MessageList MessageName MessageObject MessageOptions MessagePacket Messages MessagesNotebook MetaCharacters MetaInformation MeteorShowerData Method MethodOptions MexicanHatWavelet MeyerWavelet Midpoint Min MinColorDistance MinDetect MineralData MinFilter MinimalBy MinimalPolynomial MinimalStateSpaceModel Minimize MinimumTimeIncrement MinIntervalSize MinkowskiQuestionMark MinLimit MinMax MinorPlanetData Minors MinRecursion MinSize MinStableDistribution Minus MinusPlus MinValue Missing MissingBehavior MissingDataMethod MissingDataRules MissingQ MissingString MissingStyle MissingValuePattern MittagLefflerE MixedFractionParts MixedGraphQ MixedMagnitude MixedRadix MixedRadixQuantity MixedUnit MixtureDistribution Mod Modal Mode Modular ModularInverse ModularLambda Module Modulus MoebiusMu Molecule MoleculeContainsQ MoleculeEquivalentQ MoleculeGraph MoleculeModify MoleculePattern MoleculePlot MoleculePlot3D MoleculeProperty MoleculeQ MoleculeValue Moment Momentary MomentConvert MomentEvaluate MomentGeneratingFunction MomentOfInertia Monday Monitor MonomialList MonomialOrder MonsterGroupM MoonPhase MoonPosition MorletWavelet MorphologicalBinarize MorphologicalBranchPoints MorphologicalComponents MorphologicalEulerNumber MorphologicalGraph MorphologicalPerimeter MorphologicalTransform MortalityData Most MountainData MouseAnnotation MouseAppearance MouseAppearanceTag MouseButtons Mouseover MousePointerNote MousePosition MovieData MovingAverage MovingMap MovingMedian MoyalDistribution Multicolumn MultiedgeStyle MultigraphQ MultilaunchWarning MultiLetterItalics MultiLetterStyle MultilineFunction Multinomial MultinomialDistribution MultinormalDistribution MultiplicativeOrder Multiplicity MultiplySides Multiselection MultivariateHypergeometricDistribution MultivariatePoissonDistribution MultivariateTDistribution"+"N NakagamiDistribution NameQ Names NamespaceBox NamespaceBoxOptions Nand NArgMax NArgMin NBernoulliB NBodySimulation NBodySimulationData NCache NDEigensystem NDEigenvalues NDSolve NDSolveValue Nearest NearestFunction NearestNeighborGraph NearestTo NebulaData NeedCurrentFrontEndPackagePacket NeedCurrentFrontEndSymbolsPacket NeedlemanWunschSimilarity Needs Negative NegativeBinomialDistribution NegativeDefiniteMatrixQ NegativeIntegers NegativeMultinomialDistribution NegativeRationals NegativeReals NegativeSemidefiniteMatrixQ NeighborhoodData NeighborhoodGraph Nest NestedGreaterGreater NestedLessLess NestedScriptRules NestGraph NestList NestWhile NestWhileList NetAppend NetBidirectionalOperator NetChain NetDecoder NetDelete NetDrop NetEncoder NetEvaluationMode NetExtract NetFlatten NetFoldOperator NetGraph NetInformation NetInitialize NetInsert NetInsertSharedArrays NetJoin NetMapOperator NetMapThreadOperator NetMeasurements NetModel NetNestOperator NetPairEmbeddingOperator NetPort NetPortGradient NetPrepend NetRename NetReplace NetReplacePart NetSharedArray NetStateObject NetTake NetTrain NetTrainResultsObject NetworkPacketCapture NetworkPacketRecording NetworkPacketRecordingDuring NetworkPacketTrace NeumannValue NevilleThetaC NevilleThetaD NevilleThetaN NevilleThetaS NewPrimitiveStyle NExpectation Next NextCell NextDate NextPrime NextScheduledTaskTime NHoldAll NHoldFirst NHoldRest NicholsGridLines NicholsPlot NightHemisphere NIntegrate NMaximize NMaxValue NMinimize NMinValue NominalVariables NonAssociative NoncentralBetaDistribution NoncentralChiSquareDistribution NoncentralFRatioDistribution NoncentralStudentTDistribution NonCommutativeMultiply NonConstants NondimensionalizationTransform None NoneTrue NonlinearModelFit NonlinearStateSpaceModel NonlocalMeansFilter NonNegative NonNegativeIntegers NonNegativeRationals NonNegativeReals NonPositive NonPositiveIntegers NonPositiveRationals NonPositiveReals Nor NorlundB Norm Normal NormalDistribution NormalGrouping NormalizationLayer Normalize Normalized NormalizedSquaredEuclideanDistance NormalMatrixQ NormalsFunction NormFunction Not NotCongruent NotCupCap NotDoubleVerticalBar Notebook NotebookApply NotebookAutoSave NotebookClose NotebookConvertSettings NotebookCreate NotebookCreateReturnObject NotebookDefault NotebookDelete NotebookDirectory NotebookDynamicExpression NotebookEvaluate NotebookEventActions NotebookFileName NotebookFind NotebookFindReturnObject NotebookGet NotebookGetLayoutInformationPacket NotebookGetMisspellingsPacket NotebookImport NotebookInformation NotebookInterfaceObject NotebookLocate NotebookObject NotebookOpen NotebookOpenReturnObject NotebookPath NotebookPrint NotebookPut NotebookPutReturnObject NotebookRead NotebookResetGeneratedCells Notebooks NotebookSave NotebookSaveAs NotebookSelection NotebookSetupLayoutInformationPacket NotebooksMenu NotebookTemplate NotebookWrite NotElement NotEqualTilde NotExists NotGreater NotGreaterEqual NotGreaterFullEqual NotGreaterGreater NotGreaterLess NotGreaterSlantEqual NotGreaterTilde Nothing NotHumpDownHump NotHumpEqual NotificationFunction NotLeftTriangle NotLeftTriangleBar NotLeftTriangleEqual NotLess NotLessEqual NotLessFullEqual NotLessGreater NotLessLess NotLessSlantEqual NotLessTilde NotNestedGreaterGreater NotNestedLessLess NotPrecedes NotPrecedesEqual NotPrecedesSlantEqual NotPrecedesTilde NotReverseElement NotRightTriangle NotRightTriangleBar NotRightTriangleEqual NotSquareSubset NotSquareSubsetEqual NotSquareSuperset NotSquareSupersetEqual NotSubset NotSubsetEqual NotSucceeds NotSucceedsEqual NotSucceedsSlantEqual NotSucceedsTilde NotSuperset NotSupersetEqual NotTilde NotTildeEqual NotTildeFullEqual NotTildeTilde NotVerticalBar Now NoWhitespace NProbability NProduct NProductFactors NRoots NSolve NSum NSumTerms NuclearExplosionData NuclearReactorData Null NullRecords NullSpace NullWords Number NumberCompose NumberDecompose NumberExpand NumberFieldClassNumber NumberFieldDiscriminant NumberFieldFundamentalUnits NumberFieldIntegralBasis NumberFieldNormRepresentatives NumberFieldRegulator NumberFieldRootsOfUnity NumberFieldSignature NumberForm NumberFormat NumberLinePlot NumberMarks NumberMultiplier NumberPadding NumberPoint NumberQ NumberSeparator NumberSigns NumberString Numerator NumeratorDenominator NumericalOrder NumericalSort NumericArray NumericArrayQ NumericArrayType NumericFunction NumericQ NuttallWindow NValues NyquistGridLines NyquistPlot"+"O ObservabilityGramian ObservabilityMatrix ObservableDecomposition ObservableModelQ OceanData Octahedron OddQ Off Offset OLEData On ONanGroupON Once OneIdentity Opacity OpacityFunction OpacityFunctionScaling Open OpenAppend Opener OpenerBox OpenerBoxOptions OpenerView OpenFunctionInspectorPacket Opening OpenRead OpenSpecialOptions OpenTemporary OpenWrite Operate OperatingSystem OptimumFlowData Optional OptionalElement OptionInspectorSettings OptionQ Options OptionsPacket OptionsPattern OptionValue OptionValueBox OptionValueBoxOptions Or Orange Order OrderDistribution OrderedQ Ordering OrderingBy OrderingLayer Orderless OrderlessPatternSequence OrnsteinUhlenbeckProcess Orthogonalize OrthogonalMatrixQ Out Outer OuterPolygon OuterPolyhedron OutputAutoOverwrite OutputControllabilityMatrix OutputControllableModelQ OutputForm OutputFormData OutputGrouping OutputMathEditExpression OutputNamePacket OutputResponse OutputSizeLimit OutputStream Over OverBar OverDot Overflow OverHat Overlaps Overlay OverlayBox OverlayBoxOptions Overscript OverscriptBox OverscriptBoxOptions OverTilde OverVector OverwriteTarget OwenT OwnValues"+"Package PackingMethod PaddedForm Padding PaddingLayer PaddingSize PadeApproximant PadLeft PadRight PageBreakAbove PageBreakBelow PageBreakWithin PageFooterLines PageFooters PageHeaderLines PageHeaders PageHeight PageRankCentrality PageTheme PageWidth Pagination PairedBarChart PairedHistogram PairedSmoothHistogram PairedTTest PairedZTest PaletteNotebook PalettePath PalindromeQ Pane PaneBox PaneBoxOptions Panel PanelBox PanelBoxOptions Paneled PaneSelector PaneSelectorBox PaneSelectorBoxOptions PaperWidth ParabolicCylinderD ParagraphIndent ParagraphSpacing ParallelArray ParallelCombine ParallelDo Parallelepiped ParallelEvaluate Parallelization Parallelize ParallelMap ParallelNeeds Parallelogram ParallelProduct ParallelSubmit ParallelSum ParallelTable ParallelTry Parameter ParameterEstimator ParameterMixtureDistribution ParameterVariables ParametricFunction ParametricNDSolve ParametricNDSolveValue ParametricPlot ParametricPlot3D ParametricRegion ParentBox ParentCell ParentConnect ParentDirectory ParentForm Parenthesize ParentList ParentNotebook ParetoDistribution ParetoPickandsDistribution ParkData Part PartBehavior PartialCorrelationFunction PartialD ParticleAcceleratorData ParticleData Partition PartitionGranularity PartitionsP PartitionsQ PartLayer PartOfSpeech PartProtection ParzenWindow PascalDistribution PassEventsDown PassEventsUp Paste PasteAutoQuoteCharacters PasteBoxFormInlineCells PasteButton Path PathGraph PathGraphQ Pattern PatternSequence PatternTest PauliMatrix PaulWavelet Pause PausedTime PDF PeakDetect PeanoCurve PearsonChiSquareTest PearsonCorrelationTest PearsonDistribution PercentForm PerfectNumber PerfectNumberQ PerformanceGoal Perimeter PeriodicBoundaryCondition PeriodicInterpolation Periodogram PeriodogramArray Permanent Permissions PermissionsGroup PermissionsGroupMemberQ PermissionsGroups PermissionsKey PermissionsKeys PermutationCycles PermutationCyclesQ PermutationGroup PermutationLength PermutationList PermutationListQ PermutationMax PermutationMin PermutationOrder PermutationPower PermutationProduct PermutationReplace Permutations PermutationSupport Permute PeronaMalikFilter Perpendicular PerpendicularBisector PersistenceLocation PersistenceTime PersistentObject PersistentObjects PersistentValue PersonData PERTDistribution PetersenGraph PhaseMargins PhaseRange PhysicalSystemData Pi Pick PIDData PIDDerivativeFilter PIDFeedforward PIDTune Piecewise PiecewiseExpand PieChart PieChart3D PillaiTrace PillaiTraceTest PingTime Pink PitchRecognize Pivoting PixelConstrained PixelValue PixelValuePositions Placed Placeholder PlaceholderReplace Plain PlanarAngle PlanarGraph PlanarGraphQ PlanckRadiationLaw PlaneCurveData PlanetaryMoonData PlanetData PlantData Play PlayRange Plot Plot3D Plot3Matrix PlotDivision PlotJoined PlotLabel PlotLabels PlotLayout PlotLegends PlotMarkers PlotPoints PlotRange PlotRangeClipping PlotRangeClipPlanesStyle PlotRangePadding PlotRegion PlotStyle PlotTheme Pluralize Plus PlusMinus Pochhammer PodStates PodWidth Point Point3DBox Point3DBoxOptions PointBox PointBoxOptions PointFigureChart PointLegend PointSize PoissonConsulDistribution PoissonDistribution PoissonProcess PoissonWindow PolarAxes PolarAxesOrigin PolarGridLines PolarPlot PolarTicks PoleZeroMarkers PolyaAeppliDistribution PolyGamma Polygon Polygon3DBox Polygon3DBoxOptions PolygonalNumber PolygonAngle PolygonBox PolygonBoxOptions PolygonCoordinates PolygonDecomposition PolygonHoleScale PolygonIntersections PolygonScale Polyhedron PolyhedronAngle PolyhedronCoordinates PolyhedronData PolyhedronDecomposition PolyhedronGenus PolyLog PolynomialExtendedGCD PolynomialForm PolynomialGCD PolynomialLCM PolynomialMod PolynomialQ PolynomialQuotient PolynomialQuotientRemainder PolynomialReduce PolynomialRemainder Polynomials PoolingLayer PopupMenu PopupMenuBox PopupMenuBoxOptions PopupView PopupWindow Position PositionIndex Positive PositiveDefiniteMatrixQ PositiveIntegers PositiveRationals PositiveReals PositiveSemidefiniteMatrixQ PossibleZeroQ Postfix PostScript Power PowerDistribution PowerExpand PowerMod PowerModList PowerRange PowerSpectralDensity PowersRepresentations PowerSymmetricPolynomial Precedence PrecedenceForm Precedes PrecedesEqual PrecedesSlantEqual PrecedesTilde Precision PrecisionGoal PreDecrement Predict PredictionRoot PredictorFunction PredictorInformation PredictorMeasurements PredictorMeasurementsObject PreemptProtect PreferencesPath Prefix PreIncrement Prepend PrependLayer PrependTo PreprocessingRules PreserveColor PreserveImageOptions Previous PreviousCell PreviousDate PriceGraphDistribution PrimaryPlaceholder Prime PrimeNu PrimeOmega PrimePi PrimePowerQ PrimeQ Primes PrimeZetaP PrimitivePolynomialQ PrimitiveRoot PrimitiveRootList PrincipalComponents PrincipalValue Print PrintableASCIIQ PrintAction PrintForm PrintingCopies PrintingOptions PrintingPageRange PrintingStartingPageNumber PrintingStyleEnvironment Printout3D Printout3DPreviewer PrintPrecision PrintTemporary Prism PrismBox PrismBoxOptions PrivateCellOptions PrivateEvaluationOptions PrivateFontOptions PrivateFrontEndOptions PrivateKey PrivateNotebookOptions PrivatePaths Probability ProbabilityDistribution ProbabilityPlot ProbabilityPr ProbabilityScalePlot ProbitModelFit ProcessConnection ProcessDirectory ProcessEnvironment Processes ProcessEstimator ProcessInformation ProcessObject ProcessParameterAssumptions ProcessParameterQ ProcessStateDomain ProcessStatus ProcessTimeDomain Product ProductDistribution ProductLog ProgressIndicator ProgressIndicatorBox ProgressIndicatorBoxOptions Projection Prolog PromptForm ProofObject Properties Property PropertyList PropertyValue Proportion Proportional Protect Protected ProteinData Pruning PseudoInverse PsychrometricPropertyData PublicKey PublisherID PulsarData PunctuationCharacter Purple Put PutAppend Pyramid PyramidBox PyramidBoxOptions"+"QBinomial QFactorial QGamma QHypergeometricPFQ QnDispersion QPochhammer QPolyGamma QRDecomposition QuadraticIrrationalQ QuadraticOptimization Quantile QuantilePlot Quantity QuantityArray QuantityDistribution QuantityForm QuantityMagnitude QuantityQ QuantityUnit QuantityVariable QuantityVariableCanonicalUnit QuantityVariableDimensions QuantityVariableIdentifier QuantityVariablePhysicalQuantity Quartics QuartileDeviation Quartiles QuartileSkewness Query QueueingNetworkProcess QueueingProcess QueueProperties Quiet Quit Quotient QuotientRemainder"+"RadialGradientImage RadialityCentrality RadicalBox RadicalBoxOptions RadioButton RadioButtonBar RadioButtonBox RadioButtonBoxOptions Radon RadonTransform RamanujanTau RamanujanTauL RamanujanTauTheta RamanujanTauZ Ramp Random RandomChoice RandomColor RandomComplex RandomEntity RandomFunction RandomGeoPosition RandomGraph RandomImage RandomInstance RandomInteger RandomPermutation RandomPoint RandomPolygon RandomPolyhedron RandomPrime RandomReal RandomSample RandomSeed RandomSeeding RandomVariate RandomWalkProcess RandomWord Range RangeFilter RangeSpecification RankedMax RankedMin RarerProbability Raster Raster3D Raster3DBox Raster3DBoxOptions RasterArray RasterBox RasterBoxOptions Rasterize RasterSize Rational RationalFunctions Rationalize Rationals Ratios RawArray RawBoxes RawData RawMedium RayleighDistribution Re Read ReadByteArray ReadLine ReadList ReadProtected ReadString Real RealAbs RealBlockDiagonalForm RealDigits RealExponent Reals RealSign Reap RecognitionPrior RecognitionThreshold Record RecordLists RecordSeparators Rectangle RectangleBox RectangleBoxOptions RectangleChart RectangleChart3D RectangularRepeatingElement RecurrenceFilter RecurrenceTable RecurringDigitsForm Red Reduce RefBox ReferenceLineStyle ReferenceMarkers ReferenceMarkerStyle Refine ReflectionMatrix ReflectionTransform Refresh RefreshRate Region RegionBinarize RegionBoundary RegionBounds RegionCentroid RegionDifference RegionDimension RegionDisjoint RegionDistance RegionDistanceFunction RegionEmbeddingDimension RegionEqual RegionFunction RegionImage RegionIntersection RegionMeasure RegionMember RegionMemberFunction RegionMoment RegionNearest RegionNearestFunction RegionPlot RegionPlot3D RegionProduct RegionQ RegionResize RegionSize RegionSymmetricDifference RegionUnion RegionWithin RegisterExternalEvaluator RegularExpression Regularization RegularlySampledQ RegularPolygon ReIm ReImLabels ReImPlot ReImStyle Reinstall RelationalDatabase RelationGraph Release ReleaseHold ReliabilityDistribution ReliefImage ReliefPlot RemoteAuthorizationCaching RemoteConnect RemoteConnectionObject RemoteFile RemoteRun RemoteRunProcess Remove RemoveAlphaChannel RemoveAsynchronousTask RemoveAudioStream RemoveBackground RemoveChannelListener RemoveChannelSubscribers Removed RemoveDiacritics RemoveInputStreamMethod RemoveOutputStreamMethod RemoveProperty RemoveScheduledTask RemoveUsers RenameDirectory RenameFile RenderAll RenderingOptions RenewalProcess RenkoChart RepairMesh Repeated RepeatedNull RepeatedString RepeatedTiming RepeatingElement Replace ReplaceAll ReplaceHeldPart ReplaceImageValue ReplaceList ReplacePart ReplacePixelValue ReplaceRepeated ReplicateLayer RequiredPhysicalQuantities Resampling ResamplingAlgorithmData ResamplingMethod Rescale RescalingTransform ResetDirectory ResetMenusPacket ResetScheduledTask ReshapeLayer Residue ResizeLayer Resolve ResourceAcquire ResourceData ResourceFunction ResourceObject ResourceRegister ResourceRemove ResourceSearch ResourceSubmissionObject ResourceSubmit ResourceSystemBase ResourceUpdate ResponseForm Rest RestartInterval Restricted Resultant ResumePacket Return ReturnEntersInput ReturnExpressionPacket ReturnInputFormPacket ReturnPacket ReturnReceiptFunction ReturnTextPacket Reverse ReverseBiorthogonalSplineWavelet ReverseElement ReverseEquilibrium ReverseGraph ReverseSort ReverseSortBy ReverseUpEquilibrium RevolutionAxis RevolutionPlot3D RGBColor RiccatiSolve RiceDistribution RidgeFilter RiemannR RiemannSiegelTheta RiemannSiegelZ RiemannXi Riffle Right RightArrow RightArrowBar RightArrowLeftArrow RightComposition RightCosetRepresentative RightDownTeeVector RightDownVector RightDownVectorBar RightTee RightTeeArrow RightTeeVector RightTriangle RightTriangleBar RightTriangleEqual RightUpDownVector RightUpTeeVector RightUpVector RightUpVectorBar RightVector RightVectorBar RiskAchievementImportance RiskReductionImportance RogersTanimotoDissimilarity RollPitchYawAngles RollPitchYawMatrix RomanNumeral Root RootApproximant RootIntervals RootLocusPlot RootMeanSquare RootOfUnityQ RootReduce Roots RootSum Rotate RotateLabel RotateLeft RotateRight RotationAction RotationBox RotationBoxOptions RotationMatrix RotationTransform Round RoundImplies RoundingRadius Row RowAlignments RowBackgrounds RowBox RowHeights RowLines RowMinHeight RowReduce RowsEqual RowSpacings RSolve RSolveValue RudinShapiro RudvalisGroupRu Rule RuleCondition RuleDelayed RuleForm RulePlot RulerUnits Run RunProcess RunScheduledTask RunThrough RuntimeAttributes RuntimeOptions RussellRaoDissimilarity"+"SameQ SameTest SampledEntityClass SampleDepth SampledSoundFunction SampledSoundList SampleRate SamplingPeriod SARIMAProcess SARMAProcess SASTriangle SatelliteData SatisfiabilityCount SatisfiabilityInstances SatisfiableQ Saturday Save Saveable SaveAutoDelete SaveConnection SaveDefinitions SavitzkyGolayMatrix SawtoothWave Scale Scaled ScaleDivisions ScaledMousePosition ScaleOrigin ScalePadding ScaleRanges ScaleRangeStyle ScalingFunctions ScalingMatrix ScalingTransform Scan ScheduledTask ScheduledTaskActiveQ ScheduledTaskInformation ScheduledTaskInformationData ScheduledTaskObject ScheduledTasks SchurDecomposition ScientificForm ScientificNotationThreshold ScorerGi ScorerGiPrime ScorerHi ScorerHiPrime ScreenRectangle ScreenStyleEnvironment ScriptBaselineShifts ScriptForm ScriptLevel ScriptMinSize ScriptRules ScriptSizeMultipliers Scrollbars ScrollingOptions ScrollPosition SearchAdjustment SearchIndexObject SearchIndices SearchQueryString SearchResultObject Sec Sech SechDistribution SecondOrderConeOptimization SectionGrouping SectorChart SectorChart3D SectorOrigin SectorSpacing SecuredAuthenticationKey SecuredAuthenticationKeys SeedRandom Select Selectable SelectComponents SelectedCells SelectedNotebook SelectFirst Selection SelectionAnimate SelectionCell SelectionCellCreateCell SelectionCellDefaultStyle SelectionCellParentStyle SelectionCreateCell SelectionDebuggerTag SelectionDuplicateCell SelectionEvaluate SelectionEvaluateCreateCell SelectionMove SelectionPlaceholder SelectionSetStyle SelectWithContents SelfLoops SelfLoopStyle SemanticImport SemanticImportString SemanticInterpretation SemialgebraicComponentInstances SemidefiniteOptimization SendMail SendMessage Sequence SequenceAlignment SequenceAttentionLayer SequenceCases SequenceCount SequenceFold SequenceFoldList SequenceForm SequenceHold SequenceLastLayer SequenceMostLayer SequencePosition SequencePredict SequencePredictorFunction SequenceReplace SequenceRestLayer SequenceReverseLayer SequenceSplit Series SeriesCoefficient SeriesData ServiceConnect ServiceDisconnect ServiceExecute ServiceObject ServiceRequest ServiceResponse ServiceSubmit SessionSubmit SessionTime Set SetAccuracy SetAlphaChannel SetAttributes Setbacks SetBoxFormNamesPacket SetCloudDirectory SetCookies SetDelayed SetDirectory SetEnvironment SetEvaluationNotebook SetFileDate SetFileLoadingContext SetNotebookStatusLine SetOptions SetOptionsPacket SetPermissions SetPrecision SetProperty SetSecuredAuthenticationKey SetSelectedNotebook SetSharedFunction SetSharedVariable SetSpeechParametersPacket SetStreamPosition SetSystemModel SetSystemOptions Setter SetterBar SetterBox SetterBoxOptions Setting SetUsers SetValue Shading Shallow ShannonWavelet ShapiroWilkTest Share SharingList Sharpen ShearingMatrix ShearingTransform ShellRegion ShenCastanMatrix ShiftedGompertzDistribution ShiftRegisterSequence Short ShortDownArrow Shortest ShortestMatch ShortestPathFunction ShortLeftArrow ShortRightArrow ShortTimeFourier ShortTimeFourierData ShortUpArrow Show ShowAutoConvert ShowAutoSpellCheck ShowAutoStyles ShowCellBracket ShowCellLabel ShowCellTags ShowClosedCellArea ShowCodeAssist ShowContents ShowControls ShowCursorTracker ShowGroupOpenCloseIcon ShowGroupOpener ShowInvisibleCharacters ShowPageBreaks ShowPredictiveInterface ShowSelection ShowShortBoxForm ShowSpecialCharacters ShowStringCharacters ShowSyntaxStyles ShrinkingDelay ShrinkWrapBoundingBox SiderealTime SiegelTheta SiegelTukeyTest SierpinskiCurve SierpinskiMesh Sign Signature SignedRankTest SignedRegionDistance SignificanceLevel SignPadding SignTest SimilarityRules SimpleGraph SimpleGraphQ SimplePolygonQ SimplePolyhedronQ Simplex Simplify Sin Sinc SinghMaddalaDistribution SingleEvaluation SingleLetterItalics SingleLetterStyle SingularValueDecomposition SingularValueList SingularValuePlot SingularValues Sinh SinhIntegral SinIntegral SixJSymbol Skeleton SkeletonTransform SkellamDistribution Skewness SkewNormalDistribution SkinStyle Skip SliceContourPlot3D SliceDensityPlot3D SliceDistribution SliceVectorPlot3D Slider Slider2D Slider2DBox Slider2DBoxOptions SliderBox SliderBoxOptions SlideView Slot SlotSequence Small SmallCircle Smaller SmithDecomposition SmithDelayCompensator SmithWatermanSimilarity SmoothDensityHistogram SmoothHistogram SmoothHistogram3D SmoothKernelDistribution SnDispersion Snippet SnubPolyhedron SocialMediaData Socket SocketConnect SocketListen SocketListener SocketObject SocketOpen SocketReadMessage SocketReadyQ Sockets SocketWaitAll SocketWaitNext SoftmaxLayer SokalSneathDissimilarity SolarEclipse SolarSystemFeatureData SolidAngle SolidData SolidRegionQ Solve SolveAlways SolveDelayed Sort SortBy SortedBy SortedEntityClass Sound SoundAndGraphics SoundNote SoundVolume SourceLink Sow Space SpaceCurveData SpaceForm Spacer Spacings Span SpanAdjustments SpanCharacterRounding SpanFromAbove SpanFromBoth SpanFromLeft SpanLineThickness SpanMaxSize SpanMinSize SpanningCharacters SpanSymmetric SparseArray SpatialGraphDistribution SpatialMedian SpatialTransformationLayer Speak SpeakTextPacket SpearmanRankTest SpearmanRho SpeciesData SpecificityGoal SpectralLineData Spectrogram SpectrogramArray Specularity SpeechRecognize SpeechSynthesize SpellingCorrection SpellingCorrectionList SpellingDictionaries SpellingDictionariesPath SpellingOptions SpellingSuggestionsPacket Sphere SphereBox SpherePoints SphericalBesselJ SphericalBesselY SphericalHankelH1 SphericalHankelH2 SphericalHarmonicY SphericalPlot3D SphericalRegion SphericalShell SpheroidalEigenvalue SpheroidalJoiningFactor SpheroidalPS SpheroidalPSPrime SpheroidalQS SpheroidalQSPrime SpheroidalRadialFactor SpheroidalS1 SpheroidalS1Prime SpheroidalS2 SpheroidalS2Prime Splice SplicedDistribution SplineClosed SplineDegree SplineKnots SplineWeights Split SplitBy SpokenString Sqrt SqrtBox SqrtBoxOptions Square SquaredEuclideanDistance SquareFreeQ SquareIntersection SquareMatrixQ SquareRepeatingElement SquaresR SquareSubset SquareSubsetEqual SquareSuperset SquareSupersetEqual SquareUnion SquareWave SSSTriangle StabilityMargins StabilityMarginsStyle StableDistribution Stack StackBegin StackComplete StackedDateListPlot StackedListPlot StackInhibit StadiumShape StandardAtmosphereData StandardDeviation StandardDeviationFilter StandardForm Standardize Standardized StandardOceanData StandbyDistribution Star StarClusterData StarData StarGraph StartAsynchronousTask StartExternalSession StartingStepSize StartOfLine StartOfString StartProcess StartScheduledTask StartupSound StartWebSession StateDimensions StateFeedbackGains StateOutputEstimator StateResponse StateSpaceModel StateSpaceRealization StateSpaceTransform StateTransformationLinearize StationaryDistribution StationaryWaveletPacketTransform StationaryWaveletTransform StatusArea StatusCentrality StepMonitor StereochemistryElements StieltjesGamma StirlingS1 StirlingS2 StopAsynchronousTask StoppingPowerData StopScheduledTask StrataVariables StratonovichProcess StreamColorFunction StreamColorFunctionScaling StreamDensityPlot StreamMarkers StreamPlot StreamPoints StreamPosition Streams StreamScale StreamStyle String StringBreak StringByteCount StringCases StringContainsQ StringCount StringDelete StringDrop StringEndsQ StringExpression StringExtract StringForm StringFormat StringFreeQ StringInsert StringJoin StringLength StringMatchQ StringPadLeft StringPadRight StringPart StringPartition StringPosition StringQ StringRepeat StringReplace StringReplaceList StringReplacePart StringReverse StringRiffle StringRotateLeft StringRotateRight StringSkeleton StringSplit StringStartsQ StringTake StringTemplate StringToByteArray StringToStream StringTrim StripBoxes StripOnInput StripWrapperBoxes StrokeForm StructuralImportance StructuredArray StructuredSelection StruveH StruveL Stub StudentTDistribution Style StyleBox StyleBoxAutoDelete StyleData StyleDefinitions StyleForm StyleHints StyleKeyMapping StyleMenuListing StyleNameDialogSettings StyleNames StylePrint StyleSheetPath Subdivide Subfactorial Subgraph SubMinus SubPlus SubresultantPolynomialRemainders SubresultantPolynomials Subresultants Subscript SubscriptBox SubscriptBoxOptions Subscripted Subsequences Subset SubsetEqual SubsetMap SubsetQ Subsets SubStar SubstitutionSystem Subsuperscript SubsuperscriptBox SubsuperscriptBoxOptions Subtract SubtractFrom SubtractSides SubValues Succeeds SucceedsEqual SucceedsSlantEqual SucceedsTilde Success SuchThat Sum SumConvergence SummationLayer Sunday SunPosition Sunrise Sunset SuperDagger SuperMinus SupernovaData SuperPlus Superscript SuperscriptBox SuperscriptBoxOptions Superset SupersetEqual SuperStar Surd SurdForm SurfaceArea SurfaceColor SurfaceData SurfaceGraphics SurvivalDistribution SurvivalFunction SurvivalModel SurvivalModelFit SuspendPacket SuzukiDistribution SuzukiGroupSuz SwatchLegend Switch Symbol SymbolName SymletWavelet Symmetric SymmetricGroup SymmetricKey SymmetricMatrixQ SymmetricPolynomial SymmetricReduction Symmetrize SymmetrizedArray SymmetrizedArrayRules SymmetrizedDependentComponents SymmetrizedIndependentComponents SymmetrizedReplacePart SynchronousInitialization SynchronousUpdating Synonyms Syntax SyntaxForm SyntaxInformation SyntaxLength SyntaxPacket SyntaxQ SynthesizeMissingValues SystemDialogInput SystemException SystemGet SystemHelpPath SystemInformation SystemInformationData SystemInstall SystemModel SystemModeler SystemModelExamples SystemModelLinearize SystemModelParametricSimulate SystemModelPlot SystemModelProgressReporting SystemModelReliability SystemModels SystemModelSimulate SystemModelSimulateSensitivity SystemModelSimulationData SystemOpen SystemOptions SystemProcessData SystemProcesses SystemsConnectionsModel SystemsModelDelay SystemsModelDelayApproximate SystemsModelDelete SystemsModelDimensions SystemsModelExtract SystemsModelFeedbackConnect SystemsModelLabels SystemsModelLinearity SystemsModelMerge SystemsModelOrder SystemsModelParallelConnect SystemsModelSeriesConnect SystemsModelStateFeedbackConnect SystemsModelVectorRelativeOrders SystemStub SystemTest"+"Tab TabFilling Table TableAlignments TableDepth TableDirections TableForm TableHeadings TableSpacing TableView TableViewBox TableViewBoxBackground TableViewBoxOptions TabSpacings TabView TabViewBox TabViewBoxOptions TagBox TagBoxNote TagBoxOptions TaggingRules TagSet TagSetDelayed TagStyle TagUnset Take TakeDrop TakeLargest TakeLargestBy TakeList TakeSmallest TakeSmallestBy TakeWhile Tally Tan Tanh TargetDevice TargetFunctions TargetSystem TargetUnits TaskAbort TaskExecute TaskObject TaskRemove TaskResume Tasks TaskSuspend TaskWait TautologyQ TelegraphProcess TemplateApply TemplateArgBox TemplateBox TemplateBoxOptions TemplateEvaluate TemplateExpression TemplateIf TemplateObject TemplateSequence TemplateSlot TemplateSlotSequence TemplateUnevaluated TemplateVerbatim TemplateWith TemporalData TemporalRegularity Temporary TemporaryVariable TensorContract TensorDimensions TensorExpand TensorProduct TensorQ TensorRank TensorReduce TensorSymmetry TensorTranspose TensorWedge TestID TestReport TestReportObject TestResultObject Tetrahedron TetrahedronBox TetrahedronBoxOptions TeXForm TeXSave Text Text3DBox Text3DBoxOptions TextAlignment TextBand TextBoundingBox TextBox TextCases TextCell TextClipboardType TextContents TextData TextElement TextForm TextGrid TextJustification TextLine TextPacket TextParagraph TextPosition TextRecognize TextSearch TextSearchReport TextSentences TextString TextStructure TextStyle TextTranslation Texture TextureCoordinateFunction TextureCoordinateScaling TextWords Therefore ThermodynamicData ThermometerGauge Thick Thickness Thin Thinning ThisLink ThompsonGroupTh Thread ThreadingLayer ThreeJSymbol Threshold Through Throw ThueMorse Thumbnail Thursday Ticks TicksStyle TideData Tilde TildeEqual TildeFullEqual TildeTilde TimeConstrained TimeConstraint TimeDirection TimeFormat TimeGoal TimelinePlot TimeObject TimeObjectQ Times TimesBy TimeSeries TimeSeriesAggregate TimeSeriesForecast TimeSeriesInsert TimeSeriesInvertibility TimeSeriesMap TimeSeriesMapThread TimeSeriesModel TimeSeriesModelFit TimeSeriesResample TimeSeriesRescale TimeSeriesShift TimeSeriesThread TimeSeriesWindow TimeUsed TimeValue TimeWarpingCorrespondence TimeWarpingDistance TimeZone TimeZoneConvert TimeZoneOffset Timing Tiny TitleGrouping TitsGroupT ToBoxes ToCharacterCode ToColor ToContinuousTimeModel ToDate Today ToDiscreteTimeModel ToEntity ToeplitzMatrix ToExpression ToFileName Together Toggle ToggleFalse Toggler TogglerBar TogglerBox TogglerBoxOptions ToHeldExpression ToInvertibleTimeSeries TokenWords Tolerance ToLowerCase Tomorrow ToNumberField TooBig Tooltip TooltipBox TooltipBoxOptions TooltipDelay TooltipStyle Top TopHatTransform ToPolarCoordinates TopologicalSort ToRadicals ToRules ToSphericalCoordinates ToString Total TotalHeight TotalLayer TotalVariationFilter TotalWidth TouchPosition TouchscreenAutoZoom TouchscreenControlPlacement ToUpperCase Tr Trace TraceAbove TraceAction TraceBackward TraceDepth TraceDialog TraceForward TraceInternal TraceLevel TraceOff TraceOn TraceOriginal TracePrint TraceScan TrackedSymbols TrackingFunction TracyWidomDistribution TradingChart TraditionalForm TraditionalFunctionNotation TraditionalNotation TraditionalOrder TrainingProgressCheckpointing TrainingProgressFunction TrainingProgressMeasurements TrainingProgressReporting TrainingStoppingCriterion TransferFunctionCancel TransferFunctionExpand TransferFunctionFactor TransferFunctionModel TransferFunctionPoles TransferFunctionTransform TransferFunctionZeros TransformationClass TransformationFunction TransformationFunctions TransformationMatrix TransformedDistribution TransformedField TransformedProcess TransformedRegion TransitionDirection TransitionDuration TransitionEffect TransitiveClosureGraph TransitiveReductionGraph Translate TranslationOptions TranslationTransform Transliterate Transparent TransparentColor Transpose TransposeLayer TrapSelection TravelDirections TravelDirectionsData TravelDistance TravelDistanceList TravelMethod TravelTime TreeForm TreeGraph TreeGraphQ TreePlot TrendStyle Triangle TriangleCenter TriangleConstruct TriangleMeasurement TriangleWave TriangularDistribution TriangulateMesh Trig TrigExpand TrigFactor TrigFactorList Trigger TrigReduce TrigToExp TrimmedMean TrimmedVariance TropicalStormData True TrueQ TruncatedDistribution TruncatedPolyhedron TsallisQExponentialDistribution TsallisQGaussianDistribution TTest Tube TubeBezierCurveBox TubeBezierCurveBoxOptions TubeBox TubeBoxOptions TubeBSplineCurveBox TubeBSplineCurveBoxOptions Tuesday TukeyLambdaDistribution TukeyWindow TunnelData Tuples TuranGraph TuringMachine TuttePolynomial TwoWayRule Typed TypeSpecifier"+"UnateQ Uncompress UnconstrainedParameters Undefined UnderBar Underflow Underlined Underoverscript UnderoverscriptBox UnderoverscriptBoxOptions Underscript UnderscriptBox UnderscriptBoxOptions UnderseaFeatureData UndirectedEdge UndirectedGraph UndirectedGraphQ UndoOptions UndoTrackedVariables Unequal UnequalTo Unevaluated UniformDistribution UniformGraphDistribution UniformPolyhedron UniformSumDistribution Uninstall Union UnionPlus Unique UnitaryMatrixQ UnitBox UnitConvert UnitDimensions Unitize UnitRootTest UnitSimplify UnitStep UnitSystem UnitTriangle UnitVector UnitVectorLayer UnityDimensions UniverseModelData UniversityData UnixTime Unprotect UnregisterExternalEvaluator UnsameQ UnsavedVariables Unset UnsetShared UntrackedVariables Up UpArrow UpArrowBar UpArrowDownArrow Update UpdateDynamicObjects UpdateDynamicObjectsSynchronous UpdateInterval UpdateSearchIndex UpDownArrow UpEquilibrium UpperCaseQ UpperLeftArrow UpperRightArrow UpperTriangularize UpperTriangularMatrixQ Upsample UpSet UpSetDelayed UpTee UpTeeArrow UpTo UpValues URL URLBuild URLDecode URLDispatcher URLDownload URLDownloadSubmit URLEncode URLExecute URLExpand URLFetch URLFetchAsynchronous URLParse URLQueryDecode URLQueryEncode URLRead URLResponseTime URLSave URLSaveAsynchronous URLShorten URLSubmit UseGraphicsRange UserDefinedWavelet Using UsingFrontEnd UtilityFunction"+"V2Get ValenceErrorHandling ValidationLength ValidationSet Value ValueBox ValueBoxOptions ValueDimensions ValueForm ValuePreprocessingFunction ValueQ Values ValuesData Variables Variance VarianceEquivalenceTest VarianceEstimatorFunction VarianceGammaDistribution VarianceTest VectorAngle VectorAround VectorColorFunction VectorColorFunctionScaling VectorDensityPlot VectorGlyphData VectorGreater VectorGreaterEqual VectorLess VectorLessEqual VectorMarkers VectorPlot VectorPlot3D VectorPoints VectorQ Vectors VectorScale VectorStyle Vee Verbatim Verbose VerboseConvertToPostScriptPacket VerificationTest VerifyConvergence VerifyDerivedKey VerifyDigitalSignature VerifyInterpretation VerifySecurityCertificates VerifySolutions VerifyTestAssumptions Version VersionNumber VertexAdd VertexCapacity VertexColors VertexComponent VertexConnectivity VertexContract VertexCoordinateRules VertexCoordinates VertexCorrelationSimilarity VertexCosineSimilarity VertexCount VertexCoverQ VertexDataCoordinates VertexDegree VertexDelete VertexDiceSimilarity VertexEccentricity VertexInComponent VertexInDegree VertexIndex VertexJaccardSimilarity VertexLabeling VertexLabels VertexLabelStyle VertexList VertexNormals VertexOutComponent VertexOutDegree VertexQ VertexRenderingFunction VertexReplace VertexShape VertexShapeFunction VertexSize VertexStyle VertexTextureCoordinates VertexWeight VertexWeightedGraphQ Vertical VerticalBar VerticalForm VerticalGauge VerticalSeparator VerticalSlider VerticalTilde ViewAngle ViewCenter ViewMatrix ViewPoint ViewPointSelectorSettings ViewPort ViewProjection ViewRange ViewVector ViewVertical VirtualGroupData Visible VisibleCell VoiceStyleData VoigtDistribution VolcanoData Volume VonMisesDistribution VoronoiMesh"+"WaitAll WaitAsynchronousTask WaitNext WaitUntil WakebyDistribution WalleniusHypergeometricDistribution WaringYuleDistribution WarpingCorrespondence WarpingDistance WatershedComponents WatsonUSquareTest WattsStrogatzGraphDistribution WaveletBestBasis WaveletFilterCoefficients WaveletImagePlot WaveletListPlot WaveletMapIndexed WaveletMatrixPlot WaveletPhi WaveletPsi WaveletScale WaveletScalogram WaveletThreshold WeaklyConnectedComponents WeaklyConnectedGraphComponents WeaklyConnectedGraphQ WeakStationarity WeatherData WeatherForecastData WebAudioSearch WebElementObject WeberE WebExecute WebImage WebImageSearch WebSearch WebSessionObject WebSessions WebWindowObject Wedge Wednesday WeibullDistribution WeierstrassE1 WeierstrassE2 WeierstrassE3 WeierstrassEta1 WeierstrassEta2 WeierstrassEta3 WeierstrassHalfPeriods WeierstrassHalfPeriodW1 WeierstrassHalfPeriodW2 WeierstrassHalfPeriodW3 WeierstrassInvariantG2 WeierstrassInvariantG3 WeierstrassInvariants WeierstrassP WeierstrassPPrime WeierstrassSigma WeierstrassZeta WeightedAdjacencyGraph WeightedAdjacencyMatrix WeightedData WeightedGraphQ Weights WelchWindow WheelGraph WhenEvent Which While White WhiteNoiseProcess WhitePoint Whitespace WhitespaceCharacter WhittakerM WhittakerW WienerFilter WienerProcess WignerD WignerSemicircleDistribution WikipediaData WikipediaSearch WilksW WilksWTest WindDirectionData WindingCount WindingPolygon WindowClickSelect WindowElements WindowFloating WindowFrame WindowFrameElements WindowMargins WindowMovable WindowOpacity WindowPersistentStyles WindowSelected WindowSize WindowStatusArea WindowTitle WindowToolbars WindowWidth WindSpeedData WindVectorData WinsorizedMean WinsorizedVariance WishartMatrixDistribution With WolframAlpha WolframAlphaDate WolframAlphaQuantity WolframAlphaResult WolframLanguageData Word WordBoundary WordCharacter WordCloud WordCount WordCounts WordData WordDefinition WordFrequency WordFrequencyData WordList WordOrientation WordSearch WordSelectionFunction WordSeparators WordSpacings WordStem WordTranslation WorkingPrecision WrapAround Write WriteLine WriteString Wronskian"+"XMLElement XMLObject XMLTemplate Xnor Xor XYZColor"+"Yellow Yesterday YuleDissimilarity"+"ZernikeR ZeroSymmetric ZeroTest ZeroWidthTimes Zeta ZetaZero ZIPCodeData ZipfDistribution ZoomCenter ZoomFactor ZTest ZTransform"+"$Aborted $ActivationGroupID $ActivationKey $ActivationUserRegistered $AddOnsDirectory $AllowExternalChannelFunctions $AssertFunction $Assumptions $AsynchronousTask $AudioInputDevices $AudioOutputDevices $BaseDirectory $BatchInput $BatchOutput $BlockchainBase $BoxForms $ByteOrdering $CacheBaseDirectory $Canceled $ChannelBase $CharacterEncoding $CharacterEncodings $CloudBase $CloudConnected $CloudCreditsAvailable $CloudEvaluation $CloudExpressionBase $CloudObjectNameFormat $CloudObjectURLType $CloudRootDirectory $CloudSymbolBase $CloudUserID $CloudUserUUID $CloudVersion $CloudVersionNumber $CloudWolframEngineVersionNumber $CommandLine $CompilationTarget $ConditionHold $ConfiguredKernels $Context $ContextPath $ControlActiveSetting $Cookies $CookieStore $CreationDate $CurrentLink $CurrentTask $CurrentWebSession $DateStringFormat $DefaultAudioInputDevice $DefaultAudioOutputDevice $DefaultFont $DefaultFrontEnd $DefaultImagingDevice $DefaultLocalBase $DefaultMailbox $DefaultNetworkInterface $DefaultPath $Display $DisplayFunction $DistributedContexts $DynamicEvaluation $Echo $EmbedCodeEnvironments $EmbeddableServices $EntityStores $Epilog $EvaluationCloudBase $EvaluationCloudObject $EvaluationEnvironment $ExportFormats $Failed $FinancialDataSource $FontFamilies $FormatType $FrontEnd $FrontEndSession $GeoEntityTypes $GeoLocation $GeoLocationCity $GeoLocationCountry $GeoLocationPrecision $GeoLocationSource $HistoryLength $HomeDirectory $HTMLExportRules $HTTPCookies $HTTPRequest $IgnoreEOF $ImageFormattingWidth $ImagingDevice $ImagingDevices $ImportFormats $IncomingMailSettings $InitialDirectory $Initialization $InitializationContexts $Input $InputFileName $InputStreamMethods $Inspector $InstallationDate $InstallationDirectory $InterfaceEnvironment $InterpreterTypes $IterationLimit $KernelCount $KernelID $Language $LaunchDirectory $LibraryPath $LicenseExpirationDate $LicenseID $LicenseProcesses $LicenseServer $LicenseSubprocesses $LicenseType $Line $Linked $LinkSupported $LoadedFiles $LocalBase $LocalSymbolBase $MachineAddresses $MachineDomain $MachineDomains $MachineEpsilon $MachineID $MachineName $MachinePrecision $MachineType $MaxExtraPrecision $MaxLicenseProcesses $MaxLicenseSubprocesses $MaxMachineNumber $MaxNumber $MaxPiecewiseCases $MaxPrecision $MaxRootDegree $MessageGroups $MessageList $MessagePrePrint $Messages $MinMachineNumber $MinNumber $MinorReleaseNumber $MinPrecision $MobilePhone $ModuleNumber $NetworkConnected $NetworkInterfaces $NetworkLicense $NewMessage $NewSymbol $Notebooks $NoValue $NumberMarks $Off $OperatingSystem $Output $OutputForms $OutputSizeLimit $OutputStreamMethods $Packages $ParentLink $ParentProcessID $PasswordFile $PatchLevelID $Path $PathnameSeparator $PerformanceGoal $Permissions $PermissionsGroupBase $PersistenceBase $PersistencePath $PipeSupported $PlotTheme $Post $Pre $PreferencesDirectory $PreInitialization $PrePrint $PreRead $PrintForms $PrintLiteral $Printout3DPreviewer $ProcessID $ProcessorCount $ProcessorType $ProductInformation $ProgramName $PublisherID $RandomState $RecursionLimit $RegisteredDeviceClasses $RegisteredUserName $ReleaseNumber $RequesterAddress $RequesterWolframID $RequesterWolframUUID $ResourceSystemBase $RootDirectory $ScheduledTask $ScriptCommandLine $ScriptInputString $SecuredAuthenticationKeyTokens $ServiceCreditsAvailable $Services $SessionID $SetParentLink $SharedFunctions $SharedVariables $SoundDisplay $SoundDisplayFunction $SourceLink $SSHAuthentication $SummaryBoxDataSizeLimit $SuppressInputFormHeads $SynchronousEvaluation $SyntaxHandler $System $SystemCharacterEncoding $SystemID $SystemMemory $SystemShell $SystemTimeZone $SystemWordLength $TemplatePath $TemporaryDirectory $TemporaryPrefix $TestFileName $TextStyle $TimedOut $TimeUnit $TimeZone $TimeZoneEntity $TopDirectory $TraceOff $TraceOn $TracePattern $TracePostAction $TracePreAction $UnitSystem $Urgent $UserAddOnsDirectory $UserAgentLanguages $UserAgentMachine $UserAgentName $UserAgentOperatingSystem $UserAgentString $UserAgentVersion $UserBaseDirectory $UserDocumentsDirectory $Username $UserName $UserURLBase $Version $VersionNumber $VoiceStyles $WolframID $WolframUUID",contains:[hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("nginx",function(hljs){var VAR={className:"variable",variants:[{begin:/\$\d+/},{begin:/\$\{/,end:/}/},{begin:"[\\$\\@]"+hljs.UNDERSCORE_IDENT_RE}]};var DEFAULT={endsWithParent:true,lexemes:"[a-z/_]+",keywords:{literal:"on off yes no true false none blocked debug info notice warn error crit "+"select break last permanent redirect kqueue rtsig epoll poll /dev/poll"},relevance:0,illegal:"=>",contains:[hljs.HASH_COMMENT_MODE,{className:"string",contains:[hljs.BACKSLASH_ESCAPE,VAR],variants:[{begin:/"/,end:/"/},{begin:/'/,end:/'/}]},{begin:"([a-z]+):/",end:"\\s",endsWithParent:true,excludeEnd:true,contains:[VAR]},{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,VAR],variants:[{begin:"\\s\\^",end:"\\s|{|;",returnEnd:true},{begin:"~\\*?\\s+",end:"\\s|{|;",returnEnd:true},{begin:"\\*(\\.[a-z\\-]+)+"},{begin:"([a-z\\-]+\\.)+\\*"}]},{className:"number",begin:"\\b\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b"},{className:"number",begin:"\\b\\d+[kKmMgGdshdwy]*\\b",relevance:0},VAR]};return{aliases:["nginxconf"],contains:[hljs.HASH_COMMENT_MODE,{begin:hljs.UNDERSCORE_IDENT_RE+"\\s+{",returnBegin:true,end:"{",contains:[{className:"section",begin:hljs.UNDERSCORE_IDENT_RE}],relevance:0},{begin:hljs.UNDERSCORE_IDENT_RE+"\\s",end:";|{",returnBegin:true,contains:[{className:"attribute",begin:hljs.UNDERSCORE_IDENT_RE,starts:DEFAULT}],relevance:0}],illegal:"[^\\s\\}]"}});hljs.registerLanguage("n1ql",function(hljs){return{case_insensitive:true,contains:[{beginKeywords:"build create index delete drop explain infer|10 insert merge prepare select update upsert|10",end:/;/,endsWithParent:true,keywords:{keyword:"all alter analyze and any array as asc begin between binary boolean break bucket build by call "+"case cast cluster collate collection commit connect continue correlate cover create database "+"dataset datastore declare decrement delete derived desc describe distinct do drop each element "+"else end every except exclude execute exists explain fetch first flatten for force from "+"function grant group gsi having if ignore ilike in include increment index infer inline inner "+"insert intersect into is join key keys keyspace known last left let letting like limit lsm map "+"mapping matched materialized merge minus namespace nest not number object offset on "+"option or order outer over parse partition password path pool prepare primary private privilege "+"procedure public raw realm reduce rename return returning revoke right role rollback satisfies "+"schema select self semi set show some start statistics string system then to transaction trigger "+"truncate under union unique unknown unnest unset update upsert use user using validate value "+"valued values via view when where while with within work xor",literal:"true false null missing|5",built_in:"array_agg array_append array_concat array_contains array_count array_distinct array_ifnull array_length "+"array_max array_min array_position array_prepend array_put array_range array_remove array_repeat array_replace "+"array_reverse array_sort array_sum avg count max min sum greatest least ifmissing ifmissingornull ifnull "+"missingif nullif ifinf ifnan ifnanorinf naninf neginfif posinfif clock_millis clock_str date_add_millis "+"date_add_str date_diff_millis date_diff_str date_part_millis date_part_str date_trunc_millis date_trunc_str "+"duration_to_str millis str_to_millis millis_to_str millis_to_utc millis_to_zone_name now_millis now_str "+"str_to_duration str_to_utc str_to_zone_name decode_json encode_json encoded_size poly_length base64 base64_encode "+"base64_decode meta uuid abs acos asin atan atan2 ceil cos degrees e exp ln log floor pi power radians random "+"round sign sin sqrt tan trunc object_length object_names object_pairs object_inner_pairs object_values "+"object_inner_values object_add object_put object_remove object_unwrap regexp_contains regexp_like regexp_position "+"regexp_replace contains initcap length lower ltrim position repeat replace rtrim split substr title trim upper "+"isarray isatom isboolean isnumber isobject isstring type toarray toatom toboolean tonumber toobject tostring"},contains:[{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"symbol",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE],relevance:2},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("pf",function(hljs){var MACRO={className:"variable",begin:/\$[\w\d#@][\w\d_]*/};var TABLE={className:"variable",begin:/<(?!\/)/,end:/>/};var QUOTE_STRING={className:"string",begin:/"/,end:/"/};return{aliases:["pf.conf"],lexemes:/[a-z0-9_<>-]+/,keywords:{built_in:"block match pass load anchor|5 antispoof|10 set table",keyword:"in out log quick on rdomain inet inet6 proto from port os to route"+"allow-opts divert-packet divert-reply divert-to flags group icmp-type"+"icmp6-type label once probability recieved-on rtable prio queue"+"tos tag tagged user keep fragment for os drop"+"af-to|10 binat-to|10 nat-to|10 rdr-to|10 bitmask least-stats random round-robin"+"source-hash static-port"+"dup-to reply-to route-to"+"parent bandwidth default min max qlimit"+"block-policy debug fingerprints hostid limit loginterface optimization"+"reassemble ruleset-optimization basic none profile skip state-defaults"+"state-policy timeout"+"const counters persist"+"no modulate synproxy state|5 floating if-bound no-sync pflow|10 sloppy"+"source-track global rule max-src-nodes max-src-states max-src-conn"+"max-src-conn-rate overload flush"+"scrub|5 max-mss min-ttl no-df|10 random-id",literal:"all any no-route self urpf-failed egress|5 unknown"},contains:[hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE,hljs.QUOTE_STRING_MODE,MACRO,TABLE]}});hljs.registerLanguage("llvm",function(hljs){var identifier="([-a-zA-Z$._][\\w\\-$.]*)";return{keywords:"begin end true false declare define global "+"constant private linker_private internal "+"available_externally linkonce linkonce_odr weak "+"weak_odr appending dllimport dllexport common "+"default hidden protected extern_weak external "+"thread_local zeroinitializer undef null to tail "+"target triple datalayout volatile nuw nsw nnan "+"ninf nsz arcp fast exact inbounds align "+"addrspace section alias module asm sideeffect "+"gc dbg linker_private_weak attributes blockaddress "+"initialexec localdynamic localexec prefix unnamed_addr "+"ccc fastcc coldcc x86_stdcallcc x86_fastcallcc "+"arm_apcscc arm_aapcscc arm_aapcs_vfpcc ptx_device "+"ptx_kernel intel_ocl_bicc msp430_intrcc spir_func "+"spir_kernel x86_64_sysvcc x86_64_win64cc x86_thiscallcc "+"cc c signext zeroext inreg sret nounwind "+"noreturn noalias nocapture byval nest readnone "+"readonly inlinehint noinline alwaysinline optsize ssp "+"sspreq noredzone noimplicitfloat naked builtin cold "+"nobuiltin noduplicate nonlazybind optnone returns_twice "+"sanitize_address sanitize_memory sanitize_thread sspstrong "+"uwtable returned type opaque eq ne slt sgt "+"sle sge ult ugt ule uge oeq one olt ogt "+"ole oge ord uno ueq une x acq_rel acquire "+"alignstack atomic catch cleanup filter inteldialect "+"max min monotonic nand personality release seq_cst "+"singlethread umax umin unordered xchg add fadd "+"sub fsub mul fmul udiv sdiv fdiv urem srem "+"frem shl lshr ashr and or xor icmp fcmp "+"phi call trunc zext sext fptrunc fpext uitofp "+"sitofp fptoui fptosi inttoptr ptrtoint bitcast "+"addrspacecast select va_arg ret br switch invoke "+"unwind unreachable indirectbr landingpad resume "+"malloc alloca free load store getelementptr "+"extractelement insertelement shufflevector getresult "+"extractvalue insertvalue atomicrmw cmpxchg fence "+"argmemonly double",contains:[{className:"keyword",begin:"i\\d+"},hljs.COMMENT(";","\\n",{relevance:0}),hljs.QUOTE_STRING_MODE,{className:"string",variants:[{begin:'"',end:'[^\\\\]"'}],relevance:0},{className:"title",variants:[{begin:"@"+identifier},{begin:"@\\d+"},{begin:"!"+identifier},{begin:"!\\d+"+identifier}]},{className:"symbol",variants:[{begin:"%"+identifier},{begin:"%\\d+"},{begin:"#\\d+"}]},{className:"number",variants:[{begin:"0[xX][a-fA-F0-9]+"},{begin:"-?\\d+(?:[.]\\d+)?(?:[eE][-+]?\\d+(?:[.]\\d+)?)?"}],relevance:0}]}});hljs.registerLanguage("avrasm",function(hljs){return{case_insensitive:true,lexemes:"\\.?"+hljs.IDENT_RE,keywords:{keyword:"adc add adiw and andi asr bclr bld brbc brbs brcc brcs break breq brge brhc brhs "+"brid brie brlo brlt brmi brne brpl brsh brtc brts brvc brvs bset bst call cbi cbr "+"clc clh cli cln clr cls clt clv clz com cp cpc cpi cpse dec eicall eijmp elpm eor "+"fmul fmuls fmulsu icall ijmp in inc jmp ld ldd ldi lds lpm lsl lsr mov movw mul "+"muls mulsu neg nop or ori out pop push rcall ret reti rjmp rol ror sbc sbr sbrc sbrs "+"sec seh sbi sbci sbic sbis sbiw sei sen ser ses set sev sez sleep spm st std sts sub "+"subi swap tst wdr",built_in:"r0 r1 r2 r3 r4 r5 r6 r7 r8 r9 r10 r11 r12 r13 r14 r15 r16 r17 r18 r19 r20 r21 r22 "+"r23 r24 r25 r26 r27 r28 r29 r30 r31 x|0 xh xl y|0 yh yl z|0 zh zl "+"ucsr1c udr1 ucsr1a ucsr1b ubrr1l ubrr1h ucsr0c ubrr0h tccr3c tccr3a tccr3b tcnt3h "+"tcnt3l ocr3ah ocr3al ocr3bh ocr3bl ocr3ch ocr3cl icr3h icr3l etimsk etifr tccr1c "+"ocr1ch ocr1cl twcr twdr twar twsr twbr osccal xmcra xmcrb eicra spmcsr spmcr portg "+"ddrg ping portf ddrf sreg sph spl xdiv rampz eicrb eimsk gimsk gicr eifr gifr timsk "+"tifr mcucr mcucsr tccr0 tcnt0 ocr0 assr tccr1a tccr1b tcnt1h tcnt1l ocr1ah ocr1al "+"ocr1bh ocr1bl icr1h icr1l tccr2 tcnt2 ocr2 ocdr wdtcr sfior eearh eearl eedr eecr "+"porta ddra pina portb ddrb pinb portc ddrc pinc portd ddrd pind spdr spsr spcr udr0 "+"ucsr0a ucsr0b ubrr0l acsr admux adcsr adch adcl porte ddre pine pinf",meta:".byte .cseg .db .def .device .dseg .dw .endmacro .equ .eseg .exit .include .list "+".listmac .macro .nolist .org .set"},contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(";","$",{relevance:0}),hljs.C_NUMBER_MODE,hljs.BINARY_NUMBER_MODE,{className:"number",begin:"\\b(\\$[a-zA-Z0-9]+|0o[0-7]+)"},hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",illegal:"[^\\\\][^']"},{className:"symbol",begin:"^[A-Za-z0-9_.$]+:"},{className:"meta",begin:"#",end:"$"},{className:"subst",begin:"@[0-9]+"}]}});</script>
+
+<style type="text/css">
+  code{white-space: pre-wrap;}
+  span.smallcaps{font-variant: small-caps;}
+  span.underline{text-decoration: underline;}
+  div.column{display: inline-block; vertical-align: top; width: 50%;}
+  div.hanging-indent{margin-left: 1.5em; text-indent: -1.5em;}
+  ul.task-list{list-style: none;}
+    </style>
+
+<style type="text/css">code{white-space: pre;}</style>
+<script type="text/javascript">
+if (window.hljs) {
+  hljs.configure({languages: []});
+  hljs.initHighlightingOnLoad();
+  if (document.readyState && document.readyState === "complete") {
+    window.setTimeout(function() { hljs.initHighlighting(); }, 0);
+  }
+}
+</script>
+
+
+
+
+
+
+
+
+
+<style type="text/css">
+.main-container {
+  max-width: 940px;
+  margin-left: auto;
+  margin-right: auto;
+}
+img {
+  max-width:100%;
+}
+.tabbed-pane {
+  padding-top: 12px;
+}
+.html-widget {
+  margin-bottom: 20px;
+}
+button.code-folding-btn:focus {
+  outline: none;
+}
+summary {
+  display: list-item;
+}
+details > summary > p:only-child {
+  display: inline;
+}
+pre code {
+  padding: 0;
+}
+</style>
+
+
+
+<!-- tabsets -->
+
+<style type="text/css">
+.tabset-dropdown > .nav-tabs {
+  display: inline-table;
+  max-height: 500px;
+  min-height: 44px;
+  overflow-y: auto;
+  border: 1px solid #ddd;
+  border-radius: 4px;
+}
+
+.tabset-dropdown > .nav-tabs > li.active:before, .tabset-dropdown > .nav-tabs.nav-tabs-open:before {
+  content: "\e259";
+  font-family: 'Glyphicons Halflings';
+  display: inline-block;
+  padding: 10px;
+  border-right: 1px solid #ddd;
+}
+
+.tabset-dropdown > .nav-tabs.nav-tabs-open > li.active:before {
+  content: "\e258";
+  font-family: 'Glyphicons Halflings';
+  border: none;
+}
+
+.tabset-dropdown > .nav-tabs > li.active {
+  display: block;
+}
+
+.tabset-dropdown > .nav-tabs > li > a,
+.tabset-dropdown > .nav-tabs > li > a:focus,
+.tabset-dropdown > .nav-tabs > li > a:hover {
+  border: none;
+  display: inline-block;
+  border-radius: 4px;
+  background-color: transparent;
+}
+
+.tabset-dropdown > .nav-tabs.nav-tabs-open > li {
+  display: block;
+  float: none;
+}
+
+.tabset-dropdown > .nav-tabs > li {
+  display: none;
+}
+</style>
+
+<!-- code folding -->
+
+
+
+
+</head>
+
+<body>
+
+
+<div class="container-fluid main-container">
+
+
+
+
+<div id="header">
+
+
+
+<h1 class="title toc-ignore">SMR Vignette</h1>
+<h4 class="author">Matti Rantanen</h4>
+<h4 class="date">2023-01-19</h4>
+
+</div>
+
+<div id="TOC">
+<ul>
+<li><a href="#introduction">Introduction</a></li>
+<li><a href="#splines">Splines</a>
+<ul>
+<li><a href="#knots">Knots</a></li>
+</ul></li>
+<li><a href="#smr">SMR</a>
+<ul>
+<li><a href="#mortality-external-cohort-and-popmort-data">Mortality:
+External cohort and popmort data</a></li>
+<li><a href="#splines-1">splines</a></li>
+</ul></li>
+</ul>
+</div>
+
+<pre class="r"><code>library(popEpi)
+library(Epi)</code></pre>
+<div id="introduction" class="section level1">
+<h1>Introduction</h1>
+<p>Standardized incidence ratio (SIR) or mortality ratio (SMR) is a
+ratio of observed and expected cases. Observed cases is the absolute
+number of cases in the cohort. The expected cases are derived by
+multiplying the cohort person-years with reference populations rate. The
+rate should be stratified or adjusted by confounding factors. Usually
+these are age group, gender, calendar period and possibly a cancer type
+or other confounding variable. Also a social economic status or area
+variable can be used.</p>
+<p>In reference population the expected rate in strata <span class="math inline">\(j\)</span> is <span class="math inline">\(\lambda_j = d_j\)</span> / <span class="math inline">\(n_j\)</span>, where <span class="math inline">\(d_j\)</span> is observed cases and <span class="math inline">\(n_j\)</span> is observed person years. Now the SIR
+can be written as a ratio <span class="math display">\[
+SIR = \frac{ \sum d_j }{\sum n_j \lambda_j} = \frac{D}{E}
+\]</span> where <span class="math inline">\(D\)</span> is the observed
+cases in cohort population and <span class="math inline">\(E\)</span> is
+the expected number. Univariate confidence intervals are based on exact
+values of Poisson distribution and the formula for p-value is <span class="math display">\[
+\chi^2 = \frac{ (|O - E| -0.5)^2 }{E}.
+\]</span> Modelled SIR is a Poisson regression model with log-link and
+cohorts person-years as a offset.</p>
+<p>The homogeneity of SIR’s can be tested using a likelihood ratio test
+in Poisson modelled SIRs.</p>
+<p>The same workflow applies for standardised mortality ratios.</p>
+</div>
+<div id="splines" class="section level1">
+<h1>Splines</h1>
+<p>A continuous spline function can be fitted for time variables,
+e.g. age-group. Idea of the splines is to smooth the SMR estimates and
+do inference from the curve figure. This requires pre-defined
+knots/nodes that are used to fit the spline curve. Selecting the number
+of knots and knot places is a very subjective matter and there are three
+options to pass spline knots to function.</p>
+<p>It’s good practice to try between different knot settings for
+realistic spline estimates. Overfitting might cause unintentional
+artefacts in the estimate and underfitting might smooth away interesting
+patterns.</p>
+<p>The spline variable should be as continuous as possible, say from 18
+to 100 time points. But when splitting time in too narrow intervals,
+random variation might occur in the expected or population rate values.
+Therefore it’s also possible to do two variables for age or period:
+first with wider intervals for standardisation and second with narrow
+intervals for the spline.</p>
+<div id="knots" class="section level2">
+<h2>Knots</h2>
+<p>There are three options to for assigning knots to the spline:</p>
+<ol style="list-style-type: decimal">
+<li><p>A vector of numbers of knots for each spline variable. Number of
+knots includes the boundary knots, so that the minimum number of knots
+is 2, which is a log linear association. The knots are placed
+automatically using the quantiles of observed cases.</p></li>
+<li><p>A list of vectors of predefined knot places. Number of vectors
+needs to match the length of spline variables. And each vector has to
+have at least the minimum and maximum for boundary knots.</p></li>
+<li><p>NULL will automatically finds the optimal number of knots based
+on AIC. Knots are placed according the quantiles of observed cases. This
+is usually a good place to start the fitting process.</p></li>
+</ol>
+<p>Number of knots and knot places are always found in output.</p>
+</div>
+</div>
+<div id="smr" class="section level1">
+<h1>SMR</h1>
+<div id="mortality-external-cohort-and-popmort-data" class="section level2">
+<h2>Mortality: External cohort and popmort data</h2>
+<p>Estimate SMR of a simulated cohort of Finnish female rectal cancer
+patients, <code>sire</code>. Death rates for each age, period and sex is
+available in <code>popmort</code> dataset.</p>
+<p>For more information about the dataset see <code>help(popmort)</code>
+and <code>help(sire)</code>.</p>
+<pre class="r"><code>data(sire)
+data(popmort)
+c &lt;- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )</code></pre>
+<pre><code>## dropped 16 rows where entry == exit</code></pre>
+<pre class="r"><code>se &lt;- sir( coh.data = c, coh.obs = &#39;from0to2&#39;, coh.pyrs = &#39;pyrs&#39;,
+           ref.data = popmort, ref.rate = &#39;haz&#39;, 
+           adjust = c(&#39;agegroup&#39;,&#39;year&#39;,&#39;sex&#39;), print =&#39;fot&#39;)
+se</code></pre>
+<pre><code>## SIR (adjusted by agegroup, year, sex) with 95% confidence intervals (profile) 
+## Test for homogeneity: p = 0.735 
+## 
+##  Total sir: 1.01 (0.95-1.06)
+##  Total observed: 1490
+##  Total expected: 1482.13
+##  Total person-years: 39906 
+## 
+## 
+##    fot observed expected     pyrs  sir sir.lo sir.hi p_value
+## 1:   0     1226  1214.54 34445.96 1.01   0.95   1.07  0.7423
+## 2:  10      264   267.59  5459.96 0.99   0.87   1.11  0.8262</code></pre>
+<p>SMR’s for other causes is 1 for both follow-up intervals. Also the
+p-value suggest that there is no heterogeneity between SMR estimates
+(p=0.735).</p>
+<p>The total mortality can be estimated by modifying the
+<code>status</code> argument. Now we want to account all deaths,
+i.e. status is 1 or 2.</p>
+<pre class="r"><code>c &lt;- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )</code></pre>
+<pre><code>## dropped 16 rows where entry == exit</code></pre>
+<pre class="r"><code>se &lt;- sir( coh.data = c, coh.obs = &#39;from0to1&#39;, coh.pyrs = &#39;pyrs&#39;,
+           ref.data = popmort, ref.rate = &#39;haz&#39;, 
+           adjust = c(&#39;agegroup&#39;,&#39;year&#39;,&#39;sex&#39;), print =&#39;fot&#39;)
+se</code></pre>
+<pre><code>## SIR (adjusted by agegroup, year, sex) with 95% confidence intervals (profile) 
+## Test for homogeneity: p &lt; 0.001 
+## 
+##  Total sir: 3.08 (2.99-3.17)
+##  Total observed: 4559
+##  Total expected: 1482.13
+##  Total person-years: 39906 
+## 
+## 
+##    fot observed expected     pyrs  sir sir.lo sir.hi p_value
+## 1:   0     4264  1214.54 34445.96 3.51   3.41   3.62   0.000
+## 2:  10      295   267.59  5459.96 1.10   0.98   1.23   0.094</code></pre>
+<p>Now the estimates for follow-up intervals seems to differ
+significantly, p = 0. Plotting SMR (S3-method for
+<code>sir</code>-object) is easily done using default plot-function.</p>
+<pre class="r"><code>plot(se, col = 2:3)
+title(&#39;SMR for follow-up categories&#39;)</code></pre>
+<p><img src="" width="576" /></p>
+</div>
+<div id="splines-1" class="section level2">
+<h2>splines</h2>
+<p>Lets fit splines for the follow-up time and age group using two
+different options: the splines are fitted in different model and in same
+model, <code>dependent.splines</code>.</p>
+<pre class="r"><code>c &lt;- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
+              aggre = list(fot, agegroup = age, year = per, sex) )</code></pre>
+<pre><code>## dropped 16 rows where entry == exit</code></pre>
+<pre class="r"><code>sf &lt;- sirspline( coh.data = c, coh.obs = &#39;from0to1&#39;, coh.pyrs = &#39;pyrs&#39;, 
+                 ref.data = popmort, ref.rate = &#39;haz&#39;, 
+                 adjust = c(&#39;agegroup&#39;,&#39;year&#39;,&#39;sex&#39;),
+                 spline = c(&#39;agegroup&#39;,&#39;fot&#39;), dependent.splines=FALSE)
+
+st &lt;- sirspline( coh.data = c, coh.obs = &#39;from0to1&#39;, coh.pyrs = &#39;pyrs&#39;, 
+                 ref.data = popmort, ref.rate = &#39;haz&#39;, 
+                 adjust = c(&#39;agegroup&#39;,&#39;year&#39;,&#39;sex&#39;),
+                 spline = c(&#39;agegroup&#39;,&#39;fot&#39;), dependent.splines = TRUE)
+
+plot(sf, col=2, log=TRUE)
+title(&#39;Splines fitted in different models&#39;)</code></pre>
+<p><img src="" width="576" /></p>
+<pre class="r"><code>plot(st, col=4, log=TRUE)
+title(&#39;Splines are dependent&#39;)</code></pre>
+<p><img src="" width="576" /></p>
+<p>In dependent spline the <code>fot</code> is the ratio with zero time
+as reference point. Reference points can be altered. Here age group
+profile is assumed to be same for every follow-up time. SMR is 0.2 times
+from 0 to 10 years of follow-up.</p>
+<p>Splines can also be stratified using the <code>print</code> argument.
+For example we split the death time in two time periods and test if the
+age group splines are equal.</p>
+<pre class="r"><code>c$year.cat &lt;- ifelse(c$year &lt; 2002, 1, 2)
+sy &lt;- sirspline( coh.data = c, coh.obs = &#39;from0to1&#39;, coh.pyrs = &#39;pyrs&#39;, 
+                 ref.data = popmort, ref.rate = &#39;haz&#39;, 
+                 adjust = c(&#39;agegroup&#39;,&#39;year&#39;,&#39;sex&#39;),
+                 spline = c(&#39;agegroup&#39;), print = &#39;year.cat&#39;)
+plot(sy, log=TRUE)
+legend(&#39;topright&#39;, c(&#39;before 2002&#39;,&#39;after 2002&#39;), lty=1, col=c(1,2))</code></pre>
+<p><img src="" width="576" /></p>
+<p>For category before 2002 the SMR seems to be higher after the age of
+50. Also the p-value (&lt;0.0001) indicates that there is a difference
+in age group trends before and after year 2002. P-value is a likelihood
+ratio test that compares models where splines are fitted together and
+separately.</p>
+<pre class="r"><code>print(sy)</code></pre>
+<pre><code>## agegroup: p &lt; 0.001
+## NA: p = NA
+## NA: p = NA
+## 
+##   levels  colour
+## 1      1   black
+## 2      2 #DF536B</code></pre>
+</div>
+</div>
+
+
+
+
+</div>
+
+<script>
+
+// add bootstrap table styles to pandoc tables
+function bootstrapStylePandocTables() {
+  $('tr.odd').parent('tbody').parent('table').addClass('table table-condensed');
+}
+$(document).ready(function () {
+  bootstrapStylePandocTables();
+});
+
+
+</script>
+
+<!-- tabsets -->
+
+<script>
+$(document).ready(function () {
+  window.buildTabsets("TOC");
+});
+
+$(document).ready(function () {
+  $('.tabset-dropdown > .nav-tabs > li').click(function () {
+    $(this).parent().toggleClass('nav-tabs-open');
+  });
+});
+</script>
+
+<!-- code folding -->
+
+
+<!-- dynamically load mathjax for compatibility with self-contained -->
+<script>
+  (function () {
+    var script = document.createElement("script");
+    script.type = "text/javascript";
+    script.src  = "https://mathjax.rstudio.com/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML";
+    document.getElementsByTagName("head")[0].appendChild(script);
+  })();
+</script>
+
+</body>
+</html>
diff --git a/inst/doc/survtab_examples.R b/inst/doc/survtab_examples.R
index 454584b..a1753d4 100644
--- a/inst/doc/survtab_examples.R
+++ b/inst/doc/survtab_examples.R
@@ -1,169 +1,169 @@
-## ----pkgs, eval = TRUE, echo = TRUE, message = FALSE--------------------------
-library(popEpi)
-library(Epi)
-
-## -----------------------------------------------------------------------------
-data(sire)
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-
-## pretend some are male
-set.seed(1L)
-x$sex <- rbinom(nrow(x), 1, 0.5)
-
-## observed survival - explicit method
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## observed survival - easy method (assumes lex.Xst in x is the status variable)
-st <- survtab(FUT ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## printing gives the used settings and 
-## estimates at the middle and end of the estimated
-## curves; more information available using summary()
-st
-
-
-## -----------------------------------------------------------------------------
-plot(st, col = c("blue", "red"))
-
-## ----popmort------------------------------------------------------------------
-data(popmort)
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-head(pm)
-
-## ----survtab_e2---------------------------------------------------------------
-st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-## -----------------------------------------------------------------------------
-plot(st.e2, y = "r.e2", col = c("blue", "red"))
-
-## ----survtab_pp---------------------------------------------------------------
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-## -----------------------------------------------------------------------------
-plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
-lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
-
-## ----survtab_adjust-----------------------------------------------------------
-## an age group variable
-x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
-
-## using "internal weights" - see ?ICSS for international weights standards
-w <- table(x$agegr)
-w
-
-w <- list(agegr = as.numeric(w))
-
-## ----survtab_adjust_2---------------------------------------------------------
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-## -----------------------------------------------------------------------------
-plot(st.as, y = "r.e2.as", col = c("blue", "red"))
-
-## ----weights_examples, eval = TRUE--------------------------------------------
-list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
-
-wdf <- merge(0:1, 1:4)
-names(wdf) <- c("sex", "agegr")
-wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
-wdf
-
-## ----survtab_adjust_3---------------------------------------------------------
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
-                 adjust = "agegr",
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-## ----survtab_cause------------------------------------------------------------
-st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                 data = x, 
-                 surv.type = "surv.cause",
-                 breaks = list(FUT = seq(0, 5, 1/12)))
-
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-plot(st.ca, y = "surv.obs.canD", col = "blue")
-lines(st.pp, y = "r.pp", col = "red")
-
-## ----survtab_cif--------------------------------------------------------------
-st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.obs",
-                  breaks = list(FUT = seq(0, 5, 1/12)))
-
-plot(st.cif, y = "CIF_canD", conf.int = FALSE)
-lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
-
-## ----survtab_relcif-----------------------------------------------------------
-st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.rel",
-                  breaks = list(FUT = seq(0, 5, 1/12)),
-                  pophaz = pm)
-plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
-lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
-
-## -----------------------------------------------------------------------------
-sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              aggre = list(sex, fot))
-head(ag)
-
-## ----survtab_ag_example1------------------------------------------------------
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "hazard",
-                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
-
-## ----survtab_ag_example2------------------------------------------------------
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "lifetable",
-                 d = c("from0to1", "from0to2"), n = "at.risk",
-                 n.cens = "from0to0")
-
-## ----survtab_ag_cause---------------------------------------------------------
-st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
-                    surv.method = "hazard",
-                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
-plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
-
-## -----------------------------------------------------------------------------
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              pophaz = popmort, pp = TRUE,
-              aggre = list(sex, fot))
-
-st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
-                    surv.method = "hazard", relsurv.method = "pp",
-                    d = list(from0to1 + from0to2), pyrs = "pyrs",
-                    d.pp = list(from0to1.pp + from0to2.pp),
-                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
-                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
-plot(st.pp, y = "r.pp", col = c("blue", "red"))
-
+## ----pkgs, eval = TRUE, echo = TRUE, message = FALSE--------------------------
+library(popEpi)
+library(Epi)
+
+## -----------------------------------------------------------------------------
+data(sire)
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+
+## pretend some are male
+set.seed(1L)
+x$sex <- rbinom(nrow(x), 1, 0.5)
+
+## observed survival - explicit method
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## observed survival - easy method (assumes lex.Xst in x is the status variable)
+st <- survtab(FUT ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## printing gives the used settings and 
+## estimates at the middle and end of the estimated
+## curves; more information available using summary()
+st
+
+
+## -----------------------------------------------------------------------------
+plot(st, col = c("blue", "red"))
+
+## ----popmort------------------------------------------------------------------
+data(popmort)
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+head(pm)
+
+## ----survtab_e2---------------------------------------------------------------
+st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+## -----------------------------------------------------------------------------
+plot(st.e2, y = "r.e2", col = c("blue", "red"))
+
+## ----survtab_pp---------------------------------------------------------------
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+## -----------------------------------------------------------------------------
+plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
+lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
+
+## ----survtab_adjust-----------------------------------------------------------
+## an age group variable
+x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
+
+## using "internal weights" - see ?ICSS for international weights standards
+w <- table(x$agegr)
+w
+
+w <- list(agegr = as.numeric(w))
+
+## ----survtab_adjust_2---------------------------------------------------------
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+## -----------------------------------------------------------------------------
+plot(st.as, y = "r.e2.as", col = c("blue", "red"))
+
+## ----weights_examples, eval = TRUE--------------------------------------------
+list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
+
+wdf <- merge(0:1, 1:4)
+names(wdf) <- c("sex", "agegr")
+wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
+wdf
+
+## ----survtab_adjust_3---------------------------------------------------------
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
+                 adjust = "agegr",
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+## ----survtab_cause------------------------------------------------------------
+st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                 data = x, 
+                 surv.type = "surv.cause",
+                 breaks = list(FUT = seq(0, 5, 1/12)))
+
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+plot(st.ca, y = "surv.obs.canD", col = "blue")
+lines(st.pp, y = "r.pp", col = "red")
+
+## ----survtab_cif--------------------------------------------------------------
+st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.obs",
+                  breaks = list(FUT = seq(0, 5, 1/12)))
+
+plot(st.cif, y = "CIF_canD", conf.int = FALSE)
+lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
+
+## ----survtab_relcif-----------------------------------------------------------
+st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.rel",
+                  breaks = list(FUT = seq(0, 5, 1/12)),
+                  pophaz = pm)
+plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
+lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
+
+## -----------------------------------------------------------------------------
+sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              aggre = list(sex, fot))
+head(ag)
+
+## ----survtab_ag_example1------------------------------------------------------
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "hazard",
+                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
+
+## ----survtab_ag_example2------------------------------------------------------
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "lifetable",
+                 d = c("from0to1", "from0to2"), n = "at.risk",
+                 n.cens = "from0to0")
+
+## ----survtab_ag_cause---------------------------------------------------------
+st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
+                    surv.method = "hazard",
+                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
+plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
+
+## -----------------------------------------------------------------------------
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              pophaz = popmort, pp = TRUE,
+              aggre = list(sex, fot))
+
+st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
+                    surv.method = "hazard", relsurv.method = "pp",
+                    d = list(from0to1 + from0to2), pyrs = "pyrs",
+                    d.pp = list(from0to1.pp + from0to2.pp),
+                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
+                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
+plot(st.pp, y = "r.pp", col = c("blue", "red"))
+
diff --git a/inst/doc/survtab_examples.Rmd b/inst/doc/survtab_examples.Rmd
index 341f23c..ece43b6 100644
--- a/inst/doc/survtab_examples.Rmd
+++ b/inst/doc/survtab_examples.Rmd
@@ -1,295 +1,295 @@
----
-title: "Examples of using survtab"
-author: "Joonas Miettinen"
-date: "`r Sys.Date()`"
-output: 
-  html_document:
-    toc: true
-    toc_depth: 2
-    fig_width: 6
-    fig_height: 6
-vignette: >
-  %\VignetteEngine{knitr::rmarkdown}
-  %\VignetteIndexEntry{survtab examples}
-  %\usepackage[utf8]{inputenc}
----
-
-# Overview
-  
-This vignette aims to clarify the usage of the `survtab_ag` and `survtab` functions included in this package. `survtab_ag` estimates various survival functions and cumulative incidence functions (CIFs) non-parametrically using aggregated data, and `survtab` is a wrapper for `survtab_ag`, to which `Lexis` data is supplied. 
-  
-Two methods (`surv.method`) are currently supported: The `"lifetable"` (actuarial) method only makes use of counts when estimating any of the supported survival time functions. The default method (`"hazard"`}) estimates appropriate hazards and transforms them into survival function or CIF estimates.
-  
-For relative survival estimation we need also to enumerate the expected hazard levels for the subjects in the data. This is done by merging expected hazards to individuals' subintervals (which divide their survival time lines to a number of small intervals). For Pohar-Perme-weighted analyses one must additionally compute various weighted figures at the level of split subject data. 
-  
-If one has subject-level data, the simplest way of computing survival function estimates with `popEpi` is by defining a `Lexis` object and using `survtab`, which will do the rest. For pre-aggregated data one may use the `survtab_ag` function instead. One can also use the `lexpand` function to split, merge population hazards, and aggregate in a single function call and then use `survtab_ag` if that is convenient.
-  
-# Using `survtab`
-  
-It is straightforward to estimate various survival time functions with `survtab` once a `Lexis` object has been defined (see `?Lexis` in package `Epi` for details):
-  
-```{r pkgs, eval = TRUE, echo = TRUE, message = FALSE}
-library(popEpi)
-library(Epi)
-```
-
-```{r}
-data(sire)
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-
-## pretend some are male
-set.seed(1L)
-x$sex <- rbinom(nrow(x), 1, 0.5)
-
-## observed survival - explicit method
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## observed survival - easy method (assumes lex.Xst in x is the status variable)
-st <- survtab(FUT ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## printing gives the used settings and 
-## estimates at the middle and end of the estimated
-## curves; more information available using summary()
-st
-
-```
-
-Plotting by strata (men = blue, women = red):
-
-```{r}
-plot(st, col = c("blue", "red"))
-```
-
-Note that the correct usage of the `formula` argument in `survtab` specifies the time scale in the `Lexis` object over which survival is computed (here `"FUT"` for follow-up time). This is used to identify the appropriate time scale in the data. When only supplying the survival time scale as the right-hand-side of the formula, the column `lex.Xst` in the supplied `Lexis` object is assumed to be the (correctly formatted!) status variable. When using `Surv()` to be explicit, we effectively (and exceptionally) pass the starting times to the `time` argument in `Surv()`, and `time2` is ignored entirely. The function will fail if `time` does not match exactly with a time scale in data.
-
-When using `Surv()`, one must also pass the status variable, which can be something other than the `lex.Xst` variable created by `Lexis()`, though usually ``lex.Xst` is what you want to use (especially if the data has already been split using e.g. `splitLexis` or `splitMulti`, which is allowed). It is recommended to use a factor status variable to pass to `Surv()`, though a numeric variable will work in simple cases (0 = alive, 1 = dead; also `FALSE`  = alive, `TRUE` = dead). Using `Surv()` also allows easy passing of transformations of `lex.Xst`, e.g. `Surv(FUT, lex.Xst %in% 1:2)`.
-
-The argument `breaks` must be a named list of breaks by which to split the `Lexis` data (see `?splitMulti`). It is mandatory to assign breaks at least to the survival time scale (`"FUT"` in our example) so that `survtab` knows what intervals to use to estimate the requested survival time function(s). The breaks also determine the window used: It is therefore easy to compute so called period estimates by defining the roof and floor along the calendar time scale, e.g. 
-
-`breaks = list(FUT = seq(0, 5, 1/12), CAL = c(2000, 2005))`
-
-would cause `survtab` to compute period estimates for 2000-2004 (breaks given here as fractional years, so 2005 is effectively 2004.99999...).
-
-## Relative/net survival
-
-Relative/net survival estimation requires knowledge of the expected hazard levels for the individuals in the data. In `survtab` this is accomplished by passing a long-format `data.frame` of population hazards via the `pophaz` argument. E.g. the `popmort` dataset included in `popEpi` (Finnish overall mortality rates for men and women).
-
-```{r popmort}
-data(popmort)
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-head(pm)
-```
-
-The `data.frame` should contain a variable named `"haz"` indicating the population hazard at the level of one subject-year. Any other variables are considered to be variables, by which to merge population hazards to the (split) subject-level data within `survtab`. These merging variables may correspond to the time scales in the used `Lexis` object. This allows for e.g. merging in different population hazards for the same subject as they get older.
-
-The following causes `survtab` to estimate EdererII relative survival:
-
-```{r survtab_e2}
-st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-```{r}
-plot(st.e2, y = "r.e2", col = c("blue", "red"))
-```
-
-Note that the curves diverge due to merging in the "wrong" population hazards for some individuals which we randomized earlier to be male though all the individuals in data are actually female. Pohar-Perme-weighted estimates can be computed by
-
-```{r survtab_pp}
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-Compare with EdererII estimates:
-
-```{r}
-plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
-lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
-```
-
-## Adjusting estimates
-
-`survtab` also allows for adjusting the survival curves by categorical variables --- typically by age groups. The following demonstrates how:
-
-```{r survtab_adjust}
-## an age group variable
-x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
-
-## using "internal weights" - see ?ICSS for international weights standards
-w <- table(x$agegr)
-w
-
-w <- list(agegr = as.numeric(w))
-```
-
-```{r survtab_adjust_2}
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-```{r}
-plot(st.as, y = "r.e2.as", col = c("blue", "red"))
-```
-
-We now have age-adjusted EdererII relative/net survival estimates. The `weights` argument allows for either a list of weights (with one or multiple variables to adjust by) or a `data.frame` of weights. Examples:
-
-```{r weights_examples, eval = TRUE}
-list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
-
-wdf <- merge(0:1, 1:4)
-names(wdf) <- c("sex", "agegr")
-wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
-wdf
-```
-
-The weights do not have to sum to one when supplied as they are internally forced to do so within each stratum. In the `data.frame` of weights, the column of actual weights to use must be named "weights". When there are more than one variable to adjust by, and a list of weights has been supplied, the variable-specific weights are first multiplied together (cumulatively) and then scaled to sum to one.
-
-This adjusting can be done to any survival time function that `survtab` (and `survtab_ag`) estimates. One can also supply adjusting variables via the `adjust` argument if convenient:
-
-```{r survtab_adjust_3}
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
-                 adjust = "agegr",
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-Where `adjust` could also be `adjust = agegr`, `adjust = list(agegr)` or 
-
-`adjust = list(agegr = cut(dg_age, c(0, 60, 70, 80, Inf), right = FALSE))`
-
-for exactly the same results. When adjusting by multiple variables, one must supply a vector of variable names in data or a list of multiple elements (as in the base function `aggregate`).
-
-## Other survival time functions
-
-One can also estimate cause-specific survival functions, cumulative incidence functions (CIFs, a.k.a. crude risk a.k.a. absolute risk functions), and CIFs based on the excess numbers of events. Cause-specific survival is close to net survival as they are philosophically highly similar concepts: 
-
-```{r survtab_cause}
-st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                 data = x, 
-                 surv.type = "surv.cause",
-                 breaks = list(FUT = seq(0, 5, 1/12)))
-
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-plot(st.ca, y = "surv.obs.canD", col = "blue")
-lines(st.pp, y = "r.pp", col = "red")
-```
-
-Absolute risk:
-
-```{r survtab_cif}
-st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.obs",
-                  breaks = list(FUT = seq(0, 5, 1/12)))
-
-plot(st.cif, y = "CIF_canD", conf.int = FALSE)
-lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
-```
-
-The "relative CIF" attempts to be close to the true CIF without using knowledge about the types of events, e.g. causes of death:
-
-```{r survtab_relcif}
-st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.rel",
-                  breaks = list(FUT = seq(0, 5, 1/12)),
-                  pophaz = pm)
-plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
-lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
-```
-
-
-# Using `survtab_ag`
-
-Arguments concerning the types and methods of estimating of survival time functions work the same in `survtab_ag` as in `survtab` (the latter uses the former). However, with aggregated data one must explicitly supply the various count and person-time variables. Also, usage of the `formula` argument is different.
-
-For demonstration purposes we form an aggregated data set using `lexpand`; see `?lexpand` for more information on that function.
-
-```{r}
-sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              aggre = list(sex, fot))
-head(ag)
-```
-
-Now simply do:
-
-```{r survtab_ag_example1}
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "hazard",
-                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
-```
-
-Or:
-
-```{r survtab_ag_example2}
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "lifetable",
-                 d = c("from0to1", "from0to2"), n = "at.risk",
-                 n.cens = "from0to0")
-```
-
-Note that e.g. argument `d` could also have been supplied as 
-
-`list(from0to1, from0to2)`
-
-or
-
-`list(canD = from0to1, othD = from0to2)`
-
-for identical results. The last is convenient for e.g. `surv.cause` computations:
-
-```{r survtab_ag_cause}
-st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
-                    surv.method = "hazard",
-                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
-plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
-```
-
-One has to supply the most variables when computing Pohar-Perme estimates (though it is probably rare to have third-source aggregated data with Pohar-Perme weighted figures, it is implemented here to be used as a workhorse for `survtab`). For this we must aggregate again to get the Pohar-Perme weighted counts and subject-times:
-
-```{r}
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              pophaz = popmort, pp = TRUE,
-              aggre = list(sex, fot))
-
-st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
-                    surv.method = "hazard", relsurv.method = "pp",
-                    d = list(from0to1 + from0to2), pyrs = "pyrs",
-                    d.pp = list(from0to1.pp + from0to2.pp),
-                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
-                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
-plot(st.pp, y = "r.pp", col = c("blue", "red"))
-```
-
-Here it is best to supply only one column to each argument since Pohar-Perme estimates will not be computed for several types of events at the same time.
-
-
-
+---
+title: "Examples of using survtab"
+author: "Joonas Miettinen"
+date: "`r Sys.Date()`"
+output: 
+  html_document:
+    toc: true
+    toc_depth: 2
+    fig_width: 6
+    fig_height: 6
+vignette: >
+  %\VignetteEngine{knitr::rmarkdown}
+  %\VignetteIndexEntry{survtab examples}
+  %\usepackage[utf8]{inputenc}
+---
+
+# Overview
+  
+This vignette aims to clarify the usage of the `survtab_ag` and `survtab` functions included in this package. `survtab_ag` estimates various survival functions and cumulative incidence functions (CIFs) non-parametrically using aggregated data, and `survtab` is a wrapper for `survtab_ag`, to which `Lexis` data is supplied. 
+  
+Two methods (`surv.method`) are currently supported: The `"lifetable"` (actuarial) method only makes use of counts when estimating any of the supported survival time functions. The default method (`"hazard"`}) estimates appropriate hazards and transforms them into survival function or CIF estimates.
+  
+For relative survival estimation we need also to enumerate the expected hazard levels for the subjects in the data. This is done by merging expected hazards to individuals' subintervals (which divide their survival time lines to a number of small intervals). For Pohar-Perme-weighted analyses one must additionally compute various weighted figures at the level of split subject data. 
+  
+If one has subject-level data, the simplest way of computing survival function estimates with `popEpi` is by defining a `Lexis` object and using `survtab`, which will do the rest. For pre-aggregated data one may use the `survtab_ag` function instead. One can also use the `lexpand` function to split, merge population hazards, and aggregate in a single function call and then use `survtab_ag` if that is convenient.
+  
+# Using `survtab`
+  
+It is straightforward to estimate various survival time functions with `survtab` once a `Lexis` object has been defined (see `?Lexis` in package `Epi` for details):
+  
+```{r pkgs, eval = TRUE, echo = TRUE, message = FALSE}
+library(popEpi)
+library(Epi)
+```
+
+```{r}
+data(sire)
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+
+## pretend some are male
+set.seed(1L)
+x$sex <- rbinom(nrow(x), 1, 0.5)
+
+## observed survival - explicit method
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## observed survival - easy method (assumes lex.Xst in x is the status variable)
+st <- survtab(FUT ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## printing gives the used settings and 
+## estimates at the middle and end of the estimated
+## curves; more information available using summary()
+st
+
+```
+
+Plotting by strata (men = blue, women = red):
+
+```{r}
+plot(st, col = c("blue", "red"))
+```
+
+Note that the correct usage of the `formula` argument in `survtab` specifies the time scale in the `Lexis` object over which survival is computed (here `"FUT"` for follow-up time). This is used to identify the appropriate time scale in the data. When only supplying the survival time scale as the right-hand-side of the formula, the column `lex.Xst` in the supplied `Lexis` object is assumed to be the (correctly formatted!) status variable. When using `Surv()` to be explicit, we effectively (and exceptionally) pass the starting times to the `time` argument in `Surv()`, and `time2` is ignored entirely. The function will fail if `time` does not match exactly with a time scale in data.
+
+When using `Surv()`, one must also pass the status variable, which can be something other than the `lex.Xst` variable created by `Lexis()`, though usually ``lex.Xst` is what you want to use (especially if the data has already been split using e.g. `splitLexis` or `splitMulti`, which is allowed). It is recommended to use a factor status variable to pass to `Surv()`, though a numeric variable will work in simple cases (0 = alive, 1 = dead; also `FALSE`  = alive, `TRUE` = dead). Using `Surv()` also allows easy passing of transformations of `lex.Xst`, e.g. `Surv(FUT, lex.Xst %in% 1:2)`.
+
+The argument `breaks` must be a named list of breaks by which to split the `Lexis` data (see `?splitMulti`). It is mandatory to assign breaks at least to the survival time scale (`"FUT"` in our example) so that `survtab` knows what intervals to use to estimate the requested survival time function(s). The breaks also determine the window used: It is therefore easy to compute so called period estimates by defining the roof and floor along the calendar time scale, e.g. 
+
+`breaks = list(FUT = seq(0, 5, 1/12), CAL = c(2000, 2005))`
+
+would cause `survtab` to compute period estimates for 2000-2004 (breaks given here as fractional years, so 2005 is effectively 2004.99999...).
+
+## Relative/net survival
+
+Relative/net survival estimation requires knowledge of the expected hazard levels for the individuals in the data. In `survtab` this is accomplished by passing a long-format `data.frame` of population hazards via the `pophaz` argument. E.g. the `popmort` dataset included in `popEpi` (Finnish overall mortality rates for men and women).
+
+```{r popmort}
+data(popmort)
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+head(pm)
+```
+
+The `data.frame` should contain a variable named `"haz"` indicating the population hazard at the level of one subject-year. Any other variables are considered to be variables, by which to merge population hazards to the (split) subject-level data within `survtab`. These merging variables may correspond to the time scales in the used `Lexis` object. This allows for e.g. merging in different population hazards for the same subject as they get older.
+
+The following causes `survtab` to estimate EdererII relative survival:
+
+```{r survtab_e2}
+st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+```{r}
+plot(st.e2, y = "r.e2", col = c("blue", "red"))
+```
+
+Note that the curves diverge due to merging in the "wrong" population hazards for some individuals which we randomized earlier to be male though all the individuals in data are actually female. Pohar-Perme-weighted estimates can be computed by
+
+```{r survtab_pp}
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+Compare with EdererII estimates:
+
+```{r}
+plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
+lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
+```
+
+## Adjusting estimates
+
+`survtab` also allows for adjusting the survival curves by categorical variables --- typically by age groups. The following demonstrates how:
+
+```{r survtab_adjust}
+## an age group variable
+x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
+
+## using "internal weights" - see ?ICSS for international weights standards
+w <- table(x$agegr)
+w
+
+w <- list(agegr = as.numeric(w))
+```
+
+```{r survtab_adjust_2}
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+```{r}
+plot(st.as, y = "r.e2.as", col = c("blue", "red"))
+```
+
+We now have age-adjusted EdererII relative/net survival estimates. The `weights` argument allows for either a list of weights (with one or multiple variables to adjust by) or a `data.frame` of weights. Examples:
+
+```{r weights_examples, eval = TRUE}
+list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
+
+wdf <- merge(0:1, 1:4)
+names(wdf) <- c("sex", "agegr")
+wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
+wdf
+```
+
+The weights do not have to sum to one when supplied as they are internally forced to do so within each stratum. In the `data.frame` of weights, the column of actual weights to use must be named "weights". When there are more than one variable to adjust by, and a list of weights has been supplied, the variable-specific weights are first multiplied together (cumulatively) and then scaled to sum to one.
+
+This adjusting can be done to any survival time function that `survtab` (and `survtab_ag`) estimates. One can also supply adjusting variables via the `adjust` argument if convenient:
+
+```{r survtab_adjust_3}
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
+                 adjust = "agegr",
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+Where `adjust` could also be `adjust = agegr`, `adjust = list(agegr)` or 
+
+`adjust = list(agegr = cut(dg_age, c(0, 60, 70, 80, Inf), right = FALSE))`
+
+for exactly the same results. When adjusting by multiple variables, one must supply a vector of variable names in data or a list of multiple elements (as in the base function `aggregate`).
+
+## Other survival time functions
+
+One can also estimate cause-specific survival functions, cumulative incidence functions (CIFs, a.k.a. crude risk a.k.a. absolute risk functions), and CIFs based on the excess numbers of events. Cause-specific survival is close to net survival as they are philosophically highly similar concepts: 
+
+```{r survtab_cause}
+st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                 data = x, 
+                 surv.type = "surv.cause",
+                 breaks = list(FUT = seq(0, 5, 1/12)))
+
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+plot(st.ca, y = "surv.obs.canD", col = "blue")
+lines(st.pp, y = "r.pp", col = "red")
+```
+
+Absolute risk:
+
+```{r survtab_cif}
+st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.obs",
+                  breaks = list(FUT = seq(0, 5, 1/12)))
+
+plot(st.cif, y = "CIF_canD", conf.int = FALSE)
+lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
+```
+
+The "relative CIF" attempts to be close to the true CIF without using knowledge about the types of events, e.g. causes of death:
+
+```{r survtab_relcif}
+st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.rel",
+                  breaks = list(FUT = seq(0, 5, 1/12)),
+                  pophaz = pm)
+plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
+lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
+```
+
+
+# Using `survtab_ag`
+
+Arguments concerning the types and methods of estimating of survival time functions work the same in `survtab_ag` as in `survtab` (the latter uses the former). However, with aggregated data one must explicitly supply the various count and person-time variables. Also, usage of the `formula` argument is different.
+
+For demonstration purposes we form an aggregated data set using `lexpand`; see `?lexpand` for more information on that function.
+
+```{r}
+sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              aggre = list(sex, fot))
+head(ag)
+```
+
+Now simply do:
+
+```{r survtab_ag_example1}
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "hazard",
+                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
+```
+
+Or:
+
+```{r survtab_ag_example2}
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "lifetable",
+                 d = c("from0to1", "from0to2"), n = "at.risk",
+                 n.cens = "from0to0")
+```
+
+Note that e.g. argument `d` could also have been supplied as 
+
+`list(from0to1, from0to2)`
+
+or
+
+`list(canD = from0to1, othD = from0to2)`
+
+for identical results. The last is convenient for e.g. `surv.cause` computations:
+
+```{r survtab_ag_cause}
+st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
+                    surv.method = "hazard",
+                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
+plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
+```
+
+One has to supply the most variables when computing Pohar-Perme estimates (though it is probably rare to have third-source aggregated data with Pohar-Perme weighted figures, it is implemented here to be used as a workhorse for `survtab`). For this we must aggregate again to get the Pohar-Perme weighted counts and subject-times:
+
+```{r}
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              pophaz = popmort, pp = TRUE,
+              aggre = list(sex, fot))
+
+st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
+                    surv.method = "hazard", relsurv.method = "pp",
+                    d = list(from0to1 + from0to2), pyrs = "pyrs",
+                    d.pp = list(from0to1.pp + from0to2.pp),
+                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
+                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
+plot(st.pp, y = "r.pp", col = c("blue", "red"))
+```
+
+Here it is best to supply only one column to each argument since Pohar-Perme estimates will not be computed for several types of events at the same time.
+
+
+
diff --git a/inst/doc/survtab_examples.html b/inst/doc/survtab_examples.html
new file mode 100644
index 0000000..c58cff1
--- /dev/null
+++ b/inst/doc/survtab_examples.html
@@ -0,0 +1,815 @@
+<!DOCTYPE html>
+
+<html>
+
+<head>
+
+<meta charset="utf-8" />
+<meta name="generator" content="pandoc" />
+<meta http-equiv="X-UA-Compatible" content="IE=EDGE" />
+
+
+<meta name="author" content="Joonas Miettinen" />
+
+<meta name="date" content="2023-01-19" />
+
+<title>Examples of using survtab</title>
+
+<script>// Pandoc 2.9 adds attributes on both header and div. We remove the former (to
+// be compatible with the behavior of Pandoc < 2.8).
+document.addEventListener('DOMContentLoaded', function(e) {
+  var hs = document.querySelectorAll("div.section[class*='level'] > :first-child");
+  var i, h, a;
+  for (i = 0; i < hs.length; i++) {
+    h = hs[i];
+    if (!/^h[1-6]$/i.test(h.tagName)) continue;  // it should be a header h1-h6
+    a = h.attributes;
+    while (a.length > 0) h.removeAttribute(a[0].name);
+  }
+});
+</script>
+<script>/*! jQuery v3.6.1 | (c) OpenJS Foundation and other contributors | jquery.org/license */
+!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(e.document)return t(e);throw new Error("jQuery requires a window with a document")}:t(e)}("undefined"!=typeof window?window:this,function(w,R){"use strict";function v(e){return"function"==typeof e&&"number"!=typeof e.nodeType&&"function"!=typeof e.item}function g(e){return null!=e&&e===e.window}var t=[],M=Object.getPrototypeOf,s=t.slice,I=t.flat?function(e){return t.flat.call(e)}:function(e){return t.concat.apply([],e)},W=t.push,F=t.indexOf,$={},B=$.toString,_=$.hasOwnProperty,z=_.toString,U=z.call(Object),y={},T=w.document,X={type:!0,src:!0,nonce:!0,noModule:!0};function V(e,t,n){var r,i,o=(n=n||T).createElement("script");if(o.text=e,t)for(r in X)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function h(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?$[B.call(e)]||"object":typeof e}var e="3.6.1",C=function(e,t){return new C.fn.init(e,t)};function G(e){var t=!!e&&"length"in e&&e.length,n=h(e);return!v(e)&&!g(e)&&("array"===n||0===t||"number"==typeof t&&0<t&&t-1 in e)}C.fn=C.prototype={jquery:e,constructor:C,length:0,toArray:function(){return s.call(this)},get:function(e){return null==e?s.call(this):e<0?this[e+this.length]:this[e]},pushStack:function(e){e=C.merge(this.constructor(),e);return e.prevObject=this,e},each:function(e){return C.each(this,e)},map:function(n){return this.pushStack(C.map(this,function(e,t){return n.call(e,t,e)}))},slice:function(){return this.pushStack(s.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},even:function(){return this.pushStack(C.grep(this,function(e,t){return(t+1)%2}))},odd:function(){return this.pushStack(C.grep(this,function(e,t){return t%2}))},eq:function(e){var t=this.length,e=+e+(e<0?t:0);return this.pushStack(0<=e&&e<t?[this[e]]:[])},end:function(){return this.prevObject||this.constructor()},push:W,sort:t.sort,splice:t.splice},C.extend=C.fn.extend=function(){var e,t,n,r,i,o=arguments[0]||{},a=1,s=arguments.length,u=!1;for("boolean"==typeof o&&(u=o,o=arguments[a]||{},a++),"object"==typeof o||v(o)||(o={}),a===s&&(o=this,a--);a<s;a++)if(null!=(e=arguments[a]))for(t in e)n=e[t],"__proto__"!==t&&o!==n&&(u&&n&&(C.isPlainObject(n)||(r=Array.isArray(n)))?(i=o[t],i=r&&!Array.isArray(i)?[]:r||C.isPlainObject(i)?i:{},r=!1,o[t]=C.extend(u,i,n)):void 0!==n&&(o[t]=n));return o},C.extend({expando:"jQuery"+(e+Math.random()).replace(/\D/g,""),isReady:!0,error:function(e){throw new Error(e)},noop:function(){},isPlainObject:function(e){return!(!e||"[object Object]"!==B.call(e))&&(!(e=M(e))||"function"==typeof(e=_.call(e,"constructor")&&e.constructor)&&z.call(e)===U)},isEmptyObject:function(e){for(var t in e)return!1;return!0},globalEval:function(e,t,n){V(e,{nonce:t&&t.nonce},n)},each:function(e,t){var n,r=0;if(G(e)){for(n=e.length;r<n;r++)if(!1===t.call(e[r],r,e[r]))break}else for(r in e)if(!1===t.call(e[r],r,e[r]))break;return e},makeArray:function(e,t){t=t||[];return null!=e&&(G(Object(e))?C.merge(t,"string"==typeof e?[e]:e):W.call(t,e)),t},inArray:function(e,t,n){return null==t?-1:F.call(t,e,n)},merge:function(e,t){for(var n=+t.length,r=0,i=e.length;r<n;r++)e[i++]=t[r];return e.length=i,e},grep:function(e,t,n){for(var r=[],i=0,o=e.length,a=!n;i<o;i++)!t(e[i],i)!=a&&r.push(e[i]);return r},map:function(e,t,n){var r,i,o=0,a=[];if(G(e))for(r=e.length;o<r;o++)null!=(i=t(e[o],o,n))&&a.push(i);else for(o in e)null!=(i=t(e[o],o,n))&&a.push(i);return I(a)},guid:1,support:y}),"function"==typeof Symbol&&(C.fn[Symbol.iterator]=t[Symbol.iterator]),C.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(e,t){$["[object "+t+"]"]=t.toLowerCase()});function r(e,t,n){var r=[],i=void 0!==n;while((e=e[t])&&9!==e.nodeType)if(1===e.nodeType){if(i&&C(e).is(n))break;r.push(e)}return r}function Y(e,t){for(var n=[];e;e=e.nextSibling)1===e.nodeType&&e!==t&&n.push(e);return n}var e=function(R){function c(e,t){return e="0x"+e.slice(1)-65536,t||(e<0?String.fromCharCode(65536+e):String.fromCharCode(e>>10|55296,1023&e|56320))}function M(e,t){return t?"\0"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e}function I(){T()}var e,p,b,o,W,d,F,$,w,u,l,T,C,n,E,h,r,i,g,S="sizzle"+ +new Date,f=R.document,k=0,B=0,_=q(),z=q(),U=q(),y=q(),X=function(e,t){return e===t&&(l=!0),0},V={}.hasOwnProperty,t=[],G=t.pop,Y=t.push,A=t.push,Q=t.slice,v=function(e,t){for(var n=0,r=e.length;n<r;n++)if(e[n]===t)return n;return-1},J="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",a="[\\x20\\t\\r\\n\\f]",s="(?:\\\\[\\da-fA-F]{1,6}"+a+"?|\\\\[^\\r\\n\\f]|[\\w-]|[^\0-\\x7f])+",K="\\["+a+"*("+s+")(?:"+a+"*([*^$|!~]?=)"+a+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+s+"))|)"+a+"*\\]",Z=":("+s+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+K+")*)|.*)\\)|)",ee=new RegExp(a+"+","g"),m=new RegExp("^"+a+"+|((?:^|[^\\\\])(?:\\\\.)*)"+a+"+$","g"),te=new RegExp("^"+a+"*,"+a+"*"),ne=new RegExp("^"+a+"*([>+~]|"+a+")"+a+"*"),re=new RegExp(a+"|>"),ie=new RegExp(Z),oe=new RegExp("^"+s+"$"),x={ID:new RegExp("^#("+s+")"),CLASS:new RegExp("^\\.("+s+")"),TAG:new RegExp("^("+s+"|[*])"),ATTR:new RegExp("^"+K),PSEUDO:new RegExp("^"+Z),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+a+"*(even|odd|(([+-]|)(\\d*)n|)"+a+"*(?:([+-]|)"+a+"*(\\d+)|))"+a+"*\\)|)","i"),bool:new RegExp("^(?:"+J+")$","i"),needsContext:new RegExp("^"+a+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+a+"*((?:-\\d)?\\d*)"+a+"*\\)|)(?=[^-]|$)","i")},ae=/HTML$/i,se=/^(?:input|select|textarea|button)$/i,ue=/^h\d$/i,N=/^[^{]+\{\s*\[native \w/,le=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ce=/[+~]/,j=new RegExp("\\\\[\\da-fA-F]{1,6}"+a+"?|\\\\([^\\r\\n\\f])","g"),fe=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,pe=ve(function(e){return!0===e.disabled&&"fieldset"===e.nodeName.toLowerCase()},{dir:"parentNode",next:"legend"});try{A.apply(t=Q.call(f.childNodes),f.childNodes),t[f.childNodes.length].nodeType}catch(e){A={apply:t.length?function(e,t){Y.apply(e,Q.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function D(t,e,n,r){var i,o,a,s,u,l,c=e&&e.ownerDocument,f=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==f&&9!==f&&11!==f)return n;if(!r&&(T(e),e=e||C,E)){if(11!==f&&(s=le.exec(t)))if(i=s[1]){if(9===f){if(!(l=e.getElementById(i)))return n;if(l.id===i)return n.push(l),n}else if(c&&(l=c.getElementById(i))&&g(e,l)&&l.id===i)return n.push(l),n}else{if(s[2])return A.apply(n,e.getElementsByTagName(t)),n;if((i=s[3])&&p.getElementsByClassName&&e.getElementsByClassName)return A.apply(n,e.getElementsByClassName(i)),n}if(p.qsa&&!y[t+" "]&&(!h||!h.test(t))&&(1!==f||"object"!==e.nodeName.toLowerCase())){if(l=t,c=e,1===f&&(re.test(t)||ne.test(t))){(c=ce.test(t)&&ye(e.parentNode)||e)===e&&p.scope||((a=e.getAttribute("id"))?a=a.replace(fe,M):e.setAttribute("id",a=S)),o=(u=d(t)).length;while(o--)u[o]=(a?"#"+a:":scope")+" "+P(u[o]);l=u.join(",")}try{return A.apply(n,c.querySelectorAll(l)),n}catch(e){y(t,!0)}finally{a===S&&e.removeAttribute("id")}}}return $(t.replace(m,"$1"),e,n,r)}function q(){var n=[];function r(e,t){return n.push(e+" ")>b.cacheLength&&delete r[n.shift()],r[e+" "]=t}return r}function L(e){return e[S]=!0,e}function H(e){var t=C.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t)}}function de(e,t){var n=e.split("|"),r=n.length;while(r--)b.attrHandle[n[r]]=t}function he(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function ge(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&pe(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function O(a){return L(function(o){return o=+o,L(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function ye(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}for(e in p=D.support={},W=D.isXML=function(e){var t=e&&e.namespaceURI,e=e&&(e.ownerDocument||e).documentElement;return!ae.test(t||e&&e.nodeName||"HTML")},T=D.setDocument=function(e){var e=e?e.ownerDocument||e:f;return e!=C&&9===e.nodeType&&e.documentElement&&(n=(C=e).documentElement,E=!W(C),f!=C&&(e=C.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",I,!1):e.attachEvent&&e.attachEvent("onunload",I)),p.scope=H(function(e){return n.appendChild(e).appendChild(C.createElement("div")),"undefined"!=typeof e.querySelectorAll&&!e.querySelectorAll(":scope fieldset div").length}),p.attributes=H(function(e){return e.className="i",!e.getAttribute("className")}),p.getElementsByTagName=H(function(e){return e.appendChild(C.createComment("")),!e.getElementsByTagName("*").length}),p.getElementsByClassName=N.test(C.getElementsByClassName),p.getById=H(function(e){return n.appendChild(e).id=S,!C.getElementsByName||!C.getElementsByName(S).length}),p.getById?(b.filter.ID=function(e){var t=e.replace(j,c);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E)return(t=t.getElementById(e))?[t]:[]}):(b.filter.ID=function(e){var t=e.replace(j,c);return function(e){e="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return e&&e.value===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=p.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):p.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"!==e)return o;while(n=o[i++])1===n.nodeType&&r.push(n);return r},b.find.CLASS=p.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&E)return t.getElementsByClassName(e)},r=[],h=[],(p.qsa=N.test(C.querySelectorAll))&&(H(function(e){var t;n.appendChild(e).innerHTML="<a id='"+S+"'></a><select id='"+S+"-\r\\' msallowcapture=''><option selected=''></option></select>",e.querySelectorAll("[msallowcapture^='']").length&&h.push("[*^$]="+a+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||h.push("\\["+a+"*(?:value|"+J+")"),e.querySelectorAll("[id~="+S+"-]").length||h.push("~="),(t=C.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||h.push("\\["+a+"*name"+a+"*="+a+"*(?:''|\"\")"),e.querySelectorAll(":checked").length||h.push(":checked"),e.querySelectorAll("a#"+S+"+*").length||h.push(".#.+[+~]"),e.querySelectorAll("\\\f"),h.push("[\\r\\n\\f]")}),H(function(e){e.innerHTML="<a href='' disabled='disabled'></a><select disabled='disabled'><option/></select>";var t=C.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&h.push("name"+a+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&h.push(":enabled",":disabled"),n.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&h.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),h.push(",.*:")})),(p.matchesSelector=N.test(i=n.matches||n.webkitMatchesSelector||n.mozMatchesSelector||n.oMatchesSelector||n.msMatchesSelector))&&H(function(e){p.disconnectedMatch=i.call(e,"*"),i.call(e,"[s!='']:x"),r.push("!=",Z)}),h=h.length&&new RegExp(h.join("|")),r=r.length&&new RegExp(r.join("|")),e=N.test(n.compareDocumentPosition),g=e||N.test(n.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,t=t&&t.parentNode;return e===t||!(!t||1!==t.nodeType||!(n.contains?n.contains(t):e.compareDocumentPosition&&16&e.compareDocumentPosition(t)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},X=e?function(e,t){var n;return e===t?(l=!0,0):(n=!e.compareDocumentPosition-!t.compareDocumentPosition)||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!p.sortDetached&&t.compareDocumentPosition(e)===n?e==C||e.ownerDocument==f&&g(f,e)?-1:t==C||t.ownerDocument==f&&g(f,t)?1:u?v(u,e)-v(u,t):0:4&n?-1:1)}:function(e,t){if(e===t)return l=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e==C?-1:t==C?1:i?-1:o?1:u?v(u,e)-v(u,t):0;if(i===o)return he(e,t);n=e;while(n=n.parentNode)a.unshift(n);n=t;while(n=n.parentNode)s.unshift(n);while(a[r]===s[r])r++;return r?he(a[r],s[r]):a[r]==f?-1:s[r]==f?1:0}),C},D.matches=function(e,t){return D(e,null,null,t)},D.matchesSelector=function(e,t){if(T(e),p.matchesSelector&&E&&!y[t+" "]&&(!r||!r.test(t))&&(!h||!h.test(t)))try{var n=i.call(e,t);if(n||p.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){y(t,!0)}return 0<D(t,C,null,[e]).length},D.contains=function(e,t){return(e.ownerDocument||e)!=C&&T(e),g(e,t)},D.attr=function(e,t){(e.ownerDocument||e)!=C&&T(e);var n=b.attrHandle[t.toLowerCase()],n=n&&V.call(b.attrHandle,t.toLowerCase())?n(e,t,!E):void 0;return void 0!==n?n:p.attributes||!E?e.getAttribute(t):(n=e.getAttributeNode(t))&&n.specified?n.value:null},D.escape=function(e){return(e+"").replace(fe,M)},D.error=function(e){throw new Error("Syntax error, unrecognized expression: "+e)},D.uniqueSort=function(e){var t,n=[],r=0,i=0;if(l=!p.detectDuplicates,u=!p.sortStable&&e.slice(0),e.sort(X),l){while(t=e[i++])t===e[i]&&(r=n.push(i));while(r--)e.splice(n[r],1)}return u=null,e},o=D.getText=function(e){var t,n="",r=0,i=e.nodeType;if(i){if(1===i||9===i||11===i){if("string"==typeof e.textContent)return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=o(e)}else if(3===i||4===i)return e.nodeValue}else while(t=e[r++])n+=o(t);return n},(b=D.selectors={cacheLength:50,createPseudo:L,match:x,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(j,c),e[3]=(e[3]||e[4]||e[5]||"").replace(j,c),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||D.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&D.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return x.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&ie.test(n)&&(t=d(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(j,c).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=_[e+" "];return t||(t=new RegExp("(^|"+a+")"+e+"("+a+"|$)"))&&_(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(t,n,r){return function(e){e=D.attr(e,t);return null==e?"!="===n:!n||(e+="","="===n?e===r:"!="===n?e!==r:"^="===n?r&&0===e.indexOf(r):"*="===n?r&&-1<e.indexOf(r):"$="===n?r&&e.slice(-r.length)===r:"~="===n?-1<(" "+e.replace(ee," ")+" ").indexOf(r):"|="===n&&(e===r||e.slice(0,r.length+1)===r+"-"))}},CHILD:function(h,e,t,g,y){var m="nth"!==h.slice(0,3),v="last"!==h.slice(-4),x="of-type"===e;return 1===g&&0===y?function(e){return!!e.parentNode}:function(e,t,n){var r,i,o,a,s,u,l=m!=v?"nextSibling":"previousSibling",c=e.parentNode,f=x&&e.nodeName.toLowerCase(),p=!n&&!x,d=!1;if(c){if(m){while(l){a=e;while(a=a[l])if(x?a.nodeName.toLowerCase()===f:1===a.nodeType)return!1;u=l="only"===h&&!u&&"nextSibling"}return!0}if(u=[v?c.firstChild:c.lastChild],v&&p){d=(s=(r=(i=(o=(a=c)[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]||[])[0]===k&&r[1])&&r[2],a=s&&c.childNodes[s];while(a=++s&&a&&a[l]||(d=s=0)||u.pop())if(1===a.nodeType&&++d&&a===e){i[h]=[k,s,d];break}}else if(!1===(d=p?s=(r=(i=(o=(a=e)[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]||[])[0]===k&&r[1]:d))while(a=++s&&a&&a[l]||(d=s=0)||u.pop())if((x?a.nodeName.toLowerCase()===f:1===a.nodeType)&&++d&&(p&&((i=(o=a[S]||(a[S]={}))[a.uniqueID]||(o[a.uniqueID]={}))[h]=[k,d]),a===e))break;return(d-=y)===g||d%g==0&&0<=d/g}}},PSEUDO:function(e,o){var t,a=b.pseudos[e]||b.setFilters[e.toLowerCase()]||D.error("unsupported pseudo: "+e);return a[S]?a(o):1<a.length?(t=[e,e,"",o],b.setFilters.hasOwnProperty(e.toLowerCase())?L(function(e,t){var n,r=a(e,o),i=r.length;while(i--)e[n=v(e,r[i])]=!(t[n]=r[i])}):function(e){return a(e,0,t)}):a}},pseudos:{not:L(function(e){var r=[],i=[],s=F(e.replace(m,"$1"));return s[S]?L(function(e,t,n,r){var i,o=s(e,null,r,[]),a=e.length;while(a--)(i=o[a])&&(e[a]=!(t[a]=i))}):function(e,t,n){return r[0]=e,s(r,null,n,i),r[0]=null,!i.pop()}}),has:L(function(t){return function(e){return 0<D(t,e).length}}),contains:L(function(t){return t=t.replace(j,c),function(e){return-1<(e.textContent||o(e)).indexOf(t)}}),lang:L(function(n){return oe.test(n||"")||D.error("unsupported lang: "+n),n=n.replace(j,c).toLowerCase(),function(e){var t;do{if(t=E?e.lang:e.getAttribute("xml:lang")||e.getAttribute("lang"))return(t=t.toLowerCase())===n||0===t.indexOf(n+"-")}while((e=e.parentNode)&&1===e.nodeType);return!1}}),target:function(e){var t=R.location&&R.location.hash;return t&&t.slice(1)===e.id},root:function(e){return e===n},focus:function(e){return e===C.activeElement&&(!C.hasFocus||C.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},enabled:ge(!1),disabled:ge(!0),checked:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&!!e.checked||"option"===t&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,!0===e.selected},empty:function(e){for(e=e.firstChild;e;e=e.nextSibling)if(e.nodeType<6)return!1;return!0},parent:function(e){return!b.pseudos.empty(e)},header:function(e){return ue.test(e.nodeName)},input:function(e){return se.test(e.nodeName)},button:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&"button"===e.type||"button"===t},text:function(e){return"input"===e.nodeName.toLowerCase()&&"text"===e.type&&(null==(e=e.getAttribute("type"))||"text"===e.toLowerCase())},first:O(function(){return[0]}),last:O(function(e,t){return[t-1]}),eq:O(function(e,t,n){return[n<0?n+t:n]}),even:O(function(e,t){for(var n=0;n<t;n+=2)e.push(n);return e}),odd:O(function(e,t){for(var n=1;n<t;n+=2)e.push(n);return e}),lt:O(function(e,t,n){for(var r=n<0?n+t:t<n?t:n;0<=--r;)e.push(r);return e}),gt:O(function(e,t,n){for(var r=n<0?n+t:n;++r<t;)e.push(r);return e})}}).pseudos.nth=b.pseudos.eq,{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})b.pseudos[e]=function(t){return function(e){return"input"===e.nodeName.toLowerCase()&&e.type===t}}(e);for(e in{submit:!0,reset:!0})b.pseudos[e]=function(n){return function(e){var t=e.nodeName.toLowerCase();return("input"===t||"button"===t)&&e.type===n}}(e);function me(){}function P(e){for(var t=0,n=e.length,r="";t<n;t++)r+=e[t].value;return r}function ve(a,e,t){var s=e.dir,u=e.next,l=u||s,c=t&&"parentNode"===l,f=B++;return e.first?function(e,t,n){while(e=e[s])if(1===e.nodeType||c)return a(e,t,n);return!1}:function(e,t,n){var r,i,o=[k,f];if(n){while(e=e[s])if((1===e.nodeType||c)&&a(e,t,n))return!0}else while(e=e[s])if(1===e.nodeType||c)if(i=(i=e[S]||(e[S]={}))[e.uniqueID]||(i[e.uniqueID]={}),u&&u===e.nodeName.toLowerCase())e=e[s]||e;else{if((r=i[l])&&r[0]===k&&r[1]===f)return o[2]=r[2];if((i[l]=o)[2]=a(e,t,n))return!0}return!1}}function xe(i){return 1<i.length?function(e,t,n){var r=i.length;while(r--)if(!i[r](e,t,n))return!1;return!0}:i[0]}function be(e,t,n,r,i){for(var o,a=[],s=0,u=e.length,l=null!=t;s<u;s++)!(o=e[s])||n&&!n(o,r,i)||(a.push(o),l&&t.push(s));return a}function we(d,h,g,y,m,e){return y&&!y[S]&&(y=we(y)),m&&!m[S]&&(m=we(m,e)),L(function(e,t,n,r){var i,o,a,s=[],u=[],l=t.length,c=e||function(e,t,n){for(var r=0,i=t.length;r<i;r++)D(e,t[r],n);return n}(h||"*",n.nodeType?[n]:n,[]),f=!d||!e&&h?c:be(c,s,d,n,r),p=g?m||(e?d:l||y)?[]:t:f;if(g&&g(f,p,n,r),y){i=be(p,u),y(i,[],n,r),o=i.length;while(o--)(a=i[o])&&(p[u[o]]=!(f[u[o]]=a))}if(e){if(m||d){if(m){i=[],o=p.length;while(o--)(a=p[o])&&i.push(f[o]=a);m(null,p=[],i,r)}o=p.length;while(o--)(a=p[o])&&-1<(i=m?v(e,a):s[o])&&(e[i]=!(t[i]=a))}}else p=be(p===t?p.splice(l,p.length):p),m?m(null,t,p,r):A.apply(t,p)})}function Te(y,m){function e(e,t,n,r,i){var o,a,s,u=0,l="0",c=e&&[],f=[],p=w,d=e||x&&b.find.TAG("*",i),h=k+=null==p?1:Math.random()||.1,g=d.length;for(i&&(w=t==C||t||i);l!==g&&null!=(o=d[l]);l++){if(x&&o){a=0,t||o.ownerDocument==C||(T(o),n=!E);while(s=y[a++])if(s(o,t||C,n)){r.push(o);break}i&&(k=h)}v&&((o=!s&&o)&&u--,e&&c.push(o))}if(u+=l,v&&l!==u){a=0;while(s=m[a++])s(c,f,t,n);if(e){if(0<u)while(l--)c[l]||f[l]||(f[l]=G.call(r));f=be(f)}A.apply(r,f),i&&!e&&0<f.length&&1<u+m.length&&D.uniqueSort(r)}return i&&(k=h,w=p),c}var v=0<m.length,x=0<y.length;return v?L(e):e}return me.prototype=b.filters=b.pseudos,b.setFilters=new me,d=D.tokenize=function(e,t){var n,r,i,o,a,s,u,l=z[e+" "];if(l)return t?0:l.slice(0);a=e,s=[],u=b.preFilter;while(a){for(o in n&&!(r=te.exec(a))||(r&&(a=a.slice(r[0].length)||a),s.push(i=[])),n=!1,(r=ne.exec(a))&&(n=r.shift(),i.push({value:n,type:r[0].replace(m," ")}),a=a.slice(n.length)),b.filter)!(r=x[o].exec(a))||u[o]&&!(r=u[o](r))||(n=r.shift(),i.push({value:n,type:o,matches:r}),a=a.slice(n.length));if(!n)break}return t?a.length:a?D.error(e):z(e,s).slice(0)},F=D.compile=function(e,t){var n,r=[],i=[],o=U[e+" "];if(!o){n=(t=t||d(e)).length;while(n--)((o=function e(t){for(var r,n,i,o=t.length,a=b.relative[t[0].type],s=a||b.relative[" "],u=a?1:0,l=ve(function(e){return e===r},s,!0),c=ve(function(e){return-1<v(r,e)},s,!0),f=[function(e,t,n){return e=!a&&(n||t!==w)||((r=t).nodeType?l:c)(e,t,n),r=null,e}];u<o;u++)if(n=b.relative[t[u].type])f=[ve(xe(f),n)];else{if((n=b.filter[t[u].type].apply(null,t[u].matches))[S]){for(i=++u;i<o;i++)if(b.relative[t[i].type])break;return we(1<u&&xe(f),1<u&&P(t.slice(0,u-1).concat({value:" "===t[u-2].type?"*":""})).replace(m,"$1"),n,u<i&&e(t.slice(u,i)),i<o&&e(t=t.slice(i)),i<o&&P(t))}f.push(n)}return xe(f)}(t[n]))[S]?r:i).push(o);(o=U(e,Te(i,r))).selector=e}return o},$=D.select=function(e,t,n,r){var i,o,a,s,u="function"==typeof e&&e,l=!r&&d(e=u.selector||e);if(n=n||[],1===l.length){if(2<(o=l[0]=l[0].slice(0)).length&&"ID"===(a=o[0]).type&&9===t.nodeType&&E&&b.relative[o[1].type]){if(!(t=(b.find.ID(a.matches[0].replace(j,c),t)||[])[0]))return n;u&&(t=t.parentNode),e=e.slice(o.shift().value.length)}i=x.needsContext.test(e)?0:o.length;while(i--){if(a=o[i],b.relative[s=a.type])break;if((s=b.find[s])&&(r=s(a.matches[0].replace(j,c),ce.test(o[0].type)&&ye(t.parentNode)||t))){if(o.splice(i,1),e=r.length&&P(o))break;return A.apply(n,r),n}}}return(u||F(e,l))(r,t,!E,n,!t||ce.test(e)&&ye(t.parentNode)||t),n},p.sortStable=S.split("").sort(X).join("")===S,p.detectDuplicates=!!l,T(),p.sortDetached=H(function(e){return 1&e.compareDocumentPosition(C.createElement("fieldset"))}),H(function(e){return e.innerHTML="<a href='#'></a>","#"===e.firstChild.getAttribute("href")})||de("type|href|height|width",function(e,t,n){if(!n)return e.getAttribute(t,"type"===t.toLowerCase()?1:2)}),p.attributes&&H(function(e){return e.innerHTML="<input/>",e.firstChild.setAttribute("value",""),""===e.firstChild.getAttribute("value")})||de("value",function(e,t,n){if(!n&&"input"===e.nodeName.toLowerCase())return e.defaultValue}),H(function(e){return null==e.getAttribute("disabled")})||de(J,function(e,t,n){if(!n)return!0===e[t]?t.toLowerCase():(n=e.getAttributeNode(t))&&n.specified?n.value:null}),D}(w),Q=(C.find=e,C.expr=e.selectors,C.expr[":"]=C.expr.pseudos,C.uniqueSort=C.unique=e.uniqueSort,C.text=e.getText,C.isXMLDoc=e.isXML,C.contains=e.contains,C.escapeSelector=e.escape,C.expr.match.needsContext);function u(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()}var J=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function K(e,n,r){return v(n)?C.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?C.grep(e,function(e){return e===n!==r}):"string"!=typeof n?C.grep(e,function(e){return-1<F.call(n,e)!==r}):C.filter(n,e,r)}C.filter=function(e,t,n){var r=t[0];return n&&(e=":not("+e+")"),1===t.length&&1===r.nodeType?C.find.matchesSelector(r,e)?[r]:[]:C.find.matches(e,C.grep(t,function(e){return 1===e.nodeType}))},C.fn.extend({find:function(e){var t,n,r=this.length,i=this;if("string"!=typeof e)return this.pushStack(C(e).filter(function(){for(t=0;t<r;t++)if(C.contains(i[t],this))return!0}));for(n=this.pushStack([]),t=0;t<r;t++)C.find(e,i[t],n);return 1<r?C.uniqueSort(n):n},filter:function(e){return this.pushStack(K(this,e||[],!1))},not:function(e){return this.pushStack(K(this,e||[],!0))},is:function(e){return!!K(this,"string"==typeof e&&Q.test(e)?C(e):e||[],!1).length}});var Z,ee=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,te=((C.fn.init=function(e,t,n){if(e){if(n=n||Z,"string"!=typeof e)return e.nodeType?(this[0]=e,this.length=1,this):v(e)?void 0!==n.ready?n.ready(e):e(C):C.makeArray(e,this);if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:ee.exec(e))||!r[1]&&t)return(!t||t.jquery?t||n:this.constructor(t)).find(e);if(r[1]){if(t=t instanceof C?t[0]:t,C.merge(this,C.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:T,!0)),J.test(r[1])&&C.isPlainObject(t))for(var r in t)v(this[r])?this[r](t[r]):this.attr(r,t[r])}else(n=T.getElementById(r[2]))&&(this[0]=n,this.length=1)}return this}).prototype=C.fn,Z=C(T),/^(?:parents|prev(?:Until|All))/),ne={children:!0,contents:!0,next:!0,prev:!0};function re(e,t){while((e=e[t])&&1!==e.nodeType);return e}C.fn.extend({has:function(e){var t=C(e,this),n=t.length;return this.filter(function(){for(var e=0;e<n;e++)if(C.contains(this,t[e]))return!0})},closest:function(e,t){var n,r=0,i=this.length,o=[],a="string"!=typeof e&&C(e);if(!Q.test(e))for(;r<i;r++)for(n=this[r];n&&n!==t;n=n.parentNode)if(n.nodeType<11&&(a?-1<a.index(n):1===n.nodeType&&C.find.matchesSelector(n,e))){o.push(n);break}return this.pushStack(1<o.length?C.uniqueSort(o):o)},index:function(e){return e?"string"==typeof e?F.call(C(e),this[0]):F.call(this,e.jquery?e[0]:e):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(e,t){return this.pushStack(C.uniqueSort(C.merge(this.get(),C(e,t))))},addBack:function(e){return this.add(null==e?this.prevObject:this.prevObject.filter(e))}}),C.each({parent:function(e){e=e.parentNode;return e&&11!==e.nodeType?e:null},parents:function(e){return r(e,"parentNode")},parentsUntil:function(e,t,n){return r(e,"parentNode",n)},next:function(e){return re(e,"nextSibling")},prev:function(e){return re(e,"previousSibling")},nextAll:function(e){return r(e,"nextSibling")},prevAll:function(e){return r(e,"previousSibling")},nextUntil:function(e,t,n){return r(e,"nextSibling",n)},prevUntil:function(e,t,n){return r(e,"previousSibling",n)},siblings:function(e){return Y((e.parentNode||{}).firstChild,e)},children:function(e){return Y(e.firstChild)},contents:function(e){return null!=e.contentDocument&&M(e.contentDocument)?e.contentDocument:(u(e,"template")&&(e=e.content||e),C.merge([],e.childNodes))}},function(r,i){C.fn[r]=function(e,t){var n=C.map(this,i,e);return(t="Until"!==r.slice(-5)?e:t)&&"string"==typeof t&&(n=C.filter(t,n)),1<this.length&&(ne[r]||C.uniqueSort(n),te.test(r)&&n.reverse()),this.pushStack(n)}});var E=/[^\x20\t\r\n\f]+/g;function c(e){return e}function ie(e){throw e}function oe(e,t,n,r){var i;try{e&&v(i=e.promise)?i.call(e).done(t).fail(n):e&&v(i=e.then)?i.call(e,t,n):t.apply(void 0,[e].slice(r))}catch(e){n.apply(void 0,[e])}}C.Callbacks=function(r){var e,n;r="string"==typeof r?(e=r,n={},C.each(e.match(E)||[],function(e,t){n[t]=!0}),n):C.extend({},r);function i(){for(s=s||r.once,a=o=!0;l.length;c=-1){t=l.shift();while(++c<u.length)!1===u[c].apply(t[0],t[1])&&r.stopOnFalse&&(c=u.length,t=!1)}r.memory||(t=!1),o=!1,s&&(u=t?[]:"")}var o,t,a,s,u=[],l=[],c=-1,f={add:function(){return u&&(t&&!o&&(c=u.length-1,l.push(t)),function n(e){C.each(e,function(e,t){v(t)?r.unique&&f.has(t)||u.push(t):t&&t.length&&"string"!==h(t)&&n(t)})}(arguments),t&&!o&&i()),this},remove:function(){return C.each(arguments,function(e,t){var n;while(-1<(n=C.inArray(t,u,n)))u.splice(n,1),n<=c&&c--}),this},has:function(e){return e?-1<C.inArray(e,u):0<u.length},empty:function(){return u=u&&[],this},disable:function(){return s=l=[],u=t="",this},disabled:function(){return!u},lock:function(){return s=l=[],t||o||(u=t=""),this},locked:function(){return!!s},fireWith:function(e,t){return s||(t=[e,(t=t||[]).slice?t.slice():t],l.push(t),o||i()),this},fire:function(){return f.fireWith(this,arguments),this},fired:function(){return!!a}};return f},C.extend({Deferred:function(e){var o=[["notify","progress",C.Callbacks("memory"),C.Callbacks("memory"),2],["resolve","done",C.Callbacks("once memory"),C.Callbacks("once memory"),0,"resolved"],["reject","fail",C.Callbacks("once memory"),C.Callbacks("once memory"),1,"rejected"]],i="pending",a={state:function(){return i},always:function(){return s.done(arguments).fail(arguments),this},catch:function(e){return a.then(null,e)},pipe:function(){var i=arguments;return C.Deferred(function(r){C.each(o,function(e,t){var n=v(i[t[4]])&&i[t[4]];s[t[1]](function(){var e=n&&n.apply(this,arguments);e&&v(e.promise)?e.promise().progress(r.notify).done(r.resolve).fail(r.reject):r[t[0]+"With"](this,n?[e]:arguments)})}),i=null}).promise()},then:function(t,n,r){var u=0;function l(i,o,a,s){return function(){function e(){var e,t;if(!(i<u)){if((e=a.apply(n,r))===o.promise())throw new TypeError("Thenable self-resolution");t=e&&("object"==typeof e||"function"==typeof e)&&e.then,v(t)?s?t.call(e,l(u,o,c,s),l(u,o,ie,s)):(u++,t.call(e,l(u,o,c,s),l(u,o,ie,s),l(u,o,c,o.notifyWith))):(a!==c&&(n=void 0,r=[e]),(s||o.resolveWith)(n,r))}}var n=this,r=arguments,t=s?e:function(){try{e()}catch(e){C.Deferred.exceptionHook&&C.Deferred.exceptionHook(e,t.stackTrace),u<=i+1&&(a!==ie&&(n=void 0,r=[e]),o.rejectWith(n,r))}};i?t():(C.Deferred.getStackHook&&(t.stackTrace=C.Deferred.getStackHook()),w.setTimeout(t))}}return C.Deferred(function(e){o[0][3].add(l(0,e,v(r)?r:c,e.notifyWith)),o[1][3].add(l(0,e,v(t)?t:c)),o[2][3].add(l(0,e,v(n)?n:ie))}).promise()},promise:function(e){return null!=e?C.extend(e,a):a}},s={};return C.each(o,function(e,t){var n=t[2],r=t[5];a[t[1]]=n.add,r&&n.add(function(){i=r},o[3-e][2].disable,o[3-e][3].disable,o[0][2].lock,o[0][3].lock),n.add(t[3].fire),s[t[0]]=function(){return s[t[0]+"With"](this===s?void 0:this,arguments),this},s[t[0]+"With"]=n.fireWith}),a.promise(s),e&&e.call(s,s),s},when:function(e){function t(t){return function(e){i[t]=this,o[t]=1<arguments.length?s.call(arguments):e,--n||a.resolveWith(i,o)}}var n=arguments.length,r=n,i=Array(r),o=s.call(arguments),a=C.Deferred();if(n<=1&&(oe(e,a.done(t(r)).resolve,a.reject,!n),"pending"===a.state()||v(o[r]&&o[r].then)))return a.then();while(r--)oe(o[r],t(r),a.reject);return a.promise()}});var ae=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/,se=(C.Deferred.exceptionHook=function(e,t){w.console&&w.console.warn&&e&&ae.test(e.name)&&w.console.warn("jQuery.Deferred exception: "+e.message,e.stack,t)},C.readyException=function(e){w.setTimeout(function(){throw e})},C.Deferred());function ue(){T.removeEventListener("DOMContentLoaded",ue),w.removeEventListener("load",ue),C.ready()}C.fn.ready=function(e){return se.then(e).catch(function(e){C.readyException(e)}),this},C.extend({isReady:!1,readyWait:1,ready:function(e){(!0===e?--C.readyWait:C.isReady)||(C.isReady=!0)!==e&&0<--C.readyWait||se.resolveWith(T,[C])}}),C.ready.then=se.then,"complete"===T.readyState||"loading"!==T.readyState&&!T.documentElement.doScroll?w.setTimeout(C.ready):(T.addEventListener("DOMContentLoaded",ue),w.addEventListener("load",ue));function f(e,t,n,r,i,o,a){var s=0,u=e.length,l=null==n;if("object"===h(n))for(s in i=!0,n)f(e,t,s,n[s],!0,o,a);else if(void 0!==r&&(i=!0,v(r)||(a=!0),t=l?a?(t.call(e,r),null):(l=t,function(e,t,n){return l.call(C(e),n)}):t))for(;s<u;s++)t(e[s],n,a?r:r.call(e[s],s,t(e[s],n)));return i?e:l?t.call(e):u?t(e[0],n):o}var le=/^-ms-/,ce=/-([a-z])/g;function fe(e,t){return t.toUpperCase()}function x(e){return e.replace(le,"ms-").replace(ce,fe)}function m(e){return 1===e.nodeType||9===e.nodeType||!+e.nodeType}function pe(){this.expando=C.expando+pe.uid++}pe.uid=1,pe.prototype={cache:function(e){var t=e[this.expando];return t||(t={},m(e)&&(e.nodeType?e[this.expando]=t:Object.defineProperty(e,this.expando,{value:t,configurable:!0}))),t},set:function(e,t,n){var r,i=this.cache(e);if("string"==typeof t)i[x(t)]=n;else for(r in t)i[x(r)]=t[r];return i},get:function(e,t){return void 0===t?this.cache(e):e[this.expando]&&e[this.expando][x(t)]},access:function(e,t,n){return void 0===t||t&&"string"==typeof t&&void 0===n?this.get(e,t):(this.set(e,t,n),void 0!==n?n:t)},remove:function(e,t){var n,r=e[this.expando];if(void 0!==r){if(void 0!==t){n=(t=Array.isArray(t)?t.map(x):(t=x(t))in r?[t]:t.match(E)||[]).length;while(n--)delete r[t[n]]}void 0!==t&&!C.isEmptyObject(r)||(e.nodeType?e[this.expando]=void 0:delete e[this.expando])}},hasData:function(e){e=e[this.expando];return void 0!==e&&!C.isEmptyObject(e)}};var b=new pe,l=new pe,de=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,he=/[A-Z]/g;function ge(e,t,n){var r,i;if(void 0===n&&1===e.nodeType)if(r="data-"+t.replace(he,"-$&").toLowerCase(),"string"==typeof(n=e.getAttribute(r))){try{n="true"===(i=n)||"false"!==i&&("null"===i?null:i===+i+""?+i:de.test(i)?JSON.parse(i):i)}catch(e){}l.set(e,t,n)}else n=void 0;return n}C.extend({hasData:function(e){return l.hasData(e)||b.hasData(e)},data:function(e,t,n){return l.access(e,t,n)},removeData:function(e,t){l.remove(e,t)},_data:function(e,t,n){return b.access(e,t,n)},_removeData:function(e,t){b.remove(e,t)}}),C.fn.extend({data:function(n,e){var t,r,i,o=this[0],a=o&&o.attributes;if(void 0!==n)return"object"==typeof n?this.each(function(){l.set(this,n)}):f(this,function(e){var t;if(o&&void 0===e)return void 0!==(t=l.get(o,n))||void 0!==(t=ge(o,n))?t:void 0;this.each(function(){l.set(this,n,e)})},null,e,1<arguments.length,null,!0);if(this.length&&(i=l.get(o),1===o.nodeType&&!b.get(o,"hasDataAttrs"))){t=a.length;while(t--)a[t]&&0===(r=a[t].name).indexOf("data-")&&(r=x(r.slice(5)),ge(o,r,i[r]));b.set(o,"hasDataAttrs",!0)}return i},removeData:function(e){return this.each(function(){l.remove(this,e)})}}),C.extend({queue:function(e,t,n){var r;if(e)return r=b.get(e,t=(t||"fx")+"queue"),n&&(!r||Array.isArray(n)?r=b.access(e,t,C.makeArray(n)):r.push(n)),r||[]},dequeue:function(e,t){t=t||"fx";var n=C.queue(e,t),r=n.length,i=n.shift(),o=C._queueHooks(e,t);"inprogress"===i&&(i=n.shift(),r--),i&&("fx"===t&&n.unshift("inprogress"),delete o.stop,i.call(e,function(){C.dequeue(e,t)},o)),!r&&o&&o.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return b.get(e,n)||b.access(e,n,{empty:C.Callbacks("once memory").add(function(){b.remove(e,[t+"queue",n])})})}}),C.fn.extend({queue:function(t,n){var e=2;return"string"!=typeof t&&(n=t,t="fx",e--),arguments.length<e?C.queue(this[0],t):void 0===n?this:this.each(function(){var e=C.queue(this,t,n);C._queueHooks(this,t),"fx"===t&&"inprogress"!==e[0]&&C.dequeue(this,t)})},dequeue:function(e){return this.each(function(){C.dequeue(this,e)})},clearQueue:function(e){return this.queue(e||"fx",[])},promise:function(e,t){function n(){--i||o.resolveWith(a,[a])}var r,i=1,o=C.Deferred(),a=this,s=this.length;"string"!=typeof e&&(t=e,e=void 0),e=e||"fx";while(s--)(r=b.get(a[s],e+"queueHooks"))&&r.empty&&(i++,r.empty.add(n));return n(),o.promise(t)}});function ye(e,t){return"none"===(e=t||e).style.display||""===e.style.display&&k(e)&&"none"===C.css(e,"display")}var e=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,me=new RegExp("^(?:([+-])=|)("+e+")([a-z%]*)$","i"),p=["Top","Right","Bottom","Left"],S=T.documentElement,k=function(e){return C.contains(e.ownerDocument,e)},ve={composed:!0};S.getRootNode&&(k=function(e){return C.contains(e.ownerDocument,e)||e.getRootNode(ve)===e.ownerDocument});function xe(e,t,n,r){var i,o,a=20,s=r?function(){return r.cur()}:function(){return C.css(e,t,"")},u=s(),l=n&&n[3]||(C.cssNumber[t]?"":"px"),c=e.nodeType&&(C.cssNumber[t]||"px"!==l&&+u)&&me.exec(C.css(e,t));if(c&&c[3]!==l){l=l||c[3],c=+(u/=2)||1;while(a--)C.style(e,t,c+l),(1-o)*(1-(o=s()/u||.5))<=0&&(a=0),c/=o;C.style(e,t,(c*=2)+l),n=n||[]}return n&&(c=+c||+u||0,i=n[1]?c+(n[1]+1)*n[2]:+n[2],r&&(r.unit=l,r.start=c,r.end=i)),i}var be={};function A(e,t){for(var n,r,i,o,a,s=[],u=0,l=e.length;u<l;u++)(r=e[u]).style&&(n=r.style.display,t?("none"===n&&(s[u]=b.get(r,"display")||null,s[u]||(r.style.display="")),""===r.style.display&&ye(r)&&(s[u]=(a=o=void 0,o=(i=r).ownerDocument,i=i.nodeName,(a=be[i])||(o=o.body.appendChild(o.createElement(i)),a=C.css(o,"display"),o.parentNode.removeChild(o),be[i]=a="none"===a?"block":a),a))):"none"!==n&&(s[u]="none",b.set(r,"display",n)));for(u=0;u<l;u++)null!=s[u]&&(e[u].style.display=s[u]);return e}C.fn.extend({show:function(){return A(this,!0)},hide:function(){return A(this)},toggle:function(e){return"boolean"==typeof e?e?this.show():this.hide():this.each(function(){ye(this)?C(this).show():C(this).hide()})}});var we=/^(?:checkbox|radio)$/i,Te=/<([a-z][^\/\0>\x20\t\r\n\f]*)/i,Ce=/^$|^module$|\/(?:java|ecma)script/i,N=(L=T.createDocumentFragment().appendChild(T.createElement("div")),(o=T.createElement("input")).setAttribute("type","radio"),o.setAttribute("checked","checked"),o.setAttribute("name","t"),L.appendChild(o),y.checkClone=L.cloneNode(!0).cloneNode(!0).lastChild.checked,L.innerHTML="<textarea>x</textarea>",y.noCloneChecked=!!L.cloneNode(!0).lastChild.defaultValue,L.innerHTML="<option></option>",y.option=!!L.lastChild,{thead:[1,"<table>","</table>"],col:[2,"<table><colgroup>","</colgroup></table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:[0,"",""]});function j(e,t){var n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[];return void 0===t||t&&u(e,t)?C.merge([e],n):n}function Ee(e,t){for(var n=0,r=e.length;n<r;n++)b.set(e[n],"globalEval",!t||b.get(t[n],"globalEval"))}N.tbody=N.tfoot=N.colgroup=N.caption=N.thead,N.th=N.td,y.option||(N.optgroup=N.option=[1,"<select multiple='multiple'>","</select>"]);var Se=/<|&#?\w+;/;function ke(e,t,n,r,i){for(var o,a,s,u,l,c=t.createDocumentFragment(),f=[],p=0,d=e.length;p<d;p++)if((o=e[p])||0===o)if("object"===h(o))C.merge(f,o.nodeType?[o]:o);else if(Se.test(o)){a=a||c.appendChild(t.createElement("div")),s=(Te.exec(o)||["",""])[1].toLowerCase(),s=N[s]||N._default,a.innerHTML=s[1]+C.htmlPrefilter(o)+s[2],l=s[0];while(l--)a=a.lastChild;C.merge(f,a.childNodes),(a=c.firstChild).textContent=""}else f.push(t.createTextNode(o));c.textContent="",p=0;while(o=f[p++])if(r&&-1<C.inArray(o,r))i&&i.push(o);else if(u=k(o),a=j(c.appendChild(o),"script"),u&&Ee(a),n){l=0;while(o=a[l++])Ce.test(o.type||"")&&n.push(o)}return c}var Ae=/^([^.]*)(?:\.(.+)|)/;function n(){return!0}function d(){return!1}function Ne(e,t){return e===function(){try{return T.activeElement}catch(e){}}()==("focus"===t)}function je(e,t,n,r,i,o){var a,s;if("object"==typeof t){for(s in"string"!=typeof n&&(r=r||n,n=void 0),t)je(e,s,n,r,t[s],o);return e}if(null==r&&null==i?(i=n,r=n=void 0):null==i&&("string"==typeof n?(i=r,r=void 0):(i=r,r=n,n=void 0)),!1===i)i=d;else if(!i)return e;return 1===o&&(a=i,(i=function(e){return C().off(e),a.apply(this,arguments)}).guid=a.guid||(a.guid=C.guid++)),e.each(function(){C.event.add(this,t,i,r,n)})}function De(e,i,o){o?(b.set(e,i,!1),C.event.add(e,i,{namespace:!1,handler:function(e){var t,n,r=b.get(this,i);if(1&e.isTrigger&&this[i]){if(r.length)(C.event.special[i]||{}).delegateType&&e.stopPropagation();else if(r=s.call(arguments),b.set(this,i,r),t=o(this,i),this[i](),r!==(n=b.get(this,i))||t?b.set(this,i,!1):n={},r!==n)return e.stopImmediatePropagation(),e.preventDefault(),n&&n.value}else r.length&&(b.set(this,i,{value:C.event.trigger(C.extend(r[0],C.Event.prototype),r.slice(1),this)}),e.stopImmediatePropagation())}})):void 0===b.get(e,i)&&C.event.add(e,i,n)}C.event={global:{},add:function(t,e,n,r,i){var o,a,s,u,l,c,f,p,d,h=b.get(t);if(m(t)){n.handler&&(n=(o=n).handler,i=o.selector),i&&C.find.matchesSelector(S,i),n.guid||(n.guid=C.guid++),(s=h.events)||(s=h.events=Object.create(null)),(a=h.handle)||(a=h.handle=function(e){return"undefined"!=typeof C&&C.event.triggered!==e.type?C.event.dispatch.apply(t,arguments):void 0}),u=(e=(e||"").match(E)||[""]).length;while(u--)f=d=(p=Ae.exec(e[u])||[])[1],p=(p[2]||"").split(".").sort(),f&&(l=C.event.special[f]||{},f=(i?l.delegateType:l.bindType)||f,l=C.event.special[f]||{},d=C.extend({type:f,origType:d,data:r,handler:n,guid:n.guid,selector:i,needsContext:i&&C.expr.match.needsContext.test(i),namespace:p.join(".")},o),(c=s[f])||((c=s[f]=[]).delegateCount=0,l.setup&&!1!==l.setup.call(t,r,p,a)||t.addEventListener&&t.addEventListener(f,a)),l.add&&(l.add.call(t,d),d.handler.guid||(d.handler.guid=n.guid)),i?c.splice(c.delegateCount++,0,d):c.push(d),C.event.global[f]=!0)}},remove:function(e,t,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,y=b.hasData(e)&&b.get(e);if(y&&(u=y.events)){l=(t=(t||"").match(E)||[""]).length;while(l--)if(d=g=(s=Ae.exec(t[l])||[])[1],h=(s[2]||"").split(".").sort(),d){f=C.event.special[d]||{},p=u[d=(r?f.delegateType:f.bindType)||d]||[],s=s[2]&&new RegExp("(^|\\.)"+h.join("\\.(?:.*\\.|)")+"(\\.|$)"),a=o=p.length;while(o--)c=p[o],!i&&g!==c.origType||n&&n.guid!==c.guid||s&&!s.test(c.namespace)||r&&r!==c.selector&&("**"!==r||!c.selector)||(p.splice(o,1),c.selector&&p.delegateCount--,f.remove&&f.remove.call(e,c));a&&!p.length&&(f.teardown&&!1!==f.teardown.call(e,h,y.handle)||C.removeEvent(e,d,y.handle),delete u[d])}else for(d in u)C.event.remove(e,d+t[l],n,r,!0);C.isEmptyObject(u)&&b.remove(e,"handle events")}},dispatch:function(e){var t,n,r,i,o,a=new Array(arguments.length),s=C.event.fix(e),e=(b.get(this,"events")||Object.create(null))[s.type]||[],u=C.event.special[s.type]||{};for(a[0]=s,t=1;t<arguments.length;t++)a[t]=arguments[t];if(s.delegateTarget=this,!u.preDispatch||!1!==u.preDispatch.call(this,s)){o=C.event.handlers.call(this,s,e),t=0;while((r=o[t++])&&!s.isPropagationStopped()){s.currentTarget=r.elem,n=0;while((i=r.handlers[n++])&&!s.isImmediatePropagationStopped())s.rnamespace&&!1!==i.namespace&&!s.rnamespace.test(i.namespace)||(s.handleObj=i,s.data=i.data,void 0!==(i=((C.event.special[i.origType]||{}).handle||i.handler).apply(r.elem,a))&&!1===(s.result=i)&&(s.preventDefault(),s.stopPropagation()))}return u.postDispatch&&u.postDispatch.call(this,s),s.result}},handlers:function(e,t){var n,r,i,o,a,s=[],u=t.delegateCount,l=e.target;if(u&&l.nodeType&&!("click"===e.type&&1<=e.button))for(;l!==this;l=l.parentNode||this)if(1===l.nodeType&&("click"!==e.type||!0!==l.disabled)){for(o=[],a={},n=0;n<u;n++)void 0===a[i=(r=t[n]).selector+" "]&&(a[i]=r.needsContext?-1<C(i,this).index(l):C.find(i,this,null,[l]).length),a[i]&&o.push(r);o.length&&s.push({elem:l,handlers:o})}return l=this,u<t.length&&s.push({elem:l,handlers:t.slice(u)}),s},addProp:function(t,e){Object.defineProperty(C.Event.prototype,t,{enumerable:!0,configurable:!0,get:v(e)?function(){if(this.originalEvent)return e(this.originalEvent)}:function(){if(this.originalEvent)return this.originalEvent[t]},set:function(e){Object.defineProperty(this,t,{enumerable:!0,configurable:!0,writable:!0,value:e})}})},fix:function(e){return e[C.expando]?e:new C.Event(e)},special:{load:{noBubble:!0},click:{setup:function(e){e=this||e;return we.test(e.type)&&e.click&&u(e,"input")&&De(e,"click",n),!1},trigger:function(e){e=this||e;return we.test(e.type)&&e.click&&u(e,"input")&&De(e,"click"),!0},_default:function(e){e=e.target;return we.test(e.type)&&e.click&&u(e,"input")&&b.get(e,"click")||u(e,"a")}},beforeunload:{postDispatch:function(e){void 0!==e.result&&e.originalEvent&&(e.originalEvent.returnValue=e.result)}}}},C.removeEvent=function(e,t,n){e.removeEventListener&&e.removeEventListener(t,n)},C.Event=function(e,t){if(!(this instanceof C.Event))return new C.Event(e,t);e&&e.type?(this.originalEvent=e,this.type=e.type,this.isDefaultPrevented=e.defaultPrevented||void 0===e.defaultPrevented&&!1===e.returnValue?n:d,this.target=e.target&&3===e.target.nodeType?e.target.parentNode:e.target,this.currentTarget=e.currentTarget,this.relatedTarget=e.relatedTarget):this.type=e,t&&C.extend(this,t),this.timeStamp=e&&e.timeStamp||Date.now(),this[C.expando]=!0},C.Event.prototype={constructor:C.Event,isDefaultPrevented:d,isPropagationStopped:d,isImmediatePropagationStopped:d,isSimulated:!1,preventDefault:function(){var e=this.originalEvent;this.isDefaultPrevented=n,e&&!this.isSimulated&&e.preventDefault()},stopPropagation:function(){var e=this.originalEvent;this.isPropagationStopped=n,e&&!this.isSimulated&&e.stopPropagation()},stopImmediatePropagation:function(){var e=this.originalEvent;this.isImmediatePropagationStopped=n,e&&!this.isSimulated&&e.stopImmediatePropagation(),this.stopPropagation()}},C.each({altKey:!0,bubbles:!0,cancelable:!0,changedTouches:!0,ctrlKey:!0,detail:!0,eventPhase:!0,metaKey:!0,pageX:!0,pageY:!0,shiftKey:!0,view:!0,char:!0,code:!0,charCode:!0,key:!0,keyCode:!0,button:!0,buttons:!0,clientX:!0,clientY:!0,offsetX:!0,offsetY:!0,pointerId:!0,pointerType:!0,screenX:!0,screenY:!0,targetTouches:!0,toElement:!0,touches:!0,which:!0},C.event.addProp),C.each({focus:"focusin",blur:"focusout"},function(t,e){C.event.special[t]={setup:function(){return De(this,t,Ne),!1},trigger:function(){return De(this,t),!0},_default:function(e){return b.get(e.target,t)},delegateType:e}}),C.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(e,i){C.event.special[e]={delegateType:i,bindType:i,handle:function(e){var t,n=e.relatedTarget,r=e.handleObj;return n&&(n===this||C.contains(this,n))||(e.type=r.origType,t=r.handler.apply(this,arguments),e.type=i),t}}}),C.fn.extend({on:function(e,t,n,r){return je(this,e,t,n,r)},one:function(e,t,n,r){return je(this,e,t,n,r,1)},off:function(e,t,n){var r,i;if(e&&e.preventDefault&&e.handleObj)r=e.handleObj,C(e.delegateTarget).off(r.namespace?r.origType+"."+r.namespace:r.origType,r.selector,r.handler);else{if("object"!=typeof e)return!1!==t&&"function"!=typeof t||(n=t,t=void 0),!1===n&&(n=d),this.each(function(){C.event.remove(this,e,n,t)});for(i in e)this.off(i,t,e[i])}return this}});var qe=/<script|<style|<link/i,Le=/checked\s*(?:[^=]|=\s*.checked.)/i,He=/^\s*<!\[CDATA\[|\]\]>\s*$/g;function Oe(e,t){return u(e,"table")&&u(11!==t.nodeType?t:t.firstChild,"tr")&&C(e).children("tbody")[0]||e}function Pe(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function Re(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Me(e,t){var n,r,i,o;if(1===t.nodeType){if(b.hasData(e)&&(o=b.get(e).events))for(i in b.remove(t,"handle events"),o)for(n=0,r=o[i].length;n<r;n++)C.event.add(t,i,o[i][n]);l.hasData(e)&&(e=l.access(e),e=C.extend({},e),l.set(t,e))}}function D(n,r,i,o){r=I(r);var e,t,a,s,u,l,c=0,f=n.length,p=f-1,d=r[0],h=v(d);if(h||1<f&&"string"==typeof d&&!y.checkClone&&Le.test(d))return n.each(function(e){var t=n.eq(e);h&&(r[0]=d.call(this,e,t.html())),D(t,r,i,o)});if(f&&(t=(e=ke(r,n[0].ownerDocument,!1,n,o)).firstChild,1===e.childNodes.length&&(e=t),t||o)){for(s=(a=C.map(j(e,"script"),Pe)).length;c<f;c++)u=e,c!==p&&(u=C.clone(u,!0,!0),s&&C.merge(a,j(u,"script"))),i.call(n[c],u,c);if(s)for(l=a[a.length-1].ownerDocument,C.map(a,Re),c=0;c<s;c++)u=a[c],Ce.test(u.type||"")&&!b.access(u,"globalEval")&&C.contains(l,u)&&(u.src&&"module"!==(u.type||"").toLowerCase()?C._evalUrl&&!u.noModule&&C._evalUrl(u.src,{nonce:u.nonce||u.getAttribute("nonce")},l):V(u.textContent.replace(He,""),u,l))}return n}function Ie(e,t,n){for(var r,i=t?C.filter(t,e):e,o=0;null!=(r=i[o]);o++)n||1!==r.nodeType||C.cleanData(j(r)),r.parentNode&&(n&&k(r)&&Ee(j(r,"script")),r.parentNode.removeChild(r));return e}C.extend({htmlPrefilter:function(e){return e},clone:function(e,t,n){var r,i,o,a,s,u,l,c=e.cloneNode(!0),f=k(e);if(!(y.noCloneChecked||1!==e.nodeType&&11!==e.nodeType||C.isXMLDoc(e)))for(a=j(c),r=0,i=(o=j(e)).length;r<i;r++)s=o[r],u=a[r],l=void 0,"input"===(l=u.nodeName.toLowerCase())&&we.test(s.type)?u.checked=s.checked:"input"!==l&&"textarea"!==l||(u.defaultValue=s.defaultValue);if(t)if(n)for(o=o||j(e),a=a||j(c),r=0,i=o.length;r<i;r++)Me(o[r],a[r]);else Me(e,c);return 0<(a=j(c,"script")).length&&Ee(a,!f&&j(e,"script")),c},cleanData:function(e){for(var t,n,r,i=C.event.special,o=0;void 0!==(n=e[o]);o++)if(m(n)){if(t=n[b.expando]){if(t.events)for(r in t.events)i[r]?C.event.remove(n,r):C.removeEvent(n,r,t.handle);n[b.expando]=void 0}n[l.expando]&&(n[l.expando]=void 0)}}}),C.fn.extend({detach:function(e){return Ie(this,e,!0)},remove:function(e){return Ie(this,e)},text:function(e){return f(this,function(e){return void 0===e?C.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=e)})},null,e,arguments.length)},append:function(){return D(this,arguments,function(e){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||Oe(this,e).appendChild(e)})},prepend:function(){return D(this,arguments,function(e){var t;1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(t=Oe(this,e)).insertBefore(e,t.firstChild)})},before:function(){return D(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this)})},after:function(){return D(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this.nextSibling)})},empty:function(){for(var e,t=0;null!=(e=this[t]);t++)1===e.nodeType&&(C.cleanData(j(e,!1)),e.textContent="");return this},clone:function(e,t){return e=null!=e&&e,t=null==t?e:t,this.map(function(){return C.clone(this,e,t)})},html:function(e){return f(this,function(e){var t=this[0]||{},n=0,r=this.length;if(void 0===e&&1===t.nodeType)return t.innerHTML;if("string"==typeof e&&!qe.test(e)&&!N[(Te.exec(e)||["",""])[1].toLowerCase()]){e=C.htmlPrefilter(e);try{for(;n<r;n++)1===(t=this[n]||{}).nodeType&&(C.cleanData(j(t,!1)),t.innerHTML=e);t=0}catch(e){}}t&&this.empty().append(e)},null,e,arguments.length)},replaceWith:function(){var n=[];return D(this,arguments,function(e){var t=this.parentNode;C.inArray(this,n)<0&&(C.cleanData(j(this)),t&&t.replaceChild(e,this))},n)}}),C.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(e,a){C.fn[e]=function(e){for(var t,n=[],r=C(e),i=r.length-1,o=0;o<=i;o++)t=o===i?this:this.clone(!0),C(r[o])[a](t),W.apply(n,t.get());return this.pushStack(n)}});function We(e){var t=e.ownerDocument.defaultView;return(t=t&&t.opener?t:w).getComputedStyle(e)}function Fe(e,t,n){var r,i={};for(r in t)i[r]=e.style[r],e.style[r]=t[r];for(r in n=n.call(e),t)e.style[r]=i[r];return n}var $e,Be,_e,ze,Ue,Xe,Ve,i,Ge=new RegExp("^("+e+")(?!px)[a-z%]+$","i"),Ye=/^--/,Qe=new RegExp(p.join("|"),"i"),o="[\\x20\\t\\r\\n\\f]",Je=new RegExp("^"+o+"+|((?:^|[^\\\\])(?:\\\\.)*)"+o+"+$","g");function Ke(){var e;i&&(Ve.style.cssText="position:absolute;left:-11111px;width:60px;margin-top:1px;padding:0;border:0",i.style.cssText="position:relative;display:block;box-sizing:border-box;overflow:scroll;margin:auto;border:1px;padding:1px;width:60%;top:1%",S.appendChild(Ve).appendChild(i),e=w.getComputedStyle(i),$e="1%"!==e.top,Xe=12===Ze(e.marginLeft),i.style.right="60%",ze=36===Ze(e.right),Be=36===Ze(e.width),i.style.position="absolute",_e=12===Ze(i.offsetWidth/3),S.removeChild(Ve),i=null)}function Ze(e){return Math.round(parseFloat(e))}function et(e,t,n){var r,i=Ye.test(t),o=e.style;return(n=n||We(e))&&(r=n.getPropertyValue(t)||n[t],""!==(r=i?r.replace(Je,"$1"):r)||k(e)||(r=C.style(e,t)),!y.pixelBoxStyles()&&Ge.test(r)&&Qe.test(t)&&(i=o.width,e=o.minWidth,t=o.maxWidth,o.minWidth=o.maxWidth=o.width=r,r=n.width,o.width=i,o.minWidth=e,o.maxWidth=t)),void 0!==r?r+"":r}function tt(e,t){return{get:function(){if(!e())return(this.get=t).apply(this,arguments);delete this.get}}}Ve=T.createElement("div"),(i=T.createElement("div")).style&&(i.style.backgroundClip="content-box",i.cloneNode(!0).style.backgroundClip="",y.clearCloneStyle="content-box"===i.style.backgroundClip,C.extend(y,{boxSizingReliable:function(){return Ke(),Be},pixelBoxStyles:function(){return Ke(),ze},pixelPosition:function(){return Ke(),$e},reliableMarginLeft:function(){return Ke(),Xe},scrollboxSize:function(){return Ke(),_e},reliableTrDimensions:function(){var e,t,n;return null==Ue&&(e=T.createElement("table"),t=T.createElement("tr"),n=T.createElement("div"),e.style.cssText="position:absolute;left:-11111px;border-collapse:separate",t.style.cssText="border:1px solid",t.style.height="1px",n.style.height="9px",n.style.display="block",S.appendChild(e).appendChild(t).appendChild(n),n=w.getComputedStyle(t),Ue=parseInt(n.height,10)+parseInt(n.borderTopWidth,10)+parseInt(n.borderBottomWidth,10)===t.offsetHeight,S.removeChild(e)),Ue}}));var nt=["Webkit","Moz","ms"],rt=T.createElement("div").style,it={};function ot(e){var t=C.cssProps[e]||it[e];return t||(e in rt?e:it[e]=function(e){var t=e[0].toUpperCase()+e.slice(1),n=nt.length;while(n--)if((e=nt[n]+t)in rt)return e}(e)||e)}var at=/^(none|table(?!-c[ea]).+)/,st={position:"absolute",visibility:"hidden",display:"block"},ut={letterSpacing:"0",fontWeight:"400"};function lt(e,t,n){var r=me.exec(t);return r?Math.max(0,r[2]-(n||0))+(r[3]||"px"):t}function ct(e,t,n,r,i,o){var a="width"===t?1:0,s=0,u=0;if(n===(r?"border":"content"))return 0;for(;a<4;a+=2)"margin"===n&&(u+=C.css(e,n+p[a],!0,i)),r?("content"===n&&(u-=C.css(e,"padding"+p[a],!0,i)),"margin"!==n&&(u-=C.css(e,"border"+p[a]+"Width",!0,i))):(u+=C.css(e,"padding"+p[a],!0,i),"padding"!==n?u+=C.css(e,"border"+p[a]+"Width",!0,i):s+=C.css(e,"border"+p[a]+"Width",!0,i));return!r&&0<=o&&(u+=Math.max(0,Math.ceil(e["offset"+t[0].toUpperCase()+t.slice(1)]-o-u-s-.5))||0),u}function ft(e,t,n){var r=We(e),i=(!y.boxSizingReliable()||n)&&"border-box"===C.css(e,"boxSizing",!1,r),o=i,a=et(e,t,r),s="offset"+t[0].toUpperCase()+t.slice(1);if(Ge.test(a)){if(!n)return a;a="auto"}return(!y.boxSizingReliable()&&i||!y.reliableTrDimensions()&&u(e,"tr")||"auto"===a||!parseFloat(a)&&"inline"===C.css(e,"display",!1,r))&&e.getClientRects().length&&(i="border-box"===C.css(e,"boxSizing",!1,r),(o=s in e)&&(a=e[s])),(a=parseFloat(a)||0)+ct(e,t,n||(i?"border":"content"),o,r,a)+"px"}function a(e,t,n,r,i){return new a.prototype.init(e,t,n,r,i)}C.extend({cssHooks:{opacity:{get:function(e,t){if(t)return""===(t=et(e,"opacity"))?"1":t}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,gridArea:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnStart:!0,gridRow:!0,gridRowEnd:!0,gridRowStart:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{},style:function(e,t,n,r){if(e&&3!==e.nodeType&&8!==e.nodeType&&e.style){var i,o,a,s=x(t),u=Ye.test(t),l=e.style;if(u||(t=ot(s)),a=C.cssHooks[t]||C.cssHooks[s],void 0===n)return a&&"get"in a&&void 0!==(i=a.get(e,!1,r))?i:l[t];"string"===(o=typeof n)&&(i=me.exec(n))&&i[1]&&(n=xe(e,t,i),o="number"),null!=n&&n==n&&("number"!==o||u||(n+=i&&i[3]||(C.cssNumber[s]?"":"px")),y.clearCloneStyle||""!==n||0!==t.indexOf("background")||(l[t]="inherit"),a&&"set"in a&&void 0===(n=a.set(e,n,r))||(u?l.setProperty(t,n):l[t]=n))}},css:function(e,t,n,r){var i,o=x(t);return Ye.test(t)||(t=ot(o)),"normal"===(i=void 0===(i=(o=C.cssHooks[t]||C.cssHooks[o])&&"get"in o?o.get(e,!0,n):i)?et(e,t,r):i)&&t in ut&&(i=ut[t]),(""===n||n)&&(o=parseFloat(i),!0===n||isFinite(o))?o||0:i}}),C.each(["height","width"],function(e,a){C.cssHooks[a]={get:function(e,t,n){if(t)return!at.test(C.css(e,"display"))||e.getClientRects().length&&e.getBoundingClientRect().width?ft(e,a,n):Fe(e,st,function(){return ft(e,a,n)})},set:function(e,t,n){var r=We(e),i=!y.scrollboxSize()&&"absolute"===r.position,o=(i||n)&&"border-box"===C.css(e,"boxSizing",!1,r),n=n?ct(e,a,n,o,r):0;return o&&i&&(n-=Math.ceil(e["offset"+a[0].toUpperCase()+a.slice(1)]-parseFloat(r[a])-ct(e,a,"border",!1,r)-.5)),n&&(o=me.exec(t))&&"px"!==(o[3]||"px")&&(e.style[a]=t,t=C.css(e,a)),lt(0,t,n)}}}),C.cssHooks.marginLeft=tt(y.reliableMarginLeft,function(e,t){if(t)return(parseFloat(et(e,"marginLeft"))||e.getBoundingClientRect().left-Fe(e,{marginLeft:0},function(){return e.getBoundingClientRect().left}))+"px"}),C.each({margin:"",padding:"",border:"Width"},function(i,o){C.cssHooks[i+o]={expand:function(e){for(var t=0,n={},r="string"==typeof e?e.split(" "):[e];t<4;t++)n[i+p[t]+o]=r[t]||r[t-2]||r[0];return n}},"margin"!==i&&(C.cssHooks[i+o].set=lt)}),C.fn.extend({css:function(e,t){return f(this,function(e,t,n){var r,i,o={},a=0;if(Array.isArray(t)){for(r=We(e),i=t.length;a<i;a++)o[t[a]]=C.css(e,t[a],!1,r);return o}return void 0!==n?C.style(e,t,n):C.css(e,t)},e,t,1<arguments.length)}}),((C.Tween=a).prototype={constructor:a,init:function(e,t,n,r,i,o){this.elem=e,this.prop=n,this.easing=i||C.easing._default,this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=o||(C.cssNumber[n]?"":"px")},cur:function(){var e=a.propHooks[this.prop];return(e&&e.get?e:a.propHooks._default).get(this)},run:function(e){var t,n=a.propHooks[this.prop];return this.options.duration?this.pos=t=C.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):this.pos=t=e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),(n&&n.set?n:a.propHooks._default).set(this),this}}).init.prototype=a.prototype,(a.propHooks={_default:{get:function(e){return 1!==e.elem.nodeType||null!=e.elem[e.prop]&&null==e.elem.style[e.prop]?e.elem[e.prop]:(e=C.css(e.elem,e.prop,""))&&"auto"!==e?e:0},set:function(e){C.fx.step[e.prop]?C.fx.step[e.prop](e):1!==e.elem.nodeType||!C.cssHooks[e.prop]&&null==e.elem.style[ot(e.prop)]?e.elem[e.prop]=e.now:C.style(e.elem,e.prop,e.now+e.unit)}}}).scrollTop=a.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},C.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2},_default:"swing"},C.fx=a.prototype.init,C.fx.step={};var q,pt,L,dt=/^(?:toggle|show|hide)$/,ht=/queueHooks$/;function gt(){pt&&(!1===T.hidden&&w.requestAnimationFrame?w.requestAnimationFrame(gt):w.setTimeout(gt,C.fx.interval),C.fx.tick())}function yt(){return w.setTimeout(function(){q=void 0}),q=Date.now()}function mt(e,t){var n,r=0,i={height:e};for(t=t?1:0;r<4;r+=2-t)i["margin"+(n=p[r])]=i["padding"+n]=e;return t&&(i.opacity=i.width=e),i}function vt(e,t,n){for(var r,i=(H.tweeners[t]||[]).concat(H.tweeners["*"]),o=0,a=i.length;o<a;o++)if(r=i[o].call(n,t,e))return r}function H(i,e,t){var n,o,r,a,s,u,l,c=0,f=H.prefilters.length,p=C.Deferred().always(function(){delete d.elem}),d=function(){if(!o){for(var e=q||yt(),e=Math.max(0,h.startTime+h.duration-e),t=1-(e/h.duration||0),n=0,r=h.tweens.length;n<r;n++)h.tweens[n].run(t);if(p.notifyWith(i,[h,t,e]),t<1&&r)return e;r||p.notifyWith(i,[h,1,0]),p.resolveWith(i,[h])}return!1},h=p.promise({elem:i,props:C.extend({},e),opts:C.extend(!0,{specialEasing:{},easing:C.easing._default},t),originalProperties:e,originalOptions:t,startTime:q||yt(),duration:t.duration,tweens:[],createTween:function(e,t){t=C.Tween(i,h.opts,e,t,h.opts.specialEasing[e]||h.opts.easing);return h.tweens.push(t),t},stop:function(e){var t=0,n=e?h.tweens.length:0;if(!o){for(o=!0;t<n;t++)h.tweens[t].run(1);e?(p.notifyWith(i,[h,1,0]),p.resolveWith(i,[h,e])):p.rejectWith(i,[h,e])}return this}}),g=h.props,y=g,m=h.opts.specialEasing;for(r in y)if(s=m[a=x(r)],u=y[r],Array.isArray(u)&&(s=u[1],u=y[r]=u[0]),r!==a&&(y[a]=u,delete y[r]),(l=C.cssHooks[a])&&"expand"in l)for(r in u=l.expand(u),delete y[a],u)r in y||(y[r]=u[r],m[r]=s);else m[a]=s;for(;c<f;c++)if(n=H.prefilters[c].call(h,i,g,h.opts))return v(n.stop)&&(C._queueHooks(h.elem,h.opts.queue).stop=n.stop.bind(n)),n;return C.map(g,vt,h),v(h.opts.start)&&h.opts.start.call(i,h),h.progress(h.opts.progress).done(h.opts.done,h.opts.complete).fail(h.opts.fail).always(h.opts.always),C.fx.timer(C.extend(d,{elem:i,anim:h,queue:h.opts.queue})),h}C.Animation=C.extend(H,{tweeners:{"*":[function(e,t){var n=this.createTween(e,t);return xe(n.elem,e,me.exec(t),n),n}]},tweener:function(e,t){for(var n,r=0,i=(e=v(e)?(t=e,["*"]):e.match(E)).length;r<i;r++)n=e[r],H.tweeners[n]=H.tweeners[n]||[],H.tweeners[n].unshift(t)},prefilters:[function(e,t,n){var r,i,o,a,s,u,l,c="width"in t||"height"in t,f=this,p={},d=e.style,h=e.nodeType&&ye(e),g=b.get(e,"fxshow");for(r in n.queue||(null==(a=C._queueHooks(e,"fx")).unqueued&&(a.unqueued=0,s=a.empty.fire,a.empty.fire=function(){a.unqueued||s()}),a.unqueued++,f.always(function(){f.always(function(){a.unqueued--,C.queue(e,"fx").length||a.empty.fire()})})),t)if(i=t[r],dt.test(i)){if(delete t[r],o=o||"toggle"===i,i===(h?"hide":"show")){if("show"!==i||!g||void 0===g[r])continue;h=!0}p[r]=g&&g[r]||C.style(e,r)}if((u=!C.isEmptyObject(t))||!C.isEmptyObject(p))for(r in c&&1===e.nodeType&&(n.overflow=[d.overflow,d.overflowX,d.overflowY],null==(l=g&&g.display)&&(l=b.get(e,"display")),"none"===(c=C.css(e,"display"))&&(l?c=l:(A([e],!0),l=e.style.display||l,c=C.css(e,"display"),A([e]))),("inline"===c||"inline-block"===c&&null!=l)&&"none"===C.css(e,"float")&&(u||(f.done(function(){d.display=l}),null==l&&(c=d.display,l="none"===c?"":c)),d.display="inline-block")),n.overflow&&(d.overflow="hidden",f.always(function(){d.overflow=n.overflow[0],d.overflowX=n.overflow[1],d.overflowY=n.overflow[2]})),u=!1,p)u||(g?"hidden"in g&&(h=g.hidden):g=b.access(e,"fxshow",{display:l}),o&&(g.hidden=!h),h&&A([e],!0),f.done(function(){for(r in h||A([e]),b.remove(e,"fxshow"),p)C.style(e,r,p[r])})),u=vt(h?g[r]:0,r,f),r in g||(g[r]=u.start,h&&(u.end=u.start,u.start=0))}],prefilter:function(e,t){t?H.prefilters.unshift(e):H.prefilters.push(e)}}),C.speed=function(e,t,n){var r=e&&"object"==typeof e?C.extend({},e):{complete:n||!n&&t||v(e)&&e,duration:e,easing:n&&t||t&&!v(t)&&t};return C.fx.off?r.duration=0:"number"!=typeof r.duration&&(r.duration in C.fx.speeds?r.duration=C.fx.speeds[r.duration]:r.duration=C.fx.speeds._default),null!=r.queue&&!0!==r.queue||(r.queue="fx"),r.old=r.complete,r.complete=function(){v(r.old)&&r.old.call(this),r.queue&&C.dequeue(this,r.queue)},r},C.fn.extend({fadeTo:function(e,t,n,r){return this.filter(ye).css("opacity",0).show().end().animate({opacity:t},e,n,r)},animate:function(t,e,n,r){function i(){var e=H(this,C.extend({},t),a);(o||b.get(this,"finish"))&&e.stop(!0)}var o=C.isEmptyObject(t),a=C.speed(e,n,r);return i.finish=i,o||!1===a.queue?this.each(i):this.queue(a.queue,i)},stop:function(i,e,o){function a(e){var t=e.stop;delete e.stop,t(o)}return"string"!=typeof i&&(o=e,e=i,i=void 0),e&&this.queue(i||"fx",[]),this.each(function(){var e=!0,t=null!=i&&i+"queueHooks",n=C.timers,r=b.get(this);if(t)r[t]&&r[t].stop&&a(r[t]);else for(t in r)r[t]&&r[t].stop&&ht.test(t)&&a(r[t]);for(t=n.length;t--;)n[t].elem!==this||null!=i&&n[t].queue!==i||(n[t].anim.stop(o),e=!1,n.splice(t,1));!e&&o||C.dequeue(this,i)})},finish:function(a){return!1!==a&&(a=a||"fx"),this.each(function(){var e,t=b.get(this),n=t[a+"queue"],r=t[a+"queueHooks"],i=C.timers,o=n?n.length:0;for(t.finish=!0,C.queue(this,a,[]),r&&r.stop&&r.stop.call(this,!0),e=i.length;e--;)i[e].elem===this&&i[e].queue===a&&(i[e].anim.stop(!0),i.splice(e,1));for(e=0;e<o;e++)n[e]&&n[e].finish&&n[e].finish.call(this);delete t.finish})}}),C.each(["toggle","show","hide"],function(e,r){var i=C.fn[r];C.fn[r]=function(e,t,n){return null==e||"boolean"==typeof e?i.apply(this,arguments):this.animate(mt(r,!0),e,t,n)}}),C.each({slideDown:mt("show"),slideUp:mt("hide"),slideToggle:mt("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(e,r){C.fn[e]=function(e,t,n){return this.animate(r,e,t,n)}}),C.timers=[],C.fx.tick=function(){var e,t=0,n=C.timers;for(q=Date.now();t<n.length;t++)(e=n[t])()||n[t]!==e||n.splice(t--,1);n.length||C.fx.stop(),q=void 0},C.fx.timer=function(e){C.timers.push(e),C.fx.start()},C.fx.interval=13,C.fx.start=function(){pt||(pt=!0,gt())},C.fx.stop=function(){pt=null},C.fx.speeds={slow:600,fast:200,_default:400},C.fn.delay=function(r,e){return r=C.fx&&C.fx.speeds[r]||r,this.queue(e=e||"fx",function(e,t){var n=w.setTimeout(e,r);t.stop=function(){w.clearTimeout(n)}})},L=T.createElement("input"),e=T.createElement("select").appendChild(T.createElement("option")),L.type="checkbox",y.checkOn=""!==L.value,y.optSelected=e.selected,(L=T.createElement("input")).value="t",L.type="radio",y.radioValue="t"===L.value;var xt,bt=C.expr.attrHandle,wt=(C.fn.extend({attr:function(e,t){return f(this,C.attr,e,t,1<arguments.length)},removeAttr:function(e){return this.each(function(){C.removeAttr(this,e)})}}),C.extend({attr:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return"undefined"==typeof e.getAttribute?C.prop(e,t,n):(1===o&&C.isXMLDoc(e)||(i=C.attrHooks[t.toLowerCase()]||(C.expr.match.bool.test(t)?xt:void 0)),void 0!==n?null===n?void C.removeAttr(e,t):i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:(e.setAttribute(t,n+""),n):!(i&&"get"in i&&null!==(r=i.get(e,t)))&&null==(r=C.find.attr(e,t))?void 0:r)},attrHooks:{type:{set:function(e,t){var n;if(!y.radioValue&&"radio"===t&&u(e,"input"))return n=e.value,e.setAttribute("type",t),n&&(e.value=n),t}}},removeAttr:function(e,t){var n,r=0,i=t&&t.match(E);if(i&&1===e.nodeType)while(n=i[r++])e.removeAttribute(n)}}),xt={set:function(e,t,n){return!1===t?C.removeAttr(e,n):e.setAttribute(n,n),n}},C.each(C.expr.match.bool.source.match(/\w+/g),function(e,t){var a=bt[t]||C.find.attr;bt[t]=function(e,t,n){var r,i,o=t.toLowerCase();return n||(i=bt[o],bt[o]=r,r=null!=a(e,t,n)?o:null,bt[o]=i),r}}),/^(?:input|select|textarea|button)$/i),Tt=/^(?:a|area)$/i;function O(e){return(e.match(E)||[]).join(" ")}function P(e){return e.getAttribute&&e.getAttribute("class")||""}function Ct(e){return Array.isArray(e)?e:"string"==typeof e&&e.match(E)||[]}C.fn.extend({prop:function(e,t){return f(this,C.prop,e,t,1<arguments.length)},removeProp:function(e){return this.each(function(){delete this[C.propFix[e]||e]})}}),C.extend({prop:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return 1===o&&C.isXMLDoc(e)||(t=C.propFix[t]||t,i=C.propHooks[t]),void 0!==n?i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:e[t]=n:i&&"get"in i&&null!==(r=i.get(e,t))?r:e[t]},propHooks:{tabIndex:{get:function(e){var t=C.find.attr(e,"tabindex");return t?parseInt(t,10):wt.test(e.nodeName)||Tt.test(e.nodeName)&&e.href?0:-1}}},propFix:{for:"htmlFor",class:"className"}}),y.optSelected||(C.propHooks.selected={get:function(e){e=e.parentNode;return e&&e.parentNode&&e.parentNode.selectedIndex,null},set:function(e){e=e.parentNode;e&&(e.selectedIndex,e.parentNode&&e.parentNode.selectedIndex)}}),C.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){C.propFix[this.toLowerCase()]=this}),C.fn.extend({addClass:function(t){var e,n,r,i,o,a;return v(t)?this.each(function(e){C(this).addClass(t.call(this,e,P(this)))}):(e=Ct(t)).length?this.each(function(){if(r=P(this),n=1===this.nodeType&&" "+O(r)+" "){for(o=0;o<e.length;o++)i=e[o],n.indexOf(" "+i+" ")<0&&(n+=i+" ");a=O(n),r!==a&&this.setAttribute("class",a)}}):this},removeClass:function(t){var e,n,r,i,o,a;return v(t)?this.each(function(e){C(this).removeClass(t.call(this,e,P(this)))}):arguments.length?(e=Ct(t)).length?this.each(function(){if(r=P(this),n=1===this.nodeType&&" "+O(r)+" "){for(o=0;o<e.length;o++){i=e[o];while(-1<n.indexOf(" "+i+" "))n=n.replace(" "+i+" "," ")}a=O(n),r!==a&&this.setAttribute("class",a)}}):this:this.attr("class","")},toggleClass:function(t,n){var e,r,i,o,a=typeof t,s="string"==a||Array.isArray(t);return v(t)?this.each(function(e){C(this).toggleClass(t.call(this,e,P(this),n),n)}):"boolean"==typeof n&&s?n?this.addClass(t):this.removeClass(t):(e=Ct(t),this.each(function(){if(s)for(o=C(this),i=0;i<e.length;i++)r=e[i],o.hasClass(r)?o.removeClass(r):o.addClass(r);else void 0!==t&&"boolean"!=a||((r=P(this))&&b.set(this,"__className__",r),this.setAttribute&&this.setAttribute("class",!r&&!1!==t&&b.get(this,"__className__")||""))}))},hasClass:function(e){var t,n=0,r=" "+e+" ";while(t=this[n++])if(1===t.nodeType&&-1<(" "+O(P(t))+" ").indexOf(r))return!0;return!1}});function Et(e){e.stopPropagation()}var St=/\r/g,kt=(C.fn.extend({val:function(t){var n,e,r,i=this[0];return arguments.length?(r=v(t),this.each(function(e){1===this.nodeType&&(null==(e=r?t.call(this,e,C(this).val()):t)?e="":"number"==typeof e?e+="":Array.isArray(e)&&(e=C.map(e,function(e){return null==e?"":e+""})),(n=C.valHooks[this.type]||C.valHooks[this.nodeName.toLowerCase()])&&"set"in n&&void 0!==n.set(this,e,"value")||(this.value=e))})):i?(n=C.valHooks[i.type]||C.valHooks[i.nodeName.toLowerCase()])&&"get"in n&&void 0!==(e=n.get(i,"value"))?e:"string"==typeof(e=i.value)?e.replace(St,""):null==e?"":e:void 0}}),C.extend({valHooks:{option:{get:function(e){var t=C.find.attr(e,"value");return null!=t?t:O(C.text(e))}},select:{get:function(e){for(var t,n=e.options,r=e.selectedIndex,i="select-one"===e.type,o=i?null:[],a=i?r+1:n.length,s=r<0?a:i?r:0;s<a;s++)if(((t=n[s]).selected||s===r)&&!t.disabled&&(!t.parentNode.disabled||!u(t.parentNode,"optgroup"))){if(t=C(t).val(),i)return t;o.push(t)}return o},set:function(e,t){var n,r,i=e.options,o=C.makeArray(t),a=i.length;while(a--)((r=i[a]).selected=-1<C.inArray(C.valHooks.option.get(r),o))&&(n=!0);return n||(e.selectedIndex=-1),o}}}}),C.each(["radio","checkbox"],function(){C.valHooks[this]={set:function(e,t){if(Array.isArray(t))return e.checked=-1<C.inArray(C(e).val(),t)}},y.checkOn||(C.valHooks[this].get=function(e){return null===e.getAttribute("value")?"on":e.value})}),y.focusin="onfocusin"in w,/^(?:focusinfocus|focusoutblur)$/),At=(C.extend(C.event,{trigger:function(e,t,n,r){var i,o,a,s,u,l,c,f=[n||T],p=_.call(e,"type")?e.type:e,d=_.call(e,"namespace")?e.namespace.split("."):[],h=c=o=n=n||T;if(3!==n.nodeType&&8!==n.nodeType&&!kt.test(p+C.event.triggered)&&(-1<p.indexOf(".")&&(p=(d=p.split(".")).shift(),d.sort()),s=p.indexOf(":")<0&&"on"+p,(e=e[C.expando]?e:new C.Event(p,"object"==typeof e&&e)).isTrigger=r?2:3,e.namespace=d.join("."),e.rnamespace=e.namespace?new RegExp("(^|\\.)"+d.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,e.result=void 0,e.target||(e.target=n),t=null==t?[e]:C.makeArray(t,[e]),l=C.event.special[p]||{},r||!l.trigger||!1!==l.trigger.apply(n,t))){if(!r&&!l.noBubble&&!g(n)){for(a=l.delegateType||p,kt.test(a+p)||(h=h.parentNode);h;h=h.parentNode)f.push(h),o=h;o===(n.ownerDocument||T)&&f.push(o.defaultView||o.parentWindow||w)}i=0;while((h=f[i++])&&!e.isPropagationStopped())c=h,e.type=1<i?a:l.bindType||p,(u=(b.get(h,"events")||Object.create(null))[e.type]&&b.get(h,"handle"))&&u.apply(h,t),(u=s&&h[s])&&u.apply&&m(h)&&(e.result=u.apply(h,t),!1===e.result&&e.preventDefault());return e.type=p,r||e.isDefaultPrevented()||l._default&&!1!==l._default.apply(f.pop(),t)||!m(n)||s&&v(n[p])&&!g(n)&&((o=n[s])&&(n[s]=null),C.event.triggered=p,e.isPropagationStopped()&&c.addEventListener(p,Et),n[p](),e.isPropagationStopped()&&c.removeEventListener(p,Et),C.event.triggered=void 0,o&&(n[s]=o)),e.result}},simulate:function(e,t,n){n=C.extend(new C.Event,n,{type:e,isSimulated:!0});C.event.trigger(n,null,t)}}),C.fn.extend({trigger:function(e,t){return this.each(function(){C.event.trigger(e,t,this)})},triggerHandler:function(e,t){var n=this[0];if(n)return C.event.trigger(e,t,n,!0)}}),y.focusin||C.each({focus:"focusin",blur:"focusout"},function(n,r){function i(e){C.event.simulate(r,e.target,C.event.fix(e))}C.event.special[r]={setup:function(){var e=this.ownerDocument||this.document||this,t=b.access(e,r);t||e.addEventListener(n,i,!0),b.access(e,r,(t||0)+1)},teardown:function(){var e=this.ownerDocument||this.document||this,t=b.access(e,r)-1;t?b.access(e,r,t):(e.removeEventListener(n,i,!0),b.remove(e,r))}}}),w.location),Nt={guid:Date.now()},jt=/\?/,Dt=(C.parseXML=function(e){var t,n;if(!e||"string"!=typeof e)return null;try{t=(new w.DOMParser).parseFromString(e,"text/xml")}catch(e){}return n=t&&t.getElementsByTagName("parsererror")[0],t&&!n||C.error("Invalid XML: "+(n?C.map(n.childNodes,function(e){return e.textContent}).join("\n"):e)),t},/\[\]$/),qt=/\r?\n/g,Lt=/^(?:submit|button|image|reset|file)$/i,Ht=/^(?:input|select|textarea|keygen)/i;C.param=function(e,t){function n(e,t){t=v(t)?t():t,i[i.length]=encodeURIComponent(e)+"="+encodeURIComponent(null==t?"":t)}var r,i=[];if(null==e)return"";if(Array.isArray(e)||e.jquery&&!C.isPlainObject(e))C.each(e,function(){n(this.name,this.value)});else for(r in e)!function n(r,e,i,o){if(Array.isArray(e))C.each(e,function(e,t){i||Dt.test(r)?o(r,t):n(r+"["+("object"==typeof t&&null!=t?e:"")+"]",t,i,o)});else if(i||"object"!==h(e))o(r,e);else for(var t in e)n(r+"["+t+"]",e[t],i,o)}(r,e[r],t,n);return i.join("&")},C.fn.extend({serialize:function(){return C.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var e=C.prop(this,"elements");return e?C.makeArray(e):this}).filter(function(){var e=this.type;return this.name&&!C(this).is(":disabled")&&Ht.test(this.nodeName)&&!Lt.test(e)&&(this.checked||!we.test(e))}).map(function(e,t){var n=C(this).val();return null==n?null:Array.isArray(n)?C.map(n,function(e){return{name:t.name,value:e.replace(qt,"\r\n")}}):{name:t.name,value:n.replace(qt,"\r\n")}}).get()}});var Ot=/%20/g,Pt=/#.*$/,Rt=/([?&])_=[^&]*/,Mt=/^(.*?):[ \t]*([^\r\n]*)$/gm,It=/^(?:GET|HEAD)$/,Wt=/^\/\//,Ft={},$t={},Bt="*/".concat("*"),_t=T.createElement("a");function zt(o){return function(e,t){"string"!=typeof e&&(t=e,e="*");var n,r=0,i=e.toLowerCase().match(E)||[];if(v(t))while(n=i[r++])"+"===n[0]?(n=n.slice(1)||"*",(o[n]=o[n]||[]).unshift(t)):(o[n]=o[n]||[]).push(t)}}function Ut(t,r,i,o){var a={},s=t===$t;function u(e){var n;return a[e]=!0,C.each(t[e]||[],function(e,t){t=t(r,i,o);return"string"!=typeof t||s||a[t]?s?!(n=t):void 0:(r.dataTypes.unshift(t),u(t),!1)}),n}return u(r.dataTypes[0])||!a["*"]&&u("*")}function Xt(e,t){var n,r,i=C.ajaxSettings.flatOptions||{};for(n in t)void 0!==t[n]&&((i[n]?e:r=r||{})[n]=t[n]);return r&&C.extend(!0,e,r),e}_t.href=At.href,C.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:At.href,type:"GET",isLocal:/^(?:about|app|app-storage|.+-extension|file|res|widget):$/.test(At.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Bt,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":JSON.parse,"text xml":C.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(e,t){return t?Xt(Xt(e,C.ajaxSettings),t):Xt(C.ajaxSettings,e)},ajaxPrefilter:zt(Ft),ajaxTransport:zt($t),ajax:function(e,t){"object"==typeof e&&(t=e,e=void 0);var u,l,c,n,f,p,d,r,h=C.ajaxSetup({},t=t||{}),g=h.context||h,y=h.context&&(g.nodeType||g.jquery)?C(g):C.event,m=C.Deferred(),v=C.Callbacks("once memory"),x=h.statusCode||{},i={},o={},a="canceled",b={readyState:0,getResponseHeader:function(e){var t;if(p){if(!n){n={};while(t=Mt.exec(c))n[t[1].toLowerCase()+" "]=(n[t[1].toLowerCase()+" "]||[]).concat(t[2])}t=n[e.toLowerCase()+" "]}return null==t?null:t.join(", ")},getAllResponseHeaders:function(){return p?c:null},setRequestHeader:function(e,t){return null==p&&(e=o[e.toLowerCase()]=o[e.toLowerCase()]||e,i[e]=t),this},overrideMimeType:function(e){return null==p&&(h.mimeType=e),this},statusCode:function(e){if(e)if(p)b.always(e[b.status]);else for(var t in e)x[t]=[x[t],e[t]];return this},abort:function(e){e=e||a;return u&&u.abort(e),s(0,e),this}};if(m.promise(b),h.url=((e||h.url||At.href)+"").replace(Wt,At.protocol+"//"),h.type=t.method||t.type||h.method||h.type,h.dataTypes=(h.dataType||"*").toLowerCase().match(E)||[""],null==h.crossDomain){e=T.createElement("a");try{e.href=h.url,e.href=e.href,h.crossDomain=_t.protocol+"//"+_t.host!=e.protocol+"//"+e.host}catch(e){h.crossDomain=!0}}if(h.data&&h.processData&&"string"!=typeof h.data&&(h.data=C.param(h.data,h.traditional)),Ut(Ft,h,t,b),!p){for(r in(d=C.event&&h.global)&&0==C.active++&&C.event.trigger("ajaxStart"),h.type=h.type.toUpperCase(),h.hasContent=!It.test(h.type),l=h.url.replace(Pt,""),h.hasContent?h.data&&h.processData&&0===(h.contentType||"").indexOf("application/x-www-form-urlencoded")&&(h.data=h.data.replace(Ot,"+")):(e=h.url.slice(l.length),h.data&&(h.processData||"string"==typeof h.data)&&(l+=(jt.test(l)?"&":"?")+h.data,delete h.data),!1===h.cache&&(l=l.replace(Rt,"$1"),e=(jt.test(l)?"&":"?")+"_="+Nt.guid+++e),h.url=l+e),h.ifModified&&(C.lastModified[l]&&b.setRequestHeader("If-Modified-Since",C.lastModified[l]),C.etag[l]&&b.setRequestHeader("If-None-Match",C.etag[l])),(h.data&&h.hasContent&&!1!==h.contentType||t.contentType)&&b.setRequestHeader("Content-Type",h.contentType),b.setRequestHeader("Accept",h.dataTypes[0]&&h.accepts[h.dataTypes[0]]?h.accepts[h.dataTypes[0]]+("*"!==h.dataTypes[0]?", "+Bt+"; q=0.01":""):h.accepts["*"]),h.headers)b.setRequestHeader(r,h.headers[r]);if(h.beforeSend&&(!1===h.beforeSend.call(g,b,h)||p))return b.abort();if(a="abort",v.add(h.complete),b.done(h.success),b.fail(h.error),u=Ut($t,h,t,b)){if(b.readyState=1,d&&y.trigger("ajaxSend",[b,h]),p)return b;h.async&&0<h.timeout&&(f=w.setTimeout(function(){b.abort("timeout")},h.timeout));try{p=!1,u.send(i,s)}catch(e){if(p)throw e;s(-1,e)}}else s(-1,"No Transport")}return b;function s(e,t,n,r){var i,o,a,s=t;p||(p=!0,f&&w.clearTimeout(f),u=void 0,c=r||"",b.readyState=0<e?4:0,r=200<=e&&e<300||304===e,n&&(a=function(e,t,n){var r,i,o,a,s=e.contents,u=e.dataTypes;while("*"===u[0])u.shift(),void 0===r&&(r=e.mimeType||t.getResponseHeader("Content-Type"));if(r)for(i in s)if(s[i]&&s[i].test(r)){u.unshift(i);break}if(u[0]in n)o=u[0];else{for(i in n){if(!u[0]||e.converters[i+" "+u[0]]){o=i;break}a=a||i}o=o||a}if(o)return o!==u[0]&&u.unshift(o),n[o]}(h,b,n)),!r&&-1<C.inArray("script",h.dataTypes)&&C.inArray("json",h.dataTypes)<0&&(h.converters["text script"]=function(){}),a=function(e,t,n,r){var i,o,a,s,u,l={},c=e.dataTypes.slice();if(c[1])for(a in e.converters)l[a.toLowerCase()]=e.converters[a];o=c.shift();while(o)if(e.responseFields[o]&&(n[e.responseFields[o]]=t),!u&&r&&e.dataFilter&&(t=e.dataFilter(t,e.dataType)),u=o,o=c.shift())if("*"===o)o=u;else if("*"!==u&&u!==o){if(!(a=l[u+" "+o]||l["* "+o]))for(i in l)if((s=i.split(" "))[1]===o&&(a=l[u+" "+s[0]]||l["* "+s[0]])){!0===a?a=l[i]:!0!==l[i]&&(o=s[0],c.unshift(s[1]));break}if(!0!==a)if(a&&e.throws)t=a(t);else try{t=a(t)}catch(e){return{state:"parsererror",error:a?e:"No conversion from "+u+" to "+o}}}return{state:"success",data:t}}(h,a,b,r),r?(h.ifModified&&((n=b.getResponseHeader("Last-Modified"))&&(C.lastModified[l]=n),(n=b.getResponseHeader("etag"))&&(C.etag[l]=n)),204===e||"HEAD"===h.type?s="nocontent":304===e?s="notmodified":(s=a.state,i=a.data,r=!(o=a.error))):(o=s,!e&&s||(s="error",e<0&&(e=0))),b.status=e,b.statusText=(t||s)+"",r?m.resolveWith(g,[i,s,b]):m.rejectWith(g,[b,s,o]),b.statusCode(x),x=void 0,d&&y.trigger(r?"ajaxSuccess":"ajaxError",[b,h,r?i:o]),v.fireWith(g,[b,s]),d&&(y.trigger("ajaxComplete",[b,h]),--C.active||C.event.trigger("ajaxStop")))}},getJSON:function(e,t,n){return C.get(e,t,n,"json")},getScript:function(e,t){return C.get(e,void 0,t,"script")}}),C.each(["get","post"],function(e,i){C[i]=function(e,t,n,r){return v(t)&&(r=r||n,n=t,t=void 0),C.ajax(C.extend({url:e,type:i,dataType:r,data:t,success:n},C.isPlainObject(e)&&e))}}),C.ajaxPrefilter(function(e){for(var t in e.headers)"content-type"===t.toLowerCase()&&(e.contentType=e.headers[t]||"")}),C._evalUrl=function(e,t,n){return C.ajax({url:e,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,converters:{"text script":function(){}},dataFilter:function(e){C.globalEval(e,t,n)}})},C.fn.extend({wrapAll:function(e){return this[0]&&(v(e)&&(e=e.call(this[0])),e=C(e,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&e.insertBefore(this[0]),e.map(function(){var e=this;while(e.firstElementChild)e=e.firstElementChild;return e}).append(this)),this},wrapInner:function(n){return v(n)?this.each(function(e){C(this).wrapInner(n.call(this,e))}):this.each(function(){var e=C(this),t=e.contents();t.length?t.wrapAll(n):e.append(n)})},wrap:function(t){var n=v(t);return this.each(function(e){C(this).wrapAll(n?t.call(this,e):t)})},unwrap:function(e){return this.parent(e).not("body").each(function(){C(this).replaceWith(this.childNodes)}),this}}),C.expr.pseudos.hidden=function(e){return!C.expr.pseudos.visible(e)},C.expr.pseudos.visible=function(e){return!!(e.offsetWidth||e.offsetHeight||e.getClientRects().length)},C.ajaxSettings.xhr=function(){try{return new w.XMLHttpRequest}catch(e){}};var Vt={0:200,1223:204},Gt=C.ajaxSettings.xhr(),Yt=(y.cors=!!Gt&&"withCredentials"in Gt,y.ajax=Gt=!!Gt,C.ajaxTransport(function(i){var o,a;if(y.cors||Gt&&!i.crossDomain)return{send:function(e,t){var n,r=i.xhr();if(r.open(i.type,i.url,i.async,i.username,i.password),i.xhrFields)for(n in i.xhrFields)r[n]=i.xhrFields[n];for(n in i.mimeType&&r.overrideMimeType&&r.overrideMimeType(i.mimeType),i.crossDomain||e["X-Requested-With"]||(e["X-Requested-With"]="XMLHttpRequest"),e)r.setRequestHeader(n,e[n]);o=function(e){return function(){o&&(o=a=r.onload=r.onerror=r.onabort=r.ontimeout=r.onreadystatechange=null,"abort"===e?r.abort():"error"===e?"number"!=typeof r.status?t(0,"error"):t(r.status,r.statusText):t(Vt[r.status]||r.status,r.statusText,"text"!==(r.responseType||"text")||"string"!=typeof r.responseText?{binary:r.response}:{text:r.responseText},r.getAllResponseHeaders()))}},r.onload=o(),a=r.onerror=r.ontimeout=o("error"),void 0!==r.onabort?r.onabort=a:r.onreadystatechange=function(){4===r.readyState&&w.setTimeout(function(){o&&a()})},o=o("abort");try{r.send(i.hasContent&&i.data||null)}catch(e){if(o)throw e}},abort:function(){o&&o()}}}),C.ajaxPrefilter(function(e){e.crossDomain&&(e.contents.script=!1)}),C.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(e){return C.globalEval(e),e}}}),C.ajaxPrefilter("script",function(e){void 0===e.cache&&(e.cache=!1),e.crossDomain&&(e.type="GET")}),C.ajaxTransport("script",function(n){var r,i;if(n.crossDomain||n.scriptAttrs)return{send:function(e,t){r=C("<script>").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),T.head.appendChild(r[0])},abort:function(){i&&i()}}}),[]),Qt=/(=)\?(?=&|$)|\?\?/,Jt=(C.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Yt.pop()||C.expando+"_"+Nt.guid++;return this[e]=!0,e}}),C.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Qt.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Qt.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=v(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Qt,"$1"+r):!1!==e.jsonp&&(e.url+=(jt.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||C.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=w[r],w[r]=function(){o=arguments},n.always(function(){void 0===i?C(w).removeProp(r):w[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,Yt.push(r)),o&&v(i)&&i(o[0]),o=i=void 0}),"script"}),y.createHTMLDocument=((o=T.implementation.createHTMLDocument("").body).innerHTML="<form></form><form></form>",2===o.childNodes.length),C.parseHTML=function(e,t,n){var r;return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(y.createHTMLDocument?((r=(t=T.implementation.createHTMLDocument("")).createElement("base")).href=T.location.href,t.head.appendChild(r)):t=T),r=!n&&[],(n=J.exec(e))?[t.createElement(n[1])]:(n=ke([e],t,r),r&&r.length&&C(r).remove(),C.merge([],n.childNodes)))},C.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1<s&&(r=O(e.slice(s)),e=e.slice(0,s)),v(t)?(n=t,t=void 0):t&&"object"==typeof t&&(i="POST"),0<a.length&&C.ajax({url:e,type:i||"GET",dataType:"html",data:t}).done(function(e){o=arguments,a.html(r?C("<div>").append(C.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},C.expr.pseudos.animated=function(t){return C.grep(C.timers,function(e){return t===e.elem}).length},C.offset={setOffset:function(e,t,n){var r,i,o,a,s=C.css(e,"position"),u=C(e),l={};"static"===s&&(e.style.position="relative"),o=u.offset(),r=C.css(e,"top"),a=C.css(e,"left"),s=("absolute"===s||"fixed"===s)&&-1<(r+a).indexOf("auto")?(i=(s=u.position()).top,s.left):(i=parseFloat(r)||0,parseFloat(a)||0),null!=(t=v(t)?t.call(e,n,C.extend({},o)):t).top&&(l.top=t.top-o.top+i),null!=t.left&&(l.left=t.left-o.left+s),"using"in t?t.using.call(e,l):u.css(l)}},C.fn.extend({offset:function(t){var e,n;return arguments.length?void 0===t?this:this.each(function(e){C.offset.setOffset(this,t,e)}):(n=this[0])?n.getClientRects().length?(e=n.getBoundingClientRect(),n=n.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===C.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===C.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=C(e).offset()).top+=C.css(e,"borderTopWidth",!0),i.left+=C.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-C.css(r,"marginTop",!0),left:t.left-i.left-C.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===C.css(e,"position"))e=e.offsetParent;return e||S})}}),C.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;C.fn[t]=function(e){return f(this,function(e,t,n){var r;if(g(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),C.each(["top","left"],function(e,n){C.cssHooks[n]=tt(y.pixelPosition,function(e,t){if(t)return t=et(e,n),Ge.test(t)?C(e).position()[n]+"px":t})}),C.each({Height:"height",Width:"width"},function(a,s){C.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){C.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return f(this,function(e,t,n){var r;return g(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?C.css(e,t,i):C.style(e,t,n,i)},s,n?e:void 0,n)}})}),C.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){C.fn[t]=function(e){return this.on(t,e)}}),C.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),C.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){C.fn[n]=function(e,t){return 0<arguments.length?this.on(n,null,e,t):this.trigger(n)}}),/^[\s\uFEFF\xA0]+|([^\s\uFEFF\xA0])[\s\uFEFF\xA0]+$/g),Kt=(C.proxy=function(e,t){var n,r;if("string"==typeof t&&(r=e[t],t=e,e=r),v(e))return n=s.call(arguments,2),(r=function(){return e.apply(t||this,n.concat(s.call(arguments)))}).guid=e.guid=e.guid||C.guid++,r},C.holdReady=function(e){e?C.readyWait++:C.ready(!0)},C.isArray=Array.isArray,C.parseJSON=JSON.parse,C.nodeName=u,C.isFunction=v,C.isWindow=g,C.camelCase=x,C.type=h,C.now=Date.now,C.isNumeric=function(e){var t=C.type(e);return("number"===t||"string"===t)&&!isNaN(e-parseFloat(e))},C.trim=function(e){return null==e?"":(e+"").replace(Jt,"$1")},"function"==typeof define&&define.amd&&define("jquery",[],function(){return C}),w.jQuery),Zt=w.$;return C.noConflict=function(e){return w.$===C&&(w.$=Zt),e&&w.jQuery===C&&(w.jQuery=Kt),C},"undefined"==typeof R&&(w.jQuery=w.$=C),C});
+</script>
+<meta name="viewport" content="width=device-width, initial-scale=1" />
+<style type="text/css">  html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,hgroup,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block;vertical-align:baseline}audio:not([controls]){display:none;height:0}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}abbr[title]{border-bottom:none;text-decoration:underline;text-decoration:underline dotted}b,strong{font-weight:bold}dfn{font-style:italic}h1{font-size:2em;margin:.67em 0}mark{background:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-0.5em}sub{bottom:-0.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:1em 40px}hr{box-sizing:content-box;height:0}pre{overflow:auto}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}button,input,optgroup,select,textarea{color:inherit;font:inherit;margin:0}button{overflow:visible}button,select{text-transform:none}button,html input[type="button"],input[type="reset"],input[type="submit"]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}input{line-height:normal}input[type="checkbox"],input[type="radio"]{box-sizing:border-box;padding:0}input[type="number"]::-webkit-inner-spin-button,input[type="number"]::-webkit-outer-spin-button{height:auto}input[type="search"]{-webkit-appearance:textfield;box-sizing:content-box}input[type="search"]::-webkit-search-cancel-button,input[type="search"]::-webkit-search-decoration{-webkit-appearance:none}fieldset{border:1px solid #c0c0c0;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}textarea{overflow:auto}optgroup{font-weight:bold}table{border-collapse:collapse;border-spacing:0}td,th{padding:0} @media print{*,*:before,*:after{color:#000!important;text-shadow:none!important;background:transparent!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:"(" attr(href) ")"}abbr[title]:after{content:"(" attr(title) ")"}a[href^="#"]:after,a[href^="javascript:"]:after{content:""}pre,blockquote{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100%!important}p,h2,h3{orphans:3;widows:3}h2,h3{page-break-after:avoid}.navbar{display:none}.btn>.caret,.dropup>.btn>.caret{border-top-color:#000!important}.label{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered th,.table-bordered td{border:1px solid #ddd!important}}@font-face{font-family:"Glyphicons Halflings";src:url(data:application/vnd.ms-fontobject;base64,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);src:url(data:application/vnd.ms-fontobject;base64,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) format("embedded-opentype"),url(data:font/woff2;base64,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) format("woff2"),url(data:application/font-woff;base64,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) format("woff"),url(data:application/font-sfnt;base64,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JbwmFCZ+JuYnUCe8J/goNChwKKwpoCnMKiYqSiqEKworeiwILGgsuizsLRwtiC30LiguZi6iLtgvDi9GL34vsi/4MD4whDDSMRIxYDGuMegyJDJeMpoy3jMiMz4zaDO2NBg0YDSoNNI1LDWeNeg2PjZ8Ntw3GjdON5I31DgQOEI4hjjIOQo5SjmIOcw6HDpsOpo63jugO9w8GDxQPKI8+D0yPew+Oj6MPtQ/KD9uP6o/+kBIQIBAxkECQX5CGEKoQu5DGENCQ3ZDoEPKRBBEYESuRPZFWkW2RgZGdEa0RvZHNkd2R7ZH9kgWSDJITkhqSIZIzEkSSThJXkmESapKAkouSlIAAQAAARcApwARAAAAAAACAAAAAQABAAAAQAAuAAAAAAAAABAAxgABAAAAAAATABIAAAADAAEECQAAAGoAEgADAAEECQABACgAfAADAAEECQACAA4ApAADAAEECQADAEwAsgADAAEECQAEADgA/gADAAEECQAFAHgBNgADAAEECQAGADYBrgADAAEECQAIABYB5AADAAEECQAJABYB+gADAAEECQALACQCEAADAAEECQAMACQCNAADAAEECQATACQCWAADAAEECQDIABYCfAADAAEECQDJADACkgADAAEECdkDABoCwnd3dy5nbHlwaGljb25zLmNvbQBDAG8AcAB5AHIAaQBnAGgAdAAgAKkAIAAyADAAMQA0ACAAYgB5ACAASgBhAG4AIABLAG8AdgBhAHIAaQBrAC4AIABBAGwAbAAgAHIAaQBnAGgAdABzACAAcgBlAHMAZQByAHYAZQBkAC4ARwBMAFkAUABIAEkAQwBPAE4AUwAgAEgAYQBsAGYAbABpAG4AZwBzAFIAZQBnAHUAbABhAHIAMQAuADAAMAA5ADsAVQBLAFcATgA7AEcATABZAFAASABJAEMATwBOAFMASABhAGwAZgBsAGkAbgBnAHMALQBSAGUAZwB1AGwAYQByAEcATABZAFAASABJAEMATwBOAFMAIABIAGEAbABmAGwAaQBuAGcAcwAgAFIAZQBnAHUAbABhAHIAVgBlAHIAcwBpAG8AbgAgADEALgAwADAAOQA7AFAAUwAgADAAMAAxAC4AMAAwADkAOwBoAG8AdABjAG8AbgB2ACAAMQAuADAALgA3ADAAOwBtAGEAawBlAG8AdABmAC4AbABpAGIAMgAuADUALgA1ADgAMwAyADkARwBMAFkAUABIAEkAQwBPAE4AUwBIAGEAbABmAGwAaQBuAGcAcwAtAFIAZQBnAHUAbABhAHIASgBhAG4AIABLAG8AdgBhAHIAaQBrAEoAYQBuACAASwBvAHYAYQByAGkAawB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQB3AHcAdwAuAGcAbAB5AHAAaABpAGMAbwBuAHMALgBjAG8AbQBXAGUAYgBmAG8AbgB0ACAAMQAuADAAVwBlAGQAIABPAGMAdAAgADIAOQAgADAANgA6ADMANgA6ADAANwAgADIAMAAxADQARgBvAG4AdAAgAFMAcQB1AGkAcgByAGUAbAAAAAIAAAAAAAD/tQAyAAAAAAAAAAAAAAAAAAAAAAAAAAABFwAAAQIBAwADAA0ADgEEAJYBBQEGAQcBCAEJAQoBCwEMAQ0BDgEPARABEQESARMA7wEUARUBFgEXARgBGQEaARsBHAEdAR4BHwEgASEBIgEjASQBJQEmAScBKAEpASoBKwEsAS0BLgEvATABMQEyATMBNAE1ATYBNwE4ATkBOgE7ATwBPQE+AT8BQAFBAUIBQwFEAUUBRgFHAUgBSQFKAUsBTAFNAU4BTwFQAVEBUgFTAVQBVQFWAVcBWAFZAVoBWwFcAV0BXgFfAWABYQFiAWMBZAFlAWYBZwFoAWkBagFrAWwBbQFuAW8BcAFxAXIBcwF0AXUBdgF3AXgBeQF6AXsBfAF9AX4BfwGAAYEBggGDAYQBhQGGAYcBiAGJAYoBiwGMAY0BjgGPAZABkQGSAZMBlAGVAZYBlwGYAZkBmgGbAZwBnQGeAZ8BoAGhAaIBowGkAaUBpgGnAagBqQGqAasBrAGtAa4BrwGwAbEBsgGzAbQBtQG2AbcBuAG5AboBuwG8Ab0BvgG/AcABwQHCAcMBxAHFAcYBxwHIAckBygHLAcwBzQHOAc8B0AHRAdIB0wHUAdUB1gHXAdgB2QHaAdsB3AHdAd4B3wHgAeEB4gHjAeQB5QHmAecB6AHpAeoB6wHsAe0B7gHvAfAB8QHyAfMB9AH1AfYB9wH4AfkB+gH7AfwB/QH+Af8CAAIBAgICAwIEAgUCBgIHAggCCQIKAgsCDAINAg4CDwIQAhECEgZnbHlwaDEGZ2x5cGgyB3VuaTAwQTAHdW5pMjAwMAd1bmkyMDAxB3VuaTIwMDIHdW5pMjAwMwd1bmkyMDA0B3VuaTIwMDUHdW5pMjAwNgd1bmkyMDA3B3VuaTIwMDgHdW5pMjAwOQd1bmkyMDBBB3VuaTIwMkYHdW5pMjA1RgRFdXJvB3VuaTIwQkQHdW5pMjMxQgd1bmkyNUZDB3VuaTI2MDEHdW5pMjZGQQd1bmkyNzA5B3VuaTI3MEYHdW5pRTAwMQd1bmlFMDAyB3VuaUUwMDMHdW5pRTAwNQd1bmlFMDA2B3VuaUUwMDcHdW5pRTAwOAd1bmlFMDA5B3VuaUUwMTAHdW5pRTAxMQd1bmlFMDEyB3VuaUUwMTMHdW5pRTAxNAd1bmlFMDE1B3VuaUUwMTYHdW5pRTAxNwd1bmlFMDE4B3VuaUUwMTkHdW5pRTAyMAd1bmlFMDIxB3VuaUUwMjIHdW5pRTAyMwd1bmlFMDI0B3VuaUUwMjUHdW5pRTAyNgd1bmlFMDI3B3VuaUUwMjgHdW5pRTAyOQd1bmlFMDMwB3VuaUUwMzEHdW5pRTAzMgd1bmlFMDMzB3VuaUUwMzQHdW5pRTAzNQd1bmlFMDM2B3VuaUUwMzcHdW5pRTAzOAd1bmlFMDM5B3VuaUUwNDAHdW5pRTA0MQd1bmlFMDQyB3VuaUUwNDMHdW5pRTA0NAd1bmlFMDQ1B3VuaUUwNDYHdW5pRTA0Nwd1bmlFMDQ4B3VuaUUwNDkHdW5pRTA1MAd1bmlFMDUxB3VuaUUwNTIHdW5pRTA1Mwd1bmlFMDU0B3VuaUUwNTUHdW5pRTA1Ngd1bmlFMDU3B3VuaUUwNTgHdW5pRTA1OQd1bmlFMDYwB3VuaUUwNjIHdW5pRTA2Mwd1bmlFMDY0B3VuaUUwNjUHdW5pRTA2Ngd1bmlFMDY3B3VuaUUwNjgHdW5pRTA2OQd1bmlFMDcwB3VuaUUwNzEHdW5pRTA3Mgd1bmlFMDczB3VuaUUwNzQHdW5pRTA3NQd1bmlFMDc2B3VuaUUwNzcHdW5pRTA3OAd1bmlFMDc5B3VuaUUwODAHdW5pRTA4MQd1bmlFMDgyB3VuaUUwODMHdW5pRTA4NAd1bmlFMDg1B3VuaUUwODYHdW5pRTA4Nwd1bmlFMDg4B3VuaUUwODkHdW5pRTA5MAd1bmlFMDkxB3VuaUUwOTIHdW5pRTA5Mwd1bmlFMDk0B3VuaUUwOTUHdW5pRTA5Ngd1bmlFMDk3B3VuaUUxMDEHdW5pRTEwMgd1bmlFMTAzB3VuaUUxMDQHdW5pRTEwNQd1bmlFMTA2B3VuaUUxMDcHdW5pRTEwOAd1bmlFMTA5B3VuaUUxMTAHdW5pRTExMQd1bmlFMTEyB3VuaUUxMTMHdW5pRTExNAd1bmlFMTE1B3VuaUUxMTYHdW5pRTExNwd1bmlFMTE4B3VuaUUxMTkHdW5pRTEyMAd1bmlFMTIxB3VuaUUxMjIHdW5pRTEyMwd1bmlFMTI0B3VuaUUxMjUHdW5pRTEyNgd1bmlFMTI3B3VuaUUxMjgHdW5pRTEyOQd1bmlFMTMwB3VuaUUxMzEHdW5pRTEzMgd1bmlFMTMzB3VuaUUxMzQHdW5pRTEzNQd1bmlFMTM2B3VuaUUxMzcHdW5pRTEzOAd1bmlFMTM5B3VuaUUxNDAHdW5pRTE0MQd1bmlFMTQyB3VuaUUxNDMHdW5pRTE0NAd1bmlFMTQ1B3VuaUUxNDYHdW5pRTE0OAd1bmlFMTQ5B3VuaUUxNTAHdW5pRTE1MQd1bmlFMTUyB3VuaUUxNTMHdW5pRTE1NAd1bmlFMTU1B3VuaUUxNTYHdW5pRTE1Nwd1bmlFMTU4B3VuaUUxNTkHdW5pRTE2MAd1bmlFMTYxB3VuaUUxNjIHdW5pRTE2Mwd1bmlFMTY0B3VuaUUxNjUHdW5pRTE2Ngd1bmlFMTY3B3VuaUUxNjgHdW5pRTE2OQd1bmlFMTcwB3VuaUUxNzEHdW5pRTE3Mgd1bmlFMTczB3VuaUUxNzQHdW5pRTE3NQd1bmlFMTc2B3VuaUUxNzcHdW5pRTE3OAd1bmlFMTc5B3VuaUUxODAHdW5pRTE4MQd1bmlFMTgyB3VuaUUxODMHdW5pRTE4NAd1bmlFMTg1B3VuaUUxODYHdW5pRTE4Nwd1bmlFMTg4B3VuaUUxODkHdW5pRTE5MAd1bmlFMTkxB3VuaUUxOTIHdW5pRTE5Mwd1bmlFMTk0B3VuaUUxOTUHdW5pRTE5Nwd1bmlFMTk4B3VuaUUxOTkHdW5pRTIwMAd1bmlFMjAxB3VuaUUyMDIHdW5pRTIwMwd1bmlFMjA0B3VuaUUyMDUHdW5pRTIwNgd1bmlFMjA5B3VuaUUyMTAHdW5pRTIxMQd1bmlFMjEyB3VuaUUyMTMHdW5pRTIxNAd1bmlFMjE1B3VuaUUyMTYHdW5pRTIxOAd1bmlFMjE5B3VuaUUyMjEHdW5pRTIyMwd1bmlFMjI0B3VuaUUyMjUHdW5pRTIyNgd1bmlFMjI3B3VuaUUyMzAHdW5pRTIzMQd1bmlFMjMyB3VuaUUyMzMHdW5pRTIzNAd1bmlFMjM1B3VuaUUyMzYHdW5pRTIzNwd1bmlFMjM4B3VuaUUyMzkHdW5pRTI0MAd1bmlFMjQxB3VuaUUyNDIHdW5pRTI0Mwd1bmlFMjQ0B3VuaUUyNDUHdW5pRTI0Ngd1bmlFMjQ3B3VuaUUyNDgHdW5pRTI0OQd1bmlFMjUwB3VuaUUyNTEHdW5pRTI1Mgd1bmlFMjUzB3VuaUUyNTQHdW5pRTI1NQd1bmlFMjU2B3VuaUUyNTcHdW5pRTI1OAd1bmlFMjU5B3VuaUUyNjAHdW5pRjhGRgZ1MUY1MTEGdTFGNkFBAAAAAAFUUMMXAAA=) format("truetype"),url() format("svg")}.glyphicon{position:relative;top:1px;display:inline-block;font-family:"Glyphicons Halflings";font-style:normal;font-weight:400;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.glyphicon-asterisk:before{content:"\002a"}.glyphicon-plus:before{content:"\002b"}.glyphicon-euro:before,.glyphicon-eur:before{content:"\20ac"}.glyphicon-minus:before{content:"\2212"}.glyphicon-cloud:before{content:"\2601"}.glyphicon-envelope:before{content:"\2709"}.glyphicon-pencil:before{content:"\270f"}.glyphicon-glass:before{content:"\e001"}.glyphicon-music:before{content:"\e002"}.glyphicon-search:before{content:"\e003"}.glyphicon-heart:before{content:"\e005"}.glyphicon-star:before{content:"\e006"}.glyphicon-star-empty:before{content:"\e007"}.glyphicon-user:before{content:"\e008"}.glyphicon-film:before{content:"\e009"}.glyphicon-th-large:before{content:"\e010"}.glyphicon-th:before{content:"\e011"}.glyphicon-th-list:before{content:"\e012"}.glyphicon-ok:before{content:"\e013"}.glyphicon-remove:before{content:"\e014"}.glyphicon-zoom-in:before{content:"\e015"}.glyphicon-zoom-out:before{content:"\e016"}.glyphicon-off:before{content:"\e017"}.glyphicon-signal:before{content:"\e018"}.glyphicon-cog:before{content:"\e019"}.glyphicon-trash:before{content:"\e020"}.glyphicon-home:before{content:"\e021"}.glyphicon-file:before{content:"\e022"}.glyphicon-time:before{content:"\e023"}.glyphicon-road:before{content:"\e024"}.glyphicon-download-alt:before{content:"\e025"}.glyphicon-download:before{content:"\e026"}.glyphicon-upload:before{content:"\e027"}.glyphicon-inbox:before{content:"\e028"}.glyphicon-play-circle:before{content:"\e029"}.glyphicon-repeat:before{content:"\e030"}.glyphicon-refresh:before{content:"\e031"}.glyphicon-list-alt:before{content:"\e032"}.glyphicon-lock:before{content:"\e033"}.glyphicon-flag:before{content:"\e034"}.glyphicon-headphones:before{content:"\e035"}.glyphicon-volume-off:before{content:"\e036"}.glyphicon-volume-down:before{content:"\e037"}.glyphicon-volume-up:before{content:"\e038"}.glyphicon-qrcode:before{content:"\e039"}.glyphicon-barcode:before{content:"\e040"}.glyphicon-tag:before{content:"\e041"}.glyphicon-tags:before{content:"\e042"}.glyphicon-book:before{content:"\e043"}.glyphicon-bookmark:before{content:"\e044"}.glyphicon-print:before{content:"\e045"}.glyphicon-camera:before{content:"\e046"}.glyphicon-font:before{content:"\e047"}.glyphicon-bold:before{content:"\e048"}.glyphicon-italic:before{content:"\e049"}.glyphicon-text-height:before{content:"\e050"}.glyphicon-text-width:before{content:"\e051"}.glyphicon-align-left:before{content:"\e052"}.glyphicon-align-center:before{content:"\e053"}.glyphicon-align-right:before{content:"\e054"}.glyphicon-align-justify:before{content:"\e055"}.glyphicon-list:before{content:"\e056"}.glyphicon-indent-left:before{content:"\e057"}.glyphicon-indent-right:before{content:"\e058"}.glyphicon-facetime-video:before{content:"\e059"}.glyphicon-picture:before{content:"\e060"}.glyphicon-map-marker:before{content:"\e062"}.glyphicon-adjust:before{content:"\e063"}.glyphicon-tint:before{content:"\e064"}.glyphicon-edit:before{content:"\e065"}.glyphicon-share:before{content:"\e066"}.glyphicon-check:before{content:"\e067"}.glyphicon-move:before{content:"\e068"}.glyphicon-step-backward:before{content:"\e069"}.glyphicon-fast-backward:before{content:"\e070"}.glyphicon-backward:before{content:"\e071"}.glyphicon-play:before{content:"\e072"}.glyphicon-pause:before{content:"\e073"}.glyphicon-stop:before{content:"\e074"}.glyphicon-forward:before{content:"\e075"}.glyphicon-fast-forward:before{content:"\e076"}.glyphicon-step-forward:before{content:"\e077"}.glyphicon-eject:before{content:"\e078"}.glyphicon-chevron-left:before{content:"\e079"}.glyphicon-chevron-right:before{content:"\e080"}.glyphicon-plus-sign:before{content:"\e081"}.glyphicon-minus-sign:before{content:"\e082"}.glyphicon-remove-sign:before{content:"\e083"}.glyphicon-ok-sign:before{content:"\e084"}.glyphicon-question-sign:before{content:"\e085"}.glyphicon-info-sign:before{content:"\e086"}.glyphicon-screenshot:before{content:"\e087"}.glyphicon-remove-circle:before{content:"\e088"}.glyphicon-ok-circle:before{content:"\e089"}.glyphicon-ban-circle:before{content:"\e090"}.glyphicon-arrow-left:before{content:"\e091"}.glyphicon-arrow-right:before{content:"\e092"}.glyphicon-arrow-up:before{content:"\e093"}.glyphicon-arrow-down:before{content:"\e094"}.glyphicon-share-alt:before{content:"\e095"}.glyphicon-resize-full:before{content:"\e096"}.glyphicon-resize-small:before{content:"\e097"}.glyphicon-exclamation-sign:before{content:"\e101"}.glyphicon-gift:before{content:"\e102"}.glyphicon-leaf:before{content:"\e103"}.glyphicon-fire:before{content:"\e104"}.glyphicon-eye-open:before{content:"\e105"}.glyphicon-eye-close:before{content:"\e106"}.glyphicon-warning-sign:before{content:"\e107"}.glyphicon-plane:before{content:"\e108"}.glyphicon-calendar:before{content:"\e109"}.glyphicon-random:before{content:"\e110"}.glyphicon-comment:before{content:"\e111"}.glyphicon-magnet:before{content:"\e112"}.glyphicon-chevron-up:before{content:"\e113"}.glyphicon-chevron-down:before{content:"\e114"}.glyphicon-retweet:before{content:"\e115"}.glyphicon-shopping-cart:before{content:"\e116"}.glyphicon-folder-close:before{content:"\e117"}.glyphicon-folder-open:before{content:"\e118"}.glyphicon-resize-vertical:before{content:"\e119"}.glyphicon-resize-horizontal:before{content:"\e120"}.glyphicon-hdd:before{content:"\e121"}.glyphicon-bullhorn:before{content:"\e122"}.glyphicon-bell:before{content:"\e123"}.glyphicon-certificate:before{content:"\e124"}.glyphicon-thumbs-up:before{content:"\e125"}.glyphicon-thumbs-down:before{content:"\e126"}.glyphicon-hand-right:before{content:"\e127"}.glyphicon-hand-left:before{content:"\e128"}.glyphicon-hand-up:before{content:"\e129"}.glyphicon-hand-down:before{content:"\e130"}.glyphicon-circle-arrow-right:before{content:"\e131"}.glyphicon-circle-arrow-left:before{content:"\e132"}.glyphicon-circle-arrow-up:before{content:"\e133"}.glyphicon-circle-arrow-down:before{content:"\e134"}.glyphicon-globe:before{content:"\e135"}.glyphicon-wrench:before{content:"\e136"}.glyphicon-tasks:before{content:"\e137"}.glyphicon-filter:before{content:"\e138"}.glyphicon-briefcase:before{content:"\e139"}.glyphicon-fullscreen:before{content:"\e140"}.glyphicon-dashboard:before{content:"\e141"}.glyphicon-paperclip:before{content:"\e142"}.glyphicon-heart-empty:before{content:"\e143"}.glyphicon-link:before{content:"\e144"}.glyphicon-phone:before{content:"\e145"}.glyphicon-pushpin:before{content:"\e146"}.glyphicon-usd:before{content:"\e148"}.glyphicon-gbp:before{content:"\e149"}.glyphicon-sort:before{content:"\e150"}.glyphicon-sort-by-alphabet:before{content:"\e151"}.glyphicon-sort-by-alphabet-alt:before{content:"\e152"}.glyphicon-sort-by-order:before{content:"\e153"}.glyphicon-sort-by-order-alt:before{content:"\e154"}.glyphicon-sort-by-attributes:before{content:"\e155"}.glyphicon-sort-by-attributes-alt:before{content:"\e156"}.glyphicon-unchecked:before{content:"\e157"}.glyphicon-expand:before{content:"\e158"}.glyphicon-collapse-down:before{content:"\e159"}.glyphicon-collapse-up:before{content:"\e160"}.glyphicon-log-in:before{content:"\e161"}.glyphicon-flash:before{content:"\e162"}.glyphicon-log-out:before{content:"\e163"}.glyphicon-new-window:before{content:"\e164"}.glyphicon-record:before{content:"\e165"}.glyphicon-save:before{content:"\e166"}.glyphicon-open:before{content:"\e167"}.glyphicon-saved:before{content:"\e168"}.glyphicon-import:before{content:"\e169"}.glyphicon-export:before{content:"\e170"}.glyphicon-send:before{content:"\e171"}.glyphicon-floppy-disk:before{content:"\e172"}.glyphicon-floppy-saved:before{content:"\e173"}.glyphicon-floppy-remove:before{content:"\e174"}.glyphicon-floppy-save:before{content:"\e175"}.glyphicon-floppy-open:before{content:"\e176"}.glyphicon-credit-card:before{content:"\e177"}.glyphicon-transfer:before{content:"\e178"}.glyphicon-cutlery:before{content:"\e179"}.glyphicon-header:before{content:"\e180"}.glyphicon-compressed:before{content:"\e181"}.glyphicon-earphone:before{content:"\e182"}.glyphicon-phone-alt:before{content:"\e183"}.glyphicon-tower:before{content:"\e184"}.glyphicon-stats:before{content:"\e185"}.glyphicon-sd-video:before{content:"\e186"}.glyphicon-hd-video:before{content:"\e187"}.glyphicon-subtitles:before{content:"\e188"}.glyphicon-sound-stereo:before{content:"\e189"}.glyphicon-sound-dolby:before{content:"\e190"}.glyphicon-sound-5-1:before{content:"\e191"}.glyphicon-sound-6-1:before{content:"\e192"}.glyphicon-sound-7-1:before{content:"\e193"}.glyphicon-copyright-mark:before{content:"\e194"}.glyphicon-registration-mark:before{content:"\e195"}.glyphicon-cloud-download:before{content:"\e197"}.glyphicon-cloud-upload:before{content:"\e198"}.glyphicon-tree-conifer:before{content:"\e199"}.glyphicon-tree-deciduous:before{content:"\e200"}.glyphicon-cd:before{content:"\e201"}.glyphicon-save-file:before{content:"\e202"}.glyphicon-open-file:before{content:"\e203"}.glyphicon-level-up:before{content:"\e204"}.glyphicon-copy:before{content:"\e205"}.glyphicon-paste:before{content:"\e206"}.glyphicon-alert:before{content:"\e209"}.glyphicon-equalizer:before{content:"\e210"}.glyphicon-king:before{content:"\e211"}.glyphicon-queen:before{content:"\e212"}.glyphicon-pawn:before{content:"\e213"}.glyphicon-bishop:before{content:"\e214"}.glyphicon-knight:before{content:"\e215"}.glyphicon-baby-formula:before{content:"\e216"}.glyphicon-tent:before{content:"\26fa"}.glyphicon-blackboard:before{content:"\e218"}.glyphicon-bed:before{content:"\e219"}.glyphicon-apple:before{content:"\f8ff"}.glyphicon-erase:before{content:"\e221"}.glyphicon-hourglass:before{content:"\231b"}.glyphicon-lamp:before{content:"\e223"}.glyphicon-duplicate:before{content:"\e224"}.glyphicon-piggy-bank:before{content:"\e225"}.glyphicon-scissors:before{content:"\e226"}.glyphicon-bitcoin:before{content:"\e227"}.glyphicon-btc:before{content:"\e227"}.glyphicon-xbt:before{content:"\e227"}.glyphicon-yen:before{content:"\00a5"}.glyphicon-jpy:before{content:"\00a5"}.glyphicon-ruble:before{content:"\20bd"}.glyphicon-rub:before{content:"\20bd"}.glyphicon-scale:before{content:"\e230"}.glyphicon-ice-lolly:before{content:"\e231"}.glyphicon-ice-lolly-tasted:before{content:"\e232"}.glyphicon-education:before{content:"\e233"}.glyphicon-option-horizontal:before{content:"\e234"}.glyphicon-option-vertical:before{content:"\e235"}.glyphicon-menu-hamburger:before{content:"\e236"}.glyphicon-modal-window:before{content:"\e237"}.glyphicon-oil:before{content:"\e238"}.glyphicon-grain:before{content:"\e239"}.glyphicon-sunglasses:before{content:"\e240"}.glyphicon-text-size:before{content:"\e241"}.glyphicon-text-color:before{content:"\e242"}.glyphicon-text-background:before{content:"\e243"}.glyphicon-object-align-top:before{content:"\e244"}.glyphicon-object-align-bottom:before{content:"\e245"}.glyphicon-object-align-horizontal:before{content:"\e246"}.glyphicon-object-align-left:before{content:"\e247"}.glyphicon-object-align-vertical:before{content:"\e248"}.glyphicon-object-align-right:before{content:"\e249"}.glyphicon-triangle-right:before{content:"\e250"}.glyphicon-triangle-left:before{content:"\e251"}.glyphicon-triangle-bottom:before{content:"\e252"}.glyphicon-triangle-top:before{content:"\e253"}.glyphicon-console:before{content:"\e254"}.glyphicon-superscript:before{content:"\e255"}.glyphicon-subscript:before{content:"\e256"}.glyphicon-menu-left:before{content:"\e257"}.glyphicon-menu-right:before{content:"\e258"}.glyphicon-menu-down:before{content:"\e259"}.glyphicon-menu-up:before{content:"\e260"}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}*:before,*:after{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}input,button,select,textarea{font-family:inherit;font-size:inherit;line-height:inherit}a{color:#337ab7;text-decoration:none}a:hover,a:focus{color:#23527c;text-decoration:underline}a:focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}figure{margin:0}img{vertical-align:middle}.img-responsive,.thumbnail>img,.thumbnail a>img,.carousel-inner>.item>img,.carousel-inner>.item>a>img{display:block;max-width:100%;height:auto}.img-rounded{border-radius:6px}.img-thumbnail{padding:4px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:all .2s ease-in-out;-o-transition:all .2s ease-in-out;transition:all .2s ease-in-out;display:inline-block;max-width:100%;height:auto}.img-circle{border-radius:50%}hr{margin-top:20px;margin-bottom:20px;border:0;border-top:1px solid #eee}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;margin:0;overflow:visible;clip:auto}[role="button"]{cursor:pointer}h1,h2,h3,h4,h5,h6,.h1,.h2,.h3,.h4,.h5,.h6{font-family:inherit;font-weight:500;line-height:1.1;color:inherit}h1 small,h2 small,h3 small,h4 small,h5 small,h6 small,.h1 small,.h2 small,.h3 small,.h4 small,.h5 small,.h6 small,h1 .small,h2 .small,h3 .small,h4 .small,h5 .small,h6 .small,.h1 .small,.h2 .small,.h3 .small,.h4 .small,.h5 .small,.h6 .small{font-weight:400;line-height:1;color:#777}h1,.h1,h2,.h2,h3,.h3{margin-top:20px;margin-bottom:10px}h1 small,.h1 small,h2 small,.h2 small,h3 small,.h3 small,h1 .small,.h1 .small,h2 .small,.h2 .small,h3 .small,.h3 .small{font-size:65%}h4,.h4,h5,.h5,h6,.h6{margin-top:10px;margin-bottom:10px}h4 small,.h4 small,h5 small,.h5 small,h6 small,.h6 small,h4 .small,.h4 .small,h5 .small,.h5 .small,h6 .small,.h6 .small{font-size:75%}h1,.h1{font-size:36px}h2,.h2{font-size:30px}h3,.h3{font-size:24px}h4,.h4{font-size:18px}h5,.h5{font-size:14px}h6,.h6{font-size:12px}p{margin:0 0 10px}.lead{margin-bottom:20px;font-size:16px;font-weight:300;line-height:1.4}@media(min-width:768px){.lead{font-size:21px}}small,.small{font-size:85%}mark,.mark{padding:.2em;background-color:#fcf8e3}.text-left{text-align:left}.text-right{text-align:right}.text-center{text-align:center}.text-justify{text-align:justify}.text-nowrap{white-space:nowrap}.text-lowercase{text-transform:lowercase}.text-uppercase{text-transform:uppercase}.text-capitalize{text-transform:capitalize}.text-muted{color:#777}.text-primary{color:#337ab7}a.text-primary:hover,a.text-primary:focus{color:#286090}.text-success{color:#3c763d}a.text-success:hover,a.text-success:focus{color:#2b542c}.text-info{color:#31708f}a.text-info:hover,a.text-info:focus{color:#245269}.text-warning{color:#8a6d3b}a.text-warning:hover,a.text-warning:focus{color:#66512c}.text-danger{color:#a94442}a.text-danger:hover,a.text-danger:focus{color:#843534}.bg-primary{color:#fff;background-color:#337ab7}a.bg-primary:hover,a.bg-primary:focus{background-color:#286090}.bg-success{background-color:#dff0d8}a.bg-success:hover,a.bg-success:focus{background-color:#c1e2b3}.bg-info{background-color:#d9edf7}a.bg-info:hover,a.bg-info:focus{background-color:#afd9ee}.bg-warning{background-color:#fcf8e3}a.bg-warning:hover,a.bg-warning:focus{background-color:#f7ecb5}.bg-danger{background-color:#f2dede}a.bg-danger:hover,a.bg-danger:focus{background-color:#e4b9b9}.page-header{padding-bottom:9px;margin:40px 0 20px;border-bottom:1px solid #eee}ul,ol{margin-top:0;margin-bottom:10px}ul ul,ol ul,ul ol,ol ol{margin-bottom:0}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none;margin-left:-5px}.list-inline>li{display:inline-block;padding-right:5px;padding-left:5px}dl{margin-top:0;margin-bottom:20px}dt,dd{line-height:1.42857143}dt{font-weight:700}dd{margin-left:0}@media(min-width:768px){.dl-horizontal dt{float:left;width:160px;clear:left;text-align:right;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.dl-horizontal dd{margin-left:180px}}abbr[title],abbr[data-original-title]{cursor:help}.initialism{font-size:90%;text-transform:uppercase}blockquote{padding:10px 20px;margin:0 0 20px;font-size:17.5px;border-left:5px solid #eee}blockquote p:last-child,blockquote ul:last-child,blockquote ol:last-child{margin-bottom:0}blockquote footer,blockquote small,blockquote .small{display:block;font-size:80%;line-height:1.42857143;color:#777}blockquote footer:before,blockquote small:before,blockquote .small:before{content:"\2014 \00A0"}.blockquote-reverse,blockquote.pull-right{padding-right:15px;padding-left:0;text-align:right;border-right:5px solid #eee;border-left:0}.blockquote-reverse footer:before,blockquote.pull-right footer:before,.blockquote-reverse small:before,blockquote.pull-right small:before,.blockquote-reverse .small:before,blockquote.pull-right .small:before{content:""}.blockquote-reverse footer:after,blockquote.pull-right footer:after,.blockquote-reverse small:after,blockquote.pull-right small:after,.blockquote-reverse .small:after,blockquote.pull-right .small:after{content:"\00A0 \2014"}address{margin-bottom:20px;font-style:normal;line-height:1.42857143}code,kbd,pre,samp{font-family:Menlo,Monaco,Consolas,"Courier New",monospace}code{padding:2px 4px;font-size:90%;color:#c7254e;background-color:#f9f2f4;border-radius:4px}kbd{padding:2px 4px;font-size:90%;color:#fff;background-color:#333;border-radius:3px;box-shadow:inset 0 -1px 0 rgba(0,0,0,0.25)}kbd kbd{padding:0;font-size:100%;font-weight:700;box-shadow:none}pre{display:block;padding:9.5px;margin:0 0 10px;font-size:13px;line-height:1.42857143;color:#333;word-break:break-all;word-wrap:break-word;background-color:#f5f5f5;border:1px solid #ccc;border-radius:4px}pre code{padding:0;font-size:inherit;color:inherit;white-space:pre-wrap;background-color:transparent;border-radius:0}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media(min-width:768px){.container{width:750px}}@media(min-width:992px){.container{width:970px}}@media(min-width:1200px){.container{width:1170px}}.container-fluid{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{margin-right:-15px;margin-left:-15px}.row-no-gutters{margin-right:0;margin-left:0}.row-no-gutters [class*="col-"]{padding-right:0;padding-left:0}.col-xs-1,.col-sm-1,.col-md-1,.col-lg-1,.col-xs-2,.col-sm-2,.col-md-2,.col-lg-2,.col-xs-3,.col-sm-3,.col-md-3,.col-lg-3,.col-xs-4,.col-sm-4,.col-md-4,.col-lg-4,.col-xs-5,.col-sm-5,.col-md-5,.col-lg-5,.col-xs-6,.col-sm-6,.col-md-6,.col-lg-6,.col-xs-7,.col-sm-7,.col-md-7,.col-lg-7,.col-xs-8,.col-sm-8,.col-md-8,.col-lg-8,.col-xs-9,.col-sm-9,.col-md-9,.col-lg-9,.col-xs-10,.col-sm-10,.col-md-10,.col-lg-10,.col-xs-11,.col-sm-11,.col-md-11,.col-lg-11,.col-xs-12,.col-sm-12,.col-md-12,.col-lg-12{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-1,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9,.col-xs-10,.col-xs-11,.col-xs-12{float:left}.col-xs-12{width:100%}.col-xs-11{width:91.66666667%}.col-xs-10{width:83.33333333%}.col-xs-9{width:75%}.col-xs-8{width:66.66666667%}.col-xs-7{width:58.33333333%}.col-xs-6{width:50%}.col-xs-5{width:41.66666667%}.col-xs-4{width:33.33333333%}.col-xs-3{width:25%}.col-xs-2{width:16.66666667%}.col-xs-1{width:8.33333333%}.col-xs-pull-12{right:100%}.col-xs-pull-11{right:91.66666667%}.col-xs-pull-10{right:83.33333333%}.col-xs-pull-9{right:75%}.col-xs-pull-8{right:66.66666667%}.col-xs-pull-7{right:58.33333333%}.col-xs-pull-6{right:50%}.col-xs-pull-5{right:41.66666667%}.col-xs-pull-4{right:33.33333333%}.col-xs-pull-3{right:25%}.col-xs-pull-2{right:16.66666667%}.col-xs-pull-1{right:8.33333333%}.col-xs-pull-0{right:auto}.col-xs-push-12{left:100%}.col-xs-push-11{left:91.66666667%}.col-xs-push-10{left:83.33333333%}.col-xs-push-9{left:75%}.col-xs-push-8{left:66.66666667%}.col-xs-push-7{left:58.33333333%}.col-xs-push-6{left:50%}.col-xs-push-5{left:41.66666667%}.col-xs-push-4{left:33.33333333%}.col-xs-push-3{left:25%}.col-xs-push-2{left:16.66666667%}.col-xs-push-1{left:8.33333333%}.col-xs-push-0{left:auto}.col-xs-offset-12{margin-left:100%}.col-xs-offset-11{margin-left:91.66666667%}.col-xs-offset-10{margin-left:83.33333333%}.col-xs-offset-9{margin-left:75%}.col-xs-offset-8{margin-left:66.66666667%}.col-xs-offset-7{margin-left:58.33333333%}.col-xs-offset-6{margin-left:50%}.col-xs-offset-5{margin-left:41.66666667%}.col-xs-offset-4{margin-left:33.33333333%}.col-xs-offset-3{margin-left:25%}.col-xs-offset-2{margin-left:16.66666667%}.col-xs-offset-1{margin-left:8.33333333%}.col-xs-offset-0{margin-left:0}@media(min-width:768px){.col-sm-1,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-10,.col-sm-11,.col-sm-12{float:left}.col-sm-12{width:100%}.col-sm-11{width:91.66666667%}.col-sm-10{width:83.33333333%}.col-sm-9{width:75%}.col-sm-8{width:66.66666667%}.col-sm-7{width:58.33333333%}.col-sm-6{width:50%}.col-sm-5{width:41.66666667%}.col-sm-4{width:33.33333333%}.col-sm-3{width:25%}.col-sm-2{width:16.66666667%}.col-sm-1{width:8.33333333%}.col-sm-pull-12{right:100%}.col-sm-pull-11{right:91.66666667%}.col-sm-pull-10{right:83.33333333%}.col-sm-pull-9{right:75%}.col-sm-pull-8{right:66.66666667%}.col-sm-pull-7{right:58.33333333%}.col-sm-pull-6{right:50%}.col-sm-pull-5{right:41.66666667%}.col-sm-pull-4{right:33.33333333%}.col-sm-pull-3{right:25%}.col-sm-pull-2{right:16.66666667%}.col-sm-pull-1{right:8.33333333%}.col-sm-pull-0{right:auto}.col-sm-push-12{left:100%}.col-sm-push-11{left:91.66666667%}.col-sm-push-10{left:83.33333333%}.col-sm-push-9{left:75%}.col-sm-push-8{left:66.66666667%}.col-sm-push-7{left:58.33333333%}.col-sm-push-6{left:50%}.col-sm-push-5{left:41.66666667%}.col-sm-push-4{left:33.33333333%}.col-sm-push-3{left:25%}.col-sm-push-2{left:16.66666667%}.col-sm-push-1{left:8.33333333%}.col-sm-push-0{left:auto}.col-sm-offset-12{margin-left:100%}.col-sm-offset-11{margin-left:91.66666667%}.col-sm-offset-10{margin-left:83.33333333%}.col-sm-offset-9{margin-left:75%}.col-sm-offset-8{margin-left:66.66666667%}.col-sm-offset-7{margin-left:58.33333333%}.col-sm-offset-6{margin-left:50%}.col-sm-offset-5{margin-left:41.66666667%}.col-sm-offset-4{margin-left:33.33333333%}.col-sm-offset-3{margin-left:25%}.col-sm-offset-2{margin-left:16.66666667%}.col-sm-offset-1{margin-left:8.33333333%}.col-sm-offset-0{margin-left:0}}@media(min-width:992px){.col-md-1,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-10,.col-md-11,.col-md-12{float:left}.col-md-12{width:100%}.col-md-11{width:91.66666667%}.col-md-10{width:83.33333333%}.col-md-9{width:75%}.col-md-8{width:66.66666667%}.col-md-7{width:58.33333333%}.col-md-6{width:50%}.col-md-5{width:41.66666667%}.col-md-4{width:33.33333333%}.col-md-3{width:25%}.col-md-2{width:16.66666667%}.col-md-1{width:8.33333333%}.col-md-pull-12{right:100%}.col-md-pull-11{right:91.66666667%}.col-md-pull-10{right:83.33333333%}.col-md-pull-9{right:75%}.col-md-pull-8{right:66.66666667%}.col-md-pull-7{right:58.33333333%}.col-md-pull-6{right:50%}.col-md-pull-5{right:41.66666667%}.col-md-pull-4{right:33.33333333%}.col-md-pull-3{right:25%}.col-md-pull-2{right:16.66666667%}.col-md-pull-1{right:8.33333333%}.col-md-pull-0{right:auto}.col-md-push-12{left:100%}.col-md-push-11{left:91.66666667%}.col-md-push-10{left:83.33333333%}.col-md-push-9{left:75%}.col-md-push-8{left:66.66666667%}.col-md-push-7{left:58.33333333%}.col-md-push-6{left:50%}.col-md-push-5{left:41.66666667%}.col-md-push-4{left:33.33333333%}.col-md-push-3{left:25%}.col-md-push-2{left:16.66666667%}.col-md-push-1{left:8.33333333%}.col-md-push-0{left:auto}.col-md-offset-12{margin-left:100%}.col-md-offset-11{margin-left:91.66666667%}.col-md-offset-10{margin-left:83.33333333%}.col-md-offset-9{margin-left:75%}.col-md-offset-8{margin-left:66.66666667%}.col-md-offset-7{margin-left:58.33333333%}.col-md-offset-6{margin-left:50%}.col-md-offset-5{margin-left:41.66666667%}.col-md-offset-4{margin-left:33.33333333%}.col-md-offset-3{margin-left:25%}.col-md-offset-2{margin-left:16.66666667%}.col-md-offset-1{margin-left:8.33333333%}.col-md-offset-0{margin-left:0}}@media(min-width:1200px){.col-lg-1,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-10,.col-lg-11,.col-lg-12{float:left}.col-lg-12{width:100%}.col-lg-11{width:91.66666667%}.col-lg-10{width:83.33333333%}.col-lg-9{width:75%}.col-lg-8{width:66.66666667%}.col-lg-7{width:58.33333333%}.col-lg-6{width:50%}.col-lg-5{width:41.66666667%}.col-lg-4{width:33.33333333%}.col-lg-3{width:25%}.col-lg-2{width:16.66666667%}.col-lg-1{width:8.33333333%}.col-lg-pull-12{right:100%}.col-lg-pull-11{right:91.66666667%}.col-lg-pull-10{right:83.33333333%}.col-lg-pull-9{right:75%}.col-lg-pull-8{right:66.66666667%}.col-lg-pull-7{right:58.33333333%}.col-lg-pull-6{right:50%}.col-lg-pull-5{right:41.66666667%}.col-lg-pull-4{right:33.33333333%}.col-lg-pull-3{right:25%}.col-lg-pull-2{right:16.66666667%}.col-lg-pull-1{right:8.33333333%}.col-lg-pull-0{right:auto}.col-lg-push-12{left:100%}.col-lg-push-11{left:91.66666667%}.col-lg-push-10{left:83.33333333%}.col-lg-push-9{left:75%}.col-lg-push-8{left:66.66666667%}.col-lg-push-7{left:58.33333333%}.col-lg-push-6{left:50%}.col-lg-push-5{left:41.66666667%}.col-lg-push-4{left:33.33333333%}.col-lg-push-3{left:25%}.col-lg-push-2{left:16.66666667%}.col-lg-push-1{left:8.33333333%}.col-lg-push-0{left:auto}.col-lg-offset-12{margin-left:100%}.col-lg-offset-11{margin-left:91.66666667%}.col-lg-offset-10{margin-left:83.33333333%}.col-lg-offset-9{margin-left:75%}.col-lg-offset-8{margin-left:66.66666667%}.col-lg-offset-7{margin-left:58.33333333%}.col-lg-offset-6{margin-left:50%}.col-lg-offset-5{margin-left:41.66666667%}.col-lg-offset-4{margin-left:33.33333333%}.col-lg-offset-3{margin-left:25%}.col-lg-offset-2{margin-left:16.66666667%}.col-lg-offset-1{margin-left:8.33333333%}.col-lg-offset-0{margin-left:0}}table{background-color:transparent}table col[class*="col-"]{position:static;display:table-column;float:none}table td[class*="col-"],table th[class*="col-"]{position:static;display:table-cell;float:none}caption{padding-top:8px;padding-bottom:8px;color:#777;text-align:left}th{text-align:left}.table{width:100%;max-width:100%;margin-bottom:20px}.table>thead>tr>th,.table>tbody>tr>th,.table>tfoot>tr>th,.table>thead>tr>td,.table>tbody>tr>td,.table>tfoot>tr>td{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table>thead>tr>th{vertical-align:bottom;border-bottom:2px solid #ddd}.table>caption+thead>tr:first-child>th,.table>colgroup+thead>tr:first-child>th,.table>thead:first-child>tr:first-child>th,.table>caption+thead>tr:first-child>td,.table>colgroup+thead>tr:first-child>td,.table>thead:first-child>tr:first-child>td{border-top:0}.table>tbody+tbody{border-top:2px solid #ddd}.table .table{background-color:#fff}.table-condensed>thead>tr>th,.table-condensed>tbody>tr>th,.table-condensed>tfoot>tr>th,.table-condensed>thead>tr>td,.table-condensed>tbody>tr>td,.table-condensed>tfoot>tr>td{padding:5px}.table-bordered{border:1px solid #ddd}.table-bordered>thead>tr>th,.table-bordered>tbody>tr>th,.table-bordered>tfoot>tr>th,.table-bordered>thead>tr>td,.table-bordered>tbody>tr>td,.table-bordered>tfoot>tr>td{border:1px solid #ddd}.table-bordered>thead>tr>th,.table-bordered>thead>tr>td{border-bottom-width:2px}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.table-hover>tbody>tr:hover{background-color:#f5f5f5}.table>thead>tr>td.active,.table>tbody>tr>td.active,.table>tfoot>tr>td.active,.table>thead>tr>th.active,.table>tbody>tr>th.active,.table>tfoot>tr>th.active,.table>thead>tr.active>td,.table>tbody>tr.active>td,.table>tfoot>tr.active>td,.table>thead>tr.active>th,.table>tbody>tr.active>th,.table>tfoot>tr.active>th{background-color:#f5f5f5}.table-hover>tbody>tr>td.active:hover,.table-hover>tbody>tr>th.active:hover,.table-hover>tbody>tr.active:hover>td,.table-hover>tbody>tr:hover>.active,.table-hover>tbody>tr.active:hover>th{background-color:#e8e8e8}.table>thead>tr>td.success,.table>tbody>tr>td.success,.table>tfoot>tr>td.success,.table>thead>tr>th.success,.table>tbody>tr>th.success,.table>tfoot>tr>th.success,.table>thead>tr.success>td,.table>tbody>tr.success>td,.table>tfoot>tr.success>td,.table>thead>tr.success>th,.table>tbody>tr.success>th,.table>tfoot>tr.success>th{background-color:#dff0d8}.table-hover>tbody>tr>td.success:hover,.table-hover>tbody>tr>th.success:hover,.table-hover>tbody>tr.success:hover>td,.table-hover>tbody>tr:hover>.success,.table-hover>tbody>tr.success:hover>th{background-color:#d0e9c6}.table>thead>tr>td.info,.table>tbody>tr>td.info,.table>tfoot>tr>td.info,.table>thead>tr>th.info,.table>tbody>tr>th.info,.table>tfoot>tr>th.info,.table>thead>tr.info>td,.table>tbody>tr.info>td,.table>tfoot>tr.info>td,.table>thead>tr.info>th,.table>tbody>tr.info>th,.table>tfoot>tr.info>th{background-color:#d9edf7}.table-hover>tbody>tr>td.info:hover,.table-hover>tbody>tr>th.info:hover,.table-hover>tbody>tr.info:hover>td,.table-hover>tbody>tr:hover>.info,.table-hover>tbody>tr.info:hover>th{background-color:#c4e3f3}.table>thead>tr>td.warning,.table>tbody>tr>td.warning,.table>tfoot>tr>td.warning,.table>thead>tr>th.warning,.table>tbody>tr>th.warning,.table>tfoot>tr>th.warning,.table>thead>tr.warning>td,.table>tbody>tr.warning>td,.table>tfoot>tr.warning>td,.table>thead>tr.warning>th,.table>tbody>tr.warning>th,.table>tfoot>tr.warning>th{background-color:#fcf8e3}.table-hover>tbody>tr>td.warning:hover,.table-hover>tbody>tr>th.warning:hover,.table-hover>tbody>tr.warning:hover>td,.table-hover>tbody>tr:hover>.warning,.table-hover>tbody>tr.warning:hover>th{background-color:#faf2cc}.table>thead>tr>td.danger,.table>tbody>tr>td.danger,.table>tfoot>tr>td.danger,.table>thead>tr>th.danger,.table>tbody>tr>th.danger,.table>tfoot>tr>th.danger,.table>thead>tr.danger>td,.table>tbody>tr.danger>td,.table>tfoot>tr.danger>td,.table>thead>tr.danger>th,.table>tbody>tr.danger>th,.table>tfoot>tr.danger>th{background-color:#f2dede}.table-hover>tbody>tr>td.danger:hover,.table-hover>tbody>tr>th.danger:hover,.table-hover>tbody>tr.danger:hover>td,.table-hover>tbody>tr:hover>.danger,.table-hover>tbody>tr.danger:hover>th{background-color:#ebcccc}.table-responsive{min-height:.01%;overflow-x:auto}@media screen and (max-width:767px){.table-responsive{width:100%;margin-bottom:15px;overflow-y:hidden;-ms-overflow-style:-ms-autohiding-scrollbar;border:1px solid #ddd}.table-responsive>.table{margin-bottom:0}.table-responsive>.table>thead>tr>th,.table-responsive>.table>tbody>tr>th,.table-responsive>.table>tfoot>tr>th,.table-responsive>.table>thead>tr>td,.table-responsive>.table>tbody>tr>td,.table-responsive>.table>tfoot>tr>td{white-space:nowrap}.table-responsive>.table-bordered{border:0}.table-responsive>.table-bordered>thead>tr>th:first-child,.table-responsive>.table-bordered>tbody>tr>th:first-child,.table-responsive>.table-bordered>tfoot>tr>th:first-child,.table-responsive>.table-bordered>thead>tr>td:first-child,.table-responsive>.table-bordered>tbody>tr>td:first-child,.table-responsive>.table-bordered>tfoot>tr>td:first-child{border-left:0}.table-responsive>.table-bordered>thead>tr>th:last-child,.table-responsive>.table-bordered>tbody>tr>th:last-child,.table-responsive>.table-bordered>tfoot>tr>th:last-child,.table-responsive>.table-bordered>thead>tr>td:last-child,.table-responsive>.table-bordered>tbody>tr>td:last-child,.table-responsive>.table-bordered>tfoot>tr>td:last-child{border-right:0}.table-responsive>.table-bordered>tbody>tr:last-child>th,.table-responsive>.table-bordered>tfoot>tr:last-child>th,.table-responsive>.table-bordered>tbody>tr:last-child>td,.table-responsive>.table-bordered>tfoot>tr:last-child>td{border-bottom:0}}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;padding:0;margin-bottom:20px;font-size:21px;line-height:inherit;color:#333;border:0;border-bottom:1px solid #e5e5e5}label{display:inline-block;max-width:100%;margin-bottom:5px;font-weight:700}input[type="search"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;-webkit-appearance:none;appearance:none}input[type="radio"],input[type="checkbox"]{margin:4px 0 0;margin-top:1px \9;line-height:normal}input[type="radio"][disabled],input[type="checkbox"][disabled],input[type="radio"].disabled,input[type="checkbox"].disabled,fieldset[disabled] input[type="radio"],fieldset[disabled] input[type="checkbox"]{cursor:not-allowed}input[type="file"]{display:block}input[type="range"]{display:block;width:100%}select[multiple],select[size]{height:auto}input[type="file"]:focus,input[type="radio"]:focus,input[type="checkbox"]:focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}output{display:block;padding-top:7px;font-size:14px;line-height:1.42857143;color:#555}.form-control{display:block;width:100%;height:34px;padding:6px 12px;font-size:14px;line-height:1.42857143;color:#555;background-color:#fff;background-image:none;border:1px solid #ccc;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);-webkit-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;-o-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s}.form-control:focus{border-color:#66afe9;outline:0;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,0.6);box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,0.6)}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.form-control::-webkit-input-placeholder{color:#999}.form-control::-ms-expand{background-color:transparent;border:0}.form-control[disabled],.form-control[readonly],fieldset[disabled] .form-control{background-color:#eee;opacity:1}.form-control[disabled],fieldset[disabled] .form-control{cursor:not-allowed}textarea.form-control{height:auto}@media screen and (-webkit-min-device-pixel-ratio:0){input[type="date"].form-control,input[type="time"].form-control,input[type="datetime-local"].form-control,input[type="month"].form-control{line-height:34px}input[type="date"].input-sm,input[type="time"].input-sm,input[type="datetime-local"].input-sm,input[type="month"].input-sm,.input-group-sm input[type="date"],.input-group-sm input[type="time"],.input-group-sm input[type="datetime-local"],.input-group-sm input[type="month"]{line-height:30px}input[type="date"].input-lg,input[type="time"].input-lg,input[type="datetime-local"].input-lg,input[type="month"].input-lg,.input-group-lg input[type="date"],.input-group-lg input[type="time"],.input-group-lg input[type="datetime-local"],.input-group-lg input[type="month"]{line-height:46px}}.form-group{margin-bottom:15px}.radio,.checkbox{position:relative;display:block;margin-top:10px;margin-bottom:10px}.radio.disabled label,.checkbox.disabled label,fieldset[disabled] .radio label,fieldset[disabled] .checkbox label{cursor:not-allowed}.radio label,.checkbox label{min-height:20px;padding-left:20px;margin-bottom:0;font-weight:400;cursor:pointer}.radio input[type="radio"],.radio-inline input[type="radio"],.checkbox input[type="checkbox"],.checkbox-inline input[type="checkbox"]{position:absolute;margin-top:4px \9;margin-left:-20px}.radio+.radio,.checkbox+.checkbox{margin-top:-5px}.radio-inline,.checkbox-inline{position:relative;display:inline-block;padding-left:20px;margin-bottom:0;font-weight:400;vertical-align:middle;cursor:pointer}.radio-inline.disabled,.checkbox-inline.disabled,fieldset[disabled] .radio-inline,fieldset[disabled] .checkbox-inline{cursor:not-allowed}.radio-inline+.radio-inline,.checkbox-inline+.checkbox-inline{margin-top:0;margin-left:10px}.form-control-static{min-height:34px;padding-top:7px;padding-bottom:7px;margin-bottom:0}.form-control-static.input-lg,.form-control-static.input-sm{padding-right:0;padding-left:0}.input-sm{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-sm{height:30px;line-height:30px}textarea.input-sm,select[multiple].input-sm{height:auto}.form-group-sm .form-control{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.form-group-sm select.form-control{height:30px;line-height:30px}.form-group-sm textarea.form-control,.form-group-sm select[multiple].form-control{height:auto}.form-group-sm .form-control-static{height:30px;min-height:32px;padding:6px 10px;font-size:12px;line-height:1.5}.input-lg{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-lg{height:46px;line-height:46px}textarea.input-lg,select[multiple].input-lg{height:auto}.form-group-lg .form-control{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.form-group-lg select.form-control{height:46px;line-height:46px}.form-group-lg textarea.form-control,.form-group-lg select[multiple].form-control{height:auto}.form-group-lg .form-control-static{height:46px;min-height:38px;padding:11px 16px;font-size:18px;line-height:1.3333333}.has-feedback{position:relative}.has-feedback .form-control{padding-right:42.5px}.form-control-feedback{position:absolute;top:0;right:0;z-index:2;display:block;width:34px;height:34px;line-height:34px;text-align:center;pointer-events:none}.input-lg+.form-control-feedback,.input-group-lg+.form-control-feedback,.form-group-lg .form-control+.form-control-feedback{width:46px;height:46px;line-height:46px}.input-sm+.form-control-feedback,.input-group-sm+.form-control-feedback,.form-group-sm .form-control+.form-control-feedback{width:30px;height:30px;line-height:30px}.has-success .help-block,.has-success .control-label,.has-success .radio,.has-success .checkbox,.has-success .radio-inline,.has-success .checkbox-inline,.has-success.radio label,.has-success.checkbox label,.has-success.radio-inline label,.has-success.checkbox-inline label{color:#3c763d}.has-success .form-control{border-color:#3c763d;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-success .form-control:focus{border-color:#2b542c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #67b168;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #67b168}.has-success .input-group-addon{color:#3c763d;background-color:#dff0d8;border-color:#3c763d}.has-success .form-control-feedback{color:#3c763d}.has-warning .help-block,.has-warning .control-label,.has-warning .radio,.has-warning .checkbox,.has-warning .radio-inline,.has-warning .checkbox-inline,.has-warning.radio label,.has-warning.checkbox label,.has-warning.radio-inline label,.has-warning.checkbox-inline label{color:#8a6d3b}.has-warning .form-control{border-color:#8a6d3b;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-warning .form-control:focus{border-color:#66512c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #c0a16b;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #c0a16b}.has-warning .input-group-addon{color:#8a6d3b;background-color:#fcf8e3;border-color:#8a6d3b}.has-warning .form-control-feedback{color:#8a6d3b}.has-error .help-block,.has-error .control-label,.has-error .radio,.has-error .checkbox,.has-error .radio-inline,.has-error .checkbox-inline,.has-error.radio label,.has-error.checkbox label,.has-error.radio-inline label,.has-error.checkbox-inline label{color:#a94442}.has-error .form-control{border-color:#a94442;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075)}.has-error .form-control:focus{border-color:#843534;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #ce8483;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 6px #ce8483}.has-error .input-group-addon{color:#a94442;background-color:#f2dede;border-color:#a94442}.has-error .form-control-feedback{color:#a94442}.has-feedback label ~ .form-control-feedback{top:25px}.has-feedback label.sr-only ~ .form-control-feedback{top:0}.help-block{display:block;margin-top:5px;margin-bottom:10px;color:#737373}@media(min-width:768px){.form-inline .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-static{display:inline-block}.form-inline .input-group{display:inline-table;vertical-align:middle}.form-inline .input-group .input-group-addon,.form-inline .input-group .input-group-btn,.form-inline .input-group .form-control{width:auto}.form-inline .input-group>.form-control{width:100%}.form-inline .control-label{margin-bottom:0;vertical-align:middle}.form-inline .radio,.form-inline .checkbox{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.form-inline .radio label,.form-inline .checkbox label{padding-left:0}.form-inline .radio input[type="radio"],.form-inline .checkbox input[type="checkbox"]{position:relative;margin-left:0}.form-inline .has-feedback .form-control-feedback{top:0}}.form-horizontal .radio,.form-horizontal .checkbox,.form-horizontal .radio-inline,.form-horizontal .checkbox-inline{padding-top:7px;margin-top:0;margin-bottom:0}.form-horizontal .radio,.form-horizontal .checkbox{min-height:27px}.form-horizontal .form-group{margin-right:-15px;margin-left:-15px}@media(min-width:768px){.form-horizontal .control-label{padding-top:7px;margin-bottom:0;text-align:right}}.form-horizontal .has-feedback .form-control-feedback{right:15px}@media(min-width:768px){.form-horizontal .form-group-lg .control-label{padding-top:11px;font-size:18px}}@media(min-width:768px){.form-horizontal .form-group-sm .control-label{padding-top:6px;font-size:12px}}.btn{display:inline-block;margin-bottom:0;font-weight:normal;text-align:center;white-space:nowrap;vertical-align:middle;touch-action:manipulation;cursor:pointer;background-image:none;border:1px solid transparent;padding:6px 12px;font-size:14px;line-height:1.42857143;border-radius:4px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.btn:focus,.btn:active:focus,.btn.active:focus,.btn.focus,.btn:active.focus,.btn.active.focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.btn:hover,.btn:focus,.btn.focus{color:#333;text-decoration:none}.btn:active,.btn.active{background-image:none;outline:0;-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,0.125);box-shadow:inset 0 3px 5px rgba(0,0,0,0.125)}.btn.disabled,.btn[disabled],fieldset[disabled] .btn{cursor:not-allowed;filter:alpha(opacity=65);opacity:.65;-webkit-box-shadow:none;box-shadow:none}a.btn.disabled,fieldset[disabled] a.btn{pointer-events:none}.btn-default{color:#333;background-color:#fff;border-color:#ccc}.btn-default:focus,.btn-default.focus{color:#333;background-color:#e6e6e6;border-color:#8c8c8c}.btn-default:hover{color:#333;background-color:#e6e6e6;border-color:#adadad}.btn-default:active,.btn-default.active,.open>.dropdown-toggle.btn-default{color:#333;background-color:#e6e6e6;background-image:none;border-color:#adadad}.btn-default:active:hover,.btn-default.active:hover,.open>.dropdown-toggle.btn-default:hover,.btn-default:active:focus,.btn-default.active:focus,.open>.dropdown-toggle.btn-default:focus,.btn-default:active.focus,.btn-default.active.focus,.open>.dropdown-toggle.btn-default.focus{color:#333;background-color:#d4d4d4;border-color:#8c8c8c}.btn-default.disabled:hover,.btn-default[disabled]:hover,fieldset[disabled] .btn-default:hover,.btn-default.disabled:focus,.btn-default[disabled]:focus,fieldset[disabled] .btn-default:focus,.btn-default.disabled.focus,.btn-default[disabled].focus,fieldset[disabled] .btn-default.focus{background-color:#fff;border-color:#ccc}.btn-default .badge{color:#fff;background-color:#333}.btn-primary{color:#fff;background-color:#337ab7;border-color:#2e6da4}.btn-primary:focus,.btn-primary.focus{color:#fff;background-color:#286090;border-color:#122b40}.btn-primary:hover{color:#fff;background-color:#286090;border-color:#204d74}.btn-primary:active,.btn-primary.active,.open>.dropdown-toggle.btn-primary{color:#fff;background-color:#286090;background-image:none;border-color:#204d74}.btn-primary:active:hover,.btn-primary.active:hover,.open>.dropdown-toggle.btn-primary:hover,.btn-primary:active:focus,.btn-primary.active:focus,.open>.dropdown-toggle.btn-primary:focus,.btn-primary:active.focus,.btn-primary.active.focus,.open>.dropdown-toggle.btn-primary.focus{color:#fff;background-color:#204d74;border-color:#122b40}.btn-primary.disabled:hover,.btn-primary[disabled]:hover,fieldset[disabled] .btn-primary:hover,.btn-primary.disabled:focus,.btn-primary[disabled]:focus,fieldset[disabled] .btn-primary:focus,.btn-primary.disabled.focus,.btn-primary[disabled].focus,fieldset[disabled] .btn-primary.focus{background-color:#337ab7;border-color:#2e6da4}.btn-primary .badge{color:#337ab7;background-color:#fff}.btn-success{color:#fff;background-color:#5cb85c;border-color:#4cae4c}.btn-success:focus,.btn-success.focus{color:#fff;background-color:#449d44;border-color:#255625}.btn-success:hover{color:#fff;background-color:#449d44;border-color:#398439}.btn-success:active,.btn-success.active,.open>.dropdown-toggle.btn-success{color:#fff;background-color:#449d44;background-image:none;border-color:#398439}.btn-success:active:hover,.btn-success.active:hover,.open>.dropdown-toggle.btn-success:hover,.btn-success:active:focus,.btn-success.active:focus,.open>.dropdown-toggle.btn-success:focus,.btn-success:active.focus,.btn-success.active.focus,.open>.dropdown-toggle.btn-success.focus{color:#fff;background-color:#398439;border-color:#255625}.btn-success.disabled:hover,.btn-success[disabled]:hover,fieldset[disabled] .btn-success:hover,.btn-success.disabled:focus,.btn-success[disabled]:focus,fieldset[disabled] .btn-success:focus,.btn-success.disabled.focus,.btn-success[disabled].focus,fieldset[disabled] .btn-success.focus{background-color:#5cb85c;border-color:#4cae4c}.btn-success .badge{color:#5cb85c;background-color:#fff}.btn-info{color:#fff;background-color:#5bc0de;border-color:#46b8da}.btn-info:focus,.btn-info.focus{color:#fff;background-color:#31b0d5;border-color:#1b6d85}.btn-info:hover{color:#fff;background-color:#31b0d5;border-color:#269abc}.btn-info:active,.btn-info.active,.open>.dropdown-toggle.btn-info{color:#fff;background-color:#31b0d5;background-image:none;border-color:#269abc}.btn-info:active:hover,.btn-info.active:hover,.open>.dropdown-toggle.btn-info:hover,.btn-info:active:focus,.btn-info.active:focus,.open>.dropdown-toggle.btn-info:focus,.btn-info:active.focus,.btn-info.active.focus,.open>.dropdown-toggle.btn-info.focus{color:#fff;background-color:#269abc;border-color:#1b6d85}.btn-info.disabled:hover,.btn-info[disabled]:hover,fieldset[disabled] .btn-info:hover,.btn-info.disabled:focus,.btn-info[disabled]:focus,fieldset[disabled] .btn-info:focus,.btn-info.disabled.focus,.btn-info[disabled].focus,fieldset[disabled] .btn-info.focus{background-color:#5bc0de;border-color:#46b8da}.btn-info .badge{color:#5bc0de;background-color:#fff}.btn-warning{color:#fff;background-color:#f0ad4e;border-color:#eea236}.btn-warning:focus,.btn-warning.focus{color:#fff;background-color:#ec971f;border-color:#985f0d}.btn-warning:hover{color:#fff;background-color:#ec971f;border-color:#d58512}.btn-warning:active,.btn-warning.active,.open>.dropdown-toggle.btn-warning{color:#fff;background-color:#ec971f;background-image:none;border-color:#d58512}.btn-warning:active:hover,.btn-warning.active:hover,.open>.dropdown-toggle.btn-warning:hover,.btn-warning:active:focus,.btn-warning.active:focus,.open>.dropdown-toggle.btn-warning:focus,.btn-warning:active.focus,.btn-warning.active.focus,.open>.dropdown-toggle.btn-warning.focus{color:#fff;background-color:#d58512;border-color:#985f0d}.btn-warning.disabled:hover,.btn-warning[disabled]:hover,fieldset[disabled] .btn-warning:hover,.btn-warning.disabled:focus,.btn-warning[disabled]:focus,fieldset[disabled] .btn-warning:focus,.btn-warning.disabled.focus,.btn-warning[disabled].focus,fieldset[disabled] .btn-warning.focus{background-color:#f0ad4e;border-color:#eea236}.btn-warning .badge{color:#f0ad4e;background-color:#fff}.btn-danger{color:#fff;background-color:#d9534f;border-color:#d43f3a}.btn-danger:focus,.btn-danger.focus{color:#fff;background-color:#c9302c;border-color:#761c19}.btn-danger:hover{color:#fff;background-color:#c9302c;border-color:#ac2925}.btn-danger:active,.btn-danger.active,.open>.dropdown-toggle.btn-danger{color:#fff;background-color:#c9302c;background-image:none;border-color:#ac2925}.btn-danger:active:hover,.btn-danger.active:hover,.open>.dropdown-toggle.btn-danger:hover,.btn-danger:active:focus,.btn-danger.active:focus,.open>.dropdown-toggle.btn-danger:focus,.btn-danger:active.focus,.btn-danger.active.focus,.open>.dropdown-toggle.btn-danger.focus{color:#fff;background-color:#ac2925;border-color:#761c19}.btn-danger.disabled:hover,.btn-danger[disabled]:hover,fieldset[disabled] .btn-danger:hover,.btn-danger.disabled:focus,.btn-danger[disabled]:focus,fieldset[disabled] .btn-danger:focus,.btn-danger.disabled.focus,.btn-danger[disabled].focus,fieldset[disabled] .btn-danger.focus{background-color:#d9534f;border-color:#d43f3a}.btn-danger .badge{color:#d9534f;background-color:#fff}.btn-link{font-weight:400;color:#337ab7;border-radius:0}.btn-link,.btn-link:active,.btn-link.active,.btn-link[disabled],fieldset[disabled] .btn-link{background-color:transparent;-webkit-box-shadow:none;box-shadow:none}.btn-link,.btn-link:hover,.btn-link:focus,.btn-link:active{border-color:transparent}.btn-link:hover,.btn-link:focus{color:#23527c;text-decoration:underline;background-color:transparent}.btn-link[disabled]:hover,fieldset[disabled] .btn-link:hover,.btn-link[disabled]:focus,fieldset[disabled] .btn-link:focus{color:#777;text-decoration:none}.btn-lg,.btn-group-lg>.btn{padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.btn-sm,.btn-group-sm>.btn{padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.btn-xs,.btn-group-xs>.btn{padding:1px 5px;font-size:12px;line-height:1.5;border-radius:3px}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:5px}input[type="submit"].btn-block,input[type="reset"].btn-block,input[type="button"].btn-block{width:100%}.fade{opacity:0;-webkit-transition:opacity .15s linear;-o-transition:opacity .15s linear;transition:opacity .15s linear}.fade.in{opacity:1}.collapse{display:none}.collapse.in{display:block}tr.collapse.in{display:table-row}tbody.collapse.in{display:table-row-group}.collapsing{position:relative;height:0;overflow:hidden;-webkit-transition-property:height,visibility;transition-property:height,visibility;-webkit-transition-duration:.35s;transition-duration:.35s;-webkit-transition-timing-function:ease;transition-timing-function:ease}.caret{display:inline-block;width:0;height:0;margin-left:2px;vertical-align:middle;border-top:4px dashed;border-top:4px solid \9;border-right:4px solid transparent;border-left:4px solid transparent}.dropup,.dropdown{position:relative}.dropdown-toggle:focus{outline:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:160px;padding:5px 0;margin:2px 0 0;font-size:14px;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,0.15);border-radius:4px;-webkit-box-shadow:0 6px 12px rgba(0,0,0,0.175);box-shadow:0 6px 12px rgba(0,0,0,0.175)}.dropdown-menu.pull-right{right:0;left:auto}.dropdown-menu .divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.dropdown-menu>li>a{display:block;padding:3px 20px;clear:both;font-weight:400;line-height:1.42857143;color:#333;white-space:nowrap}.dropdown-menu>li>a:hover,.dropdown-menu>li>a:focus{color:#262626;text-decoration:none;background-color:#f5f5f5}.dropdown-menu>.active>a,.dropdown-menu>.active>a:hover,.dropdown-menu>.active>a:focus{color:#fff;text-decoration:none;background-color:#337ab7;outline:0}.dropdown-menu>.disabled>a,.dropdown-menu>.disabled>a:hover,.dropdown-menu>.disabled>a:focus{color:#777}.dropdown-menu>.disabled>a:hover,.dropdown-menu>.disabled>a:focus{text-decoration:none;cursor:not-allowed;background-color:transparent;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false)}.open>.dropdown-menu{display:block}.open>a{outline:0}.dropdown-menu-right{right:0;left:auto}.dropdown-menu-left{right:auto;left:0}.dropdown-header{display:block;padding:3px 20px;font-size:12px;line-height:1.42857143;color:#777;white-space:nowrap}.dropdown-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:990}.pull-right>.dropdown-menu{right:0;left:auto}.dropup .caret,.navbar-fixed-bottom .dropdown .caret{content:"";border-top:0;border-bottom:4px dashed;border-bottom:4px solid \9}.dropup .dropdown-menu,.navbar-fixed-bottom .dropdown .dropdown-menu{top:auto;bottom:100%;margin-bottom:2px}@media(min-width:768px){.navbar-right .dropdown-menu{right:0;left:auto}.navbar-right .dropdown-menu-left{right:auto;left:0}}.btn-group,.btn-group-vertical{position:relative;display:inline-block;vertical-align:middle}.btn-group>.btn,.btn-group-vertical>.btn{position:relative;float:left}.btn-group>.btn:hover,.btn-group-vertical>.btn:hover,.btn-group>.btn:focus,.btn-group-vertical>.btn:focus,.btn-group>.btn:active,.btn-group-vertical>.btn:active,.btn-group>.btn.active,.btn-group-vertical>.btn.active{z-index:2}.btn-group .btn+.btn,.btn-group .btn+.btn-group,.btn-group .btn-group+.btn,.btn-group .btn-group+.btn-group{margin-left:-1px}.btn-toolbar{margin-left:-5px}.btn-toolbar .btn,.btn-toolbar .btn-group,.btn-toolbar .input-group{float:left}.btn-toolbar>.btn,.btn-toolbar>.btn-group,.btn-toolbar>.input-group{margin-left:5px}.btn-group>.btn:not(:first-child):not(:last-child):not(.dropdown-toggle){border-radius:0}.btn-group>.btn:first-child{margin-left:0}.btn-group>.btn:first-child:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn:last-child:not(:first-child),.btn-group>.dropdown-toggle:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.btn-group>.btn-group{float:left}.btn-group>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-bottom-left-radius:0}.btn-group .dropdown-toggle:active,.btn-group.open .dropdown-toggle{outline:0}.btn-group>.btn+.dropdown-toggle{padding-right:8px;padding-left:8px}.btn-group>.btn-lg+.dropdown-toggle{padding-right:12px;padding-left:12px}.btn-group.open .dropdown-toggle{-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,0.125);box-shadow:inset 0 3px 5px rgba(0,0,0,0.125)}.btn-group.open .dropdown-toggle.btn-link{-webkit-box-shadow:none;box-shadow:none}.btn .caret{margin-left:0}.btn-lg .caret{border-width:5px 5px 0;border-bottom-width:0}.dropup .btn-lg .caret{border-width:0 5px 5px}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group,.btn-group-vertical>.btn-group>.btn{display:block;float:none;width:100%;max-width:100%}.btn-group-vertical>.btn-group>.btn{float:none}.btn-group-vertical>.btn+.btn,.btn-group-vertical>.btn+.btn-group,.btn-group-vertical>.btn-group+.btn,.btn-group-vertical>.btn-group+.btn-group{margin-top:-1px;margin-left:0}.btn-group-vertical>.btn:not(:first-child):not(:last-child){border-radius:0}.btn-group-vertical>.btn:first-child:not(:last-child){border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn:last-child:not(:first-child){border-top-left-radius:0;border-top-right-radius:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}.btn-group-vertical>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group-vertical>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group-vertical>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-top-right-radius:0}.btn-group-justified{display:table;width:100%;table-layout:fixed;border-collapse:separate}.btn-group-justified>.btn,.btn-group-justified>.btn-group{display:table-cell;float:none;width:1%}.btn-group-justified>.btn-group .btn{width:100%}.btn-group-justified>.btn-group .dropdown-menu{left:auto}[data-toggle="buttons"]>.btn input[type="radio"],[data-toggle="buttons"]>.btn-group>.btn input[type="radio"],[data-toggle="buttons"]>.btn input[type="checkbox"],[data-toggle="buttons"]>.btn-group>.btn input[type="checkbox"]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:table;border-collapse:separate}.input-group[class*="col-"]{float:none;padding-right:0;padding-left:0}.input-group .form-control{position:relative;z-index:2;float:left;width:100%;margin-bottom:0}.input-group .form-control:focus{z-index:3}.input-group-lg>.form-control,.input-group-lg>.input-group-addon,.input-group-lg>.input-group-btn>.btn{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-group-lg>.form-control,select.input-group-lg>.input-group-addon,select.input-group-lg>.input-group-btn>.btn{height:46px;line-height:46px}textarea.input-group-lg>.form-control,textarea.input-group-lg>.input-group-addon,textarea.input-group-lg>.input-group-btn>.btn,select[multiple].input-group-lg>.form-control,select[multiple].input-group-lg>.input-group-addon,select[multiple].input-group-lg>.input-group-btn>.btn{height:auto}.input-group-sm>.form-control,.input-group-sm>.input-group-addon,.input-group-sm>.input-group-btn>.btn{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-group-sm>.form-control,select.input-group-sm>.input-group-addon,select.input-group-sm>.input-group-btn>.btn{height:30px;line-height:30px}textarea.input-group-sm>.form-control,textarea.input-group-sm>.input-group-addon,textarea.input-group-sm>.input-group-btn>.btn,select[multiple].input-group-sm>.form-control,select[multiple].input-group-sm>.input-group-addon,select[multiple].input-group-sm>.input-group-btn>.btn{height:auto}.input-group-addon,.input-group-btn,.input-group .form-control{display:table-cell}.input-group-addon:not(:first-child):not(:last-child),.input-group-btn:not(:first-child):not(:last-child),.input-group .form-control:not(:first-child):not(:last-child){border-radius:0}.input-group-addon,.input-group-btn{width:1%;white-space:nowrap;vertical-align:middle}.input-group-addon{padding:6px 12px;font-size:14px;font-weight:400;line-height:1;color:#555;text-align:center;background-color:#eee;border:1px solid #ccc;border-radius:4px}.input-group-addon.input-sm{padding:5px 10px;font-size:12px;border-radius:3px}.input-group-addon.input-lg{padding:10px 16px;font-size:18px;border-radius:6px}.input-group-addon input[type="radio"],.input-group-addon input[type="checkbox"]{margin-top:0}.input-group .form-control:first-child,.input-group-addon:first-child,.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group>.btn,.input-group-btn:first-child>.dropdown-toggle,.input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group-btn:last-child>.btn-group:not(:last-child)>.btn{border-top-right-radius:0;border-bottom-right-radius:0}.input-group-addon:first-child{border-right:0}.input-group .form-control:last-child,.input-group-addon:last-child,.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group>.btn,.input-group-btn:last-child>.dropdown-toggle,.input-group-btn:first-child>.btn:not(:first-child),.input-group-btn:first-child>.btn-group:not(:first-child)>.btn{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-addon:last-child{border-left:0}.input-group-btn{position:relative;font-size:0;white-space:nowrap}.input-group-btn>.btn{position:relative}.input-group-btn>.btn+.btn{margin-left:-1px}.input-group-btn>.btn:hover,.input-group-btn>.btn:focus,.input-group-btn>.btn:active{z-index:2}.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group{margin-right:-1px}.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group{z-index:2;margin-left:-1px}.nav{padding-left:0;margin-bottom:0;list-style:none}.nav>li{position:relative;display:block}.nav>li>a{position:relative;display:block;padding:10px 15px}.nav>li>a:hover,.nav>li>a:focus{text-decoration:none;background-color:#eee}.nav>li.disabled>a{color:#777}.nav>li.disabled>a:hover,.nav>li.disabled>a:focus{color:#777;text-decoration:none;cursor:not-allowed;background-color:transparent}.nav .open>a,.nav .open>a:hover,.nav .open>a:focus{background-color:#eee;border-color:#337ab7}.nav .nav-divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.nav>li>a>img{max-width:none}.nav-tabs{border-bottom:1px solid #ddd}.nav-tabs>li{float:left;margin-bottom:-1px}.nav-tabs>li>a{margin-right:2px;line-height:1.42857143;border:1px solid transparent;border-radius:4px 4px 0 0}.nav-tabs>li>a:hover{border-color:#eee #eee #ddd}.nav-tabs>li.active>a,.nav-tabs>li.active>a:hover,.nav-tabs>li.active>a:focus{color:#555;cursor:default;background-color:#fff;border:1px solid #ddd;border-bottom-color:transparent}.nav-tabs.nav-justified{width:100%;border-bottom:0}.nav-tabs.nav-justified>li{float:none}.nav-tabs.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-tabs.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media(min-width:768px){.nav-tabs.nav-justified>li{display:table-cell;width:1%}.nav-tabs.nav-justified>li>a{margin-bottom:0}}.nav-tabs.nav-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:hover,.nav-tabs.nav-justified>.active>a:focus{border:1px solid #ddd}@media(min-width:768px){.nav-tabs.nav-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:hover,.nav-tabs.nav-justified>.active>a:focus{border-bottom-color:#fff}}.nav-pills>li{float:left}.nav-pills>li>a{border-radius:4px}.nav-pills>li+li{margin-left:2px}.nav-pills>li.active>a,.nav-pills>li.active>a:hover,.nav-pills>li.active>a:focus{color:#fff;background-color:#337ab7}.nav-stacked>li{float:none}.nav-stacked>li+li{margin-top:2px;margin-left:0}.nav-justified{width:100%}.nav-justified>li{float:none}.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media(min-width:768px){.nav-justified>li{display:table-cell;width:1%}.nav-justified>li>a{margin-bottom:0}}.nav-tabs-justified{border-bottom:0}.nav-tabs-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:hover,.nav-tabs-justified>.active>a:focus{border:1px solid #ddd}@media(min-width:768px){.nav-tabs-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:hover,.nav-tabs-justified>.active>a:focus{border-bottom-color:#fff}}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.navbar{position:relative;min-height:50px;margin-bottom:20px;border:1px solid transparent}@media(min-width:768px){.navbar{border-radius:4px}}@media(min-width:768px){.navbar-header{float:left}}.navbar-collapse{padding-right:15px;padding-left:15px;overflow-x:visible;border-top:1px solid transparent;box-shadow:inset 0 1px 0 rgba(255,255,255,0.1);-webkit-overflow-scrolling:touch}.navbar-collapse.in{overflow-y:auto}@media(min-width:768px){.navbar-collapse{width:auto;border-top:0;box-shadow:none}.navbar-collapse.collapse{display:block!important;height:auto!important;padding-bottom:0;overflow:visible!important}.navbar-collapse.in{overflow-y:visible}.navbar-fixed-top .navbar-collapse,.navbar-static-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{padding-right:0;padding-left:0}}.navbar-fixed-top,.navbar-fixed-bottom{position:fixed;right:0;left:0;z-index:1030}.navbar-fixed-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{max-height:340px}@media(max-device-width:480px) and (orientation:landscape){.navbar-fixed-top .navbar-collapse,.navbar-fixed-bottom .navbar-collapse{max-height:200px}}@media(min-width:768px){.navbar-fixed-top,.navbar-fixed-bottom{border-radius:0}}.navbar-fixed-top{top:0;border-width:0 0 1px}.navbar-fixed-bottom{bottom:0;margin-bottom:0;border-width:1px 0 0}.container>.navbar-header,.container-fluid>.navbar-header,.container>.navbar-collapse,.container-fluid>.navbar-collapse{margin-right:-15px;margin-left:-15px}@media(min-width:768px){.container>.navbar-header,.container-fluid>.navbar-header,.container>.navbar-collapse,.container-fluid>.navbar-collapse{margin-right:0;margin-left:0}}.navbar-static-top{z-index:1000;border-width:0 0 1px}@media(min-width:768px){.navbar-static-top{border-radius:0}}.navbar-brand{float:left;height:50px;padding:15px 15px;font-size:18px;line-height:20px}.navbar-brand:hover,.navbar-brand:focus{text-decoration:none}.navbar-brand>img{display:block}@media(min-width:768px){.navbar>.container .navbar-brand,.navbar>.container-fluid .navbar-brand{margin-left:-15px}}.navbar-toggle{position:relative;float:right;padding:9px 10px;margin-right:15px;margin-top:8px;margin-bottom:8px;background-color:transparent;background-image:none;border:1px solid transparent;border-radius:4px}.navbar-toggle:focus{outline:0}.navbar-toggle .icon-bar{display:block;width:22px;height:2px;border-radius:1px}.navbar-toggle .icon-bar+.icon-bar{margin-top:4px}@media(min-width:768px){.navbar-toggle{display:none}}.navbar-nav{margin:7.5px -15px}.navbar-nav>li>a{padding-top:10px;padding-bottom:10px;line-height:20px}@media(max-width:767px){.navbar-nav .open .dropdown-menu{position:static;float:none;width:auto;margin-top:0;background-color:transparent;border:0;box-shadow:none}.navbar-nav .open .dropdown-menu>li>a,.navbar-nav .open .dropdown-menu .dropdown-header{padding:5px 15px 5px 25px}.navbar-nav .open .dropdown-menu>li>a{line-height:20px}.navbar-nav .open .dropdown-menu>li>a:hover,.navbar-nav .open .dropdown-menu>li>a:focus{background-image:none}}@media(min-width:768px){.navbar-nav{float:left;margin:0}.navbar-nav>li{float:left}.navbar-nav>li>a{padding-top:15px;padding-bottom:15px}}.navbar-form{padding:10px 15px;margin-right:-15px;margin-left:-15px;border-top:1px solid transparent;border-bottom:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);margin-top:8px;margin-bottom:8px}@media(min-width:768px){.navbar-form .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.navbar-form .form-control{display:inline-block;width:auto;vertical-align:middle}.navbar-form .form-control-static{display:inline-block}.navbar-form .input-group{display:inline-table;vertical-align:middle}.navbar-form .input-group .input-group-addon,.navbar-form .input-group .input-group-btn,.navbar-form .input-group .form-control{width:auto}.navbar-form .input-group>.form-control{width:100%}.navbar-form .control-label{margin-bottom:0;vertical-align:middle}.navbar-form .radio,.navbar-form .checkbox{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.navbar-form .radio label,.navbar-form .checkbox label{padding-left:0}.navbar-form .radio input[type="radio"],.navbar-form .checkbox input[type="checkbox"]{position:relative;margin-left:0}.navbar-form .has-feedback .form-control-feedback{top:0}}@media(max-width:767px){.navbar-form .form-group{margin-bottom:5px}.navbar-form .form-group:last-child{margin-bottom:0}}@media(min-width:768px){.navbar-form{width:auto;padding-top:0;padding-bottom:0;margin-right:0;margin-left:0;border:0;-webkit-box-shadow:none;box-shadow:none}}.navbar-nav>li>.dropdown-menu{margin-top:0;border-top-left-radius:0;border-top-right-radius:0}.navbar-fixed-bottom .navbar-nav>li>.dropdown-menu{margin-bottom:0;border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.navbar-btn{margin-top:8px;margin-bottom:8px}.navbar-btn.btn-sm{margin-top:10px;margin-bottom:10px}.navbar-btn.btn-xs{margin-top:14px;margin-bottom:14px}.navbar-text{margin-top:15px;margin-bottom:15px}@media(min-width:768px){.navbar-text{float:left;margin-right:15px;margin-left:15px}}@media(min-width:768px){.navbar-left{float:left!important}.navbar-right{float:right!important;margin-right:-15px}.navbar-right ~ .navbar-right{margin-right:0}}.navbar-default{background-color:#f8f8f8;border-color:#e7e7e7}.navbar-default .navbar-brand{color:#777}.navbar-default .navbar-brand:hover,.navbar-default .navbar-brand:focus{color:#5e5e5e;background-color:transparent}.navbar-default .navbar-text{color:#777}.navbar-default .navbar-nav>li>a{color:#777}.navbar-default .navbar-nav>li>a:hover,.navbar-default .navbar-nav>li>a:focus{color:#333;background-color:transparent}.navbar-default .navbar-nav>.active>a,.navbar-default .navbar-nav>.active>a:hover,.navbar-default .navbar-nav>.active>a:focus{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav>.disabled>a,.navbar-default .navbar-nav>.disabled>a:hover,.navbar-default .navbar-nav>.disabled>a:focus{color:#ccc;background-color:transparent}.navbar-default .navbar-nav>.open>a,.navbar-default .navbar-nav>.open>a:hover,.navbar-default .navbar-nav>.open>a:focus{color:#555;background-color:#e7e7e7}@media(max-width:767px){.navbar-default .navbar-nav .open .dropdown-menu>li>a{color:#777}.navbar-default .navbar-nav .open .dropdown-menu>li>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>li>a:focus{color:#333;background-color:transparent}.navbar-default .navbar-nav .open .dropdown-menu>.active>a,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:focus{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:hover,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:focus{color:#ccc;background-color:transparent}}.navbar-default .navbar-toggle{border-color:#ddd}.navbar-default .navbar-toggle:hover,.navbar-default .navbar-toggle:focus{background-color:#ddd}.navbar-default .navbar-toggle .icon-bar{background-color:#888}.navbar-default .navbar-collapse,.navbar-default .navbar-form{border-color:#e7e7e7}.navbar-default .navbar-link{color:#777}.navbar-default .navbar-link:hover{color:#333}.navbar-default .btn-link{color:#777}.navbar-default .btn-link:hover,.navbar-default .btn-link:focus{color:#333}.navbar-default .btn-link[disabled]:hover,fieldset[disabled] .navbar-default .btn-link:hover,.navbar-default .btn-link[disabled]:focus,fieldset[disabled] .navbar-default .btn-link:focus{color:#ccc}.navbar-inverse{background-color:#222;border-color:#080808}.navbar-inverse .navbar-brand{color:#9d9d9d}.navbar-inverse .navbar-brand:hover,.navbar-inverse .navbar-brand:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-text{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a:hover,.navbar-inverse .navbar-nav>li>a:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav>.active>a,.navbar-inverse .navbar-nav>.active>a:hover,.navbar-inverse .navbar-nav>.active>a:focus{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav>.disabled>a,.navbar-inverse .navbar-nav>.disabled>a:hover,.navbar-inverse .navbar-nav>.disabled>a:focus{color:#444;background-color:transparent}.navbar-inverse .navbar-nav>.open>a,.navbar-inverse .navbar-nav>.open>a:hover,.navbar-inverse .navbar-nav>.open>a:focus{color:#fff;background-color:#080808}@media(max-width:767px){.navbar-inverse .navbar-nav .open .dropdown-menu>.dropdown-header{border-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu .divider{background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:focus{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:focus{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:hover,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:focus{color:#444;background-color:transparent}}.navbar-inverse .navbar-toggle{border-color:#333}.navbar-inverse .navbar-toggle:hover,.navbar-inverse .navbar-toggle:focus{background-color:#333}.navbar-inverse .navbar-toggle .icon-bar{background-color:#fff}.navbar-inverse .navbar-collapse,.navbar-inverse .navbar-form{border-color:#101010}.navbar-inverse .navbar-link{color:#9d9d9d}.navbar-inverse .navbar-link:hover{color:#fff}.navbar-inverse .btn-link{color:#9d9d9d}.navbar-inverse .btn-link:hover,.navbar-inverse .btn-link:focus{color:#fff}.navbar-inverse .btn-link[disabled]:hover,fieldset[disabled] .navbar-inverse .btn-link:hover,.navbar-inverse .btn-link[disabled]:focus,fieldset[disabled] .navbar-inverse .btn-link:focus{color:#444}.breadcrumb{padding:8px 15px;margin-bottom:20px;list-style:none;background-color:#f5f5f5;border-radius:4px}.breadcrumb>li{display:inline-block}.breadcrumb>li+li:before{padding:0 5px;color:#ccc;content:"/\00a0"}.breadcrumb>.active{color:#777}.pagination{display:inline-block;padding-left:0;margin:20px 0;border-radius:4px}.pagination>li{display:inline}.pagination>li>a,.pagination>li>span{position:relative;float:left;padding:6px 12px;margin-left:-1px;line-height:1.42857143;color:#337ab7;text-decoration:none;background-color:#fff;border:1px solid #ddd}.pagination>li>a:hover,.pagination>li>span:hover,.pagination>li>a:focus,.pagination>li>span:focus{z-index:2;color:#23527c;background-color:#eee;border-color:#ddd}.pagination>li:first-child>a,.pagination>li:first-child>span{margin-left:0;border-top-left-radius:4px;border-bottom-left-radius:4px}.pagination>li:last-child>a,.pagination>li:last-child>span{border-top-right-radius:4px;border-bottom-right-radius:4px}.pagination>.active>a,.pagination>.active>span,.pagination>.active>a:hover,.pagination>.active>span:hover,.pagination>.active>a:focus,.pagination>.active>span:focus{z-index:3;color:#fff;cursor:default;background-color:#337ab7;border-color:#337ab7}.pagination>.disabled>span,.pagination>.disabled>span:hover,.pagination>.disabled>span:focus,.pagination>.disabled>a,.pagination>.disabled>a:hover,.pagination>.disabled>a:focus{color:#777;cursor:not-allowed;background-color:#fff;border-color:#ddd}.pagination-lg>li>a,.pagination-lg>li>span{padding:10px 16px;font-size:18px;line-height:1.3333333}.pagination-lg>li:first-child>a,.pagination-lg>li:first-child>span{border-top-left-radius:6px;border-bottom-left-radius:6px}.pagination-lg>li:last-child>a,.pagination-lg>li:last-child>span{border-top-right-radius:6px;border-bottom-right-radius:6px}.pagination-sm>li>a,.pagination-sm>li>span{padding:5px 10px;font-size:12px;line-height:1.5}.pagination-sm>li:first-child>a,.pagination-sm>li:first-child>span{border-top-left-radius:3px;border-bottom-left-radius:3px}.pagination-sm>li:last-child>a,.pagination-sm>li:last-child>span{border-top-right-radius:3px;border-bottom-right-radius:3px}.pager{padding-left:0;margin:20px 0;text-align:center;list-style:none}.pager li{display:inline}.pager li>a,.pager li>span{display:inline-block;padding:5px 14px;background-color:#fff;border:1px solid #ddd;border-radius:15px}.pager li>a:hover,.pager li>a:focus{text-decoration:none;background-color:#eee}.pager .next>a,.pager .next>span{float:right}.pager .previous>a,.pager .previous>span{float:left}.pager .disabled>a,.pager .disabled>a:hover,.pager .disabled>a:focus,.pager .disabled>span{color:#777;cursor:not-allowed;background-color:#fff}.label{display:inline;padding:.2em .6em .3em;font-size:75%;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25em}a.label:hover,a.label:focus{color:#fff;text-decoration:none;cursor:pointer}.label:empty{display:none}.btn .label{position:relative;top:-1px}.label-default{background-color:#777}.label-default[href]:hover,.label-default[href]:focus{background-color:#5e5e5e}.label-primary{background-color:#337ab7}.label-primary[href]:hover,.label-primary[href]:focus{background-color:#286090}.label-success{background-color:#5cb85c}.label-success[href]:hover,.label-success[href]:focus{background-color:#449d44}.label-info{background-color:#5bc0de}.label-info[href]:hover,.label-info[href]:focus{background-color:#31b0d5}.label-warning{background-color:#f0ad4e}.label-warning[href]:hover,.label-warning[href]:focus{background-color:#ec971f}.label-danger{background-color:#d9534f}.label-danger[href]:hover,.label-danger[href]:focus{background-color:#c9302c}.badge{display:inline-block;min-width:10px;padding:3px 7px;font-size:12px;font-weight:bold;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:middle;background-color:#777;border-radius:10px}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.btn-xs .badge,.btn-group-xs>.btn .badge{top:0;padding:1px 5px}a.badge:hover,a.badge:focus{color:#fff;text-decoration:none;cursor:pointer}.list-group-item.active>.badge,.nav-pills>.active>a>.badge{color:#337ab7;background-color:#fff}.list-group-item>.badge{float:right}.list-group-item>.badge+.badge{margin-right:5px}.nav-pills>li>a>.badge{margin-left:3px}.jumbotron{padding-top:30px;padding-bottom:30px;margin-bottom:30px;color:inherit;background-color:#eee}.jumbotron h1,.jumbotron .h1{color:inherit}.jumbotron p{margin-bottom:15px;font-size:21px;font-weight:200}.jumbotron>hr{border-top-color:#d5d5d5}.container .jumbotron,.container-fluid .jumbotron{padding-right:15px;padding-left:15px;border-radius:6px}.jumbotron .container{max-width:100%}@media screen and (min-width:768px){.jumbotron{padding-top:48px;padding-bottom:48px}.container .jumbotron,.container-fluid .jumbotron{padding-right:60px;padding-left:60px}.jumbotron h1,.jumbotron .h1{font-size:63px}}.thumbnail{display:block;padding:4px;margin-bottom:20px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:border .2s ease-in-out;-o-transition:border .2s ease-in-out;transition:border .2s ease-in-out}.thumbnail>img,.thumbnail a>img{margin-right:auto;margin-left:auto}a.thumbnail:hover,a.thumbnail:focus,a.thumbnail.active{border-color:#337ab7}.thumbnail .caption{padding:9px;color:#333}.alert{padding:15px;margin-bottom:20px;border:1px solid transparent;border-radius:4px}.alert h4{margin-top:0;color:inherit}.alert .alert-link{font-weight:bold}.alert>p,.alert>ul{margin-bottom:0}.alert>p+p{margin-top:5px}.alert-dismissable,.alert-dismissible{padding-right:35px}.alert-dismissable .close,.alert-dismissible .close{position:relative;top:-2px;right:-21px;color:inherit}.alert-success{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.alert-success hr{border-top-color:#c9e2b3}.alert-success .alert-link{color:#2b542c}.alert-info{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.alert-info hr{border-top-color:#a6e1ec}.alert-info .alert-link{color:#245269}.alert-warning{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.alert-warning hr{border-top-color:#f7e1b5}.alert-warning .alert-link{color:#66512c}.alert-danger{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.alert-danger hr{border-top-color:#e4b9c0}.alert-danger .alert-link{color:#843534}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}.progress{height:20px;margin-bottom:20px;overflow:hidden;background-color:#f5f5f5;border-radius:4px;-webkit-box-shadow:inset 0 1px 2px rgba(0,0,0,0.1);box-shadow:inset 0 1px 2px rgba(0,0,0,0.1)}.progress-bar{float:left;width:0;height:100%;font-size:12px;line-height:20px;color:#fff;text-align:center;background-color:#337ab7;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,0.15);box-shadow:inset 0 -1px 0 rgba(0,0,0,0.15);-webkit-transition:width .6s ease;-o-transition:width .6s ease;transition:width .6s ease}.progress-striped .progress-bar,.progress-bar-striped{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-size:40px 40px}.progress.active .progress-bar,.progress-bar.active{-webkit-animation:progress-bar-stripes 2s linear infinite;-o-animation:progress-bar-stripes 2s linear infinite;animation:progress-bar-stripes 2s linear infinite}.progress-bar-success{background-color:#5cb85c}.progress-striped .progress-bar-success{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-info{background-color:#5bc0de}.progress-striped .progress-bar-info{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-warning{background-color:#f0ad4e}.progress-striped .progress-bar-warning{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.progress-bar-danger{background-color:#d9534f}.progress-striped .progress-bar-danger{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,0.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,0.15) 50%,rgba(255,255,255,0.15) 75%,transparent 75%,transparent)}.media{margin-top:15px}.media:first-child{margin-top:0}.media,.media-body{overflow:hidden;zoom:1}.media-body{width:10000px}.media-object{display:block}.media-object.img-thumbnail{max-width:none}.media-right,.media>.pull-right{padding-left:10px}.media-left,.media>.pull-left{padding-right:10px}.media-left,.media-right,.media-body{display:table-cell;vertical-align:top}.media-middle{vertical-align:middle}.media-bottom{vertical-align:bottom}.media-heading{margin-top:0;margin-bottom:5px}.media-list{padding-left:0;list-style:none}.list-group{padding-left:0;margin-bottom:20px}.list-group-item{position:relative;display:block;padding:10px 15px;margin-bottom:-1px;background-color:#fff;border:1px solid #ddd}.list-group-item:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}.list-group-item.disabled,.list-group-item.disabled:hover,.list-group-item.disabled:focus{color:#777;cursor:not-allowed;background-color:#eee}.list-group-item.disabled .list-group-item-heading,.list-group-item.disabled:hover .list-group-item-heading,.list-group-item.disabled:focus .list-group-item-heading{color:inherit}.list-group-item.disabled .list-group-item-text,.list-group-item.disabled:hover .list-group-item-text,.list-group-item.disabled:focus .list-group-item-text{color:#777}.list-group-item.active,.list-group-item.active:hover,.list-group-item.active:focus{z-index:2;color:#fff;background-color:#337ab7;border-color:#337ab7}.list-group-item.active .list-group-item-heading,.list-group-item.active:hover .list-group-item-heading,.list-group-item.active:focus .list-group-item-heading,.list-group-item.active .list-group-item-heading>small,.list-group-item.active:hover .list-group-item-heading>small,.list-group-item.active:focus .list-group-item-heading>small,.list-group-item.active .list-group-item-heading>.small,.list-group-item.active:hover .list-group-item-heading>.small,.list-group-item.active:focus .list-group-item-heading>.small{color:inherit}.list-group-item.active .list-group-item-text,.list-group-item.active:hover .list-group-item-text,.list-group-item.active:focus .list-group-item-text{color:#c7ddef}a.list-group-item,button.list-group-item{color:#555}a.list-group-item .list-group-item-heading,button.list-group-item .list-group-item-heading{color:#333}a.list-group-item:hover,button.list-group-item:hover,a.list-group-item:focus,button.list-group-item:focus{color:#555;text-decoration:none;background-color:#f5f5f5}button.list-group-item{width:100%;text-align:left}.list-group-item-success{color:#3c763d;background-color:#dff0d8}a.list-group-item-success,button.list-group-item-success{color:#3c763d}a.list-group-item-success .list-group-item-heading,button.list-group-item-success .list-group-item-heading{color:inherit}a.list-group-item-success:hover,button.list-group-item-success:hover,a.list-group-item-success:focus,button.list-group-item-success:focus{color:#3c763d;background-color:#d0e9c6}a.list-group-item-success.active,button.list-group-item-success.active,a.list-group-item-success.active:hover,button.list-group-item-success.active:hover,a.list-group-item-success.active:focus,button.list-group-item-success.active:focus{color:#fff;background-color:#3c763d;border-color:#3c763d}.list-group-item-info{color:#31708f;background-color:#d9edf7}a.list-group-item-info,button.list-group-item-info{color:#31708f}a.list-group-item-info .list-group-item-heading,button.list-group-item-info .list-group-item-heading{color:inherit}a.list-group-item-info:hover,button.list-group-item-info:hover,a.list-group-item-info:focus,button.list-group-item-info:focus{color:#31708f;background-color:#c4e3f3}a.list-group-item-info.active,button.list-group-item-info.active,a.list-group-item-info.active:hover,button.list-group-item-info.active:hover,a.list-group-item-info.active:focus,button.list-group-item-info.active:focus{color:#fff;background-color:#31708f;border-color:#31708f}.list-group-item-warning{color:#8a6d3b;background-color:#fcf8e3}a.list-group-item-warning,button.list-group-item-warning{color:#8a6d3b}a.list-group-item-warning .list-group-item-heading,button.list-group-item-warning .list-group-item-heading{color:inherit}a.list-group-item-warning:hover,button.list-group-item-warning:hover,a.list-group-item-warning:focus,button.list-group-item-warning:focus{color:#8a6d3b;background-color:#faf2cc}a.list-group-item-warning.active,button.list-group-item-warning.active,a.list-group-item-warning.active:hover,button.list-group-item-warning.active:hover,a.list-group-item-warning.active:focus,button.list-group-item-warning.active:focus{color:#fff;background-color:#8a6d3b;border-color:#8a6d3b}.list-group-item-danger{color:#a94442;background-color:#f2dede}a.list-group-item-danger,button.list-group-item-danger{color:#a94442}a.list-group-item-danger .list-group-item-heading,button.list-group-item-danger .list-group-item-heading{color:inherit}a.list-group-item-danger:hover,button.list-group-item-danger:hover,a.list-group-item-danger:focus,button.list-group-item-danger:focus{color:#a94442;background-color:#ebcccc}a.list-group-item-danger.active,button.list-group-item-danger.active,a.list-group-item-danger.active:hover,button.list-group-item-danger.active:hover,a.list-group-item-danger.active:focus,button.list-group-item-danger.active:focus{color:#fff;background-color:#a94442;border-color:#a94442}.list-group-item-heading{margin-top:0;margin-bottom:5px}.list-group-item-text{margin-bottom:0;line-height:1.3}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,0.05);box-shadow:0 1px 1px rgba(0,0,0,0.05)}.panel-body{padding:15px}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel-heading>.dropdown .dropdown-toggle{color:inherit}.panel-title{margin-top:0;margin-bottom:0;font-size:16px;color:inherit}.panel-title>a,.panel-title>small,.panel-title>.small,.panel-title>small>a,.panel-title>.small>a{color:inherit}.panel-footer{padding:10px 15px;background-color:#f5f5f5;border-top:1px solid #ddd;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.list-group,.panel>.panel-collapse>.list-group{margin-bottom:0}.panel>.list-group .list-group-item,.panel>.panel-collapse>.list-group .list-group-item{border-width:1px 0;border-radius:0}.panel>.list-group:first-child .list-group-item:first-child,.panel>.panel-collapse>.list-group:first-child .list-group-item:first-child{border-top:0;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.list-group:last-child .list-group-item:last-child,.panel>.panel-collapse>.list-group:last-child .list-group-item:last-child{border-bottom:0;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.panel-heading+.panel-collapse>.list-group .list-group-item:first-child{border-top-left-radius:0;border-top-right-radius:0}.panel-heading+.list-group .list-group-item:first-child{border-top-width:0}.list-group+.panel-footer{border-top-width:0}.panel>.table,.panel>.table-responsive>.table,.panel>.panel-collapse>.table{margin-bottom:0}.panel>.table caption,.panel>.table-responsive>.table caption,.panel>.panel-collapse>.table caption{padding-right:15px;padding-left:15px}.panel>.table:first-child,.panel>.table-responsive:first-child>.table:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:first-child{border-top-left-radius:3px}.panel>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:last-child{border-top-right-radius:3px}.panel>.table:last-child,.panel>.table-responsive:last-child>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:first-child{border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:last-child{border-bottom-right-radius:3px}.panel>.panel-body+.table,.panel>.panel-body+.table-responsive,.panel>.table+.panel-body,.panel>.table-responsive+.panel-body{border-top:1px solid #ddd}.panel>.table>tbody:first-child>tr:first-child th,.panel>.table>tbody:first-child>tr:first-child td{border-top:0}.panel>.table-bordered,.panel>.table-responsive>.table-bordered{border:0}.panel>.table-bordered>thead>tr>th:first-child,.panel>.table-responsive>.table-bordered>thead>tr>th:first-child,.panel>.table-bordered>tbody>tr>th:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:first-child,.panel>.table-bordered>tfoot>tr>th:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:first-child,.panel>.table-bordered>thead>tr>td:first-child,.panel>.table-responsive>.table-bordered>thead>tr>td:first-child,.panel>.table-bordered>tbody>tr>td:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:first-child,.panel>.table-bordered>tfoot>tr>td:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:first-child{border-left:0}.panel>.table-bordered>thead>tr>th:last-child,.panel>.table-responsive>.table-bordered>thead>tr>th:last-child,.panel>.table-bordered>tbody>tr>th:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:last-child,.panel>.table-bordered>tfoot>tr>th:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:last-child,.panel>.table-bordered>thead>tr>td:last-child,.panel>.table-responsive>.table-bordered>thead>tr>td:last-child,.panel>.table-bordered>tbody>tr>td:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:last-child,.panel>.table-bordered>tfoot>tr>td:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:last-child{border-right:0}.panel>.table-bordered>thead>tr:first-child>td,.panel>.table-responsive>.table-bordered>thead>tr:first-child>td,.panel>.table-bordered>tbody>tr:first-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>td,.panel>.table-bordered>thead>tr:first-child>th,.panel>.table-responsive>.table-bordered>thead>tr:first-child>th,.panel>.table-bordered>tbody>tr:first-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>th{border-bottom:0}.panel>.table-bordered>tbody>tr:last-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>td,.panel>.table-bordered>tfoot>tr:last-child>td,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>td,.panel>.table-bordered>tbody>tr:last-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>th,.panel>.table-bordered>tfoot>tr:last-child>th,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}.panel>.table-responsive{margin-bottom:0;border:0}.panel-group{margin-bottom:20px}.panel-group .panel{margin-bottom:0;border-radius:4px}.panel-group .panel+.panel{margin-top:5px}.panel-group .panel-heading{border-bottom:0}.panel-group .panel-heading+.panel-collapse>.panel-body,.panel-group .panel-heading+.panel-collapse>.list-group{border-top:1px solid #ddd}.panel-group .panel-footer{border-top:0}.panel-group .panel-footer+.panel-collapse .panel-body{border-bottom:1px solid #ddd}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.panel-default>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ddd}.panel-default>.panel-heading .badge{color:#f5f5f5;background-color:#333}.panel-default>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ddd}.panel-primary{border-color:#337ab7}.panel-primary>.panel-heading{color:#fff;background-color:#337ab7;border-color:#337ab7}.panel-primary>.panel-heading+.panel-collapse>.panel-body{border-top-color:#337ab7}.panel-primary>.panel-heading .badge{color:#337ab7;background-color:#fff}.panel-primary>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#337ab7}.panel-success{border-color:#d6e9c6}.panel-success>.panel-heading{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.panel-success>.panel-heading+.panel-collapse>.panel-body{border-top-color:#d6e9c6}.panel-success>.panel-heading .badge{color:#dff0d8;background-color:#3c763d}.panel-success>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#d6e9c6}.panel-info{border-color:#bce8f1}.panel-info>.panel-heading{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.panel-info>.panel-heading+.panel-collapse>.panel-body{border-top-color:#bce8f1}.panel-info>.panel-heading .badge{color:#d9edf7;background-color:#31708f}.panel-info>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#bce8f1}.panel-warning{border-color:#faebcc}.panel-warning>.panel-heading{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.panel-warning>.panel-heading+.panel-collapse>.panel-body{border-top-color:#faebcc}.panel-warning>.panel-heading .badge{color:#fcf8e3;background-color:#8a6d3b}.panel-warning>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#faebcc}.panel-danger{border-color:#ebccd1}.panel-danger>.panel-heading{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.panel-danger>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ebccd1}.panel-danger>.panel-heading .badge{color:#f2dede;background-color:#a94442}.panel-danger>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ebccd1}.embed-responsive{position:relative;display:block;height:0;padding:0;overflow:hidden}.embed-responsive .embed-responsive-item,.embed-responsive iframe,.embed-responsive embed,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-16by9{padding-bottom:56.25%}.embed-responsive-4by3{padding-bottom:75%}.well{min-height:20px;padding:19px;margin-bottom:20px;background-color:#f5f5f5;border:1px solid #e3e3e3;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.05);box-shadow:inset 0 1px 1px rgba(0,0,0,0.05)}.well blockquote{border-color:#ddd;border-color:rgba(0,0,0,0.15)}.well-lg{padding:24px;border-radius:6px}.well-sm{padding:9px;border-radius:3px}.close{float:right;font-size:21px;font-weight:bold;line-height:1;color:#000;text-shadow:0 1px 0 #fff;filter:alpha(opacity=20);opacity:.2}.close:hover,.close:focus{color:#000;text-decoration:none;cursor:pointer;filter:alpha(opacity=50);opacity:.5}button.close{padding:0;cursor:pointer;background:transparent;border:0;-webkit-appearance:none;appearance:none}.modal-open{overflow:hidden}.modal{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1050;display:none;overflow:hidden;-webkit-overflow-scrolling:touch;outline:0}.modal.fade .modal-dialog{-webkit-transform:translate(0,-25%);-ms-transform:translate(0,-25%);-o-transform:translate(0,-25%);transform:translate(0,-25%);-webkit-transition:-webkit-transform .3s ease-out;-moz-transition:-moz-transform .3s ease-out;-o-transition:-o-transform .3s ease-out;transition:transform .3s ease-out}.modal.in .modal-dialog{-webkit-transform:translate(0,0);-ms-transform:translate(0,0);-o-transform:translate(0,0);transform:translate(0,0)}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal-dialog{position:relative;width:auto;margin:10px}.modal-content{position:relative;background-color:#fff;background-clip:padding-box;border:1px solid #999;border:1px solid rgba(0,0,0,0.2);border-radius:6px;-webkit-box-shadow:0 3px 9px rgba(0,0,0,0.5);box-shadow:0 3px 9px rgba(0,0,0,0.5);outline:0}.modal-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1040;background-color:#000}.modal-backdrop.fade{filter:alpha(opacity=0);opacity:0}.modal-backdrop.in{filter:alpha(opacity=50);opacity:.5}.modal-header{padding:15px;border-bottom:1px solid #e5e5e5}.modal-header .close{margin-top:-2px}.modal-title{margin:0;line-height:1.42857143}.modal-body{position:relative;padding:15px}.modal-footer{padding:15px;text-align:right;border-top:1px solid #e5e5e5}.modal-footer .btn+.btn{margin-bottom:0;margin-left:5px}.modal-footer .btn-group .btn+.btn{margin-left:-1px}.modal-footer .btn-block+.btn-block{margin-left:0}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media(min-width:768px){.modal-dialog{width:600px;margin:30px auto}.modal-content{-webkit-box-shadow:0 5px 15px rgba(0,0,0,0.5);box-shadow:0 5px 15px rgba(0,0,0,0.5)}.modal-sm{width:300px}}@media(min-width:992px){.modal-lg{width:900px}}.tooltip{position:absolute;z-index:1070;display:block;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-style:normal;font-weight:400;line-height:1.42857143;line-break:auto;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;word-wrap:normal;white-space:normal;font-size:12px;filter:alpha(opacity=0);opacity:0}.tooltip.in{filter:alpha(opacity=90);opacity:.9}.tooltip.top{padding:5px 0;margin-top:-3px}.tooltip.right{padding:0 5px;margin-left:3px}.tooltip.bottom{padding:5px 0;margin-top:3px}.tooltip.left{padding:0 5px;margin-left:-3px}.tooltip.top .tooltip-arrow{bottom:0;left:50%;margin-left:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-left .tooltip-arrow{right:5px;bottom:0;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-right .tooltip-arrow{bottom:0;left:5px;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.right .tooltip-arrow{top:50%;left:0;margin-top:-5px;border-width:5px 5px 5px 0;border-right-color:#000}.tooltip.left .tooltip-arrow{top:50%;right:0;margin-top:-5px;border-width:5px 0 5px 5px;border-left-color:#000}.tooltip.bottom .tooltip-arrow{top:0;left:50%;margin-left:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-left .tooltip-arrow{top:0;right:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-right .tooltip-arrow{top:0;left:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip-inner{max-width:200px;padding:3px 8px;color:#fff;text-align:center;background-color:#000;border-radius:4px}.tooltip-arrow{position:absolute;width:0;height:0;border-color:transparent;border-style:solid}.popover{position:absolute;top:0;left:0;z-index:1060;display:none;max-width:276px;padding:1px;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-style:normal;font-weight:400;line-height:1.42857143;line-break:auto;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;word-wrap:normal;white-space:normal;font-size:14px;background-color:#fff;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,0.2);border-radius:6px;-webkit-box-shadow:0 5px 10px rgba(0,0,0,0.2);box-shadow:0 5px 10px rgba(0,0,0,0.2)}.popover.top{margin-top:-10px}.popover.right{margin-left:10px}.popover.bottom{margin-top:10px}.popover.left{margin-left:-10px}.popover>.arrow{border-width:11px}.popover>.arrow,.popover>.arrow:after{position:absolute;display:block;width:0;height:0;border-color:transparent;border-style:solid}.popover>.arrow:after{content:"";border-width:10px}.popover.top>.arrow{bottom:-11px;left:50%;margin-left:-11px;border-top-color:#999;border-top-color:rgba(0,0,0,0.25);border-bottom-width:0}.popover.top>.arrow:after{bottom:1px;margin-left:-10px;content:" ";border-top-color:#fff;border-bottom-width:0}.popover.right>.arrow{top:50%;left:-11px;margin-top:-11px;border-right-color:#999;border-right-color:rgba(0,0,0,0.25);border-left-width:0}.popover.right>.arrow:after{bottom:-10px;left:1px;content:" ";border-right-color:#fff;border-left-width:0}.popover.bottom>.arrow{top:-11px;left:50%;margin-left:-11px;border-top-width:0;border-bottom-color:#999;border-bottom-color:rgba(0,0,0,0.25)}.popover.bottom>.arrow:after{top:1px;margin-left:-10px;content:" ";border-top-width:0;border-bottom-color:#fff}.popover.left>.arrow{top:50%;right:-11px;margin-top:-11px;border-right-width:0;border-left-color:#999;border-left-color:rgba(0,0,0,0.25)}.popover.left>.arrow:after{right:1px;bottom:-10px;content:" ";border-right-width:0;border-left-color:#fff}.popover-title{padding:8px 14px;margin:0;font-size:14px;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-radius:5px 5px 0 0}.popover-content{padding:9px 14px}.carousel{position:relative}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner>.item{position:relative;display:none;-webkit-transition:.6s ease-in-out left;-o-transition:.6s ease-in-out left;transition:.6s ease-in-out left}.carousel-inner>.item>img,.carousel-inner>.item>a>img{line-height:1}@media all and (transform-3d),(-webkit-transform-3d){.carousel-inner>.item{-webkit-transition:-webkit-transform .6s ease-in-out;-moz-transition:-moz-transform .6s ease-in-out;-o-transition:-o-transform .6s ease-in-out;transition:transform .6s ease-in-out;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;backface-visibility:hidden;-webkit-perspective:1000px;-moz-perspective:1000px;perspective:1000px}.carousel-inner>.item.next,.carousel-inner>.item.active.right{-webkit-transform:translate3d(100%,0,0);transform:translate3d(100%,0,0);left:0}.carousel-inner>.item.prev,.carousel-inner>.item.active.left{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0);left:0}.carousel-inner>.item.next.left,.carousel-inner>.item.prev.right,.carousel-inner>.item.active{-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0);left:0}}.carousel-inner>.active,.carousel-inner>.next,.carousel-inner>.prev{display:block}.carousel-inner>.active{left:0}.carousel-inner>.next,.carousel-inner>.prev{position:absolute;top:0;width:100%}.carousel-inner>.next{left:100%}.carousel-inner>.prev{left:-100%}.carousel-inner>.next.left,.carousel-inner>.prev.right{left:0}.carousel-inner>.active.left{left:-100%}.carousel-inner>.active.right{left:100%}.carousel-control{position:absolute;top:0;bottom:0;left:0;width:15%;font-size:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,0.6);background-color:rgba(0,0,0,0);filter:alpha(opacity=50);opacity:.5}.carousel-control.left{background-image:-webkit-linear-gradient(left,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);background-image:linear-gradient(to right,rgba(0,0,0,0.5) 0,rgba(0,0,0,0.0001) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#80000000',endColorstr='#00000000',GradientType=1);background-repeat:repeat-x}.carousel-control.right{right:0;left:auto;background-image:-webkit-linear-gradient(left,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);background-image:linear-gradient(to right,rgba(0,0,0,0.0001) 0,rgba(0,0,0,0.5) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#00000000',endColorstr='#80000000',GradientType=1);background-repeat:repeat-x}.carousel-control:hover,.carousel-control:focus{color:#fff;text-decoration:none;outline:0;filter:alpha(opacity=90);opacity:.9}.carousel-control .icon-prev,.carousel-control .icon-next,.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right{position:absolute;top:50%;z-index:5;display:inline-block;margin-top:-10px}.carousel-control .icon-prev,.carousel-control .glyphicon-chevron-left{left:50%;margin-left:-10px}.carousel-control .icon-next,.carousel-control .glyphicon-chevron-right{right:50%;margin-right:-10px}.carousel-control .icon-prev,.carousel-control .icon-next{width:20px;height:20px;font-family:serif;line-height:1}.carousel-control .icon-prev:before{content:"\2039"}.carousel-control .icon-next:before{content:"\203a"}.carousel-indicators{position:absolute;bottom:10px;left:50%;z-index:15;width:60%;padding-left:0;margin-left:-30%;text-align:center;list-style:none}.carousel-indicators li{display:inline-block;width:10px;height:10px;margin:1px;text-indent:-999px;cursor:pointer;background-color:#000 \9;background-color:rgba(0,0,0,0);border:1px solid #fff;border-radius:10px}.carousel-indicators .active{width:12px;height:12px;margin:0;background-color:#fff}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,0.6)}.carousel-caption .btn{text-shadow:none}@media screen and (min-width:768px){.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-prev,.carousel-control .icon-next{width:30px;height:30px;margin-top:-10px;font-size:30px}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{margin-left:-10px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{margin-right:-10px}.carousel-caption{right:20%;left:20%;padding-bottom:30px}.carousel-indicators{bottom:20px}}.clearfix:before,.clearfix:after,.dl-horizontal dd:before,.dl-horizontal dd:after,.container:before,.container:after,.container-fluid:before,.container-fluid:after,.row:before,.row:after,.form-horizontal .form-group:before,.form-horizontal .form-group:after,.btn-toolbar:before,.btn-toolbar:after,.btn-group-vertical>.btn-group:before,.btn-group-vertical>.btn-group:after,.nav:before,.nav:after,.navbar:before,.navbar:after,.navbar-header:before,.navbar-header:after,.navbar-collapse:before,.navbar-collapse:after,.pager:before,.pager:after,.panel-body:before,.panel-body:after,.modal-header:before,.modal-header:after,.modal-footer:before,.modal-footer:after{display:table;content:" "}.clearfix:after,.dl-horizontal dd:after,.container:after,.container-fluid:after,.row:after,.form-horizontal .form-group:after,.btn-toolbar:after,.btn-group-vertical>.btn-group:after,.nav:after,.navbar:after,.navbar-header:after,.navbar-collapse:after,.pager:after,.panel-body:after,.modal-header:after,.modal-footer:after{clear:both}.center-block{display:block;margin-right:auto;margin-left:auto}.pull-right{float:right!important}.pull-left{float:left!important}.hide{display:none!important}.show{display:block!important}.invisible{visibility:hidden}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.hidden{display:none!important}.affix{position:fixed}@-ms-viewport{width:device-width}.visible-xs,.visible-sm,.visible-md,.visible-lg{display:none!important}.visible-xs-block,.visible-xs-inline,.visible-xs-inline-block,.visible-sm-block,.visible-sm-inline,.visible-sm-inline-block,.visible-md-block,.visible-md-inline,.visible-md-inline-block,.visible-lg-block,.visible-lg-inline,.visible-lg-inline-block{display:none!important}@media(max-width:767px){.visible-xs{display:block!important}table.visible-xs{display:table!important}tr.visible-xs{display:table-row!important}th.visible-xs,td.visible-xs{display:table-cell!important}}@media(max-width:767px){.visible-xs-block{display:block!important}}@media(max-width:767px){.visible-xs-inline{display:inline!important}}@media(max-width:767px){.visible-xs-inline-block{display:inline-block!important}}@media(min-width:768px) and (max-width:991px){.visible-sm{display:block!important}table.visible-sm{display:table!important}tr.visible-sm{display:table-row!important}th.visible-sm,td.visible-sm{display:table-cell!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-block{display:block!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-inline{display:inline!important}}@media(min-width:768px) and (max-width:991px){.visible-sm-inline-block{display:inline-block!important}}@media(min-width:992px) and (max-width:1199px){.visible-md{display:block!important}table.visible-md{display:table!important}tr.visible-md{display:table-row!important}th.visible-md,td.visible-md{display:table-cell!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-block{display:block!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-inline{display:inline!important}}@media(min-width:992px) and (max-width:1199px){.visible-md-inline-block{display:inline-block!important}}@media(min-width:1200px){.visible-lg{display:block!important}table.visible-lg{display:table!important}tr.visible-lg{display:table-row!important}th.visible-lg,td.visible-lg{display:table-cell!important}}@media(min-width:1200px){.visible-lg-block{display:block!important}}@media(min-width:1200px){.visible-lg-inline{display:inline!important}}@media(min-width:1200px){.visible-lg-inline-block{display:inline-block!important}}@media(max-width:767px){.hidden-xs{display:none!important}}@media(min-width:768px) and (max-width:991px){.hidden-sm{display:none!important}}@media(min-width:992px) and (max-width:1199px){.hidden-md{display:none!important}}@media(min-width:1200px){.hidden-lg{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:block!important}table.visible-print{display:table!important}tr.visible-print{display:table-row!important}th.visible-print,td.visible-print{display:table-cell!important}}.visible-print-block{display:none!important}@media print{.visible-print-block{display:block!important}}.visible-print-inline{display:none!important}@media print{.visible-print-inline{display:inline!important}}.visible-print-inline-block{display:none!important}@media print{.visible-print-inline-block{display:inline-block!important}}@media print{.hidden-print{display:none!important}}</style>
+<script>if(typeof jQuery==="undefined"){throw new Error("Bootstrap's JavaScript requires jQuery")}+function($){"use strict";var version=$.fn.jquery.split(" ")[0].split(".");if(version[0]<2&&version[1]<9||version[0]==1&&version[1]==9&&version[2]<1||version[0]>3){throw new Error("Bootstrap's JavaScript requires jQuery version 1.9.1 or higher, but lower than version 4")}}(jQuery);+function($){"use strict";function transitionEnd(){var el=document.createElement("bootstrap");var transEndEventNames={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"};for(var name in transEndEventNames){if(el.style[name]!==undefined){return{end:transEndEventNames[name]}}}return false}$.fn.emulateTransitionEnd=function(duration){var called=false;var $el=this;$(this).one("bsTransitionEnd",function(){called=true});var callback=function(){if(!called)$($el).trigger($.support.transition.end)};setTimeout(callback,duration);return this};$(function(){$.support.transition=transitionEnd();if(!$.support.transition)return;$.event.special.bsTransitionEnd={bindType:$.support.transition.end,delegateType:$.support.transition.end,handle:function(e){if($(e.target).is(this))return e.handleObj.handler.apply(this,arguments)}}})}(jQuery);+function($){"use strict";var dismiss='[data-dismiss="alert"]';var Alert=function(el){$(el).on("click",dismiss,this.close)};Alert.VERSION="3.4.1";Alert.TRANSITION_DURATION=150;Alert.prototype.close=function(e){var $this=$(this);var selector=$this.attr("data-target");if(!selector){selector=$this.attr("href");selector=selector&&selector.replace(/.*(?=#[^\s]*$)/,"")}selector=selector==="#"?[]:selector;var $parent=$(document).find(selector);if(e)e.preventDefault();if(!$parent.length){$parent=$this.closest(".alert")}$parent.trigger(e=$.Event("close.bs.alert"));if(e.isDefaultPrevented())return;$parent.removeClass("in");function removeElement(){$parent.detach().trigger("closed.bs.alert").remove()}$.support.transition&&$parent.hasClass("fade")?$parent.one("bsTransitionEnd",removeElement).emulateTransitionEnd(Alert.TRANSITION_DURATION):removeElement()};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.alert");if(!data)$this.data("bs.alert",data=new Alert(this));if(typeof option=="string")data[option].call($this)})}var old=$.fn.alert;$.fn.alert=Plugin;$.fn.alert.Constructor=Alert;$.fn.alert.noConflict=function(){$.fn.alert=old;return this};$(document).on("click.bs.alert.data-api",dismiss,Alert.prototype.close)}(jQuery);+function($){"use strict";var Button=function(element,options){this.$element=$(element);this.options=$.extend({},Button.DEFAULTS,options);this.isLoading=false};Button.VERSION="3.4.1";Button.DEFAULTS={loadingText:"loading..."};Button.prototype.setState=function(state){var d="disabled";var $el=this.$element;var val=$el.is("input")?"val":"html";var data=$el.data();state+="Text";if(data.resetText==null)$el.data("resetText",$el[val]());setTimeout($.proxy(function(){$el[val](data[state]==null?this.options[state]:data[state]);if(state=="loadingText"){this.isLoading=true;$el.addClass(d).attr(d,d).prop(d,true)}else if(this.isLoading){this.isLoading=false;$el.removeClass(d).removeAttr(d).prop(d,false)}},this),0)};Button.prototype.toggle=function(){var changed=true;var $parent=this.$element.closest('[data-toggle="buttons"]');if($parent.length){var $input=this.$element.find("input");if($input.prop("type")=="radio"){if($input.prop("checked"))changed=false;$parent.find(".active").removeClass("active");this.$element.addClass("active")}else if($input.prop("type")=="checkbox"){if($input.prop("checked")!==this.$element.hasClass("active"))changed=false;this.$element.toggleClass("active")}$input.prop("checked",this.$element.hasClass("active"));if(changed)$input.trigger("change")}else{this.$element.attr("aria-pressed",!this.$element.hasClass("active"));this.$element.toggleClass("active")}};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.button");var options=typeof option=="object"&&option;if(!data)$this.data("bs.button",data=new Button(this,options));if(option=="toggle")data.toggle();else if(option)data.setState(option)})}var old=$.fn.button;$.fn.button=Plugin;$.fn.button.Constructor=Button;$.fn.button.noConflict=function(){$.fn.button=old;return this};$(document).on("click.bs.button.data-api",'[data-toggle^="button"]',function(e){var $btn=$(e.target).closest(".btn");Plugin.call($btn,"toggle");if(!$(e.target).is('input[type="radio"], input[type="checkbox"]')){e.preventDefault();if($btn.is("input,button"))$btn.trigger("focus");else $btn.find("input:visible,button:visible").first().trigger("focus")}}).on("focus.bs.button.data-api blur.bs.button.data-api",'[data-toggle^="button"]',function(e){$(e.target).closest(".btn").toggleClass("focus",/^focus(in)?$/.test(e.type))})}(jQuery);+function($){"use strict";var Carousel=function(element,options){this.$element=$(element);this.$indicators=this.$element.find(".carousel-indicators");this.options=options;this.paused=null;this.sliding=null;this.interval=null;this.$active=null;this.$items=null;this.options.keyboard&&this.$element.on("keydown.bs.carousel",$.proxy(this.keydown,this));this.options.pause=="hover"&&!("ontouchstart"in document.documentElement)&&this.$element.on("mouseenter.bs.carousel",$.proxy(this.pause,this)).on("mouseleave.bs.carousel",$.proxy(this.cycle,this))};Carousel.VERSION="3.4.1";Carousel.TRANSITION_DURATION=600;Carousel.DEFAULTS={interval:5e3,pause:"hover",wrap:true,keyboard:true};Carousel.prototype.keydown=function(e){if(/input|textarea/i.test(e.target.tagName))return;switch(e.which){case 37:this.prev();break;case 39:this.next();break;default:return}e.preventDefault()};Carousel.prototype.cycle=function(e){e||(this.paused=false);this.interval&&clearInterval(this.interval);this.options.interval&&!this.paused&&(this.interval=setInterval($.proxy(this.next,this),this.options.interval));return this};Carousel.prototype.getItemIndex=function(item){this.$items=item.parent().children(".item");return this.$items.index(item||this.$active)};Carousel.prototype.getItemForDirection=function(direction,active){var activeIndex=this.getItemIndex(active);var willWrap=direction=="prev"&&activeIndex===0||direction=="next"&&activeIndex==this.$items.length-1;if(willWrap&&!this.options.wrap)return active;var delta=direction=="prev"?-1:1;var itemIndex=(activeIndex+delta)%this.$items.length;return this.$items.eq(itemIndex)};Carousel.prototype.to=function(pos){var that=this;var activeIndex=this.getItemIndex(this.$active=this.$element.find(".item.active"));if(pos>this.$items.length-1||pos<0)return;if(this.sliding)return this.$element.one("slid.bs.carousel",function(){that.to(pos)});if(activeIndex==pos)return this.pause().cycle();return this.slide(pos>activeIndex?"next":"prev",this.$items.eq(pos))};Carousel.prototype.pause=function(e){e||(this.paused=true);if(this.$element.find(".next, .prev").length&&$.support.transition){this.$element.trigger($.support.transition.end);this.cycle(true)}this.interval=clearInterval(this.interval);return this};Carousel.prototype.next=function(){if(this.sliding)return;return this.slide("next")};Carousel.prototype.prev=function(){if(this.sliding)return;return this.slide("prev")};Carousel.prototype.slide=function(type,next){var $active=this.$element.find(".item.active");var $next=next||this.getItemForDirection(type,$active);var isCycling=this.interval;var direction=type=="next"?"left":"right";var that=this;if($next.hasClass("active"))return this.sliding=false;var relatedTarget=$next[0];var slideEvent=$.Event("slide.bs.carousel",{relatedTarget:relatedTarget,direction:direction});this.$element.trigger(slideEvent);if(slideEvent.isDefaultPrevented())return;this.sliding=true;isCycling&&this.pause();if(this.$indicators.length){this.$indicators.find(".active").removeClass("active");var $nextIndicator=$(this.$indicators.children()[this.getItemIndex($next)]);$nextIndicator&&$nextIndicator.addClass("active")}var slidEvent=$.Event("slid.bs.carousel",{relatedTarget:relatedTarget,direction:direction});if($.support.transition&&this.$element.hasClass("slide")){$next.addClass(type);if(typeof $next==="object"&&$next.length){$next[0].offsetWidth}$active.addClass(direction);$next.addClass(direction);$active.one("bsTransitionEnd",function(){$next.removeClass([type,direction].join(" ")).addClass("active");$active.removeClass(["active",direction].join(" "));that.sliding=false;setTimeout(function(){that.$element.trigger(slidEvent)},0)}).emulateTransitionEnd(Carousel.TRANSITION_DURATION)}else{$active.removeClass("active");$next.addClass("active");this.sliding=false;this.$element.trigger(slidEvent)}isCycling&&this.cycle();return this};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.carousel");var options=$.extend({},Carousel.DEFAULTS,$this.data(),typeof option=="object"&&option);var action=typeof option=="string"?option:options.slide;if(!data)$this.data("bs.carousel",data=new Carousel(this,options));if(typeof option=="number")data.to(option);else if(action)data[action]();else if(options.interval)data.pause().cycle()})}var old=$.fn.carousel;$.fn.carousel=Plugin;$.fn.carousel.Constructor=Carousel;$.fn.carousel.noConflict=function(){$.fn.carousel=old;return this};var clickHandler=function(e){var $this=$(this);var href=$this.attr("href");if(href){href=href.replace(/.*(?=#[^\s]+$)/,"")}var target=$this.attr("data-target")||href;var $target=$(document).find(target);if(!$target.hasClass("carousel"))return;var options=$.extend({},$target.data(),$this.data());var slideIndex=$this.attr("data-slide-to");if(slideIndex)options.interval=false;Plugin.call($target,options);if(slideIndex){$target.data("bs.carousel").to(slideIndex)}e.preventDefault()};$(document).on("click.bs.carousel.data-api","[data-slide]",clickHandler).on("click.bs.carousel.data-api","[data-slide-to]",clickHandler);$(window).on("load",function(){$('[data-ride="carousel"]').each(function(){var $carousel=$(this);Plugin.call($carousel,$carousel.data())})})}(jQuery);+function($){"use strict";var Collapse=function(element,options){this.$element=$(element);this.options=$.extend({},Collapse.DEFAULTS,options);this.$trigger=$('[data-toggle="collapse"][href="#'+element.id+'"],'+'[data-toggle="collapse"][data-target="#'+element.id+'"]');this.transitioning=null;if(this.options.parent){this.$parent=this.getParent()}else{this.addAriaAndCollapsedClass(this.$element,this.$trigger)}if(this.options.toggle)this.toggle()};Collapse.VERSION="3.4.1";Collapse.TRANSITION_DURATION=350;Collapse.DEFAULTS={toggle:true};Collapse.prototype.dimension=function(){var hasWidth=this.$element.hasClass("width");return hasWidth?"width":"height"};Collapse.prototype.show=function(){if(this.transitioning||this.$element.hasClass("in"))return;var activesData;var actives=this.$parent&&this.$parent.children(".panel").children(".in, .collapsing");if(actives&&actives.length){activesData=actives.data("bs.collapse");if(activesData&&activesData.transitioning)return}var startEvent=$.Event("show.bs.collapse");this.$element.trigger(startEvent);if(startEvent.isDefaultPrevented())return;if(actives&&actives.length){Plugin.call(actives,"hide");activesData||actives.data("bs.collapse",null)}var dimension=this.dimension();this.$element.removeClass("collapse").addClass("collapsing")[dimension](0).attr("aria-expanded",true);this.$trigger.removeClass("collapsed").attr("aria-expanded",true);this.transitioning=1;var complete=function(){this.$element.removeClass("collapsing").addClass("collapse in")[dimension]("");this.transitioning=0;this.$element.trigger("shown.bs.collapse")};if(!$.support.transition)return complete.call(this);var scrollSize=$.camelCase(["scroll",dimension].join("-"));this.$element.one("bsTransitionEnd",$.proxy(complete,this)).emulateTransitionEnd(Collapse.TRANSITION_DURATION)[dimension](this.$element[0][scrollSize])};Collapse.prototype.hide=function(){if(this.transitioning||!this.$element.hasClass("in"))return;var startEvent=$.Event("hide.bs.collapse");this.$element.trigger(startEvent);if(startEvent.isDefaultPrevented())return;var dimension=this.dimension();this.$element[dimension](this.$element[dimension]())[0].offsetHeight;this.$element.addClass("collapsing").removeClass("collapse in").attr("aria-expanded",false);this.$trigger.addClass("collapsed").attr("aria-expanded",false);this.transitioning=1;var complete=function(){this.transitioning=0;this.$element.removeClass("collapsing").addClass("collapse").trigger("hidden.bs.collapse")};if(!$.support.transition)return complete.call(this);this.$element[dimension](0).one("bsTransitionEnd",$.proxy(complete,this)).emulateTransitionEnd(Collapse.TRANSITION_DURATION)};Collapse.prototype.toggle=function(){this[this.$element.hasClass("in")?"hide":"show"]()};Collapse.prototype.getParent=function(){return $(document).find(this.options.parent).find('[data-toggle="collapse"][data-parent="'+this.options.parent+'"]').each($.proxy(function(i,element){var $element=$(element);this.addAriaAndCollapsedClass(getTargetFromTrigger($element),$element)},this)).end()};Collapse.prototype.addAriaAndCollapsedClass=function($element,$trigger){var isOpen=$element.hasClass("in");$element.attr("aria-expanded",isOpen);$trigger.toggleClass("collapsed",!isOpen).attr("aria-expanded",isOpen)};function getTargetFromTrigger($trigger){var href;var target=$trigger.attr("data-target")||(href=$trigger.attr("href"))&&href.replace(/.*(?=#[^\s]+$)/,"");return $(document).find(target)}function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.collapse");var options=$.extend({},Collapse.DEFAULTS,$this.data(),typeof option=="object"&&option);if(!data&&options.toggle&&/show|hide/.test(option))options.toggle=false;if(!data)$this.data("bs.collapse",data=new Collapse(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.collapse;$.fn.collapse=Plugin;$.fn.collapse.Constructor=Collapse;$.fn.collapse.noConflict=function(){$.fn.collapse=old;return this};$(document).on("click.bs.collapse.data-api",'[data-toggle="collapse"]',function(e){var $this=$(this);if(!$this.attr("data-target"))e.preventDefault();var $target=getTargetFromTrigger($this);var data=$target.data("bs.collapse");var option=data?"toggle":$this.data();Plugin.call($target,option)})}(jQuery);+function($){"use strict";var backdrop=".dropdown-backdrop";var toggle='[data-toggle="dropdown"]';var Dropdown=function(element){$(element).on("click.bs.dropdown",this.toggle)};Dropdown.VERSION="3.4.1";function getParent($this){var selector=$this.attr("data-target");if(!selector){selector=$this.attr("href");selector=selector&&/#[A-Za-z]/.test(selector)&&selector.replace(/.*(?=#[^\s]*$)/,"")}var $parent=selector!=="#"?$(document).find(selector):null;return $parent&&$parent.length?$parent:$this.parent()}function clearMenus(e){if(e&&e.which===3)return;$(backdrop).remove();$(toggle).each(function(){var $this=$(this);var $parent=getParent($this);var relatedTarget={relatedTarget:this};if(!$parent.hasClass("open"))return;if(e&&e.type=="click"&&/input|textarea/i.test(e.target.tagName)&&$.contains($parent[0],e.target))return;$parent.trigger(e=$.Event("hide.bs.dropdown",relatedTarget));if(e.isDefaultPrevented())return;$this.attr("aria-expanded","false");$parent.removeClass("open").trigger($.Event("hidden.bs.dropdown",relatedTarget))})}Dropdown.prototype.toggle=function(e){var $this=$(this);if($this.is(".disabled, :disabled"))return;var $parent=getParent($this);var isActive=$parent.hasClass("open");clearMenus();if(!isActive){if("ontouchstart"in document.documentElement&&!$parent.closest(".navbar-nav").length){$(document.createElement("div")).addClass("dropdown-backdrop").insertAfter($(this)).on("click",clearMenus)}var relatedTarget={relatedTarget:this};$parent.trigger(e=$.Event("show.bs.dropdown",relatedTarget));if(e.isDefaultPrevented())return;$this.trigger("focus").attr("aria-expanded","true");$parent.toggleClass("open").trigger($.Event("shown.bs.dropdown",relatedTarget))}return false};Dropdown.prototype.keydown=function(e){if(!/(38|40|27|32)/.test(e.which)||/input|textarea/i.test(e.target.tagName))return;var $this=$(this);e.preventDefault();e.stopPropagation();if($this.is(".disabled, :disabled"))return;var $parent=getParent($this);var isActive=$parent.hasClass("open");if(!isActive&&e.which!=27||isActive&&e.which==27){if(e.which==27)$parent.find(toggle).trigger("focus");return $this.trigger("click")}var desc=" li:not(.disabled):visible a";var $items=$parent.find(".dropdown-menu"+desc);if(!$items.length)return;var index=$items.index(e.target);if(e.which==38&&index>0)index--;if(e.which==40&&index<$items.length-1)index++;if(!~index)index=0;$items.eq(index).trigger("focus")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.dropdown");if(!data)$this.data("bs.dropdown",data=new Dropdown(this));if(typeof option=="string")data[option].call($this)})}var old=$.fn.dropdown;$.fn.dropdown=Plugin;$.fn.dropdown.Constructor=Dropdown;$.fn.dropdown.noConflict=function(){$.fn.dropdown=old;return this};$(document).on("click.bs.dropdown.data-api",clearMenus).on("click.bs.dropdown.data-api",".dropdown form",function(e){e.stopPropagation()}).on("click.bs.dropdown.data-api",toggle,Dropdown.prototype.toggle).on("keydown.bs.dropdown.data-api",toggle,Dropdown.prototype.keydown).on("keydown.bs.dropdown.data-api",".dropdown-menu",Dropdown.prototype.keydown)}(jQuery);+function($){"use strict";var Modal=function(element,options){this.options=options;this.$body=$(document.body);this.$element=$(element);this.$dialog=this.$element.find(".modal-dialog");this.$backdrop=null;this.isShown=null;this.originalBodyPad=null;this.scrollbarWidth=0;this.ignoreBackdropClick=false;this.fixedContent=".navbar-fixed-top, .navbar-fixed-bottom";if(this.options.remote){this.$element.find(".modal-content").load(this.options.remote,$.proxy(function(){this.$element.trigger("loaded.bs.modal")},this))}};Modal.VERSION="3.4.1";Modal.TRANSITION_DURATION=300;Modal.BACKDROP_TRANSITION_DURATION=150;Modal.DEFAULTS={backdrop:true,keyboard:true,show:true};Modal.prototype.toggle=function(_relatedTarget){return this.isShown?this.hide():this.show(_relatedTarget)};Modal.prototype.show=function(_relatedTarget){var that=this;var e=$.Event("show.bs.modal",{relatedTarget:_relatedTarget});this.$element.trigger(e);if(this.isShown||e.isDefaultPrevented())return;this.isShown=true;this.checkScrollbar();this.setScrollbar();this.$body.addClass("modal-open");this.escape();this.resize();this.$element.on("click.dismiss.bs.modal",'[data-dismiss="modal"]',$.proxy(this.hide,this));this.$dialog.on("mousedown.dismiss.bs.modal",function(){that.$element.one("mouseup.dismiss.bs.modal",function(e){if($(e.target).is(that.$element))that.ignoreBackdropClick=true})});this.backdrop(function(){var transition=$.support.transition&&that.$element.hasClass("fade");if(!that.$element.parent().length){that.$element.appendTo(that.$body)}that.$element.show().scrollTop(0);that.adjustDialog();if(transition){that.$element[0].offsetWidth}that.$element.addClass("in");that.enforceFocus();var e=$.Event("shown.bs.modal",{relatedTarget:_relatedTarget});transition?that.$dialog.one("bsTransitionEnd",function(){that.$element.trigger("focus").trigger(e)}).emulateTransitionEnd(Modal.TRANSITION_DURATION):that.$element.trigger("focus").trigger(e)})};Modal.prototype.hide=function(e){if(e)e.preventDefault();e=$.Event("hide.bs.modal");this.$element.trigger(e);if(!this.isShown||e.isDefaultPrevented())return;this.isShown=false;this.escape();this.resize();$(document).off("focusin.bs.modal");this.$element.removeClass("in").off("click.dismiss.bs.modal").off("mouseup.dismiss.bs.modal");this.$dialog.off("mousedown.dismiss.bs.modal");$.support.transition&&this.$element.hasClass("fade")?this.$element.one("bsTransitionEnd",$.proxy(this.hideModal,this)).emulateTransitionEnd(Modal.TRANSITION_DURATION):this.hideModal()};Modal.prototype.enforceFocus=function(){$(document).off("focusin.bs.modal").on("focusin.bs.modal",$.proxy(function(e){if(document!==e.target&&this.$element[0]!==e.target&&!this.$element.has(e.target).length){this.$element.trigger("focus")}},this))};Modal.prototype.escape=function(){if(this.isShown&&this.options.keyboard){this.$element.on("keydown.dismiss.bs.modal",$.proxy(function(e){e.which==27&&this.hide()},this))}else if(!this.isShown){this.$element.off("keydown.dismiss.bs.modal")}};Modal.prototype.resize=function(){if(this.isShown){$(window).on("resize.bs.modal",$.proxy(this.handleUpdate,this))}else{$(window).off("resize.bs.modal")}};Modal.prototype.hideModal=function(){var that=this;this.$element.hide();this.backdrop(function(){that.$body.removeClass("modal-open");that.resetAdjustments();that.resetScrollbar();that.$element.trigger("hidden.bs.modal")})};Modal.prototype.removeBackdrop=function(){this.$backdrop&&this.$backdrop.remove();this.$backdrop=null};Modal.prototype.backdrop=function(callback){var that=this;var animate=this.$element.hasClass("fade")?"fade":"";if(this.isShown&&this.options.backdrop){var doAnimate=$.support.transition&&animate;this.$backdrop=$(document.createElement("div")).addClass("modal-backdrop "+animate).appendTo(this.$body);this.$element.on("click.dismiss.bs.modal",$.proxy(function(e){if(this.ignoreBackdropClick){this.ignoreBackdropClick=false;return}if(e.target!==e.currentTarget)return;this.options.backdrop=="static"?this.$element[0].focus():this.hide()},this));if(doAnimate)this.$backdrop[0].offsetWidth;this.$backdrop.addClass("in");if(!callback)return;doAnimate?this.$backdrop.one("bsTransitionEnd",callback).emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION):callback()}else if(!this.isShown&&this.$backdrop){this.$backdrop.removeClass("in");var callbackRemove=function(){that.removeBackdrop();callback&&callback()};$.support.transition&&this.$element.hasClass("fade")?this.$backdrop.one("bsTransitionEnd",callbackRemove).emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION):callbackRemove()}else if(callback){callback()}};Modal.prototype.handleUpdate=function(){this.adjustDialog()};Modal.prototype.adjustDialog=function(){var modalIsOverflowing=this.$element[0].scrollHeight>document.documentElement.clientHeight;this.$element.css({paddingLeft:!this.bodyIsOverflowing&&modalIsOverflowing?this.scrollbarWidth:"",paddingRight:this.bodyIsOverflowing&&!modalIsOverflowing?this.scrollbarWidth:""})};Modal.prototype.resetAdjustments=function(){this.$element.css({paddingLeft:"",paddingRight:""})};Modal.prototype.checkScrollbar=function(){var fullWindowWidth=window.innerWidth;if(!fullWindowWidth){var documentElementRect=document.documentElement.getBoundingClientRect();fullWindowWidth=documentElementRect.right-Math.abs(documentElementRect.left)}this.bodyIsOverflowing=document.body.clientWidth<fullWindowWidth;this.scrollbarWidth=this.measureScrollbar()};Modal.prototype.setScrollbar=function(){var bodyPad=parseInt(this.$body.css("padding-right")||0,10);this.originalBodyPad=document.body.style.paddingRight||"";var scrollbarWidth=this.scrollbarWidth;if(this.bodyIsOverflowing){this.$body.css("padding-right",bodyPad+scrollbarWidth);$(this.fixedContent).each(function(index,element){var actualPadding=element.style.paddingRight;var calculatedPadding=$(element).css("padding-right");$(element).data("padding-right",actualPadding).css("padding-right",parseFloat(calculatedPadding)+scrollbarWidth+"px")})}};Modal.prototype.resetScrollbar=function(){this.$body.css("padding-right",this.originalBodyPad);$(this.fixedContent).each(function(index,element){var padding=$(element).data("padding-right");$(element).removeData("padding-right");element.style.paddingRight=padding?padding:""})};Modal.prototype.measureScrollbar=function(){var scrollDiv=document.createElement("div");scrollDiv.className="modal-scrollbar-measure";this.$body.append(scrollDiv);var scrollbarWidth=scrollDiv.offsetWidth-scrollDiv.clientWidth;this.$body[0].removeChild(scrollDiv);return scrollbarWidth};function Plugin(option,_relatedTarget){return this.each(function(){var $this=$(this);var data=$this.data("bs.modal");var options=$.extend({},Modal.DEFAULTS,$this.data(),typeof option=="object"&&option);if(!data)$this.data("bs.modal",data=new Modal(this,options));if(typeof option=="string")data[option](_relatedTarget);else if(options.show)data.show(_relatedTarget)})}var old=$.fn.modal;$.fn.modal=Plugin;$.fn.modal.Constructor=Modal;$.fn.modal.noConflict=function(){$.fn.modal=old;return this};$(document).on("click.bs.modal.data-api",'[data-toggle="modal"]',function(e){var $this=$(this);var href=$this.attr("href");var target=$this.attr("data-target")||href&&href.replace(/.*(?=#[^\s]+$)/,"");var $target=$(document).find(target);var option=$target.data("bs.modal")?"toggle":$.extend({remote:!/#/.test(href)&&href},$target.data(),$this.data());if($this.is("a"))e.preventDefault();$target.one("show.bs.modal",function(showEvent){if(showEvent.isDefaultPrevented())return;$target.one("hidden.bs.modal",function(){$this.is(":visible")&&$this.trigger("focus")})});Plugin.call($target,option,this)})}(jQuery);+function($){"use strict";var DISALLOWED_ATTRIBUTES=["sanitize","whiteList","sanitizeFn"];var uriAttrs=["background","cite","href","itemtype","longdesc","poster","src","xlink:href"];var ARIA_ATTRIBUTE_PATTERN=/^aria-[\w-]*$/i;var DefaultWhitelist={"*":["class","dir","id","lang","role",ARIA_ATTRIBUTE_PATTERN],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],div:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]};var SAFE_URL_PATTERN=/^(?:(?:https?|mailto|ftp|tel|file):|[^&:/?#]*(?:[/?#]|$))/gi;var DATA_URL_PATTERN=/^data:(?:image\/(?:bmp|gif|jpeg|jpg|png|tiff|webp)|video\/(?:mpeg|mp4|ogg|webm)|audio\/(?:mp3|oga|ogg|opus));base64,[a-z0-9+/]+=*$/i;function allowedAttribute(attr,allowedAttributeList){var attrName=attr.nodeName.toLowerCase();if($.inArray(attrName,allowedAttributeList)!==-1){if($.inArray(attrName,uriAttrs)!==-1){return Boolean(attr.nodeValue.match(SAFE_URL_PATTERN)||attr.nodeValue.match(DATA_URL_PATTERN))}return true}var regExp=$(allowedAttributeList).filter(function(index,value){return value instanceof RegExp});for(var i=0,l=regExp.length;i<l;i++){if(attrName.match(regExp[i])){return true}}return false}function sanitizeHtml(unsafeHtml,whiteList,sanitizeFn){if(unsafeHtml.length===0){return unsafeHtml}if(sanitizeFn&&typeof sanitizeFn==="function"){return sanitizeFn(unsafeHtml)}if(!document.implementation||!document.implementation.createHTMLDocument){return unsafeHtml}var createdDocument=document.implementation.createHTMLDocument("sanitization");createdDocument.body.innerHTML=unsafeHtml;var whitelistKeys=$.map(whiteList,function(el,i){return i});var elements=$(createdDocument.body).find("*");for(var i=0,len=elements.length;i<len;i++){var el=elements[i];var elName=el.nodeName.toLowerCase();if($.inArray(elName,whitelistKeys)===-1){el.parentNode.removeChild(el);continue}var attributeList=$.map(el.attributes,function(el){return el});var whitelistedAttributes=[].concat(whiteList["*"]||[],whiteList[elName]||[]);for(var j=0,len2=attributeList.length;j<len2;j++){if(!allowedAttribute(attributeList[j],whitelistedAttributes)){el.removeAttribute(attributeList[j].nodeName)}}}return createdDocument.body.innerHTML}var Tooltip=function(element,options){this.type=null;this.options=null;this.enabled=null;this.timeout=null;this.hoverState=null;this.$element=null;this.inState=null;this.init("tooltip",element,options)};Tooltip.VERSION="3.4.1";Tooltip.TRANSITION_DURATION=150;Tooltip.DEFAULTS={animation:true,placement:"top",selector:false,template:'<div class="tooltip" role="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>',trigger:"hover focus",title:"",delay:0,html:false,container:false,viewport:{selector:"body",padding:0},sanitize:true,sanitizeFn:null,whiteList:DefaultWhitelist};Tooltip.prototype.init=function(type,element,options){this.enabled=true;this.type=type;this.$element=$(element);this.options=this.getOptions(options);this.$viewport=this.options.viewport&&$(document).find($.isFunction(this.options.viewport)?this.options.viewport.call(this,this.$element):this.options.viewport.selector||this.options.viewport);this.inState={click:false,hover:false,focus:false};if(this.$element[0]instanceof document.constructor&&!this.options.selector){throw new Error("`selector` option must be specified when initializing "+this.type+" on the window.document object!")}var triggers=this.options.trigger.split(" ");for(var i=triggers.length;i--;){var trigger=triggers[i];if(trigger=="click"){this.$element.on("click."+this.type,this.options.selector,$.proxy(this.toggle,this))}else if(trigger!="manual"){var eventIn=trigger=="hover"?"mouseenter":"focusin";var eventOut=trigger=="hover"?"mouseleave":"focusout";this.$element.on(eventIn+"."+this.type,this.options.selector,$.proxy(this.enter,this));this.$element.on(eventOut+"."+this.type,this.options.selector,$.proxy(this.leave,this))}}this.options.selector?this._options=$.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()};Tooltip.prototype.getDefaults=function(){return Tooltip.DEFAULTS};Tooltip.prototype.getOptions=function(options){var dataAttributes=this.$element.data();for(var dataAttr in dataAttributes){if(dataAttributes.hasOwnProperty(dataAttr)&&$.inArray(dataAttr,DISALLOWED_ATTRIBUTES)!==-1){delete dataAttributes[dataAttr]}}options=$.extend({},this.getDefaults(),dataAttributes,options);if(options.delay&&typeof options.delay=="number"){options.delay={show:options.delay,hide:options.delay}}if(options.sanitize){options.template=sanitizeHtml(options.template,options.whiteList,options.sanitizeFn)}return options};Tooltip.prototype.getDelegateOptions=function(){var options={};var defaults=this.getDefaults();this._options&&$.each(this._options,function(key,value){if(defaults[key]!=value)options[key]=value});return options};Tooltip.prototype.enter=function(obj){var self=obj instanceof this.constructor?obj:$(obj.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(obj.currentTarget,this.getDelegateOptions());$(obj.currentTarget).data("bs."+this.type,self)}if(obj instanceof $.Event){self.inState[obj.type=="focusin"?"focus":"hover"]=true}if(self.tip().hasClass("in")||self.hoverState=="in"){self.hoverState="in";return}clearTimeout(self.timeout);self.hoverState="in";if(!self.options.delay||!self.options.delay.show)return self.show();self.timeout=setTimeout(function(){if(self.hoverState=="in")self.show()},self.options.delay.show)};Tooltip.prototype.isInStateTrue=function(){for(var key in this.inState){if(this.inState[key])return true}return false};Tooltip.prototype.leave=function(obj){var self=obj instanceof this.constructor?obj:$(obj.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(obj.currentTarget,this.getDelegateOptions());$(obj.currentTarget).data("bs."+this.type,self)}if(obj instanceof $.Event){self.inState[obj.type=="focusout"?"focus":"hover"]=false}if(self.isInStateTrue())return;clearTimeout(self.timeout);self.hoverState="out";if(!self.options.delay||!self.options.delay.hide)return self.hide();self.timeout=setTimeout(function(){if(self.hoverState=="out")self.hide()},self.options.delay.hide)};Tooltip.prototype.show=function(){var e=$.Event("show.bs."+this.type);if(this.hasContent()&&this.enabled){this.$element.trigger(e);var inDom=$.contains(this.$element[0].ownerDocument.documentElement,this.$element[0]);if(e.isDefaultPrevented()||!inDom)return;var that=this;var $tip=this.tip();var tipId=this.getUID(this.type);this.setContent();$tip.attr("id",tipId);this.$element.attr("aria-describedby",tipId);if(this.options.animation)$tip.addClass("fade");var placement=typeof this.options.placement=="function"?this.options.placement.call(this,$tip[0],this.$element[0]):this.options.placement;var autoToken=/\s?auto?\s?/i;var autoPlace=autoToken.test(placement);if(autoPlace)placement=placement.replace(autoToken,"")||"top";$tip.detach().css({top:0,left:0,display:"block"}).addClass(placement).data("bs."+this.type,this);this.options.container?$tip.appendTo($(document).find(this.options.container)):$tip.insertAfter(this.$element);this.$element.trigger("inserted.bs."+this.type);var pos=this.getPosition();var actualWidth=$tip[0].offsetWidth;var actualHeight=$tip[0].offsetHeight;if(autoPlace){var orgPlacement=placement;var viewportDim=this.getPosition(this.$viewport);placement=placement=="bottom"&&pos.bottom+actualHeight>viewportDim.bottom?"top":placement=="top"&&pos.top-actualHeight<viewportDim.top?"bottom":placement=="right"&&pos.right+actualWidth>viewportDim.width?"left":placement=="left"&&pos.left-actualWidth<viewportDim.left?"right":placement;$tip.removeClass(orgPlacement).addClass(placement)}var calculatedOffset=this.getCalculatedOffset(placement,pos,actualWidth,actualHeight);this.applyPlacement(calculatedOffset,placement);var complete=function(){var prevHoverState=that.hoverState;that.$element.trigger("shown.bs."+that.type);that.hoverState=null;if(prevHoverState=="out")that.leave(that)};$.support.transition&&this.$tip.hasClass("fade")?$tip.one("bsTransitionEnd",complete).emulateTransitionEnd(Tooltip.TRANSITION_DURATION):complete()}};Tooltip.prototype.applyPlacement=function(offset,placement){var $tip=this.tip();var width=$tip[0].offsetWidth;var height=$tip[0].offsetHeight;var marginTop=parseInt($tip.css("margin-top"),10);var marginLeft=parseInt($tip.css("margin-left"),10);if(isNaN(marginTop))marginTop=0;if(isNaN(marginLeft))marginLeft=0;offset.top+=marginTop;offset.left+=marginLeft;$.offset.setOffset($tip[0],$.extend({using:function(props){$tip.css({top:Math.round(props.top),left:Math.round(props.left)})}},offset),0);$tip.addClass("in");var actualWidth=$tip[0].offsetWidth;var actualHeight=$tip[0].offsetHeight;if(placement=="top"&&actualHeight!=height){offset.top=offset.top+height-actualHeight}var delta=this.getViewportAdjustedDelta(placement,offset,actualWidth,actualHeight);if(delta.left)offset.left+=delta.left;else offset.top+=delta.top;var isVertical=/top|bottom/.test(placement);var arrowDelta=isVertical?delta.left*2-width+actualWidth:delta.top*2-height+actualHeight;var arrowOffsetPosition=isVertical?"offsetWidth":"offsetHeight";$tip.offset(offset);this.replaceArrow(arrowDelta,$tip[0][arrowOffsetPosition],isVertical)};Tooltip.prototype.replaceArrow=function(delta,dimension,isVertical){this.arrow().css(isVertical?"left":"top",50*(1-delta/dimension)+"%").css(isVertical?"top":"left","")};Tooltip.prototype.setContent=function(){var $tip=this.tip();var title=this.getTitle();if(this.options.html){if(this.options.sanitize){title=sanitizeHtml(title,this.options.whiteList,this.options.sanitizeFn)}$tip.find(".tooltip-inner").html(title)}else{$tip.find(".tooltip-inner").text(title)}$tip.removeClass("fade in top bottom left right")};Tooltip.prototype.hide=function(callback){var that=this;var $tip=$(this.$tip);var e=$.Event("hide.bs."+this.type);function complete(){if(that.hoverState!="in")$tip.detach();if(that.$element){that.$element.removeAttr("aria-describedby").trigger("hidden.bs."+that.type)}callback&&callback()}this.$element.trigger(e);if(e.isDefaultPrevented())return;$tip.removeClass("in");$.support.transition&&$tip.hasClass("fade")?$tip.one("bsTransitionEnd",complete).emulateTransitionEnd(Tooltip.TRANSITION_DURATION):complete();this.hoverState=null;return this};Tooltip.prototype.fixTitle=function(){var $e=this.$element;if($e.attr("title")||typeof $e.attr("data-original-title")!="string"){$e.attr("data-original-title",$e.attr("title")||"").attr("title","")}};Tooltip.prototype.hasContent=function(){return this.getTitle()};Tooltip.prototype.getPosition=function($element){$element=$element||this.$element;var el=$element[0];var isBody=el.tagName=="BODY";var elRect=el.getBoundingClientRect();if(elRect.width==null){elRect=$.extend({},elRect,{width:elRect.right-elRect.left,height:elRect.bottom-elRect.top})}var isSvg=window.SVGElement&&el instanceof window.SVGElement;var elOffset=isBody?{top:0,left:0}:isSvg?null:$element.offset();var scroll={scroll:isBody?document.documentElement.scrollTop||document.body.scrollTop:$element.scrollTop()};var outerDims=isBody?{width:$(window).width(),height:$(window).height()}:null;return $.extend({},elRect,scroll,outerDims,elOffset)};Tooltip.prototype.getCalculatedOffset=function(placement,pos,actualWidth,actualHeight){return placement=="bottom"?{top:pos.top+pos.height,left:pos.left+pos.width/2-actualWidth/2}:placement=="top"?{top:pos.top-actualHeight,left:pos.left+pos.width/2-actualWidth/2}:placement=="left"?{top:pos.top+pos.height/2-actualHeight/2,left:pos.left-actualWidth}:{top:pos.top+pos.height/2-actualHeight/2,left:pos.left+pos.width}};Tooltip.prototype.getViewportAdjustedDelta=function(placement,pos,actualWidth,actualHeight){var delta={top:0,left:0};if(!this.$viewport)return delta;var viewportPadding=this.options.viewport&&this.options.viewport.padding||0;var viewportDimensions=this.getPosition(this.$viewport);if(/right|left/.test(placement)){var topEdgeOffset=pos.top-viewportPadding-viewportDimensions.scroll;var bottomEdgeOffset=pos.top+viewportPadding-viewportDimensions.scroll+actualHeight;if(topEdgeOffset<viewportDimensions.top){delta.top=viewportDimensions.top-topEdgeOffset}else if(bottomEdgeOffset>viewportDimensions.top+viewportDimensions.height){delta.top=viewportDimensions.top+viewportDimensions.height-bottomEdgeOffset}}else{var leftEdgeOffset=pos.left-viewportPadding;var rightEdgeOffset=pos.left+viewportPadding+actualWidth;if(leftEdgeOffset<viewportDimensions.left){delta.left=viewportDimensions.left-leftEdgeOffset}else if(rightEdgeOffset>viewportDimensions.right){delta.left=viewportDimensions.left+viewportDimensions.width-rightEdgeOffset}}return delta};Tooltip.prototype.getTitle=function(){var title;var $e=this.$element;var o=this.options;title=$e.attr("data-original-title")||(typeof o.title=="function"?o.title.call($e[0]):o.title);return title};Tooltip.prototype.getUID=function(prefix){do{prefix+=~~(Math.random()*1e6)}while(document.getElementById(prefix));return prefix};Tooltip.prototype.tip=function(){if(!this.$tip){this.$tip=$(this.options.template);if(this.$tip.length!=1){throw new Error(this.type+" `template` option must consist of exactly 1 top-level element!")}}return this.$tip};Tooltip.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")};Tooltip.prototype.enable=function(){this.enabled=true};Tooltip.prototype.disable=function(){this.enabled=false};Tooltip.prototype.toggleEnabled=function(){this.enabled=!this.enabled};Tooltip.prototype.toggle=function(e){var self=this;if(e){self=$(e.currentTarget).data("bs."+this.type);if(!self){self=new this.constructor(e.currentTarget,this.getDelegateOptions());$(e.currentTarget).data("bs."+this.type,self)}}if(e){self.inState.click=!self.inState.click;if(self.isInStateTrue())self.enter(self);else self.leave(self)}else{self.tip().hasClass("in")?self.leave(self):self.enter(self)}};Tooltip.prototype.destroy=function(){var that=this;clearTimeout(this.timeout);this.hide(function(){that.$element.off("."+that.type).removeData("bs."+that.type);if(that.$tip){that.$tip.detach()}that.$tip=null;that.$arrow=null;that.$viewport=null;that.$element=null})};Tooltip.prototype.sanitizeHtml=function(unsafeHtml){return sanitizeHtml(unsafeHtml,this.options.whiteList,this.options.sanitizeFn)};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.tooltip");var options=typeof option=="object"&&option;if(!data&&/destroy|hide/.test(option))return;if(!data)$this.data("bs.tooltip",data=new Tooltip(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.tooltip;$.fn.tooltip=Plugin;$.fn.tooltip.Constructor=Tooltip;$.fn.tooltip.noConflict=function(){$.fn.tooltip=old;return this}}(jQuery);+function($){"use strict";var Popover=function(element,options){this.init("popover",element,options)};if(!$.fn.tooltip)throw new Error("Popover requires tooltip.js");Popover.VERSION="3.4.1";Popover.DEFAULTS=$.extend({},$.fn.tooltip.Constructor.DEFAULTS,{placement:"right",trigger:"click",content:"",template:'<div class="popover" role="tooltip"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'});Popover.prototype=$.extend({},$.fn.tooltip.Constructor.prototype);Popover.prototype.constructor=Popover;Popover.prototype.getDefaults=function(){return Popover.DEFAULTS};Popover.prototype.setContent=function(){var $tip=this.tip();var title=this.getTitle();var content=this.getContent();if(this.options.html){var typeContent=typeof content;if(this.options.sanitize){title=this.sanitizeHtml(title);if(typeContent==="string"){content=this.sanitizeHtml(content)}}$tip.find(".popover-title").html(title);$tip.find(".popover-content").children().detach().end()[typeContent==="string"?"html":"append"](content)}else{$tip.find(".popover-title").text(title);$tip.find(".popover-content").children().detach().end().text(content)}$tip.removeClass("fade top bottom left right in");if(!$tip.find(".popover-title").html())$tip.find(".popover-title").hide()};Popover.prototype.hasContent=function(){return this.getTitle()||this.getContent()};Popover.prototype.getContent=function(){var $e=this.$element;var o=this.options;return $e.attr("data-content")||(typeof o.content=="function"?o.content.call($e[0]):o.content)};Popover.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".arrow")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.popover");var options=typeof option=="object"&&option;if(!data&&/destroy|hide/.test(option))return;if(!data)$this.data("bs.popover",data=new Popover(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.popover;$.fn.popover=Plugin;$.fn.popover.Constructor=Popover;$.fn.popover.noConflict=function(){$.fn.popover=old;return this}}(jQuery);+function($){"use strict";function ScrollSpy(element,options){this.$body=$(document.body);this.$scrollElement=$(element).is(document.body)?$(window):$(element);this.options=$.extend({},ScrollSpy.DEFAULTS,options);this.selector=(this.options.target||"")+" .nav li > a";this.offsets=[];this.targets=[];this.activeTarget=null;this.scrollHeight=0;this.$scrollElement.on("scroll.bs.scrollspy",$.proxy(this.process,this));this.refresh();this.process()}ScrollSpy.VERSION="3.4.1";ScrollSpy.DEFAULTS={offset:10};ScrollSpy.prototype.getScrollHeight=function(){return this.$scrollElement[0].scrollHeight||Math.max(this.$body[0].scrollHeight,document.documentElement.scrollHeight)};ScrollSpy.prototype.refresh=function(){var that=this;var offsetMethod="offset";var offsetBase=0;this.offsets=[];this.targets=[];this.scrollHeight=this.getScrollHeight();if(!$.isWindow(this.$scrollElement[0])){offsetMethod="position";offsetBase=this.$scrollElement.scrollTop()}this.$body.find(this.selector).map(function(){var $el=$(this);var href=$el.data("target")||$el.attr("href");var $href=/^#./.test(href)&&$(href);return $href&&$href.length&&$href.is(":visible")&&[[$href[offsetMethod]().top+offsetBase,href]]||null}).sort(function(a,b){return a[0]-b[0]}).each(function(){that.offsets.push(this[0]);that.targets.push(this[1])})};ScrollSpy.prototype.process=function(){var scrollTop=this.$scrollElement.scrollTop()+this.options.offset;var scrollHeight=this.getScrollHeight();var maxScroll=this.options.offset+scrollHeight-this.$scrollElement.height();var offsets=this.offsets;var targets=this.targets;var activeTarget=this.activeTarget;var i;if(this.scrollHeight!=scrollHeight){this.refresh()}if(scrollTop>=maxScroll){return activeTarget!=(i=targets[targets.length-1])&&this.activate(i)}if(activeTarget&&scrollTop<offsets[0]){this.activeTarget=null;return this.clear()}for(i=offsets.length;i--;){activeTarget!=targets[i]&&scrollTop>=offsets[i]&&(offsets[i+1]===undefined||scrollTop<offsets[i+1])&&this.activate(targets[i])}};ScrollSpy.prototype.activate=function(target){this.activeTarget=target;this.clear();var selector=this.selector+'[data-target="'+target+'"],'+this.selector+'[href="'+target+'"]';var active=$(selector).parents("li").addClass("active");if(active.parent(".dropdown-menu").length){active=active.closest("li.dropdown").addClass("active")}active.trigger("activate.bs.scrollspy")};ScrollSpy.prototype.clear=function(){$(this.selector).parentsUntil(this.options.target,".active").removeClass("active")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.scrollspy");var options=typeof option=="object"&&option;if(!data)$this.data("bs.scrollspy",data=new ScrollSpy(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.scrollspy;$.fn.scrollspy=Plugin;$.fn.scrollspy.Constructor=ScrollSpy;$.fn.scrollspy.noConflict=function(){$.fn.scrollspy=old;return this};$(window).on("load.bs.scrollspy.data-api",function(){$('[data-spy="scroll"]').each(function(){var $spy=$(this);Plugin.call($spy,$spy.data())})})}(jQuery);+function($){"use strict";var Tab=function(element){this.element=$(element)};Tab.VERSION="3.4.1";Tab.TRANSITION_DURATION=150;Tab.prototype.show=function(){var $this=this.element;var $ul=$this.closest("ul:not(.dropdown-menu)");var selector=$this.data("target");if(!selector){selector=$this.attr("href");selector=selector&&selector.replace(/.*(?=#[^\s]*$)/,"")}if($this.parent("li").hasClass("active"))return;var $previous=$ul.find(".active:last a");var hideEvent=$.Event("hide.bs.tab",{relatedTarget:$this[0]});var showEvent=$.Event("show.bs.tab",{relatedTarget:$previous[0]});$previous.trigger(hideEvent);$this.trigger(showEvent);if(showEvent.isDefaultPrevented()||hideEvent.isDefaultPrevented())return;var $target=$(document).find(selector);this.activate($this.closest("li"),$ul);this.activate($target,$target.parent(),function(){$previous.trigger({type:"hidden.bs.tab",relatedTarget:$this[0]});$this.trigger({type:"shown.bs.tab",relatedTarget:$previous[0]})})};Tab.prototype.activate=function(element,container,callback){var $active=container.find("> .active");var transition=callback&&$.support.transition&&($active.length&&$active.hasClass("fade")||!!container.find("> .fade").length);function next(){$active.removeClass("active").find("> .dropdown-menu > .active").removeClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",false);element.addClass("active").find('[data-toggle="tab"]').attr("aria-expanded",true);if(transition){element[0].offsetWidth;element.addClass("in")}else{element.removeClass("fade")}if(element.parent(".dropdown-menu").length){element.closest("li.dropdown").addClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",true)}callback&&callback()}$active.length&&transition?$active.one("bsTransitionEnd",next).emulateTransitionEnd(Tab.TRANSITION_DURATION):next();$active.removeClass("in")};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.tab");if(!data)$this.data("bs.tab",data=new Tab(this));if(typeof option=="string")data[option]()})}var old=$.fn.tab;$.fn.tab=Plugin;$.fn.tab.Constructor=Tab;$.fn.tab.noConflict=function(){$.fn.tab=old;return this};var clickHandler=function(e){e.preventDefault();Plugin.call($(this),"show")};$(document).on("click.bs.tab.data-api",'[data-toggle="tab"]',clickHandler).on("click.bs.tab.data-api",'[data-toggle="pill"]',clickHandler)}(jQuery);+function($){"use strict";var Affix=function(element,options){this.options=$.extend({},Affix.DEFAULTS,options);var target=this.options.target===Affix.DEFAULTS.target?$(this.options.target):$(document).find(this.options.target);this.$target=target.on("scroll.bs.affix.data-api",$.proxy(this.checkPosition,this)).on("click.bs.affix.data-api",$.proxy(this.checkPositionWithEventLoop,this));this.$element=$(element);this.affixed=null;this.unpin=null;this.pinnedOffset=null;this.checkPosition()};Affix.VERSION="3.4.1";Affix.RESET="affix affix-top affix-bottom";Affix.DEFAULTS={offset:0,target:window};Affix.prototype.getState=function(scrollHeight,height,offsetTop,offsetBottom){var scrollTop=this.$target.scrollTop();var position=this.$element.offset();var targetHeight=this.$target.height();if(offsetTop!=null&&this.affixed=="top")return scrollTop<offsetTop?"top":false;if(this.affixed=="bottom"){if(offsetTop!=null)return scrollTop+this.unpin<=position.top?false:"bottom";return scrollTop+targetHeight<=scrollHeight-offsetBottom?false:"bottom"}var initializing=this.affixed==null;var colliderTop=initializing?scrollTop:position.top;var colliderHeight=initializing?targetHeight:height;if(offsetTop!=null&&scrollTop<=offsetTop)return"top";if(offsetBottom!=null&&colliderTop+colliderHeight>=scrollHeight-offsetBottom)return"bottom";return false};Affix.prototype.getPinnedOffset=function(){if(this.pinnedOffset)return this.pinnedOffset;this.$element.removeClass(Affix.RESET).addClass("affix");var scrollTop=this.$target.scrollTop();var position=this.$element.offset();return this.pinnedOffset=position.top-scrollTop};Affix.prototype.checkPositionWithEventLoop=function(){setTimeout($.proxy(this.checkPosition,this),1)};Affix.prototype.checkPosition=function(){if(!this.$element.is(":visible"))return;var height=this.$element.height();var offset=this.options.offset;var offsetTop=offset.top;var offsetBottom=offset.bottom;var scrollHeight=Math.max($(document).height(),$(document.body).height());if(typeof offset!="object")offsetBottom=offsetTop=offset;if(typeof offsetTop=="function")offsetTop=offset.top(this.$element);if(typeof offsetBottom=="function")offsetBottom=offset.bottom(this.$element);var affix=this.getState(scrollHeight,height,offsetTop,offsetBottom);if(this.affixed!=affix){if(this.unpin!=null)this.$element.css("top","");var affixType="affix"+(affix?"-"+affix:"");var e=$.Event(affixType+".bs.affix");this.$element.trigger(e);if(e.isDefaultPrevented())return;this.affixed=affix;this.unpin=affix=="bottom"?this.getPinnedOffset():null;this.$element.removeClass(Affix.RESET).addClass(affixType).trigger(affixType.replace("affix","affixed")+".bs.affix")}if(affix=="bottom"){this.$element.offset({top:scrollHeight-height-offsetBottom})}};function Plugin(option){return this.each(function(){var $this=$(this);var data=$this.data("bs.affix");var options=typeof option=="object"&&option;if(!data)$this.data("bs.affix",data=new Affix(this,options));if(typeof option=="string")data[option]()})}var old=$.fn.affix;$.fn.affix=Plugin;$.fn.affix.Constructor=Affix;$.fn.affix.noConflict=function(){$.fn.affix=old;return this};$(window).on("load",function(){$('[data-spy="affix"]').each(function(){var $spy=$(this);var data=$spy.data();data.offset=data.offset||{};if(data.offsetBottom!=null)data.offset.bottom=data.offsetBottom;if(data.offsetTop!=null)data.offset.top=data.offsetTop;Plugin.call($spy,data)})})}(jQuery);</script>
+<script>!function(e, i) {
+    var l, m, t, n = e.html5 || {}, r = /^<|^(?:button|map|select|textarea|object|iframe|option|optgroup)$/i, a = /^(?:a|b|code|div|fieldset|h1|h2|h3|h4|h5|h6|i|label|li|ol|p|q|span|strong|style|table|tbody|td|th|tr|ul)$/i, o = "_html5shiv", c = 0, s = {};
+    try {
+        var d = i.createElement("a");
+        d.innerHTML = "<xyz></xyz>", l = "hidden" in d, m = 1 == d.childNodes.length || (i.createElement("a"), 
+        void 0 === (t = i.createDocumentFragment()).cloneNode || void 0 === t.createDocumentFragment || void 0 === t.createElement);
+    } catch (e) {
+        m = l = !0;
+    }
+    function u() {
+        var e = p.elements;
+        return "string" == typeof e ? e.split(" ") : e;
+    }
+    function h(e) {
+        var t = s[e[o]];
+        return t || (t = {}, c++, e[o] = c, s[c] = t), t;
+    }
+    function f(e, t, n) {
+        return t = t || i, m ? t.createElement(e) : !(t = (n = n || h(t)).cache[e] ? n.cache[e].cloneNode() : a.test(e) ? (n.cache[e] = n.createElem(e)).cloneNode() : n.createElem(e)).canHaveChildren || r.test(e) || t.tagUrn ? t : n.frag.appendChild(t);
+    }
+    function g(e) {
+        var t, n, r, a, o, c = h(e = e || i);
+        return !p.shivCSS || l || c.hasCSS || (c.hasCSS = (n = "article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}mark{background:#FF0;color:#000}template{display:none}", 
+        r = (t = e).createElement("p"), t = t.getElementsByTagName("head")[0] || t.documentElement, 
+        r.innerHTML = "x<style>" + n + "</style>", !!t.insertBefore(r.lastChild, t.firstChild))), 
+        m || (a = e, (o = c).cache || (o.cache = {}, o.createElem = a.createElement, 
+        o.createFrag = a.createDocumentFragment, o.frag = o.createFrag()), a.createElement = function(e) {
+            return p.shivMethods ? f(e, a, o) : o.createElem(e);
+        }, a.createDocumentFragment = Function("h,f", "return function(){var n=f.cloneNode(),c=n.createElement;h.shivMethods&&(" + u().join().replace(/[\w\-:]+/g, function(e) {
+            return o.createElem(e), o.frag.createElement(e), 'c("' + e + '")';
+        }) + ");return n}")(p, o.frag)), e;
+    }
+    var p = {
+        elements: n.elements || "abbr article aside audio bdi canvas data datalist details dialog figcaption figure footer header hgroup main mark meter nav output picture progress section summary template time video",
+        version: "3.7.3",
+        shivCSS: !1 !== n.shivCSS,
+        supportsUnknownElements: m,
+        shivMethods: !1 !== n.shivMethods,
+        type: "default",
+        shivDocument: g,
+        createElement: f,
+        createDocumentFragment: function(e, t) {
+            if (e = e || i, m) return e.createDocumentFragment();
+            for (var n = (t = t || h(e)).frag.cloneNode(), r = 0, a = u(), o = a.length; r < o; r++) n.createElement(a[r]);
+            return n;
+        },
+        addElements: function(e, t) {
+            var n = p.elements;
+            "string" != typeof n && (n = n.join(" ")), "string" != typeof e && (e = e.join(" ")), 
+            p.elements = n + " " + e, g(t);
+        }
+    };
+    e.html5 = p, g(i), "object" == typeof module && module.exports && (module.exports = p);
+}("undefined" != typeof window ? window : this, document);</script>
+<script>(function(c){c.matchMedia=c.matchMedia||function(c,e){var h,r=c.documentElement,t=r.firstElementChild||r.firstChild,m=c.createElement("body"),f=c.createElement("div");f.id="mq-test-1";f.style.cssText="position:absolute;top:-100em";m.style.background="none";m.appendChild(f);return function(c){f.innerHTML='&shy;<style media="'+c+'"> #mq-test-1 { width: 42px; }</style>';r.insertBefore(m,t);h=42===f.offsetWidth;r.removeChild(m);return{matches:h,media:c}}}(c.document)})(this);
+(function(c){function D(){y(!0)}var e={};c.respond=e;e.update=function(){};var h=[],r=function(){var d=!1;try{d=new c.XMLHttpRequest}catch(b){d=new c.ActiveXObject("Microsoft.XMLHTTP")}return function(){return d}}(),t=function(d,b){var a=r();a&&(a.open("GET",d,!0),a.onreadystatechange=function(){4!==a.readyState||200!==a.status&&304!==a.status||b(a.responseText)},4!==a.readyState&&a.send(null))},m=function(d){return d.replace(e.regex.minmaxwh,"").match(e.regex.other)};e.ajax=t;e.queue=h;e.unsupportedmq=
+m;e.regex={media:/@media[^\{]+\{([^\{\}]*\{[^\}\{]*\})+/gi,keyframes:/@(?:\-(?:o|moz|webkit)\-)?keyframes[^\{]+\{(?:[^\{\}]*\{[^\}\{]*\})+[^\}]*\}/gi,comments:/\/\*[^*]*\*+([^/][^*]*\*+)*\//gi,urls:/(url\()['"]?([^\/\)'"][^:\)'"]+)['"]?(\))/g,findStyles:/@media *([^\{]+)\{([\S\s]+?)$/,only:/(only\s+)?([a-zA-Z]+)\s?/,minw:/\(\s*min\-width\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/,maxw:/\(\s*max\-width\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/,minmaxwh:/\(\s*m(in|ax)\-(height|width)\s*:\s*(\s*[0-9\.]+)(px|em)\s*\)/gi,
+other:/\([^\)]*\)/g};e.mediaQueriesSupported=c.matchMedia&&null!==c.matchMedia("only all")&&c.matchMedia("only all").matches;if(!e.mediaQueriesSupported){var f=c.document,k=f.documentElement,u=[],v=[],n=[],z={},w=f.getElementsByTagName("head")[0]||k,I=f.getElementsByTagName("base")[0],x=w.getElementsByTagName("link"),A,E,B,C=function(){var d,b=f.createElement("div"),a=f.body,c=k.style.fontSize,e=a&&a.style.fontSize,g=!1;b.style.cssText="position:absolute;font-size:1em;width:1em";a||(a=g=f.createElement("body"),
+a.style.background="none");k.style.fontSize="100%";a.style.fontSize="100%";a.appendChild(b);g&&k.insertBefore(a,k.firstChild);d=b.offsetWidth;g?k.removeChild(a):a.removeChild(b);k.style.fontSize=c;e&&(a.style.fontSize=e);return d=B=parseFloat(d)},y=function(d){var b=k.clientWidth,a="CSS1Compat"===f.compatMode&&b||f.body.clientWidth||b,b={},e=x[x.length-1],q=(new Date).getTime();if(d&&A&&30>q-A)c.clearTimeout(E),E=c.setTimeout(y,30);else{A=q;for(var g in u)if(u.hasOwnProperty(g)){d=u[g];var q=d.minw,
+s=d.maxw,l=null===q,h=null===s;q&&(q=parseFloat(q)*(-1<q.indexOf("em")?B||C():1));s&&(s=parseFloat(s)*(-1<s.indexOf("em")?B||C():1));if(!d.hasquery||(!l||!h)&&(l||a>=q)&&(h||a<=s))b[d.media]||(b[d.media]=[]),b[d.media].push(v[d.rules])}for(var p in n)n.hasOwnProperty(p)&&n[p]&&n[p].parentNode===w&&w.removeChild(n[p]);n.length=0;for(var m in b)b.hasOwnProperty(m)&&(g=f.createElement("style"),p=b[m].join("\n"),g.type="text/css",g.media=m,w.insertBefore(g,e.nextSibling),g.styleSheet?g.styleSheet.cssText=
+p:g.appendChild(f.createTextNode(p)),n.push(g))}},F=function(d,b,a){var c=d.replace(e.regex.comments,"").replace(e.regex.keyframes,"").match(e.regex.media),f=c&&c.length||0;b=b.substring(0,b.lastIndexOf("/"));var g=!f&&a;b.length&&(b+="/");g&&(f=1);for(var h=0;h<f;h++){var l,k,p;g?(l=a,v.push(d.replace(e.regex.urls,"$1"+b+"$2$3"))):(l=c[h].match(e.regex.findStyles)&&RegExp.$1,v.push(RegExp.$2&&RegExp.$2.replace(e.regex.urls,"$1"+b+"$2$3")));k=l.split(",");p=k.length;for(var n=0;n<p;n++)l=k[n],m(l)||
+u.push({media:l.split("(")[0].match(e.regex.only)&&RegExp.$2||"all",rules:v.length-1,hasquery:-1<l.indexOf("("),minw:l.match(e.regex.minw)&&parseFloat(RegExp.$1)+(RegExp.$2||""),maxw:l.match(e.regex.maxw)&&parseFloat(RegExp.$1)+(RegExp.$2||"")})}y()},G=function(){if(h.length){var d=h.shift();t(d.href,function(b){F(b,d.href,d.media);z[d.href]=!0;c.setTimeout(function(){G()},0)})}},H=function(){for(var d=0;d<x.length;d++){var b=x[d],a=b.href,e=b.media,f=b.rel&&"stylesheet"===b.rel.toLowerCase();if(a&&
+f&&!z[a])if(b.styleSheet&&b.styleSheet.rawCssText)F(b.styleSheet.rawCssText,a,e),z[a]=!0;else if(!/^([a-zA-Z:]*\/\/)/.test(a)&&!I||a.replace(RegExp.$1,"").split("/")[0]===c.location.host)"//"===a.substring(0,2)&&(a=c.location.protocol+a),h.push({href:a,media:e})}G()};H();e.update=H;e.getEmValue=C;c.addEventListener?c.addEventListener("resize",D,!1):c.attachEvent&&c.attachEvent("onresize",D)}})(this);
+</script>
+<style>h1 {font-size: 34px;}
+       h1.title {font-size: 38px;}
+       h2 {font-size: 30px;}
+       h3 {font-size: 24px;}
+       h4 {font-size: 18px;}
+       h5 {font-size: 16px;}
+       h6 {font-size: 12px;}
+       code {color: inherit; background-color: rgba(0, 0, 0, 0.04);}
+       pre:not([class]) { background-color: white }</style>
+<script>
+
+/**
+ * jQuery Plugin: Sticky Tabs
+ *
+ * @author Aidan Lister <aidan@php.net>
+ * adapted by Ruben Arslan to activate parent tabs too
+ * http://www.aidanlister.com/2014/03/persisting-the-tab-state-in-bootstrap/
+ */
+(function($) {
+  "use strict";
+  $.fn.rmarkdownStickyTabs = function() {
+    var context = this;
+    // Show the tab corresponding with the hash in the URL, or the first tab
+    var showStuffFromHash = function() {
+      var hash = window.location.hash;
+      var selector = hash ? 'a[href="' + hash + '"]' : 'li.active > a';
+      var $selector = $(selector, context);
+      if($selector.data('toggle') === "tab") {
+        $selector.tab('show');
+        // walk up the ancestors of this element, show any hidden tabs
+        $selector.parents('.section.tabset').each(function(i, elm) {
+          var link = $('a[href="#' + $(elm).attr('id') + '"]');
+          if(link.data('toggle') === "tab") {
+            link.tab("show");
+          }
+        });
+      }
+    };
+
+
+    // Set the correct tab when the page loads
+    showStuffFromHash(context);
+
+    // Set the correct tab when a user uses their back/forward button
+    $(window).on('hashchange', function() {
+      showStuffFromHash(context);
+    });
+
+    // Change the URL when tabs are clicked
+    $('a', context).on('click', function(e) {
+      history.pushState(null, null, this.href);
+      showStuffFromHash(context);
+    });
+
+    return this;
+  };
+}(jQuery));
+
+window.buildTabsets = function(tocID) {
+
+  // build a tabset from a section div with the .tabset class
+  function buildTabset(tabset) {
+
+    // check for fade and pills options
+    var fade = tabset.hasClass("tabset-fade");
+    var pills = tabset.hasClass("tabset-pills");
+    var navClass = pills ? "nav-pills" : "nav-tabs";
+
+    // determine the heading level of the tabset and tabs
+    var match = tabset.attr('class').match(/level(\d) /);
+    if (match === null)
+      return;
+    var tabsetLevel = Number(match[1]);
+    var tabLevel = tabsetLevel + 1;
+
+    // find all subheadings immediately below
+    var tabs = tabset.find("div.section.level" + tabLevel);
+    if (!tabs.length)
+      return;
+
+    // create tablist and tab-content elements
+    var tabList = $('<ul class="nav ' + navClass + '" role="tablist"></ul>');
+    $(tabs[0]).before(tabList);
+    var tabContent = $('<div class="tab-content"></div>');
+    $(tabs[0]).before(tabContent);
+
+    // build the tabset
+    var activeTab = 0;
+    tabs.each(function(i) {
+
+      // get the tab div
+      var tab = $(tabs[i]);
+
+      // get the id then sanitize it for use with bootstrap tabs
+      var id = tab.attr('id');
+
+      // see if this is marked as the active tab
+      if (tab.hasClass('active'))
+        activeTab = i;
+
+      // remove any table of contents entries associated with
+      // this ID (since we'll be removing the heading element)
+      $("div#" + tocID + " li a[href='#" + id + "']").parent().remove();
+
+      // sanitize the id for use with bootstrap tabs
+      id = id.replace(/[.\/?&!#<>]/g, '').replace(/\s/g, '_');
+      tab.attr('id', id);
+
+      // get the heading element within it, grab it's text, then remove it
+      var heading = tab.find('h' + tabLevel + ':first');
+      var headingText = heading.html();
+      heading.remove();
+
+      // build and append the tab list item
+      var a = $('<a role="tab" data-toggle="tab">' + headingText + '</a>');
+      a.attr('href', '#' + id);
+      a.attr('aria-controls', id);
+      var li = $('<li role="presentation"></li>');
+      li.append(a);
+      tabList.append(li);
+
+      // set it's attributes
+      tab.attr('role', 'tabpanel');
+      tab.addClass('tab-pane');
+      tab.addClass('tabbed-pane');
+      if (fade)
+        tab.addClass('fade');
+
+      // move it into the tab content div
+      tab.detach().appendTo(tabContent);
+    });
+
+    // set active tab
+    $(tabList.children('li')[activeTab]).addClass('active');
+    var active = $(tabContent.children('div.section')[activeTab]);
+    active.addClass('active');
+    if (fade)
+      active.addClass('in');
+
+    if (tabset.hasClass("tabset-sticky"))
+      tabset.rmarkdownStickyTabs();
+  }
+
+  // convert section divs with the .tabset class to tabsets
+  var tabsets = $("div.section.tabset");
+  tabsets.each(function(i) {
+    buildTabset($(tabsets[i]));
+  });
+};
+
+</script>
+<style type="text/css">.hljs-literal {
+color: #990073;
+}
+.hljs-number {
+color: #099;
+}
+.hljs-comment {
+color: #998;
+font-style: italic;
+}
+.hljs-keyword {
+color: #900;
+font-weight: bold;
+}
+.hljs-string {
+color: #d14;
+}
+</style>
+<script>var hljs=function(factory){var globalObject=typeof window==="object"&&window||typeof self==="object"&&self;if(typeof exports!=="undefined"&&!exports.nodeType){return factory(exports)}else if(globalObject){globalObject.hljs=factory({});if(typeof define==="function"&&define.amd){define([],function(){return globalObject.hljs})}return globalObject.hljs}}(function(hljs){var showedUpgradeWarning=false;var ArrayProto=[],objectKeys=Object.keys;var languages=Object.create(null),aliases=Object.create(null);var SAFE_MODE=true;var noHighlightRe=/^(no-?highlight|plain|text)$/i,languagePrefixRe=/\blang(?:uage)?-([\w-]+)\b/i,fixMarkupRe=/((^(<[^>]+>|\t|)+|(?:\n)))/gm;var API_REPLACES;var spanEndTag="</span>";var LANGUAGE_NOT_FOUND="Could not find the language '{}', did you forget to load/include a language module?";var options={hideUpgradeWarningAcceptNoSupportOrSecurityUpdates:false,classPrefix:"hljs-",tabReplace:null,useBR:false,languages:undefined};var COMMON_KEYWORDS="of and for in not or if then".split(" ");function escape(value){return value.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;")}function tag(node){return node.nodeName.toLowerCase()}function testRe(re,lexeme){var match=re&&re.exec(lexeme);return match&&match.index===0}function isNotHighlighted(language){return noHighlightRe.test(language)}function blockLanguage(block){var i,match,length,_class;var classes=block.className+" ";classes+=block.parentNode?block.parentNode.className:"";match=languagePrefixRe.exec(classes);if(match){var language=getLanguage(match[1]);if(!language){console.warn(LANGUAGE_NOT_FOUND.replace("{}",match[1]));console.warn("Falling back to no-highlight mode for this block.",block)}return language?match[1]:"no-highlight"}classes=classes.split(/\s+/);for(i=0,length=classes.length;i<length;i++){_class=classes[i];if(isNotHighlighted(_class)||getLanguage(_class)){return _class}}}function inherit(parent){var key;var result={};var objects=Array.prototype.slice.call(arguments,1);for(key in parent)result[key]=parent[key];objects.forEach(function(obj){for(key in obj)result[key]=obj[key]});return result}function nodeStream(node){var result=[];(function _nodeStream(node,offset){for(var child=node.firstChild;child;child=child.nextSibling){if(child.nodeType===3)offset+=child.nodeValue.length;else if(child.nodeType===1){result.push({event:"start",offset:offset,node:child});offset=_nodeStream(child,offset);if(!tag(child).match(/br|hr|img|input/)){result.push({event:"stop",offset:offset,node:child})}}}return offset})(node,0);return result}function mergeStreams(original,highlighted,value){var processed=0;var result="";var nodeStack=[];function selectStream(){if(!original.length||!highlighted.length){return original.length?original:highlighted}if(original[0].offset!==highlighted[0].offset){return original[0].offset<highlighted[0].offset?original:highlighted}return highlighted[0].event==="start"?original:highlighted}function open(node){function attr_str(a){return" "+a.nodeName+'="'+escape(a.value).replace(/"/g,"&quot;")+'"'}result+="<"+tag(node)+ArrayProto.map.call(node.attributes,attr_str).join("")+">"}function close(node){result+="</"+tag(node)+">"}function render(event){(event.event==="start"?open:close)(event.node)}while(original.length||highlighted.length){var stream=selectStream();result+=escape(value.substring(processed,stream[0].offset));processed=stream[0].offset;if(stream===original){nodeStack.reverse().forEach(close);do{render(stream.splice(0,1)[0]);stream=selectStream()}while(stream===original&&stream.length&&stream[0].offset===processed);nodeStack.reverse().forEach(open)}else{if(stream[0].event==="start"){nodeStack.push(stream[0].node)}else{nodeStack.pop()}render(stream.splice(0,1)[0])}}return result+escape(value.substr(processed))}function dependencyOnParent(mode){if(!mode)return false;return mode.endsWithParent||dependencyOnParent(mode.starts)}function expand_or_clone_mode(mode){if(mode.variants&&!mode.cached_variants){mode.cached_variants=mode.variants.map(function(variant){return inherit(mode,{variants:null},variant)})}if(mode.cached_variants)return mode.cached_variants;if(dependencyOnParent(mode))return[inherit(mode,{starts:mode.starts?inherit(mode.starts):null})];if(Object.isFrozen(mode))return[inherit(mode)];return[mode]}function restoreLanguageApi(obj){if(API_REPLACES&&!obj.langApiRestored){obj.langApiRestored=true;for(var key in API_REPLACES){if(obj[key]){obj[API_REPLACES[key]]=obj[key]}}(obj.contains||[]).concat(obj.variants||[]).forEach(restoreLanguageApi)}}function compileKeywords(rawKeywords,case_insensitive){var compiled_keywords={};if(typeof rawKeywords==="string"){splitAndCompile("keyword",rawKeywords)}else{objectKeys(rawKeywords).forEach(function(className){splitAndCompile(className,rawKeywords[className])})}return compiled_keywords;function splitAndCompile(className,str){if(case_insensitive){str=str.toLowerCase()}str.split(" ").forEach(function(keyword){var pair=keyword.split("|");compiled_keywords[pair[0]]=[className,scoreForKeyword(pair[0],pair[1])]})}}function scoreForKeyword(keyword,providedScore){if(providedScore)return Number(providedScore);return commonKeyword(keyword)?0:1}function commonKeyword(word){return COMMON_KEYWORDS.indexOf(word.toLowerCase())!=-1}function compileLanguage(language){function reStr(re){return re&&re.source||re}function langRe(value,global){return new RegExp(reStr(value),"m"+(language.case_insensitive?"i":"")+(global?"g":""))}function reCountMatchGroups(re){return new RegExp(re.toString()+"|").exec("").length-1}function joinRe(regexps,separator){var backreferenceRe=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./;var numCaptures=0;var ret="";for(var i=0;i<regexps.length;i++){numCaptures+=1;var offset=numCaptures;var re=reStr(regexps[i]);if(i>0){ret+=separator}ret+="(";while(re.length>0){var match=backreferenceRe.exec(re);if(match==null){ret+=re;break}ret+=re.substring(0,match.index);re=re.substring(match.index+match[0].length);if(match[0][0]=="\\"&&match[1]){ret+="\\"+String(Number(match[1])+offset)}else{ret+=match[0];if(match[0]=="("){numCaptures++}}}ret+=")"}return ret}function buildModeRegex(mode){var matchIndexes={};var matcherRe;var regexes=[];var matcher={};var matchAt=1;function addRule(rule,regex){matchIndexes[matchAt]=rule;regexes.push([rule,regex]);matchAt+=reCountMatchGroups(regex)+1}var term;for(var i=0;i<mode.contains.length;i++){var re;term=mode.contains[i];if(term.beginKeywords){re="\\.?(?:"+term.begin+")\\.?"}else{re=term.begin}addRule(term,re)}if(mode.terminator_end)addRule("end",mode.terminator_end);if(mode.illegal)addRule("illegal",mode.illegal);var terminators=regexes.map(function(el){return el[1]});matcherRe=langRe(joinRe(terminators,"|"),true);matcher.lastIndex=0;matcher.exec=function(s){var rule;if(regexes.length===0)return null;matcherRe.lastIndex=matcher.lastIndex;var match=matcherRe.exec(s);if(!match){return null}for(var i=0;i<match.length;i++){if(match[i]!=undefined&&matchIndexes[""+i]!=undefined){rule=matchIndexes[""+i];break}}if(typeof rule==="string"){match.type=rule;match.extra=[mode.illegal,mode.terminator_end]}else{match.type="begin";match.rule=rule}return match};return matcher}function compileMode(mode,parent){if(mode.compiled)return;mode.compiled=true;mode.keywords=mode.keywords||mode.beginKeywords;if(mode.keywords)mode.keywords=compileKeywords(mode.keywords,language.case_insensitive);mode.lexemesRe=langRe(mode.lexemes||/\w+/,true);if(parent){if(mode.beginKeywords){mode.begin="\\b("+mode.beginKeywords.split(" ").join("|")+")\\b"}if(!mode.begin)mode.begin=/\B|\b/;mode.beginRe=langRe(mode.begin);if(mode.endSameAsBegin)mode.end=mode.begin;if(!mode.end&&!mode.endsWithParent)mode.end=/\B|\b/;if(mode.end)mode.endRe=langRe(mode.end);mode.terminator_end=reStr(mode.end)||"";if(mode.endsWithParent&&parent.terminator_end)mode.terminator_end+=(mode.end?"|":"")+parent.terminator_end}if(mode.illegal)mode.illegalRe=langRe(mode.illegal);if(mode.relevance==null)mode.relevance=1;if(!mode.contains){mode.contains=[]}mode.contains=Array.prototype.concat.apply([],mode.contains.map(function(c){return expand_or_clone_mode(c==="self"?mode:c)}));mode.contains.forEach(function(c){compileMode(c,mode)});if(mode.starts){compileMode(mode.starts,parent)}mode.terminators=buildModeRegex(mode)}if(language.contains&&language.contains.indexOf("self")!=-1){if(!SAFE_MODE){throw new Error("ERR: contains `self` is not supported at the top-level of a language.  See documentation.")}else{language.contains=language.contains.filter(function(mode){return mode!="self"})}}compileMode(language)}function hideUpgradeWarning(){if(options.hideUpgradeWarningAcceptNoSupportOrSecurityUpdates)return true;if(typeof process==="object"&&typeof process.env==="object"&&process.env["HLJS_HIDE_UPGRADE_WARNING"])return true}function highlight(languageName,code,ignore_illegals,continuation){if(!hideUpgradeWarning()){if(!showedUpgradeWarning){showedUpgradeWarning=true;console.log("Version 9 of Highlight.js has reached EOL and is no longer supported.\n"+"Please upgrade or ask whatever dependency you are using to upgrade.\n"+"https://github.com/highlightjs/highlight.js/issues/2877")}}var codeToHighlight=code;function escapeRe(value){return new RegExp(value.replace(/[-\/\\^$*+?.()|[\]{}]/g,"\\$&"),"m")}function endOfMode(mode,lexeme){if(testRe(mode.endRe,lexeme)){while(mode.endsParent&&mode.parent){mode=mode.parent}return mode}if(mode.endsWithParent){return endOfMode(mode.parent,lexeme)}}function keywordMatch(mode,match){var match_str=language.case_insensitive?match[0].toLowerCase():match[0];return mode.keywords.hasOwnProperty(match_str)&&mode.keywords[match_str]}function buildSpan(className,insideSpan,leaveOpen,noPrefix){if(!leaveOpen&&insideSpan==="")return"";if(!className)return insideSpan;var classPrefix=noPrefix?"":options.classPrefix,openSpan='<span class="'+classPrefix,closeSpan=leaveOpen?"":spanEndTag;openSpan+=className+'">';return openSpan+insideSpan+closeSpan}function processKeywords(){var keyword_match,last_index,match,result;if(!top.keywords)return escape(mode_buffer);result="";last_index=0;top.lexemesRe.lastIndex=0;match=top.lexemesRe.exec(mode_buffer);while(match){result+=escape(mode_buffer.substring(last_index,match.index));keyword_match=keywordMatch(top,match);if(keyword_match){relevance+=keyword_match[1];result+=buildSpan(keyword_match[0],escape(match[0]))}else{result+=escape(match[0])}last_index=top.lexemesRe.lastIndex;match=top.lexemesRe.exec(mode_buffer)}return result+escape(mode_buffer.substr(last_index))}function processSubLanguage(){var explicit=typeof top.subLanguage==="string";if(explicit&&!languages[top.subLanguage]){return escape(mode_buffer)}var result=explicit?highlight(top.subLanguage,mode_buffer,true,continuations[top.subLanguage]):highlightAuto(mode_buffer,top.subLanguage.length?top.subLanguage:undefined);if(top.relevance>0){relevance+=result.relevance}if(explicit){continuations[top.subLanguage]=result.top}return buildSpan(result.language,result.value,false,true)}function processBuffer(){result+=top.subLanguage!=null?processSubLanguage():processKeywords();mode_buffer=""}function startNewMode(mode){result+=mode.className?buildSpan(mode.className,"",true):"";top=Object.create(mode,{parent:{value:top}})}function doBeginMatch(match){var lexeme=match[0];var new_mode=match.rule;if(new_mode&&new_mode.endSameAsBegin){new_mode.endRe=escapeRe(lexeme)}if(new_mode.skip){mode_buffer+=lexeme}else{if(new_mode.excludeBegin){mode_buffer+=lexeme}processBuffer();if(!new_mode.returnBegin&&!new_mode.excludeBegin){mode_buffer=lexeme}}startNewMode(new_mode);return new_mode.returnBegin?0:lexeme.length}function doEndMatch(match){var lexeme=match[0];var matchPlusRemainder=codeToHighlight.substr(match.index);var end_mode=endOfMode(top,matchPlusRemainder);if(!end_mode){return}var origin=top;if(origin.skip){mode_buffer+=lexeme}else{if(!(origin.returnEnd||origin.excludeEnd)){mode_buffer+=lexeme}processBuffer();if(origin.excludeEnd){mode_buffer=lexeme}}do{if(top.className){result+=spanEndTag}if(!top.skip&&!top.subLanguage){relevance+=top.relevance}top=top.parent}while(top!==end_mode.parent);if(end_mode.starts){if(end_mode.endSameAsBegin){end_mode.starts.endRe=end_mode.endRe}startNewMode(end_mode.starts)}return origin.returnEnd?0:lexeme.length}var lastMatch={};function processLexeme(text_before_match,match){var lexeme=match&&match[0];mode_buffer+=text_before_match;if(lexeme==null){processBuffer();return 0}if(lastMatch.type=="begin"&&match.type=="end"&&lastMatch.index==match.index&&lexeme===""){mode_buffer+=codeToHighlight.slice(match.index,match.index+1);return 1}if(lastMatch.type==="illegal"&&lexeme===""){mode_buffer+=codeToHighlight.slice(match.index,match.index+1);return 1}lastMatch=match;if(match.type==="begin"){return doBeginMatch(match)}else if(match.type==="illegal"&&!ignore_illegals){throw new Error('Illegal lexeme "'+lexeme+'" for mode "'+(top.className||"<unnamed>")+'"')}else if(match.type==="end"){var processed=doEndMatch(match);if(processed!=undefined)return processed}mode_buffer+=lexeme;return lexeme.length}var language=getLanguage(languageName);if(!language){console.error(LANGUAGE_NOT_FOUND.replace("{}",languageName));throw new Error('Unknown language: "'+languageName+'"')}compileLanguage(language);var top=continuation||language;var continuations={};var result="",current;for(current=top;current!==language;current=current.parent){if(current.className){result=buildSpan(current.className,"",true)+result}}var mode_buffer="";var relevance=0;try{var match,count,index=0;while(true){top.terminators.lastIndex=index;match=top.terminators.exec(codeToHighlight);if(!match)break;count=processLexeme(codeToHighlight.substring(index,match.index),match);index=match.index+count}processLexeme(codeToHighlight.substr(index));for(current=top;current.parent;current=current.parent){if(current.className){result+=spanEndTag}}return{relevance:relevance,value:result,illegal:false,language:languageName,top:top}}catch(err){if(err.message&&err.message.indexOf("Illegal")!==-1){return{illegal:true,relevance:0,value:escape(codeToHighlight)}}else if(SAFE_MODE){return{relevance:0,value:escape(codeToHighlight),language:languageName,top:top,errorRaised:err}}else{throw err}}}function highlightAuto(code,languageSubset){languageSubset=languageSubset||options.languages||objectKeys(languages);var result={relevance:0,value:escape(code)};var second_best=result;languageSubset.filter(getLanguage).filter(autoDetection).forEach(function(name){var current=highlight(name,code,false);current.language=name;if(current.relevance>second_best.relevance){second_best=current}if(current.relevance>result.relevance){second_best=result;result=current}});if(second_best.language){result.second_best=second_best}return result}function fixMarkup(value){if(!(options.tabReplace||options.useBR)){return value}return value.replace(fixMarkupRe,function(match,p1){if(options.useBR&&match==="\n"){return"<br>"}else if(options.tabReplace){return p1.replace(/\t/g,options.tabReplace)}return""})}function buildClassName(prevClassName,currentLang,resultLang){var language=currentLang?aliases[currentLang]:resultLang,result=[prevClassName.trim()];if(!prevClassName.match(/\bhljs\b/)){result.push("hljs")}if(prevClassName.indexOf(language)===-1){result.push(language)}return result.join(" ").trim()}function highlightBlock(block){var node,originalStream,result,resultNode,text;var language=blockLanguage(block);if(isNotHighlighted(language))return;if(options.useBR){node=document.createElement("div");node.innerHTML=block.innerHTML.replace(/\n/g,"").replace(/<br[ \/]*>/g,"\n")}else{node=block}text=node.textContent;result=language?highlight(language,text,true):highlightAuto(text);originalStream=nodeStream(node);if(originalStream.length){resultNode=document.createElement("div");resultNode.innerHTML=result.value;result.value=mergeStreams(originalStream,nodeStream(resultNode),text)}result.value=fixMarkup(result.value);block.innerHTML=result.value;block.className=buildClassName(block.className,language,result.language);block.result={language:result.language,re:result.relevance};if(result.second_best){block.second_best={language:result.second_best.language,re:result.second_best.relevance}}}function configure(user_options){options=inherit(options,user_options)}function initHighlighting(){if(initHighlighting.called)return;initHighlighting.called=true;var blocks=document.querySelectorAll("pre code");ArrayProto.forEach.call(blocks,highlightBlock)}function initHighlightingOnLoad(){window.addEventListener("DOMContentLoaded",initHighlighting,false);window.addEventListener("load",initHighlighting,false)}var PLAINTEXT_LANGUAGE={disableAutodetect:true};function registerLanguage(name,language){var lang;try{lang=language(hljs)}catch(error){console.error("Language definition for '{}' could not be registered.".replace("{}",name));if(!SAFE_MODE){throw error}else{console.error(error)}lang=PLAINTEXT_LANGUAGE}languages[name]=lang;restoreLanguageApi(lang);lang.rawDefinition=language.bind(null,hljs);if(lang.aliases){lang.aliases.forEach(function(alias){aliases[alias]=name})}}function listLanguages(){return objectKeys(languages)}function requireLanguage(name){var lang=getLanguage(name);if(lang){return lang}var err=new Error("The '{}' language is required, but not loaded.".replace("{}",name));throw err}function getLanguage(name){name=(name||"").toLowerCase();return languages[name]||languages[aliases[name]]}function autoDetection(name){var lang=getLanguage(name);return lang&&!lang.disableAutodetect}hljs.highlight=highlight;hljs.highlightAuto=highlightAuto;hljs.fixMarkup=fixMarkup;hljs.highlightBlock=highlightBlock;hljs.configure=configure;hljs.initHighlighting=initHighlighting;hljs.initHighlightingOnLoad=initHighlightingOnLoad;hljs.registerLanguage=registerLanguage;hljs.listLanguages=listLanguages;hljs.getLanguage=getLanguage;hljs.requireLanguage=requireLanguage;hljs.autoDetection=autoDetection;hljs.inherit=inherit;hljs.debugMode=function(){SAFE_MODE=false};hljs.IDENT_RE="[a-zA-Z]\\w*";hljs.UNDERSCORE_IDENT_RE="[a-zA-Z_]\\w*";hljs.NUMBER_RE="\\b\\d+(\\.\\d+)?";hljs.C_NUMBER_RE="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)";hljs.BINARY_NUMBER_RE="\\b(0b[01]+)";hljs.RE_STARTERS_RE="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~";hljs.BACKSLASH_ESCAPE={begin:"\\\\[\\s\\S]",relevance:0};hljs.APOS_STRING_MODE={className:"string",begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};hljs.QUOTE_STRING_MODE={className:"string",begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};hljs.PHRASAL_WORDS_MODE={begin:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/};hljs.COMMENT=function(begin,end,inherits){var mode=hljs.inherit({className:"comment",begin:begin,end:end,contains:[]},inherits||{});mode.contains.push(hljs.PHRASAL_WORDS_MODE);mode.contains.push({className:"doctag",begin:"(?:TODO|FIXME|NOTE|BUG|XXX):",relevance:0});return mode};hljs.C_LINE_COMMENT_MODE=hljs.COMMENT("//","$");hljs.C_BLOCK_COMMENT_MODE=hljs.COMMENT("/\\*","\\*/");hljs.HASH_COMMENT_MODE=hljs.COMMENT("#","$");hljs.NUMBER_MODE={className:"number",begin:hljs.NUMBER_RE,relevance:0};hljs.C_NUMBER_MODE={className:"number",begin:hljs.C_NUMBER_RE,relevance:0};hljs.BINARY_NUMBER_MODE={className:"number",begin:hljs.BINARY_NUMBER_RE,relevance:0};hljs.CSS_NUMBER_MODE={className:"number",begin:hljs.NUMBER_RE+"("+"%|em|ex|ch|rem"+"|vw|vh|vmin|vmax"+"|cm|mm|in|pt|pc|px"+"|deg|grad|rad|turn"+"|s|ms"+"|Hz|kHz"+"|dpi|dpcm|dppx"+")?",relevance:0};hljs.REGEXP_MODE={className:"regexp",begin:/\//,end:/\/[gimuy]*/,illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE,{begin:/\[/,end:/\]/,relevance:0,contains:[hljs.BACKSLASH_ESCAPE]}]};hljs.TITLE_MODE={className:"title",begin:hljs.IDENT_RE,relevance:0};hljs.UNDERSCORE_TITLE_MODE={className:"title",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0};hljs.METHOD_GUARD={begin:"\\.\\s*"+hljs.UNDERSCORE_IDENT_RE,relevance:0};var constants=[hljs.BACKSLASH_ESCAPE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.PHRASAL_WORDS_MODE,hljs.COMMENT,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE,hljs.C_NUMBER_MODE,hljs.BINARY_NUMBER_MODE,hljs.CSS_NUMBER_MODE,hljs.REGEXP_MODE,hljs.TITLE_MODE,hljs.UNDERSCORE_TITLE_MODE,hljs.METHOD_GUARD];constants.forEach(function(obj){deepFreeze(obj)});function deepFreeze(o){Object.freeze(o);var objIsFunction=typeof o==="function";Object.getOwnPropertyNames(o).forEach(function(prop){if(o.hasOwnProperty(prop)&&o[prop]!==null&&(typeof o[prop]==="object"||typeof o[prop]==="function")&&(objIsFunction?prop!=="caller"&&prop!=="callee"&&prop!=="arguments":true)&&!Object.isFrozen(o[prop])){deepFreeze(o[prop])}});return o}return hljs});hljs.registerLanguage("hsp",function(hljs){return{case_insensitive:true,lexemes:/[\w\._]+/,keywords:"goto gosub return break repeat loop continue wait await dim sdim foreach dimtype dup dupptr end stop newmod delmod mref run exgoto on mcall assert logmes newlab resume yield onexit onerror onkey onclick oncmd exist delete mkdir chdir dirlist bload bsave bcopy memfile if else poke wpoke lpoke getstr chdpm memexpand memcpy memset notesel noteadd notedel noteload notesave randomize noteunsel noteget split strrep setease button chgdisp exec dialog mmload mmplay mmstop mci pset pget syscolor mes print title pos circle cls font sysfont objsize picload color palcolor palette redraw width gsel gcopy gzoom gmode bmpsave hsvcolor getkey listbox chkbox combox input mesbox buffer screen bgscr mouse objsel groll line clrobj boxf objprm objmode stick grect grotate gsquare gradf objimage objskip objenable celload celdiv celput newcom querycom delcom cnvstow comres axobj winobj sendmsg comevent comevarg sarrayconv callfunc cnvwtos comevdisp libptr system hspstat hspver stat cnt err strsize looplev sublev iparam wparam lparam refstr refdval int rnd strlen length length2 length3 length4 vartype gettime peek wpeek lpeek varptr varuse noteinfo instr abs limit getease str strmid strf getpath strtrim sin cos tan atan sqrt double absf expf logf limitf powf geteasef mousex mousey mousew hwnd hinstance hdc ginfo objinfo dirinfo sysinfo thismod __hspver__ __hsp30__ __date__ __time__ __line__ __file__ _debug __hspdef__ and or xor not screen_normal screen_palette screen_hide screen_fixedsize screen_tool screen_frame gmode_gdi gmode_mem gmode_rgb0 gmode_alpha gmode_rgb0alpha gmode_add gmode_sub gmode_pixela ginfo_mx ginfo_my ginfo_act ginfo_sel ginfo_wx1 ginfo_wy1 ginfo_wx2 ginfo_wy2 ginfo_vx ginfo_vy ginfo_sizex ginfo_sizey ginfo_winx ginfo_winy ginfo_mesx ginfo_mesy ginfo_r ginfo_g ginfo_b ginfo_paluse ginfo_dispx ginfo_dispy ginfo_cx ginfo_cy ginfo_intid ginfo_newid ginfo_sx ginfo_sy objinfo_mode objinfo_bmscr objinfo_hwnd notemax notesize dir_cur dir_exe dir_win dir_sys dir_cmdline dir_desktop dir_mydoc dir_tv font_normal font_bold font_italic font_underline font_strikeout font_antialias objmode_normal objmode_guifont objmode_usefont gsquare_grad msgothic msmincho do until while wend for next _break _continue switch case default swbreak swend ddim ldim alloc m_pi rad2deg deg2rad ease_linear ease_quad_in ease_quad_out ease_quad_inout ease_cubic_in ease_cubic_out ease_cubic_inout ease_quartic_in ease_quartic_out ease_quartic_inout ease_bounce_in ease_bounce_out ease_bounce_inout ease_shake_in ease_shake_out ease_shake_inout ease_loop",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"string",begin:'{"',end:'"}',contains:[hljs.BACKSLASH_ESCAPE]},hljs.COMMENT(";","$",{relevance:0}),{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"addion cfunc cmd cmpopt comfunc const defcfunc deffunc define else endif enum epack func global if ifdef ifndef include modcfunc modfunc modinit modterm module pack packopt regcmd runtime undef usecom uselib"},contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"}),hljs.NUMBER_MODE,hljs.C_NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"symbol",begin:"^\\*(\\w+|@)"},hljs.NUMBER_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("applescript",function(hljs){var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:""});var PARAMS={className:"params",begin:"\\(",end:"\\)",contains:["self",hljs.C_NUMBER_MODE,STRING]};var COMMENT_MODE_1=hljs.COMMENT("--","$");var COMMENT_MODE_2=hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self",COMMENT_MODE_1]});var COMMENTS=[COMMENT_MODE_1,COMMENT_MODE_2,hljs.HASH_COMMENT_MODE];return{aliases:["osascript"],keywords:{keyword:"about above after against and around as at back before beginning "+"behind below beneath beside between but by considering "+"contain contains continue copy div does eighth else end equal "+"equals error every exit fifth first for fourth from front "+"get given global if ignoring in into is it its last local me "+"middle mod my ninth not of on onto or over prop property put ref "+"reference repeat returning script second set seventh since "+"sixth some tell tenth that the|0 then third through thru "+"timeout times to transaction try until where while whose with "+"without",literal:"AppleScript false linefeed return pi quote result space tab true",built_in:"alias application boolean class constant date file integer list "+"number real record string text "+"activate beep count delay launch log offset read round "+"run say summarize write "+"character characters contents day frontmost id item length "+"month name paragraph paragraphs rest reverse running time version "+"weekday word words year"},contains:[STRING,hljs.C_NUMBER_MODE,{className:"built_in",begin:"\\b(clipboard info|the clipboard|info for|list (disks|folder)|"+"mount volume|path to|(close|open for) access|(get|set) eof|"+"current date|do shell script|get volume settings|random number|"+"set volume|system attribute|system info|time to GMT|"+"(load|run|store) script|scripting components|"+"ASCII (character|number)|localized string|"+"choose (application|color|file|file name|"+"folder|from list|remote application|URL)|"+"display (alert|dialog))\\b|^\\s*return\\b"},{className:"literal",begin:"\\b(text item delimiters|current application|missing value)\\b"},{className:"keyword",begin:"\\b(apart from|aside from|instead of|out of|greater than|"+"isn't|(doesn't|does not) (equal|come before|come after|contain)|"+"(greater|less) than( or equal)?|(starts?|ends|begins?) with|"+"contained by|comes (before|after)|a (ref|reference)|POSIX file|"+"POSIX path|(date|time) string|quoted form)\\b"},{beginKeywords:"on",illegal:"[${=;\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]}].concat(COMMENTS),illegal:"//|->|=>|\\[\\["}});hljs.registerLanguage("matlab",function(hljs){var TRANSPOSE_RE="('|\\.')+";var TRANSPOSE={relevance:0,contains:[{begin:TRANSPOSE_RE}]};return{keywords:{keyword:"break case catch classdef continue else elseif end enumerated events for function "+"global if methods otherwise parfor persistent properties return spmd switch try while",built_in:"sin sind sinh asin asind asinh cos cosd cosh acos acosd acosh tan tand tanh atan "+"atand atan2 atanh sec secd sech asec asecd asech csc cscd csch acsc acscd acsch cot "+"cotd coth acot acotd acoth hypot exp expm1 log log1p log10 log2 pow2 realpow reallog "+"realsqrt sqrt nthroot nextpow2 abs angle complex conj imag real unwrap isreal "+"cplxpair fix floor ceil round mod rem sign airy besselj bessely besselh besseli "+"besselk beta betainc betaln ellipj ellipke erf erfc erfcx erfinv expint gamma "+"gammainc gammaln psi legendre cross dot factor isprime primes gcd lcm rat rats perms "+"nchoosek factorial cart2sph cart2pol pol2cart sph2cart hsv2rgb rgb2hsv zeros ones "+"eye repmat rand randn linspace logspace freqspace meshgrid accumarray size length "+"ndims numel disp isempty isequal isequalwithequalnans cat reshape diag blkdiag tril "+"triu fliplr flipud flipdim rot90 find sub2ind ind2sub bsxfun ndgrid permute ipermute "+"shiftdim circshift squeeze isscalar isvector ans eps realmax realmin pi i inf nan "+"isnan isinf isfinite j why compan gallery hadamard hankel hilb invhilb magic pascal "+"rosser toeplitz vander wilkinson max min nanmax nanmin mean nanmean type table "+"readtable writetable sortrows sort figure plot plot3 scatter scatter3 cellfun "+"legend intersect ismember procrustes hold num2cell "},illegal:'(//|"|#|/\\*|\\s+/\\w+)',contains:[{className:"function",beginKeywords:"function",end:"$",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",variants:[{begin:"\\(",end:"\\)"},{begin:"\\[",end:"\\]"}]}]},{className:"built_in",begin:/true|false/,relevance:0,starts:TRANSPOSE},{begin:"[a-zA-Z][a-zA-Z_0-9]*"+TRANSPOSE_RE,relevance:0},{className:"number",begin:hljs.C_NUMBER_RE,relevance:0,starts:TRANSPOSE},{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE,{begin:"''"}]},{begin:/\]|}|\)/,relevance:0,starts:TRANSPOSE},{className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE,{begin:'""'}],starts:TRANSPOSE},hljs.COMMENT("^\\s*\\%\\{\\s*$","^\\s*\\%\\}\\s*$"),hljs.COMMENT("\\%","$")]}});hljs.registerLanguage("gml",function(hljs){var GML_KEYWORDS={keyword:"begin end if then else while do for break continue with until "+"repeat exit and or xor not return mod div switch case default var "+"globalvar enum #macro #region #endregion",built_in:"is_real is_string is_array is_undefined is_int32 is_int64 "+"is_ptr is_vec3 is_vec4 is_matrix is_bool typeof "+"variable_global_exists variable_global_get variable_global_set "+"variable_instance_exists variable_instance_get variable_instance_set "+"variable_instance_get_names array_length_1d array_length_2d "+"array_height_2d array_equals array_create array_copy random "+"random_range irandom irandom_range random_set_seed random_get_seed "+"randomize randomise choose abs round floor ceil sign frac sqrt sqr "+"exp ln log2 log10 sin cos tan arcsin arccos arctan arctan2 dsin dcos "+"dtan darcsin darccos darctan darctan2 degtorad radtodeg power logn "+"min max mean median clamp lerp dot_product dot_product_3d "+"dot_product_normalised dot_product_3d_normalised "+"dot_product_normalized dot_product_3d_normalized math_set_epsilon "+"math_get_epsilon angle_difference point_distance_3d point_distance "+"point_direction lengthdir_x lengthdir_y real string int64 ptr "+"string_format chr ansi_char ord string_length string_byte_length "+"string_pos string_copy string_char_at string_ord_at string_byte_at "+"string_set_byte_at string_delete string_insert string_lower "+"string_upper string_repeat string_letters string_digits "+"string_lettersdigits string_replace string_replace_all string_count "+"string_hash_to_newline clipboard_has_text clipboard_set_text "+"clipboard_get_text date_current_datetime date_create_datetime "+"date_valid_datetime date_inc_year date_inc_month date_inc_week "+"date_inc_day date_inc_hour date_inc_minute date_inc_second "+"date_get_year date_get_month date_get_week date_get_day "+"date_get_hour date_get_minute date_get_second date_get_weekday "+"date_get_day_of_year date_get_hour_of_year date_get_minute_of_year "+"date_get_second_of_year date_year_span date_month_span "+"date_week_span date_day_span date_hour_span date_minute_span "+"date_second_span date_compare_datetime date_compare_date "+"date_compare_time date_date_of date_time_of date_datetime_string "+"date_date_string date_time_string date_days_in_month "+"date_days_in_year date_leap_year date_is_today date_set_timezone "+"date_get_timezone game_set_speed game_get_speed motion_set "+"motion_add place_free place_empty place_meeting place_snapped "+"move_random move_snap move_towards_point move_contact_solid "+"move_contact_all move_outside_solid move_outside_all "+"move_bounce_solid move_bounce_all move_wrap distance_to_point "+"distance_to_object position_empty position_meeting path_start "+"path_end mp_linear_step mp_potential_step mp_linear_step_object "+"mp_potential_step_object mp_potential_settings mp_linear_path "+"mp_potential_path mp_linear_path_object mp_potential_path_object "+"mp_grid_create mp_grid_destroy mp_grid_clear_all mp_grid_clear_cell "+"mp_grid_clear_rectangle mp_grid_add_cell mp_grid_get_cell "+"mp_grid_add_rectangle mp_grid_add_instances mp_grid_path "+"mp_grid_draw mp_grid_to_ds_grid collision_point collision_rectangle "+"collision_circle collision_ellipse collision_line "+"collision_point_list collision_rectangle_list collision_circle_list "+"collision_ellipse_list collision_line_list instance_position_list "+"instance_place_list point_in_rectangle "+"point_in_triangle point_in_circle rectangle_in_rectangle "+"rectangle_in_triangle rectangle_in_circle instance_find "+"instance_exists instance_number instance_position instance_nearest "+"instance_furthest instance_place instance_create_depth "+"instance_create_layer instance_copy instance_change instance_destroy "+"position_destroy position_change instance_id_get "+"instance_deactivate_all instance_deactivate_object "+"instance_deactivate_region instance_activate_all "+"instance_activate_object instance_activate_region room_goto "+"room_goto_previous room_goto_next room_previous room_next "+"room_restart game_end game_restart game_load game_save "+"game_save_buffer game_load_buffer event_perform event_user "+"event_perform_object event_inherited show_debug_message "+"show_debug_overlay debug_event debug_get_callstack alarm_get "+"alarm_set font_texture_page_size keyboard_set_map keyboard_get_map "+"keyboard_unset_map keyboard_check keyboard_check_pressed "+"keyboard_check_released keyboard_check_direct keyboard_get_numlock "+"keyboard_set_numlock keyboard_key_press keyboard_key_release "+"keyboard_clear io_clear mouse_check_button "+"mouse_check_button_pressed mouse_check_button_released "+"mouse_wheel_up mouse_wheel_down mouse_clear draw_self draw_sprite "+"draw_sprite_pos draw_sprite_ext draw_sprite_stretched "+"draw_sprite_stretched_ext draw_sprite_tiled draw_sprite_tiled_ext "+"draw_sprite_part draw_sprite_part_ext draw_sprite_general draw_clear "+"draw_clear_alpha draw_point draw_line draw_line_width draw_rectangle "+"draw_roundrect draw_roundrect_ext draw_triangle draw_circle "+"draw_ellipse draw_set_circle_precision draw_arrow draw_button "+"draw_path draw_healthbar draw_getpixel draw_getpixel_ext "+"draw_set_colour draw_set_color draw_set_alpha draw_get_colour "+"draw_get_color draw_get_alpha merge_colour make_colour_rgb "+"make_colour_hsv colour_get_red colour_get_green colour_get_blue "+"colour_get_hue colour_get_saturation colour_get_value merge_color "+"make_color_rgb make_color_hsv color_get_red color_get_green "+"color_get_blue color_get_hue color_get_saturation color_get_value "+"merge_color screen_save screen_save_part draw_set_font "+"draw_set_halign draw_set_valign draw_text draw_text_ext string_width "+"string_height string_width_ext string_height_ext "+"draw_text_transformed draw_text_ext_transformed draw_text_colour "+"draw_text_ext_colour draw_text_transformed_colour "+"draw_text_ext_transformed_colour draw_text_color draw_text_ext_color "+"draw_text_transformed_color draw_text_ext_transformed_color "+"draw_point_colour draw_line_colour draw_line_width_colour "+"draw_rectangle_colour draw_roundrect_colour "+"draw_roundrect_colour_ext draw_triangle_colour draw_circle_colour "+"draw_ellipse_colour draw_point_color draw_line_color "+"draw_line_width_color draw_rectangle_color draw_roundrect_color "+"draw_roundrect_color_ext draw_triangle_color draw_circle_color "+"draw_ellipse_color draw_primitive_begin draw_vertex "+"draw_vertex_colour draw_vertex_color draw_primitive_end "+"sprite_get_uvs font_get_uvs sprite_get_texture font_get_texture "+"texture_get_width texture_get_height texture_get_uvs "+"draw_primitive_begin_texture draw_vertex_texture "+"draw_vertex_texture_colour draw_vertex_texture_color "+"texture_global_scale surface_create surface_create_ext "+"surface_resize surface_free surface_exists surface_get_width "+"surface_get_height surface_get_texture surface_set_target "+"surface_set_target_ext surface_reset_target surface_depth_disable "+"surface_get_depth_disable draw_surface draw_surface_stretched "+"draw_surface_tiled draw_surface_part draw_surface_ext "+"draw_surface_stretched_ext draw_surface_tiled_ext "+"draw_surface_part_ext draw_surface_general surface_getpixel "+"surface_getpixel_ext surface_save surface_save_part surface_copy "+"surface_copy_part application_surface_draw_enable "+"application_get_position application_surface_enable "+"application_surface_is_enabled display_get_width display_get_height "+"display_get_orientation display_get_gui_width display_get_gui_height "+"display_reset display_mouse_get_x display_mouse_get_y "+"display_mouse_set display_set_ui_visibility "+"window_set_fullscreen window_get_fullscreen "+"window_set_caption window_set_min_width window_set_max_width "+"window_set_min_height window_set_max_height window_get_visible_rects "+"window_get_caption window_set_cursor window_get_cursor "+"window_set_colour window_get_colour window_set_color "+"window_get_color window_set_position window_set_size "+"window_set_rectangle window_center window_get_x window_get_y "+"window_get_width window_get_height window_mouse_get_x "+"window_mouse_get_y window_mouse_set window_view_mouse_get_x "+"window_view_mouse_get_y window_views_mouse_get_x "+"window_views_mouse_get_y audio_listener_position "+"audio_listener_velocity audio_listener_orientation "+"audio_emitter_position audio_emitter_create audio_emitter_free "+"audio_emitter_exists audio_emitter_pitch audio_emitter_velocity "+"audio_emitter_falloff audio_emitter_gain audio_play_sound "+"audio_play_sound_on audio_play_sound_at audio_stop_sound "+"audio_resume_music audio_music_is_playing audio_resume_sound "+"audio_pause_sound audio_pause_music audio_channel_num "+"audio_sound_length audio_get_type audio_falloff_set_model "+"audio_play_music audio_stop_music audio_master_gain audio_music_gain "+"audio_sound_gain audio_sound_pitch audio_stop_all audio_resume_all "+"audio_pause_all audio_is_playing audio_is_paused audio_exists "+"audio_sound_set_track_position audio_sound_get_track_position "+"audio_emitter_get_gain audio_emitter_get_pitch audio_emitter_get_x "+"audio_emitter_get_y audio_emitter_get_z audio_emitter_get_vx "+"audio_emitter_get_vy audio_emitter_get_vz "+"audio_listener_set_position audio_listener_set_velocity "+"audio_listener_set_orientation audio_listener_get_data "+"audio_set_master_gain audio_get_master_gain audio_sound_get_gain "+"audio_sound_get_pitch audio_get_name audio_sound_set_track_position "+"audio_sound_get_track_position audio_create_stream "+"audio_destroy_stream audio_create_sync_group "+"audio_destroy_sync_group audio_play_in_sync_group "+"audio_start_sync_group audio_stop_sync_group audio_pause_sync_group "+"audio_resume_sync_group audio_sync_group_get_track_pos "+"audio_sync_group_debug audio_sync_group_is_playing audio_debug "+"audio_group_load audio_group_unload audio_group_is_loaded "+"audio_group_load_progress audio_group_name audio_group_stop_all "+"audio_group_set_gain audio_create_buffer_sound "+"audio_free_buffer_sound audio_create_play_queue "+"audio_free_play_queue audio_queue_sound audio_get_recorder_count "+"audio_get_recorder_info audio_start_recording audio_stop_recording "+"audio_sound_get_listener_mask audio_emitter_get_listener_mask "+"audio_get_listener_mask audio_sound_set_listener_mask "+"audio_emitter_set_listener_mask audio_set_listener_mask "+"audio_get_listener_count audio_get_listener_info audio_system "+"show_message show_message_async clickable_add clickable_add_ext "+"clickable_change clickable_change_ext clickable_delete "+"clickable_exists clickable_set_style show_question "+"show_question_async get_integer get_string get_integer_async "+"get_string_async get_login_async get_open_filename get_save_filename "+"get_open_filename_ext get_save_filename_ext show_error "+"highscore_clear highscore_add highscore_value highscore_name "+"draw_highscore sprite_exists sprite_get_name sprite_get_number "+"sprite_get_width sprite_get_height sprite_get_xoffset "+"sprite_get_yoffset sprite_get_bbox_left sprite_get_bbox_right "+"sprite_get_bbox_top sprite_get_bbox_bottom sprite_save "+"sprite_save_strip sprite_set_cache_size sprite_set_cache_size_ext "+"sprite_get_tpe sprite_prefetch sprite_prefetch_multi sprite_flush "+"sprite_flush_multi sprite_set_speed sprite_get_speed_type "+"sprite_get_speed font_exists font_get_name font_get_fontname "+"font_get_bold font_get_italic font_get_first font_get_last "+"font_get_size font_set_cache_size path_exists path_get_name "+"path_get_length path_get_time path_get_kind path_get_closed "+"path_get_precision path_get_number path_get_point_x path_get_point_y "+"path_get_point_speed path_get_x path_get_y path_get_speed "+"script_exists script_get_name timeline_add timeline_delete "+"timeline_clear timeline_exists timeline_get_name "+"timeline_moment_clear timeline_moment_add_script timeline_size "+"timeline_max_moment object_exists object_get_name object_get_sprite "+"object_get_solid object_get_visible object_get_persistent "+"object_get_mask object_get_parent object_get_physics "+"object_is_ancestor room_exists room_get_name sprite_set_offset "+"sprite_duplicate sprite_assign sprite_merge sprite_add "+"sprite_replace sprite_create_from_surface sprite_add_from_surface "+"sprite_delete sprite_set_alpha_from_sprite sprite_collision_mask "+"font_add_enable_aa font_add_get_enable_aa font_add font_add_sprite "+"font_add_sprite_ext font_replace font_replace_sprite "+"font_replace_sprite_ext font_delete path_set_kind path_set_closed "+"path_set_precision path_add path_assign path_duplicate path_append "+"path_delete path_add_point path_insert_point path_change_point "+"path_delete_point path_clear_points path_reverse path_mirror "+"path_flip path_rotate path_rescale path_shift script_execute "+"object_set_sprite object_set_solid object_set_visible "+"object_set_persistent object_set_mask room_set_width room_set_height "+"room_set_persistent room_set_background_colour "+"room_set_background_color room_set_view room_set_viewport "+"room_get_viewport room_set_view_enabled room_add room_duplicate "+"room_assign room_instance_add room_instance_clear room_get_camera "+"room_set_camera asset_get_index asset_get_type "+"file_text_open_from_string file_text_open_read file_text_open_write "+"file_text_open_append file_text_close file_text_write_string "+"file_text_write_real file_text_writeln file_text_read_string "+"file_text_read_real file_text_readln file_text_eof file_text_eoln "+"file_exists file_delete file_rename file_copy directory_exists "+"directory_create directory_destroy file_find_first file_find_next "+"file_find_close file_attributes filename_name filename_path "+"filename_dir filename_drive filename_ext filename_change_ext "+"file_bin_open file_bin_rewrite file_bin_close file_bin_position "+"file_bin_size file_bin_seek file_bin_write_byte file_bin_read_byte "+"parameter_count parameter_string environment_get_variable "+"ini_open_from_string ini_open ini_close ini_read_string "+"ini_read_real ini_write_string ini_write_real ini_key_exists "+"ini_section_exists ini_key_delete ini_section_delete "+"ds_set_precision ds_exists ds_stack_create ds_stack_destroy "+"ds_stack_clear ds_stack_copy ds_stack_size ds_stack_empty "+"ds_stack_push ds_stack_pop ds_stack_top ds_stack_write ds_stack_read "+"ds_queue_create ds_queue_destroy ds_queue_clear ds_queue_copy "+"ds_queue_size ds_queue_empty ds_queue_enqueue ds_queue_dequeue "+"ds_queue_head ds_queue_tail ds_queue_write ds_queue_read "+"ds_list_create ds_list_destroy ds_list_clear ds_list_copy "+"ds_list_size ds_list_empty ds_list_add ds_list_insert "+"ds_list_replace ds_list_delete ds_list_find_index ds_list_find_value "+"ds_list_mark_as_list ds_list_mark_as_map ds_list_sort "+"ds_list_shuffle ds_list_write ds_list_read ds_list_set ds_map_create "+"ds_map_destroy ds_map_clear ds_map_copy ds_map_size ds_map_empty "+"ds_map_add ds_map_add_list ds_map_add_map ds_map_replace "+"ds_map_replace_map ds_map_replace_list ds_map_delete ds_map_exists "+"ds_map_find_value ds_map_find_previous ds_map_find_next "+"ds_map_find_first ds_map_find_last ds_map_write ds_map_read "+"ds_map_secure_save ds_map_secure_load ds_map_secure_load_buffer "+"ds_map_secure_save_buffer ds_map_set ds_priority_create "+"ds_priority_destroy ds_priority_clear ds_priority_copy "+"ds_priority_size ds_priority_empty ds_priority_add "+"ds_priority_change_priority ds_priority_find_priority "+"ds_priority_delete_value ds_priority_delete_min ds_priority_find_min "+"ds_priority_delete_max ds_priority_find_max ds_priority_write "+"ds_priority_read ds_grid_create ds_grid_destroy ds_grid_copy "+"ds_grid_resize ds_grid_width ds_grid_height ds_grid_clear "+"ds_grid_set ds_grid_add ds_grid_multiply ds_grid_set_region "+"ds_grid_add_region ds_grid_multiply_region ds_grid_set_disk "+"ds_grid_add_disk ds_grid_multiply_disk ds_grid_set_grid_region "+"ds_grid_add_grid_region ds_grid_multiply_grid_region ds_grid_get "+"ds_grid_get_sum ds_grid_get_max ds_grid_get_min ds_grid_get_mean "+"ds_grid_get_disk_sum ds_grid_get_disk_min ds_grid_get_disk_max "+"ds_grid_get_disk_mean ds_grid_value_exists ds_grid_value_x "+"ds_grid_value_y ds_grid_value_disk_exists ds_grid_value_disk_x "+"ds_grid_value_disk_y ds_grid_shuffle ds_grid_write ds_grid_read "+"ds_grid_sort ds_grid_set ds_grid_get effect_create_below "+"effect_create_above effect_clear part_type_create part_type_destroy "+"part_type_exists part_type_clear part_type_shape part_type_sprite "+"part_type_size part_type_scale part_type_orientation part_type_life "+"part_type_step part_type_death part_type_speed part_type_direction "+"part_type_gravity part_type_colour1 part_type_colour2 "+"part_type_colour3 part_type_colour_mix part_type_colour_rgb "+"part_type_colour_hsv part_type_color1 part_type_color2 "+"part_type_color3 part_type_color_mix part_type_color_rgb "+"part_type_color_hsv part_type_alpha1 part_type_alpha2 "+"part_type_alpha3 part_type_blend part_system_create "+"part_system_create_layer part_system_destroy part_system_exists "+"part_system_clear part_system_draw_order part_system_depth "+"part_system_position part_system_automatic_update "+"part_system_automatic_draw part_system_update part_system_drawit "+"part_system_get_layer part_system_layer part_particles_create "+"part_particles_create_colour part_particles_create_color "+"part_particles_clear part_particles_count part_emitter_create "+"part_emitter_destroy part_emitter_destroy_all part_emitter_exists "+"part_emitter_clear part_emitter_region part_emitter_burst "+"part_emitter_stream external_call external_define external_free "+"window_handle window_device matrix_get matrix_set "+"matrix_build_identity matrix_build matrix_build_lookat "+"matrix_build_projection_ortho matrix_build_projection_perspective "+"matrix_build_projection_perspective_fov matrix_multiply "+"matrix_transform_vertex matrix_stack_push matrix_stack_pop "+"matrix_stack_multiply matrix_stack_set matrix_stack_clear "+"matrix_stack_top matrix_stack_is_empty browser_input_capture "+"os_get_config os_get_info os_get_language os_get_region "+"os_lock_orientation display_get_dpi_x display_get_dpi_y "+"display_set_gui_size display_set_gui_maximise "+"display_set_gui_maximize device_mouse_dbclick_enable "+"display_set_timing_method display_get_timing_method "+"display_set_sleep_margin display_get_sleep_margin virtual_key_add "+"virtual_key_hide virtual_key_delete virtual_key_show "+"draw_enable_drawevent draw_enable_swf_aa draw_set_swf_aa_level "+"draw_get_swf_aa_level draw_texture_flush draw_flush "+"gpu_set_blendenable gpu_set_ztestenable gpu_set_zfunc "+"gpu_set_zwriteenable gpu_set_lightingenable gpu_set_fog "+"gpu_set_cullmode gpu_set_blendmode gpu_set_blendmode_ext "+"gpu_set_blendmode_ext_sepalpha gpu_set_colorwriteenable "+"gpu_set_colourwriteenable gpu_set_alphatestenable "+"gpu_set_alphatestref gpu_set_alphatestfunc gpu_set_texfilter "+"gpu_set_texfilter_ext gpu_set_texrepeat gpu_set_texrepeat_ext "+"gpu_set_tex_filter gpu_set_tex_filter_ext gpu_set_tex_repeat "+"gpu_set_tex_repeat_ext gpu_set_tex_mip_filter "+"gpu_set_tex_mip_filter_ext gpu_set_tex_mip_bias "+"gpu_set_tex_mip_bias_ext gpu_set_tex_min_mip gpu_set_tex_min_mip_ext "+"gpu_set_tex_max_mip gpu_set_tex_max_mip_ext gpu_set_tex_max_aniso "+"gpu_set_tex_max_aniso_ext gpu_set_tex_mip_enable "+"gpu_set_tex_mip_enable_ext gpu_get_blendenable gpu_get_ztestenable "+"gpu_get_zfunc gpu_get_zwriteenable gpu_get_lightingenable "+"gpu_get_fog gpu_get_cullmode gpu_get_blendmode gpu_get_blendmode_ext "+"gpu_get_blendmode_ext_sepalpha gpu_get_blendmode_src "+"gpu_get_blendmode_dest gpu_get_blendmode_srcalpha "+"gpu_get_blendmode_destalpha gpu_get_colorwriteenable "+"gpu_get_colourwriteenable gpu_get_alphatestenable "+"gpu_get_alphatestref gpu_get_alphatestfunc gpu_get_texfilter "+"gpu_get_texfilter_ext gpu_get_texrepeat gpu_get_texrepeat_ext "+"gpu_get_tex_filter gpu_get_tex_filter_ext gpu_get_tex_repeat "+"gpu_get_tex_repeat_ext gpu_get_tex_mip_filter "+"gpu_get_tex_mip_filter_ext gpu_get_tex_mip_bias "+"gpu_get_tex_mip_bias_ext gpu_get_tex_min_mip gpu_get_tex_min_mip_ext "+"gpu_get_tex_max_mip gpu_get_tex_max_mip_ext gpu_get_tex_max_aniso "+"gpu_get_tex_max_aniso_ext gpu_get_tex_mip_enable "+"gpu_get_tex_mip_enable_ext gpu_push_state gpu_pop_state "+"gpu_get_state gpu_set_state draw_light_define_ambient "+"draw_light_define_direction draw_light_define_point "+"draw_light_enable draw_set_lighting draw_light_get_ambient "+"draw_light_get draw_get_lighting shop_leave_rating url_get_domain "+"url_open url_open_ext url_open_full get_timer achievement_login "+"achievement_logout achievement_post achievement_increment "+"achievement_post_score achievement_available "+"achievement_show_achievements achievement_show_leaderboards "+"achievement_load_friends achievement_load_leaderboard "+"achievement_send_challenge achievement_load_progress "+"achievement_reset achievement_login_status achievement_get_pic "+"achievement_show_challenge_notifications achievement_get_challenges "+"achievement_event achievement_show achievement_get_info "+"cloud_file_save cloud_string_save cloud_synchronise ads_enable "+"ads_disable ads_setup ads_engagement_launch ads_engagement_available "+"ads_engagement_active ads_event ads_event_preload "+"ads_set_reward_callback ads_get_display_height ads_get_display_width "+"ads_move ads_interstitial_available ads_interstitial_display "+"device_get_tilt_x device_get_tilt_y device_get_tilt_z "+"device_is_keypad_open device_mouse_check_button "+"device_mouse_check_button_pressed device_mouse_check_button_released "+"device_mouse_x device_mouse_y device_mouse_raw_x device_mouse_raw_y "+"device_mouse_x_to_gui device_mouse_y_to_gui iap_activate iap_status "+"iap_enumerate_products iap_restore_all iap_acquire iap_consume "+"iap_product_details iap_purchase_details facebook_init "+"facebook_login facebook_status facebook_graph_request "+"facebook_dialog facebook_logout facebook_launch_offerwall "+"facebook_post_message facebook_send_invite facebook_user_id "+"facebook_accesstoken facebook_check_permission "+"facebook_request_read_permissions "+"facebook_request_publish_permissions gamepad_is_supported "+"gamepad_get_device_count gamepad_is_connected "+"gamepad_get_description gamepad_get_button_threshold "+"gamepad_set_button_threshold gamepad_get_axis_deadzone "+"gamepad_set_axis_deadzone gamepad_button_count gamepad_button_check "+"gamepad_button_check_pressed gamepad_button_check_released "+"gamepad_button_value gamepad_axis_count gamepad_axis_value "+"gamepad_set_vibration gamepad_set_colour gamepad_set_color "+"os_is_paused window_has_focus code_is_compiled http_get "+"http_get_file http_post_string http_request json_encode json_decode "+"zip_unzip load_csv base64_encode base64_decode md5_string_unicode "+"md5_string_utf8 md5_file os_is_network_connected sha1_string_unicode "+"sha1_string_utf8 sha1_file os_powersave_enable analytics_event "+"analytics_event_ext win8_livetile_tile_notification "+"win8_livetile_tile_clear win8_livetile_badge_notification "+"win8_livetile_badge_clear win8_livetile_queue_enable "+"win8_secondarytile_pin win8_secondarytile_badge_notification "+"win8_secondarytile_delete win8_livetile_notification_begin "+"win8_livetile_notification_secondary_begin "+"win8_livetile_notification_expiry win8_livetile_notification_tag "+"win8_livetile_notification_text_add "+"win8_livetile_notification_image_add win8_livetile_notification_end "+"win8_appbar_enable win8_appbar_add_element "+"win8_appbar_remove_element win8_settingscharm_add_entry "+"win8_settingscharm_add_html_entry win8_settingscharm_add_xaml_entry "+"win8_settingscharm_set_xaml_property "+"win8_settingscharm_get_xaml_property win8_settingscharm_remove_entry "+"win8_share_image win8_share_screenshot win8_share_file "+"win8_share_url win8_share_text win8_search_enable "+"win8_search_disable win8_search_add_suggestions "+"win8_device_touchscreen_available win8_license_initialize_sandbox "+"win8_license_trial_version winphone_license_trial_version "+"winphone_tile_title winphone_tile_count winphone_tile_back_title "+"winphone_tile_back_content winphone_tile_back_content_wide "+"winphone_tile_front_image winphone_tile_front_image_small "+"winphone_tile_front_image_wide winphone_tile_back_image "+"winphone_tile_back_image_wide winphone_tile_background_colour "+"winphone_tile_background_color winphone_tile_icon_image "+"winphone_tile_small_icon_image winphone_tile_wide_content "+"winphone_tile_cycle_images winphone_tile_small_background_image "+"physics_world_create physics_world_gravity "+"physics_world_update_speed physics_world_update_iterations "+"physics_world_draw_debug physics_pause_enable physics_fixture_create "+"physics_fixture_set_kinematic physics_fixture_set_density "+"physics_fixture_set_awake physics_fixture_set_restitution "+"physics_fixture_set_friction physics_fixture_set_collision_group "+"physics_fixture_set_sensor physics_fixture_set_linear_damping "+"physics_fixture_set_angular_damping physics_fixture_set_circle_shape "+"physics_fixture_set_box_shape physics_fixture_set_edge_shape "+"physics_fixture_set_polygon_shape physics_fixture_set_chain_shape "+"physics_fixture_add_point physics_fixture_bind "+"physics_fixture_bind_ext physics_fixture_delete physics_apply_force "+"physics_apply_impulse physics_apply_angular_impulse "+"physics_apply_local_force physics_apply_local_impulse "+"physics_apply_torque physics_mass_properties physics_draw_debug "+"physics_test_overlap physics_remove_fixture physics_set_friction "+"physics_set_density physics_set_restitution physics_get_friction "+"physics_get_density physics_get_restitution "+"physics_joint_distance_create physics_joint_rope_create "+"physics_joint_revolute_create physics_joint_prismatic_create "+"physics_joint_pulley_create physics_joint_wheel_create "+"physics_joint_weld_create physics_joint_friction_create "+"physics_joint_gear_create physics_joint_enable_motor "+"physics_joint_get_value physics_joint_set_value physics_joint_delete "+"physics_particle_create physics_particle_delete "+"physics_particle_delete_region_circle "+"physics_particle_delete_region_box "+"physics_particle_delete_region_poly physics_particle_set_flags "+"physics_particle_set_category_flags physics_particle_draw "+"physics_particle_draw_ext physics_particle_count "+"physics_particle_get_data physics_particle_get_data_particle "+"physics_particle_group_begin physics_particle_group_circle "+"physics_particle_group_box physics_particle_group_polygon "+"physics_particle_group_add_point physics_particle_group_end "+"physics_particle_group_join physics_particle_group_delete "+"physics_particle_group_count physics_particle_group_get_data "+"physics_particle_group_get_mass physics_particle_group_get_inertia "+"physics_particle_group_get_centre_x "+"physics_particle_group_get_centre_y physics_particle_group_get_vel_x "+"physics_particle_group_get_vel_y physics_particle_group_get_ang_vel "+"physics_particle_group_get_x physics_particle_group_get_y "+"physics_particle_group_get_angle physics_particle_set_group_flags "+"physics_particle_get_group_flags physics_particle_get_max_count "+"physics_particle_get_radius physics_particle_get_density "+"physics_particle_get_damping physics_particle_get_gravity_scale "+"physics_particle_set_max_count physics_particle_set_radius "+"physics_particle_set_density physics_particle_set_damping "+"physics_particle_set_gravity_scale network_create_socket "+"network_create_socket_ext network_create_server "+"network_create_server_raw network_connect network_connect_raw "+"network_send_packet network_send_raw network_send_broadcast "+"network_send_udp network_send_udp_raw network_set_timeout "+"network_set_config network_resolve network_destroy buffer_create "+"buffer_write buffer_read buffer_seek buffer_get_surface "+"buffer_set_surface buffer_delete buffer_exists buffer_get_type "+"buffer_get_alignment buffer_poke buffer_peek buffer_save "+"buffer_save_ext buffer_load buffer_load_ext buffer_load_partial "+"buffer_copy buffer_fill buffer_get_size buffer_tell buffer_resize "+"buffer_md5 buffer_sha1 buffer_base64_encode buffer_base64_decode "+"buffer_base64_decode_ext buffer_sizeof buffer_get_address "+"buffer_create_from_vertex_buffer "+"buffer_create_from_vertex_buffer_ext buffer_copy_from_vertex_buffer "+"buffer_async_group_begin buffer_async_group_option "+"buffer_async_group_end buffer_load_async buffer_save_async "+"gml_release_mode gml_pragma steam_activate_overlay "+"steam_is_overlay_enabled steam_is_overlay_activated "+"steam_get_persona_name steam_initialised "+"steam_is_cloud_enabled_for_app steam_is_cloud_enabled_for_account "+"steam_file_persisted steam_get_quota_total steam_get_quota_free "+"steam_file_write steam_file_write_file steam_file_read "+"steam_file_delete steam_file_exists steam_file_size steam_file_share "+"steam_is_screenshot_requested steam_send_screenshot "+"steam_is_user_logged_on steam_get_user_steam_id steam_user_owns_dlc "+"steam_user_installed_dlc steam_set_achievement steam_get_achievement "+"steam_clear_achievement steam_set_stat_int steam_set_stat_float "+"steam_set_stat_avg_rate steam_get_stat_int steam_get_stat_float "+"steam_get_stat_avg_rate steam_reset_all_stats "+"steam_reset_all_stats_achievements steam_stats_ready "+"steam_create_leaderboard steam_upload_score steam_upload_score_ext "+"steam_download_scores_around_user steam_download_scores "+"steam_download_friends_scores steam_upload_score_buffer "+"steam_upload_score_buffer_ext steam_current_game_language "+"steam_available_languages steam_activate_overlay_browser "+"steam_activate_overlay_user steam_activate_overlay_store "+"steam_get_user_persona_name steam_get_app_id "+"steam_get_user_account_id steam_ugc_download steam_ugc_create_item "+"steam_ugc_start_item_update steam_ugc_set_item_title "+"steam_ugc_set_item_description steam_ugc_set_item_visibility "+"steam_ugc_set_item_tags steam_ugc_set_item_content "+"steam_ugc_set_item_preview steam_ugc_submit_item_update "+"steam_ugc_get_item_update_progress steam_ugc_subscribe_item "+"steam_ugc_unsubscribe_item steam_ugc_num_subscribed_items "+"steam_ugc_get_subscribed_items steam_ugc_get_item_install_info "+"steam_ugc_get_item_update_info steam_ugc_request_item_details "+"steam_ugc_create_query_user steam_ugc_create_query_user_ex "+"steam_ugc_create_query_all steam_ugc_create_query_all_ex "+"steam_ugc_query_set_cloud_filename_filter "+"steam_ugc_query_set_match_any_tag steam_ugc_query_set_search_text "+"steam_ugc_query_set_ranked_by_trend_days "+"steam_ugc_query_add_required_tag steam_ugc_query_add_excluded_tag "+"steam_ugc_query_set_return_long_description "+"steam_ugc_query_set_return_total_only "+"steam_ugc_query_set_allow_cached_response steam_ugc_send_query "+"shader_set shader_get_name shader_reset shader_current "+"shader_is_compiled shader_get_sampler_index shader_get_uniform "+"shader_set_uniform_i shader_set_uniform_i_array shader_set_uniform_f "+"shader_set_uniform_f_array shader_set_uniform_matrix "+"shader_set_uniform_matrix_array shader_enable_corner_id "+"texture_set_stage texture_get_texel_width texture_get_texel_height "+"shaders_are_supported vertex_format_begin vertex_format_end "+"vertex_format_delete vertex_format_add_position "+"vertex_format_add_position_3d vertex_format_add_colour "+"vertex_format_add_color vertex_format_add_normal "+"vertex_format_add_texcoord vertex_format_add_textcoord "+"vertex_format_add_custom vertex_create_buffer "+"vertex_create_buffer_ext vertex_delete_buffer vertex_begin "+"vertex_end vertex_position vertex_position_3d vertex_colour "+"vertex_color vertex_argb vertex_texcoord vertex_normal vertex_float1 "+"vertex_float2 vertex_float3 vertex_float4 vertex_ubyte4 "+"vertex_submit vertex_freeze vertex_get_number vertex_get_buffer_size "+"vertex_create_buffer_from_buffer "+"vertex_create_buffer_from_buffer_ext push_local_notification "+"push_get_first_local_notification push_get_next_local_notification "+"push_cancel_local_notification skeleton_animation_set "+"skeleton_animation_get skeleton_animation_mix "+"skeleton_animation_set_ext skeleton_animation_get_ext "+"skeleton_animation_get_duration skeleton_animation_get_frames "+"skeleton_animation_clear skeleton_skin_set skeleton_skin_get "+"skeleton_attachment_set skeleton_attachment_get "+"skeleton_attachment_create skeleton_collision_draw_set "+"skeleton_bone_data_get skeleton_bone_data_set "+"skeleton_bone_state_get skeleton_bone_state_set skeleton_get_minmax "+"skeleton_get_num_bounds skeleton_get_bounds "+"skeleton_animation_get_frame skeleton_animation_set_frame "+"draw_skeleton draw_skeleton_time draw_skeleton_instance "+"draw_skeleton_collision skeleton_animation_list skeleton_skin_list "+"skeleton_slot_data layer_get_id layer_get_id_at_depth "+"layer_get_depth layer_create layer_destroy layer_destroy_instances "+"layer_add_instance layer_has_instance layer_set_visible "+"layer_get_visible layer_exists layer_x layer_y layer_get_x "+"layer_get_y layer_hspeed layer_vspeed layer_get_hspeed "+"layer_get_vspeed layer_script_begin layer_script_end layer_shader "+"layer_get_script_begin layer_get_script_end layer_get_shader "+"layer_set_target_room layer_get_target_room layer_reset_target_room "+"layer_get_all layer_get_all_elements layer_get_name layer_depth "+"layer_get_element_layer layer_get_element_type layer_element_move "+"layer_force_draw_depth layer_is_draw_depth_forced "+"layer_get_forced_depth layer_background_get_id "+"layer_background_exists layer_background_create "+"layer_background_destroy layer_background_visible "+"layer_background_change layer_background_sprite "+"layer_background_htiled layer_background_vtiled "+"layer_background_stretch layer_background_yscale "+"layer_background_xscale layer_background_blend "+"layer_background_alpha layer_background_index layer_background_speed "+"layer_background_get_visible layer_background_get_sprite "+"layer_background_get_htiled layer_background_get_vtiled "+"layer_background_get_stretch layer_background_get_yscale "+"layer_background_get_xscale layer_background_get_blend "+"layer_background_get_alpha layer_background_get_index "+"layer_background_get_speed layer_sprite_get_id layer_sprite_exists "+"layer_sprite_create layer_sprite_destroy layer_sprite_change "+"layer_sprite_index layer_sprite_speed layer_sprite_xscale "+"layer_sprite_yscale layer_sprite_angle layer_sprite_blend "+"layer_sprite_alpha layer_sprite_x layer_sprite_y "+"layer_sprite_get_sprite layer_sprite_get_index "+"layer_sprite_get_speed layer_sprite_get_xscale "+"layer_sprite_get_yscale layer_sprite_get_angle "+"layer_sprite_get_blend layer_sprite_get_alpha layer_sprite_get_x "+"layer_sprite_get_y layer_tilemap_get_id layer_tilemap_exists "+"layer_tilemap_create layer_tilemap_destroy tilemap_tileset tilemap_x "+"tilemap_y tilemap_set tilemap_set_at_pixel tilemap_get_tileset "+"tilemap_get_tile_width tilemap_get_tile_height tilemap_get_width "+"tilemap_get_height tilemap_get_x tilemap_get_y tilemap_get "+"tilemap_get_at_pixel tilemap_get_cell_x_at_pixel "+"tilemap_get_cell_y_at_pixel tilemap_clear draw_tilemap draw_tile "+"tilemap_set_global_mask tilemap_get_global_mask tilemap_set_mask "+"tilemap_get_mask tilemap_get_frame tile_set_empty tile_set_index "+"tile_set_flip tile_set_mirror tile_set_rotate tile_get_empty "+"tile_get_index tile_get_flip tile_get_mirror tile_get_rotate "+"layer_tile_exists layer_tile_create layer_tile_destroy "+"layer_tile_change layer_tile_xscale layer_tile_yscale "+"layer_tile_blend layer_tile_alpha layer_tile_x layer_tile_y "+"layer_tile_region layer_tile_visible layer_tile_get_sprite "+"layer_tile_get_xscale layer_tile_get_yscale layer_tile_get_blend "+"layer_tile_get_alpha layer_tile_get_x layer_tile_get_y "+"layer_tile_get_region layer_tile_get_visible "+"layer_instance_get_instance instance_activate_layer "+"instance_deactivate_layer camera_create camera_create_view "+"camera_destroy camera_apply camera_get_active camera_get_default "+"camera_set_default camera_set_view_mat camera_set_proj_mat "+"camera_set_update_script camera_set_begin_script "+"camera_set_end_script camera_set_view_pos camera_set_view_size "+"camera_set_view_speed camera_set_view_border camera_set_view_angle "+"camera_set_view_target camera_get_view_mat camera_get_proj_mat "+"camera_get_update_script camera_get_begin_script "+"camera_get_end_script camera_get_view_x camera_get_view_y "+"camera_get_view_width camera_get_view_height camera_get_view_speed_x "+"camera_get_view_speed_y camera_get_view_border_x "+"camera_get_view_border_y camera_get_view_angle "+"camera_get_view_target view_get_camera view_get_visible "+"view_get_xport view_get_yport view_get_wport view_get_hport "+"view_get_surface_id view_set_camera view_set_visible view_set_xport "+"view_set_yport view_set_wport view_set_hport view_set_surface_id "+"gesture_drag_time gesture_drag_distance gesture_flick_speed "+"gesture_double_tap_time gesture_double_tap_distance "+"gesture_pinch_distance gesture_pinch_angle_towards "+"gesture_pinch_angle_away gesture_rotate_time gesture_rotate_angle "+"gesture_tap_count gesture_get_drag_time gesture_get_drag_distance "+"gesture_get_flick_speed gesture_get_double_tap_time "+"gesture_get_double_tap_distance gesture_get_pinch_distance "+"gesture_get_pinch_angle_towards gesture_get_pinch_angle_away "+"gesture_get_rotate_time gesture_get_rotate_angle "+"gesture_get_tap_count keyboard_virtual_show keyboard_virtual_hide "+"keyboard_virtual_status keyboard_virtual_height",literal:"self other all noone global local undefined pointer_invalid "+"pointer_null path_action_stop path_action_restart "+"path_action_continue path_action_reverse true false pi GM_build_date "+"GM_version GM_runtime_version  timezone_local timezone_utc "+"gamespeed_fps gamespeed_microseconds  ev_create ev_destroy ev_step "+"ev_alarm ev_keyboard ev_mouse ev_collision ev_other ev_draw "+"ev_draw_begin ev_draw_end ev_draw_pre ev_draw_post ev_keypress "+"ev_keyrelease ev_trigger ev_left_button ev_right_button "+"ev_middle_button ev_no_button ev_left_press ev_right_press "+"ev_middle_press ev_left_release ev_right_release ev_middle_release "+"ev_mouse_enter ev_mouse_leave ev_mouse_wheel_up ev_mouse_wheel_down "+"ev_global_left_button ev_global_right_button ev_global_middle_button "+"ev_global_left_press ev_global_right_press ev_global_middle_press "+"ev_global_left_release ev_global_right_release "+"ev_global_middle_release ev_joystick1_left ev_joystick1_right "+"ev_joystick1_up ev_joystick1_down ev_joystick1_button1 "+"ev_joystick1_button2 ev_joystick1_button3 ev_joystick1_button4 "+"ev_joystick1_button5 ev_joystick1_button6 ev_joystick1_button7 "+"ev_joystick1_button8 ev_joystick2_left ev_joystick2_right "+"ev_joystick2_up ev_joystick2_down ev_joystick2_button1 "+"ev_joystick2_button2 ev_joystick2_button3 ev_joystick2_button4 "+"ev_joystick2_button5 ev_joystick2_button6 ev_joystick2_button7 "+"ev_joystick2_button8 ev_outside ev_boundary ev_game_start "+"ev_game_end ev_room_start ev_room_end ev_no_more_lives "+"ev_animation_end ev_end_of_path ev_no_more_health ev_close_button "+"ev_user0 ev_user1 ev_user2 ev_user3 ev_user4 ev_user5 ev_user6 "+"ev_user7 ev_user8 ev_user9 ev_user10 ev_user11 ev_user12 ev_user13 "+"ev_user14 ev_user15 ev_step_normal ev_step_begin ev_step_end ev_gui "+"ev_gui_begin ev_gui_end ev_cleanup ev_gesture ev_gesture_tap "+"ev_gesture_double_tap ev_gesture_drag_start ev_gesture_dragging "+"ev_gesture_drag_end ev_gesture_flick ev_gesture_pinch_start "+"ev_gesture_pinch_in ev_gesture_pinch_out ev_gesture_pinch_end "+"ev_gesture_rotate_start ev_gesture_rotating ev_gesture_rotate_end "+"ev_global_gesture_tap ev_global_gesture_double_tap "+"ev_global_gesture_drag_start ev_global_gesture_dragging "+"ev_global_gesture_drag_end ev_global_gesture_flick "+"ev_global_gesture_pinch_start ev_global_gesture_pinch_in "+"ev_global_gesture_pinch_out ev_global_gesture_pinch_end "+"ev_global_gesture_rotate_start ev_global_gesture_rotating "+"ev_global_gesture_rotate_end vk_nokey vk_anykey vk_enter vk_return "+"vk_shift vk_control vk_alt vk_escape vk_space vk_backspace vk_tab "+"vk_pause vk_printscreen vk_left vk_right vk_up vk_down vk_home "+"vk_end vk_delete vk_insert vk_pageup vk_pagedown vk_f1 vk_f2 vk_f3 "+"vk_f4 vk_f5 vk_f6 vk_f7 vk_f8 vk_f9 vk_f10 vk_f11 vk_f12 vk_numpad0 "+"vk_numpad1 vk_numpad2 vk_numpad3 vk_numpad4 vk_numpad5 vk_numpad6 "+"vk_numpad7 vk_numpad8 vk_numpad9 vk_divide vk_multiply vk_subtract "+"vk_add vk_decimal vk_lshift vk_lcontrol vk_lalt vk_rshift "+"vk_rcontrol vk_ralt  mb_any mb_none mb_left mb_right mb_middle "+"c_aqua c_black c_blue c_dkgray c_fuchsia c_gray c_green c_lime "+"c_ltgray c_maroon c_navy c_olive c_purple c_red c_silver c_teal "+"c_white c_yellow c_orange fa_left fa_center fa_right fa_top "+"fa_middle fa_bottom pr_pointlist pr_linelist pr_linestrip "+"pr_trianglelist pr_trianglestrip pr_trianglefan bm_complex bm_normal "+"bm_add bm_max bm_subtract bm_zero bm_one bm_src_colour "+"bm_inv_src_colour bm_src_color bm_inv_src_color bm_src_alpha "+"bm_inv_src_alpha bm_dest_alpha bm_inv_dest_alpha bm_dest_colour "+"bm_inv_dest_colour bm_dest_color bm_inv_dest_color bm_src_alpha_sat "+"tf_point tf_linear tf_anisotropic mip_off mip_on mip_markedonly "+"audio_falloff_none audio_falloff_inverse_distance "+"audio_falloff_inverse_distance_clamped audio_falloff_linear_distance "+"audio_falloff_linear_distance_clamped "+"audio_falloff_exponent_distance "+"audio_falloff_exponent_distance_clamped audio_old_system "+"audio_new_system audio_mono audio_stereo audio_3d cr_default cr_none "+"cr_arrow cr_cross cr_beam cr_size_nesw cr_size_ns cr_size_nwse "+"cr_size_we cr_uparrow cr_hourglass cr_drag cr_appstart cr_handpoint "+"cr_size_all spritespeed_framespersecond "+"spritespeed_framespergameframe asset_object asset_unknown "+"asset_sprite asset_sound asset_room asset_path asset_script "+"asset_font asset_timeline asset_tiles asset_shader fa_readonly "+"fa_hidden fa_sysfile fa_volumeid fa_directory fa_archive  "+"ds_type_map ds_type_list ds_type_stack ds_type_queue ds_type_grid "+"ds_type_priority ef_explosion ef_ring ef_ellipse ef_firework "+"ef_smoke ef_smokeup ef_star ef_spark ef_flare ef_cloud ef_rain "+"ef_snow pt_shape_pixel pt_shape_disk pt_shape_square pt_shape_line "+"pt_shape_star pt_shape_circle pt_shape_ring pt_shape_sphere "+"pt_shape_flare pt_shape_spark pt_shape_explosion pt_shape_cloud "+"pt_shape_smoke pt_shape_snow ps_distr_linear ps_distr_gaussian "+"ps_distr_invgaussian ps_shape_rectangle ps_shape_ellipse "+"ps_shape_diamond ps_shape_line ty_real ty_string dll_cdecl "+"dll_stdcall matrix_view matrix_projection matrix_world os_win32 "+"os_windows os_macosx os_ios os_android os_symbian os_linux "+"os_unknown os_winphone os_tizen os_win8native "+"os_wiiu os_3ds  os_psvita os_bb10 os_ps4 os_xboxone "+"os_ps3 os_xbox360 os_uwp os_tvos os_switch "+"browser_not_a_browser browser_unknown browser_ie browser_firefox "+"browser_chrome browser_safari browser_safari_mobile browser_opera "+"browser_tizen browser_edge browser_windows_store browser_ie_mobile  "+"device_ios_unknown device_ios_iphone device_ios_iphone_retina "+"device_ios_ipad device_ios_ipad_retina device_ios_iphone5 "+"device_ios_iphone6 device_ios_iphone6plus device_emulator "+"device_tablet display_landscape display_landscape_flipped "+"display_portrait display_portrait_flipped tm_sleep tm_countvsyncs "+"of_challenge_win of_challen ge_lose of_challenge_tie "+"leaderboard_type_number leaderboard_type_time_mins_secs "+"cmpfunc_never cmpfunc_less cmpfunc_equal cmpfunc_lessequal "+"cmpfunc_greater cmpfunc_notequal cmpfunc_greaterequal cmpfunc_always "+"cull_noculling cull_clockwise cull_counterclockwise lighttype_dir "+"lighttype_point iap_ev_storeload iap_ev_product iap_ev_purchase "+"iap_ev_consume iap_ev_restore iap_storeload_ok iap_storeload_failed "+"iap_status_uninitialised iap_status_unavailable iap_status_loading "+"iap_status_available iap_status_processing iap_status_restoring "+"iap_failed iap_unavailable iap_available iap_purchased iap_canceled "+"iap_refunded fb_login_default fb_login_fallback_to_webview "+"fb_login_no_fallback_to_webview fb_login_forcing_webview "+"fb_login_use_system_account fb_login_forcing_safari  "+"phy_joint_anchor_1_x phy_joint_anchor_1_y phy_joint_anchor_2_x "+"phy_joint_anchor_2_y phy_joint_reaction_force_x "+"phy_joint_reaction_force_y phy_joint_reaction_torque "+"phy_joint_motor_speed phy_joint_angle phy_joint_motor_torque "+"phy_joint_max_motor_torque phy_joint_translation phy_joint_speed "+"phy_joint_motor_force phy_joint_max_motor_force phy_joint_length_1 "+"phy_joint_length_2 phy_joint_damping_ratio phy_joint_frequency "+"phy_joint_lower_angle_limit phy_joint_upper_angle_limit "+"phy_joint_angle_limits phy_joint_max_length phy_joint_max_torque "+"phy_joint_max_force phy_debug_render_aabb "+"phy_debug_render_collision_pairs phy_debug_render_coms "+"phy_debug_render_core_shapes phy_debug_render_joints "+"phy_debug_render_obb phy_debug_render_shapes  "+"phy_particle_flag_water phy_particle_flag_zombie "+"phy_particle_flag_wall phy_particle_flag_spring "+"phy_particle_flag_elastic phy_particle_flag_viscous "+"phy_particle_flag_powder phy_particle_flag_tensile "+"phy_particle_flag_colourmixing phy_particle_flag_colormixing "+"phy_particle_group_flag_solid phy_particle_group_flag_rigid "+"phy_particle_data_flag_typeflags phy_particle_data_flag_position "+"phy_particle_data_flag_velocity phy_particle_data_flag_colour "+"phy_particle_data_flag_color phy_particle_data_flag_category  "+"achievement_our_info achievement_friends_info "+"achievement_leaderboard_info achievement_achievement_info "+"achievement_filter_all_players achievement_filter_friends_only "+"achievement_filter_favorites_only "+"achievement_type_achievement_challenge "+"achievement_type_score_challenge achievement_pic_loaded  "+"achievement_show_ui achievement_show_profile "+"achievement_show_leaderboard achievement_show_achievement "+"achievement_show_bank achievement_show_friend_picker "+"achievement_show_purchase_prompt network_socket_tcp "+"network_socket_udp network_socket_bluetooth network_type_connect "+"network_type_disconnect network_type_data "+"network_type_non_blocking_connect network_config_connect_timeout "+"network_config_use_non_blocking_socket "+"network_config_enable_reliable_udp "+"network_config_disable_reliable_udp buffer_fixed buffer_grow "+"buffer_wrap buffer_fast buffer_vbuffer buffer_network buffer_u8 "+"buffer_s8 buffer_u16 buffer_s16 buffer_u32 buffer_s32 buffer_u64 "+"buffer_f16 buffer_f32 buffer_f64 buffer_bool buffer_text "+"buffer_string buffer_surface_copy buffer_seek_start "+"buffer_seek_relative buffer_seek_end "+"buffer_generalerror buffer_outofspace buffer_outofbounds "+"buffer_invalidtype  text_type button_type input_type ANSI_CHARSET "+"DEFAULT_CHARSET EASTEUROPE_CHARSET RUSSIAN_CHARSET SYMBOL_CHARSET "+"SHIFTJIS_CHARSET HANGEUL_CHARSET GB2312_CHARSET CHINESEBIG5_CHARSET "+"JOHAB_CHARSET HEBREW_CHARSET ARABIC_CHARSET GREEK_CHARSET "+"TURKISH_CHARSET VIETNAMESE_CHARSET THAI_CHARSET MAC_CHARSET "+"BALTIC_CHARSET OEM_CHARSET  gp_face1 gp_face2 gp_face3 gp_face4 "+"gp_shoulderl gp_shoulderr gp_shoulderlb gp_shoulderrb gp_select "+"gp_start gp_stickl gp_stickr gp_padu gp_padd gp_padl gp_padr "+"gp_axislh gp_axislv gp_axisrh gp_axisrv ov_friends ov_community "+"ov_players ov_settings ov_gamegroup ov_achievements lb_sort_none "+"lb_sort_ascending lb_sort_descending lb_disp_none lb_disp_numeric "+"lb_disp_time_sec lb_disp_time_ms ugc_result_success "+"ugc_filetype_community ugc_filetype_microtrans ugc_visibility_public "+"ugc_visibility_friends_only ugc_visibility_private "+"ugc_query_RankedByVote ugc_query_RankedByPublicationDate "+"ugc_query_AcceptedForGameRankedByAcceptanceDate "+"ugc_query_RankedByTrend "+"ugc_query_FavoritedByFriendsRankedByPublicationDate "+"ugc_query_CreatedByFriendsRankedByPublicationDate "+"ugc_query_RankedByNumTimesReported "+"ugc_query_CreatedByFollowedUsersRankedByPublicationDate "+"ugc_query_NotYetRated ugc_query_RankedByTotalVotesAsc "+"ugc_query_RankedByVotesUp ugc_query_RankedByTextSearch "+"ugc_sortorder_CreationOrderDesc ugc_sortorder_CreationOrderAsc "+"ugc_sortorder_TitleAsc ugc_sortorder_LastUpdatedDesc "+"ugc_sortorder_SubscriptionDateDesc ugc_sortorder_VoteScoreDesc "+"ugc_sortorder_ForModeration ugc_list_Published ugc_list_VotedOn "+"ugc_list_VotedUp ugc_list_VotedDown ugc_list_WillVoteLater "+"ugc_list_Favorited ugc_list_Subscribed ugc_list_UsedOrPlayed "+"ugc_list_Followed ugc_match_Items ugc_match_Items_Mtx "+"ugc_match_Items_ReadyToUse ugc_match_Collections ugc_match_Artwork "+"ugc_match_Videos ugc_match_Screenshots ugc_match_AllGuides "+"ugc_match_WebGuides ugc_match_IntegratedGuides "+"ugc_match_UsableInGame ugc_match_ControllerBindings  "+"vertex_usage_position vertex_usage_colour vertex_usage_color "+"vertex_usage_normal vertex_usage_texcoord vertex_usage_textcoord "+"vertex_usage_blendweight vertex_usage_blendindices "+"vertex_usage_psize vertex_usage_tangent vertex_usage_binormal "+"vertex_usage_fog vertex_usage_depth vertex_usage_sample "+"vertex_type_float1 vertex_type_float2 vertex_type_float3 "+"vertex_type_float4 vertex_type_colour vertex_type_color "+"vertex_type_ubyte4 layerelementtype_undefined "+"layerelementtype_background layerelementtype_instance "+"layerelementtype_oldtilemap layerelementtype_sprite "+"layerelementtype_tilemap layerelementtype_particlesystem "+"layerelementtype_tile tile_rotate tile_flip tile_mirror "+"tile_index_mask kbv_type_default kbv_type_ascii kbv_type_url "+"kbv_type_email kbv_type_numbers kbv_type_phone kbv_type_phone_name "+"kbv_returnkey_default kbv_returnkey_go kbv_returnkey_google "+"kbv_returnkey_join kbv_returnkey_next kbv_returnkey_route "+"kbv_returnkey_search kbv_returnkey_send kbv_returnkey_yahoo "+"kbv_returnkey_done kbv_returnkey_continue kbv_returnkey_emergency "+"kbv_autocapitalize_none kbv_autocapitalize_words "+"kbv_autocapitalize_sentences kbv_autocapitalize_characters",symbol:"argument_relative argument argument0 argument1 argument2 "+"argument3 argument4 argument5 argument6 argument7 argument8 "+"argument9 argument10 argument11 argument12 argument13 argument14 "+"argument15 argument_count x y xprevious yprevious xstart ystart "+"hspeed vspeed direction speed friction gravity gravity_direction "+"path_index path_position path_positionprevious path_speed "+"path_scale path_orientation path_endaction object_index id solid "+"persistent mask_index instance_count instance_id room_speed fps "+"fps_real current_time current_year current_month current_day "+"current_weekday current_hour current_minute current_second alarm "+"timeline_index timeline_position timeline_speed timeline_running "+"timeline_loop room room_first room_last room_width room_height "+"room_caption room_persistent score lives health show_score "+"show_lives show_health caption_score caption_lives caption_health "+"event_type event_number event_object event_action "+"application_surface gamemaker_pro gamemaker_registered "+"gamemaker_version error_occurred error_last debug_mode "+"keyboard_key keyboard_lastkey keyboard_lastchar keyboard_string "+"mouse_x mouse_y mouse_button mouse_lastbutton cursor_sprite "+"visible sprite_index sprite_width sprite_height sprite_xoffset "+"sprite_yoffset image_number image_index image_speed depth "+"image_xscale image_yscale image_angle image_alpha image_blend "+"bbox_left bbox_right bbox_top bbox_bottom layer background_colour  "+"background_showcolour background_color background_showcolor "+"view_enabled view_current view_visible view_xview view_yview "+"view_wview view_hview view_xport view_yport view_wport view_hport "+"view_angle view_hborder view_vborder view_hspeed view_vspeed "+"view_object view_surface_id view_camera game_id game_display_name "+"game_project_name game_save_id working_directory temp_directory "+"program_directory browser_width browser_height os_type os_device "+"os_browser os_version display_aa async_load delta_time "+"webgl_enabled event_data iap_data phy_rotation phy_position_x "+"phy_position_y phy_angular_velocity phy_linear_velocity_x "+"phy_linear_velocity_y phy_speed_x phy_speed_y phy_speed "+"phy_angular_damping phy_linear_damping phy_bullet "+"phy_fixed_rotation phy_active phy_mass phy_inertia phy_com_x "+"phy_com_y phy_dynamic phy_kinematic phy_sleeping "+"phy_collision_points phy_collision_x phy_collision_y "+"phy_col_normal_x phy_col_normal_y phy_position_xprevious "+"phy_position_yprevious"};return{aliases:["gml","GML"],case_insensitive:false,keywords:GML_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("smali",function(hljs){var smali_instr_low_prio=["add","and","cmp","cmpg","cmpl","const","div","double","float","goto","if","int","long","move","mul","neg","new","nop","not","or","rem","return","shl","shr","sput","sub","throw","ushr","xor"];var smali_instr_high_prio=["aget","aput","array","check","execute","fill","filled","goto/16","goto/32","iget","instance","invoke","iput","monitor","packed","sget","sparse"];var smali_keywords=["transient","constructor","abstract","final","synthetic","public","private","protected","static","bridge","system"];return{aliases:["smali"],contains:[{className:"string",begin:'"',end:'"',relevance:0},hljs.COMMENT("#","$",{relevance:0}),{className:"keyword",variants:[{begin:"\\s*\\.end\\s[a-zA-Z0-9]*"},{begin:"^[ ]*\\.[a-zA-Z]*",relevance:0},{begin:"\\s:[a-zA-Z_0-9]*",relevance:0},{begin:"\\s("+smali_keywords.join("|")+")"}]},{className:"built_in",variants:[{begin:"\\s("+smali_instr_low_prio.join("|")+")\\s"},{begin:"\\s("+smali_instr_low_prio.join("|")+")((\\-|/)[a-zA-Z0-9]+)+\\s",relevance:10},{begin:"\\s("+smali_instr_high_prio.join("|")+")((\\-|/)[a-zA-Z0-9]+)*\\s",relevance:10}]},{className:"class",begin:"L[^(;:\n]*;",relevance:0},{begin:"[vp][0-9]+"}]}});hljs.registerLanguage("lsl",function(hljs){var LSL_STRING_ESCAPE_CHARS={className:"subst",begin:/\\[tn"\\]/};var LSL_STRINGS={className:"string",begin:'"',end:'"',contains:[LSL_STRING_ESCAPE_CHARS]};var LSL_NUMBERS={className:"number",begin:hljs.C_NUMBER_RE};var LSL_CONSTANTS={className:"literal",variants:[{begin:"\\b(?:PI|TWO_PI|PI_BY_TWO|DEG_TO_RAD|RAD_TO_DEG|SQRT2)\\b"},{begin:"\\b(?:XP_ERROR_(?:EXPERIENCES_DISABLED|EXPERIENCE_(?:DISABLED|SUSPENDED)|INVALID_(?:EXPERIENCE|PARAMETERS)|KEY_NOT_FOUND|MATURITY_EXCEEDED|NONE|NOT_(?:FOUND|PERMITTED(?:_LAND)?)|NO_EXPERIENCE|QUOTA_EXCEEDED|RETRY_UPDATE|STORAGE_EXCEPTION|STORE_DISABLED|THROTTLED|UNKNOWN_ERROR)|JSON_APPEND|STATUS_(?:PHYSICS|ROTATE_[XYZ]|PHANTOM|SANDBOX|BLOCK_GRAB(?:_OBJECT)?|(?:DIE|RETURN)_AT_EDGE|CAST_SHADOWS|OK|MALFORMED_PARAMS|TYPE_MISMATCH|BOUNDS_ERROR|NOT_(?:FOUND|SUPPORTED)|INTERNAL_ERROR|WHITELIST_FAILED)|AGENT(?:_(?:BY_(?:LEGACY_|USER)NAME|FLYING|ATTACHMENTS|SCRIPTED|MOUSELOOK|SITTING|ON_OBJECT|AWAY|WALKING|IN_AIR|TYPING|CROUCHING|BUSY|ALWAYS_RUN|AUTOPILOT|LIST_(?:PARCEL(?:_OWNER)?|REGION)))?|CAMERA_(?:PITCH|DISTANCE|BEHINDNESS_(?:ANGLE|LAG)|(?:FOCUS|POSITION)(?:_(?:THRESHOLD|LOCKED|LAG))?|FOCUS_OFFSET|ACTIVE)|ANIM_ON|LOOP|REVERSE|PING_PONG|SMOOTH|ROTATE|SCALE|ALL_SIDES|LINK_(?:ROOT|SET|ALL_(?:OTHERS|CHILDREN)|THIS)|ACTIVE|PASS(?:IVE|_(?:ALWAYS|IF_NOT_HANDLED|NEVER))|SCRIPTED|CONTROL_(?:FWD|BACK|(?:ROT_)?(?:LEFT|RIGHT)|UP|DOWN|(?:ML_)?LBUTTON)|PERMISSION_(?:RETURN_OBJECTS|DEBIT|OVERRIDE_ANIMATIONS|SILENT_ESTATE_MANAGEMENT|TAKE_CONTROLS|TRIGGER_ANIMATION|ATTACH|CHANGE_LINKS|(?:CONTROL|TRACK)_CAMERA|TELEPORT)|INVENTORY_(?:TEXTURE|SOUND|OBJECT|SCRIPT|LANDMARK|CLOTHING|NOTECARD|BODYPART|ANIMATION|GESTURE|ALL|NONE)|CHANGED_(?:INVENTORY|COLOR|SHAPE|SCALE|TEXTURE|LINK|ALLOWED_DROP|OWNER|REGION(?:_START)?|TELEPORT|MEDIA)|OBJECT_(?:CLICK_ACTION|HOVER_HEIGHT|LAST_OWNER_ID|(?:PHYSICS|SERVER|STREAMING)_COST|UNKNOWN_DETAIL|CHARACTER_TIME|PHANTOM|PHYSICS|TEMP_(?:ATTACHED|ON_REZ)|NAME|DESC|POS|PRIM_(?:COUNT|EQUIVALENCE)|RETURN_(?:PARCEL(?:_OWNER)?|REGION)|REZZER_KEY|ROO?T|VELOCITY|OMEGA|OWNER|GROUP(?:_TAG)?|CREATOR|ATTACHED_(?:POINT|SLOTS_AVAILABLE)|RENDER_WEIGHT|(?:BODY_SHAPE|PATHFINDING)_TYPE|(?:RUNNING|TOTAL)_SCRIPT_COUNT|TOTAL_INVENTORY_COUNT|SCRIPT_(?:MEMORY|TIME))|TYPE_(?:INTEGER|FLOAT|STRING|KEY|VECTOR|ROTATION|INVALID)|(?:DEBUG|PUBLIC)_CHANNEL|ATTACH_(?:AVATAR_CENTER|CHEST|HEAD|BACK|PELVIS|MOUTH|CHIN|NECK|NOSE|BELLY|[LR](?:SHOULDER|HAND|FOOT|EAR|EYE|[UL](?:ARM|LEG)|HIP)|(?:LEFT|RIGHT)_PEC|HUD_(?:CENTER_[12]|TOP_(?:RIGHT|CENTER|LEFT)|BOTTOM(?:_(?:RIGHT|LEFT))?)|[LR]HAND_RING1|TAIL_(?:BASE|TIP)|[LR]WING|FACE_(?:JAW|[LR]EAR|[LR]EYE|TOUNGE)|GROIN|HIND_[LR]FOOT)|LAND_(?:LEVEL|RAISE|LOWER|SMOOTH|NOISE|REVERT)|DATA_(?:ONLINE|NAME|BORN|SIM_(?:POS|STATUS|RATING)|PAYINFO)|PAYMENT_INFO_(?:ON_FILE|USED)|REMOTE_DATA_(?:CHANNEL|REQUEST|REPLY)|PSYS_(?:PART_(?:BF_(?:ZERO|ONE(?:_MINUS_(?:DEST_COLOR|SOURCE_(ALPHA|COLOR)))?|DEST_COLOR|SOURCE_(ALPHA|COLOR))|BLEND_FUNC_(DEST|SOURCE)|FLAGS|(?:START|END)_(?:COLOR|ALPHA|SCALE|GLOW)|MAX_AGE|(?:RIBBON|WIND|INTERP_(?:COLOR|SCALE)|BOUNCE|FOLLOW_(?:SRC|VELOCITY)|TARGET_(?:POS|LINEAR)|EMISSIVE)_MASK)|SRC_(?:MAX_AGE|PATTERN|ANGLE_(?:BEGIN|END)|BURST_(?:RATE|PART_COUNT|RADIUS|SPEED_(?:MIN|MAX))|ACCEL|TEXTURE|TARGET_KEY|OMEGA|PATTERN_(?:DROP|EXPLODE|ANGLE(?:_CONE(?:_EMPTY)?)?)))|VEHICLE_(?:REFERENCE_FRAME|TYPE_(?:NONE|SLED|CAR|BOAT|AIRPLANE|BALLOON)|(?:LINEAR|ANGULAR)_(?:FRICTION_TIMESCALE|MOTOR_DIRECTION)|LINEAR_MOTOR_OFFSET|HOVER_(?:HEIGHT|EFFICIENCY|TIMESCALE)|BUOYANCY|(?:LINEAR|ANGULAR)_(?:DEFLECTION_(?:EFFICIENCY|TIMESCALE)|MOTOR_(?:DECAY_)?TIMESCALE)|VERTICAL_ATTRACTION_(?:EFFICIENCY|TIMESCALE)|BANKING_(?:EFFICIENCY|MIX|TIMESCALE)|FLAG_(?:NO_DEFLECTION_UP|LIMIT_(?:ROLL_ONLY|MOTOR_UP)|HOVER_(?:(?:WATER|TERRAIN|UP)_ONLY|GLOBAL_HEIGHT)|MOUSELOOK_(?:STEER|BANK)|CAMERA_DECOUPLED))|PRIM_(?:ALLOW_UNSIT|ALPHA_MODE(?:_(?:BLEND|EMISSIVE|MASK|NONE))?|NORMAL|SPECULAR|TYPE(?:_(?:BOX|CYLINDER|PRISM|SPHERE|TORUS|TUBE|RING|SCULPT))?|HOLE_(?:DEFAULT|CIRCLE|SQUARE|TRIANGLE)|MATERIAL(?:_(?:STONE|METAL|GLASS|WOOD|FLESH|PLASTIC|RUBBER))?|SHINY_(?:NONE|LOW|MEDIUM|HIGH)|BUMP_(?:NONE|BRIGHT|DARK|WOOD|BARK|BRICKS|CHECKER|CONCRETE|TILE|STONE|DISKS|GRAVEL|BLOBS|SIDING|LARGETILE|STUCCO|SUCTION|WEAVE)|TEXGEN_(?:DEFAULT|PLANAR)|SCRIPTED_SIT_ONLY|SCULPT_(?:TYPE_(?:SPHERE|TORUS|PLANE|CYLINDER|MASK)|FLAG_(?:MIRROR|INVERT))|PHYSICS(?:_(?:SHAPE_(?:CONVEX|NONE|PRIM|TYPE)))?|(?:POS|ROT)_LOCAL|SLICE|TEXT|FLEXIBLE|POINT_LIGHT|TEMP_ON_REZ|PHANTOM|POSITION|SIT_TARGET|SIZE|ROTATION|TEXTURE|NAME|OMEGA|DESC|LINK_TARGET|COLOR|BUMP_SHINY|FULLBRIGHT|TEXGEN|GLOW|MEDIA_(?:ALT_IMAGE_ENABLE|CONTROLS|(?:CURRENT|HOME)_URL|AUTO_(?:LOOP|PLAY|SCALE|ZOOM)|FIRST_CLICK_INTERACT|(?:WIDTH|HEIGHT)_PIXELS|WHITELIST(?:_ENABLE)?|PERMS_(?:INTERACT|CONTROL)|PARAM_MAX|CONTROLS_(?:STANDARD|MINI)|PERM_(?:NONE|OWNER|GROUP|ANYONE)|MAX_(?:URL_LENGTH|WHITELIST_(?:SIZE|COUNT)|(?:WIDTH|HEIGHT)_PIXELS)))|MASK_(?:BASE|OWNER|GROUP|EVERYONE|NEXT)|PERM_(?:TRANSFER|MODIFY|COPY|MOVE|ALL)|PARCEL_(?:MEDIA_COMMAND_(?:STOP|PAUSE|PLAY|LOOP|TEXTURE|URL|TIME|AGENT|UNLOAD|AUTO_ALIGN|TYPE|SIZE|DESC|LOOP_SET)|FLAG_(?:ALLOW_(?:FLY|(?:GROUP_)?SCRIPTS|LANDMARK|TERRAFORM|DAMAGE|CREATE_(?:GROUP_)?OBJECTS)|USE_(?:ACCESS_(?:GROUP|LIST)|BAN_LIST|LAND_PASS_LIST)|LOCAL_SOUND_ONLY|RESTRICT_PUSHOBJECT|ALLOW_(?:GROUP|ALL)_OBJECT_ENTRY)|COUNT_(?:TOTAL|OWNER|GROUP|OTHER|SELECTED|TEMP)|DETAILS_(?:NAME|DESC|OWNER|GROUP|AREA|ID|SEE_AVATARS))|LIST_STAT_(?:MAX|MIN|MEAN|MEDIAN|STD_DEV|SUM(?:_SQUARES)?|NUM_COUNT|GEOMETRIC_MEAN|RANGE)|PAY_(?:HIDE|DEFAULT)|REGION_FLAG_(?:ALLOW_DAMAGE|FIXED_SUN|BLOCK_TERRAFORM|SANDBOX|DISABLE_(?:COLLISIONS|PHYSICS)|BLOCK_FLY|ALLOW_DIRECT_TELEPORT|RESTRICT_PUSHOBJECT)|HTTP_(?:METHOD|MIMETYPE|BODY_(?:MAXLENGTH|TRUNCATED)|CUSTOM_HEADER|PRAGMA_NO_CACHE|VERBOSE_THROTTLE|VERIFY_CERT)|SIT_(?:INVALID_(?:AGENT|LINK_OBJECT)|NO(?:T_EXPERIENCE|_(?:ACCESS|EXPERIENCE_PERMISSION|SIT_TARGET)))|STRING_(?:TRIM(?:_(?:HEAD|TAIL))?)|CLICK_ACTION_(?:NONE|TOUCH|SIT|BUY|PAY|OPEN(?:_MEDIA)?|PLAY|ZOOM)|TOUCH_INVALID_FACE|PROFILE_(?:NONE|SCRIPT_MEMORY)|RC_(?:DATA_FLAGS|DETECT_PHANTOM|GET_(?:LINK_NUM|NORMAL|ROOT_KEY)|MAX_HITS|REJECT_(?:TYPES|AGENTS|(?:NON)?PHYSICAL|LAND))|RCERR_(?:CAST_TIME_EXCEEDED|SIM_PERF_LOW|UNKNOWN)|ESTATE_ACCESS_(?:ALLOWED_(?:AGENT|GROUP)_(?:ADD|REMOVE)|BANNED_AGENT_(?:ADD|REMOVE))|DENSITY|FRICTION|RESTITUTION|GRAVITY_MULTIPLIER|KFM_(?:COMMAND|CMD_(?:PLAY|STOP|PAUSE)|MODE|FORWARD|LOOP|PING_PONG|REVERSE|DATA|ROTATION|TRANSLATION)|ERR_(?:GENERIC|PARCEL_PERMISSIONS|MALFORMED_PARAMS|RUNTIME_PERMISSIONS|THROTTLED)|CHARACTER_(?:CMD_(?:(?:SMOOTH_)?STOP|JUMP)|DESIRED_(?:TURN_)?SPEED|RADIUS|STAY_WITHIN_PARCEL|LENGTH|ORIENTATION|ACCOUNT_FOR_SKIPPED_FRAMES|AVOIDANCE_MODE|TYPE(?:_(?:[ABCD]|NONE))?|MAX_(?:DECEL|TURN_RADIUS|(?:ACCEL|SPEED)))|PURSUIT_(?:OFFSET|FUZZ_FACTOR|GOAL_TOLERANCE|INTERCEPT)|REQUIRE_LINE_OF_SIGHT|FORCE_DIRECT_PATH|VERTICAL|HORIZONTAL|AVOID_(?:CHARACTERS|DYNAMIC_OBSTACLES|NONE)|PU_(?:EVADE_(?:HIDDEN|SPOTTED)|FAILURE_(?:DYNAMIC_PATHFINDING_DISABLED|INVALID_(?:GOAL|START)|NO_(?:NAVMESH|VALID_DESTINATION)|OTHER|TARGET_GONE|(?:PARCEL_)?UNREACHABLE)|(?:GOAL|SLOWDOWN_DISTANCE)_REACHED)|TRAVERSAL_TYPE(?:_(?:FAST|NONE|SLOW))?|CONTENT_TYPE_(?:ATOM|FORM|HTML|JSON|LLSD|RSS|TEXT|XHTML|XML)|GCNP_(?:RADIUS|STATIC)|(?:PATROL|WANDER)_PAUSE_AT_WAYPOINTS|OPT_(?:AVATAR|CHARACTER|EXCLUSION_VOLUME|LEGACY_LINKSET|MATERIAL_VOLUME|OTHER|STATIC_OBSTACLE|WALKABLE)|SIM_STAT_PCT_CHARS_STEPPED)\\b"},{begin:"\\b(?:FALSE|TRUE)\\b"},{begin:"\\b(?:ZERO_ROTATION)\\b"},{begin:"\\b(?:EOF|JSON_(?:ARRAY|DELETE|FALSE|INVALID|NULL|NUMBER|OBJECT|STRING|TRUE)|NULL_KEY|TEXTURE_(?:BLANK|DEFAULT|MEDIA|PLYWOOD|TRANSPARENT)|URL_REQUEST_(?:GRANTED|DENIED))\\b"},{begin:"\\b(?:ZERO_VECTOR|TOUCH_INVALID_(?:TEXCOORD|VECTOR))\\b"}]};var LSL_FUNCTIONS={className:"built_in",begin:"\\b(?:ll(?:AgentInExperience|(?:Create|DataSize|Delete|KeyCount|Keys|Read|Update)KeyValue|GetExperience(?:Details|ErrorMessage)|ReturnObjectsBy(?:ID|Owner)|Json(?:2List|[GS]etValue|ValueType)|Sin|Cos|Tan|Atan2|Sqrt|Pow|Abs|Fabs|Frand|Floor|Ceil|Round|Vec(?:Mag|Norm|Dist)|Rot(?:Between|2(?:Euler|Fwd|Left|Up))|(?:Euler|Axes)2Rot|Whisper|(?:Region|Owner)?Say|Shout|Listen(?:Control|Remove)?|Sensor(?:Repeat|Remove)?|Detected(?:Name|Key|Owner|Type|Pos|Vel|Grab|Rot|Group|LinkNumber)|Die|Ground|Wind|(?:[GS]et)(?:AnimationOverride|MemoryLimit|PrimMediaParams|ParcelMusicURL|Object(?:Desc|Name)|PhysicsMaterial|Status|Scale|Color|Alpha|Texture|Pos|Rot|Force|Torque)|ResetAnimationOverride|(?:Scale|Offset|Rotate)Texture|(?:Rot)?Target(?:Remove)?|(?:Stop)?MoveToTarget|Apply(?:Rotational)?Impulse|Set(?:KeyframedMotion|ContentType|RegionPos|(?:Angular)?Velocity|Buoyancy|HoverHeight|ForceAndTorque|TimerEvent|ScriptState|Damage|TextureAnim|Sound(?:Queueing|Radius)|Vehicle(?:Type|(?:Float|Vector|Rotation)Param)|(?:Touch|Sit)?Text|Camera(?:Eye|At)Offset|PrimitiveParams|ClickAction|Link(?:Alpha|Color|PrimitiveParams(?:Fast)?|Texture(?:Anim)?|Camera|Media)|RemoteScriptAccessPin|PayPrice|LocalRot)|ScaleByFactor|Get(?:(?:Max|Min)ScaleFactor|ClosestNavPoint|StaticPath|SimStats|Env|PrimitiveParams|Link(?:PrimitiveParams|Number(?:OfSides)?|Key|Name|Media)|HTTPHeader|FreeURLs|Object(?:Details|PermMask|PrimCount)|Parcel(?:MaxPrims|Details|Prim(?:Count|Owners))|Attached(?:List)?|(?:SPMax|Free|Used)Memory|Region(?:Name|TimeDilation|FPS|Corner|AgentCount)|Root(?:Position|Rotation)|UnixTime|(?:Parcel|Region)Flags|(?:Wall|GMT)clock|SimulatorHostname|BoundingBox|GeometricCenter|Creator|NumberOf(?:Prims|NotecardLines|Sides)|Animation(?:List)?|(?:Camera|Local)(?:Pos|Rot)|Vel|Accel|Omega|Time(?:stamp|OfDay)|(?:Object|CenterOf)?Mass|MassMKS|Energy|Owner|(?:Owner)?Key|SunDirection|Texture(?:Offset|Scale|Rot)|Inventory(?:Number|Name|Key|Type|Creator|PermMask)|Permissions(?:Key)?|StartParameter|List(?:Length|EntryType)|Date|Agent(?:Size|Info|Language|List)|LandOwnerAt|NotecardLine|Script(?:Name|State))|(?:Get|Reset|GetAndReset)Time|PlaySound(?:Slave)?|LoopSound(?:Master|Slave)?|(?:Trigger|Stop|Preload)Sound|(?:(?:Get|Delete)Sub|Insert)String|To(?:Upper|Lower)|Give(?:InventoryList|Money)|RezObject|(?:Stop)?LookAt|Sleep|CollisionFilter|(?:Take|Release)Controls|DetachFromAvatar|AttachToAvatar(?:Temp)?|InstantMessage|(?:GetNext)?Email|StopHover|MinEventDelay|RotLookAt|String(?:Length|Trim)|(?:Start|Stop)Animation|TargetOmega|Request(?:Experience)?Permissions|(?:Create|Break)Link|BreakAllLinks|(?:Give|Remove)Inventory|Water|PassTouches|Request(?:Agent|Inventory)Data|TeleportAgent(?:Home|GlobalCoords)?|ModifyLand|CollisionSound|ResetScript|MessageLinked|PushObject|PassCollisions|AxisAngle2Rot|Rot2(?:Axis|Angle)|A(?:cos|sin)|AngleBetween|AllowInventoryDrop|SubStringIndex|List2(?:CSV|Integer|Json|Float|String|Key|Vector|Rot|List(?:Strided)?)|DeleteSubList|List(?:Statistics|Sort|Randomize|(?:Insert|Find|Replace)List)|EdgeOfWorld|AdjustSoundVolume|Key2Name|TriggerSoundLimited|EjectFromLand|(?:CSV|ParseString)2List|OverMyLand|SameGroup|UnSit|Ground(?:Slope|Normal|Contour)|GroundRepel|(?:Set|Remove)VehicleFlags|SitOnLink|(?:AvatarOn)?(?:Link)?SitTarget|Script(?:Danger|Profiler)|Dialog|VolumeDetect|ResetOtherScript|RemoteLoadScriptPin|(?:Open|Close)RemoteDataChannel|SendRemoteData|RemoteDataReply|(?:Integer|String)ToBase64|XorBase64|Log(?:10)?|Base64To(?:String|Integer)|ParseStringKeepNulls|RezAtRoot|RequestSimulatorData|ForceMouselook|(?:Load|Release|(?:E|Une)scape)URL|ParcelMedia(?:CommandList|Query)|ModPow|MapDestination|(?:RemoveFrom|AddTo|Reset)Land(?:Pass|Ban)List|(?:Set|Clear)CameraParams|HTTP(?:Request|Response)|TextBox|DetectedTouch(?:UV|Face|Pos|(?:N|Bin)ormal|ST)|(?:MD5|SHA1|DumpList2)String|Request(?:Secure)?URL|Clear(?:Prim|Link)Media|(?:Link)?ParticleSystem|(?:Get|Request)(?:Username|DisplayName)|RegionSayTo|CastRay|GenerateKey|TransferLindenDollars|ManageEstateAccess|(?:Create|Delete)Character|ExecCharacterCmd|Evade|FleeFrom|NavigateTo|PatrolPoints|Pursue|UpdateCharacter|WanderWithin))\\b"};return{illegal:":",contains:[LSL_STRINGS,{className:"comment",variants:[hljs.COMMENT("//","$"),hljs.COMMENT("/\\*","\\*/")],relevance:0},LSL_NUMBERS,{className:"section",variants:[{begin:"\\b(?:state|default)\\b"},{begin:"\\b(?:state_(?:entry|exit)|touch(?:_(?:start|end))?|(?:land_)?collision(?:_(?:start|end))?|timer|listen|(?:no_)?sensor|control|(?:not_)?at_(?:rot_)?target|money|email|experience_permissions(?:_denied)?|run_time_permissions|changed|attach|dataserver|moving_(?:start|end)|link_message|(?:on|object)_rez|remote_data|http_re(?:sponse|quest)|path_update|transaction_result)\\b"}]},LSL_FUNCTIONS,LSL_CONSTANTS,{className:"type",begin:"\\b(?:integer|float|string|key|vector|quaternion|rotation|list)\\b"}]}});hljs.registerLanguage("markdown",function(hljs){return{aliases:["md","mkdown","mkd"],contains:[{className:"section",variants:[{begin:"^#{1,6}",end:"$"},{begin:"^.+?\\n[=-]{2,}$"}]},{begin:"<",end:">",subLanguage:"xml",relevance:0},{className:"bullet",begin:"^\\s*([*+-]|(\\d+\\.))\\s+"},{className:"strong",begin:"[*_]{2}.+?[*_]{2}"},{className:"emphasis",variants:[{begin:"\\*.+?\\*"},{begin:"_.+?_",relevance:0}]},{className:"quote",begin:"^>\\s+",end:"$"},{className:"code",variants:[{begin:"^```\\w*\\s*$",end:"^```[ ]*$"},{begin:"`.+?`"},{begin:"^( {4}|\\t)",end:"$",relevance:0}]},{begin:"^[-\\*]{3,}",end:"$"},{begin:"\\[.+?\\][\\(\\[].*?[\\)\\]]",returnBegin:true,contains:[{className:"string",begin:"\\[",end:"\\]",excludeBegin:true,returnEnd:true,relevance:0},{className:"link",begin:"\\]\\(",end:"\\)",excludeBegin:true,excludeEnd:true},{className:"symbol",begin:"\\]\\[",end:"\\]",excludeBegin:true,excludeEnd:true}],relevance:10},{begin:/^\[[^\n]+\]:/,returnBegin:true,contains:[{className:"symbol",begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true},{className:"link",begin:/:\s*/,end:/$/,excludeBegin:true}]}]}});hljs.registerLanguage("dart",function(hljs){var SUBST={className:"subst",variants:[{begin:"\\$[A-Za-z0-9_]+"}]};var BRACED_SUBST={className:"subst",variants:[{begin:"\\${",end:"}"}],keywords:"true false null this is new super"};var STRING={className:"string",variants:[{begin:"r'''",end:"'''"},{begin:'r"""',end:'"""'},{begin:"r'",end:"'",illegal:"\\n"},{begin:'r"',end:'"',illegal:"\\n"},{begin:"'''",end:"'''",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:'"""',end:'"""',contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]},{begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST,BRACED_SUBST]}]};BRACED_SUBST.contains=[hljs.C_NUMBER_MODE,STRING];var KEYWORDS={keyword:"abstract as assert async await break case catch class const continue covariant default deferred do "+"dynamic else enum export extends extension external factory false final finally for Function get hide if "+"implements import in inferface is library mixin new null on operator part rethrow return set show static "+"super switch sync this throw true try typedef var void while with yield",built_in:"Comparable DateTime Duration Function Iterable Iterator List Map Match Null Object Pattern RegExp Set "+"Stopwatch String StringBuffer StringSink Symbol Type Uri bool double dynamic int num print "+"Element ElementList document querySelector querySelectorAll window"};return{keywords:KEYWORDS,contains:[STRING,hljs.COMMENT("/\\*\\*","\\*/",{subLanguage:"markdown"}),hljs.COMMENT("///+\\s*","$",{contains:[{subLanguage:"markdown",begin:".",end:"$"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"},{begin:"=>"}]}});hljs.registerLanguage("xml",function(hljs){var XML_IDENT_RE="[A-Za-z0-9\\._:-]+";var XML_ENTITIES={className:"symbol",begin:"&[a-z]+;|&#[0-9]+;|&#x[a-f0-9]+;"};var XML_META_KEYWORDS={begin:"\\s",contains:[{className:"meta-keyword",begin:"#?[a-z_][a-z1-9_-]+",illegal:"\\n"}]};var XML_META_PAR_KEYWORDS=hljs.inherit(XML_META_KEYWORDS,{begin:"\\(",end:"\\)"});var APOS_META_STRING_MODE=hljs.inherit(hljs.APOS_STRING_MODE,{className:"meta-string"});var QUOTE_META_STRING_MODE=hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"});var TAG_INTERNALS={endsWithParent:true,illegal:/</,relevance:0,contains:[{className:"attr",begin:XML_IDENT_RE,relevance:0},{begin:/=\s*/,relevance:0,contains:[{className:"string",endsParent:true,variants:[{begin:/"/,end:/"/,contains:[XML_ENTITIES]},{begin:/'/,end:/'/,contains:[XML_ENTITIES]},{begin:/[^\s"'=<>`]+/}]}]}]};return{aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist","wsf","svg"],case_insensitive:true,contains:[{className:"meta",begin:"<![a-z]",end:">",relevance:10,contains:[XML_META_KEYWORDS,QUOTE_META_STRING_MODE,APOS_META_STRING_MODE,XML_META_PAR_KEYWORDS,{begin:"\\[",end:"\\]",contains:[{className:"meta",begin:"<![a-z]",end:">",contains:[XML_META_KEYWORDS,XML_META_PAR_KEYWORDS,QUOTE_META_STRING_MODE,APOS_META_STRING_MODE]}]}]},hljs.COMMENT("\x3c!--","--\x3e",{relevance:10}),{begin:"<\\!\\[CDATA\\[",end:"\\]\\]>",relevance:10},XML_ENTITIES,{className:"meta",begin:/<\?xml/,end:/\?>/,relevance:10},{begin:/<\?(php)?/,end:/\?>/,subLanguage:"php",contains:[{begin:"/\\*",end:"\\*/",skip:true},{begin:'b"',end:'"',skip:true},{begin:"b'",end:"'",skip:true},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null,className:null,contains:null,skip:true}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null,className:null,contains:null,skip:true})]},{className:"tag",begin:"<style(?=\\s|>)",end:">",keywords:{name:"style"},contains:[TAG_INTERNALS],starts:{end:"</style>",returnEnd:true,subLanguage:["css","xml"]}},{className:"tag",begin:"<script(?=\\s|>)",end:">",keywords:{name:"script"},contains:[TAG_INTERNALS],starts:{end:"<\/script>",returnEnd:true,subLanguage:["actionscript","javascript","handlebars","xml"]}},{className:"tag",begin:"</?",end:"/?>",contains:[{className:"name",begin:/[^\/><\s]+/,relevance:0},TAG_INTERNALS]}]}});hljs.registerLanguage("parser3",function(hljs){var CURLY_SUBCOMMENT=hljs.COMMENT("{","}",{contains:["self"]});return{subLanguage:"xml",relevance:0,contains:[hljs.COMMENT("^#","$"),hljs.COMMENT("\\^rem{","}",{relevance:10,contains:[CURLY_SUBCOMMENT]}),{className:"meta",begin:"^@(?:BASE|USE|CLASS|OPTIONS)$",relevance:10},{className:"title",begin:"@[\\w\\-]+\\[[\\w^;\\-]*\\](?:\\[[\\w^;\\-]*\\])?(?:.*)$"},{className:"variable",begin:"\\$\\{?[\\w\\-\\.\\:]+\\}?"},{className:"keyword",begin:"\\^[\\w\\-\\.\\:]+"},{className:"number",begin:"\\^#[0-9a-fA-F]+"},hljs.C_NUMBER_MODE]}});hljs.registerLanguage("sql",function(hljs){var COMMENT_MODE=hljs.COMMENT("--","$");return{case_insensitive:true,illegal:/[<>{}*]/,contains:[{beginKeywords:"begin end start commit rollback savepoint lock alter create drop rename call "+"delete do handler insert load replace select truncate update set show pragma grant "+"merge describe use explain help declare prepare execute deallocate release "+"unlock purge reset change stop analyze cache flush optimize repair kill "+"install uninstall checksum restore check backup revoke comment values with",end:/;/,endsWithParent:true,lexemes:/[\w\.]+/,keywords:{keyword:"as abort abs absolute acc acce accep accept access accessed accessible account acos action activate add "+"addtime admin administer advanced advise aes_decrypt aes_encrypt after agent aggregate ali alia alias "+"all allocate allow alter always analyze ancillary and anti any anydata anydataset anyschema anytype apply "+"archive archived archivelog are as asc ascii asin assembly assertion associate asynchronous at atan "+"atn2 attr attri attrib attribu attribut attribute attributes audit authenticated authentication authid "+"authors auto autoallocate autodblink autoextend automatic availability avg backup badfile basicfile "+"before begin beginning benchmark between bfile bfile_base big bigfile bin binary_double binary_float "+"binlog bit_and bit_count bit_length bit_or bit_xor bitmap blob_base block blocksize body both bound "+"bucket buffer_cache buffer_pool build bulk by byte byteordermark bytes cache caching call calling cancel "+"capacity cascade cascaded case cast catalog category ceil ceiling chain change changed char_base "+"char_length character_length characters characterset charindex charset charsetform charsetid check "+"checksum checksum_agg child choose chr chunk class cleanup clear client clob clob_base clone close "+"cluster_id cluster_probability cluster_set clustering coalesce coercibility col collate collation "+"collect colu colum column column_value columns columns_updated comment commit compact compatibility "+"compiled complete composite_limit compound compress compute concat concat_ws concurrent confirm conn "+"connec connect connect_by_iscycle connect_by_isleaf connect_by_root connect_time connection "+"consider consistent constant constraint constraints constructor container content contents context "+"contributors controlfile conv convert convert_tz corr corr_k corr_s corresponding corruption cos cost "+"count count_big counted covar_pop covar_samp cpu_per_call cpu_per_session crc32 create creation "+"critical cross cube cume_dist curdate current current_date current_time current_timestamp current_user "+"cursor curtime customdatum cycle data database databases datafile datafiles datalength date_add "+"date_cache date_format date_sub dateadd datediff datefromparts datename datepart datetime2fromparts "+"day day_to_second dayname dayofmonth dayofweek dayofyear days db_role_change dbtimezone ddl deallocate "+"declare decode decompose decrement decrypt deduplicate def defa defau defaul default defaults "+"deferred defi defin define degrees delayed delegate delete delete_all delimited demand dense_rank "+"depth dequeue des_decrypt des_encrypt des_key_file desc descr descri describ describe descriptor "+"deterministic diagnostics difference dimension direct_load directory disable disable_all "+"disallow disassociate discardfile disconnect diskgroup distinct distinctrow distribute distributed div "+"do document domain dotnet double downgrade drop dumpfile duplicate duration each edition editionable "+"editions element ellipsis else elsif elt empty enable enable_all enclosed encode encoding encrypt "+"end end-exec endian enforced engine engines enqueue enterprise entityescaping eomonth error errors "+"escaped evalname evaluate event eventdata events except exception exceptions exchange exclude excluding "+"execu execut execute exempt exists exit exp expire explain explode export export_set extended extent external "+"external_1 external_2 externally extract failed failed_login_attempts failover failure far fast "+"feature_set feature_value fetch field fields file file_name_convert filesystem_like_logging final "+"finish first first_value fixed flash_cache flashback floor flush following follows for forall force foreign "+"form forma format found found_rows freelist freelists freepools fresh from from_base64 from_days "+"ftp full function general generated get get_format get_lock getdate getutcdate global global_name "+"globally go goto grant grants greatest group group_concat group_id grouping grouping_id groups "+"gtid_subtract guarantee guard handler hash hashkeys having hea head headi headin heading heap help hex "+"hierarchy high high_priority hosts hour hours http id ident_current ident_incr ident_seed identified "+"identity idle_time if ifnull ignore iif ilike ilm immediate import in include including increment "+"index indexes indexing indextype indicator indices inet6_aton inet6_ntoa inet_aton inet_ntoa infile "+"initial initialized initially initrans inmemory inner innodb input insert install instance instantiable "+"instr interface interleaved intersect into invalidate invisible is is_free_lock is_ipv4 is_ipv4_compat "+"is_not is_not_null is_used_lock isdate isnull isolation iterate java join json json_exists "+"keep keep_duplicates key keys kill language large last last_day last_insert_id last_value lateral lax lcase "+"lead leading least leaves left len lenght length less level levels library like like2 like4 likec limit "+"lines link list listagg little ln load load_file lob lobs local localtime localtimestamp locate "+"locator lock locked log log10 log2 logfile logfiles logging logical logical_reads_per_call "+"logoff logon logs long loop low low_priority lower lpad lrtrim ltrim main make_set makedate maketime "+"managed management manual map mapping mask master master_pos_wait match matched materialized max "+"maxextents maximize maxinstances maxlen maxlogfiles maxloghistory maxlogmembers maxsize maxtrans "+"md5 measures median medium member memcompress memory merge microsecond mid migration min minextents "+"minimum mining minus minute minutes minvalue missing mod mode model modification modify module monitoring month "+"months mount move movement multiset mutex name name_const names nan national native natural nav nchar "+"nclob nested never new newline next nextval no no_write_to_binlog noarchivelog noaudit nobadfile "+"nocheck nocompress nocopy nocycle nodelay nodiscardfile noentityescaping noguarantee nokeep nologfile "+"nomapping nomaxvalue nominimize nominvalue nomonitoring none noneditionable nonschema noorder "+"nopr nopro noprom nopromp noprompt norely noresetlogs noreverse normal norowdependencies noschemacheck "+"noswitch not nothing notice notnull notrim novalidate now nowait nth_value nullif nulls num numb numbe "+"nvarchar nvarchar2 object ocicoll ocidate ocidatetime ociduration ociinterval ociloblocator ocinumber "+"ociref ocirefcursor ocirowid ocistring ocitype oct octet_length of off offline offset oid oidindex old "+"on online only opaque open operations operator optimal optimize option optionally or oracle oracle_date "+"oradata ord ordaudio orddicom orddoc order ordimage ordinality ordvideo organization orlany orlvary "+"out outer outfile outline output over overflow overriding package pad parallel parallel_enable "+"parameters parent parse partial partition partitions pascal passing password password_grace_time "+"password_lock_time password_reuse_max password_reuse_time password_verify_function patch path patindex "+"pctincrease pctthreshold pctused pctversion percent percent_rank percentile_cont percentile_disc "+"performance period period_add period_diff permanent physical pi pipe pipelined pivot pluggable plugin "+"policy position post_transaction pow power pragma prebuilt precedes preceding precision prediction "+"prediction_cost prediction_details prediction_probability prediction_set prepare present preserve "+"prior priority private private_sga privileges procedural procedure procedure_analyze processlist "+"profiles project prompt protection public publishingservername purge quarter query quick quiesce quota "+"quotename radians raise rand range rank raw read reads readsize rebuild record records "+"recover recovery recursive recycle redo reduced ref reference referenced references referencing refresh "+"regexp_like register regr_avgx regr_avgy regr_count regr_intercept regr_r2 regr_slope regr_sxx regr_sxy "+"reject rekey relational relative relaylog release release_lock relies_on relocate rely rem remainder rename "+"repair repeat replace replicate replication required reset resetlogs resize resource respect restore "+"restricted result result_cache resumable resume retention return returning returns reuse reverse revoke "+"right rlike role roles rollback rolling rollup round row row_count rowdependencies rowid rownum rows "+"rtrim rules safe salt sample save savepoint sb1 sb2 sb4 scan schema schemacheck scn scope scroll "+"sdo_georaster sdo_topo_geometry search sec_to_time second seconds section securefile security seed segment select "+"self semi sequence sequential serializable server servererror session session_user sessions_per_user set "+"sets settings sha sha1 sha2 share shared shared_pool short show shrink shutdown si_averagecolor "+"si_colorhistogram si_featurelist si_positionalcolor si_stillimage si_texture siblings sid sign sin "+"size size_t sizes skip slave sleep smalldatetimefromparts smallfile snapshot some soname sort soundex "+"source space sparse spfile split sql sql_big_result sql_buffer_result sql_cache sql_calc_found_rows "+"sql_small_result sql_variant_property sqlcode sqldata sqlerror sqlname sqlstate sqrt square standalone "+"standby start starting startup statement static statistics stats_binomial_test stats_crosstab "+"stats_ks_test stats_mode stats_mw_test stats_one_way_anova stats_t_test_ stats_t_test_indep "+"stats_t_test_one stats_t_test_paired stats_wsr_test status std stddev stddev_pop stddev_samp stdev "+"stop storage store stored str str_to_date straight_join strcmp strict string struct stuff style subdate "+"subpartition subpartitions substitutable substr substring subtime subtring_index subtype success sum "+"suspend switch switchoffset switchover sync synchronous synonym sys sys_xmlagg sysasm sysaux sysdate "+"sysdatetimeoffset sysdba sysoper system system_user sysutcdatetime table tables tablespace tablesample tan tdo "+"template temporary terminated tertiary_weights test than then thread through tier ties time time_format "+"time_zone timediff timefromparts timeout timestamp timestampadd timestampdiff timezone_abbr "+"timezone_minute timezone_region to to_base64 to_date to_days to_seconds todatetimeoffset trace tracking "+"transaction transactional translate translation treat trigger trigger_nestlevel triggers trim truncate "+"try_cast try_convert try_parse type ub1 ub2 ub4 ucase unarchived unbounded uncompress "+"under undo unhex unicode uniform uninstall union unique unix_timestamp unknown unlimited unlock unnest unpivot "+"unrecoverable unsafe unsigned until untrusted unusable unused update updated upgrade upped upper upsert "+"url urowid usable usage use use_stored_outlines user user_data user_resources users using utc_date "+"utc_timestamp uuid uuid_short validate validate_password_strength validation valist value values var "+"var_samp varcharc vari varia variab variabl variable variables variance varp varraw varrawc varray "+"verify version versions view virtual visible void wait wallet warning warnings week weekday weekofyear "+"wellformed when whene whenev wheneve whenever where while whitespace window with within without work wrapped "+"xdb xml xmlagg xmlattributes xmlcast xmlcolattval xmlelement xmlexists xmlforest xmlindex xmlnamespaces "+"xmlpi xmlquery xmlroot xmlschema xmlserialize xmltable xmltype xor year year_to_month years yearweek",literal:"true false null unknown",built_in:"array bigint binary bit blob bool boolean char character date dec decimal float int int8 integer interval number "+"numeric real record serial serial8 smallint text time timestamp tinyint varchar varchar2 varying void"},contains:[{className:"string",begin:"'",end:"'",contains:[{begin:"''"}]},{className:"string",begin:'"',end:'"',contains:[{begin:'""'}]},{className:"string",begin:"`",end:"`"},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,hljs.HASH_COMMENT_MODE]},hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("perl",function(hljs){var PERL_KEYWORDS="getpwent getservent quotemeta msgrcv scalar kill dbmclose undef lc "+"ma syswrite tr send umask sysopen shmwrite vec qx utime local oct semctl localtime "+"readpipe do return format read sprintf dbmopen pop getpgrp not getpwnam rewinddir qq"+"fileno qw endprotoent wait sethostent bless s|0 opendir continue each sleep endgrent "+"shutdown dump chomp connect getsockname die socketpair close flock exists index shmget"+"sub for endpwent redo lstat msgctl setpgrp abs exit select print ref gethostbyaddr "+"unshift fcntl syscall goto getnetbyaddr join gmtime symlink semget splice x|0 "+"getpeername recv log setsockopt cos last reverse gethostbyname getgrnam study formline "+"endhostent times chop length gethostent getnetent pack getprotoent getservbyname rand "+"mkdir pos chmod y|0 substr endnetent printf next open msgsnd readdir use unlink "+"getsockopt getpriority rindex wantarray hex system getservbyport endservent int chr "+"untie rmdir prototype tell listen fork shmread ucfirst setprotoent else sysseek link "+"getgrgid shmctl waitpid unpack getnetbyname reset chdir grep split require caller "+"lcfirst until warn while values shift telldir getpwuid my getprotobynumber delete and "+"sort uc defined srand accept package seekdir getprotobyname semop our rename seek if q|0 "+"chroot sysread setpwent no crypt getc chown sqrt write setnetent setpriority foreach "+"tie sin msgget map stat getlogin unless elsif truncate exec keys glob tied closedir"+"ioctl socket readlink eval xor readline binmode setservent eof ord bind alarm pipe "+"atan2 getgrent exp time push setgrent gt lt or ne m|0 break given say state when";var SUBST={className:"subst",begin:"[$@]\\{",end:"\\}",keywords:PERL_KEYWORDS};var METHOD={begin:"->{",end:"}"};var VAR={variants:[{begin:/\$\d/},{begin:/[\$%@](\^\w\b|#\w+(::\w+)*|{\w+}|\w+(::\w*)*)/},{begin:/[\$%@][^\s\w{]/,relevance:0}]};var STRING_CONTAINS=[hljs.BACKSLASH_ESCAPE,SUBST,VAR];var PERL_DEFAULT_CONTAINS=[VAR,hljs.HASH_COMMENT_MODE,hljs.COMMENT("^\\=\\w","\\=cut",{endsWithParent:true}),METHOD,{className:"string",contains:STRING_CONTAINS,variants:[{begin:"q[qwxr]?\\s*\\(",end:"\\)",relevance:5},{begin:"q[qwxr]?\\s*\\[",end:"\\]",relevance:5},{begin:"q[qwxr]?\\s*\\{",end:"\\}",relevance:5},{begin:"q[qwxr]?\\s*\\|",end:"\\|",relevance:5},{begin:"q[qwxr]?\\s*\\<",end:"\\>",relevance:5},{begin:"qw\\s+q",end:"q",relevance:5},{begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"',end:'"'},{begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"{\\w+}",contains:[],relevance:0},{begin:"-?\\w+\\s*\\=\\>",contains:[],relevance:0}]},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},{begin:"(\\/\\/|"+hljs.RE_STARTERS_RE+"|\\b(split|return|print|reverse|grep)\\b)\\s*",keywords:"split return print reverse grep",relevance:0,contains:[hljs.HASH_COMMENT_MODE,{className:"regexp",begin:"(s|tr|y)/(\\\\.|[^/])*/(\\\\.|[^/])*/[a-z]*",relevance:10},{className:"regexp",begin:"(m|qr)?/",end:"/[a-z]*",contains:[hljs.BACKSLASH_ESCAPE],relevance:0}]},{className:"function",beginKeywords:"sub",end:"(\\s*\\(.*?\\))?[;{]",excludeEnd:true,relevance:5,contains:[hljs.TITLE_MODE]},{begin:"-\\w\\b",relevance:0},{begin:"^__DATA__$",end:"^__END__$",subLanguage:"mojolicious",contains:[{begin:"^@@.*",end:"$",className:"comment"}]}];SUBST.contains=PERL_DEFAULT_CONTAINS;METHOD.contains=PERL_DEFAULT_CONTAINS;return{aliases:["pl","pm"],lexemes:/[\w\.]+/,keywords:PERL_KEYWORDS,contains:PERL_DEFAULT_CONTAINS}});hljs.registerLanguage("gauss",function(hljs){var KEYWORDS={keyword:"bool break call callexe checkinterrupt clear clearg closeall cls comlog compile "+"continue create debug declare delete disable dlibrary dllcall do dos ed edit else "+"elseif enable end endfor endif endp endo errorlog errorlogat expr external fn "+"for format goto gosub graph if keyword let lib library line load loadarray loadexe "+"loadf loadk loadm loadp loads loadx local locate loopnextindex lprint lpwidth lshow "+"matrix msym ndpclex new open output outwidth plot plotsym pop prcsn print "+"printdos proc push retp return rndcon rndmod rndmult rndseed run save saveall screen "+"scroll setarray show sparse stop string struct system trace trap threadfor "+"threadendfor threadbegin threadjoin threadstat threadend until use while winprint "+"ne ge le gt lt and xor or not eq eqv",built_in:"abs acf aconcat aeye amax amean AmericanBinomCall AmericanBinomCall_Greeks AmericanBinomCall_ImpVol "+"AmericanBinomPut AmericanBinomPut_Greeks AmericanBinomPut_ImpVol AmericanBSCall AmericanBSCall_Greeks "+"AmericanBSCall_ImpVol AmericanBSPut AmericanBSPut_Greeks AmericanBSPut_ImpVol amin amult annotationGetDefaults "+"annotationSetBkd annotationSetFont annotationSetLineColor annotationSetLineStyle annotationSetLineThickness "+"annualTradingDays arccos arcsin areshape arrayalloc arrayindex arrayinit arraytomat asciiload asclabel astd "+"astds asum atan atan2 atranspose axmargin balance band bandchol bandcholsol bandltsol bandrv bandsolpd bar "+"base10 begwind besselj bessely beta box boxcox cdfBeta cdfBetaInv cdfBinomial cdfBinomialInv cdfBvn cdfBvn2 "+"cdfBvn2e cdfCauchy cdfCauchyInv cdfChic cdfChii cdfChinc cdfChincInv cdfExp cdfExpInv cdfFc cdfFnc cdfFncInv "+"cdfGam cdfGenPareto cdfHyperGeo cdfLaplace cdfLaplaceInv cdfLogistic cdfLogisticInv cdfmControlCreate cdfMvn "+"cdfMvn2e cdfMvnce cdfMvne cdfMvt2e cdfMvtce cdfMvte cdfN cdfN2 cdfNc cdfNegBinomial cdfNegBinomialInv cdfNi "+"cdfPoisson cdfPoissonInv cdfRayleigh cdfRayleighInv cdfTc cdfTci cdfTnc cdfTvn cdfWeibull cdfWeibullInv cdir "+"ceil ChangeDir chdir chiBarSquare chol choldn cholsol cholup chrs close code cols colsf combinate combinated "+"complex con cond conj cons ConScore contour conv convertsatostr convertstrtosa corrm corrms corrvc corrx corrxs "+"cos cosh counts countwts crossprd crout croutp csrcol csrlin csvReadM csvReadSA cumprodc cumsumc curve cvtos "+"datacreate datacreatecomplex datalist dataload dataloop dataopen datasave date datestr datestring datestrymd "+"dayinyr dayofweek dbAddDatabase dbClose dbCommit dbCreateQuery dbExecQuery dbGetConnectOptions dbGetDatabaseName "+"dbGetDriverName dbGetDrivers dbGetHostName dbGetLastErrorNum dbGetLastErrorText dbGetNumericalPrecPolicy "+"dbGetPassword dbGetPort dbGetTableHeaders dbGetTables dbGetUserName dbHasFeature dbIsDriverAvailable dbIsOpen "+"dbIsOpenError dbOpen dbQueryBindValue dbQueryClear dbQueryCols dbQueryExecPrepared dbQueryFetchAllM dbQueryFetchAllSA "+"dbQueryFetchOneM dbQueryFetchOneSA dbQueryFinish dbQueryGetBoundValue dbQueryGetBoundValues dbQueryGetField "+"dbQueryGetLastErrorNum dbQueryGetLastErrorText dbQueryGetLastInsertID dbQueryGetLastQuery dbQueryGetPosition "+"dbQueryIsActive dbQueryIsForwardOnly dbQueryIsNull dbQueryIsSelect dbQueryIsValid dbQueryPrepare dbQueryRows "+"dbQuerySeek dbQuerySeekFirst dbQuerySeekLast dbQuerySeekNext dbQuerySeekPrevious dbQuerySetForwardOnly "+"dbRemoveDatabase dbRollback dbSetConnectOptions dbSetDatabaseName dbSetHostName dbSetNumericalPrecPolicy "+"dbSetPort dbSetUserName dbTransaction DeleteFile delif delrows denseToSp denseToSpRE denToZero design det detl "+"dfft dffti diag diagrv digamma doswin DOSWinCloseall DOSWinOpen dotfeq dotfeqmt dotfge dotfgemt dotfgt dotfgtmt "+"dotfle dotflemt dotflt dotfltmt dotfne dotfnemt draw drop dsCreate dstat dstatmt dstatmtControlCreate dtdate dtday "+"dttime dttodtv dttostr dttoutc dtvnormal dtvtodt dtvtoutc dummy dummybr dummydn eig eigh eighv eigv elapsedTradingDays "+"endwind envget eof eqSolve eqSolvemt eqSolvemtControlCreate eqSolvemtOutCreate eqSolveset erf erfc erfccplx erfcplx error "+"etdays ethsec etstr EuropeanBinomCall EuropeanBinomCall_Greeks EuropeanBinomCall_ImpVol EuropeanBinomPut "+"EuropeanBinomPut_Greeks EuropeanBinomPut_ImpVol EuropeanBSCall EuropeanBSCall_Greeks EuropeanBSCall_ImpVol "+"EuropeanBSPut EuropeanBSPut_Greeks EuropeanBSPut_ImpVol exctsmpl exec execbg exp extern eye fcheckerr fclearerr feq "+"feqmt fflush fft ffti fftm fftmi fftn fge fgemt fgets fgetsa fgetsat fgetst fgt fgtmt fileinfo filesa fle flemt "+"floor flt fltmt fmod fne fnemt fonts fopen formatcv formatnv fputs fputst fseek fstrerror ftell ftocv ftos ftostrC "+"gamma gammacplx gammaii gausset gdaAppend gdaCreate gdaDStat gdaDStatMat gdaGetIndex gdaGetName gdaGetNames gdaGetOrders "+"gdaGetType gdaGetTypes gdaGetVarInfo gdaIsCplx gdaLoad gdaPack gdaRead gdaReadByIndex gdaReadSome gdaReadSparse "+"gdaReadStruct gdaReportVarInfo gdaSave gdaUpdate gdaUpdateAndPack gdaVars gdaWrite gdaWrite32 gdaWriteSome getarray "+"getdims getf getGAUSShome getmatrix getmatrix4D getname getnamef getNextTradingDay getNextWeekDay getnr getorders "+"getpath getPreviousTradingDay getPreviousWeekDay getRow getscalar3D getscalar4D getTrRow getwind glm gradcplx gradMT "+"gradMTm gradMTT gradMTTm gradp graphprt graphset hasimag header headermt hess hessMT hessMTg hessMTgw hessMTm "+"hessMTmw hessMTT hessMTTg hessMTTgw hessMTTm hessMTw hessp hist histf histp hsec imag indcv indexcat indices indices2 "+"indicesf indicesfn indnv indsav integrate1d integrateControlCreate intgrat2 intgrat3 inthp1 inthp2 inthp3 inthp4 "+"inthpControlCreate intquad1 intquad2 intquad3 intrleav intrleavsa intrsect intsimp inv invpd invswp iscplx iscplxf "+"isden isinfnanmiss ismiss key keyav keyw lag lag1 lagn lapEighb lapEighi lapEighvb lapEighvi lapgEig lapgEigh lapgEighv "+"lapgEigv lapgSchur lapgSvdcst lapgSvds lapgSvdst lapSvdcusv lapSvds lapSvdusv ldlp ldlsol linSolve listwise ln lncdfbvn "+"lncdfbvn2 lncdfmvn lncdfn lncdfn2 lncdfnc lnfact lngammacplx lnpdfmvn lnpdfmvt lnpdfn lnpdft loadd loadstruct loadwind "+"loess loessmt loessmtControlCreate log loglog logx logy lower lowmat lowmat1 ltrisol lu lusol machEpsilon make makevars "+"makewind margin matalloc matinit mattoarray maxbytes maxc maxindc maxv maxvec mbesselei mbesselei0 mbesselei1 mbesseli "+"mbesseli0 mbesseli1 meanc median mergeby mergevar minc minindc minv miss missex missrv moment momentd movingave "+"movingaveExpwgt movingaveWgt nextindex nextn nextnevn nextwind ntos null null1 numCombinations ols olsmt olsmtControlCreate "+"olsqr olsqr2 olsqrmt ones optn optnevn orth outtyp pacf packedToSp packr parse pause pdfCauchy pdfChi pdfExp pdfGenPareto "+"pdfHyperGeo pdfLaplace pdfLogistic pdfn pdfPoisson pdfRayleigh pdfWeibull pi pinv pinvmt plotAddArrow plotAddBar plotAddBox "+"plotAddHist plotAddHistF plotAddHistP plotAddPolar plotAddScatter plotAddShape plotAddTextbox plotAddTS plotAddXY plotArea "+"plotBar plotBox plotClearLayout plotContour plotCustomLayout plotGetDefaults plotHist plotHistF plotHistP plotLayout "+"plotLogLog plotLogX plotLogY plotOpenWindow plotPolar plotSave plotScatter plotSetAxesPen plotSetBar plotSetBarFill "+"plotSetBarStacked plotSetBkdColor plotSetFill plotSetGrid plotSetLegend plotSetLineColor plotSetLineStyle plotSetLineSymbol "+"plotSetLineThickness plotSetNewWindow plotSetTitle plotSetWhichYAxis plotSetXAxisShow plotSetXLabel plotSetXRange "+"plotSetXTicInterval plotSetXTicLabel plotSetYAxisShow plotSetYLabel plotSetYRange plotSetZAxisShow plotSetZLabel "+"plotSurface plotTS plotXY polar polychar polyeval polygamma polyint polymake polymat polymroot polymult polyroot "+"pqgwin previousindex princomp printfm printfmt prodc psi putarray putf putvals pvCreate pvGetIndex pvGetParNames "+"pvGetParVector pvLength pvList pvPack pvPacki pvPackm pvPackmi pvPacks pvPacksi pvPacksm pvPacksmi pvPutParVector "+"pvTest pvUnpack QNewton QNewtonmt QNewtonmtControlCreate QNewtonmtOutCreate QNewtonSet QProg QProgmt QProgmtInCreate "+"qqr qqre qqrep qr qre qrep qrsol qrtsol qtyr qtyre qtyrep quantile quantiled qyr qyre qyrep qz rank rankindx readr "+"real reclassify reclassifyCuts recode recserar recsercp recserrc rerun rescale reshape rets rev rfft rffti rfftip rfftn "+"rfftnp rfftp rndBernoulli rndBeta rndBinomial rndCauchy rndChiSquare rndCon rndCreateState rndExp rndGamma rndGeo rndGumbel "+"rndHyperGeo rndi rndKMbeta rndKMgam rndKMi rndKMn rndKMnb rndKMp rndKMu rndKMvm rndLaplace rndLCbeta rndLCgam rndLCi rndLCn "+"rndLCnb rndLCp rndLCu rndLCvm rndLogNorm rndMTu rndMVn rndMVt rndn rndnb rndNegBinomial rndp rndPoisson rndRayleigh "+"rndStateSkip rndu rndvm rndWeibull rndWishart rotater round rows rowsf rref sampleData satostrC saved saveStruct savewind "+"scale scale3d scalerr scalinfnanmiss scalmiss schtoc schur searchsourcepath seekr select selif seqa seqm setdif setdifsa "+"setvars setvwrmode setwind shell shiftr sin singleindex sinh sleep solpd sortc sortcc sortd sorthc sorthcc sortind "+"sortindc sortmc sortr sortrc spBiconjGradSol spChol spConjGradSol spCreate spDenseSubmat spDiagRvMat spEigv spEye spLDL "+"spline spLU spNumNZE spOnes spreadSheetReadM spreadSheetReadSA spreadSheetWrite spScale spSubmat spToDense spTrTDense "+"spTScalar spZeros sqpSolve sqpSolveMT sqpSolveMTControlCreate sqpSolveMTlagrangeCreate sqpSolveMToutCreate sqpSolveSet "+"sqrt statements stdc stdsc stocv stof strcombine strindx strlen strput strrindx strsect strsplit strsplitPad strtodt "+"strtof strtofcplx strtriml strtrimr strtrunc strtruncl strtruncpad strtruncr submat subscat substute subvec sumc sumr "+"surface svd svd1 svd2 svdcusv svds svdusv sysstate tab tan tanh tempname "+"time timedt timestr timeutc title tkf2eps tkf2ps tocart todaydt toeplitz token topolar trapchk "+"trigamma trimr trunc type typecv typef union unionsa uniqindx uniqindxsa unique uniquesa upmat upmat1 upper utctodt "+"utctodtv utrisol vals varCovMS varCovXS varget vargetl varmall varmares varput varputl vartypef vcm vcms vcx vcxs "+"vec vech vecr vector vget view viewxyz vlist vnamecv volume vput vread vtypecv wait waitc walkindex where window "+"writer xlabel xlsGetSheetCount xlsGetSheetSize xlsGetSheetTypes xlsMakeRange xlsReadM xlsReadSA xlsWrite xlsWriteM "+"xlsWriteSA xpnd xtics xy xyz ylabel ytics zeros zeta zlabel ztics cdfEmpirical dot h5create h5open h5read h5readAttribute "+"h5write h5writeAttribute ldl plotAddErrorBar plotAddSurface plotCDFEmpirical plotSetColormap plotSetContourLabels "+"plotSetLegendFont plotSetTextInterpreter plotSetXTicCount plotSetYTicCount plotSetZLevels powerm strjoin sylvester "+"strtrim",literal:"DB_AFTER_LAST_ROW DB_ALL_TABLES DB_BATCH_OPERATIONS DB_BEFORE_FIRST_ROW DB_BLOB DB_EVENT_NOTIFICATIONS "+"DB_FINISH_QUERY DB_HIGH_PRECISION DB_LAST_INSERT_ID DB_LOW_PRECISION_DOUBLE DB_LOW_PRECISION_INT32 "+"DB_LOW_PRECISION_INT64 DB_LOW_PRECISION_NUMBERS DB_MULTIPLE_RESULT_SETS DB_NAMED_PLACEHOLDERS "+"DB_POSITIONAL_PLACEHOLDERS DB_PREPARED_QUERIES DB_QUERY_SIZE DB_SIMPLE_LOCKING DB_SYSTEM_TABLES DB_TABLES "+"DB_TRANSACTIONS DB_UNICODE DB_VIEWS __STDIN __STDOUT __STDERR __FILE_DIR"};var AT_COMMENT_MODE=hljs.COMMENT("@","@");var PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"define definecs|10 undef ifdef ifndef iflight ifdllcall ifmac ifos2win ifunix else endif lineson linesoff srcfile srcline"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",end:"$",keywords:{"meta-keyword":"include"},contains:[{className:"meta-string",begin:'"',end:'"',illegal:"\\n"}]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE]};var STRUCT_TYPE={begin:/\bstruct\s+/,end:/\s/,keywords:"struct",contains:[{className:"type",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0}]};var PARSE_PARAMS=[{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,endsWithParent:true,relevance:0,contains:[{className:"literal",begin:/\.\.\./},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,STRUCT_TYPE]}];var FUNCTION_DEF={className:"title",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0};var DEFINITION=function(beginKeywords,end,inherits){var mode=hljs.inherit({className:"function",beginKeywords:beginKeywords,end:end,excludeEnd:true,contains:[].concat(PARSE_PARAMS)},inherits||{});mode.contains.push(FUNCTION_DEF);mode.contains.push(hljs.C_NUMBER_MODE);mode.contains.push(hljs.C_BLOCK_COMMENT_MODE);mode.contains.push(AT_COMMENT_MODE);return mode};var BUILT_IN_REF={className:"built_in",begin:"\\b("+KEYWORDS.built_in.split(" ").join("|")+")\\b"};var STRING_REF={className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE],relevance:0};var FUNCTION_REF={begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,keywords:KEYWORDS,relevance:0,contains:[{beginKeywords:KEYWORDS.keyword},BUILT_IN_REF,{className:"built_in",begin:hljs.UNDERSCORE_IDENT_RE,relevance:0}]};var FUNCTION_REF_PARAMS={begin:/\(/,end:/\)/,relevance:0,keywords:{built_in:KEYWORDS.built_in,literal:KEYWORDS.literal},contains:[hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,BUILT_IN_REF,FUNCTION_REF,STRING_REF,"self"]};FUNCTION_REF.contains.push(FUNCTION_REF_PARAMS);return{aliases:["gss"],case_insensitive:true,keywords:KEYWORDS,illegal:/(\{[%#]|[%#]\}| <- )/,contains:[hljs.C_NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,STRING_REF,PREPROCESSOR,{className:"keyword",begin:/\bexternal (matrix|string|array|sparse matrix|struct|proc|keyword|fn)/},DEFINITION("proc keyword",";"),DEFINITION("fn","="),{beginKeywords:"for threadfor",end:/;/,relevance:0,contains:[hljs.C_BLOCK_COMMENT_MODE,AT_COMMENT_MODE,FUNCTION_REF_PARAMS]},{variants:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\."+hljs.UNDERSCORE_IDENT_RE},{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*="}],relevance:0},FUNCTION_REF,STRUCT_TYPE]}});hljs.registerLanguage("awk",function(hljs){var VARIABLE={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var KEYWORDS="BEGIN END if else while do for in break continue delete next nextfile function func exit|10";var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:/(u|b)?r?'''/,end:/'''/,relevance:10},{begin:/(u|b)?r?"""/,end:/"""/,relevance:10},{begin:/(u|r|ur)'/,end:/'/,relevance:10},{begin:/(u|r|ur)"/,end:/"/,relevance:10},{begin:/(b|br)'/,end:/'/},{begin:/(b|br)"/,end:/"/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};return{keywords:{keyword:KEYWORDS},contains:[VARIABLE,STRING,hljs.REGEXP_MODE,hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("yaml",function(hljs){var LITERALS="true false yes no null";var KEY={className:"attr",variants:[{begin:"\\w[\\w :\\/.-]*:(?=[ \t]|$)"},{begin:'"\\w[\\w :\\/.-]*":(?=[ \t]|$)'},{begin:"'\\w[\\w :\\/.-]*':(?=[ \t]|$)"}]};var TEMPLATE_VARIABLES={className:"template-variable",variants:[{begin:"{{",end:"}}"},{begin:"%{",end:"}"}]};var STRING={className:"string",relevance:0,variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/\S+/}],contains:[hljs.BACKSLASH_ESCAPE,TEMPLATE_VARIABLES]};return{case_insensitive:true,aliases:["yml","YAML","yaml"],contains:[KEY,{className:"meta",begin:"^---s*$",relevance:10},{className:"string",begin:"[\\|>]([0-9]?[+-])?[ ]*\\n( *)[\\S ]+\\n(\\2[\\S ]+\\n?)*"},{begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:"!"+hljs.UNDERSCORE_IDENT_RE},{className:"type",begin:"!!"+hljs.UNDERSCORE_IDENT_RE},{className:"meta",begin:"&"+hljs.UNDERSCORE_IDENT_RE+"$"},{className:"meta",begin:"\\*"+hljs.UNDERSCORE_IDENT_RE+"$"},{className:"bullet",begin:"\\-(?=[ ]|$)",relevance:0},hljs.HASH_COMMENT_MODE,{beginKeywords:LITERALS,keywords:{literal:LITERALS}},{className:"number",begin:hljs.C_NUMBER_RE+"\\b"},STRING]}});hljs.registerLanguage("tap",function(hljs){return{case_insensitive:true,contains:[hljs.HASH_COMMENT_MODE,{className:"meta",variants:[{begin:"^TAP version (\\d+)$"},{begin:"^1\\.\\.(\\d+)$"}]},{begin:"(s+)?---$",end:"\\.\\.\\.$",subLanguage:"yaml",relevance:0},{className:"number",begin:" (\\d+) "},{className:"symbol",variants:[{begin:"^ok"},{begin:"^not ok"}]}]}});hljs.registerLanguage("rib",function(hljs){return{keywords:"ArchiveRecord AreaLightSource Atmosphere Attribute AttributeBegin AttributeEnd Basis "+"Begin Blobby Bound Clipping ClippingPlane Color ColorSamples ConcatTransform Cone "+"CoordinateSystem CoordSysTransform CropWindow Curves Cylinder DepthOfField Detail "+"DetailRange Disk Displacement Display End ErrorHandler Exposure Exterior Format "+"FrameAspectRatio FrameBegin FrameEnd GeneralPolygon GeometricApproximation Geometry "+"Hider Hyperboloid Identity Illuminate Imager Interior LightSource "+"MakeCubeFaceEnvironment MakeLatLongEnvironment MakeShadow MakeTexture Matte "+"MotionBegin MotionEnd NuPatch ObjectBegin ObjectEnd ObjectInstance Opacity Option "+"Orientation Paraboloid Patch PatchMesh Perspective PixelFilter PixelSamples "+"PixelVariance Points PointsGeneralPolygons PointsPolygons Polygon Procedural Projection "+"Quantize ReadArchive RelativeDetail ReverseOrientation Rotate Scale ScreenWindow "+"ShadingInterpolation ShadingRate Shutter Sides Skew SolidBegin SolidEnd Sphere "+"SubdivisionMesh Surface TextureCoordinates Torus Transform TransformBegin TransformEnd "+"TransformPoints Translate TrimCurve WorldBegin WorldEnd",illegal:"</",contains:[hljs.HASH_COMMENT_MODE,hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]}});hljs.registerLanguage("prolog",function(hljs){var ATOM={begin:/[a-z][A-Za-z0-9_]*/,relevance:0};var VAR={className:"symbol",variants:[{begin:/[A-Z][a-zA-Z0-9_]*/},{begin:/_[A-Za-z0-9_]*/}],relevance:0};var PARENTED={begin:/\(/,end:/\)/,relevance:0};var LIST={begin:/\[/,end:/\]/};var LINE_COMMENT={className:"comment",begin:/%/,end:/$/,contains:[hljs.PHRASAL_WORDS_MODE]};var BACKTICK_STRING={className:"string",begin:/`/,end:/`/,contains:[hljs.BACKSLASH_ESCAPE]};var CHAR_CODE={className:"string",begin:/0\'(\\\'|.)/};var SPACE_CODE={className:"string",begin:/0\'\\s/};var PRED_OP={begin:/:-/};var inner=[ATOM,VAR,PARENTED,PRED_OP,LIST,LINE_COMMENT,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,BACKTICK_STRING,CHAR_CODE,SPACE_CODE,hljs.C_NUMBER_MODE];PARENTED.contains=inner;LIST.contains=inner;return{contains:inner.concat([{begin:/\.$/}])}});hljs.registerLanguage("actionscript",function(hljs){var IDENT_RE="[a-zA-Z_$][a-zA-Z0-9_$]*";var IDENT_FUNC_RETURN_TYPE_RE="([*]|[a-zA-Z_$][a-zA-Z0-9_$]*)";var AS3_REST_ARG_MODE={className:"rest_arg",begin:"[.]{3}",end:IDENT_RE,relevance:10};return{aliases:["as"],keywords:{keyword:"as break case catch class const continue default delete do dynamic each "+"else extends final finally for function get if implements import in include "+"instanceof interface internal is namespace native new override package private "+"protected public return set static super switch this throw try typeof use var void "+"while with",literal:"true false null undefined"},contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"class",beginKeywords:"package",end:"{",contains:[hljs.TITLE_MODE]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,contains:[{beginKeywords:"extends implements"},hljs.TITLE_MODE]},{className:"meta",beginKeywords:"import include",end:";",keywords:{"meta-keyword":"import include"}},{className:"function",beginKeywords:"function",end:"[{;]",excludeEnd:true,illegal:"\\S",contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AS3_REST_ARG_MODE]},{begin:":\\s*"+IDENT_FUNC_RETURN_TYPE_RE}]},hljs.METHOD_GUARD],illegal:/#/}});hljs.registerLanguage("ebnf",function(hljs){var commentMode=hljs.COMMENT(/\(\*/,/\*\)/);var nonTerminalMode={className:"attribute",begin:/^[ ]*[a-zA-Z][a-zA-Z-_]*([\s-_]+[a-zA-Z][a-zA-Z]*)*/};var specialSequenceMode={className:"meta",begin:/\?.*\?/};var ruleBodyMode={begin:/=/,end:/[.;]/,contains:[commentMode,specialSequenceMode,{className:"string",variants:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"`",end:"`"}]}]};return{illegal:/\S/,contains:[commentMode,nonTerminalMode,ruleBodyMode]}});hljs.registerLanguage("fix",function(hljs){return{contains:[{begin:/[^\u2401\u0001]+/,end:/[\u2401\u0001]/,excludeEnd:true,returnBegin:true,returnEnd:false,contains:[{begin:/([^\u2401\u0001=]+)/,end:/=([^\u2401\u0001=]+)/,returnEnd:true,returnBegin:false,className:"attr"},{begin:/=/,end:/([\u2401\u0001])/,excludeEnd:true,excludeBegin:true,className:"string"}]}],case_insensitive:true}});hljs.registerLanguage("bnf",function(hljs){return{contains:[{className:"attribute",begin:/</,end:/>/},{begin:/::=/,starts:{end:/$/,contains:[{begin:/</,end:/>/},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]}}]}});hljs.registerLanguage("isbl",function(hljs){var UNDERSCORE_IDENT_RE="[A-Za-zА-Яа-яёЁ_!][A-Za-zА-Яа-яёЁ_0-9]*";var FUNCTION_NAME_IDENT_RE="[A-Za-zА-Яа-яёЁ_][A-Za-zА-Яа-яёЁ_0-9]*";var KEYWORD="and и else иначе endexcept endfinally endforeach конецвсе endif конецесли endwhile конецпока "+"except exitfor finally foreach все if если in в not не or или try while пока ";var sysres_constants="SYSRES_CONST_ACCES_RIGHT_TYPE_EDIT "+"SYSRES_CONST_ACCES_RIGHT_TYPE_FULL "+"SYSRES_CONST_ACCES_RIGHT_TYPE_VIEW "+"SYSRES_CONST_ACCESS_MODE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_NO_ACCESS_VIEW "+"SYSRES_CONST_ACCESS_NO_ACCESS_VIEW_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_ADD_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_ADD_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_CHANGE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_CHANGE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_DELETE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_DELETE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_EXECUTE_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_EXECUTE_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_NO_ACCESS_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_NO_ACCESS_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_RATIFY_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_RATIFY_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_REQUISITE_CODE "+"SYSRES_CONST_ACCESS_RIGHTS_VIEW_REQUISITE_YES_CODE "+"SYSRES_CONST_ACCESS_TYPE_CHANGE "+"SYSRES_CONST_ACCESS_TYPE_CHANGE_CODE "+"SYSRES_CONST_ACCESS_TYPE_EXISTS "+"SYSRES_CONST_ACCESS_TYPE_EXISTS_CODE "+"SYSRES_CONST_ACCESS_TYPE_FULL "+"SYSRES_CONST_ACCESS_TYPE_FULL_CODE "+"SYSRES_CONST_ACCESS_TYPE_VIEW "+"SYSRES_CONST_ACCESS_TYPE_VIEW_CODE "+"SYSRES_CONST_ACTION_TYPE_ABORT "+"SYSRES_CONST_ACTION_TYPE_ACCEPT "+"SYSRES_CONST_ACTION_TYPE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ADD_ATTACHMENT "+"SYSRES_CONST_ACTION_TYPE_CHANGE_CARD "+"SYSRES_CONST_ACTION_TYPE_CHANGE_KIND "+"SYSRES_CONST_ACTION_TYPE_CHANGE_STORAGE "+"SYSRES_CONST_ACTION_TYPE_CONTINUE "+"SYSRES_CONST_ACTION_TYPE_COPY "+"SYSRES_CONST_ACTION_TYPE_CREATE "+"SYSRES_CONST_ACTION_TYPE_CREATE_VERSION "+"SYSRES_CONST_ACTION_TYPE_DELETE "+"SYSRES_CONST_ACTION_TYPE_DELETE_ATTACHMENT "+"SYSRES_CONST_ACTION_TYPE_DELETE_VERSION "+"SYSRES_CONST_ACTION_TYPE_DISABLE_DELEGATE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ENABLE_DELEGATE_ACCESS_RIGHTS "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_CERTIFICATE "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_CERTIFICATE_AND_PASSWORD "+"SYSRES_CONST_ACTION_TYPE_ENCRYPTION_BY_PASSWORD "+"SYSRES_CONST_ACTION_TYPE_EXPORT_WITH_LOCK "+"SYSRES_CONST_ACTION_TYPE_EXPORT_WITHOUT_LOCK "+"SYSRES_CONST_ACTION_TYPE_IMPORT_WITH_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_IMPORT_WITHOUT_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_LIFE_CYCLE_STAGE "+"SYSRES_CONST_ACTION_TYPE_LOCK "+"SYSRES_CONST_ACTION_TYPE_LOCK_FOR_SERVER "+"SYSRES_CONST_ACTION_TYPE_LOCK_MODIFY "+"SYSRES_CONST_ACTION_TYPE_MARK_AS_READED "+"SYSRES_CONST_ACTION_TYPE_MARK_AS_UNREADED "+"SYSRES_CONST_ACTION_TYPE_MODIFY "+"SYSRES_CONST_ACTION_TYPE_MODIFY_CARD "+"SYSRES_CONST_ACTION_TYPE_MOVE_TO_ARCHIVE "+"SYSRES_CONST_ACTION_TYPE_OFF_ENCRYPTION "+"SYSRES_CONST_ACTION_TYPE_PASSWORD_CHANGE "+"SYSRES_CONST_ACTION_TYPE_PERFORM "+"SYSRES_CONST_ACTION_TYPE_RECOVER_FROM_LOCAL_COPY "+"SYSRES_CONST_ACTION_TYPE_RESTART "+"SYSRES_CONST_ACTION_TYPE_RESTORE_FROM_ARCHIVE "+"SYSRES_CONST_ACTION_TYPE_REVISION "+"SYSRES_CONST_ACTION_TYPE_SEND_BY_MAIL "+"SYSRES_CONST_ACTION_TYPE_SIGN "+"SYSRES_CONST_ACTION_TYPE_START "+"SYSRES_CONST_ACTION_TYPE_UNLOCK "+"SYSRES_CONST_ACTION_TYPE_UNLOCK_FROM_SERVER "+"SYSRES_CONST_ACTION_TYPE_VERSION_STATE "+"SYSRES_CONST_ACTION_TYPE_VERSION_VISIBILITY "+"SYSRES_CONST_ACTION_TYPE_VIEW "+"SYSRES_CONST_ACTION_TYPE_VIEW_SHADOW_COPY "+"SYSRES_CONST_ACTION_TYPE_WORKFLOW_DESCRIPTION_MODIFY "+"SYSRES_CONST_ACTION_TYPE_WRITE_HISTORY "+"SYSRES_CONST_ACTIVE_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_ADD_REFERENCE_MODE_NAME "+"SYSRES_CONST_ADDITION_REQUISITE_CODE "+"SYSRES_CONST_ADDITIONAL_PARAMS_REQUISITE_CODE "+"SYSRES_CONST_ADITIONAL_JOB_END_DATE_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_READ_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_START_DATE_REQUISITE_NAME "+"SYSRES_CONST_ADITIONAL_JOB_STATE_REQUISITE_NAME "+"SYSRES_CONST_ADMINISTRATION_HISTORY_ADDING_USER_TO_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_ADDING_USER_TO_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_COMP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_COMP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_USER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_CREATION_USER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_CREATION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_CREATION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_DELETION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DATABASE_USER_DELETION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_COMP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_COMP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_FROM_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_DELETION_USER_FROM_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_RESTRICTION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_FILTERER_RESTRICTION_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_PRIVILEGE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_PRIVILEGE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_RIGHTS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_GRANTING_RIGHTS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_MAIN_SERVER_CHANGED_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_MAIN_SERVER_CHANGED_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_PUBLIC_CHANGED_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_IS_PUBLIC_CHANGED_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_RESTRICTION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_FILTERER_RESTRICTION_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_PRIVILEGE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_PRIVILEGE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_RIGHTS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_REMOVING_RIGHTS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_CREATION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_CREATION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_DELETION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_SERVER_LOGIN_DELETION_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_CATEGORY_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_CATEGORY_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_COMP_TITLE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_COMP_TITLE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_FULL_NAME_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_FULL_NAME_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_PARENT_GROUP_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_PARENT_GROUP_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_AUTH_TYPE_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_AUTH_TYPE_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_LOGIN_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_LOGIN_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_STATUS_ACTION "+"SYSRES_CONST_ADMINISTRATION_HISTORY_UPDATING_USER_STATUS_ACTION_CODE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_USER_PASSWORD_CHANGE "+"SYSRES_CONST_ADMINISTRATION_HISTORY_USER_PASSWORD_CHANGE_ACTION "+"SYSRES_CONST_ALL_ACCEPT_CONDITION_RUS "+"SYSRES_CONST_ALL_USERS_GROUP "+"SYSRES_CONST_ALL_USERS_GROUP_NAME "+"SYSRES_CONST_ALL_USERS_SERVER_GROUP_NAME "+"SYSRES_CONST_ALLOWED_ACCESS_TYPE_CODE "+"SYSRES_CONST_ALLOWED_ACCESS_TYPE_NAME "+"SYSRES_CONST_APP_VIEWER_TYPE_REQUISITE_CODE "+"SYSRES_CONST_APPROVING_SIGNATURE_NAME "+"SYSRES_CONST_APPROVING_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_ASSISTANT_SUBSTITUE_TYPE "+"SYSRES_CONST_ASSISTANT_SUBSTITUE_TYPE_CODE "+"SYSRES_CONST_ATTACH_TYPE_COMPONENT_TOKEN "+"SYSRES_CONST_ATTACH_TYPE_DOC "+"SYSRES_CONST_ATTACH_TYPE_EDOC "+"SYSRES_CONST_ATTACH_TYPE_FOLDER "+"SYSRES_CONST_ATTACH_TYPE_JOB "+"SYSRES_CONST_ATTACH_TYPE_REFERENCE "+"SYSRES_CONST_ATTACH_TYPE_TASK "+"SYSRES_CONST_AUTH_ENCODED_PASSWORD "+"SYSRES_CONST_AUTH_ENCODED_PASSWORD_CODE "+"SYSRES_CONST_AUTH_NOVELL "+"SYSRES_CONST_AUTH_PASSWORD "+"SYSRES_CONST_AUTH_PASSWORD_CODE "+"SYSRES_CONST_AUTH_WINDOWS "+"SYSRES_CONST_AUTHENTICATING_SIGNATURE_NAME "+"SYSRES_CONST_AUTHENTICATING_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_AUTO_ENUM_METHOD_FLAG "+"SYSRES_CONST_AUTO_NUMERATION_CODE "+"SYSRES_CONST_AUTO_STRONG_ENUM_METHOD_FLAG "+"SYSRES_CONST_AUTOTEXT_NAME_REQUISITE_CODE "+"SYSRES_CONST_AUTOTEXT_TEXT_REQUISITE_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_ALL "+"SYSRES_CONST_AUTOTEXT_USAGE_ALL_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_SIGN "+"SYSRES_CONST_AUTOTEXT_USAGE_SIGN_CODE "+"SYSRES_CONST_AUTOTEXT_USAGE_WORK "+"SYSRES_CONST_AUTOTEXT_USAGE_WORK_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ANYWHERE_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ON_SIGNING_CODE "+"SYSRES_CONST_AUTOTEXT_USE_ON_WORK_CODE "+"SYSRES_CONST_BEGIN_DATE_REQUISITE_CODE "+"SYSRES_CONST_BLACK_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_BLUE_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_BTN_PART "+"SYSRES_CONST_CALCULATED_ROLE_TYPE_CODE "+"SYSRES_CONST_CALL_TYPE_VARIABLE_BUTTON_VALUE "+"SYSRES_CONST_CALL_TYPE_VARIABLE_PROGRAM_VALUE "+"SYSRES_CONST_CANCEL_MESSAGE_FUNCTION_RESULT "+"SYSRES_CONST_CARD_PART "+"SYSRES_CONST_CARD_REFERENCE_MODE_NAME "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_ENCRYPT_VALUE "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_SIGN_AND_ENCRYPT_VALUE "+"SYSRES_CONST_CERTIFICATE_TYPE_REQUISITE_SIGN_VALUE "+"SYSRES_CONST_CHECK_PARAM_VALUE_DATE_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_FLOAT_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_INTEGER_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_PICK_PARAM_TYPE "+"SYSRES_CONST_CHECK_PARAM_VALUE_REEFRENCE_PARAM_TYPE "+"SYSRES_CONST_CLOSED_RECORD_FLAG_VALUE_FEMININE "+"SYSRES_CONST_CLOSED_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_ADMIN "+"SYSRES_CONST_CODE_COMPONENT_TYPE_DEVELOPER "+"SYSRES_CONST_CODE_COMPONENT_TYPE_DOCS "+"SYSRES_CONST_CODE_COMPONENT_TYPE_EDOC_CARDS "+"SYSRES_CONST_CODE_COMPONENT_TYPE_EXTERNAL_EXECUTABLE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_OTHER "+"SYSRES_CONST_CODE_COMPONENT_TYPE_REFERENCE "+"SYSRES_CONST_CODE_COMPONENT_TYPE_REPORT "+"SYSRES_CONST_CODE_COMPONENT_TYPE_SCRIPT "+"SYSRES_CONST_CODE_COMPONENT_TYPE_URL "+"SYSRES_CONST_CODE_REQUISITE_ACCESS "+"SYSRES_CONST_CODE_REQUISITE_CODE "+"SYSRES_CONST_CODE_REQUISITE_COMPONENT "+"SYSRES_CONST_CODE_REQUISITE_DESCRIPTION "+"SYSRES_CONST_CODE_REQUISITE_EXCLUDE_COMPONENT "+"SYSRES_CONST_CODE_REQUISITE_RECORD "+"SYSRES_CONST_COMMENT_REQ_CODE "+"SYSRES_CONST_COMMON_SETTINGS_REQUISITE_CODE "+"SYSRES_CONST_COMP_CODE_GRD "+"SYSRES_CONST_COMPONENT_GROUP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_COMPONENT_TYPE_ADMIN_COMPONENTS "+"SYSRES_CONST_COMPONENT_TYPE_DEVELOPER_COMPONENTS "+"SYSRES_CONST_COMPONENT_TYPE_DOCS "+"SYSRES_CONST_COMPONENT_TYPE_EDOC_CARDS "+"SYSRES_CONST_COMPONENT_TYPE_EDOCS "+"SYSRES_CONST_COMPONENT_TYPE_EXTERNAL_EXECUTABLE "+"SYSRES_CONST_COMPONENT_TYPE_OTHER "+"SYSRES_CONST_COMPONENT_TYPE_REFERENCE_TYPES "+"SYSRES_CONST_COMPONENT_TYPE_REFERENCES "+"SYSRES_CONST_COMPONENT_TYPE_REPORTS "+"SYSRES_CONST_COMPONENT_TYPE_SCRIPTS "+"SYSRES_CONST_COMPONENT_TYPE_URL "+"SYSRES_CONST_COMPONENTS_REMOTE_SERVERS_VIEW_CODE "+"SYSRES_CONST_CONDITION_BLOCK_DESCRIPTION "+"SYSRES_CONST_CONST_FIRM_STATUS_COMMON "+"SYSRES_CONST_CONST_FIRM_STATUS_INDIVIDUAL "+"SYSRES_CONST_CONST_NEGATIVE_VALUE "+"SYSRES_CONST_CONST_POSITIVE_VALUE "+"SYSRES_CONST_CONST_SERVER_STATUS_DONT_REPLICATE "+"SYSRES_CONST_CONST_SERVER_STATUS_REPLICATE "+"SYSRES_CONST_CONTENTS_REQUISITE_CODE "+"SYSRES_CONST_DATA_TYPE_BOOLEAN "+"SYSRES_CONST_DATA_TYPE_DATE "+"SYSRES_CONST_DATA_TYPE_FLOAT "+"SYSRES_CONST_DATA_TYPE_INTEGER "+"SYSRES_CONST_DATA_TYPE_PICK "+"SYSRES_CONST_DATA_TYPE_REFERENCE "+"SYSRES_CONST_DATA_TYPE_STRING "+"SYSRES_CONST_DATA_TYPE_TEXT "+"SYSRES_CONST_DATA_TYPE_VARIANT "+"SYSRES_CONST_DATE_CLOSE_REQ_CODE "+"SYSRES_CONST_DATE_FORMAT_DATE_ONLY_CHAR "+"SYSRES_CONST_DATE_OPEN_REQ_CODE "+"SYSRES_CONST_DATE_REQUISITE "+"SYSRES_CONST_DATE_REQUISITE_CODE "+"SYSRES_CONST_DATE_REQUISITE_NAME "+"SYSRES_CONST_DATE_REQUISITE_TYPE "+"SYSRES_CONST_DATE_TYPE_CHAR "+"SYSRES_CONST_DATETIME_FORMAT_VALUE "+"SYSRES_CONST_DEA_ACCESS_RIGHTS_ACTION_CODE "+"SYSRES_CONST_DESCRIPTION_LOCALIZE_ID_REQUISITE_CODE "+"SYSRES_CONST_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_DET1_PART "+"SYSRES_CONST_DET2_PART "+"SYSRES_CONST_DET3_PART "+"SYSRES_CONST_DET4_PART "+"SYSRES_CONST_DET5_PART "+"SYSRES_CONST_DET6_PART "+"SYSRES_CONST_DETAIL_DATASET_KEY_REQUISITE_CODE "+"SYSRES_CONST_DETAIL_PICK_REQUISITE_CODE "+"SYSRES_CONST_DETAIL_REQ_CODE "+"SYSRES_CONST_DO_NOT_USE_ACCESS_TYPE_CODE "+"SYSRES_CONST_DO_NOT_USE_ACCESS_TYPE_NAME "+"SYSRES_CONST_DO_NOT_USE_ON_VIEW_ACCESS_TYPE_CODE "+"SYSRES_CONST_DO_NOT_USE_ON_VIEW_ACCESS_TYPE_NAME "+"SYSRES_CONST_DOCUMENT_STORAGES_CODE "+"SYSRES_CONST_DOCUMENT_TEMPLATES_TYPE_NAME "+"SYSRES_CONST_DOUBLE_REQUISITE_CODE "+"SYSRES_CONST_EDITOR_CLOSE_FILE_OBSERV_TYPE_CODE "+"SYSRES_CONST_EDITOR_CLOSE_PROCESS_OBSERV_TYPE_CODE "+"SYSRES_CONST_EDITOR_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_APPLICATION_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_CREATE_SEVERAL_PROCESSES_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_EXTENSION_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_OBSERVER_BY_PROCESS_TYPE "+"SYSRES_CONST_EDITORS_REFERENCE_CODE "+"SYSRES_CONST_EDITORS_REPLACE_SPEC_CHARS_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_USE_PLUGINS_REQUISITE_CODE "+"SYSRES_CONST_EDITORS_VIEW_DOCUMENT_OPENED_TO_EDIT_CODE "+"SYSRES_CONST_EDOC_CARD_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_CARD_TYPES_LINK_REQUISITE_CODE "+"SYSRES_CONST_EDOC_CERTIFICATE_AND_PASSWORD_ENCODE_CODE "+"SYSRES_CONST_EDOC_CERTIFICATE_ENCODE_CODE "+"SYSRES_CONST_EDOC_DATE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_KIND_REFERENCE_CODE "+"SYSRES_CONST_EDOC_KINDS_BY_TEMPLATE_ACTION_CODE "+"SYSRES_CONST_EDOC_MANAGE_ACCESS_CODE "+"SYSRES_CONST_EDOC_NONE_ENCODE_CODE "+"SYSRES_CONST_EDOC_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_EDOC_PASSWORD_ENCODE_CODE "+"SYSRES_CONST_EDOC_READONLY_ACCESS_CODE "+"SYSRES_CONST_EDOC_SHELL_LIFE_TYPE_VIEW_VALUE "+"SYSRES_CONST_EDOC_SIZE_RESTRICTION_PRIORITY_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_CHECK_ACCESS_RIGHTS_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_COMPUTER_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_DATABASE_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_EDIT_IN_STORAGE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_LOCAL_PATH_REQUISITE_CODE "+"SYSRES_CONST_EDOC_STORAGE_SHARED_SOURCE_NAME_REQUISITE_CODE "+"SYSRES_CONST_EDOC_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_EDOC_TYPES_REFERENCE_CODE "+"SYSRES_CONST_EDOC_VERSION_ACTIVE_STAGE_CODE "+"SYSRES_CONST_EDOC_VERSION_DESIGN_STAGE_CODE "+"SYSRES_CONST_EDOC_VERSION_OBSOLETE_STAGE_CODE "+"SYSRES_CONST_EDOC_WRITE_ACCES_CODE "+"SYSRES_CONST_EDOCUMENT_CARD_REQUISITES_REFERENCE_CODE_SELECTED_REQUISITE "+"SYSRES_CONST_ENCODE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_END_DATE_REQUISITE_CODE "+"SYSRES_CONST_ENUMERATION_TYPE_REQUISITE_CODE "+"SYSRES_CONST_EXECUTE_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_EXECUTIVE_FILE_STORAGE_TYPE "+"SYSRES_CONST_EXIST_CONST "+"SYSRES_CONST_EXIST_VALUE "+"SYSRES_CONST_EXPORT_LOCK_TYPE_ASK "+"SYSRES_CONST_EXPORT_LOCK_TYPE_WITH_LOCK "+"SYSRES_CONST_EXPORT_LOCK_TYPE_WITHOUT_LOCK "+"SYSRES_CONST_EXPORT_VERSION_TYPE_ASK "+"SYSRES_CONST_EXPORT_VERSION_TYPE_LAST "+"SYSRES_CONST_EXPORT_VERSION_TYPE_LAST_ACTIVE "+"SYSRES_CONST_EXTENSION_REQUISITE_CODE "+"SYSRES_CONST_FILTER_NAME_REQUISITE_CODE "+"SYSRES_CONST_FILTER_REQUISITE_CODE "+"SYSRES_CONST_FILTER_TYPE_COMMON_CODE "+"SYSRES_CONST_FILTER_TYPE_COMMON_NAME "+"SYSRES_CONST_FILTER_TYPE_USER_CODE "+"SYSRES_CONST_FILTER_TYPE_USER_NAME "+"SYSRES_CONST_FILTER_VALUE_REQUISITE_NAME "+"SYSRES_CONST_FLOAT_NUMBER_FORMAT_CHAR "+"SYSRES_CONST_FLOAT_REQUISITE_TYPE "+"SYSRES_CONST_FOLDER_AUTHOR_VALUE "+"SYSRES_CONST_FOLDER_KIND_ANY_OBJECTS "+"SYSRES_CONST_FOLDER_KIND_COMPONENTS "+"SYSRES_CONST_FOLDER_KIND_EDOCS "+"SYSRES_CONST_FOLDER_KIND_JOBS "+"SYSRES_CONST_FOLDER_KIND_TASKS "+"SYSRES_CONST_FOLDER_TYPE_COMMON "+"SYSRES_CONST_FOLDER_TYPE_COMPONENT "+"SYSRES_CONST_FOLDER_TYPE_FAVORITES "+"SYSRES_CONST_FOLDER_TYPE_INBOX "+"SYSRES_CONST_FOLDER_TYPE_OUTBOX "+"SYSRES_CONST_FOLDER_TYPE_QUICK_LAUNCH "+"SYSRES_CONST_FOLDER_TYPE_SEARCH "+"SYSRES_CONST_FOLDER_TYPE_SHORTCUTS "+"SYSRES_CONST_FOLDER_TYPE_USER "+"SYSRES_CONST_FROM_DICTIONARY_ENUM_METHOD_FLAG "+"SYSRES_CONST_FULL_SUBSTITUTE_TYPE "+"SYSRES_CONST_FULL_SUBSTITUTE_TYPE_CODE "+"SYSRES_CONST_FUNCTION_CANCEL_RESULT "+"SYSRES_CONST_FUNCTION_CATEGORY_SYSTEM "+"SYSRES_CONST_FUNCTION_CATEGORY_USER "+"SYSRES_CONST_FUNCTION_FAILURE_RESULT "+"SYSRES_CONST_FUNCTION_SAVE_RESULT "+"SYSRES_CONST_GENERATED_REQUISITE "+"SYSRES_CONST_GREEN_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_GROUP_ACCOUNT_TYPE_VALUE_CODE "+"SYSRES_CONST_GROUP_CATEGORY_NORMAL_CODE "+"SYSRES_CONST_GROUP_CATEGORY_NORMAL_NAME "+"SYSRES_CONST_GROUP_CATEGORY_SERVICE_CODE "+"SYSRES_CONST_GROUP_CATEGORY_SERVICE_NAME "+"SYSRES_CONST_GROUP_COMMON_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_GROUP_FULL_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_RIGHTS_T_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVER_CODES_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVER_NAME_REQUISITE_CODE "+"SYSRES_CONST_GROUP_SERVICE_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_GROUP_USER_REQUISITE_CODE "+"SYSRES_CONST_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_GROUPS_REQUISITE_CODE "+"SYSRES_CONST_HIDDEN_MODE_NAME "+"SYSRES_CONST_HIGH_LVL_REQUISITE_CODE "+"SYSRES_CONST_HISTORY_ACTION_CREATE_CODE "+"SYSRES_CONST_HISTORY_ACTION_DELETE_CODE "+"SYSRES_CONST_HISTORY_ACTION_EDIT_CODE "+"SYSRES_CONST_HOUR_CHAR "+"SYSRES_CONST_ID_REQUISITE_CODE "+"SYSRES_CONST_IDSPS_REQUISITE_CODE "+"SYSRES_CONST_IMAGE_MODE_COLOR "+"SYSRES_CONST_IMAGE_MODE_GREYSCALE "+"SYSRES_CONST_IMAGE_MODE_MONOCHROME "+"SYSRES_CONST_IMPORTANCE_HIGH "+"SYSRES_CONST_IMPORTANCE_LOW "+"SYSRES_CONST_IMPORTANCE_NORMAL "+"SYSRES_CONST_IN_DESIGN_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_INCOMING_WORK_RULE_TYPE_CODE "+"SYSRES_CONST_INT_REQUISITE "+"SYSRES_CONST_INT_REQUISITE_TYPE "+"SYSRES_CONST_INTEGER_NUMBER_FORMAT_CHAR "+"SYSRES_CONST_INTEGER_TYPE_CHAR "+"SYSRES_CONST_IS_GENERATED_REQUISITE_NEGATIVE_VALUE "+"SYSRES_CONST_IS_PUBLIC_ROLE_REQUISITE_CODE "+"SYSRES_CONST_IS_REMOTE_USER_NEGATIVE_VALUE "+"SYSRES_CONST_IS_REMOTE_USER_POSITIVE_VALUE "+"SYSRES_CONST_IS_STORED_REQUISITE_NEGATIVE_VALUE "+"SYSRES_CONST_IS_STORED_REQUISITE_STORED_VALUE "+"SYSRES_CONST_ITALIC_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_JOB_BLOCK_DESCRIPTION "+"SYSRES_CONST_JOB_KIND_CONTROL_JOB "+"SYSRES_CONST_JOB_KIND_JOB "+"SYSRES_CONST_JOB_KIND_NOTICE "+"SYSRES_CONST_JOB_STATE_ABORTED "+"SYSRES_CONST_JOB_STATE_COMPLETE "+"SYSRES_CONST_JOB_STATE_WORKING "+"SYSRES_CONST_KIND_REQUISITE_CODE "+"SYSRES_CONST_KIND_REQUISITE_NAME "+"SYSRES_CONST_KINDS_CREATE_SHADOW_COPIES_REQUISITE_CODE "+"SYSRES_CONST_KINDS_DEFAULT_EDOC_LIFE_STAGE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALL_TEPLATES_ALLOWED_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALLOW_LIFE_CYCLE_STAGE_CHANGING_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_ALLOW_MULTIPLE_ACTIVE_VERSIONS_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_SHARE_ACCES_RIGHTS_BY_DEFAULT_CODE "+"SYSRES_CONST_KINDS_EDOC_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_EDOC_TYPE_REQUISITE_CODE "+"SYSRES_CONST_KINDS_SIGNERS_REQUISITES_CODE "+"SYSRES_CONST_KOD_INPUT_TYPE "+"SYSRES_CONST_LAST_UPDATE_DATE_REQUISITE_CODE "+"SYSRES_CONST_LIFE_CYCLE_START_STAGE_REQUISITE_CODE "+"SYSRES_CONST_LILAC_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_LINK_OBJECT_KIND_COMPONENT "+"SYSRES_CONST_LINK_OBJECT_KIND_DOCUMENT "+"SYSRES_CONST_LINK_OBJECT_KIND_EDOC "+"SYSRES_CONST_LINK_OBJECT_KIND_FOLDER "+"SYSRES_CONST_LINK_OBJECT_KIND_JOB "+"SYSRES_CONST_LINK_OBJECT_KIND_REFERENCE "+"SYSRES_CONST_LINK_OBJECT_KIND_TASK "+"SYSRES_CONST_LINK_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_LIST_REFERENCE_MODE_NAME "+"SYSRES_CONST_LOCALIZATION_DICTIONARY_MAIN_VIEW_CODE "+"SYSRES_CONST_MAIN_VIEW_CODE "+"SYSRES_CONST_MANUAL_ENUM_METHOD_FLAG "+"SYSRES_CONST_MASTER_COMP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_MASTER_TABLE_REC_ID_REQUISITE_CODE "+"SYSRES_CONST_MAXIMIZED_MODE_NAME "+"SYSRES_CONST_ME_VALUE "+"SYSRES_CONST_MESSAGE_ATTENTION_CAPTION "+"SYSRES_CONST_MESSAGE_CONFIRMATION_CAPTION "+"SYSRES_CONST_MESSAGE_ERROR_CAPTION "+"SYSRES_CONST_MESSAGE_INFORMATION_CAPTION "+"SYSRES_CONST_MINIMIZED_MODE_NAME "+"SYSRES_CONST_MINUTE_CHAR "+"SYSRES_CONST_MODULE_REQUISITE_CODE "+"SYSRES_CONST_MONITORING_BLOCK_DESCRIPTION "+"SYSRES_CONST_MONTH_FORMAT_VALUE "+"SYSRES_CONST_NAME_LOCALIZE_ID_REQUISITE_CODE "+"SYSRES_CONST_NAME_REQUISITE_CODE "+"SYSRES_CONST_NAME_SINGULAR_REQUISITE_CODE "+"SYSRES_CONST_NAMEAN_INPUT_TYPE "+"SYSRES_CONST_NEGATIVE_PICK_VALUE "+"SYSRES_CONST_NEGATIVE_VALUE "+"SYSRES_CONST_NO "+"SYSRES_CONST_NO_PICK_VALUE "+"SYSRES_CONST_NO_SIGNATURE_REQUISITE_CODE "+"SYSRES_CONST_NO_VALUE "+"SYSRES_CONST_NONE_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_NONOPERATING_RECORD_FLAG_VALUE "+"SYSRES_CONST_NONOPERATING_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_NORMAL_ACCESS_RIGHTS_TYPE_CODE "+"SYSRES_CONST_NORMAL_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_NORMAL_MODE_NAME "+"SYSRES_CONST_NOT_ALLOWED_ACCESS_TYPE_CODE "+"SYSRES_CONST_NOT_ALLOWED_ACCESS_TYPE_NAME "+"SYSRES_CONST_NOTE_REQUISITE_CODE "+"SYSRES_CONST_NOTICE_BLOCK_DESCRIPTION "+"SYSRES_CONST_NUM_REQUISITE "+"SYSRES_CONST_NUM_STR_REQUISITE_CODE "+"SYSRES_CONST_NUMERATION_AUTO_NOT_STRONG "+"SYSRES_CONST_NUMERATION_AUTO_STRONG "+"SYSRES_CONST_NUMERATION_FROM_DICTONARY "+"SYSRES_CONST_NUMERATION_MANUAL "+"SYSRES_CONST_NUMERIC_TYPE_CHAR "+"SYSRES_CONST_NUMREQ_REQUISITE_CODE "+"SYSRES_CONST_OBSOLETE_VERSION_STATE_PICK_VALUE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_CODE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_FEMININE "+"SYSRES_CONST_OPERATING_RECORD_FLAG_VALUE_MASCULINE "+"SYSRES_CONST_OPTIONAL_FORM_COMP_REQCODE_PREFIX "+"SYSRES_CONST_ORANGE_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_ORIGINALREF_REQUISITE_CODE "+"SYSRES_CONST_OURFIRM_REF_CODE "+"SYSRES_CONST_OURFIRM_REQUISITE_CODE "+"SYSRES_CONST_OURFIRM_VAR "+"SYSRES_CONST_OUTGOING_WORK_RULE_TYPE_CODE "+"SYSRES_CONST_PICK_NEGATIVE_RESULT "+"SYSRES_CONST_PICK_POSITIVE_RESULT "+"SYSRES_CONST_PICK_REQUISITE "+"SYSRES_CONST_PICK_REQUISITE_TYPE "+"SYSRES_CONST_PICK_TYPE_CHAR "+"SYSRES_CONST_PLAN_STATUS_REQUISITE_CODE "+"SYSRES_CONST_PLATFORM_VERSION_COMMENT "+"SYSRES_CONST_PLUGINS_SETTINGS_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_POSITIVE_PICK_VALUE "+"SYSRES_CONST_POWER_TO_CREATE_ACTION_CODE "+"SYSRES_CONST_POWER_TO_SIGN_ACTION_CODE "+"SYSRES_CONST_PRIORITY_REQUISITE_CODE "+"SYSRES_CONST_QUALIFIED_TASK_TYPE "+"SYSRES_CONST_QUALIFIED_TASK_TYPE_CODE "+"SYSRES_CONST_RECSTAT_REQUISITE_CODE "+"SYSRES_CONST_RED_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_REF_ID_T_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_REF_REQUISITE "+"SYSRES_CONST_REF_REQUISITE_TYPE "+"SYSRES_CONST_REF_REQUISITES_REFERENCE_CODE_SELECTED_REQUISITE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_CREATE_ACTION_CODE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_DELETE_ACTION_CODE "+"SYSRES_CONST_REFERENCE_RECORD_HISTORY_MODIFY_ACTION_CODE "+"SYSRES_CONST_REFERENCE_TYPE_CHAR "+"SYSRES_CONST_REFERENCE_TYPE_REQUISITE_NAME "+"SYSRES_CONST_REFERENCES_ADD_PARAMS_REQUISITE_CODE "+"SYSRES_CONST_REFERENCES_DISPLAY_REQUISITE_REQUISITE_CODE "+"SYSRES_CONST_REMOTE_SERVER_STATUS_WORKING "+"SYSRES_CONST_REMOTE_SERVER_TYPE_MAIN "+"SYSRES_CONST_REMOTE_SERVER_TYPE_SECONDARY "+"SYSRES_CONST_REMOTE_USER_FLAG_VALUE_CODE "+"SYSRES_CONST_REPORT_APP_EDITOR_INTERNAL "+"SYSRES_CONST_REPORT_BASE_REPORT_ID_REQUISITE_CODE "+"SYSRES_CONST_REPORT_BASE_REPORT_REQUISITE_CODE "+"SYSRES_CONST_REPORT_SCRIPT_REQUISITE_CODE "+"SYSRES_CONST_REPORT_TEMPLATE_REQUISITE_CODE "+"SYSRES_CONST_REPORT_VIEWER_CODE_REQUISITE_CODE "+"SYSRES_CONST_REQ_ALLOW_COMPONENT_DEFAULT_VALUE "+"SYSRES_CONST_REQ_ALLOW_RECORD_DEFAULT_VALUE "+"SYSRES_CONST_REQ_ALLOW_SERVER_COMPONENT_DEFAULT_VALUE "+"SYSRES_CONST_REQ_MODE_AVAILABLE_CODE "+"SYSRES_CONST_REQ_MODE_EDIT_CODE "+"SYSRES_CONST_REQ_MODE_HIDDEN_CODE "+"SYSRES_CONST_REQ_MODE_NOT_AVAILABLE_CODE "+"SYSRES_CONST_REQ_MODE_VIEW_CODE "+"SYSRES_CONST_REQ_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_REQ_SECTION_VALUE "+"SYSRES_CONST_REQ_TYPE_VALUE "+"SYSRES_CONST_REQUISITE_FORMAT_BY_UNIT "+"SYSRES_CONST_REQUISITE_FORMAT_DATE_FULL "+"SYSRES_CONST_REQUISITE_FORMAT_DATE_TIME "+"SYSRES_CONST_REQUISITE_FORMAT_LEFT "+"SYSRES_CONST_REQUISITE_FORMAT_RIGHT "+"SYSRES_CONST_REQUISITE_FORMAT_WITHOUT_UNIT "+"SYSRES_CONST_REQUISITE_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_REQUISITE_SECTION_ACTIONS "+"SYSRES_CONST_REQUISITE_SECTION_BUTTON "+"SYSRES_CONST_REQUISITE_SECTION_BUTTONS "+"SYSRES_CONST_REQUISITE_SECTION_CARD "+"SYSRES_CONST_REQUISITE_SECTION_TABLE "+"SYSRES_CONST_REQUISITE_SECTION_TABLE10 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE11 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE12 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE13 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE14 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE15 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE16 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE17 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE18 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE19 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE2 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE20 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE21 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE22 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE23 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE24 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE3 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE4 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE5 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE6 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE7 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE8 "+"SYSRES_CONST_REQUISITE_SECTION_TABLE9 "+"SYSRES_CONST_REQUISITES_PSEUDOREFERENCE_REQUISITE_NUMBER_REQUISITE_CODE "+"SYSRES_CONST_RIGHT_ALIGNMENT_CODE "+"SYSRES_CONST_ROLES_REFERENCE_CODE "+"SYSRES_CONST_ROUTE_STEP_AFTER_RUS "+"SYSRES_CONST_ROUTE_STEP_AND_CONDITION_RUS "+"SYSRES_CONST_ROUTE_STEP_OR_CONDITION_RUS "+"SYSRES_CONST_ROUTE_TYPE_COMPLEX "+"SYSRES_CONST_ROUTE_TYPE_PARALLEL "+"SYSRES_CONST_ROUTE_TYPE_SERIAL "+"SYSRES_CONST_SBDATASETDESC_NEGATIVE_VALUE "+"SYSRES_CONST_SBDATASETDESC_POSITIVE_VALUE "+"SYSRES_CONST_SBVIEWSDESC_POSITIVE_VALUE "+"SYSRES_CONST_SCRIPT_BLOCK_DESCRIPTION "+"SYSRES_CONST_SEARCH_BY_TEXT_REQUISITE_CODE "+"SYSRES_CONST_SEARCHES_COMPONENT_CONTENT "+"SYSRES_CONST_SEARCHES_CRITERIA_ACTION_NAME "+"SYSRES_CONST_SEARCHES_EDOC_CONTENT "+"SYSRES_CONST_SEARCHES_FOLDER_CONTENT "+"SYSRES_CONST_SEARCHES_JOB_CONTENT "+"SYSRES_CONST_SEARCHES_REFERENCE_CODE "+"SYSRES_CONST_SEARCHES_TASK_CONTENT "+"SYSRES_CONST_SECOND_CHAR "+"SYSRES_CONST_SECTION_REQUISITE_ACTIONS_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_CARD_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_CODE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_1_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_2_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_3_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_4_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_5_VALUE "+"SYSRES_CONST_SECTION_REQUISITE_DETAIL_6_VALUE "+"SYSRES_CONST_SELECT_REFERENCE_MODE_NAME "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE_ONLY_CHILD "+"SYSRES_CONST_SELECT_TYPE_SELECTABLE_WITH_CHILD "+"SYSRES_CONST_SELECT_TYPE_UNSLECTABLE "+"SYSRES_CONST_SERVER_TYPE_MAIN "+"SYSRES_CONST_SERVICE_USER_CATEGORY_FIELD_VALUE "+"SYSRES_CONST_SETTINGS_USER_REQUISITE_CODE "+"SYSRES_CONST_SIGNATURE_AND_ENCODE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_SIGNATURE_CERTIFICATE_TYPE_CODE "+"SYSRES_CONST_SINGULAR_TITLE_REQUISITE_CODE "+"SYSRES_CONST_SQL_SERVER_AUTHENTIFICATION_FLAG_VALUE_CODE "+"SYSRES_CONST_SQL_SERVER_ENCODE_AUTHENTIFICATION_FLAG_VALUE_CODE "+"SYSRES_CONST_STANDART_ROUTE_REFERENCE_CODE "+"SYSRES_CONST_STANDART_ROUTE_REFERENCE_COMMENT_REQUISITE_CODE "+"SYSRES_CONST_STANDART_ROUTES_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_STATE_REQ_NAME "+"SYSRES_CONST_STATE_REQUISITE_ACTIVE_VALUE "+"SYSRES_CONST_STATE_REQUISITE_CLOSED_VALUE "+"SYSRES_CONST_STATE_REQUISITE_CODE "+"SYSRES_CONST_STATIC_ROLE_TYPE_CODE "+"SYSRES_CONST_STATUS_PLAN_DEFAULT_VALUE "+"SYSRES_CONST_STATUS_VALUE_AUTOCLEANING "+"SYSRES_CONST_STATUS_VALUE_BLUE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_COMPLETE "+"SYSRES_CONST_STATUS_VALUE_GREEN_SQUARE "+"SYSRES_CONST_STATUS_VALUE_ORANGE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_PURPLE_SQUARE "+"SYSRES_CONST_STATUS_VALUE_RED_SQUARE "+"SYSRES_CONST_STATUS_VALUE_SUSPEND "+"SYSRES_CONST_STATUS_VALUE_YELLOW_SQUARE "+"SYSRES_CONST_STDROUTE_SHOW_TO_USERS_REQUISITE_CODE "+"SYSRES_CONST_STORAGE_TYPE_FILE "+"SYSRES_CONST_STORAGE_TYPE_SQL_SERVER "+"SYSRES_CONST_STR_REQUISITE "+"SYSRES_CONST_STRIKEOUT_LIFE_CYCLE_STAGE_DRAW_STYLE "+"SYSRES_CONST_STRING_FORMAT_LEFT_ALIGN_CHAR "+"SYSRES_CONST_STRING_FORMAT_RIGHT_ALIGN_CHAR "+"SYSRES_CONST_STRING_REQUISITE_CODE "+"SYSRES_CONST_STRING_REQUISITE_TYPE "+"SYSRES_CONST_STRING_TYPE_CHAR "+"SYSRES_CONST_SUBSTITUTES_PSEUDOREFERENCE_CODE "+"SYSRES_CONST_SUBTASK_BLOCK_DESCRIPTION "+"SYSRES_CONST_SYSTEM_SETTING_CURRENT_USER_PARAM_VALUE "+"SYSRES_CONST_SYSTEM_SETTING_EMPTY_VALUE_PARAM_VALUE "+"SYSRES_CONST_SYSTEM_VERSION_COMMENT "+"SYSRES_CONST_TASK_ACCESS_TYPE_ALL "+"SYSRES_CONST_TASK_ACCESS_TYPE_ALL_MEMBERS "+"SYSRES_CONST_TASK_ACCESS_TYPE_MANUAL "+"SYSRES_CONST_TASK_ENCODE_TYPE_CERTIFICATION "+"SYSRES_CONST_TASK_ENCODE_TYPE_CERTIFICATION_AND_PASSWORD "+"SYSRES_CONST_TASK_ENCODE_TYPE_NONE "+"SYSRES_CONST_TASK_ENCODE_TYPE_PASSWORD "+"SYSRES_CONST_TASK_ROUTE_ALL_CONDITION "+"SYSRES_CONST_TASK_ROUTE_AND_CONDITION "+"SYSRES_CONST_TASK_ROUTE_OR_CONDITION "+"SYSRES_CONST_TASK_STATE_ABORTED "+"SYSRES_CONST_TASK_STATE_COMPLETE "+"SYSRES_CONST_TASK_STATE_CONTINUED "+"SYSRES_CONST_TASK_STATE_CONTROL "+"SYSRES_CONST_TASK_STATE_INIT "+"SYSRES_CONST_TASK_STATE_WORKING "+"SYSRES_CONST_TASK_TITLE "+"SYSRES_CONST_TASK_TYPES_GROUPS_REFERENCE_CODE "+"SYSRES_CONST_TASK_TYPES_REFERENCE_CODE "+"SYSRES_CONST_TEMPLATES_REFERENCE_CODE "+"SYSRES_CONST_TEST_DATE_REQUISITE_NAME "+"SYSRES_CONST_TEST_DEV_DATABASE_NAME "+"SYSRES_CONST_TEST_DEV_SYSTEM_CODE "+"SYSRES_CONST_TEST_EDMS_DATABASE_NAME "+"SYSRES_CONST_TEST_EDMS_MAIN_CODE "+"SYSRES_CONST_TEST_EDMS_MAIN_DB_NAME "+"SYSRES_CONST_TEST_EDMS_SECOND_CODE "+"SYSRES_CONST_TEST_EDMS_SECOND_DB_NAME "+"SYSRES_CONST_TEST_EDMS_SYSTEM_CODE "+"SYSRES_CONST_TEST_NUMERIC_REQUISITE_NAME "+"SYSRES_CONST_TEXT_REQUISITE "+"SYSRES_CONST_TEXT_REQUISITE_CODE "+"SYSRES_CONST_TEXT_REQUISITE_TYPE "+"SYSRES_CONST_TEXT_TYPE_CHAR "+"SYSRES_CONST_TYPE_CODE_REQUISITE_CODE "+"SYSRES_CONST_TYPE_REQUISITE_CODE "+"SYSRES_CONST_UNDEFINED_LIFE_CYCLE_STAGE_FONT_COLOR "+"SYSRES_CONST_UNITS_SECTION_ID_REQUISITE_CODE "+"SYSRES_CONST_UNITS_SECTION_REQUISITE_CODE "+"SYSRES_CONST_UNOPERATING_RECORD_FLAG_VALUE_CODE "+"SYSRES_CONST_UNSTORED_DATA_REQUISITE_CODE "+"SYSRES_CONST_UNSTORED_DATA_REQUISITE_NAME "+"SYSRES_CONST_USE_ACCESS_TYPE_CODE "+"SYSRES_CONST_USE_ACCESS_TYPE_NAME "+"SYSRES_CONST_USER_ACCOUNT_TYPE_VALUE_CODE "+"SYSRES_CONST_USER_ADDITIONAL_INFORMATION_REQUISITE_CODE "+"SYSRES_CONST_USER_AND_GROUP_ID_FROM_PSEUDOREFERENCE_REQUISITE_CODE "+"SYSRES_CONST_USER_CATEGORY_NORMAL "+"SYSRES_CONST_USER_CERTIFICATE_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_STATE_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_SUBJECT_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_CERTIFICATE_THUMBPRINT_REQUISITE_CODE "+"SYSRES_CONST_USER_COMMON_CATEGORY "+"SYSRES_CONST_USER_COMMON_CATEGORY_CODE "+"SYSRES_CONST_USER_FULL_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_GROUP_TYPE_REQUISITE_CODE "+"SYSRES_CONST_USER_LOGIN_REQUISITE_CODE "+"SYSRES_CONST_USER_REMOTE_CONTROLLER_REQUISITE_CODE "+"SYSRES_CONST_USER_REMOTE_SYSTEM_REQUISITE_CODE "+"SYSRES_CONST_USER_RIGHTS_T_REQUISITE_CODE "+"SYSRES_CONST_USER_SERVER_NAME_REQUISITE_CODE "+"SYSRES_CONST_USER_SERVICE_CATEGORY "+"SYSRES_CONST_USER_SERVICE_CATEGORY_CODE "+"SYSRES_CONST_USER_STATUS_ADMINISTRATOR_CODE "+"SYSRES_CONST_USER_STATUS_ADMINISTRATOR_NAME "+"SYSRES_CONST_USER_STATUS_DEVELOPER_CODE "+"SYSRES_CONST_USER_STATUS_DEVELOPER_NAME "+"SYSRES_CONST_USER_STATUS_DISABLED_CODE "+"SYSRES_CONST_USER_STATUS_DISABLED_NAME "+"SYSRES_CONST_USER_STATUS_SYSTEM_DEVELOPER_CODE "+"SYSRES_CONST_USER_STATUS_USER_CODE "+"SYSRES_CONST_USER_STATUS_USER_NAME "+"SYSRES_CONST_USER_STATUS_USER_NAME_DEPRECATED "+"SYSRES_CONST_USER_TYPE_FIELD_VALUE_USER "+"SYSRES_CONST_USER_TYPE_REQUISITE_CODE "+"SYSRES_CONST_USERS_CONTROLLER_REQUISITE_CODE "+"SYSRES_CONST_USERS_IS_MAIN_SERVER_REQUISITE_CODE "+"SYSRES_CONST_USERS_REFERENCE_CODE "+"SYSRES_CONST_USERS_REGISTRATION_CERTIFICATES_ACTION_NAME "+"SYSRES_CONST_USERS_REQUISITE_CODE "+"SYSRES_CONST_USERS_SYSTEM_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_ACCESS_RIGHTS_TYPR_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_AUTHENTICATION_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_COMPONENT_REQUISITE_CODE "+"SYSRES_CONST_USERS_USER_GROUP_REQUISITE_CODE "+"SYSRES_CONST_USERS_VIEW_CERTIFICATES_ACTION_NAME "+"SYSRES_CONST_VIEW_DEFAULT_CODE "+"SYSRES_CONST_VIEW_DEFAULT_NAME "+"SYSRES_CONST_VIEWER_REQUISITE_CODE "+"SYSRES_CONST_WAITING_BLOCK_DESCRIPTION "+"SYSRES_CONST_WIZARD_FORM_LABEL_TEST_STRING  "+"SYSRES_CONST_WIZARD_QUERY_PARAM_HEIGHT_ETALON_STRING "+"SYSRES_CONST_WIZARD_REFERENCE_COMMENT_REQUISITE_CODE "+"SYSRES_CONST_WORK_RULES_DESCRIPTION_REQUISITE_CODE "+"SYSRES_CONST_WORK_TIME_CALENDAR_REFERENCE_CODE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE_CODE "+"SYSRES_CONST_WORK_WORKFLOW_HARD_ROUTE_TYPE_VALUE_CODE_RUS "+"SYSRES_CONST_WORK_WORKFLOW_SOFT_ROUTE_TYPE_VALUE_CODE_RUS "+"SYSRES_CONST_WORKFLOW_ROUTE_TYPR_HARD "+"SYSRES_CONST_WORKFLOW_ROUTE_TYPR_SOFT "+"SYSRES_CONST_XML_ENCODING "+"SYSRES_CONST_XREC_STAT_REQUISITE_CODE "+"SYSRES_CONST_XRECID_FIELD_NAME "+"SYSRES_CONST_YES "+"SYSRES_CONST_YES_NO_2_REQUISITE_CODE "+"SYSRES_CONST_YES_NO_REQUISITE_CODE "+"SYSRES_CONST_YES_NO_T_REF_TYPE_REQUISITE_CODE "+"SYSRES_CONST_YES_PICK_VALUE "+"SYSRES_CONST_YES_VALUE ";var base_constants="CR FALSE nil NO_VALUE NULL TAB TRUE YES_VALUE ";var base_group_name_constants="ADMINISTRATORS_GROUP_NAME CUSTOMIZERS_GROUP_NAME DEVELOPERS_GROUP_NAME SERVICE_USERS_GROUP_NAME ";var decision_block_properties_constants="DECISION_BLOCK_FIRST_OPERAND_PROPERTY DECISION_BLOCK_NAME_PROPERTY DECISION_BLOCK_OPERATION_PROPERTY "+"DECISION_BLOCK_RESULT_TYPE_PROPERTY DECISION_BLOCK_SECOND_OPERAND_PROPERTY ";var file_extension_constants="ANY_FILE_EXTENTION COMPRESSED_DOCUMENT_EXTENSION EXTENDED_DOCUMENT_EXTENSION "+"SHORT_COMPRESSED_DOCUMENT_EXTENSION SHORT_EXTENDED_DOCUMENT_EXTENSION ";var job_block_properties_constants="JOB_BLOCK_ABORT_DEADLINE_PROPERTY "+"JOB_BLOCK_AFTER_FINISH_EVENT "+"JOB_BLOCK_AFTER_QUERY_PARAMETERS_EVENT "+"JOB_BLOCK_ATTACHMENT_PROPERTY "+"JOB_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"JOB_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"JOB_BLOCK_BEFORE_QUERY_PARAMETERS_EVENT "+"JOB_BLOCK_BEFORE_START_EVENT "+"JOB_BLOCK_CREATED_JOBS_PROPERTY "+"JOB_BLOCK_DEADLINE_PROPERTY "+"JOB_BLOCK_EXECUTION_RESULTS_PROPERTY "+"JOB_BLOCK_IS_PARALLEL_PROPERTY "+"JOB_BLOCK_IS_RELATIVE_ABORT_DEADLINE_PROPERTY "+"JOB_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"JOB_BLOCK_JOB_TEXT_PROPERTY "+"JOB_BLOCK_NAME_PROPERTY "+"JOB_BLOCK_NEED_SIGN_ON_PERFORM_PROPERTY "+"JOB_BLOCK_PERFORMER_PROPERTY "+"JOB_BLOCK_RELATIVE_ABORT_DEADLINE_TYPE_PROPERTY "+"JOB_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"JOB_BLOCK_SUBJECT_PROPERTY ";var language_code_constants="ENGLISH_LANGUAGE_CODE RUSSIAN_LANGUAGE_CODE ";var launching_external_applications_constants="smHidden smMaximized smMinimized smNormal wmNo wmYes ";var link_kind_constants="COMPONENT_TOKEN_LINK_KIND "+"DOCUMENT_LINK_KIND "+"EDOCUMENT_LINK_KIND "+"FOLDER_LINK_KIND "+"JOB_LINK_KIND "+"REFERENCE_LINK_KIND "+"TASK_LINK_KIND ";var lock_type_constants="COMPONENT_TOKEN_LOCK_TYPE EDOCUMENT_VERSION_LOCK_TYPE ";var monitor_block_properties_constants="MONITOR_BLOCK_AFTER_FINISH_EVENT "+"MONITOR_BLOCK_BEFORE_START_EVENT "+"MONITOR_BLOCK_DEADLINE_PROPERTY "+"MONITOR_BLOCK_INTERVAL_PROPERTY "+"MONITOR_BLOCK_INTERVAL_TYPE_PROPERTY "+"MONITOR_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"MONITOR_BLOCK_NAME_PROPERTY "+"MONITOR_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"MONITOR_BLOCK_SEARCH_SCRIPT_PROPERTY ";var notice_block_properties_constants="NOTICE_BLOCK_AFTER_FINISH_EVENT "+"NOTICE_BLOCK_ATTACHMENT_PROPERTY "+"NOTICE_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"NOTICE_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"NOTICE_BLOCK_BEFORE_START_EVENT "+"NOTICE_BLOCK_CREATED_NOTICES_PROPERTY "+"NOTICE_BLOCK_DEADLINE_PROPERTY "+"NOTICE_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"NOTICE_BLOCK_NAME_PROPERTY "+"NOTICE_BLOCK_NOTICE_TEXT_PROPERTY "+"NOTICE_BLOCK_PERFORMER_PROPERTY "+"NOTICE_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"NOTICE_BLOCK_SUBJECT_PROPERTY ";var object_events_constants="dseAfterCancel "+"dseAfterClose "+"dseAfterDelete "+"dseAfterDeleteOutOfTransaction "+"dseAfterInsert "+"dseAfterOpen "+"dseAfterScroll "+"dseAfterUpdate "+"dseAfterUpdateOutOfTransaction "+"dseBeforeCancel "+"dseBeforeClose "+"dseBeforeDelete "+"dseBeforeDetailUpdate "+"dseBeforeInsert "+"dseBeforeOpen "+"dseBeforeUpdate "+"dseOnAnyRequisiteChange "+"dseOnCloseRecord "+"dseOnDeleteError "+"dseOnOpenRecord "+"dseOnPrepareUpdate "+"dseOnUpdateError "+"dseOnUpdateRatifiedRecord "+"dseOnValidDelete "+"dseOnValidUpdate "+"reOnChange "+"reOnChangeValues "+"SELECTION_BEGIN_ROUTE_EVENT "+"SELECTION_END_ROUTE_EVENT ";var object_params_constants="CURRENT_PERIOD_IS_REQUIRED "+"PREVIOUS_CARD_TYPE_NAME "+"SHOW_RECORD_PROPERTIES_FORM ";var other_constants="ACCESS_RIGHTS_SETTING_DIALOG_CODE "+"ADMINISTRATOR_USER_CODE "+"ANALYTIC_REPORT_TYPE "+"asrtHideLocal "+"asrtHideRemote "+"CALCULATED_ROLE_TYPE_CODE "+"COMPONENTS_REFERENCE_DEVELOPER_VIEW_CODE "+"DCTS_TEST_PROTOCOLS_FOLDER_PATH "+"E_EDOC_VERSION_ALREADY_APPROVINGLY_SIGNED "+"E_EDOC_VERSION_ALREADY_APPROVINGLY_SIGNED_BY_USER "+"E_EDOC_VERSION_ALREDY_SIGNED "+"E_EDOC_VERSION_ALREDY_SIGNED_BY_USER "+"EDOC_TYPES_CODE_REQUISITE_FIELD_NAME "+"EDOCUMENTS_ALIAS_NAME "+"FILES_FOLDER_PATH "+"FILTER_OPERANDS_DELIMITER "+"FILTER_OPERATIONS_DELIMITER "+"FORMCARD_NAME "+"FORMLIST_NAME "+"GET_EXTENDED_DOCUMENT_EXTENSION_CREATION_MODE "+"GET_EXTENDED_DOCUMENT_EXTENSION_IMPORT_MODE "+"INTEGRATED_REPORT_TYPE "+"IS_BUILDER_APPLICATION_ROLE "+"IS_BUILDER_APPLICATION_ROLE2 "+"IS_BUILDER_USERS "+"ISBSYSDEV "+"LOG_FOLDER_PATH "+"mbCancel "+"mbNo "+"mbNoToAll "+"mbOK "+"mbYes "+"mbYesToAll "+"MEMORY_DATASET_DESRIPTIONS_FILENAME "+"mrNo "+"mrNoToAll "+"mrYes "+"mrYesToAll "+"MULTIPLE_SELECT_DIALOG_CODE "+"NONOPERATING_RECORD_FLAG_FEMININE "+"NONOPERATING_RECORD_FLAG_MASCULINE "+"OPERATING_RECORD_FLAG_FEMININE "+"OPERATING_RECORD_FLAG_MASCULINE "+"PROFILING_SETTINGS_COMMON_SETTINGS_CODE_VALUE "+"PROGRAM_INITIATED_LOOKUP_ACTION "+"ratDelete "+"ratEdit "+"ratInsert "+"REPORT_TYPE "+"REQUIRED_PICK_VALUES_VARIABLE "+"rmCard "+"rmList "+"SBRTE_PROGID_DEV "+"SBRTE_PROGID_RELEASE "+"STATIC_ROLE_TYPE_CODE "+"SUPPRESS_EMPTY_TEMPLATE_CREATION "+"SYSTEM_USER_CODE "+"UPDATE_DIALOG_DATASET "+"USED_IN_OBJECT_HINT_PARAM "+"USER_INITIATED_LOOKUP_ACTION "+"USER_NAME_FORMAT "+"USER_SELECTION_RESTRICTIONS "+"WORKFLOW_TEST_PROTOCOLS_FOLDER_PATH "+"ELS_SUBTYPE_CONTROL_NAME "+"ELS_FOLDER_KIND_CONTROL_NAME "+"REPEAT_PROCESS_CURRENT_OBJECT_EXCEPTION_NAME ";var privileges_constants="PRIVILEGE_COMPONENT_FULL_ACCESS "+"PRIVILEGE_DEVELOPMENT_EXPORT "+"PRIVILEGE_DEVELOPMENT_IMPORT "+"PRIVILEGE_DOCUMENT_DELETE "+"PRIVILEGE_ESD "+"PRIVILEGE_FOLDER_DELETE "+"PRIVILEGE_MANAGE_ACCESS_RIGHTS "+"PRIVILEGE_MANAGE_REPLICATION "+"PRIVILEGE_MANAGE_SESSION_SERVER "+"PRIVILEGE_OBJECT_FULL_ACCESS "+"PRIVILEGE_OBJECT_VIEW "+"PRIVILEGE_RESERVE_LICENSE "+"PRIVILEGE_SYSTEM_CUSTOMIZE "+"PRIVILEGE_SYSTEM_DEVELOP "+"PRIVILEGE_SYSTEM_INSTALL "+"PRIVILEGE_TASK_DELETE "+"PRIVILEGE_USER_PLUGIN_SETTINGS_CUSTOMIZE "+"PRIVILEGES_PSEUDOREFERENCE_CODE ";var pseudoreference_code_constants="ACCESS_TYPES_PSEUDOREFERENCE_CODE "+"ALL_AVAILABLE_COMPONENTS_PSEUDOREFERENCE_CODE "+"ALL_AVAILABLE_PRIVILEGES_PSEUDOREFERENCE_CODE "+"ALL_REPLICATE_COMPONENTS_PSEUDOREFERENCE_CODE "+"AVAILABLE_DEVELOPERS_COMPONENTS_PSEUDOREFERENCE_CODE "+"COMPONENTS_PSEUDOREFERENCE_CODE "+"FILTRATER_SETTINGS_CONFLICTS_PSEUDOREFERENCE_CODE "+"GROUPS_PSEUDOREFERENCE_CODE "+"RECEIVE_PROTOCOL_PSEUDOREFERENCE_CODE "+"REFERENCE_REQUISITE_PSEUDOREFERENCE_CODE "+"REFERENCE_REQUISITES_PSEUDOREFERENCE_CODE "+"REFTYPES_PSEUDOREFERENCE_CODE "+"REPLICATION_SEANCES_DIARY_PSEUDOREFERENCE_CODE "+"SEND_PROTOCOL_PSEUDOREFERENCE_CODE "+"SUBSTITUTES_PSEUDOREFERENCE_CODE "+"SYSTEM_SETTINGS_PSEUDOREFERENCE_CODE "+"UNITS_PSEUDOREFERENCE_CODE "+"USERS_PSEUDOREFERENCE_CODE "+"VIEWERS_PSEUDOREFERENCE_CODE ";var requisite_ISBCertificateType_values_constants="CERTIFICATE_TYPE_ENCRYPT "+"CERTIFICATE_TYPE_SIGN "+"CERTIFICATE_TYPE_SIGN_AND_ENCRYPT ";var requisite_ISBEDocStorageType_values_constants="STORAGE_TYPE_FILE "+"STORAGE_TYPE_NAS_CIFS "+"STORAGE_TYPE_SAPERION "+"STORAGE_TYPE_SQL_SERVER ";var requisite_compType2_values_constants="COMPTYPE2_REQUISITE_DOCUMENTS_VALUE "+"COMPTYPE2_REQUISITE_TASKS_VALUE "+"COMPTYPE2_REQUISITE_FOLDERS_VALUE "+"COMPTYPE2_REQUISITE_REFERENCES_VALUE ";var requisite_name_constants="SYSREQ_CODE "+"SYSREQ_COMPTYPE2 "+"SYSREQ_CONST_AVAILABLE_FOR_WEB "+"SYSREQ_CONST_COMMON_CODE "+"SYSREQ_CONST_COMMON_VALUE "+"SYSREQ_CONST_FIRM_CODE "+"SYSREQ_CONST_FIRM_STATUS "+"SYSREQ_CONST_FIRM_VALUE "+"SYSREQ_CONST_SERVER_STATUS "+"SYSREQ_CONTENTS "+"SYSREQ_DATE_OPEN "+"SYSREQ_DATE_CLOSE "+"SYSREQ_DESCRIPTION "+"SYSREQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_DOUBLE "+"SYSREQ_EDOC_ACCESS_TYPE "+"SYSREQ_EDOC_AUTHOR "+"SYSREQ_EDOC_CREATED "+"SYSREQ_EDOC_DELEGATE_RIGHTS_REQUISITE_CODE "+"SYSREQ_EDOC_EDITOR "+"SYSREQ_EDOC_ENCODE_TYPE "+"SYSREQ_EDOC_ENCRYPTION_PLUGIN_NAME "+"SYSREQ_EDOC_ENCRYPTION_PLUGIN_VERSION "+"SYSREQ_EDOC_EXPORT_DATE "+"SYSREQ_EDOC_EXPORTER "+"SYSREQ_EDOC_KIND "+"SYSREQ_EDOC_LIFE_STAGE_NAME "+"SYSREQ_EDOC_LOCKED_FOR_SERVER_CODE "+"SYSREQ_EDOC_MODIFIED "+"SYSREQ_EDOC_NAME "+"SYSREQ_EDOC_NOTE "+"SYSREQ_EDOC_QUALIFIED_ID "+"SYSREQ_EDOC_SESSION_KEY "+"SYSREQ_EDOC_SESSION_KEY_ENCRYPTION_PLUGIN_NAME "+"SYSREQ_EDOC_SESSION_KEY_ENCRYPTION_PLUGIN_VERSION "+"SYSREQ_EDOC_SIGNATURE_TYPE "+"SYSREQ_EDOC_SIGNED "+"SYSREQ_EDOC_STORAGE "+"SYSREQ_EDOC_STORAGES_ARCHIVE_STORAGE "+"SYSREQ_EDOC_STORAGES_CHECK_RIGHTS "+"SYSREQ_EDOC_STORAGES_COMPUTER_NAME "+"SYSREQ_EDOC_STORAGES_EDIT_IN_STORAGE "+"SYSREQ_EDOC_STORAGES_EXECUTIVE_STORAGE "+"SYSREQ_EDOC_STORAGES_FUNCTION "+"SYSREQ_EDOC_STORAGES_INITIALIZED "+"SYSREQ_EDOC_STORAGES_LOCAL_PATH "+"SYSREQ_EDOC_STORAGES_SAPERION_DATABASE_NAME "+"SYSREQ_EDOC_STORAGES_SEARCH_BY_TEXT "+"SYSREQ_EDOC_STORAGES_SERVER_NAME "+"SYSREQ_EDOC_STORAGES_SHARED_SOURCE_NAME "+"SYSREQ_EDOC_STORAGES_TYPE "+"SYSREQ_EDOC_TEXT_MODIFIED "+"SYSREQ_EDOC_TYPE_ACT_CODE "+"SYSREQ_EDOC_TYPE_ACT_DESCRIPTION "+"SYSREQ_EDOC_TYPE_ACT_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_ACT_ON_EXECUTE "+"SYSREQ_EDOC_TYPE_ACT_ON_EXECUTE_EXISTS "+"SYSREQ_EDOC_TYPE_ACT_SECTION "+"SYSREQ_EDOC_TYPE_ADD_PARAMS "+"SYSREQ_EDOC_TYPE_COMMENT "+"SYSREQ_EDOC_TYPE_EVENT_TEXT "+"SYSREQ_EDOC_TYPE_NAME_IN_SINGULAR "+"SYSREQ_EDOC_TYPE_NAME_IN_SINGULAR_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_NAME_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_NUMERATION_METHOD "+"SYSREQ_EDOC_TYPE_PSEUDO_REQUISITE_CODE "+"SYSREQ_EDOC_TYPE_REQ_CODE "+"SYSREQ_EDOC_TYPE_REQ_DESCRIPTION "+"SYSREQ_EDOC_TYPE_REQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_EDOC_TYPE_REQ_IS_LEADING "+"SYSREQ_EDOC_TYPE_REQ_IS_REQUIRED "+"SYSREQ_EDOC_TYPE_REQ_NUMBER "+"SYSREQ_EDOC_TYPE_REQ_ON_CHANGE "+"SYSREQ_EDOC_TYPE_REQ_ON_CHANGE_EXISTS "+"SYSREQ_EDOC_TYPE_REQ_ON_SELECT "+"SYSREQ_EDOC_TYPE_REQ_ON_SELECT_KIND "+"SYSREQ_EDOC_TYPE_REQ_SECTION "+"SYSREQ_EDOC_TYPE_VIEW_CARD "+"SYSREQ_EDOC_TYPE_VIEW_CODE "+"SYSREQ_EDOC_TYPE_VIEW_COMMENT "+"SYSREQ_EDOC_TYPE_VIEW_IS_MAIN "+"SYSREQ_EDOC_TYPE_VIEW_NAME "+"SYSREQ_EDOC_TYPE_VIEW_NAME_LOCALIZE_ID "+"SYSREQ_EDOC_VERSION_AUTHOR "+"SYSREQ_EDOC_VERSION_CRC "+"SYSREQ_EDOC_VERSION_DATA "+"SYSREQ_EDOC_VERSION_EDITOR "+"SYSREQ_EDOC_VERSION_EXPORT_DATE "+"SYSREQ_EDOC_VERSION_EXPORTER "+"SYSREQ_EDOC_VERSION_HIDDEN "+"SYSREQ_EDOC_VERSION_LIFE_STAGE "+"SYSREQ_EDOC_VERSION_MODIFIED "+"SYSREQ_EDOC_VERSION_NOTE "+"SYSREQ_EDOC_VERSION_SIGNATURE_TYPE "+"SYSREQ_EDOC_VERSION_SIGNED "+"SYSREQ_EDOC_VERSION_SIZE "+"SYSREQ_EDOC_VERSION_SOURCE "+"SYSREQ_EDOC_VERSION_TEXT_MODIFIED "+"SYSREQ_EDOCKIND_DEFAULT_VERSION_STATE_CODE "+"SYSREQ_FOLDER_KIND "+"SYSREQ_FUNC_CATEGORY "+"SYSREQ_FUNC_COMMENT "+"SYSREQ_FUNC_GROUP "+"SYSREQ_FUNC_GROUP_COMMENT "+"SYSREQ_FUNC_GROUP_NUMBER "+"SYSREQ_FUNC_HELP "+"SYSREQ_FUNC_PARAM_DEF_VALUE "+"SYSREQ_FUNC_PARAM_IDENT "+"SYSREQ_FUNC_PARAM_NUMBER "+"SYSREQ_FUNC_PARAM_TYPE "+"SYSREQ_FUNC_TEXT "+"SYSREQ_GROUP_CATEGORY "+"SYSREQ_ID "+"SYSREQ_LAST_UPDATE "+"SYSREQ_LEADER_REFERENCE "+"SYSREQ_LINE_NUMBER "+"SYSREQ_MAIN_RECORD_ID "+"SYSREQ_NAME "+"SYSREQ_NAME_LOCALIZE_ID "+"SYSREQ_NOTE "+"SYSREQ_ORIGINAL_RECORD "+"SYSREQ_OUR_FIRM "+"SYSREQ_PROFILING_SETTINGS_BATCH_LOGING "+"SYSREQ_PROFILING_SETTINGS_BATCH_SIZE "+"SYSREQ_PROFILING_SETTINGS_PROFILING_ENABLED "+"SYSREQ_PROFILING_SETTINGS_SQL_PROFILING_ENABLED "+"SYSREQ_PROFILING_SETTINGS_START_LOGGED "+"SYSREQ_RECORD_STATUS "+"SYSREQ_REF_REQ_FIELD_NAME "+"SYSREQ_REF_REQ_FORMAT "+"SYSREQ_REF_REQ_GENERATED "+"SYSREQ_REF_REQ_LENGTH "+"SYSREQ_REF_REQ_PRECISION "+"SYSREQ_REF_REQ_REFERENCE "+"SYSREQ_REF_REQ_SECTION "+"SYSREQ_REF_REQ_STORED "+"SYSREQ_REF_REQ_TOKENS "+"SYSREQ_REF_REQ_TYPE "+"SYSREQ_REF_REQ_VIEW "+"SYSREQ_REF_TYPE_ACT_CODE "+"SYSREQ_REF_TYPE_ACT_DESCRIPTION "+"SYSREQ_REF_TYPE_ACT_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_REF_TYPE_ACT_ON_EXECUTE "+"SYSREQ_REF_TYPE_ACT_ON_EXECUTE_EXISTS "+"SYSREQ_REF_TYPE_ACT_SECTION "+"SYSREQ_REF_TYPE_ADD_PARAMS "+"SYSREQ_REF_TYPE_COMMENT "+"SYSREQ_REF_TYPE_COMMON_SETTINGS "+"SYSREQ_REF_TYPE_DISPLAY_REQUISITE_NAME "+"SYSREQ_REF_TYPE_EVENT_TEXT "+"SYSREQ_REF_TYPE_MAIN_LEADING_REF "+"SYSREQ_REF_TYPE_NAME_IN_SINGULAR "+"SYSREQ_REF_TYPE_NAME_IN_SINGULAR_LOCALIZE_ID "+"SYSREQ_REF_TYPE_NAME_LOCALIZE_ID "+"SYSREQ_REF_TYPE_NUMERATION_METHOD "+"SYSREQ_REF_TYPE_REQ_CODE "+"SYSREQ_REF_TYPE_REQ_DESCRIPTION "+"SYSREQ_REF_TYPE_REQ_DESCRIPTION_LOCALIZE_ID "+"SYSREQ_REF_TYPE_REQ_IS_CONTROL "+"SYSREQ_REF_TYPE_REQ_IS_FILTER "+"SYSREQ_REF_TYPE_REQ_IS_LEADING "+"SYSREQ_REF_TYPE_REQ_IS_REQUIRED "+"SYSREQ_REF_TYPE_REQ_NUMBER "+"SYSREQ_REF_TYPE_REQ_ON_CHANGE "+"SYSREQ_REF_TYPE_REQ_ON_CHANGE_EXISTS "+"SYSREQ_REF_TYPE_REQ_ON_SELECT "+"SYSREQ_REF_TYPE_REQ_ON_SELECT_KIND "+"SYSREQ_REF_TYPE_REQ_SECTION "+"SYSREQ_REF_TYPE_VIEW_CARD "+"SYSREQ_REF_TYPE_VIEW_CODE "+"SYSREQ_REF_TYPE_VIEW_COMMENT "+"SYSREQ_REF_TYPE_VIEW_IS_MAIN "+"SYSREQ_REF_TYPE_VIEW_NAME "+"SYSREQ_REF_TYPE_VIEW_NAME_LOCALIZE_ID "+"SYSREQ_REFERENCE_TYPE_ID "+"SYSREQ_STATE "+"SYSREQ_STATЕ "+"SYSREQ_SYSTEM_SETTINGS_VALUE "+"SYSREQ_TYPE "+"SYSREQ_UNIT "+"SYSREQ_UNIT_ID "+"SYSREQ_USER_GROUPS_GROUP_FULL_NAME "+"SYSREQ_USER_GROUPS_GROUP_NAME "+"SYSREQ_USER_GROUPS_GROUP_SERVER_NAME "+"SYSREQ_USERS_ACCESS_RIGHTS "+"SYSREQ_USERS_AUTHENTICATION "+"SYSREQ_USERS_CATEGORY "+"SYSREQ_USERS_COMPONENT "+"SYSREQ_USERS_COMPONENT_USER_IS_PUBLIC "+"SYSREQ_USERS_DOMAIN "+"SYSREQ_USERS_FULL_USER_NAME "+"SYSREQ_USERS_GROUP "+"SYSREQ_USERS_IS_MAIN_SERVER "+"SYSREQ_USERS_LOGIN "+"SYSREQ_USERS_REFERENCE_USER_IS_PUBLIC "+"SYSREQ_USERS_STATUS "+"SYSREQ_USERS_USER_CERTIFICATE "+"SYSREQ_USERS_USER_CERTIFICATE_INFO "+"SYSREQ_USERS_USER_CERTIFICATE_PLUGIN_NAME "+"SYSREQ_USERS_USER_CERTIFICATE_PLUGIN_VERSION "+"SYSREQ_USERS_USER_CERTIFICATE_STATE "+"SYSREQ_USERS_USER_CERTIFICATE_SUBJECT_NAME "+"SYSREQ_USERS_USER_CERTIFICATE_THUMBPRINT "+"SYSREQ_USERS_USER_DEFAULT_CERTIFICATE "+"SYSREQ_USERS_USER_DESCRIPTION "+"SYSREQ_USERS_USER_GLOBAL_NAME "+"SYSREQ_USERS_USER_LOGIN "+"SYSREQ_USERS_USER_MAIN_SERVER "+"SYSREQ_USERS_USER_TYPE "+"SYSREQ_WORK_RULES_FOLDER_ID ";var result_constants="RESULT_VAR_NAME RESULT_VAR_NAME_ENG ";var rule_identification_constants="AUTO_NUMERATION_RULE_ID "+"CANT_CHANGE_ID_REQUISITE_RULE_ID "+"CANT_CHANGE_OURFIRM_REQUISITE_RULE_ID "+"CHECK_CHANGING_REFERENCE_RECORD_USE_RULE_ID "+"CHECK_CODE_REQUISITE_RULE_ID "+"CHECK_DELETING_REFERENCE_RECORD_USE_RULE_ID "+"CHECK_FILTRATER_CHANGES_RULE_ID "+"CHECK_RECORD_INTERVAL_RULE_ID "+"CHECK_REFERENCE_INTERVAL_RULE_ID "+"CHECK_REQUIRED_DATA_FULLNESS_RULE_ID "+"CHECK_REQUIRED_REQUISITES_FULLNESS_RULE_ID "+"MAKE_RECORD_UNRATIFIED_RULE_ID "+"RESTORE_AUTO_NUMERATION_RULE_ID "+"SET_FIRM_CONTEXT_FROM_RECORD_RULE_ID "+"SET_FIRST_RECORD_IN_LIST_FORM_RULE_ID "+"SET_IDSPS_VALUE_RULE_ID "+"SET_NEXT_CODE_VALUE_RULE_ID "+"SET_OURFIRM_BOUNDS_RULE_ID "+"SET_OURFIRM_REQUISITE_RULE_ID ";var script_block_properties_constants="SCRIPT_BLOCK_AFTER_FINISH_EVENT "+"SCRIPT_BLOCK_BEFORE_START_EVENT "+"SCRIPT_BLOCK_EXECUTION_RESULTS_PROPERTY "+"SCRIPT_BLOCK_NAME_PROPERTY "+"SCRIPT_BLOCK_SCRIPT_PROPERTY ";var subtask_block_properties_constants="SUBTASK_BLOCK_ABORT_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_AFTER_FINISH_EVENT "+"SUBTASK_BLOCK_ASSIGN_PARAMS_EVENT "+"SUBTASK_BLOCK_ATTACHMENTS_PROPERTY "+"SUBTASK_BLOCK_ATTACHMENTS_RIGHTS_GROUP_PROPERTY "+"SUBTASK_BLOCK_ATTACHMENTS_RIGHTS_TYPE_PROPERTY "+"SUBTASK_BLOCK_BEFORE_START_EVENT "+"SUBTASK_BLOCK_CREATED_TASK_PROPERTY "+"SUBTASK_BLOCK_CREATION_EVENT "+"SUBTASK_BLOCK_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_IMPORTANCE_PROPERTY "+"SUBTASK_BLOCK_INITIATOR_PROPERTY "+"SUBTASK_BLOCK_IS_RELATIVE_ABORT_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"SUBTASK_BLOCK_JOBS_TYPE_PROPERTY "+"SUBTASK_BLOCK_NAME_PROPERTY "+"SUBTASK_BLOCK_PARALLEL_ROUTE_PROPERTY "+"SUBTASK_BLOCK_PERFORMERS_PROPERTY "+"SUBTASK_BLOCK_RELATIVE_ABORT_DEADLINE_TYPE_PROPERTY "+"SUBTASK_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY "+"SUBTASK_BLOCK_REQUIRE_SIGN_PROPERTY "+"SUBTASK_BLOCK_STANDARD_ROUTE_PROPERTY "+"SUBTASK_BLOCK_START_EVENT "+"SUBTASK_BLOCK_STEP_CONTROL_PROPERTY "+"SUBTASK_BLOCK_SUBJECT_PROPERTY "+"SUBTASK_BLOCK_TASK_CONTROL_PROPERTY "+"SUBTASK_BLOCK_TEXT_PROPERTY "+"SUBTASK_BLOCK_UNLOCK_ATTACHMENTS_ON_STOP_PROPERTY "+"SUBTASK_BLOCK_USE_STANDARD_ROUTE_PROPERTY "+"SUBTASK_BLOCK_WAIT_FOR_TASK_COMPLETE_PROPERTY ";var system_component_constants="SYSCOMP_CONTROL_JOBS "+"SYSCOMP_FOLDERS "+"SYSCOMP_JOBS "+"SYSCOMP_NOTICES "+"SYSCOMP_TASKS ";var system_dialogs_constants="SYSDLG_CREATE_EDOCUMENT "+"SYSDLG_CREATE_EDOCUMENT_VERSION "+"SYSDLG_CURRENT_PERIOD "+"SYSDLG_EDIT_FUNCTION_HELP "+"SYSDLG_EDOCUMENT_KINDS_FOR_TEMPLATE "+"SYSDLG_EXPORT_MULTIPLE_EDOCUMENTS "+"SYSDLG_EXPORT_SINGLE_EDOCUMENT "+"SYSDLG_IMPORT_EDOCUMENT "+"SYSDLG_MULTIPLE_SELECT "+"SYSDLG_SETUP_ACCESS_RIGHTS "+"SYSDLG_SETUP_DEFAULT_RIGHTS "+"SYSDLG_SETUP_FILTER_CONDITION "+"SYSDLG_SETUP_SIGN_RIGHTS "+"SYSDLG_SETUP_TASK_OBSERVERS "+"SYSDLG_SETUP_TASK_ROUTE "+"SYSDLG_SETUP_USERS_LIST "+"SYSDLG_SIGN_EDOCUMENT "+"SYSDLG_SIGN_MULTIPLE_EDOCUMENTS ";var system_reference_names_constants="SYSREF_ACCESS_RIGHTS_TYPES "+"SYSREF_ADMINISTRATION_HISTORY "+"SYSREF_ALL_AVAILABLE_COMPONENTS "+"SYSREF_ALL_AVAILABLE_PRIVILEGES "+"SYSREF_ALL_REPLICATING_COMPONENTS "+"SYSREF_AVAILABLE_DEVELOPERS_COMPONENTS "+"SYSREF_CALENDAR_EVENTS "+"SYSREF_COMPONENT_TOKEN_HISTORY "+"SYSREF_COMPONENT_TOKENS "+"SYSREF_COMPONENTS "+"SYSREF_CONSTANTS "+"SYSREF_DATA_RECEIVE_PROTOCOL "+"SYSREF_DATA_SEND_PROTOCOL "+"SYSREF_DIALOGS "+"SYSREF_DIALOGS_REQUISITES "+"SYSREF_EDITORS "+"SYSREF_EDOC_CARDS "+"SYSREF_EDOC_TYPES "+"SYSREF_EDOCUMENT_CARD_REQUISITES "+"SYSREF_EDOCUMENT_CARD_TYPES "+"SYSREF_EDOCUMENT_CARD_TYPES_REFERENCE "+"SYSREF_EDOCUMENT_CARDS "+"SYSREF_EDOCUMENT_HISTORY "+"SYSREF_EDOCUMENT_KINDS "+"SYSREF_EDOCUMENT_REQUISITES "+"SYSREF_EDOCUMENT_SIGNATURES "+"SYSREF_EDOCUMENT_TEMPLATES "+"SYSREF_EDOCUMENT_TEXT_STORAGES "+"SYSREF_EDOCUMENT_VIEWS "+"SYSREF_FILTERER_SETUP_CONFLICTS "+"SYSREF_FILTRATER_SETTING_CONFLICTS "+"SYSREF_FOLDER_HISTORY "+"SYSREF_FOLDERS "+"SYSREF_FUNCTION_GROUPS "+"SYSREF_FUNCTION_PARAMS "+"SYSREF_FUNCTIONS "+"SYSREF_JOB_HISTORY "+"SYSREF_LINKS "+"SYSREF_LOCALIZATION_DICTIONARY "+"SYSREF_LOCALIZATION_LANGUAGES "+"SYSREF_MODULES "+"SYSREF_PRIVILEGES "+"SYSREF_RECORD_HISTORY "+"SYSREF_REFERENCE_REQUISITES "+"SYSREF_REFERENCE_TYPE_VIEWS "+"SYSREF_REFERENCE_TYPES "+"SYSREF_REFERENCES "+"SYSREF_REFERENCES_REQUISITES "+"SYSREF_REMOTE_SERVERS "+"SYSREF_REPLICATION_SESSIONS_LOG "+"SYSREF_REPLICATION_SESSIONS_PROTOCOL "+"SYSREF_REPORTS "+"SYSREF_ROLES "+"SYSREF_ROUTE_BLOCK_GROUPS "+"SYSREF_ROUTE_BLOCKS "+"SYSREF_SCRIPTS "+"SYSREF_SEARCHES "+"SYSREF_SERVER_EVENTS "+"SYSREF_SERVER_EVENTS_HISTORY "+"SYSREF_STANDARD_ROUTE_GROUPS "+"SYSREF_STANDARD_ROUTES "+"SYSREF_STATUSES "+"SYSREF_SYSTEM_SETTINGS "+"SYSREF_TASK_HISTORY "+"SYSREF_TASK_KIND_GROUPS "+"SYSREF_TASK_KINDS "+"SYSREF_TASK_RIGHTS "+"SYSREF_TASK_SIGNATURES "+"SYSREF_TASKS "+"SYSREF_UNITS "+"SYSREF_USER_GROUPS "+"SYSREF_USER_GROUPS_REFERENCE "+"SYSREF_USER_SUBSTITUTION "+"SYSREF_USERS "+"SYSREF_USERS_REFERENCE "+"SYSREF_VIEWERS "+"SYSREF_WORKING_TIME_CALENDARS ";var table_name_constants="ACCESS_RIGHTS_TABLE_NAME "+"EDMS_ACCESS_TABLE_NAME "+"EDOC_TYPES_TABLE_NAME ";var test_constants="TEST_DEV_DB_NAME "+"TEST_DEV_SYSTEM_CODE "+"TEST_EDMS_DB_NAME "+"TEST_EDMS_MAIN_CODE "+"TEST_EDMS_MAIN_DB_NAME "+"TEST_EDMS_SECOND_CODE "+"TEST_EDMS_SECOND_DB_NAME "+"TEST_EDMS_SYSTEM_CODE "+"TEST_ISB5_MAIN_CODE "+"TEST_ISB5_SECOND_CODE "+"TEST_SQL_SERVER_2005_NAME "+"TEST_SQL_SERVER_NAME ";var using_the_dialog_windows_constants="ATTENTION_CAPTION "+"cbsCommandLinks "+"cbsDefault "+"CONFIRMATION_CAPTION "+"ERROR_CAPTION "+"INFORMATION_CAPTION "+"mrCancel "+"mrOk ";var using_the_document_constants="EDOC_VERSION_ACTIVE_STAGE_CODE "+"EDOC_VERSION_DESIGN_STAGE_CODE "+"EDOC_VERSION_OBSOLETE_STAGE_CODE ";var using_the_EA_and_encryption_constants="cpDataEnciphermentEnabled "+"cpDigitalSignatureEnabled "+"cpID "+"cpIssuer "+"cpPluginVersion "+"cpSerial "+"cpSubjectName "+"cpSubjSimpleName "+"cpValidFromDate "+"cpValidToDate ";var using_the_ISBL_editor_constants="ISBL_SYNTAX "+"NO_SYNTAX "+"XML_SYNTAX ";var wait_block_properties_constants="WAIT_BLOCK_AFTER_FINISH_EVENT "+"WAIT_BLOCK_BEFORE_START_EVENT "+"WAIT_BLOCK_DEADLINE_PROPERTY "+"WAIT_BLOCK_IS_RELATIVE_DEADLINE_PROPERTY "+"WAIT_BLOCK_NAME_PROPERTY "+"WAIT_BLOCK_RELATIVE_DEADLINE_TYPE_PROPERTY ";var sysres_common_constants="SYSRES_COMMON "+"SYSRES_CONST "+"SYSRES_MBFUNC "+"SYSRES_SBDATA "+"SYSRES_SBGUI "+"SYSRES_SBINTF "+"SYSRES_SBREFDSC "+"SYSRES_SQLERRORS "+"SYSRES_SYSCOMP ";var CONSTANTS=sysres_constants+base_constants+base_group_name_constants+decision_block_properties_constants+file_extension_constants+job_block_properties_constants+language_code_constants+launching_external_applications_constants+link_kind_constants+lock_type_constants+monitor_block_properties_constants+notice_block_properties_constants+object_events_constants+object_params_constants+other_constants+privileges_constants+pseudoreference_code_constants+requisite_ISBCertificateType_values_constants+requisite_ISBEDocStorageType_values_constants+requisite_compType2_values_constants+requisite_name_constants+result_constants+rule_identification_constants+script_block_properties_constants+subtask_block_properties_constants+system_component_constants+system_dialogs_constants+system_reference_names_constants+table_name_constants+test_constants+using_the_dialog_windows_constants+using_the_document_constants+using_the_EA_and_encryption_constants+using_the_ISBL_editor_constants+wait_block_properties_constants+sysres_common_constants;var TAccountType="atUser atGroup atRole ";var TActionEnabledMode="aemEnabledAlways "+"aemDisabledAlways "+"aemEnabledOnBrowse "+"aemEnabledOnEdit "+"aemDisabledOnBrowseEmpty ";var TAddPosition="apBegin apEnd ";var TAlignment="alLeft alRight ";var TAreaShowMode="asmNever "+"asmNoButCustomize "+"asmAsLastTime "+"asmYesButCustomize "+"asmAlways ";var TCertificateInvalidationReason="cirCommon cirRevoked ";var TCertificateType="ctSignature ctEncode ctSignatureEncode ";var TCheckListBoxItemState="clbUnchecked clbChecked clbGrayed ";var TCloseOnEsc="ceISB ceAlways ceNever ";var TCompType="ctDocument "+"ctReference "+"ctScript "+"ctUnknown "+"ctReport "+"ctDialog "+"ctFunction "+"ctFolder "+"ctEDocument "+"ctTask "+"ctJob "+"ctNotice "+"ctControlJob ";var TConditionFormat="cfInternal cfDisplay ";var TConnectionIntent="ciUnspecified ciWrite ciRead ";var TContentKind="ckFolder "+"ckEDocument "+"ckTask "+"ckJob "+"ckComponentToken "+"ckAny "+"ckReference "+"ckScript "+"ckReport "+"ckDialog ";var TControlType="ctISBLEditor "+"ctBevel "+"ctButton "+"ctCheckListBox "+"ctComboBox "+"ctComboEdit "+"ctGrid "+"ctDBCheckBox "+"ctDBComboBox "+"ctDBEdit "+"ctDBEllipsis "+"ctDBMemo "+"ctDBNavigator "+"ctDBRadioGroup "+"ctDBStatusLabel "+"ctEdit "+"ctGroupBox "+"ctInplaceHint "+"ctMemo "+"ctPanel "+"ctListBox "+"ctRadioButton "+"ctRichEdit "+"ctTabSheet "+"ctWebBrowser "+"ctImage "+"ctHyperLink "+"ctLabel "+"ctDBMultiEllipsis "+"ctRibbon "+"ctRichView "+"ctInnerPanel "+"ctPanelGroup "+"ctBitButton ";var TCriterionContentType="cctDate "+"cctInteger "+"cctNumeric "+"cctPick "+"cctReference "+"cctString "+"cctText ";var TCultureType="cltInternal cltPrimary cltGUI ";var TDataSetEventType="dseBeforeOpen "+"dseAfterOpen "+"dseBeforeClose "+"dseAfterClose "+"dseOnValidDelete "+"dseBeforeDelete "+"dseAfterDelete "+"dseAfterDeleteOutOfTransaction "+"dseOnDeleteError "+"dseBeforeInsert "+"dseAfterInsert "+"dseOnValidUpdate "+"dseBeforeUpdate "+"dseOnUpdateRatifiedRecord "+"dseAfterUpdate "+"dseAfterUpdateOutOfTransaction "+"dseOnUpdateError "+"dseAfterScroll "+"dseOnOpenRecord "+"dseOnCloseRecord "+"dseBeforeCancel "+"dseAfterCancel "+"dseOnUpdateDeadlockError "+"dseBeforeDetailUpdate "+"dseOnPrepareUpdate "+"dseOnAnyRequisiteChange ";var TDataSetState="dssEdit dssInsert dssBrowse dssInActive ";var TDateFormatType="dftDate dftShortDate dftDateTime dftTimeStamp ";var TDateOffsetType="dotDays dotHours dotMinutes dotSeconds ";var TDateTimeKind="dtkndLocal dtkndUTC ";var TDeaAccessRights="arNone arView arEdit arFull ";var TDocumentDefaultAction="ddaView ddaEdit ";var TEditMode="emLock "+"emEdit "+"emSign "+"emExportWithLock "+"emImportWithUnlock "+"emChangeVersionNote "+"emOpenForModify "+"emChangeLifeStage "+"emDelete "+"emCreateVersion "+"emImport "+"emUnlockExportedWithLock "+"emStart "+"emAbort "+"emReInit "+"emMarkAsReaded "+"emMarkAsUnreaded "+"emPerform "+"emAccept "+"emResume "+"emChangeRights "+"emEditRoute "+"emEditObserver "+"emRecoveryFromLocalCopy "+"emChangeWorkAccessType "+"emChangeEncodeTypeToCertificate "+"emChangeEncodeTypeToPassword "+"emChangeEncodeTypeToNone "+"emChangeEncodeTypeToCertificatePassword "+"emChangeStandardRoute "+"emGetText "+"emOpenForView "+"emMoveToStorage "+"emCreateObject "+"emChangeVersionHidden "+"emDeleteVersion "+"emChangeLifeCycleStage "+"emApprovingSign "+"emExport "+"emContinue "+"emLockFromEdit "+"emUnLockForEdit "+"emLockForServer "+"emUnlockFromServer "+"emDelegateAccessRights "+"emReEncode ";var TEditorCloseObservType="ecotFile ecotProcess ";var TEdmsApplicationAction="eaGet eaCopy eaCreate eaCreateStandardRoute ";var TEDocumentLockType="edltAll edltNothing edltQuery ";var TEDocumentStepShowMode="essmText essmCard ";var TEDocumentStepVersionType="esvtLast esvtLastActive esvtSpecified ";var TEDocumentStorageFunction="edsfExecutive edsfArchive ";var TEDocumentStorageType="edstSQLServer edstFile ";var TEDocumentVersionSourceType="edvstNone edvstEDocumentVersionCopy edvstFile edvstTemplate edvstScannedFile ";var TEDocumentVersionState="vsDefault vsDesign vsActive vsObsolete ";var TEncodeType="etNone etCertificate etPassword etCertificatePassword ";var TExceptionCategory="ecException ecWarning ecInformation ";var TExportedSignaturesType="estAll estApprovingOnly ";var TExportedVersionType="evtLast evtLastActive evtQuery ";var TFieldDataType="fdtString "+"fdtNumeric "+"fdtInteger "+"fdtDate "+"fdtText "+"fdtUnknown "+"fdtWideString "+"fdtLargeInteger ";var TFolderType="ftInbox "+"ftOutbox "+"ftFavorites "+"ftCommonFolder "+"ftUserFolder "+"ftComponents "+"ftQuickLaunch "+"ftShortcuts "+"ftSearch ";var TGridRowHeight="grhAuto "+"grhX1 "+"grhX2 "+"grhX3 ";var THyperlinkType="hltText "+"hltRTF "+"hltHTML ";var TImageFileFormat="iffBMP "+"iffJPEG "+"iffMultiPageTIFF "+"iffSinglePageTIFF "+"iffTIFF "+"iffPNG ";var TImageMode="im8bGrayscale "+"im24bRGB "+"im1bMonochrome ";var TImageType="itBMP "+"itJPEG "+"itWMF "+"itPNG ";var TInplaceHintKind="ikhInformation "+"ikhWarning "+"ikhError "+"ikhNoIcon ";var TISBLContext="icUnknown "+"icScript "+"icFunction "+"icIntegratedReport "+"icAnalyticReport "+"icDataSetEventHandler "+"icActionHandler "+"icFormEventHandler "+"icLookUpEventHandler "+"icRequisiteChangeEventHandler "+"icBeforeSearchEventHandler "+"icRoleCalculation "+"icSelectRouteEventHandler "+"icBlockPropertyCalculation "+"icBlockQueryParamsEventHandler "+"icChangeSearchResultEventHandler "+"icBlockEventHandler "+"icSubTaskInitEventHandler "+"icEDocDataSetEventHandler "+"icEDocLookUpEventHandler "+"icEDocActionHandler "+"icEDocFormEventHandler "+"icEDocRequisiteChangeEventHandler "+"icStructuredConversionRule "+"icStructuredConversionEventBefore "+"icStructuredConversionEventAfter "+"icWizardEventHandler "+"icWizardFinishEventHandler "+"icWizardStepEventHandler "+"icWizardStepFinishEventHandler "+"icWizardActionEnableEventHandler "+"icWizardActionExecuteEventHandler "+"icCreateJobsHandler "+"icCreateNoticesHandler "+"icBeforeLookUpEventHandler "+"icAfterLookUpEventHandler "+"icTaskAbortEventHandler "+"icWorkflowBlockActionHandler "+"icDialogDataSetEventHandler "+"icDialogActionHandler "+"icDialogLookUpEventHandler "+"icDialogRequisiteChangeEventHandler "+"icDialogFormEventHandler "+"icDialogValidCloseEventHandler "+"icBlockFormEventHandler "+"icTaskFormEventHandler "+"icReferenceMethod "+"icEDocMethod "+"icDialogMethod "+"icProcessMessageHandler ";var TItemShow="isShow "+"isHide "+"isByUserSettings ";var TJobKind="jkJob "+"jkNotice "+"jkControlJob ";var TJoinType="jtInner "+"jtLeft "+"jtRight "+"jtFull "+"jtCross ";var TLabelPos="lbpAbove "+"lbpBelow "+"lbpLeft "+"lbpRight ";var TLicensingType="eltPerConnection "+"eltPerUser ";var TLifeCycleStageFontColor="sfcUndefined "+"sfcBlack "+"sfcGreen "+"sfcRed "+"sfcBlue "+"sfcOrange "+"sfcLilac ";var TLifeCycleStageFontStyle="sfsItalic "+"sfsStrikeout "+"sfsNormal ";var TLockableDevelopmentComponentType="ldctStandardRoute "+"ldctWizard "+"ldctScript "+"ldctFunction "+"ldctRouteBlock "+"ldctIntegratedReport "+"ldctAnalyticReport "+"ldctReferenceType "+"ldctEDocumentType "+"ldctDialog "+"ldctServerEvents ";var TMaxRecordCountRestrictionType="mrcrtNone "+"mrcrtUser "+"mrcrtMaximal "+"mrcrtCustom ";var TRangeValueType="vtEqual "+"vtGreaterOrEqual "+"vtLessOrEqual "+"vtRange ";var TRelativeDate="rdYesterday "+"rdToday "+"rdTomorrow "+"rdThisWeek "+"rdThisMonth "+"rdThisYear "+"rdNextMonth "+"rdNextWeek "+"rdLastWeek "+"rdLastMonth ";var TReportDestination="rdWindow "+"rdFile "+"rdPrinter ";var TReqDataType="rdtString "+"rdtNumeric "+"rdtInteger "+"rdtDate "+"rdtReference "+"rdtAccount "+"rdtText "+"rdtPick "+"rdtUnknown "+"rdtLargeInteger "+"rdtDocument ";var TRequisiteEventType="reOnChange "+"reOnChangeValues ";var TSBTimeType="ttGlobal "+"ttLocal "+"ttUser "+"ttSystem ";var TSearchShowMode="ssmBrowse "+"ssmSelect "+"ssmMultiSelect "+"ssmBrowseModal ";var TSelectMode="smSelect "+"smLike "+"smCard ";var TSignatureType="stNone "+"stAuthenticating "+"stApproving ";var TSignerContentType="sctString "+"sctStream ";var TStringsSortType="sstAnsiSort "+"sstNaturalSort ";var TStringValueType="svtEqual "+"svtContain ";var TStructuredObjectAttributeType="soatString "+"soatNumeric "+"soatInteger "+"soatDatetime "+"soatReferenceRecord "+"soatText "+"soatPick "+"soatBoolean "+"soatEDocument "+"soatAccount "+"soatIntegerCollection "+"soatNumericCollection "+"soatStringCollection "+"soatPickCollection "+"soatDatetimeCollection "+"soatBooleanCollection "+"soatReferenceRecordCollection "+"soatEDocumentCollection "+"soatAccountCollection "+"soatContents "+"soatUnknown ";var TTaskAbortReason="tarAbortByUser "+"tarAbortByWorkflowException ";var TTextValueType="tvtAllWords "+"tvtExactPhrase "+"tvtAnyWord ";var TUserObjectStatus="usNone "+"usCompleted "+"usRedSquare "+"usBlueSquare "+"usYellowSquare "+"usGreenSquare "+"usOrangeSquare "+"usPurpleSquare "+"usFollowUp ";var TUserType="utUnknown "+"utUser "+"utDeveloper "+"utAdministrator "+"utSystemDeveloper "+"utDisconnected ";var TValuesBuildType="btAnd "+"btDetailAnd "+"btOr "+"btNotOr "+"btOnly ";var TViewMode="vmView "+"vmSelect "+"vmNavigation ";var TViewSelectionMode="vsmSingle "+"vsmMultiple "+"vsmMultipleCheck "+"vsmNoSelection ";var TWizardActionType="wfatPrevious "+"wfatNext "+"wfatCancel "+"wfatFinish ";var TWizardFormElementProperty="wfepUndefined "+"wfepText3 "+"wfepText6 "+"wfepText9 "+"wfepSpinEdit "+"wfepDropDown "+"wfepRadioGroup "+"wfepFlag "+"wfepText12 "+"wfepText15 "+"wfepText18 "+"wfepText21 "+"wfepText24 "+"wfepText27 "+"wfepText30 "+"wfepRadioGroupColumn1 "+"wfepRadioGroupColumn2 "+"wfepRadioGroupColumn3 ";var TWizardFormElementType="wfetQueryParameter "+"wfetText "+"wfetDelimiter "+"wfetLabel ";var TWizardParamType="wptString "+"wptInteger "+"wptNumeric "+"wptBoolean "+"wptDateTime "+"wptPick "+"wptText "+"wptUser "+"wptUserList "+"wptEDocumentInfo "+"wptEDocumentInfoList "+"wptReferenceRecordInfo "+"wptReferenceRecordInfoList "+"wptFolderInfo "+"wptTaskInfo "+"wptContents "+"wptFileName "+"wptDate ";var TWizardStepResult="wsrComplete "+"wsrGoNext "+"wsrGoPrevious "+"wsrCustom "+"wsrCancel "+"wsrGoFinal ";var TWizardStepType="wstForm "+"wstEDocument "+"wstTaskCard "+"wstReferenceRecordCard "+"wstFinal ";var TWorkAccessType="waAll "+"waPerformers "+"waManual ";var TWorkflowBlockType="wsbStart "+"wsbFinish "+"wsbNotice "+"wsbStep "+"wsbDecision "+"wsbWait "+"wsbMonitor "+"wsbScript "+"wsbConnector "+"wsbSubTask "+"wsbLifeCycleStage "+"wsbPause ";var TWorkflowDataType="wdtInteger "+"wdtFloat "+"wdtString "+"wdtPick "+"wdtDateTime "+"wdtBoolean "+"wdtTask "+"wdtJob "+"wdtFolder "+"wdtEDocument "+"wdtReferenceRecord "+"wdtUser "+"wdtGroup "+"wdtRole "+"wdtIntegerCollection "+"wdtFloatCollection "+"wdtStringCollection "+"wdtPickCollection "+"wdtDateTimeCollection "+"wdtBooleanCollection "+"wdtTaskCollection "+"wdtJobCollection "+"wdtFolderCollection "+"wdtEDocumentCollection "+"wdtReferenceRecordCollection "+"wdtUserCollection "+"wdtGroupCollection "+"wdtRoleCollection "+"wdtContents "+"wdtUserList "+"wdtSearchDescription "+"wdtDeadLine "+"wdtPickSet "+"wdtAccountCollection ";var TWorkImportance="wiLow "+"wiNormal "+"wiHigh ";var TWorkRouteType="wrtSoft "+"wrtHard ";var TWorkState="wsInit "+"wsRunning "+"wsDone "+"wsControlled "+"wsAborted "+"wsContinued ";var TWorkTextBuildingMode="wtmFull "+"wtmFromCurrent "+"wtmOnlyCurrent ";var ENUMS=TAccountType+TActionEnabledMode+TAddPosition+TAlignment+TAreaShowMode+TCertificateInvalidationReason+TCertificateType+TCheckListBoxItemState+TCloseOnEsc+TCompType+TConditionFormat+TConnectionIntent+TContentKind+TControlType+TCriterionContentType+TCultureType+TDataSetEventType+TDataSetState+TDateFormatType+TDateOffsetType+TDateTimeKind+TDeaAccessRights+TDocumentDefaultAction+TEditMode+TEditorCloseObservType+TEdmsApplicationAction+TEDocumentLockType+TEDocumentStepShowMode+TEDocumentStepVersionType+TEDocumentStorageFunction+TEDocumentStorageType+TEDocumentVersionSourceType+TEDocumentVersionState+TEncodeType+TExceptionCategory+TExportedSignaturesType+TExportedVersionType+TFieldDataType+TFolderType+TGridRowHeight+THyperlinkType+TImageFileFormat+TImageMode+TImageType+TInplaceHintKind+TISBLContext+TItemShow+TJobKind+TJoinType+TLabelPos+TLicensingType+TLifeCycleStageFontColor+TLifeCycleStageFontStyle+TLockableDevelopmentComponentType+TMaxRecordCountRestrictionType+TRangeValueType+TRelativeDate+TReportDestination+TReqDataType+TRequisiteEventType+TSBTimeType+TSearchShowMode+TSelectMode+TSignatureType+TSignerContentType+TStringsSortType+TStringValueType+TStructuredObjectAttributeType+TTaskAbortReason+TTextValueType+TUserObjectStatus+TUserType+TValuesBuildType+TViewMode+TViewSelectionMode+TWizardActionType+TWizardFormElementProperty+TWizardFormElementType+TWizardParamType+TWizardStepResult+TWizardStepType+TWorkAccessType+TWorkflowBlockType+TWorkflowDataType+TWorkImportance+TWorkRouteType+TWorkState+TWorkTextBuildingMode;var system_functions="AddSubString "+"AdjustLineBreaks "+"AmountInWords "+"Analysis "+"ArrayDimCount "+"ArrayHighBound "+"ArrayLowBound "+"ArrayOf "+"ArrayReDim "+"Assert "+"Assigned "+"BeginOfMonth "+"BeginOfPeriod "+"BuildProfilingOperationAnalysis "+"CallProcedure "+"CanReadFile "+"CArrayElement "+"CDataSetRequisite "+"ChangeDate "+"ChangeReferenceDataset "+"Char "+"CharPos "+"CheckParam "+"CheckParamValue "+"CompareStrings "+"ConstantExists "+"ControlState "+"ConvertDateStr "+"Copy "+"CopyFile "+"CreateArray "+"CreateCachedReference "+"CreateConnection "+"CreateDialog "+"CreateDualListDialog "+"CreateEditor "+"CreateException "+"CreateFile "+"CreateFolderDialog "+"CreateInputDialog "+"CreateLinkFile "+"CreateList "+"CreateLock "+"CreateMemoryDataSet "+"CreateObject "+"CreateOpenDialog "+"CreateProgress "+"CreateQuery "+"CreateReference "+"CreateReport "+"CreateSaveDialog "+"CreateScript "+"CreateSQLPivotFunction "+"CreateStringList "+"CreateTreeListSelectDialog "+"CSelectSQL "+"CSQL "+"CSubString "+"CurrentUserID "+"CurrentUserName "+"CurrentVersion "+"DataSetLocateEx "+"DateDiff "+"DateTimeDiff "+"DateToStr "+"DayOfWeek "+"DeleteFile "+"DirectoryExists "+"DisableCheckAccessRights "+"DisableCheckFullShowingRestriction "+"DisableMassTaskSendingRestrictions "+"DropTable "+"DupeString "+"EditText "+"EnableCheckAccessRights "+"EnableCheckFullShowingRestriction "+"EnableMassTaskSendingRestrictions "+"EndOfMonth "+"EndOfPeriod "+"ExceptionExists "+"ExceptionsOff "+"ExceptionsOn "+"Execute "+"ExecuteProcess "+"Exit "+"ExpandEnvironmentVariables "+"ExtractFileDrive "+"ExtractFileExt "+"ExtractFileName "+"ExtractFilePath "+"ExtractParams "+"FileExists "+"FileSize "+"FindFile "+"FindSubString "+"FirmContext "+"ForceDirectories "+"Format "+"FormatDate "+"FormatNumeric "+"FormatSQLDate "+"FormatString "+"FreeException "+"GetComponent "+"GetComponentLaunchParam "+"GetConstant "+"GetLastException "+"GetReferenceRecord "+"GetRefTypeByRefID "+"GetTableID "+"GetTempFolder "+"IfThen "+"In "+"IndexOf "+"InputDialog "+"InputDialogEx "+"InteractiveMode "+"IsFileLocked "+"IsGraphicFile "+"IsNumeric "+"Length "+"LoadString "+"LoadStringFmt "+"LocalTimeToUTC "+"LowerCase "+"Max "+"MessageBox "+"MessageBoxEx "+"MimeDecodeBinary "+"MimeDecodeString "+"MimeEncodeBinary "+"MimeEncodeString "+"Min "+"MoneyInWords "+"MoveFile "+"NewID "+"Now "+"OpenFile "+"Ord "+"Precision "+"Raise "+"ReadCertificateFromFile "+"ReadFile "+"ReferenceCodeByID "+"ReferenceNumber "+"ReferenceRequisiteMode "+"ReferenceRequisiteValue "+"RegionDateSettings "+"RegionNumberSettings "+"RegionTimeSettings "+"RegRead "+"RegWrite "+"RenameFile "+"Replace "+"Round "+"SelectServerCode "+"SelectSQL "+"ServerDateTime "+"SetConstant "+"SetManagedFolderFieldsState "+"ShowConstantsInputDialog "+"ShowMessage "+"Sleep "+"Split "+"SQL "+"SQL2XLSTAB "+"SQLProfilingSendReport "+"StrToDate "+"SubString "+"SubStringCount "+"SystemSetting "+"Time "+"TimeDiff "+"Today "+"Transliterate "+"Trim "+"UpperCase "+"UserStatus "+"UTCToLocalTime "+"ValidateXML "+"VarIsClear "+"VarIsEmpty "+"VarIsNull "+"WorkTimeDiff "+"WriteFile "+"WriteFileEx "+"WriteObjectHistory "+"Анализ "+"БазаДанных "+"БлокЕсть "+"БлокЕстьРасш "+"БлокИнфо "+"БлокСнять "+"БлокСнятьРасш "+"БлокУстановить "+"Ввод "+"ВводМеню "+"ВедС "+"ВедСпр "+"ВерхняяГраницаМассива "+"ВнешПрогр "+"Восст "+"ВременнаяПапка "+"Время "+"ВыборSQL "+"ВыбратьЗапись "+"ВыделитьСтр "+"Вызвать "+"Выполнить "+"ВыпПрогр "+"ГрафическийФайл "+"ГруппаДополнительно "+"ДатаВремяСерв "+"ДеньНедели "+"ДиалогДаНет "+"ДлинаСтр "+"ДобПодстр "+"ЕПусто "+"ЕслиТо "+"ЕЧисло "+"ЗамПодстр "+"ЗаписьСправочника "+"ЗначПоляСпр "+"ИДТипСпр "+"ИзвлечьДиск "+"ИзвлечьИмяФайла "+"ИзвлечьПуть "+"ИзвлечьРасширение "+"ИзмДат "+"ИзменитьРазмерМассива "+"ИзмеренийМассива "+"ИмяОрг "+"ИмяПоляСпр "+"Индекс "+"ИндикаторЗакрыть "+"ИндикаторОткрыть "+"ИндикаторШаг "+"ИнтерактивныйРежим "+"ИтогТблСпр "+"КодВидВедСпр "+"КодВидСпрПоИД "+"КодПоAnalit "+"КодСимвола "+"КодСпр "+"КолПодстр "+"КолПроп "+"КонМес "+"Конст "+"КонстЕсть "+"КонстЗнач "+"КонТран "+"КопироватьФайл "+"КопияСтр "+"КПериод "+"КСтрТблСпр "+"Макс "+"МаксСтрТблСпр "+"Массив "+"Меню "+"МенюРасш "+"Мин "+"НаборДанныхНайтиРасш "+"НаимВидСпр "+"НаимПоAnalit "+"НаимСпр "+"НастроитьПереводыСтрок "+"НачМес "+"НачТран "+"НижняяГраницаМассива "+"НомерСпр "+"НПериод "+"Окно "+"Окр "+"Окружение "+"ОтлИнфДобавить "+"ОтлИнфУдалить "+"Отчет "+"ОтчетАнал "+"ОтчетИнт "+"ПапкаСуществует "+"Пауза "+"ПВыборSQL "+"ПереименоватьФайл "+"Переменные "+"ПереместитьФайл "+"Подстр "+"ПоискПодстр "+"ПоискСтр "+"ПолучитьИДТаблицы "+"ПользовательДополнительно "+"ПользовательИД "+"ПользовательИмя "+"ПользовательСтатус "+"Прервать "+"ПроверитьПараметр "+"ПроверитьПараметрЗнач "+"ПроверитьУсловие "+"РазбСтр "+"РазнВремя "+"РазнДат "+"РазнДатаВремя "+"РазнРабВремя "+"РегУстВрем "+"РегУстДат "+"РегУстЧсл "+"РедТекст "+"РеестрЗапись "+"РеестрСписокИменПарам "+"РеестрЧтение "+"РеквСпр "+"РеквСпрПр "+"Сегодня "+"Сейчас "+"Сервер "+"СерверПроцессИД "+"СертификатФайлСчитать "+"СжПроб "+"Символ "+"СистемаДиректумКод "+"СистемаИнформация "+"СистемаКод "+"Содержит "+"СоединениеЗакрыть "+"СоединениеОткрыть "+"СоздатьДиалог "+"СоздатьДиалогВыбораИзДвухСписков "+"СоздатьДиалогВыбораПапки "+"СоздатьДиалогОткрытияФайла "+"СоздатьДиалогСохраненияФайла "+"СоздатьЗапрос "+"СоздатьИндикатор "+"СоздатьИсключение "+"СоздатьКэшированныйСправочник "+"СоздатьМассив "+"СоздатьНаборДанных "+"СоздатьОбъект "+"СоздатьОтчет "+"СоздатьПапку "+"СоздатьРедактор "+"СоздатьСоединение "+"СоздатьСписок "+"СоздатьСписокСтрок "+"СоздатьСправочник "+"СоздатьСценарий "+"СоздСпр "+"СостСпр "+"Сохр "+"СохрСпр "+"СписокСистем "+"Спр "+"Справочник "+"СпрБлокЕсть "+"СпрБлокСнять "+"СпрБлокСнятьРасш "+"СпрБлокУстановить "+"СпрИзмНабДан "+"СпрКод "+"СпрНомер "+"СпрОбновить "+"СпрОткрыть "+"СпрОтменить "+"СпрПарам "+"СпрПолеЗнач "+"СпрПолеИмя "+"СпрРекв "+"СпрРеквВведЗн "+"СпрРеквНовые "+"СпрРеквПр "+"СпрРеквПредЗн "+"СпрРеквРежим "+"СпрРеквТипТекст "+"СпрСоздать "+"СпрСост "+"СпрСохранить "+"СпрТблИтог "+"СпрТблСтр "+"СпрТблСтрКол "+"СпрТблСтрМакс "+"СпрТблСтрМин "+"СпрТблСтрПред "+"СпрТблСтрСлед "+"СпрТблСтрСозд "+"СпрТблСтрУд "+"СпрТекПредст "+"СпрУдалить "+"СравнитьСтр "+"СтрВерхРегистр "+"СтрНижнРегистр "+"СтрТблСпр "+"СумПроп "+"Сценарий "+"СценарийПарам "+"ТекВерсия "+"ТекОрг "+"Точн "+"Тран "+"Транслитерация "+"УдалитьТаблицу "+"УдалитьФайл "+"УдСпр "+"УдСтрТблСпр "+"Уст "+"УстановкиКонстант "+"ФайлАтрибутСчитать "+"ФайлАтрибутУстановить "+"ФайлВремя "+"ФайлВремяУстановить "+"ФайлВыбрать "+"ФайлЗанят "+"ФайлЗаписать "+"ФайлИскать "+"ФайлКопировать "+"ФайлМожноЧитать "+"ФайлОткрыть "+"ФайлПереименовать "+"ФайлПерекодировать "+"ФайлПереместить "+"ФайлПросмотреть "+"ФайлРазмер "+"ФайлСоздать "+"ФайлСсылкаСоздать "+"ФайлСуществует "+"ФайлСчитать "+"ФайлУдалить "+"ФмтSQLДат "+"ФмтДат "+"ФмтСтр "+"ФмтЧсл "+"Формат "+"ЦМассивЭлемент "+"ЦНаборДанныхРеквизит "+"ЦПодстр ";var predefined_variables="AltState "+"Application "+"CallType "+"ComponentTokens "+"CreatedJobs "+"CreatedNotices "+"ControlState "+"DialogResult "+"Dialogs "+"EDocuments "+"EDocumentVersionSource "+"Folders "+"GlobalIDs "+"Job "+"Jobs "+"InputValue "+"LookUpReference "+"LookUpRequisiteNames "+"LookUpSearch "+"Object "+"ParentComponent "+"Processes "+"References "+"Requisite "+"ReportName "+"Reports "+"Result "+"Scripts "+"Searches "+"SelectedAttachments "+"SelectedItems "+"SelectMode "+"Sender "+"ServerEvents "+"ServiceFactory "+"ShiftState "+"SubTask "+"SystemDialogs "+"Tasks "+"Wizard "+"Wizards "+"Work "+"ВызовСпособ "+"ИмяОтчета "+"РеквЗнач ";var interfaces="IApplication "+"IAccessRights "+"IAccountRepository "+"IAccountSelectionRestrictions "+"IAction "+"IActionList "+"IAdministrationHistoryDescription "+"IAnchors "+"IApplication "+"IArchiveInfo "+"IAttachment "+"IAttachmentList "+"ICheckListBox "+"ICheckPointedList "+"IColumn "+"IComponent "+"IComponentDescription "+"IComponentToken "+"IComponentTokenFactory "+"IComponentTokenInfo "+"ICompRecordInfo "+"IConnection "+"IContents "+"IControl "+"IControlJob "+"IControlJobInfo "+"IControlList "+"ICrypto "+"ICrypto2 "+"ICustomJob "+"ICustomJobInfo "+"ICustomListBox "+"ICustomObjectWizardStep "+"ICustomWork "+"ICustomWorkInfo "+"IDataSet "+"IDataSetAccessInfo "+"IDataSigner "+"IDateCriterion "+"IDateRequisite "+"IDateRequisiteDescription "+"IDateValue "+"IDeaAccessRights "+"IDeaObjectInfo "+"IDevelopmentComponentLock "+"IDialog "+"IDialogFactory "+"IDialogPickRequisiteItems "+"IDialogsFactory "+"IDICSFactory "+"IDocRequisite "+"IDocumentInfo "+"IDualListDialog "+"IECertificate "+"IECertificateInfo "+"IECertificates "+"IEditControl "+"IEditorForm "+"IEdmsExplorer "+"IEdmsObject "+"IEdmsObjectDescription "+"IEdmsObjectFactory "+"IEdmsObjectInfo "+"IEDocument "+"IEDocumentAccessRights "+"IEDocumentDescription "+"IEDocumentEditor "+"IEDocumentFactory "+"IEDocumentInfo "+"IEDocumentStorage "+"IEDocumentVersion "+"IEDocumentVersionListDialog "+"IEDocumentVersionSource "+"IEDocumentWizardStep "+"IEDocVerSignature "+"IEDocVersionState "+"IEnabledMode "+"IEncodeProvider "+"IEncrypter "+"IEvent "+"IEventList "+"IException "+"IExternalEvents "+"IExternalHandler "+"IFactory "+"IField "+"IFileDialog "+"IFolder "+"IFolderDescription "+"IFolderDialog "+"IFolderFactory "+"IFolderInfo "+"IForEach "+"IForm "+"IFormTitle "+"IFormWizardStep "+"IGlobalIDFactory "+"IGlobalIDInfo "+"IGrid "+"IHasher "+"IHistoryDescription "+"IHyperLinkControl "+"IImageButton "+"IImageControl "+"IInnerPanel "+"IInplaceHint "+"IIntegerCriterion "+"IIntegerList "+"IIntegerRequisite "+"IIntegerValue "+"IISBLEditorForm "+"IJob "+"IJobDescription "+"IJobFactory "+"IJobForm "+"IJobInfo "+"ILabelControl "+"ILargeIntegerCriterion "+"ILargeIntegerRequisite "+"ILargeIntegerValue "+"ILicenseInfo "+"ILifeCycleStage "+"IList "+"IListBox "+"ILocalIDInfo "+"ILocalization "+"ILock "+"IMemoryDataSet "+"IMessagingFactory "+"IMetadataRepository "+"INotice "+"INoticeInfo "+"INumericCriterion "+"INumericRequisite "+"INumericValue "+"IObject "+"IObjectDescription "+"IObjectImporter "+"IObjectInfo "+"IObserver "+"IPanelGroup "+"IPickCriterion "+"IPickProperty "+"IPickRequisite "+"IPickRequisiteDescription "+"IPickRequisiteItem "+"IPickRequisiteItems "+"IPickValue "+"IPrivilege "+"IPrivilegeList "+"IProcess "+"IProcessFactory "+"IProcessMessage "+"IProgress "+"IProperty "+"IPropertyChangeEvent "+"IQuery "+"IReference "+"IReferenceCriterion "+"IReferenceEnabledMode "+"IReferenceFactory "+"IReferenceHistoryDescription "+"IReferenceInfo "+"IReferenceRecordCardWizardStep "+"IReferenceRequisiteDescription "+"IReferencesFactory "+"IReferenceValue "+"IRefRequisite "+"IReport "+"IReportFactory "+"IRequisite "+"IRequisiteDescription "+"IRequisiteDescriptionList "+"IRequisiteFactory "+"IRichEdit "+"IRouteStep "+"IRule "+"IRuleList "+"ISchemeBlock "+"IScript "+"IScriptFactory "+"ISearchCriteria "+"ISearchCriterion "+"ISearchDescription "+"ISearchFactory "+"ISearchFolderInfo "+"ISearchForObjectDescription "+"ISearchResultRestrictions "+"ISecuredContext "+"ISelectDialog "+"IServerEvent "+"IServerEventFactory "+"IServiceDialog "+"IServiceFactory "+"ISignature "+"ISignProvider "+"ISignProvider2 "+"ISignProvider3 "+"ISimpleCriterion "+"IStringCriterion "+"IStringList "+"IStringRequisite "+"IStringRequisiteDescription "+"IStringValue "+"ISystemDialogsFactory "+"ISystemInfo "+"ITabSheet "+"ITask "+"ITaskAbortReasonInfo "+"ITaskCardWizardStep "+"ITaskDescription "+"ITaskFactory "+"ITaskInfo "+"ITaskRoute "+"ITextCriterion "+"ITextRequisite "+"ITextValue "+"ITreeListSelectDialog "+"IUser "+"IUserList "+"IValue "+"IView "+"IWebBrowserControl "+"IWizard "+"IWizardAction "+"IWizardFactory "+"IWizardFormElement "+"IWizardParam "+"IWizardPickParam "+"IWizardReferenceParam "+"IWizardStep "+"IWorkAccessRights "+"IWorkDescription "+"IWorkflowAskableParam "+"IWorkflowAskableParams "+"IWorkflowBlock "+"IWorkflowBlockResult "+"IWorkflowEnabledMode "+"IWorkflowParam "+"IWorkflowPickParam "+"IWorkflowReferenceParam "+"IWorkState "+"IWorkTreeCustomNode "+"IWorkTreeJobNode "+"IWorkTreeTaskNode "+"IXMLEditorForm "+"SBCrypto ";var BUILTIN=CONSTANTS+ENUMS;var CLASS=predefined_variables;var LITERAL="null true false nil ";var NUMBERS={className:"number",begin:hljs.NUMBER_RE,relevance:0};var STRINGS={className:"string",variants:[{begin:'"',end:'"'},{begin:"'",end:"'"}]};var DOCTAGS={className:"doctag",begin:"\\b(?:TODO|DONE|BEGIN|END|STUB|CHG|FIXME|NOTE|BUG|XXX)\\b",relevance:0};var ISBL_LINE_COMMENT_MODE={className:"comment",begin:"//",end:"$",relevance:0,contains:[hljs.PHRASAL_WORDS_MODE,DOCTAGS]};var ISBL_BLOCK_COMMENT_MODE={className:"comment",begin:"/\\*",end:"\\*/",relevance:0,contains:[hljs.PHRASAL_WORDS_MODE,DOCTAGS]};var COMMENTS={variants:[ISBL_LINE_COMMENT_MODE,ISBL_BLOCK_COMMENT_MODE]};var KEYWORDS={keyword:KEYWORD,built_in:BUILTIN,class:CLASS,literal:LITERAL};var METHODS={begin:"\\.\\s*"+hljs.UNDERSCORE_IDENT_RE,keywords:KEYWORDS,relevance:0};var TYPES={className:"type",begin:":[ \\t]*("+interfaces.trim().replace(/\s/g,"|")+")",end:"[ \\t]*=",excludeEnd:true};var VARIABLES={className:"variable",lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,begin:UNDERSCORE_IDENT_RE,relevance:0,contains:[TYPES,METHODS]};var FUNCTION_TITLE=FUNCTION_NAME_IDENT_RE+"\\(";var TITLE_MODE={className:"title",lexemes:UNDERSCORE_IDENT_RE,keywords:{built_in:system_functions},begin:FUNCTION_TITLE,end:"\\(",returnBegin:true,excludeEnd:true};var FUNCTIONS={className:"function",begin:FUNCTION_TITLE,end:"\\)$",returnBegin:true,lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,illegal:"[\\[\\]\\|\\$\\?%,~#@]",contains:[TITLE_MODE,METHODS,VARIABLES,STRINGS,NUMBERS,COMMENTS]};return{aliases:["isbl"],case_insensitive:true,lexemes:UNDERSCORE_IDENT_RE,keywords:KEYWORDS,illegal:"\\$|\\?|%|,|;$|~|#|@|</",contains:[FUNCTIONS,TYPES,METHODS,VARIABLES,STRINGS,NUMBERS,COMMENTS]}});hljs.registerLanguage("reasonml",function(hljs){function orReValues(ops){return ops.map(function(op){return op.split("").map(function(char){return"\\"+char}).join("")}).join("|")}var RE_IDENT="~?[a-z$_][0-9a-zA-Z$_]*";var RE_MODULE_IDENT="`?[A-Z$_][0-9a-zA-Z$_]*";var RE_PARAM_TYPEPARAM="'?[a-z$_][0-9a-z$_]*";var RE_PARAM_TYPE="s*:s*[a-z$_][0-9a-z$_]*((s*("+RE_PARAM_TYPEPARAM+"s*(,"+RE_PARAM_TYPEPARAM+")*)?s*))?";var RE_PARAM=RE_IDENT+"("+RE_PARAM_TYPE+")?("+RE_PARAM_TYPE+")?";var RE_OPERATOR="("+orReValues(["||","&&","++","**","+.","*","/","*.","/.","...","|>"])+"|==|===)";var RE_OPERATOR_SPACED="\\s+"+RE_OPERATOR+"\\s+";var KEYWORDS={keyword:"and as asr assert begin class constraint do done downto else end exception external"+"for fun function functor if in include inherit initializer"+"land lazy let lor lsl lsr lxor match method mod module mutable new nonrec"+"object of open or private rec sig struct then to try type val virtual when while with",built_in:"array bool bytes char exn|5 float int int32 int64 list lazy_t|5 nativeint|5 ref string unit ",literal:"true false"};var RE_NUMBER="\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)";var NUMBER_MODE={className:"number",relevance:0,variants:[{begin:RE_NUMBER},{begin:"\\(\\-"+RE_NUMBER+"\\)"}]};var OPERATOR_MODE={className:"operator",relevance:0,begin:RE_OPERATOR};var LIST_CONTENTS_MODES=[{className:"identifier",relevance:0,begin:RE_IDENT},OPERATOR_MODE,NUMBER_MODE];var MODULE_ACCESS_CONTENTS=[hljs.QUOTE_STRING_MODE,OPERATOR_MODE,{className:"module",begin:"\\b"+RE_MODULE_IDENT,returnBegin:true,end:".",contains:[{className:"identifier",begin:RE_MODULE_IDENT,relevance:0}]}];var PARAMS_CONTENTS=[{className:"module",begin:"\\b"+RE_MODULE_IDENT,returnBegin:true,end:".",relevance:0,contains:[{className:"identifier",begin:RE_MODULE_IDENT,relevance:0}]}];var PARAMS_MODE={begin:RE_IDENT,end:"(,|\\n|\\))",relevance:0,contains:[OPERATOR_MODE,{className:"typing",begin:":",end:"(,|\\n)",returnBegin:true,relevance:0,contains:PARAMS_CONTENTS}]};var FUNCTION_BLOCK_MODE={className:"function",relevance:0,keywords:KEYWORDS,variants:[{begin:"\\s(\\(\\.?.*?\\)|"+RE_IDENT+")\\s*=>",end:"\\s*=>",returnBegin:true,relevance:0,contains:[{className:"params",variants:[{begin:RE_IDENT},{begin:RE_PARAM},{begin:/\(\s*\)/}]}]},{begin:"\\s\\(\\.?[^;\\|]*\\)\\s*=>",end:"\\s=>",returnBegin:true,relevance:0,contains:[{className:"params",relevance:0,variants:[PARAMS_MODE]}]},{begin:"\\(\\.\\s"+RE_IDENT+"\\)\\s*=>"}]};MODULE_ACCESS_CONTENTS.push(FUNCTION_BLOCK_MODE);var CONSTRUCTOR_MODE={className:"constructor",begin:RE_MODULE_IDENT+"\\(",end:"\\)",illegal:"\\n",keywords:KEYWORDS,contains:[hljs.QUOTE_STRING_MODE,OPERATOR_MODE,{className:"params",begin:"\\b"+RE_IDENT}]};var PATTERN_MATCH_BLOCK_MODE={className:"pattern-match",begin:"\\|",returnBegin:true,keywords:KEYWORDS,end:"=>",relevance:0,contains:[CONSTRUCTOR_MODE,OPERATOR_MODE,{relevance:0,className:"constructor",begin:RE_MODULE_IDENT}]};var MODULE_ACCESS_MODE={className:"module-access",keywords:KEYWORDS,returnBegin:true,variants:[{begin:"\\b("+RE_MODULE_IDENT+"\\.)+"+RE_IDENT},{begin:"\\b("+RE_MODULE_IDENT+"\\.)+\\(",end:"\\)",returnBegin:true,contains:[FUNCTION_BLOCK_MODE,{begin:"\\(",end:"\\)",skip:true}].concat(MODULE_ACCESS_CONTENTS)},{begin:"\\b("+RE_MODULE_IDENT+"\\.)+{",end:"}"}],contains:MODULE_ACCESS_CONTENTS};PARAMS_CONTENTS.push(MODULE_ACCESS_MODE);return{aliases:["re"],keywords:KEYWORDS,illegal:"(:\\-|:=|\\${|\\+=)",contains:[hljs.COMMENT("/\\*","\\*/",{illegal:"^(\\#,\\/\\/)"}),{className:"character",begin:"'(\\\\[^']+|[^'])'",illegal:"\\n",relevance:0},hljs.QUOTE_STRING_MODE,{className:"literal",begin:"\\(\\)",relevance:0},{className:"literal",begin:"\\[\\|",end:"\\|\\]",relevance:0,contains:LIST_CONTENTS_MODES},{className:"literal",begin:"\\[",end:"\\]",relevance:0,contains:LIST_CONTENTS_MODES},CONSTRUCTOR_MODE,{className:"operator",begin:RE_OPERATOR_SPACED,illegal:"\\-\\->",relevance:0},NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,PATTERN_MATCH_BLOCK_MODE,FUNCTION_BLOCK_MODE,{className:"module-def",begin:"\\bmodule\\s+"+RE_IDENT+"\\s+"+RE_MODULE_IDENT+"\\s+=\\s+{",end:"}",returnBegin:true,keywords:KEYWORDS,relevance:0,contains:[{className:"module",relevance:0,begin:RE_MODULE_IDENT},{begin:"{",end:"}",skip:true}].concat(MODULE_ACCESS_CONTENTS)},MODULE_ACCESS_MODE]}});hljs.registerLanguage("maxima",function(hljs){var KEYWORDS="if then else elseif for thru do while unless step in and or not";var LITERALS="true false unknown inf minf ind und %e %i %pi %phi %gamma";var BUILTIN_FUNCTIONS=" abasep abs absint absolute_real_time acos acosh acot acoth acsc acsch activate"+" addcol add_edge add_edges addmatrices addrow add_vertex add_vertices adjacency_matrix"+" adjoin adjoint af agd airy airy_ai airy_bi airy_dai airy_dbi algsys alg_type"+" alias allroots alphacharp alphanumericp amortization %and annuity_fv"+" annuity_pv antid antidiff AntiDifference append appendfile apply apply1 apply2"+" applyb1 apropos args arit_amortization arithmetic arithsum array arrayapply"+" arrayinfo arraymake arraysetapply ascii asec asech asin asinh askinteger"+" asksign assoc assoc_legendre_p assoc_legendre_q assume assume_external_byte_order"+" asympa at atan atan2 atanh atensimp atom atvalue augcoefmatrix augmented_lagrangian_method"+" av average_degree backtrace bars barsplot barsplot_description base64 base64_decode"+" bashindices batch batchload bc2 bdvac belln benefit_cost bern bernpoly bernstein_approx"+" bernstein_expand bernstein_poly bessel bessel_i bessel_j bessel_k bessel_simplify"+" bessel_y beta beta_incomplete beta_incomplete_generalized beta_incomplete_regularized"+" bezout bfallroots bffac bf_find_root bf_fmin_cobyla bfhzeta bfloat bfloatp"+" bfpsi bfpsi0 bfzeta biconnected_components bimetric binomial bipartition"+" block blockmatrixp bode_gain bode_phase bothcoef box boxplot boxplot_description"+" break bug_report build_info|10 buildq build_sample burn cabs canform canten"+" cardinality carg cartan cartesian_product catch cauchy_matrix cbffac cdf_bernoulli"+" cdf_beta cdf_binomial cdf_cauchy cdf_chi2 cdf_continuous_uniform cdf_discrete_uniform"+" cdf_exp cdf_f cdf_gamma cdf_general_finite_discrete cdf_geometric cdf_gumbel"+" cdf_hypergeometric cdf_laplace cdf_logistic cdf_lognormal cdf_negative_binomial"+" cdf_noncentral_chi2 cdf_noncentral_student_t cdf_normal cdf_pareto cdf_poisson"+" cdf_rank_sum cdf_rayleigh cdf_signed_rank cdf_student_t cdf_weibull cdisplay"+" ceiling central_moment cequal cequalignore cf cfdisrep cfexpand cgeodesic"+" cgreaterp cgreaterpignore changename changevar chaosgame charat charfun charfun2"+" charlist charp charpoly chdir chebyshev_t chebyshev_u checkdiv check_overlaps"+" chinese cholesky christof chromatic_index chromatic_number cint circulant_graph"+" clear_edge_weight clear_rules clear_vertex_label clebsch_gordan clebsch_graph"+" clessp clesspignore close closefile cmetric coeff coefmatrix cograd col collapse"+" collectterms columnop columnspace columnswap columnvector combination combine"+" comp2pui compare compfile compile compile_file complement_graph complete_bipartite_graph"+" complete_graph complex_number_p components compose_functions concan concat"+" conjugate conmetderiv connected_components connect_vertices cons constant"+" constantp constituent constvalue cont2part content continuous_freq contortion"+" contour_plot contract contract_edge contragrad contrib_ode convert coord"+" copy copy_file copy_graph copylist copymatrix cor cos cosh cot coth cov cov1"+" covdiff covect covers crc24sum create_graph create_list csc csch csetup cspline"+" ctaylor ct_coordsys ctransform ctranspose cube_graph cuboctahedron_graph"+" cunlisp cv cycle_digraph cycle_graph cylindrical days360 dblint deactivate"+" declare declare_constvalue declare_dimensions declare_fundamental_dimensions"+" declare_fundamental_units declare_qty declare_translated declare_unit_conversion"+" declare_units declare_weights decsym defcon define define_alt_display define_variable"+" defint defmatch defrule defstruct deftaylor degree_sequence del delete deleten"+" delta demo demoivre denom depends derivdegree derivlist describe desolve"+" determinant dfloat dgauss_a dgauss_b dgeev dgemm dgeqrf dgesv dgesvd diag"+" diagmatrix diag_matrix diagmatrixp diameter diff digitcharp dimacs_export"+" dimacs_import dimension dimensionless dimensions dimensions_as_list direct"+" directory discrete_freq disjoin disjointp disolate disp dispcon dispform"+" dispfun dispJordan display disprule dispterms distrib divide divisors divsum"+" dkummer_m dkummer_u dlange dodecahedron_graph dotproduct dotsimp dpart"+" draw draw2d draw3d drawdf draw_file draw_graph dscalar echelon edge_coloring"+" edge_connectivity edges eigens_by_jacobi eigenvalues eigenvectors eighth"+" einstein eivals eivects elapsed_real_time elapsed_run_time ele2comp ele2polynome"+" ele2pui elem elementp elevation_grid elim elim_allbut eliminate eliminate_using"+" ellipse elliptic_e elliptic_ec elliptic_eu elliptic_f elliptic_kc elliptic_pi"+" ematrix empty_graph emptyp endcons entermatrix entertensor entier equal equalp"+" equiv_classes erf erfc erf_generalized erfi errcatch error errormsg errors"+" euler ev eval_string evenp every evolution evolution2d evundiff example exp"+" expand expandwrt expandwrt_factored expint expintegral_chi expintegral_ci"+" expintegral_e expintegral_e1 expintegral_ei expintegral_e_simplify expintegral_li"+" expintegral_shi expintegral_si explicit explose exponentialize express expt"+" exsec extdiff extract_linear_equations extremal_subset ezgcd %f f90 facsum"+" factcomb factor factorfacsum factorial factorout factorsum facts fast_central_elements"+" fast_linsolve fasttimes featurep fernfale fft fib fibtophi fifth filename_merge"+" file_search file_type fillarray findde find_root find_root_abs find_root_error"+" find_root_rel first fix flatten flength float floatnump floor flower_snark"+" flush flush1deriv flushd flushnd flush_output fmin_cobyla forget fortran"+" fourcos fourexpand fourier fourier_elim fourint fourintcos fourintsin foursimp"+" foursin fourth fposition frame_bracket freeof freshline fresnel_c fresnel_s"+" from_adjacency_matrix frucht_graph full_listify fullmap fullmapl fullratsimp"+" fullratsubst fullsetify funcsolve fundamental_dimensions fundamental_units"+" fundef funmake funp fv g0 g1 gamma gamma_greek gamma_incomplete gamma_incomplete_generalized"+" gamma_incomplete_regularized gauss gauss_a gauss_b gaussprob gcd gcdex gcdivide"+" gcfac gcfactor gd generalized_lambert_w genfact gen_laguerre genmatrix gensym"+" geo_amortization geo_annuity_fv geo_annuity_pv geomap geometric geometric_mean"+" geosum get getcurrentdirectory get_edge_weight getenv get_lu_factors get_output_stream_string"+" get_pixel get_plot_option get_tex_environment get_tex_environment_default"+" get_vertex_label gfactor gfactorsum ggf girth global_variances gn gnuplot_close"+" gnuplot_replot gnuplot_reset gnuplot_restart gnuplot_start go Gosper GosperSum"+" gr2d gr3d gradef gramschmidt graph6_decode graph6_encode graph6_export graph6_import"+" graph_center graph_charpoly graph_eigenvalues graph_flow graph_order graph_periphery"+" graph_product graph_size graph_union great_rhombicosidodecahedron_graph great_rhombicuboctahedron_graph"+" grid_graph grind grobner_basis grotzch_graph hamilton_cycle hamilton_path"+" hankel hankel_1 hankel_2 harmonic harmonic_mean hav heawood_graph hermite"+" hessian hgfred hilbertmap hilbert_matrix hipow histogram histogram_description"+" hodge horner hypergeometric i0 i1 %ibes ic1 ic2 ic_convert ichr1 ichr2 icosahedron_graph"+" icosidodecahedron_graph icurvature ident identfor identity idiff idim idummy"+" ieqn %if ifactors iframes ifs igcdex igeodesic_coords ilt image imagpart"+" imetric implicit implicit_derivative implicit_plot indexed_tensor indices"+" induced_subgraph inferencep inference_result infix info_display init_atensor"+" init_ctensor in_neighbors innerproduct inpart inprod inrt integerp integer_partitions"+" integrate intersect intersection intervalp intopois intosum invariant1 invariant2"+" inverse_fft inverse_jacobi_cd inverse_jacobi_cn inverse_jacobi_cs inverse_jacobi_dc"+" inverse_jacobi_dn inverse_jacobi_ds inverse_jacobi_nc inverse_jacobi_nd inverse_jacobi_ns"+" inverse_jacobi_sc inverse_jacobi_sd inverse_jacobi_sn invert invert_by_adjoint"+" invert_by_lu inv_mod irr is is_biconnected is_bipartite is_connected is_digraph"+" is_edge_in_graph is_graph is_graph_or_digraph ishow is_isomorphic isolate"+" isomorphism is_planar isqrt isreal_p is_sconnected is_tree is_vertex_in_graph"+" items_inference %j j0 j1 jacobi jacobian jacobi_cd jacobi_cn jacobi_cs jacobi_dc"+" jacobi_dn jacobi_ds jacobi_nc jacobi_nd jacobi_ns jacobi_p jacobi_sc jacobi_sd"+" jacobi_sn JF jn join jordan julia julia_set julia_sin %k kdels kdelta kill"+" killcontext kostka kron_delta kronecker_product kummer_m kummer_u kurtosis"+" kurtosis_bernoulli kurtosis_beta kurtosis_binomial kurtosis_chi2 kurtosis_continuous_uniform"+" kurtosis_discrete_uniform kurtosis_exp kurtosis_f kurtosis_gamma kurtosis_general_finite_discrete"+" kurtosis_geometric kurtosis_gumbel kurtosis_hypergeometric kurtosis_laplace"+" kurtosis_logistic kurtosis_lognormal kurtosis_negative_binomial kurtosis_noncentral_chi2"+" kurtosis_noncentral_student_t kurtosis_normal kurtosis_pareto kurtosis_poisson"+" kurtosis_rayleigh kurtosis_student_t kurtosis_weibull label labels lagrange"+" laguerre lambda lambert_w laplace laplacian_matrix last lbfgs lc2kdt lcharp"+" lc_l lcm lc_u ldefint ldisp ldisplay legendre_p legendre_q leinstein length"+" let letrules letsimp levi_civita lfreeof lgtreillis lhs li liediff limit"+" Lindstedt linear linearinterpol linear_program linear_regression line_graph"+" linsolve listarray list_correlations listify list_matrix_entries list_nc_monomials"+" listoftens listofvars listp lmax lmin load loadfile local locate_matrix_entry"+" log logcontract log_gamma lopow lorentz_gauge lowercasep lpart lratsubst"+" lreduce lriemann lsquares_estimates lsquares_estimates_approximate lsquares_estimates_exact"+" lsquares_mse lsquares_residual_mse lsquares_residuals lsum ltreillis lu_backsub"+" lucas lu_factor %m macroexpand macroexpand1 make_array makebox makefact makegamma"+" make_graph make_level_picture makelist makeOrders make_poly_continent make_poly_country"+" make_polygon make_random_state make_rgb_picture makeset make_string_input_stream"+" make_string_output_stream make_transform mandelbrot mandelbrot_set map mapatom"+" maplist matchdeclare matchfix mat_cond mat_fullunblocker mat_function mathml_display"+" mat_norm matrix matrixmap matrixp matrix_size mattrace mat_trace mat_unblocker"+" max max_clique max_degree max_flow maximize_lp max_independent_set max_matching"+" maybe md5sum mean mean_bernoulli mean_beta mean_binomial mean_chi2 mean_continuous_uniform"+" mean_deviation mean_discrete_uniform mean_exp mean_f mean_gamma mean_general_finite_discrete"+" mean_geometric mean_gumbel mean_hypergeometric mean_laplace mean_logistic"+" mean_lognormal mean_negative_binomial mean_noncentral_chi2 mean_noncentral_student_t"+" mean_normal mean_pareto mean_poisson mean_rayleigh mean_student_t mean_weibull"+" median median_deviation member mesh metricexpandall mgf1_sha1 min min_degree"+" min_edge_cut minfactorial minimalPoly minimize_lp minimum_spanning_tree minor"+" minpack_lsquares minpack_solve min_vertex_cover min_vertex_cut mkdir mnewton"+" mod mode_declare mode_identity ModeMatrix moebius mon2schur mono monomial_dimensions"+" multibernstein_poly multi_display_for_texinfo multi_elem multinomial multinomial_coeff"+" multi_orbit multiplot_mode multi_pui multsym multthru mycielski_graph nary"+" natural_unit nc_degree ncexpt ncharpoly negative_picture neighbors new newcontext"+" newdet new_graph newline newton new_variable next_prime nicedummies niceindices"+" ninth nofix nonarray noncentral_moment nonmetricity nonnegintegerp nonscalarp"+" nonzeroandfreeof notequal nounify nptetrad npv nroots nterms ntermst"+" nthroot nullity nullspace num numbered_boundaries numberp number_to_octets"+" num_distinct_partitions numerval numfactor num_partitions nusum nzeta nzetai"+" nzetar octets_to_number octets_to_oid odd_girth oddp ode2 ode_check odelin"+" oid_to_octets op opena opena_binary openr openr_binary openw openw_binary"+" operatorp opsubst optimize %or orbit orbits ordergreat ordergreatp orderless"+" orderlessp orthogonal_complement orthopoly_recur orthopoly_weight outermap"+" out_neighbors outofpois pade parabolic_cylinder_d parametric parametric_surface"+" parg parGosper parse_string parse_timedate part part2cont partfrac partition"+" partition_set partpol path_digraph path_graph pathname_directory pathname_name"+" pathname_type pdf_bernoulli pdf_beta pdf_binomial pdf_cauchy pdf_chi2 pdf_continuous_uniform"+" pdf_discrete_uniform pdf_exp pdf_f pdf_gamma pdf_general_finite_discrete"+" pdf_geometric pdf_gumbel pdf_hypergeometric pdf_laplace pdf_logistic pdf_lognormal"+" pdf_negative_binomial pdf_noncentral_chi2 pdf_noncentral_student_t pdf_normal"+" pdf_pareto pdf_poisson pdf_rank_sum pdf_rayleigh pdf_signed_rank pdf_student_t"+" pdf_weibull pearson_skewness permanent permut permutation permutations petersen_graph"+" petrov pickapart picture_equalp picturep piechart piechart_description planar_embedding"+" playback plog plot2d plot3d plotdf ploteq plsquares pochhammer points poisdiff"+" poisexpt poisint poismap poisplus poissimp poissubst poistimes poistrim polar"+" polarform polartorect polar_to_xy poly_add poly_buchberger poly_buchberger_criterion"+" poly_colon_ideal poly_content polydecomp poly_depends_p poly_elimination_ideal"+" poly_exact_divide poly_expand poly_expt poly_gcd polygon poly_grobner poly_grobner_equal"+" poly_grobner_member poly_grobner_subsetp poly_ideal_intersection poly_ideal_polysaturation"+" poly_ideal_polysaturation1 poly_ideal_saturation poly_ideal_saturation1 poly_lcm"+" poly_minimization polymod poly_multiply polynome2ele polynomialp poly_normal_form"+" poly_normalize poly_normalize_list poly_polysaturation_extension poly_primitive_part"+" poly_pseudo_divide poly_reduced_grobner poly_reduction poly_saturation_extension"+" poly_s_polynomial poly_subtract polytocompanion pop postfix potential power_mod"+" powerseries powerset prefix prev_prime primep primes principal_components"+" print printf printfile print_graph printpois printprops prodrac product properties"+" propvars psi psubst ptriangularize pui pui2comp pui2ele pui2polynome pui_direct"+" puireduc push put pv qput qrange qty quad_control quad_qag quad_qagi quad_qagp"+" quad_qags quad_qawc quad_qawf quad_qawo quad_qaws quadrilateral quantile"+" quantile_bernoulli quantile_beta quantile_binomial quantile_cauchy quantile_chi2"+" quantile_continuous_uniform quantile_discrete_uniform quantile_exp quantile_f"+" quantile_gamma quantile_general_finite_discrete quantile_geometric quantile_gumbel"+" quantile_hypergeometric quantile_laplace quantile_logistic quantile_lognormal"+" quantile_negative_binomial quantile_noncentral_chi2 quantile_noncentral_student_t"+" quantile_normal quantile_pareto quantile_poisson quantile_rayleigh quantile_student_t"+" quantile_weibull quartile_skewness quit qunit quotient racah_v racah_w radcan"+" radius random random_bernoulli random_beta random_binomial random_bipartite_graph"+" random_cauchy random_chi2 random_continuous_uniform random_digraph random_discrete_uniform"+" random_exp random_f random_gamma random_general_finite_discrete random_geometric"+" random_graph random_graph1 random_gumbel random_hypergeometric random_laplace"+" random_logistic random_lognormal random_negative_binomial random_network"+" random_noncentral_chi2 random_noncentral_student_t random_normal random_pareto"+" random_permutation random_poisson random_rayleigh random_regular_graph random_student_t"+" random_tournament random_tree random_weibull range rank rat ratcoef ratdenom"+" ratdiff ratdisrep ratexpand ratinterpol rational rationalize ratnumer ratnump"+" ratp ratsimp ratsubst ratvars ratweight read read_array read_binary_array"+" read_binary_list read_binary_matrix readbyte readchar read_hashed_array readline"+" read_list read_matrix read_nested_list readonly read_xpm real_imagpart_to_conjugate"+" realpart realroots rearray rectangle rectform rectform_log_if_constant recttopolar"+" rediff reduce_consts reduce_order region region_boundaries region_boundaries_plus"+" rem remainder remarray rembox remcomps remcon remcoord remfun remfunction"+" remlet remove remove_constvalue remove_dimensions remove_edge remove_fundamental_dimensions"+" remove_fundamental_units remove_plot_option remove_vertex rempart remrule"+" remsym remvalue rename rename_file reset reset_displays residue resolvante"+" resolvante_alternee1 resolvante_bipartite resolvante_diedrale resolvante_klein"+" resolvante_klein3 resolvante_produit_sym resolvante_unitaire resolvante_vierer"+" rest resultant return reveal reverse revert revert2 rgb2level rhs ricci riemann"+" rinvariant risch rk rmdir rncombine romberg room rootscontract round row"+" rowop rowswap rreduce run_testsuite %s save saving scalarp scaled_bessel_i"+" scaled_bessel_i0 scaled_bessel_i1 scalefactors scanmap scatterplot scatterplot_description"+" scene schur2comp sconcat scopy scsimp scurvature sdowncase sec sech second"+" sequal sequalignore set_alt_display setdifference set_draw_defaults set_edge_weight"+" setelmx setequalp setify setp set_partitions set_plot_option set_prompt set_random_state"+" set_tex_environment set_tex_environment_default setunits setup_autoload set_up_dot_simplifications"+" set_vertex_label seventh sexplode sf sha1sum sha256sum shortest_path shortest_weighted_path"+" show showcomps showratvars sierpinskiale sierpinskimap sign signum similaritytransform"+" simp_inequality simplify_sum simplode simpmetderiv simtran sin sinh sinsert"+" sinvertcase sixth skewness skewness_bernoulli skewness_beta skewness_binomial"+" skewness_chi2 skewness_continuous_uniform skewness_discrete_uniform skewness_exp"+" skewness_f skewness_gamma skewness_general_finite_discrete skewness_geometric"+" skewness_gumbel skewness_hypergeometric skewness_laplace skewness_logistic"+" skewness_lognormal skewness_negative_binomial skewness_noncentral_chi2 skewness_noncentral_student_t"+" skewness_normal skewness_pareto skewness_poisson skewness_rayleigh skewness_student_t"+" skewness_weibull slength smake small_rhombicosidodecahedron_graph small_rhombicuboctahedron_graph"+" smax smin smismatch snowmap snub_cube_graph snub_dodecahedron_graph solve"+" solve_rec solve_rec_rat some somrac sort sparse6_decode sparse6_encode sparse6_export"+" sparse6_import specint spherical spherical_bessel_j spherical_bessel_y spherical_hankel1"+" spherical_hankel2 spherical_harmonic spherical_to_xyz splice split sposition"+" sprint sqfr sqrt sqrtdenest sremove sremovefirst sreverse ssearch ssort sstatus"+" ssubst ssubstfirst staircase standardize standardize_inverse_trig starplot"+" starplot_description status std std1 std_bernoulli std_beta std_binomial"+" std_chi2 std_continuous_uniform std_discrete_uniform std_exp std_f std_gamma"+" std_general_finite_discrete std_geometric std_gumbel std_hypergeometric std_laplace"+" std_logistic std_lognormal std_negative_binomial std_noncentral_chi2 std_noncentral_student_t"+" std_normal std_pareto std_poisson std_rayleigh std_student_t std_weibull"+" stemplot stirling stirling1 stirling2 strim striml strimr string stringout"+" stringp strong_components struve_h struve_l sublis sublist sublist_indices"+" submatrix subsample subset subsetp subst substinpart subst_parallel substpart"+" substring subvar subvarp sum sumcontract summand_to_rec supcase supcontext"+" symbolp symmdifference symmetricp system take_channel take_inference tan"+" tanh taylor taylorinfo taylorp taylor_simplifier taytorat tcl_output tcontract"+" tellrat tellsimp tellsimpafter tentex tenth test_mean test_means_difference"+" test_normality test_proportion test_proportions_difference test_rank_sum"+" test_sign test_signed_rank test_variance test_variance_ratio tex tex1 tex_display"+" texput %th third throw time timedate timer timer_info tldefint tlimit todd_coxeter"+" toeplitz tokens to_lisp topological_sort to_poly to_poly_solve totaldisrep"+" totalfourier totient tpartpol trace tracematrix trace_options transform_sample"+" translate translate_file transpose treefale tree_reduce treillis treinat"+" triangle triangularize trigexpand trigrat trigreduce trigsimp trunc truncate"+" truncated_cube_graph truncated_dodecahedron_graph truncated_icosahedron_graph"+" truncated_tetrahedron_graph tr_warnings_get tube tutte_graph ueivects uforget"+" ultraspherical underlying_graph undiff union unique uniteigenvectors unitp"+" units unit_step unitvector unorder unsum untellrat untimer"+" untrace uppercasep uricci uriemann uvect vandermonde_matrix var var1 var_bernoulli"+" var_beta var_binomial var_chi2 var_continuous_uniform var_discrete_uniform"+" var_exp var_f var_gamma var_general_finite_discrete var_geometric var_gumbel"+" var_hypergeometric var_laplace var_logistic var_lognormal var_negative_binomial"+" var_noncentral_chi2 var_noncentral_student_t var_normal var_pareto var_poisson"+" var_rayleigh var_student_t var_weibull vector vectorpotential vectorsimp"+" verbify vers vertex_coloring vertex_connectivity vertex_degree vertex_distance"+" vertex_eccentricity vertex_in_degree vertex_out_degree vertices vertices_to_cycle"+" vertices_to_path %w weyl wheel_graph wiener_index wigner_3j wigner_6j"+" wigner_9j with_stdout write_binary_data writebyte write_data writefile wronskian"+" xreduce xthru %y Zeilberger zeroequiv zerofor zeromatrix zeromatrixp zeta"+" zgeev zheev zlange zn_add_table zn_carmichael_lambda zn_characteristic_factors"+" zn_determinant zn_factor_generators zn_invert_by_lu zn_log zn_mult_table"+" absboxchar activecontexts adapt_depth additive adim aform algebraic"+" algepsilon algexact aliases allbut all_dotsimp_denoms allocation allsym alphabetic"+" animation antisymmetric arrays askexp assume_pos assume_pos_pred assumescalar"+" asymbol atomgrad atrig1 axes axis_3d axis_bottom axis_left axis_right axis_top"+" azimuth background background_color backsubst berlefact bernstein_explicit"+" besselexpand beta_args_sum_to_integer beta_expand bftorat bftrunc bindtest"+" border boundaries_array box boxchar breakup %c capping cauchysum cbrange"+" cbtics center cflength cframe_flag cnonmet_flag color color_bar color_bar_tics"+" colorbox columns commutative complex cone context contexts contour contour_levels"+" cosnpiflag ctaypov ctaypt ctayswitch ctayvar ct_coords ctorsion_flag ctrgsimp"+" cube current_let_rule_package cylinder data_file_name debugmode decreasing"+" default_let_rule_package delay dependencies derivabbrev derivsubst detout"+" diagmetric diff dim dimensions dispflag display2d|10 display_format_internal"+" distribute_over doallmxops domain domxexpt domxmxops domxnctimes dontfactor"+" doscmxops doscmxplus dot0nscsimp dot0simp dot1simp dotassoc dotconstrules"+" dotdistrib dotexptsimp dotident dotscrules draw_graph_program draw_realpart"+" edge_color edge_coloring edge_partition edge_type edge_width %edispflag"+" elevation %emode endphi endtheta engineering_format_floats enhanced3d %enumer"+" epsilon_lp erfflag erf_representation errormsg error_size error_syms error_type"+" %e_to_numlog eval even evenfun evflag evfun ev_point expandwrt_denom expintexpand"+" expintrep expon expop exptdispflag exptisolate exptsubst facexpand facsum_combine"+" factlim factorflag factorial_expand factors_only fb feature features"+" file_name file_output_append file_search_demo file_search_lisp file_search_maxima|10"+" file_search_tests file_search_usage file_type_lisp file_type_maxima|10 fill_color"+" fill_density filled_func fixed_vertices flipflag float2bf font font_size"+" fortindent fortspaces fpprec fpprintprec functions gamma_expand gammalim"+" gdet genindex gensumnum GGFCFMAX GGFINFINITY globalsolve gnuplot_command"+" gnuplot_curve_styles gnuplot_curve_titles gnuplot_default_term_command gnuplot_dumb_term_command"+" gnuplot_file_args gnuplot_file_name gnuplot_out_file gnuplot_pdf_term_command"+" gnuplot_pm3d gnuplot_png_term_command gnuplot_postamble gnuplot_preamble"+" gnuplot_ps_term_command gnuplot_svg_term_command gnuplot_term gnuplot_view_args"+" Gosper_in_Zeilberger gradefs grid grid2d grind halfangles head_angle head_both"+" head_length head_type height hypergeometric_representation %iargs ibase"+" icc1 icc2 icounter idummyx ieqnprint ifb ifc1 ifc2 ifg ifgi ifr iframe_bracket_form"+" ifri igeowedge_flag ikt1 ikt2 imaginary inchar increasing infeval"+" infinity inflag infolists inm inmc1 inmc2 intanalysis integer integervalued"+" integrate_use_rootsof integration_constant integration_constant_counter interpolate_color"+" intfaclim ip_grid ip_grid_in irrational isolate_wrt_times iterations itr"+" julia_parameter %k1 %k2 keepfloat key key_pos kinvariant kt label label_alignment"+" label_orientation labels lassociative lbfgs_ncorrections lbfgs_nfeval_max"+" leftjust legend letrat let_rule_packages lfg lg lhospitallim limsubst linear"+" linear_solver linechar linel|10 linenum line_type linewidth line_width linsolve_params"+" linsolvewarn lispdisp listarith listconstvars listdummyvars lmxchar load_pathname"+" loadprint logabs logarc logcb logconcoeffp logexpand lognegint logsimp logx"+" logx_secondary logy logy_secondary logz lriem m1pbranch macroexpansion macros"+" mainvar manual_demo maperror mapprint matrix_element_add matrix_element_mult"+" matrix_element_transpose maxapplydepth maxapplyheight maxima_tempdir|10 maxima_userdir|10"+" maxnegex MAX_ORD maxposex maxpsifracdenom maxpsifracnum maxpsinegint maxpsiposint"+" maxtayorder mesh_lines_color method mod_big_prime mode_check_errorp"+" mode_checkp mode_check_warnp mod_test mod_threshold modular_linear_solver"+" modulus multiplicative multiplicities myoptions nary negdistrib negsumdispflag"+" newline newtonepsilon newtonmaxiter nextlayerfactor niceindicespref nm nmc"+" noeval nolabels nonegative_lp noninteger nonscalar noun noundisp nouns np"+" npi nticks ntrig numer numer_pbranch obase odd oddfun opacity opproperties"+" opsubst optimprefix optionset orientation origin orthopoly_returns_intervals"+" outative outchar packagefile palette partswitch pdf_file pfeformat phiresolution"+" %piargs piece pivot_count_sx pivot_max_sx plot_format plot_options plot_realpart"+" png_file pochhammer_max_index points pointsize point_size points_joined point_type"+" poislim poisson poly_coefficient_ring poly_elimination_order polyfactor poly_grobner_algorithm"+" poly_grobner_debug poly_monomial_order poly_primary_elimination_order poly_return_term_list"+" poly_secondary_elimination_order poly_top_reduction_only posfun position"+" powerdisp pred prederror primep_number_of_tests product_use_gamma program"+" programmode promote_float_to_bigfloat prompt proportional_axes props psexpand"+" ps_file radexpand radius radsubstflag rassociative ratalgdenom ratchristof"+" ratdenomdivide rateinstein ratepsilon ratfac rational ratmx ratprint ratriemann"+" ratsimpexpons ratvarswitch ratweights ratweyl ratwtlvl real realonly redraw"+" refcheck resolution restart resultant ric riem rmxchar %rnum_list rombergabs"+" rombergit rombergmin rombergtol rootsconmode rootsepsilon run_viewer same_xy"+" same_xyz savedef savefactors scalar scalarmatrixp scale scale_lp setcheck"+" setcheckbreak setval show_edge_color show_edges show_edge_type show_edge_width"+" show_id show_label showtime show_vertex_color show_vertex_size show_vertex_type"+" show_vertices show_weight simp simplified_output simplify_products simpproduct"+" simpsum sinnpiflag solvedecomposes solveexplicit solvefactors solvenullwarn"+" solveradcan solvetrigwarn space sparse sphere spring_embedding_depth sqrtdispflag"+" stardisp startphi starttheta stats_numer stringdisp structures style sublis_apply_lambda"+" subnumsimp sumexpand sumsplitfact surface surface_hide svg_file symmetric"+" tab taylordepth taylor_logexpand taylor_order_coefficients taylor_truncate_polynomials"+" tensorkill terminal testsuite_files thetaresolution timer_devalue title tlimswitch"+" tr track transcompile transform transform_xy translate_fast_arrays transparent"+" transrun tr_array_as_ref tr_bound_function_applyp tr_file_tty_messagesp tr_float_can_branch_complex"+" tr_function_call_default trigexpandplus trigexpandtimes triginverses trigsign"+" trivial_solutions tr_numer tr_optimize_max_loop tr_semicompile tr_state_vars"+" tr_warn_bad_function_calls tr_warn_fexpr tr_warn_meval tr_warn_mode"+" tr_warn_undeclared tr_warn_undefined_variable tstep ttyoff tube_extremes"+" ufg ug %unitexpand unit_vectors uric uriem use_fast_arrays user_preamble"+" usersetunits values vect_cross verbose vertex_color vertex_coloring vertex_partition"+" vertex_size vertex_type view warnings weyl width windowname windowtitle wired_surface"+" wireframe xaxis xaxis_color xaxis_secondary xaxis_type xaxis_width xlabel"+" xlabel_secondary xlength xrange xrange_secondary xtics xtics_axis xtics_rotate"+" xtics_rotate_secondary xtics_secondary xtics_secondary_axis xu_grid x_voxel"+" xy_file xyplane xy_scale yaxis yaxis_color yaxis_secondary yaxis_type yaxis_width"+" ylabel ylabel_secondary ylength yrange yrange_secondary ytics ytics_axis"+" ytics_rotate ytics_rotate_secondary ytics_secondary ytics_secondary_axis"+" yv_grid y_voxel yx_ratio zaxis zaxis_color zaxis_type zaxis_width zeroa zerob"+" zerobern zeta%pi zlabel zlabel_rotate zlength zmin zn_primroot_limit zn_primroot_pretest";var SYMBOLS="_ __ %|0 %%|0";return{lexemes:"[A-Za-z_%][0-9A-Za-z_%]*",keywords:{keyword:KEYWORDS,literal:LITERALS,built_in:BUILTIN_FUNCTIONS,symbol:SYMBOLS},contains:[{className:"comment",begin:"/\\*",end:"\\*/",contains:["self"]},hljs.QUOTE_STRING_MODE,{className:"number",relevance:0,variants:[{begin:"\\b(\\d+|\\d+\\.|\\.\\d+|\\d+\\.\\d+)[Ee][-+]?\\d+\\b"},{begin:"\\b(\\d+|\\d+\\.|\\.\\d+|\\d+\\.\\d+)[Bb][-+]?\\d+\\b",relevance:10},{begin:"\\b(\\.\\d+|\\d+\\.\\d+)\\b"},{begin:"\\b(\\d+|0[0-9A-Za-z]+)\\.?\\b"}]}],illegal:/@/}});hljs.registerLanguage("1c",function(hljs){var UNDERSCORE_IDENT_RE="[A-Za-zА-Яа-яёЁ_][A-Za-zА-Яа-яёЁ_0-9]+";var v7_keywords="далее ";var v8_keywords="возврат вызватьисключение выполнить для если и из или иначе иначеесли исключение каждого конецесли "+"конецпопытки конеццикла не новый перейти перем по пока попытка прервать продолжить тогда цикл экспорт ";var KEYWORD=v7_keywords+v8_keywords;var v7_meta_keywords="загрузитьизфайла ";var v8_meta_keywords="вебклиент вместо внешнеесоединение клиент конецобласти мобильноеприложениеклиент мобильноеприложениесервер "+"наклиенте наклиентенасервере наклиентенасерверебезконтекста насервере насерверебезконтекста область перед "+"после сервер толстыйклиентобычноеприложение толстыйклиентуправляемоеприложение тонкийклиент ";var METAKEYWORD=v7_meta_keywords+v8_meta_keywords;var v7_system_constants="разделительстраниц разделительстрок символтабуляции ";var v7_global_context_methods="ansitooem oemtoansi ввестивидсубконто ввестиперечисление ввестипериод ввестиплансчетов выбранныйплансчетов "+"датагод датамесяц датачисло заголовоксистемы значениевстроку значениеизстроки каталогиб каталогпользователя "+"кодсимв конгода конецпериодаби конецрассчитанногопериодаби конецстандартногоинтервала конквартала конмесяца "+"коннедели лог лог10 максимальноеколичествосубконто названиеинтерфейса названиенабораправ назначитьвид "+"назначитьсчет найтиссылки началопериодаби началостандартногоинтервала начгода начквартала начмесяца "+"начнедели номерднягода номерднянедели номернеделигода обработкаожидания основнойжурналрасчетов "+"основнойплансчетов основнойязык очиститьокносообщений периодстр получитьвремята получитьдатута "+"получитьдокументта получитьзначенияотбора получитьпозициюта получитьпустоезначение получитьта "+"префиксавтонумерации пропись пустоезначение разм разобратьпозициюдокумента рассчитатьрегистрына "+"рассчитатьрегистрыпо симв создатьобъект статусвозврата стрколичествострок сформироватьпозициюдокумента "+"счетпокоду текущеевремя типзначения типзначениястр установитьтана установитьтапо фиксшаблон шаблон ";var v8_global_context_methods="acos asin atan base64значение base64строка cos exp log log10 pow sin sqrt tan xmlзначение xmlстрока "+"xmlтип xmlтипзнч активноеокно безопасныйрежим безопасныйрежимразделенияданных булево ввестидату ввестизначение "+"ввестистроку ввестичисло возможностьчтенияxml вопрос восстановитьзначение врег выгрузитьжурналрегистрации "+"выполнитьобработкуоповещения выполнитьпроверкуправдоступа вычислить год данныеформывзначение дата день деньгода "+"деньнедели добавитьмесяц заблокироватьданныедляредактирования заблокироватьработупользователя завершитьработусистемы "+"загрузитьвнешнююкомпоненту закрытьсправку записатьjson записатьxml записатьдатуjson записьжурналарегистрации "+"заполнитьзначениясвойств запроситьразрешениепользователя запуститьприложение запуститьсистему зафиксироватьтранзакцию "+"значениевданныеформы значениевстрокувнутр значениевфайл значениезаполнено значениеизстрокивнутр значениеизфайла "+"изxmlтипа импортмоделиxdto имякомпьютера имяпользователя инициализироватьпредопределенныеданные информацияобошибке "+"каталогбиблиотекимобильногоустройства каталогвременныхфайлов каталогдокументов каталогпрограммы кодироватьстроку "+"кодлокализацииинформационнойбазы кодсимвола командасистемы конецгода конецдня конецквартала конецмесяца конецминуты "+"конецнедели конецчаса конфигурациябазыданныхизмененадинамически конфигурацияизменена копироватьданныеформы "+"копироватьфайл краткоепредставлениеошибки лев макс местноевремя месяц мин минута монопольныйрежим найти "+"найтинедопустимыесимволыxml найтиокнопонавигационнойссылке найтипомеченныенаудаление найтипоссылкам найтифайлы "+"началогода началодня началоквартала началомесяца началоминуты началонедели началочаса начатьзапросразрешенияпользователя "+"начатьзапускприложения начатькопированиефайла начатьперемещениефайла начатьподключениевнешнейкомпоненты "+"начатьподключениерасширенияработыскриптографией начатьподключениерасширенияработысфайлами начатьпоискфайлов "+"начатьполучениекаталогавременныхфайлов начатьполучениекаталогадокументов начатьполучениерабочегокаталогаданныхпользователя "+"начатьполучениефайлов начатьпомещениефайла начатьпомещениефайлов начатьсозданиедвоичныхданныхизфайла начатьсозданиекаталога "+"начатьтранзакцию начатьудалениефайлов начатьустановкувнешнейкомпоненты начатьустановкурасширенияработыскриптографией "+"начатьустановкурасширенияработысфайлами неделягода необходимостьзавершениясоединения номерсеансаинформационнойбазы "+"номерсоединенияинформационнойбазы нрег нстр обновитьинтерфейс обновитьнумерациюобъектов обновитьповторноиспользуемыезначения "+"обработкапрерыванияпользователя объединитьфайлы окр описаниеошибки оповестить оповеститьобизменении "+"отключитьобработчикзапросанастроекклиенталицензирования отключитьобработчикожидания отключитьобработчикоповещения "+"открытьзначение открытьиндекссправки открытьсодержаниесправки открытьсправку открытьформу открытьформумодально "+"отменитьтранзакцию очиститьжурналрегистрации очиститьнастройкипользователя очиститьсообщения параметрыдоступа "+"перейтипонавигационнойссылке переместитьфайл подключитьвнешнююкомпоненту "+"подключитьобработчикзапросанастроекклиенталицензирования подключитьобработчикожидания подключитьобработчикоповещения "+"подключитьрасширениеработыскриптографией подключитьрасширениеработысфайлами подробноепредставлениеошибки "+"показатьвводдаты показатьвводзначения показатьвводстроки показатьвводчисла показатьвопрос показатьзначение "+"показатьинформациюобошибке показатьнакарте показатьоповещениепользователя показатьпредупреждение полноеимяпользователя "+"получитьcomобъект получитьxmlтип получитьадреспоместоположению получитьблокировкусеансов получитьвремязавершенияспящегосеанса "+"получитьвремязасыпанияпассивногосеанса получитьвремяожиданияблокировкиданных получитьданныевыбора "+"получитьдополнительныйпараметрклиенталицензирования получитьдопустимыекодылокализации получитьдопустимыечасовыепояса "+"получитьзаголовокклиентскогоприложения получитьзаголовоксистемы получитьзначенияотборажурналарегистрации "+"получитьидентификаторконфигурации получитьизвременногохранилища получитьимявременногофайла "+"получитьимяклиенталицензирования получитьинформациюэкрановклиента получитьиспользованиежурналарегистрации "+"получитьиспользованиесобытияжурналарегистрации получитькраткийзаголовокприложения получитьмакетоформления "+"получитьмаскувсефайлы получитьмаскувсефайлыклиента получитьмаскувсефайлысервера получитьместоположениепоадресу "+"получитьминимальнуюдлинупаролейпользователей получитьнавигационнуюссылку получитьнавигационнуюссылкуинформационнойбазы "+"получитьобновлениеконфигурациибазыданных получитьобновлениепредопределенныхданныхинформационнойбазы получитьобщиймакет "+"получитьобщуюформу получитьокна получитьоперативнуюотметкувремени получитьотключениебезопасногорежима "+"получитьпараметрыфункциональныхопцийинтерфейса получитьполноеимяпредопределенногозначения "+"получитьпредставлениянавигационныхссылок получитьпроверкусложностипаролейпользователей получитьразделительпути "+"получитьразделительпутиклиента получитьразделительпутисервера получитьсеансыинформационнойбазы "+"получитьскоростьклиентскогосоединения получитьсоединенияинформационнойбазы получитьсообщенияпользователю "+"получитьсоответствиеобъектаиформы получитьсоставстандартногоинтерфейсаodata получитьструктурухранениябазыданных "+"получитьтекущийсеансинформационнойбазы получитьфайл получитьфайлы получитьформу получитьфункциональнуюопцию "+"получитьфункциональнуюопциюинтерфейса получитьчасовойпоясинформационнойбазы пользователиос поместитьвовременноехранилище "+"поместитьфайл поместитьфайлы прав праводоступа предопределенноезначение представлениекодалокализации представлениепериода "+"представлениеправа представлениеприложения представлениесобытияжурналарегистрации представлениечасовогопояса предупреждение "+"прекратитьработусистемы привилегированныйрежим продолжитьвызов прочитатьjson прочитатьxml прочитатьдатуjson пустаястрока "+"рабочийкаталогданныхпользователя разблокироватьданныедляредактирования разделитьфайл разорватьсоединениесвнешнимисточникомданных "+"раскодироватьстроку рольдоступна секунда сигнал символ скопироватьжурналрегистрации смещениелетнеговремени "+"смещениестандартноговремени соединитьбуферыдвоичныхданных создатькаталог создатьфабрикуxdto сокрл сокрлп сокрп сообщить "+"состояние сохранитьзначение сохранитьнастройкипользователя сред стрдлина стрзаканчиваетсяна стрзаменить стрнайти стрначинаетсяс "+"строка строкасоединенияинформационнойбазы стрполучитьстроку стрразделить стрсоединить стрсравнить стрчисловхождений "+"стрчислострок стршаблон текущаядата текущаядатасеанса текущаяуниверсальнаядата текущаяуниверсальнаядатавмиллисекундах "+"текущийвариантинтерфейсаклиентскогоприложения текущийвариантосновногошрифтаклиентскогоприложения текущийкодлокализации "+"текущийрежимзапуска текущийязык текущийязыксистемы тип типзнч транзакцияактивна трег удалитьданныеинформационнойбазы "+"удалитьизвременногохранилища удалитьобъекты удалитьфайлы универсальноевремя установитьбезопасныйрежим "+"установитьбезопасныйрежимразделенияданных установитьблокировкусеансов установитьвнешнююкомпоненту "+"установитьвремязавершенияспящегосеанса установитьвремязасыпанияпассивногосеанса установитьвремяожиданияблокировкиданных "+"установитьзаголовокклиентскогоприложения установитьзаголовоксистемы установитьиспользованиежурналарегистрации "+"установитьиспользованиесобытияжурналарегистрации установитькраткийзаголовокприложения "+"установитьминимальнуюдлинупаролейпользователей установитьмонопольныйрежим установитьнастройкиклиенталицензирования "+"установитьобновлениепредопределенныхданныхинформационнойбазы установитьотключениебезопасногорежима "+"установитьпараметрыфункциональныхопцийинтерфейса установитьпривилегированныйрежим "+"установитьпроверкусложностипаролейпользователей установитьрасширениеработыскриптографией "+"установитьрасширениеработысфайлами установитьсоединениесвнешнимисточникомданных установитьсоответствиеобъектаиформы "+"установитьсоставстандартногоинтерфейсаodata установитьчасовойпоясинформационнойбазы установитьчасовойпояссеанса "+"формат цел час часовойпояс часовойпояссеанса число числопрописью этоадресвременногохранилища ";var v8_global_context_property="wsссылки библиотекакартинок библиотекамакетовоформлениякомпоновкиданных библиотекастилей бизнеспроцессы "+"внешниеисточникиданных внешниеобработки внешниеотчеты встроенныепокупки главныйинтерфейс главныйстиль "+"документы доставляемыеуведомления журналыдокументов задачи информацияобинтернетсоединении использованиерабочейдаты "+"историяработыпользователя константы критерииотбора метаданные обработки отображениерекламы отправкадоставляемыхуведомлений "+"отчеты панельзадачос параметрзапуска параметрысеанса перечисления планывидоврасчета планывидовхарактеристик "+"планыобмена планысчетов полнотекстовыйпоиск пользователиинформационнойбазы последовательности проверкавстроенныхпокупок "+"рабочаядата расширенияконфигурации регистрыбухгалтерии регистрынакопления регистрырасчета регистрысведений "+"регламентныезадания сериализаторxdto справочники средствагеопозиционирования средствакриптографии средствамультимедиа "+"средстваотображениярекламы средствапочты средствателефонии фабрикаxdto файловыепотоки фоновыезадания хранилищанастроек "+"хранилищевариантовотчетов хранилищенастроекданныхформ хранилищеобщихнастроек хранилищепользовательскихнастроекдинамическихсписков "+"хранилищепользовательскихнастроекотчетов хранилищесистемныхнастроек ";var BUILTIN=v7_system_constants+v7_global_context_methods+v8_global_context_methods+v8_global_context_property;var v8_system_sets_of_values="webцвета windowsцвета windowsшрифты библиотекакартинок рамкистиля символы цветастиля шрифтыстиля ";var v8_system_enums_interface="автоматическоесохранениеданныхформывнастройках автонумерациявформе автораздвижениесерий "+"анимациядиаграммы вариантвыравниванияэлементовизаголовков вариантуправлениявысотойтаблицы "+"вертикальнаяпрокруткаформы вертикальноеположение вертикальноеположениеэлемента видгруппыформы "+"виддекорацииформы виддополненияэлементаформы видизмененияданных видкнопкиформы видпереключателя "+"видподписейкдиаграмме видполяформы видфлажка влияниеразмеранапузырекдиаграммы горизонтальноеположение "+"горизонтальноеположениеэлемента группировкаколонок группировкаподчиненныхэлементовформы "+"группыиэлементы действиеперетаскивания дополнительныйрежимотображения допустимыедействияперетаскивания "+"интервалмеждуэлементамиформы использованиевывода использованиеполосыпрокрутки "+"используемоезначениеточкибиржевойдиаграммы историявыборапривводе источникзначенийоситочекдиаграммы "+"источникзначенияразмерапузырькадиаграммы категориягруппыкоманд максимумсерий начальноеотображениедерева "+"начальноеотображениесписка обновлениетекстаредактирования ориентациядендрограммы ориентациядиаграммы "+"ориентацияметокдиаграммы ориентацияметоксводнойдиаграммы ориентацияэлементаформы отображениевдиаграмме "+"отображениевлегендедиаграммы отображениегруппыкнопок отображениезаголовкашкалыдиаграммы "+"отображениезначенийсводнойдиаграммы отображениезначенияизмерительнойдиаграммы "+"отображениеинтерваладиаграммыганта отображениекнопки отображениекнопкивыбора отображениеобсужденийформы "+"отображениеобычнойгруппы отображениеотрицательныхзначенийпузырьковойдиаграммы отображениепанелипоиска "+"отображениеподсказки отображениепредупрежденияприредактировании отображениеразметкиполосырегулирования "+"отображениестраницформы отображениетаблицы отображениетекстазначениядиаграммыганта "+"отображениеуправленияобычнойгруппы отображениефигурыкнопки палитрацветовдиаграммы поведениеобычнойгруппы "+"поддержкамасштабадендрограммы поддержкамасштабадиаграммыганта поддержкамасштабасводнойдиаграммы "+"поисквтаблицепривводе положениезаголовкаэлементаформы положениекартинкикнопкиформы "+"положениекартинкиэлементаграфическойсхемы положениекоманднойпанелиформы положениекоманднойпанелиэлементаформы "+"положениеопорнойточкиотрисовки положениеподписейкдиаграмме положениеподписейшкалызначенийизмерительнойдиаграммы "+"положениесостоянияпросмотра положениестрокипоиска положениетекстасоединительнойлинии положениеуправленияпоиском "+"положениешкалывремени порядокотображенияточекгоризонтальнойгистограммы порядоксерийвлегендедиаграммы "+"размеркартинки расположениезаголовкашкалыдиаграммы растягиваниеповертикалидиаграммыганта "+"режимавтоотображениясостояния режимвводастроктаблицы режимвыборанезаполненного режимвыделениядаты "+"режимвыделениястрокитаблицы режимвыделениятаблицы режимизмененияразмера режимизменениясвязанногозначения "+"режимиспользованиядиалогапечати режимиспользованияпараметракоманды режиммасштабированияпросмотра "+"режимосновногоокнаклиентскогоприложения режимоткрытияокнаформы режимотображениявыделения "+"режимотображениягеографическойсхемы режимотображениязначенийсерии режимотрисовкисеткиграфическойсхемы "+"режимполупрозрачностидиаграммы режимпробеловдиаграммы режимразмещениянастранице режимредактированияколонки "+"режимсглаживаниядиаграммы режимсглаживанияиндикатора режимсписказадач сквозноевыравнивание "+"сохранениеданныхформывнастройках способзаполнениятекстазаголовкашкалыдиаграммы "+"способопределенияограничивающегозначениядиаграммы стандартнаягруппакоманд стандартноеоформление "+"статусоповещенияпользователя стильстрелки типаппроксимациилиниитрендадиаграммы типдиаграммы "+"типединицышкалывремени типимпортасерийслоягеографическойсхемы типлиниигеографическойсхемы типлиниидиаграммы "+"типмаркерагеографическойсхемы типмаркерадиаграммы типобластиоформления "+"типорганизацииисточникаданныхгеографическойсхемы типотображениясериислоягеографическойсхемы "+"типотображенияточечногообъектагеографическойсхемы типотображенияшкалыэлементалегендыгеографическойсхемы "+"типпоискаобъектовгеографическойсхемы типпроекциигеографическойсхемы типразмещенияизмерений "+"типразмещенияреквизитовизмерений типрамкиэлементауправления типсводнойдиаграммы "+"типсвязидиаграммыганта типсоединениязначенийпосериямдиаграммы типсоединенияточекдиаграммы "+"типсоединительнойлинии типстороныэлементаграфическойсхемы типформыотчета типшкалырадарнойдиаграммы "+"факторлиниитрендадиаграммы фигуракнопки фигурыграфическойсхемы фиксациявтаблице форматдняшкалывремени "+"форматкартинки ширинаподчиненныхэлементовформы ";var v8_system_enums_objects_properties="виддвижениябухгалтерии виддвижениянакопления видпериодарегистрарасчета видсчета видточкимаршрутабизнеспроцесса "+"использованиеагрегатарегистранакопления использованиегруппиэлементов использованиережимапроведения "+"использованиесреза периодичностьагрегатарегистранакопления режимавтовремя режимзаписидокумента режимпроведениядокумента ";var v8_system_enums_exchange_plans="авторегистрацияизменений допустимыйномерсообщения отправкаэлементаданных получениеэлементаданных ";var v8_system_enums_tabular_document="использованиерасшифровкитабличногодокумента ориентациястраницы положениеитоговколоноксводнойтаблицы "+"положениеитоговстроксводнойтаблицы положениетекстаотносительнокартинки расположениезаголовкагруппировкитабличногодокумента "+"способчтениязначенийтабличногодокумента типдвустороннейпечати типзаполненияобластитабличногодокумента "+"типкурсоровтабличногодокумента типлиниирисункатабличногодокумента типлинииячейкитабличногодокумента "+"типнаправленияпереходатабличногодокумента типотображениявыделениятабличногодокумента типотображениялинийсводнойтаблицы "+"типразмещениятекстатабличногодокумента типрисункатабличногодокумента типсмещениятабличногодокумента "+"типузоратабличногодокумента типфайлатабличногодокумента точностьпечати чередованиерасположениястраниц ";var v8_system_enums_sheduler="отображениевремениэлементовпланировщика ";var v8_system_enums_formatted_document="типфайлаформатированногодокумента ";var v8_system_enums_query="обходрезультатазапроса типзаписизапроса ";var v8_system_enums_report_builder="видзаполнениярасшифровкипостроителяотчета типдобавленияпредставлений типизмеренияпостроителяотчета типразмещенияитогов ";var v8_system_enums_files="доступкфайлу режимдиалогавыборафайла режимоткрытияфайла ";var v8_system_enums_query_builder="типизмеренияпостроителязапроса ";var v8_system_enums_data_analysis="видданныханализа методкластеризации типединицыинтервалавременианализаданных типзаполнениятаблицырезультатаанализаданных "+"типиспользованиячисловыхзначенийанализаданных типисточникаданныхпоискаассоциаций типколонкианализаданныхдереворешений "+"типколонкианализаданныхкластеризация типколонкианализаданныхобщаястатистика типколонкианализаданныхпоискассоциаций "+"типколонкианализаданныхпоискпоследовательностей типколонкимоделипрогноза типмерырасстоянияанализаданных "+"типотсеченияправилассоциации типполяанализаданных типстандартизациианализаданных типупорядочиванияправилассоциациианализаданных "+"типупорядочиванияшаблоновпоследовательностейанализаданных типупрощениядереварешений ";var v8_system_enums_xml_json_xs_dom_xdto_ws="wsнаправлениепараметра вариантxpathxs вариантзаписидатыjson вариантпростоготипаxs видгруппымоделиxs видфасетаxdto "+"действиепостроителяdom завершенностьпростоготипаxs завершенностьсоставноготипаxs завершенностьсхемыxs запрещенныеподстановкиxs "+"исключениягруппподстановкиxs категорияиспользованияатрибутаxs категорияограниченияидентичностиxs категорияограниченияпространствименxs "+"методнаследованияxs модельсодержимогоxs назначениетипаxml недопустимыеподстановкиxs обработкапробельныхсимволовxs обработкасодержимогоxs "+"ограничениезначенияxs параметрыотбораузловdom переносстрокjson позициявдокументеdom пробельныесимволыxml типатрибутаxml типзначенияjson "+"типканоническогоxml типкомпонентыxs типпроверкиxml типрезультатаdomxpath типузлаdom типузлаxml формаxml формапредставленияxs "+"форматдатыjson экранированиесимволовjson ";var v8_system_enums_data_composition_system="видсравнениякомпоновкиданных действиеобработкирасшифровкикомпоновкиданных направлениесортировкикомпоновкиданных "+"расположениевложенныхэлементоврезультатакомпоновкиданных расположениеитоговкомпоновкиданных расположениегруппировкикомпоновкиданных "+"расположениеполейгруппировкикомпоновкиданных расположениеполякомпоновкиданных расположениереквизитовкомпоновкиданных "+"расположениересурсовкомпоновкиданных типбухгалтерскогоостаткакомпоновкиданных типвыводатекстакомпоновкиданных "+"типгруппировкикомпоновкиданных типгруппыэлементовотборакомпоновкиданных типдополненияпериодакомпоновкиданных "+"типзаголовкаполейкомпоновкиданных типмакетагруппировкикомпоновкиданных типмакетаобластикомпоновкиданных типостаткакомпоновкиданных "+"типпериодакомпоновкиданных типразмещениятекстакомпоновкиданных типсвязинаборовданныхкомпоновкиданных типэлементарезультатакомпоновкиданных "+"расположениелегендыдиаграммыкомпоновкиданных типпримененияотборакомпоновкиданных режимотображенияэлементанастройкикомпоновкиданных "+"режимотображениянастроеккомпоновкиданных состояниеэлементанастройкикомпоновкиданных способвосстановлениянастроеккомпоновкиданных "+"режимкомпоновкирезультата использованиепараметракомпоновкиданных автопозицияресурсовкомпоновкиданных "+"вариантиспользованиягруппировкикомпоновкиданных расположениересурсоввдиаграммекомпоновкиданных фиксациякомпоновкиданных "+"использованиеусловногооформлениякомпоновкиданных ";var v8_system_enums_email="важностьинтернетпочтовогосообщения обработкатекстаинтернетпочтовогосообщения способкодированияинтернетпочтовоговложения "+"способкодированиянеasciiсимволовинтернетпочтовогосообщения типтекстапочтовогосообщения протоколинтернетпочты "+"статусразборапочтовогосообщения ";var v8_system_enums_logbook="режимтранзакциизаписижурналарегистрации статустранзакциизаписижурналарегистрации уровеньжурналарегистрации ";var v8_system_enums_cryptography="расположениехранилищасертификатовкриптографии режимвключениясертификатовкриптографии режимпроверкисертификатакриптографии "+"типхранилищасертификатовкриптографии ";var v8_system_enums_zip="кодировкаименфайловвzipфайле методсжатияzip методшифрованияzip режимвосстановленияпутейфайловzip режимобработкиподкаталоговzip "+"режимсохраненияпутейzip уровеньсжатияzip ";var v8_system_enums_other="звуковоеоповещение направлениепереходакстроке позициявпотоке порядокбайтов режимблокировкиданных режимуправленияблокировкойданных "+"сервисвстроенныхпокупок состояниефоновогозадания типподписчикадоставляемыхуведомлений уровеньиспользованиязащищенногосоединенияftp ";var v8_system_enums_request_schema="направлениепорядкасхемызапроса типдополненияпериодамисхемызапроса типконтрольнойточкисхемызапроса типобъединениясхемызапроса "+"типпараметрадоступнойтаблицысхемызапроса типсоединениясхемызапроса ";var v8_system_enums_properties_of_metadata_objects="httpметод автоиспользованиеобщегореквизита автопрефиксномеразадачи вариантвстроенногоязыка видиерархии видрегистранакопления "+"видтаблицывнешнегоисточникаданных записьдвиженийприпроведении заполнениепоследовательностей индексирование "+"использованиебазыпланавидоврасчета использованиебыстроговыбора использованиеобщегореквизита использованиеподчинения "+"использованиеполнотекстовогопоиска использованиеразделяемыхданныхобщегореквизита использованиереквизита "+"назначениеиспользованияприложения назначениерасширенияконфигурации направлениепередачи обновлениепредопределенныхданных "+"оперативноепроведение основноепредставлениевидарасчета основноепредставлениевидахарактеристики основноепредставлениезадачи "+"основноепредставлениепланаобмена основноепредставлениесправочника основноепредставлениесчета перемещениеграницыприпроведении "+"периодичностьномерабизнеспроцесса периодичностьномерадокумента периодичностьрегистрарасчета периодичностьрегистрасведений "+"повторноеиспользованиевозвращаемыхзначений полнотекстовыйпоискпривводепостроке принадлежностьобъекта проведение "+"разделениеаутентификацииобщегореквизита разделениеданныхобщегореквизита разделениерасширенийконфигурацииобщегореквизита "+"режимавтонумерацииобъектов режимзаписирегистра режимиспользованиямодальности "+"режимиспользованиясинхронныхвызововрасширенийплатформыивнешнихкомпонент режимповторногоиспользованиясеансов "+"режимполученияданныхвыборапривводепостроке режимсовместимости режимсовместимостиинтерфейса "+"режимуправленияблокировкойданныхпоумолчанию сериикодовпланавидовхарактеристик сериикодовпланасчетов "+"сериикодовсправочника созданиепривводе способвыбора способпоискастрокипривводепостроке способредактирования "+"типданныхтаблицывнешнегоисточникаданных типкодапланавидоврасчета типкодасправочника типмакета типномерабизнеспроцесса "+"типномерадокумента типномеразадачи типформы удалениедвижений ";var v8_system_enums_differents="важностьпроблемыприменениярасширенияконфигурации вариантинтерфейсаклиентскогоприложения вариантмасштабаформклиентскогоприложения "+"вариантосновногошрифтаклиентскогоприложения вариантстандартногопериода вариантстандартнойдатыначала видграницы видкартинки "+"видотображенияполнотекстовогопоиска видрамки видсравнения видцвета видчисловогозначения видшрифта допустимаядлина допустимыйзнак "+"использованиеbyteordermark использованиеметаданныхполнотекстовогопоиска источникрасширенийконфигурации клавиша кодвозвратадиалога "+"кодировкаxbase кодировкатекста направлениепоиска направлениесортировки обновлениепредопределенныхданных обновлениеприизмененииданных "+"отображениепанелиразделов проверказаполнения режимдиалогавопрос режимзапускаклиентскогоприложения режимокругления режимоткрытияформприложения "+"режимполнотекстовогопоиска скоростьклиентскогосоединения состояниевнешнегоисточникаданных состояниеобновленияконфигурациибазыданных "+"способвыборасертификатаwindows способкодированиястроки статуссообщения типвнешнейкомпоненты типплатформы типповеденияклавишиenter "+"типэлементаинформацииовыполненииобновленияконфигурациибазыданных уровеньизоляциитранзакций хешфункция частидаты";var CLASS=v8_system_sets_of_values+v8_system_enums_interface+v8_system_enums_objects_properties+v8_system_enums_exchange_plans+v8_system_enums_tabular_document+v8_system_enums_sheduler+v8_system_enums_formatted_document+v8_system_enums_query+v8_system_enums_report_builder+v8_system_enums_files+v8_system_enums_query_builder+v8_system_enums_data_analysis+v8_system_enums_xml_json_xs_dom_xdto_ws+v8_system_enums_data_composition_system+v8_system_enums_email+v8_system_enums_logbook+v8_system_enums_cryptography+v8_system_enums_zip+v8_system_enums_other+v8_system_enums_request_schema+v8_system_enums_properties_of_metadata_objects+v8_system_enums_differents;var v8_shared_object="comобъект ftpсоединение httpзапрос httpсервисответ httpсоединение wsопределения wsпрокси xbase анализданных аннотацияxs "+"блокировкаданных буфердвоичныхданных включениеxs выражениекомпоновкиданных генераторслучайныхчисел географическаясхема "+"географическиекоординаты графическаясхема группамоделиxs данныерасшифровкикомпоновкиданных двоичныеданные дендрограмма "+"диаграмма диаграммаганта диалогвыборафайла диалогвыборацвета диалогвыборашрифта диалограсписаниярегламентногозадания "+"диалогредактированиястандартногопериода диапазон документdom документhtml документацияxs доставляемоеуведомление "+"записьdom записьfastinfoset записьhtml записьjson записьxml записьzipфайла записьданных записьтекста записьузловdom "+"запрос защищенноесоединениеopenssl значенияполейрасшифровкикомпоновкиданных извлечениетекста импортxs интернетпочта "+"интернетпочтовоесообщение интернетпочтовыйпрофиль интернетпрокси интернетсоединение информациядляприложенияxs "+"использованиеатрибутаxs использованиесобытияжурналарегистрации источникдоступныхнастроеккомпоновкиданных "+"итераторузловdom картинка квалификаторыдаты квалификаторыдвоичныхданных квалификаторыстроки квалификаторычисла "+"компоновщикмакетакомпоновкиданных компоновщикнастроеккомпоновкиданных конструктормакетаоформлениякомпоновкиданных "+"конструкторнастроеккомпоновкиданных конструкторформатнойстроки линия макеткомпоновкиданных макетобластикомпоновкиданных "+"макетоформлениякомпоновкиданных маскаxs менеджеркриптографии наборсхемxml настройкикомпоновкиданных настройкисериализацииjson "+"обработкакартинок обработкарасшифровкикомпоновкиданных обходдереваdom объявлениеатрибутаxs объявлениенотацииxs "+"объявлениеэлементаxs описаниеиспользованиясобытиядоступжурналарегистрации "+"описаниеиспользованиясобытияотказвдоступежурналарегистрации описаниеобработкирасшифровкикомпоновкиданных "+"описаниепередаваемогофайла описаниетипов определениегруппыатрибутовxs определениегруппымоделиxs "+"определениеограниченияидентичностиxs определениепростоготипаxs определениесоставноготипаxs определениетипадокументаdom "+"определенияxpathxs отборкомпоновкиданных пакетотображаемыхдокументов параметрвыбора параметркомпоновкиданных "+"параметрызаписиjson параметрызаписиxml параметрычтенияxml переопределениеxs планировщик полеанализаданных "+"полекомпоновкиданных построительdom построительзапроса построительотчета построительотчетаанализаданных "+"построительсхемxml поток потоквпамяти почта почтовоесообщение преобразованиеxsl преобразованиекканоническомуxml "+"процессорвыводарезультатакомпоновкиданныхвколлекциюзначений процессорвыводарезультатакомпоновкиданныхвтабличныйдокумент "+"процессоркомпоновкиданных разыменовательпространствименdom рамка расписаниерегламентногозадания расширенноеимяxml "+"результатчтенияданных своднаядиаграмма связьпараметравыбора связьпотипу связьпотипукомпоновкиданных сериализаторxdto "+"сертификатклиентаwindows сертификатклиентафайл сертификаткриптографии сертификатыудостоверяющихцентровwindows "+"сертификатыудостоверяющихцентровфайл сжатиеданных системнаяинформация сообщениепользователю сочетаниеклавиш "+"сравнениезначений стандартнаядатаначала стандартныйпериод схемаxml схемакомпоновкиданных табличныйдокумент "+"текстовыйдокумент тестируемоеприложение типданныхxml уникальныйидентификатор фабрикаxdto файл файловыйпоток "+"фасетдлиныxs фасетколичестваразрядовдробнойчастиxs фасетмаксимальноговключающегозначенияxs "+"фасетмаксимальногоисключающегозначенияxs фасетмаксимальнойдлиныxs фасетминимальноговключающегозначенияxs "+"фасетминимальногоисключающегозначенияxs фасетминимальнойдлиныxs фасетобразцаxs фасетобщегоколичестваразрядовxs "+"фасетперечисленияxs фасетпробельныхсимволовxs фильтрузловdom форматированнаястрока форматированныйдокумент "+"фрагментxs хешированиеданных хранилищезначения цвет чтениеfastinfoset чтениеhtml чтениеjson чтениеxml чтениеzipфайла "+"чтениеданных чтениетекста чтениеузловdom шрифт элементрезультатакомпоновкиданных ";var v8_universal_collection="comsafearray деревозначений массив соответствие списокзначений структура таблицазначений фиксированнаяструктура "+"фиксированноесоответствие фиксированныймассив ";var TYPE=v8_shared_object+v8_universal_collection;var LITERAL="null истина ложь неопределено";var NUMBERS=hljs.inherit(hljs.NUMBER_MODE);var STRINGS={className:"string",begin:'"|\\|',end:'"|$',contains:[{begin:'""'}]};var DATE={begin:"'",end:"'",excludeBegin:true,excludeEnd:true,contains:[{className:"number",begin:"\\d{4}([\\.\\\\/:-]?\\d{2}){0,5}"}]};var COMMENTS=hljs.inherit(hljs.C_LINE_COMMENT_MODE);var META={className:"meta",lexemes:UNDERSCORE_IDENT_RE,begin:"#|&",end:"$",keywords:{"meta-keyword":KEYWORD+METAKEYWORD},contains:[COMMENTS]};var SYMBOL={className:"symbol",begin:"~",end:";|:",excludeEnd:true};var FUNCTION={className:"function",lexemes:UNDERSCORE_IDENT_RE,variants:[{begin:"процедура|функция",end:"\\)",keywords:"процедура функция"},{begin:"конецпроцедуры|конецфункции",keywords:"конецпроцедуры конецфункции"}],contains:[{begin:"\\(",end:"\\)",endsParent:true,contains:[{className:"params",lexemes:UNDERSCORE_IDENT_RE,begin:UNDERSCORE_IDENT_RE,end:",",excludeEnd:true,endsWithParent:true,keywords:{keyword:"знач",literal:LITERAL},contains:[NUMBERS,STRINGS,DATE]},COMMENTS]},hljs.inherit(hljs.TITLE_MODE,{begin:UNDERSCORE_IDENT_RE})]};return{case_insensitive:true,lexemes:UNDERSCORE_IDENT_RE,keywords:{keyword:KEYWORD,built_in:BUILTIN,class:CLASS,type:TYPE,literal:LITERAL},contains:[META,FUNCTION,COMMENTS,SYMBOL,NUMBERS,STRINGS,DATE]}});hljs.registerLanguage("asciidoc",function(hljs){return{aliases:["adoc"],contains:[hljs.COMMENT("^/{4,}\\n","\\n/{4,}$",{relevance:10}),hljs.COMMENT("^//","$",{relevance:0}),{className:"title",begin:"^\\.\\w.*$"},{begin:"^[=\\*]{4,}\\n",end:"\\n^[=\\*]{4,}$",relevance:10},{className:"section",relevance:10,variants:[{begin:"^(={1,5}) .+?( \\1)?$"},{begin:"^[^\\[\\]\\n]+?\\n[=\\-~\\^\\+]{2,}$"}]},{className:"meta",begin:"^:.+?:",end:"\\s",excludeEnd:true,relevance:10},{className:"meta",begin:"^\\[.+?\\]$",relevance:0},{className:"quote",begin:"^_{4,}\\n",end:"\\n_{4,}$",relevance:10},{className:"code",begin:"^[\\-\\.]{4,}\\n",end:"\\n[\\-\\.]{4,}$",relevance:10},{begin:"^\\+{4,}\\n",end:"\\n\\+{4,}$",contains:[{begin:"<",end:">",subLanguage:"xml",relevance:0}],relevance:10},{className:"bullet",begin:"^(\\*+|\\-+|\\.+|[^\\n]+?::)\\s+"},{className:"symbol",begin:"^(NOTE|TIP|IMPORTANT|WARNING|CAUTION):\\s+",relevance:10},{className:"strong",begin:"\\B\\*(?![\\*\\s])",end:"(\\n{2}|\\*)",contains:[{begin:"\\\\*\\w",relevance:0}]},{className:"emphasis",begin:"\\B'(?!['\\s])",end:"(\\n{2}|')",contains:[{begin:"\\\\'\\w",relevance:0}],relevance:0},{className:"emphasis",begin:"_(?![_\\s])",end:"(\\n{2}|_)",relevance:0},{className:"string",variants:[{begin:"``.+?''"},{begin:"`.+?'"}]},{className:"code",begin:"(`.+?`|\\+.+?\\+)",relevance:0},{className:"code",begin:"^[ \\t]",end:"$",relevance:0},{begin:"^'{3,}[ \\t]*$",relevance:10},{begin:"(link:)?(http|https|ftp|file|irc|image:?):\\S+\\[.*?\\]",returnBegin:true,contains:[{begin:"(link|image:?):",relevance:0},{className:"link",begin:"\\w",end:"[^\\[]+",relevance:0},{className:"string",begin:"\\[",end:"\\]",excludeBegin:true,excludeEnd:true,relevance:0}],relevance:10}]}});hljs.registerLanguage("flix",function(hljs){var CHAR={className:"string",begin:/'(.|\\[xXuU][a-zA-Z0-9]+)'/};var STRING={className:"string",variants:[{begin:'"',end:'"'}]};var NAME={className:"title",begin:/[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/};var METHOD={className:"function",beginKeywords:"def",end:/[:={\[(\n;]/,excludeEnd:true,contains:[NAME]};return{keywords:{literal:"true false",keyword:"case class def else enum if impl import in lat rel index let match namespace switch type yield with"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,CHAR,STRING,METHOD,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("python",function(hljs){var KEYWORDS={keyword:"and elif is global as in if from raise for except finally print import pass return "+"exec else break not with class assert yield try while continue del or def lambda "+"async await nonlocal|10",built_in:"Ellipsis NotImplemented",literal:"False None True"};var PROMPT={className:"meta",begin:/^(>>>|\.\.\.) /};var SUBST={className:"subst",begin:/\{/,end:/\}/,keywords:KEYWORDS,illegal:/#/};var LITERAL_BRACKET={begin:/\{\{/,relevance:0};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:/(u|b)?r?'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT],relevance:10},{begin:/(u|b)?r?"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT],relevance:10},{begin:/(fr|rf|f)'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT,LITERAL_BRACKET,SUBST]},{begin:/(fr|rf|f)"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,PROMPT,LITERAL_BRACKET,SUBST]},{begin:/(u|r|ur)'/,end:/'/,relevance:10},{begin:/(u|r|ur)"/,end:/"/,relevance:10},{begin:/(b|br)'/,end:/'/},{begin:/(b|br)"/,end:/"/},{begin:/(fr|rf|f)'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE,LITERAL_BRACKET,SUBST]},{begin:/(fr|rf|f)"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,LITERAL_BRACKET,SUBST]},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};var NUMBER={className:"number",relevance:0,variants:[{begin:hljs.BINARY_NUMBER_RE+"[lLjJ]?"},{begin:"\\b(0o[0-7]+)[lLjJ]?"},{begin:hljs.C_NUMBER_RE+"[lLjJ]?"}]};var PARAMS={className:"params",begin:/\(/,end:/\)/,contains:["self",PROMPT,NUMBER,STRING,hljs.HASH_COMMENT_MODE]};SUBST.contains=[STRING,NUMBER,PROMPT];return{aliases:["py","gyp","ipython"],keywords:KEYWORDS,illegal:/(<\/|->|\?)|=>/,contains:[PROMPT,NUMBER,{beginKeywords:"if",relevance:0},STRING,hljs.HASH_COMMENT_MODE,{variants:[{className:"function",beginKeywords:"def"},{className:"class",beginKeywords:"class"}],end:/:/,illegal:/[${=;\n,]/,contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS,{begin:/->/,endsWithParent:true,keywords:"None"}]},{className:"meta",begin:/^[\t ]*@/,end:/$/},{begin:/\b(print|exec)\(/}]}});hljs.registerLanguage("leaf",function(hljs){return{contains:[{className:"function",begin:"#+"+"[A-Za-z_0-9]*"+"\\(",end:" {",returnBegin:true,excludeEnd:true,contains:[{className:"keyword",begin:"#+"},{className:"title",begin:"[A-Za-z_][A-Za-z_0-9]*"},{className:"params",begin:"\\(",end:"\\)",endsParent:true,contains:[{className:"string",begin:'"',end:'"'},{className:"variable",begin:"[A-Za-z_][A-Za-z_0-9]*"}]}]}]}});hljs.registerLanguage("groovy",function(hljs){return{keywords:{literal:"true false null",keyword:"byte short char int long boolean float double void "+"def as in assert trait "+"super this abstract static volatile transient public private protected synchronized final "+"class interface enum if else for while switch case break default continue "+"throw throws try catch finally implements extends new import package return instanceof"},contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",begin:'"""',end:'"""'},{className:"string",begin:"'''",end:"'''"},{className:"string",begin:"\\$/",end:"/\\$",relevance:10},hljs.APOS_STRING_MODE,{className:"regexp",begin:/~?\/[^\/\n]+\//,contains:[hljs.BACKSLASH_ESCAPE]},hljs.QUOTE_STRING_MODE,{className:"meta",begin:"^#!/usr/bin/env",end:"$",illegal:"\n"},hljs.BINARY_NUMBER_MODE,{className:"class",beginKeywords:"class interface trait enum",end:"{",illegal:":",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"},{className:"string",begin:/[^\?]{0}[A-Za-z0-9_$]+ *:/},{begin:/\?/,end:/\:/},{className:"symbol",begin:"^\\s*[A-Za-z0-9_$]+:",relevance:0}],illegal:/#|<\//}});hljs.registerLanguage("armasm",function(hljs){return{case_insensitive:true,aliases:["arm"],lexemes:"\\.?"+hljs.IDENT_RE,keywords:{meta:".2byte .4byte .align .ascii .asciz .balign .byte .code .data .else .end .endif .endm .endr .equ .err .exitm .extern .global .hword .if .ifdef .ifndef .include .irp .long .macro .rept .req .section .set .skip .space .text .word .arm .thumb .code16 .code32 .force_thumb .thumb_func .ltorg "+"ALIAS ALIGN ARM AREA ASSERT ATTR CN CODE CODE16 CODE32 COMMON CP DATA DCB DCD DCDU DCDO DCFD DCFDU DCI DCQ DCQU DCW DCWU DN ELIF ELSE END ENDFUNC ENDIF ENDP ENTRY EQU EXPORT EXPORTAS EXTERN FIELD FILL FUNCTION GBLA GBLL GBLS GET GLOBAL IF IMPORT INCBIN INCLUDE INFO KEEP LCLA LCLL LCLS LTORG MACRO MAP MEND MEXIT NOFP OPT PRESERVE8 PROC QN READONLY RELOC REQUIRE REQUIRE8 RLIST FN ROUT SETA SETL SETS SN SPACE SUBT THUMB THUMBX TTL WHILE WEND ",built_in:"r0 r1 r2 r3 r4 r5 r6 r7 r8 r9 r10 r11 r12 r13 r14 r15 "+"pc lr sp ip sl sb fp "+"a1 a2 a3 a4 v1 v2 v3 v4 v5 v6 v7 v8 f0 f1 f2 f3 f4 f5 f6 f7 "+"p0 p1 p2 p3 p4 p5 p6 p7 p8 p9 p10 p11 p12 p13 p14 p15 "+"c0 c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c11 c12 c13 c14 c15 "+"q0 q1 q2 q3 q4 q5 q6 q7 q8 q9 q10 q11 q12 q13 q14 q15 "+"cpsr_c cpsr_x cpsr_s cpsr_f cpsr_cx cpsr_cxs cpsr_xs cpsr_xsf cpsr_sf cpsr_cxsf "+"spsr_c spsr_x spsr_s spsr_f spsr_cx spsr_cxs spsr_xs spsr_xsf spsr_sf spsr_cxsf "+"s0 s1 s2 s3 s4 s5 s6 s7 s8 s9 s10 s11 s12 s13 s14 s15 "+"s16 s17 s18 s19 s20 s21 s22 s23 s24 s25 s26 s27 s28 s29 s30 s31 "+"d0 d1 d2 d3 d4 d5 d6 d7 d8 d9 d10 d11 d12 d13 d14 d15 "+"d16 d17 d18 d19 d20 d21 d22 d23 d24 d25 d26 d27 d28 d29 d30 d31 "+"{PC} {VAR} {TRUE} {FALSE} {OPT} {CONFIG} {ENDIAN} {CODESIZE} {CPU} {FPU} {ARCHITECTURE} {PCSTOREOFFSET} {ARMASM_VERSION} {INTER} {ROPI} {RWPI} {SWST} {NOSWST} . @"},contains:[{className:"keyword",begin:"\\b("+"adc|"+"(qd?|sh?|u[qh]?)?add(8|16)?|usada?8|(q|sh?|u[qh]?)?(as|sa)x|"+"and|adrl?|sbc|rs[bc]|asr|b[lx]?|blx|bxj|cbn?z|tb[bh]|bic|"+"bfc|bfi|[su]bfx|bkpt|cdp2?|clz|clrex|cmp|cmn|cpsi[ed]|cps|"+"setend|dbg|dmb|dsb|eor|isb|it[te]{0,3}|lsl|lsr|ror|rrx|"+"ldm(([id][ab])|f[ds])?|ldr((s|ex)?[bhd])?|movt?|mvn|mra|mar|"+"mul|[us]mull|smul[bwt][bt]|smu[as]d|smmul|smmla|"+"mla|umlaal|smlal?([wbt][bt]|d)|mls|smlsl?[ds]|smc|svc|sev|"+"mia([bt]{2}|ph)?|mrr?c2?|mcrr2?|mrs|msr|orr|orn|pkh(tb|bt)|rbit|"+"rev(16|sh)?|sel|[su]sat(16)?|nop|pop|push|rfe([id][ab])?|"+"stm([id][ab])?|str(ex)?[bhd]?|(qd?)?sub|(sh?|q|u[qh]?)?sub(8|16)|"+"[su]xt(a?h|a?b(16)?)|srs([id][ab])?|swpb?|swi|smi|tst|teq|"+"wfe|wfi|yield"+")"+"(eq|ne|cs|cc|mi|pl|vs|vc|hi|ls|ge|lt|gt|le|al|hs|lo)?"+"[sptrx]?",end:"\\s"},hljs.COMMENT("[;@]","$",{relevance:0}),hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",relevance:0},{className:"title",begin:"\\|",end:"\\|",illegal:"\\n",relevance:0},{className:"number",variants:[{begin:"[#$=]?0x[0-9a-f]+"},{begin:"[#$=]?0b[01]+"},{begin:"[#$=]\\d+"},{begin:"\\b\\d+"}],relevance:0},{className:"symbol",variants:[{begin:"^[a-z_\\.\\$][a-z0-9_\\.\\$]+"},{begin:"^\\s*[a-z_\\.\\$][a-z0-9_\\.\\$]+:"},{begin:"[=#]\\w+"}],relevance:0}]}});hljs.registerLanguage("glsl",function(hljs){return{keywords:{keyword:"break continue discard do else for if return while switch case default "+"attribute binding buffer ccw centroid centroid varying coherent column_major const cw "+"depth_any depth_greater depth_less depth_unchanged early_fragment_tests equal_spacing "+"flat fractional_even_spacing fractional_odd_spacing highp in index inout invariant "+"invocations isolines layout line_strip lines lines_adjacency local_size_x local_size_y "+"local_size_z location lowp max_vertices mediump noperspective offset origin_upper_left "+"out packed patch pixel_center_integer point_mode points precise precision quads r11f_g11f_b10f "+"r16 r16_snorm r16f r16i r16ui r32f r32i r32ui r8 r8_snorm r8i r8ui readonly restrict "+"rg16 rg16_snorm rg16f rg16i rg16ui rg32f rg32i rg32ui rg8 rg8_snorm rg8i rg8ui rgb10_a2 "+"rgb10_a2ui rgba16 rgba16_snorm rgba16f rgba16i rgba16ui rgba32f rgba32i rgba32ui rgba8 "+"rgba8_snorm rgba8i rgba8ui row_major sample shared smooth std140 std430 stream triangle_strip "+"triangles triangles_adjacency uniform varying vertices volatile writeonly",type:"atomic_uint bool bvec2 bvec3 bvec4 dmat2 dmat2x2 dmat2x3 dmat2x4 dmat3 dmat3x2 dmat3x3 "+"dmat3x4 dmat4 dmat4x2 dmat4x3 dmat4x4 double dvec2 dvec3 dvec4 float iimage1D iimage1DArray "+"iimage2D iimage2DArray iimage2DMS iimage2DMSArray iimage2DRect iimage3D iimageBuffer"+"iimageCube iimageCubeArray image1D image1DArray image2D image2DArray image2DMS image2DMSArray "+"image2DRect image3D imageBuffer imageCube imageCubeArray int isampler1D isampler1DArray "+"isampler2D isampler2DArray isampler2DMS isampler2DMSArray isampler2DRect isampler3D "+"isamplerBuffer isamplerCube isamplerCubeArray ivec2 ivec3 ivec4 mat2 mat2x2 mat2x3 "+"mat2x4 mat3 mat3x2 mat3x3 mat3x4 mat4 mat4x2 mat4x3 mat4x4 sampler1D sampler1DArray "+"sampler1DArrayShadow sampler1DShadow sampler2D sampler2DArray sampler2DArrayShadow "+"sampler2DMS sampler2DMSArray sampler2DRect sampler2DRectShadow sampler2DShadow sampler3D "+"samplerBuffer samplerCube samplerCubeArray samplerCubeArrayShadow samplerCubeShadow "+"image1D uimage1DArray uimage2D uimage2DArray uimage2DMS uimage2DMSArray uimage2DRect "+"uimage3D uimageBuffer uimageCube uimageCubeArray uint usampler1D usampler1DArray "+"usampler2D usampler2DArray usampler2DMS usampler2DMSArray usampler2DRect usampler3D "+"samplerBuffer usamplerCube usamplerCubeArray uvec2 uvec3 uvec4 vec2 vec3 vec4 void",built_in:"gl_MaxAtomicCounterBindings gl_MaxAtomicCounterBufferSize gl_MaxClipDistances gl_MaxClipPlanes "+"gl_MaxCombinedAtomicCounterBuffers gl_MaxCombinedAtomicCounters gl_MaxCombinedImageUniforms "+"gl_MaxCombinedImageUnitsAndFragmentOutputs gl_MaxCombinedTextureImageUnits gl_MaxComputeAtomicCounterBuffers "+"gl_MaxComputeAtomicCounters gl_MaxComputeImageUniforms gl_MaxComputeTextureImageUnits "+"gl_MaxComputeUniformComponents gl_MaxComputeWorkGroupCount gl_MaxComputeWorkGroupSize "+"gl_MaxDrawBuffers gl_MaxFragmentAtomicCounterBuffers gl_MaxFragmentAtomicCounters "+"gl_MaxFragmentImageUniforms gl_MaxFragmentInputComponents gl_MaxFragmentInputVectors "+"gl_MaxFragmentUniformComponents gl_MaxFragmentUniformVectors gl_MaxGeometryAtomicCounterBuffers "+"gl_MaxGeometryAtomicCounters gl_MaxGeometryImageUniforms gl_MaxGeometryInputComponents "+"gl_MaxGeometryOutputComponents gl_MaxGeometryOutputVertices gl_MaxGeometryTextureImageUnits "+"gl_MaxGeometryTotalOutputComponents gl_MaxGeometryUniformComponents gl_MaxGeometryVaryingComponents "+"gl_MaxImageSamples gl_MaxImageUnits gl_MaxLights gl_MaxPatchVertices gl_MaxProgramTexelOffset "+"gl_MaxTessControlAtomicCounterBuffers gl_MaxTessControlAtomicCounters gl_MaxTessControlImageUniforms "+"gl_MaxTessControlInputComponents gl_MaxTessControlOutputComponents gl_MaxTessControlTextureImageUnits "+"gl_MaxTessControlTotalOutputComponents gl_MaxTessControlUniformComponents "+"gl_MaxTessEvaluationAtomicCounterBuffers gl_MaxTessEvaluationAtomicCounters "+"gl_MaxTessEvaluationImageUniforms gl_MaxTessEvaluationInputComponents gl_MaxTessEvaluationOutputComponents "+"gl_MaxTessEvaluationTextureImageUnits gl_MaxTessEvaluationUniformComponents "+"gl_MaxTessGenLevel gl_MaxTessPatchComponents gl_MaxTextureCoords gl_MaxTextureImageUnits "+"gl_MaxTextureUnits gl_MaxVaryingComponents gl_MaxVaryingFloats gl_MaxVaryingVectors "+"gl_MaxVertexAtomicCounterBuffers gl_MaxVertexAtomicCounters gl_MaxVertexAttribs gl_MaxVertexImageUniforms "+"gl_MaxVertexOutputComponents gl_MaxVertexOutputVectors gl_MaxVertexTextureImageUnits "+"gl_MaxVertexUniformComponents gl_MaxVertexUniformVectors gl_MaxViewports gl_MinProgramTexelOffset "+"gl_BackColor gl_BackLightModelProduct gl_BackLightProduct gl_BackMaterial "+"gl_BackSecondaryColor gl_ClipDistance gl_ClipPlane gl_ClipVertex gl_Color "+"gl_DepthRange gl_EyePlaneQ gl_EyePlaneR gl_EyePlaneS gl_EyePlaneT gl_Fog gl_FogCoord "+"gl_FogFragCoord gl_FragColor gl_FragCoord gl_FragData gl_FragDepth gl_FrontColor "+"gl_FrontFacing gl_FrontLightModelProduct gl_FrontLightProduct gl_FrontMaterial "+"gl_FrontSecondaryColor gl_GlobalInvocationID gl_InstanceID gl_InvocationID gl_Layer gl_LightModel "+"gl_LightSource gl_LocalInvocationID gl_LocalInvocationIndex gl_ModelViewMatrix "+"gl_ModelViewMatrixInverse gl_ModelViewMatrixInverseTranspose gl_ModelViewMatrixTranspose "+"gl_ModelViewProjectionMatrix gl_ModelViewProjectionMatrixInverse gl_ModelViewProjectionMatrixInverseTranspose "+"gl_ModelViewProjectionMatrixTranspose gl_MultiTexCoord0 gl_MultiTexCoord1 gl_MultiTexCoord2 "+"gl_MultiTexCoord3 gl_MultiTexCoord4 gl_MultiTexCoord5 gl_MultiTexCoord6 gl_MultiTexCoord7 "+"gl_Normal gl_NormalMatrix gl_NormalScale gl_NumSamples gl_NumWorkGroups gl_ObjectPlaneQ "+"gl_ObjectPlaneR gl_ObjectPlaneS gl_ObjectPlaneT gl_PatchVerticesIn gl_Point gl_PointCoord "+"gl_PointSize gl_Position gl_PrimitiveID gl_PrimitiveIDIn gl_ProjectionMatrix gl_ProjectionMatrixInverse "+"gl_ProjectionMatrixInverseTranspose gl_ProjectionMatrixTranspose gl_SampleID gl_SampleMask "+"gl_SampleMaskIn gl_SamplePosition gl_SecondaryColor gl_TessCoord gl_TessLevelInner gl_TessLevelOuter "+"gl_TexCoord gl_TextureEnvColor gl_TextureMatrix gl_TextureMatrixInverse gl_TextureMatrixInverseTranspose "+"gl_TextureMatrixTranspose gl_Vertex gl_VertexID gl_ViewportIndex gl_WorkGroupID gl_WorkGroupSize gl_in gl_out "+"EmitStreamVertex EmitVertex EndPrimitive EndStreamPrimitive abs acos acosh all any asin "+"asinh atan atanh atomicAdd atomicAnd atomicCompSwap atomicCounter atomicCounterDecrement "+"atomicCounterIncrement atomicExchange atomicMax atomicMin atomicOr atomicXor barrier "+"bitCount bitfieldExtract bitfieldInsert bitfieldReverse ceil clamp cos cosh cross "+"dFdx dFdy degrees determinant distance dot equal exp exp2 faceforward findLSB findMSB "+"floatBitsToInt floatBitsToUint floor fma fract frexp ftransform fwidth greaterThan "+"greaterThanEqual groupMemoryBarrier imageAtomicAdd imageAtomicAnd imageAtomicCompSwap "+"imageAtomicExchange imageAtomicMax imageAtomicMin imageAtomicOr imageAtomicXor imageLoad "+"imageSize imageStore imulExtended intBitsToFloat interpolateAtCentroid interpolateAtOffset "+"interpolateAtSample inverse inversesqrt isinf isnan ldexp length lessThan lessThanEqual log "+"log2 matrixCompMult max memoryBarrier memoryBarrierAtomicCounter memoryBarrierBuffer "+"memoryBarrierImage memoryBarrierShared min mix mod modf noise1 noise2 noise3 noise4 "+"normalize not notEqual outerProduct packDouble2x32 packHalf2x16 packSnorm2x16 packSnorm4x8 "+"packUnorm2x16 packUnorm4x8 pow radians reflect refract round roundEven shadow1D shadow1DLod "+"shadow1DProj shadow1DProjLod shadow2D shadow2DLod shadow2DProj shadow2DProjLod sign sin sinh "+"smoothstep sqrt step tan tanh texelFetch texelFetchOffset texture texture1D texture1DLod "+"texture1DProj texture1DProjLod texture2D texture2DLod texture2DProj texture2DProjLod "+"texture3D texture3DLod texture3DProj texture3DProjLod textureCube textureCubeLod "+"textureGather textureGatherOffset textureGatherOffsets textureGrad textureGradOffset "+"textureLod textureLodOffset textureOffset textureProj textureProjGrad textureProjGradOffset "+"textureProjLod textureProjLodOffset textureProjOffset textureQueryLevels textureQueryLod "+"textureSize transpose trunc uaddCarry uintBitsToFloat umulExtended unpackDouble2x32 "+"unpackHalf2x16 unpackSnorm2x16 unpackSnorm4x8 unpackUnorm2x16 unpackUnorm4x8 usubBorrow",literal:"true false"},illegal:'"',contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"}]}});hljs.registerLanguage("q",function(hljs){var Q_KEYWORDS={keyword:"do while select delete by update from",literal:"0b 1b",built_in:"neg not null string reciprocal floor ceiling signum mod xbar xlog and or each scan over prior mmu lsq inv md5 ltime gtime count first var dev med cov cor all any rand sums prds mins maxs fills deltas ratios avgs differ prev next rank reverse iasc idesc asc desc msum mcount mavg mdev xrank mmin mmax xprev rotate distinct group where flip type key til get value attr cut set upsert raze union inter except cross sv vs sublist enlist read0 read1 hopen hclose hdel hsym hcount peach system ltrim rtrim trim lower upper ssr view tables views cols xcols keys xkey xcol xasc xdesc fkeys meta lj aj aj0 ij pj asof uj ww wj wj1 fby xgroup ungroup ej save load rsave rload show csv parse eval min max avg wavg wsum sin cos tan sum",type:"`float `double int `timestamp `timespan `datetime `time `boolean `symbol `char `byte `short `long `real `month `date `minute `second `guid"};return{aliases:["k","kdb"],keywords:Q_KEYWORDS,lexemes:/(`?)[A-Za-z0-9_]+\b/,contains:[hljs.C_LINE_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("css",function(hljs){var FUNCTION_LIKE={begin:/[\w-]+\(/,returnBegin:true,contains:[{className:"built_in",begin:/[\w-]+/},{begin:/\(/,end:/\)/,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE]}]};var ATTRIBUTE={className:"attribute",begin:/\S/,end:":",excludeEnd:true,starts:{endsWithParent:true,excludeEnd:true,contains:[FUNCTION_LIKE,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"number",begin:"#[0-9A-Fa-f]+"},{className:"meta",begin:"!important"}]}};var AT_IDENTIFIER="@[a-z-]+";var AT_MODIFIERS="and or not only";var MEDIA_TYPES="all print screen speech";var AT_PROPERTY_RE=/@\-?\w[\w]*(\-\w+)*/;var IDENT_RE="[a-zA-Z-][a-zA-Z0-9_-]*";var RULE={begin:/(?:[A-Z\_\.\-]+|--[a-zA-Z0-9_-]+)\s*:/,returnBegin:true,end:";",endsWithParent:true,contains:[ATTRIBUTE]};return{case_insensitive:true,illegal:/[=\/|'\$]/,contains:[hljs.C_BLOCK_COMMENT_MODE,{className:"selector-id",begin:/#[A-Za-z0-9_-]+/},{className:"selector-class",begin:/\.[A-Za-z0-9_-]+/},{className:"selector-attr",begin:/\[/,end:/\]/,illegal:"$",contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]},{className:"selector-pseudo",begin:/:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/},{begin:"@(page|font-face)",lexemes:AT_IDENTIFIER,keywords:"@page @font-face"},{begin:"@",end:"[{;]",illegal:/:/,returnBegin:true,contains:[{className:"keyword",begin:AT_PROPERTY_RE},{begin:/\s/,endsWithParent:true,excludeEnd:true,relevance:0,keywords:AT_MODIFIERS,contains:[{begin:/[a-z-]+:/,className:"attribute"},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE]}]},{className:"selector-tag",begin:IDENT_RE,relevance:0},{begin:"{",end:"}",illegal:/\S/,contains:[hljs.C_BLOCK_COMMENT_MODE,RULE]}]}});hljs.registerLanguage("ini",function(hljs){var NUMBERS={className:"number",relevance:0,variants:[{begin:/([\+\-]+)?[\d]+_[\d_]+/},{begin:hljs.NUMBER_RE}]};var COMMENTS=hljs.COMMENT();COMMENTS.variants=[{begin:/;/,end:/$/},{begin:/#/,end:/$/}];var VARIABLES={className:"variable",variants:[{begin:/\$[\w\d"][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var LITERALS={className:"literal",begin:/\bon|off|true|false|yes|no\b/};var STRINGS={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:"'''",end:"'''",relevance:10},{begin:'"""',end:'"""',relevance:10},{begin:'"',end:'"'},{begin:"'",end:"'"}]};var ARRAY={begin:/\[/,end:/\]/,contains:[COMMENTS,LITERALS,VARIABLES,STRINGS,NUMBERS,"self"],relevance:0};return{aliases:["toml"],case_insensitive:true,illegal:/\S/,contains:[COMMENTS,{className:"section",begin:/\[+/,end:/\]+/},{begin:/^[a-z0-9\[\]_\.-]+(?=\s*=\s*)/,className:"attr",starts:{end:/$/,contains:[COMMENTS,ARRAY,LITERALS,VARIABLES,STRINGS,NUMBERS]}}]}});hljs.registerLanguage("elm",function(hljs){var COMMENT={variants:[hljs.COMMENT("--","$"),hljs.COMMENT("{-","-}",{contains:["self"]})]};var CONSTRUCTOR={className:"type",begin:"\\b[A-Z][\\w']*",relevance:0};var LIST={begin:"\\(",end:"\\)",illegal:'"',contains:[{className:"type",begin:"\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?"},COMMENT]};var RECORD={begin:"{",end:"}",contains:LIST.contains};var CHARACTER={className:"string",begin:"'\\\\?.",end:"'",illegal:"."};return{keywords:"let in if then else case of where module import exposing "+"type alias as infix infixl infixr port effect command subscription",contains:[{beginKeywords:"port effect module",end:"exposing",keywords:"port effect module where command subscription exposing",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"import",end:"$",keywords:"import as exposing",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"type",end:"$",keywords:"type alias",contains:[CONSTRUCTOR,LIST,RECORD,COMMENT]},{beginKeywords:"infix infixl infixr",end:"$",contains:[hljs.C_NUMBER_MODE,COMMENT]},{begin:"port",end:"$",keywords:"port",contains:[COMMENT]},CHARACTER,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,CONSTRUCTOR,hljs.inherit(hljs.TITLE_MODE,{begin:"^[_a-z][\\w']*"}),COMMENT,{begin:"->|<-"}],illegal:/;/}});hljs.registerLanguage("ldif",function(hljs){return{contains:[{className:"attribute",begin:"^dn",end:": ",excludeEnd:true,starts:{end:"$",relevance:0},relevance:10},{className:"attribute",begin:"^\\w",end:": ",excludeEnd:true,starts:{end:"$",relevance:0}},{className:"literal",begin:"^-",end:"$"},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("makefile",function(hljs){var VARIABLE={className:"variable",variants:[{begin:"\\$\\("+hljs.UNDERSCORE_IDENT_RE+"\\)",contains:[hljs.BACKSLASH_ESCAPE]},{begin:/\$[@%<?\^\+\*]/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VARIABLE]};var FUNC={className:"variable",begin:/\$\([\w-]+\s/,end:/\)/,keywords:{built_in:"subst patsubst strip findstring filter filter-out sort "+"word wordlist firstword lastword dir notdir suffix basename "+"addsuffix addprefix join wildcard realpath abspath error warning "+"shell origin flavor foreach if or and call eval file value"},contains:[VARIABLE]};var ASSIGNMENT={begin:"^"+hljs.UNDERSCORE_IDENT_RE+"\\s*(?=[:+?]?=)"};var META={className:"meta",begin:/^\.PHONY:/,end:/$/,keywords:{"meta-keyword":".PHONY"},lexemes:/[\.\w]+/};var TARGET={className:"section",begin:/^[^\s]+:/,end:/$/,contains:[VARIABLE]};return{aliases:["mk","mak"],keywords:"define endef undefine ifdef ifndef ifeq ifneq else endif "+"include -include sinclude override export unexport private vpath",lexemes:/[\w-]+/,contains:[hljs.HASH_COMMENT_MODE,VARIABLE,QUOTE_STRING,FUNC,ASSIGNMENT,META,TARGET]}});hljs.registerLanguage("coffeescript",function(hljs){var KEYWORDS={keyword:"in if for while finally new do return else break catch instanceof throw try this "+"switch continue typeof delete debugger super yield import export from as default await "+"then unless until loop of by when and or is isnt not",literal:"true false null undefined "+"yes no on off",built_in:"npm require console print module global window document"};var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.BINARY_NUMBER_MODE,hljs.inherit(hljs.C_NUMBER_MODE,{starts:{end:"(\\s*/)?",relevance:0}}),{className:"string",variants:[{begin:/'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]},{className:"regexp",variants:[{begin:"///",end:"///",contains:[SUBST,hljs.HASH_COMMENT_MODE]},{begin:"//[gim]{0,3}(?=\\W)",relevance:0},{begin:/\/(?![ *]).*?(?![\\]).\/[gim]{0,3}(?=\W)/}]},{begin:"@"+JS_IDENT_RE},{subLanguage:"javascript",excludeBegin:true,excludeEnd:true,variants:[{begin:"```",end:"```"},{begin:"`",end:"`"}]}];SUBST.contains=EXPRESSIONS;var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var PARAMS_RE="(\\(.*\\))?\\s*\\B[-=]>";var PARAMS={className:"params",begin:"\\([^\\(]",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};return{aliases:["coffee","cson","iced"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("###","###"),hljs.HASH_COMMENT_MODE,{className:"function",begin:"^\\s*"+JS_IDENT_RE+"\\s*=\\s*"+PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[TITLE,PARAMS]},{begin:/[:\(,=]\s*/,relevance:0,contains:[{className:"function",begin:PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[PARAMS]}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("abnf",function(hljs){var regexes={ruleDeclaration:"^[a-zA-Z][a-zA-Z0-9-]*",unexpectedChars:"[!@#$^&',?+~`|:]"};var keywords=["ALPHA","BIT","CHAR","CR","CRLF","CTL","DIGIT","DQUOTE","HEXDIG","HTAB","LF","LWSP","OCTET","SP","VCHAR","WSP"];var commentMode=hljs.COMMENT(";","$");var terminalBinaryMode={className:"symbol",begin:/%b[0-1]+(-[0-1]+|(\.[0-1]+)+){0,1}/};var terminalDecimalMode={className:"symbol",begin:/%d[0-9]+(-[0-9]+|(\.[0-9]+)+){0,1}/};var terminalHexadecimalMode={className:"symbol",begin:/%x[0-9A-F]+(-[0-9A-F]+|(\.[0-9A-F]+)+){0,1}/};var caseSensitivityIndicatorMode={className:"symbol",begin:/%[si]/};var ruleDeclarationMode={className:"attribute",begin:regexes.ruleDeclaration+"(?=\\s*=)"};return{illegal:regexes.unexpectedChars,keywords:keywords.join(" "),contains:[ruleDeclarationMode,commentMode,terminalBinaryMode,terminalDecimalMode,terminalHexadecimalMode,caseSensitivityIndicatorMode,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("kotlin",function(hljs){var KEYWORDS={keyword:"abstract as val var vararg get set class object open private protected public noinline "+"crossinline dynamic final enum if else do while for when throw try catch finally "+"import package is in fun override companion reified inline lateinit init "+"interface annotation data sealed internal infix operator out by constructor super "+"tailrec where const inner suspend typealias external expect actual "+"trait volatile transient native default",built_in:"Byte Short Char Int Long Boolean Float Double Void Unit Nothing",literal:"true false null"};var KEYWORDS_WITH_LABEL={className:"keyword",begin:/\b(break|continue|return|this)\b/,starts:{contains:[{className:"symbol",begin:/@\w+/}]}};var LABEL={className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"@"};var SUBST={className:"subst",begin:"\\${",end:"}",contains:[hljs.C_NUMBER_MODE]};var VARIABLE={className:"variable",begin:"\\$"+hljs.UNDERSCORE_IDENT_RE};var STRING={className:"string",variants:[{begin:'"""',end:'"""(?=[^"])',contains:[VARIABLE,SUBST]},{begin:"'",end:"'",illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"',end:'"',illegal:/\n/,contains:[hljs.BACKSLASH_ESCAPE,VARIABLE,SUBST]}]};SUBST.contains.push(STRING);var ANNOTATION_USE_SITE={className:"meta",begin:"@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*"+hljs.UNDERSCORE_IDENT_RE+")?"};var ANNOTATION={className:"meta",begin:"@"+hljs.UNDERSCORE_IDENT_RE,contains:[{begin:/\(/,end:/\)/,contains:[hljs.inherit(STRING,{className:"meta-string"})]}]};var KOTLIN_NUMBER_RE="\\b"+"("+"0[bB]([01]+[01_]+[01]+|[01]+)"+"|"+"0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)"+"|"+"("+"([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?"+"|"+"\\.([\\d]+[\\d_]+[\\d]+|[\\d]+)"+")"+"([eE][-+]?\\d+)?"+")"+"[lLfF]?";var KOTLIN_NUMBER_MODE={className:"number",begin:KOTLIN_NUMBER_RE,relevance:0};var KOTLIN_NESTED_COMMENT=hljs.COMMENT("/\\*","\\*/",{contains:[hljs.C_BLOCK_COMMENT_MODE]});var KOTLIN_PAREN_TYPE={variants:[{className:"type",begin:hljs.UNDERSCORE_IDENT_RE},{begin:/\(/,end:/\)/,contains:[]}]};var KOTLIN_PAREN_TYPE2=KOTLIN_PAREN_TYPE;KOTLIN_PAREN_TYPE2.variants[1].contains=[KOTLIN_PAREN_TYPE];KOTLIN_PAREN_TYPE.variants[1].contains=[KOTLIN_PAREN_TYPE2];return{aliases:["kt"],keywords:KEYWORDS,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT,KEYWORDS_WITH_LABEL,LABEL,ANNOTATION_USE_SITE,ANNOTATION,{className:"function",beginKeywords:"fun",end:"[(]|$",returnBegin:true,excludeEnd:true,keywords:KEYWORDS,illegal:/fun\s+(<.*>)?[^\s\(]+(\s+[^\s\(]+)\s*=/,relevance:5,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"type",begin:/</,end:/>/,keywords:"reified",relevance:0},{className:"params",begin:/\(/,end:/\)/,endsParent:true,keywords:KEYWORDS,relevance:0,contains:[{begin:/:/,end:/[=,\/]/,endsWithParent:true,contains:[KOTLIN_PAREN_TYPE,hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT],relevance:0},hljs.C_LINE_COMMENT_MODE,KOTLIN_NESTED_COMMENT,ANNOTATION_USE_SITE,ANNOTATION,STRING,hljs.C_NUMBER_MODE]},KOTLIN_NESTED_COMMENT]},{className:"class",beginKeywords:"class interface trait",end:/[:\{(]|$/,excludeEnd:true,illegal:"extends implements",contains:[{beginKeywords:"public protected internal private constructor"},hljs.UNDERSCORE_TITLE_MODE,{className:"type",begin:/</,end:/>/,excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:/[,:]\s*/,end:/[<\(,]|$/,excludeBegin:true,returnEnd:true},ANNOTATION_USE_SITE,ANNOTATION]},STRING,{className:"meta",begin:"^#!/usr/bin/env",end:"$",illegal:"\n"},KOTLIN_NUMBER_MODE]}});hljs.registerLanguage("dsconfig",function(hljs){var QUOTED_PROPERTY={className:"string",begin:/"/,end:/"/};var APOS_PROPERTY={className:"string",begin:/'/,end:/'/};var UNQUOTED_PROPERTY={className:"string",begin:"[\\w-?]+:\\w+",end:"\\W",relevance:0};var VALUELESS_PROPERTY={className:"string",begin:"\\w+-?\\w+",end:"\\W",relevance:0};return{keywords:"dsconfig",contains:[{className:"keyword",begin:"^dsconfig",end:"\\s",excludeEnd:true,relevance:10},{className:"built_in",begin:"(list|create|get|set|delete)-(\\w+)",end:"\\s",excludeEnd:true,illegal:"!@#$%^&*()",relevance:10},{className:"built_in",begin:"--(\\w+)",end:"\\s",excludeEnd:true},QUOTED_PROPERTY,APOS_PROPERTY,UNQUOTED_PROPERTY,VALUELESS_PROPERTY,hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("http",function(hljs){var VERSION="HTTP/[0-9\\.]+";return{aliases:["https"],illegal:"\\S",contains:[{begin:"^"+VERSION,end:"$",contains:[{className:"number",begin:"\\b\\d{3}\\b"}]},{begin:"^[A-Z]+ (.*?) "+VERSION+"$",returnBegin:true,end:"$",contains:[{className:"string",begin:" ",end:" ",excludeBegin:true,excludeEnd:true},{begin:VERSION},{className:"keyword",begin:"[A-Z]+"}]},{className:"attribute",begin:"^\\w",end:": ",excludeEnd:true,illegal:"\\n|\\s|=",starts:{end:"$",relevance:0}},{begin:"\\n\\n",starts:{subLanguage:[],endsWithParent:true}}]}});hljs.registerLanguage("pony",function(hljs){var KEYWORDS={keyword:"actor addressof and as be break class compile_error compile_intrinsic "+"consume continue delegate digestof do else elseif embed end error "+"for fun if ifdef in interface is isnt lambda let match new not object "+"or primitive recover repeat return struct then trait try type until "+"use var where while with xor",meta:"iso val tag trn box ref",literal:"this false true"};var TRIPLE_QUOTE_STRING_MODE={className:"string",begin:'"""',end:'"""',relevance:10};var QUOTE_STRING_MODE={className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE]};var SINGLE_QUOTE_CHAR_MODE={className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE],relevance:0};var TYPE_NAME={className:"type",begin:"\\b_?[A-Z][\\w]*",relevance:0};var PRIMED_NAME={begin:hljs.IDENT_RE+"'",relevance:0};var NUMBER_MODE={className:"number",begin:"(-?)(\\b0[xX][a-fA-F0-9]+|\\b0[bB][01]+|(\\b\\d+(_\\d+)?(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",relevance:0};return{keywords:KEYWORDS,contains:[TYPE_NAME,TRIPLE_QUOTE_STRING_MODE,QUOTE_STRING_MODE,SINGLE_QUOTE_CHAR_MODE,PRIMED_NAME,NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("cpp",function(hljs){function optional(s){return"(?:"+s+")?"}var DECLTYPE_AUTO_RE="decltype\\(auto\\)";var NAMESPACE_RE="[a-zA-Z_]\\w*::";var TEMPLATE_ARGUMENT_RE="<.*?>";var FUNCTION_TYPE_RE="("+DECLTYPE_AUTO_RE+"|"+optional(NAMESPACE_RE)+"[a-zA-Z_]\\w*"+optional(TEMPLATE_ARGUMENT_RE)+")";var CPP_PRIMITIVE_TYPES={className:"keyword",begin:"\\b[a-z\\d_]*_t\\b"};var CHARACTER_ESCAPES="\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)";var STRINGS={className:"string",variants:[{begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"(u8?|U|L)?'("+CHARACTER_ESCAPES+"|.)",end:"'",illegal:"."},{begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\((?:.|\n)*?\)\1"/}]};var NUMBERS={className:"number",variants:[{begin:"\\b(0b[01']+)"},{begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],relevance:0};var PREPROCESSOR={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"if else elif endif define undef warning error line "+"pragma _Pragma ifdef ifndef include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:/<.*?>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};var TITLE_MODE={className:"title",begin:optional(NAMESPACE_RE)+hljs.IDENT_RE,relevance:0};var FUNCTION_TITLE=optional(NAMESPACE_RE)+hljs.IDENT_RE+"\\s*\\(";var CPP_KEYWORDS={keyword:"int float while private char char8_t char16_t char32_t catch import module export virtual operator sizeof "+"dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace "+"unsigned long volatile static protected bool template mutable if public friend "+"do goto auto void enum else break extern using asm case typeid wchar_t"+"short reinterpret_cast|10 default double register explicit signed typename try this "+"switch continue inline delete alignas alignof constexpr consteval constinit decltype "+"concept co_await co_return co_yield requires "+"noexcept static_assert thread_local restrict final override "+"atomic_bool atomic_char atomic_schar "+"atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong "+"atomic_ullong new throw return "+"and and_eq bitand bitor compl not not_eq or or_eq xor xor_eq",built_in:"std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream "+"auto_ptr deque list queue stack vector map set bitset multiset multimap unordered_set "+"unordered_map unordered_multiset unordered_multimap array shared_ptr abort terminate abs acos "+"asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp "+"fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper "+"isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow "+"printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp "+"strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan "+"vfprintf vprintf vsprintf endl initializer_list unique_ptr _Bool complex _Complex imaginary _Imaginary",literal:"true false nullptr NULL"};var EXPRESSION_CONTAINS=[CPP_PRIMITIVE_TYPES,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS];var EXPRESSION_CONTEXT={variants:[{begin:/=/,end:/;/},{begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}],keywords:CPP_KEYWORDS,contains:EXPRESSION_CONTAINS.concat([{begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,contains:EXPRESSION_CONTAINS.concat(["self"]),relevance:0}]),relevance:0};var FUNCTION_DECLARATION={className:"function",begin:"("+FUNCTION_TYPE_RE+"[\\*&\\s]+)+"+FUNCTION_TITLE,returnBegin:true,end:/[{;=]/,excludeEnd:true,keywords:CPP_KEYWORDS,illegal:/[^\w\s\*&:<>]/,contains:[{begin:DECLTYPE_AUTO_RE,keywords:CPP_KEYWORDS,relevance:0},{begin:FUNCTION_TITLE,returnBegin:true,contains:[TITLE_MODE],relevance:0},{className:"params",begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,relevance:0,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRINGS,NUMBERS,CPP_PRIMITIVE_TYPES,{begin:/\(/,end:/\)/,keywords:CPP_KEYWORDS,relevance:0,contains:["self",hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRINGS,NUMBERS,CPP_PRIMITIVE_TYPES]}]},CPP_PRIMITIVE_TYPES,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,PREPROCESSOR]};return{aliases:["c","cc","h","c++","h++","hpp","hh","hxx","cxx"],keywords:CPP_KEYWORDS,illegal:"</",contains:[].concat(EXPRESSION_CONTEXT,FUNCTION_DECLARATION,EXPRESSION_CONTAINS,[PREPROCESSOR,{begin:"\\b(deque|list|queue|stack|vector|map|set|bitset|multiset|multimap|unordered_map|unordered_set|unordered_multiset|unordered_multimap|array)\\s*<",end:">",keywords:CPP_KEYWORDS,contains:["self",CPP_PRIMITIVE_TYPES]},{begin:hljs.IDENT_RE+"::",keywords:CPP_KEYWORDS},{className:"class",beginKeywords:"class struct",end:/[{;:]/,contains:[{begin:/</,end:/>/,contains:["self"]},hljs.TITLE_MODE]}]),exports:{preprocessor:PREPROCESSOR,strings:STRINGS,keywords:CPP_KEYWORDS}}});hljs.registerLanguage("arduino",function(hljs){var ARDUINO_KW={keyword:"boolean byte word String",built_in:"setup loop"+"KeyboardController MouseController SoftwareSerial "+"EthernetServer EthernetClient LiquidCrystal "+"RobotControl GSMVoiceCall EthernetUDP EsploraTFT "+"HttpClient RobotMotor WiFiClient GSMScanner "+"FileSystem Scheduler GSMServer YunClient YunServer "+"IPAddress GSMClient GSMModem Keyboard Ethernet "+"Console GSMBand Esplora Stepper Process "+"WiFiUDP GSM_SMS Mailbox USBHost Firmata PImage "+"Client Server GSMPIN FileIO Bridge Serial "+"EEPROM Stream Mouse Audio Servo File Task "+"GPRS WiFi Wire TFT GSM SPI SD "+"runShellCommandAsynchronously analogWriteResolution "+"retrieveCallingNumber printFirmwareVersion "+"analogReadResolution sendDigitalPortPair "+"noListenOnLocalhost readJoystickButton setFirmwareVersion "+"readJoystickSwitch scrollDisplayRight getVoiceCallStatus "+"scrollDisplayLeft writeMicroseconds delayMicroseconds "+"beginTransmission getSignalStrength runAsynchronously "+"getAsynchronously listenOnLocalhost getCurrentCarrier "+"readAccelerometer messageAvailable sendDigitalPorts "+"lineFollowConfig countryNameWrite runShellCommand "+"readStringUntil rewindDirectory readTemperature "+"setClockDivider readLightSensor endTransmission "+"analogReference detachInterrupt countryNameRead "+"attachInterrupt encryptionType readBytesUntil "+"robotNameWrite readMicrophone robotNameRead cityNameWrite "+"userNameWrite readJoystickY readJoystickX mouseReleased "+"openNextFile scanNetworks noInterrupts digitalWrite "+"beginSpeaker mousePressed isActionDone mouseDragged "+"displayLogos noAutoscroll addParameter remoteNumber "+"getModifiers keyboardRead userNameRead waitContinue "+"processInput parseCommand printVersion readNetworks "+"writeMessage blinkVersion cityNameRead readMessage "+"setDataMode parsePacket isListening setBitOrder "+"beginPacket isDirectory motorsWrite drawCompass "+"digitalRead clearScreen serialEvent rightToLeft "+"setTextSize leftToRight requestFrom keyReleased "+"compassRead analogWrite interrupts WiFiServer "+"disconnect playMelody parseFloat autoscroll "+"getPINUsed setPINUsed setTimeout sendAnalog "+"readSlider analogRead beginWrite createChar "+"motorsStop keyPressed tempoWrite readButton "+"subnetMask debugPrint macAddress writeGreen "+"randomSeed attachGPRS readString sendString "+"remotePort releaseAll mouseMoved background "+"getXChange getYChange answerCall getResult "+"voiceCall endPacket constrain getSocket writeJSON "+"getButton available connected findUntil readBytes "+"exitValue readGreen writeBlue startLoop IPAddress "+"isPressed sendSysex pauseMode gatewayIP setCursor "+"getOemKey tuneWrite noDisplay loadImage switchPIN "+"onRequest onReceive changePIN playFile noBuffer "+"parseInt overflow checkPIN knobRead beginTFT "+"bitClear updateIR bitWrite position writeRGB "+"highByte writeRed setSpeed readBlue noStroke "+"remoteIP transfer shutdown hangCall beginSMS "+"endWrite attached maintain noCursor checkReg "+"checkPUK shiftOut isValid shiftIn pulseIn "+"connect println localIP pinMode getIMEI "+"display noBlink process getBand running beginSD "+"drawBMP lowByte setBand release bitRead prepare "+"pointTo readRed setMode noFill remove listen "+"stroke detach attach noTone exists buffer "+"height bitSet circle config cursor random "+"IRread setDNS endSMS getKey micros "+"millis begin print write ready flush width "+"isPIN blink clear press mkdir rmdir close "+"point yield image BSSID click delay "+"read text move peek beep rect line open "+"seek fill size turn stop home find "+"step tone sqrt RSSI SSID "+"end bit tan cos sin pow map abs max "+"min get run put",literal:"DIGITAL_MESSAGE FIRMATA_STRING ANALOG_MESSAGE "+"REPORT_DIGITAL REPORT_ANALOG INPUT_PULLUP "+"SET_PIN_MODE INTERNAL2V56 SYSTEM_RESET LED_BUILTIN "+"INTERNAL1V1 SYSEX_START INTERNAL EXTERNAL "+"DEFAULT OUTPUT INPUT HIGH LOW"};var ARDUINO=hljs.requireLanguage("cpp").rawDefinition();var kws=ARDUINO.keywords;kws.keyword+=" "+ARDUINO_KW.keyword;kws.literal+=" "+ARDUINO_KW.literal;kws.built_in+=" "+ARDUINO_KW.built_in;return ARDUINO});hljs.registerLanguage("ocaml",function(hljs){return{aliases:["ml"],keywords:{keyword:"and as assert asr begin class constraint do done downto else end "+"exception external for fun function functor if in include "+"inherit! inherit initializer land lazy let lor lsl lsr lxor match method!|10 method "+"mod module mutable new object of open! open or private rec sig struct "+"then to try type val! val virtual when while with "+"parser value",built_in:"array bool bytes char exn|5 float int int32 int64 list lazy_t|5 nativeint|5 string unit "+"in_channel out_channel ref",literal:"true false"},illegal:/\/\/|>>/,lexemes:"[a-z_]\\w*!?",contains:[{className:"literal",begin:"\\[(\\|\\|)?\\]|\\(\\)",relevance:0},hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),{className:"symbol",begin:"'[A-Za-z_](?!')[\\w']*"},{className:"type",begin:"`[A-Z][\\w']*"},{className:"type",begin:"\\b[A-Z][\\w']*",relevance:0},{begin:"[a-z_]\\w*'[\\w']*",relevance:0},hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"number",begin:"\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)",relevance:0},{begin:/[-=]>/}]}});hljs.registerLanguage("oxygene",function(hljs){var OXYGENE_KEYWORDS="abstract add and array as asc aspect assembly async begin break block by case class concat const copy constructor continue "+"create default delegate desc distinct div do downto dynamic each else empty end ensure enum equals event except exit extension external false "+"final finalize finalizer finally flags for forward from function future global group has if implementation implements implies in index inherited "+"inline interface into invariants is iterator join locked locking loop matching method mod module namespace nested new nil not notify nullable of "+"old on operator or order out override parallel params partial pinned private procedure property protected public queryable raise read readonly "+"record reintroduce remove repeat require result reverse sealed select self sequence set shl shr skip static step soft take then to true try tuple "+"type union unit unsafe until uses using var virtual raises volatile where while with write xor yield await mapped deprecated stdcall cdecl pascal "+"register safecall overload library platform reference packed strict published autoreleasepool selector strong weak unretained";var CURLY_COMMENT=hljs.COMMENT("{","}",{relevance:0});var PAREN_COMMENT=hljs.COMMENT("\\(\\*","\\*\\)",{relevance:10});var STRING={className:"string",begin:"'",end:"'",contains:[{begin:"''"}]};var CHAR_STRING={className:"string",begin:"(#\\d+)+"};var FUNCTION={className:"function",beginKeywords:"function constructor destructor procedure method",end:"[:;]",keywords:"function constructor|10 destructor|10 procedure|10 method|10",contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",keywords:OXYGENE_KEYWORDS,contains:[STRING,CHAR_STRING]},CURLY_COMMENT,PAREN_COMMENT]};return{case_insensitive:true,lexemes:/\.?\w+/,keywords:OXYGENE_KEYWORDS,illegal:'("|\\$[G-Zg-z]|\\/\\*|</|=>|->)',contains:[CURLY_COMMENT,PAREN_COMMENT,hljs.C_LINE_COMMENT_MODE,STRING,CHAR_STRING,hljs.NUMBER_MODE,FUNCTION,{className:"class",begin:"=\\bclass\\b",end:"end;",keywords:OXYGENE_KEYWORDS,contains:[STRING,CHAR_STRING,CURLY_COMMENT,PAREN_COMMENT,hljs.C_LINE_COMMENT_MODE,FUNCTION]}]}});hljs.registerLanguage("scilab",function(hljs){var COMMON_CONTAINS=[hljs.C_NUMBER_MODE,{className:"string",begin:"'|\"",end:"'|\"",contains:[hljs.BACKSLASH_ESCAPE,{begin:"''"}]}];return{aliases:["sci"],lexemes:/%?\w+/,keywords:{keyword:"abort break case clear catch continue do elseif else endfunction end for function "+"global if pause return resume select try then while",literal:"%f %F %t %T %pi %eps %inf %nan %e %i %z %s",built_in:"abs and acos asin atan ceil cd chdir clearglobal cosh cos cumprod deff disp error "+"exec execstr exists exp eye gettext floor fprintf fread fsolve imag isdef isempty "+"isinfisnan isvector lasterror length load linspace list listfiles log10 log2 log "+"max min msprintf mclose mopen ones or pathconvert poly printf prod pwd rand real "+"round sinh sin size gsort sprintf sqrt strcat strcmps tring sum system tanh tan "+"type typename warning zeros matrix"},illegal:'("|#|/\\*|\\s+/\\w+)',contains:[{className:"function",beginKeywords:"function",end:"$",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)"}]},{begin:"[a-zA-Z_][a-zA-Z_0-9]*('+[\\.']*|[\\.']+)",end:"",relevance:0},{begin:"\\[",end:"\\]'*[\\.']*",relevance:0,contains:COMMON_CONTAINS},hljs.COMMENT("//","$")].concat(COMMON_CONTAINS)}});hljs.registerLanguage("jboss-cli",function(hljs){var PARAM={begin:/[\w-]+ *=/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/[\w-]+/}]};var PARAMSBLOCK={className:"params",begin:/\(/,end:/\)/,contains:[PARAM],relevance:0};var OPERATION={className:"function",begin:/:[\w\-.]+/,relevance:0};var PATH={className:"string",begin:/\B(([\/.])[\w\-.\/=]+)+/};var COMMAND_PARAMS={className:"params",begin:/--[\w\-=\/]+/};return{aliases:["wildfly-cli"],lexemes:"[a-z-]+",keywords:{keyword:"alias batch cd clear command connect connection-factory connection-info data-source deploy "+"deployment-info deployment-overlay echo echo-dmr help history if jdbc-driver-info jms-queue|20 jms-topic|20 ls "+"patch pwd quit read-attribute read-operation reload rollout-plan run-batch set shutdown try unalias "+"undeploy unset version xa-data-source",literal:"true false"},contains:[hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,COMMAND_PARAMS,OPERATION,PATH,PARAMSBLOCK]}});hljs.registerLanguage("angelscript",function(hljs){var builtInTypeMode={className:"built_in",begin:"\\b(void|bool|int|int8|int16|int32|int64|uint|uint8|uint16|uint32|uint64|string|ref|array|double|float|auto|dictionary)"};var objectHandleMode={className:"symbol",begin:"[a-zA-Z0-9_]+@"};var genericMode={className:"keyword",begin:"<",end:">",contains:[builtInTypeMode,objectHandleMode]};builtInTypeMode.contains=[genericMode];objectHandleMode.contains=[genericMode];return{aliases:["asc"],keywords:"for in|0 break continue while do|0 return if else case switch namespace is cast "+"or and xor not get|0 in inout|10 out override set|0 private public const default|0 "+"final shared external mixin|10 enum typedef funcdef this super import from interface "+"abstract|0 try catch protected explicit property",illegal:"(^using\\s+[A-Za-z0-9_\\.]+;$|\\bfunctions*[^\\(])",contains:[{className:"string",begin:"'",end:"'",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"""',end:'"""'},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{beginKeywords:"interface namespace",end:"{",illegal:"[;.\\-]",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+"}]},{beginKeywords:"class",end:"{",illegal:"[;.\\-]",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+",contains:[{begin:"[:,]\\s*",contains:[{className:"symbol",begin:"[a-zA-Z0-9_]+"}]}]}]},builtInTypeMode,objectHandleMode,{className:"literal",begin:"\\b(null|true|false)"},{className:"number",begin:"(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?f?|\\.\\d+f?)([eE][-+]?\\d+f?)?)"}]}});hljs.registerLanguage("r",function(hljs){var IDENT_RE="([a-zA-Z]|\\.[a-zA-Z.])[a-zA-Z0-9._]*";return{contains:[hljs.HASH_COMMENT_MODE,{begin:IDENT_RE,lexemes:IDENT_RE,keywords:{keyword:"function if in break next repeat else for return switch while try tryCatch "+"stop warning require library attach detach source setMethod setGeneric "+"setGroupGeneric setClass ...",literal:"NULL NA TRUE FALSE T F Inf NaN NA_integer_|10 NA_real_|10 NA_character_|10 "+"NA_complex_|10"},relevance:0},{className:"number",begin:"0[xX][0-9a-fA-F]+[Li]?\\b",relevance:0},{className:"number",begin:"\\d+(?:[eE][+\\-]?\\d*)?L\\b",relevance:0},{className:"number",begin:"\\d+\\.(?!\\d)(?:i\\b)?",relevance:0},{className:"number",begin:"\\d+(?:\\.\\d*)?(?:[eE][+\\-]?\\d*)?i?\\b",relevance:0},{className:"number",begin:"\\.\\d+(?:[eE][+\\-]?\\d*)?i?\\b",relevance:0},{begin:"`",end:"`",relevance:0},{className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:'"',end:'"'},{begin:"'",end:"'"}]}]}});hljs.registerLanguage("json",function(hljs){var LITERALS={literal:"true false null"};var ALLOWED_COMMENTS=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE];var TYPES=[hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE];var VALUE_CONTAINER={end:",",endsWithParent:true,excludeEnd:true,contains:TYPES,keywords:LITERALS};var OBJECT={begin:"{",end:"}",contains:[{className:"attr",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE],illegal:"\\n"},hljs.inherit(VALUE_CONTAINER,{begin:/:/})].concat(ALLOWED_COMMENTS),illegal:"\\S"};var ARRAY={begin:"\\[",end:"\\]",contains:[hljs.inherit(VALUE_CONTAINER)],illegal:"\\S"};TYPES.push(OBJECT,ARRAY);ALLOWED_COMMENTS.forEach(function(rule){TYPES.push(rule)});return{contains:TYPES,keywords:LITERALS,illegal:"\\S"}});hljs.registerLanguage("haskell",function(hljs){var COMMENT={variants:[hljs.COMMENT("--","$"),hljs.COMMENT("{-","-}",{contains:["self"]})]};var PRAGMA={className:"meta",begin:"{-#",end:"#-}"};var PREPROCESSOR={className:"meta",begin:"^#",end:"$"};var CONSTRUCTOR={className:"type",begin:"\\b[A-Z][\\w']*",relevance:0};var LIST={begin:"\\(",end:"\\)",illegal:'"',contains:[PRAGMA,PREPROCESSOR,{className:"type",begin:"\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?"},hljs.inherit(hljs.TITLE_MODE,{begin:"[_a-z][\\w']*"}),COMMENT]};var RECORD={begin:"{",end:"}",contains:LIST.contains};return{aliases:["hs"],keywords:"let in if then else case of where do module import hiding "+"qualified type data newtype deriving class instance as default "+"infix infixl infixr foreign export ccall stdcall cplusplus "+"jvm dotnet safe unsafe family forall mdo proc rec",contains:[{beginKeywords:"module",end:"where",keywords:"module where",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{begin:"\\bimport\\b",end:"$",keywords:"import qualified as hiding",contains:[LIST,COMMENT],illegal:"\\W\\.|;"},{className:"class",begin:"^(\\s*)?(class|instance)\\b",end:"where",keywords:"class family instance where",contains:[CONSTRUCTOR,LIST,COMMENT]},{className:"class",begin:"\\b(data|(new)?type)\\b",end:"$",keywords:"data family type newtype deriving",contains:[PRAGMA,CONSTRUCTOR,LIST,RECORD,COMMENT]},{beginKeywords:"default",end:"$",contains:[CONSTRUCTOR,LIST,COMMENT]},{beginKeywords:"infix infixl infixr",end:"$",contains:[hljs.C_NUMBER_MODE,COMMENT]},{begin:"\\bforeign\\b",end:"$",keywords:"foreign import export ccall stdcall cplusplus jvm "+"dotnet safe unsafe",contains:[CONSTRUCTOR,hljs.QUOTE_STRING_MODE,COMMENT]},{className:"meta",begin:"#!\\/usr\\/bin\\/env runhaskell",end:"$"},PRAGMA,PREPROCESSOR,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,CONSTRUCTOR,hljs.inherit(hljs.TITLE_MODE,{begin:"^[_a-z][\\w']*"}),COMMENT,{begin:"->|<-"}]}});hljs.registerLanguage("taggerscript",function(hljs){var COMMENT={className:"comment",begin:/\$noop\(/,end:/\)/,contains:[{begin:/\(/,end:/\)/,contains:["self",{begin:/\\./}]}],relevance:10};var FUNCTION={className:"keyword",begin:/\$(?!noop)[a-zA-Z][_a-zA-Z0-9]*/,end:/\(/,excludeEnd:true};var VARIABLE={className:"variable",begin:/%[_a-zA-Z0-9:]*/,end:"%"};var ESCAPE_SEQUENCE={className:"symbol",begin:/\\./};return{contains:[COMMENT,FUNCTION,VARIABLE,ESCAPE_SEQUENCE]}});hljs.registerLanguage("xl",function(hljs){var BUILTIN_MODULES="ObjectLoader Animate MovieCredits Slides Filters Shading Materials LensFlare Mapping VLCAudioVideo "+"StereoDecoder PointCloud NetworkAccess RemoteControl RegExp ChromaKey Snowfall NodeJS Speech Charts";var XL_KEYWORDS={keyword:"if then else do while until for loop import with is as where when by data constant "+"integer real text name boolean symbol infix prefix postfix block tree",literal:"true false nil",built_in:"in mod rem and or xor not abs sign floor ceil sqrt sin cos tan asin "+"acos atan exp expm1 log log2 log10 log1p pi at text_length text_range "+"text_find text_replace contains page slide basic_slide title_slide "+"title subtitle fade_in fade_out fade_at clear_color color line_color "+"line_width texture_wrap texture_transform texture scale_?x scale_?y "+"scale_?z? translate_?x translate_?y translate_?z? rotate_?x rotate_?y "+"rotate_?z? rectangle circle ellipse sphere path line_to move_to "+"quad_to curve_to theme background contents locally time mouse_?x "+"mouse_?y mouse_buttons "+BUILTIN_MODULES};var DOUBLE_QUOTE_TEXT={className:"string",begin:'"',end:'"',illegal:"\\n"};var SINGLE_QUOTE_TEXT={className:"string",begin:"'",end:"'",illegal:"\\n"};var LONG_TEXT={className:"string",begin:"<<",end:">>"};var BASED_NUMBER={className:"number",begin:"[0-9]+#[0-9A-Z_]+(\\.[0-9-A-Z_]+)?#?([Ee][+-]?[0-9]+)?"};var IMPORT={beginKeywords:"import",end:"$",keywords:XL_KEYWORDS,contains:[DOUBLE_QUOTE_TEXT]};var FUNCTION_DEFINITION={className:"function",begin:/[a-z][^\n]*->/,returnBegin:true,end:/->/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,keywords:XL_KEYWORDS}})]};return{aliases:["tao"],lexemes:/[a-zA-Z][a-zA-Z0-9_?]*/,keywords:XL_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,DOUBLE_QUOTE_TEXT,SINGLE_QUOTE_TEXT,LONG_TEXT,FUNCTION_DEFINITION,IMPORT,BASED_NUMBER,hljs.NUMBER_MODE]}});hljs.registerLanguage("axapta",function(hljs){return{keywords:"false int abstract private char boolean static null if for true "+"while long throw finally protected final return void enum else "+"break new catch byte super case short default double public try this switch "+"continue reverse firstfast firstonly forupdate nofetch sum avg minof maxof count "+"order group by asc desc index hint like dispaly edit client server ttsbegin "+"ttscommit str real date container anytype common div mod",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:":",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]}]}});hljs.registerLanguage("purebasic",function(hljs){var STRINGS={className:"string",begin:'(~)?"',end:'"',illegal:"\\n"};var CONSTANTS={className:"symbol",begin:"#[a-zA-Z_]\\w*\\$?"};return{aliases:["pb","pbi"],keywords:"Align And Array As Break CallDebugger Case CompilerCase CompilerDefault "+"CompilerElse CompilerElseIf CompilerEndIf CompilerEndSelect CompilerError "+"CompilerIf CompilerSelect CompilerWarning Continue Data DataSection Debug "+"DebugLevel Declare DeclareC DeclareCDLL DeclareDLL DeclareModule Default "+"Define Dim DisableASM DisableDebugger DisableExplicit Else ElseIf EnableASM "+"EnableDebugger EnableExplicit End EndDataSection EndDeclareModule EndEnumeration "+"EndIf EndImport EndInterface EndMacro EndModule EndProcedure EndSelect "+"EndStructure EndStructureUnion EndWith Enumeration EnumerationBinary Extends "+"FakeReturn For ForEach ForEver Global Gosub Goto If Import ImportC "+"IncludeBinary IncludeFile IncludePath Interface List Macro MacroExpandedCount "+"Map Module NewList NewMap Next Not Or Procedure ProcedureC "+"ProcedureCDLL ProcedureDLL ProcedureReturn Protected Prototype PrototypeC ReDim "+"Read Repeat Restore Return Runtime Select Shared Static Step Structure "+"StructureUnion Swap Threaded To UndefineMacro Until Until  UnuseModule "+"UseModule Wend While With XIncludeFile XOr",contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"function",begin:"\\b(Procedure|Declare)(C|CDLL|DLL)?\\b",end:"\\(",excludeEnd:true,returnBegin:true,contains:[{className:"keyword",begin:"(Procedure|Declare)(C|CDLL|DLL)?",excludeEnd:true},{className:"type",begin:"\\.\\w*"},hljs.UNDERSCORE_TITLE_MODE]},STRINGS,CONSTANTS]}});hljs.registerLanguage("zephir",function(hljs){var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:'b"',end:'"'},{begin:"b'",end:"'"},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]};var NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]};return{aliases:["zep"],case_insensitive:true,keywords:"and include_once list abstract global private echo interface as static endswitch "+"array null if endwhile or const for endforeach self var let while isset public "+"protected exit foreach throw elseif include __FILE__ empty require_once do xor "+"return parent clone use __CLASS__ __LINE__ else break print eval new "+"catch __METHOD__ case exception default die require __FUNCTION__ "+"enddeclare final try switch continue endfor endif declare unset true false "+"trait goto instanceof insteadof __DIR__ __NAMESPACE__ "+"yield finally int uint long ulong char uchar double float bool boolean string"+"likely unlikely",contains:[hljs.C_LINE_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.COMMENT("__halt_compiler.+?;",false,{endsWithParent:true,keywords:"__halt_compiler",lexemes:hljs.UNDERSCORE_IDENT_RE}),{className:"string",begin:"<<<['\"]?\\w+['\"]?$",end:"^\\w+;",contains:[hljs.BACKSLASH_ESCAPE]},{begin:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{className:"function",beginKeywords:"function",end:/[;{]/,excludeEnd:true,illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:["self",hljs.C_BLOCK_COMMENT_MODE,STRING,NUMBER]}]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:/[:\(\$"]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"namespace",end:";",illegal:/[\.']/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"use",end:";",contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"=>"},STRING,NUMBER]}});hljs.registerLanguage("coq",function(hljs){return{keywords:{keyword:"_|0 as at cofix else end exists exists2 fix for forall fun if IF in let "+"match mod Prop return Set then Type using where with "+"Abort About Add Admit Admitted All Arguments Assumptions Axiom Back BackTo "+"Backtrack Bind Blacklist Canonical Cd Check Class Classes Close Coercion "+"Coercions CoFixpoint CoInductive Collection Combined Compute Conjecture "+"Conjectures Constant constr Constraint Constructors Context Corollary "+"CreateHintDb Cut Declare Defined Definition Delimit Dependencies Dependent"+"Derive Drop eauto End Equality Eval Example Existential Existentials "+"Existing Export exporting Extern Extract Extraction Fact Field Fields File "+"Fixpoint Focus for From Function Functional Generalizable Global Goal Grab "+"Grammar Graph Guarded Heap Hint HintDb Hints Hypotheses Hypothesis ident "+"Identity If Immediate Implicit Import Include Inductive Infix Info Initial "+"Inline Inspect Instance Instances Intro Intros Inversion Inversion_clear "+"Language Left Lemma Let Libraries Library Load LoadPath Local Locate Ltac ML "+"Mode Module Modules Monomorphic Morphism Next NoInline Notation Obligation "+"Obligations Opaque Open Optimize Options Parameter Parameters Parametric "+"Path Paths pattern Polymorphic Preterm Print Printing Program Projections "+"Proof Proposition Pwd Qed Quit Rec Record Recursive Redirect Relation Remark "+"Remove Require Reserved Reset Resolve Restart Rewrite Right Ring Rings Save "+"Scheme Scope Scopes Script Search SearchAbout SearchHead SearchPattern "+"SearchRewrite Section Separate Set Setoid Show Solve Sorted Step Strategies "+"Strategy Structure SubClass Table Tables Tactic Term Test Theorem Time "+"Timeout Transparent Type Typeclasses Types Undelimit Undo Unfocus Unfocused "+"Unfold Universe Universes Unset Unshelve using Variable Variables Variant "+"Verbose Visibility where with",built_in:"abstract absurd admit after apply as assert assumption at auto autorewrite "+"autounfold before bottom btauto by case case_eq cbn cbv change "+"classical_left classical_right clear clearbody cofix compare compute "+"congruence constr_eq constructor contradict contradiction cut cutrewrite "+"cycle decide decompose dependent destruct destruction dintuition "+"discriminate discrR do double dtauto eapply eassumption eauto ecase "+"econstructor edestruct ediscriminate eelim eexact eexists einduction "+"einjection eleft elim elimtype enough equality erewrite eright "+"esimplify_eq esplit evar exact exactly_once exfalso exists f_equal fail "+"field field_simplify field_simplify_eq first firstorder fix fold fourier "+"functional generalize generalizing gfail give_up has_evar hnf idtac in "+"induction injection instantiate intro intro_pattern intros intuition "+"inversion inversion_clear is_evar is_var lapply lazy left lia lra move "+"native_compute nia nsatz omega once pattern pose progress proof psatz quote "+"record red refine reflexivity remember rename repeat replace revert "+"revgoals rewrite rewrite_strat right ring ring_simplify rtauto set "+"setoid_reflexivity setoid_replace setoid_rewrite setoid_symmetry "+"setoid_transitivity shelve shelve_unifiable simpl simple simplify_eq solve "+"specialize split split_Rabs split_Rmult stepl stepr subst sum swap "+"symmetry tactic tauto time timeout top transitivity trivial try tryif "+"unfold unify until using vm_compute with"},contains:[hljs.QUOTE_STRING_MODE,hljs.COMMENT("\\(\\*","\\*\\)"),hljs.C_NUMBER_MODE,{className:"type",excludeBegin:true,begin:"\\|\\s*",end:"\\w+"},{begin:/[-=]>/}]}});hljs.registerLanguage("clojure",function(hljs){var keywords={"builtin-name":"def defonce cond apply if-not if-let if not not= = < > <= >= == + / * - rem "+"quot neg? pos? delay? symbol? keyword? true? false? integer? empty? coll? list? "+"set? ifn? fn? associative? sequential? sorted? counted? reversible? number? decimal? "+"class? distinct? isa? float? rational? reduced? ratio? odd? even? char? seq? vector? "+"string? map? nil? contains? zero? instance? not-every? not-any? libspec? -> ->> .. . "+"inc compare do dotimes mapcat take remove take-while drop letfn drop-last take-last "+"drop-while while intern condp case reduced cycle split-at split-with repeat replicate "+"iterate range merge zipmap declare line-seq sort comparator sort-by dorun doall nthnext "+"nthrest partition eval doseq await await-for let agent atom send send-off release-pending-sends "+"add-watch mapv filterv remove-watch agent-error restart-agent set-error-handler error-handler "+"set-error-mode! error-mode shutdown-agents quote var fn loop recur throw try monitor-enter "+"monitor-exit defmacro defn defn- macroexpand macroexpand-1 for dosync and or "+"when when-not when-let comp juxt partial sequence memoize constantly complement identity assert "+"peek pop doto proxy defstruct first rest cons defprotocol cast coll deftype defrecord last butlast "+"sigs reify second ffirst fnext nfirst nnext defmulti defmethod meta with-meta ns in-ns create-ns import "+"refer keys select-keys vals key val rseq name namespace promise into transient persistent! conj! "+"assoc! dissoc! pop! disj! use class type num float double short byte boolean bigint biginteger "+"bigdec print-method print-dup throw-if printf format load compile get-in update-in pr pr-on newline "+"flush read slurp read-line subvec with-open memfn time re-find re-groups rand-int rand mod locking "+"assert-valid-fdecl alias resolve ref deref refset swap! reset! set-validator! compare-and-set! alter-meta! "+"reset-meta! commute get-validator alter ref-set ref-history-count ref-min-history ref-max-history ensure sync io! "+"new next conj set! to-array future future-call into-array aset gen-class reduce map filter find empty "+"hash-map hash-set sorted-map sorted-map-by sorted-set sorted-set-by vec vector seq flatten reverse assoc dissoc list "+"disj get union difference intersection extend extend-type extend-protocol int nth delay count concat chunk chunk-buffer "+"chunk-append chunk-first chunk-rest max min dec unchecked-inc-int unchecked-inc unchecked-dec-inc unchecked-dec unchecked-negate "+"unchecked-add-int unchecked-add unchecked-subtract-int unchecked-subtract chunk-next chunk-cons chunked-seq? prn vary-meta "+"lazy-seq spread list* str find-keyword keyword symbol gensym force rationalize"};var SYMBOLSTART="a-zA-Z_\\-!.?+*=<>&#'";var SYMBOL_RE="["+SYMBOLSTART+"]["+SYMBOLSTART+"0-9/;:]*";var SIMPLE_NUMBER_RE="[-+]?\\d+(\\.\\d+)?";var SYMBOL={begin:SYMBOL_RE,relevance:0};var NUMBER={className:"number",begin:SIMPLE_NUMBER_RE,relevance:0};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var LITERAL={className:"literal",begin:/\b(true|false|nil)\b/};var COLLECTION={begin:"[\\[\\{]",end:"[\\]\\}]"};var HINT={className:"comment",begin:"\\^"+SYMBOL_RE};var HINT_COL=hljs.COMMENT("\\^\\{","\\}");var KEY={className:"symbol",begin:"[:]{1,2}"+SYMBOL_RE};var LIST={begin:"\\(",end:"\\)"};var BODY={endsWithParent:true,relevance:0};var NAME={keywords:keywords,lexemes:SYMBOL_RE,className:"name",begin:SYMBOL_RE,starts:BODY};var DEFAULT_CONTAINS=[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL,SYMBOL];LIST.contains=[hljs.COMMENT("comment",""),NAME,BODY];BODY.contains=DEFAULT_CONTAINS;COLLECTION.contains=DEFAULT_CONTAINS;HINT_COL.contains=[COLLECTION];return{aliases:["clj"],illegal:/\S/,contains:[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL]}});hljs.registerLanguage("clojure-repl",function(hljs){return{contains:[{className:"meta",begin:/^([\w.-]+|\s*#_)?=>/,starts:{end:/$/,subLanguage:"clojure"}}]}});hljs.registerLanguage("java",function(hljs){var JAVA_IDENT_RE="[À-ʸa-zA-Z_$][À-ʸa-zA-Z_$0-9]*";var GENERIC_IDENT_RE=JAVA_IDENT_RE+"(<"+JAVA_IDENT_RE+"(\\s*,\\s*"+JAVA_IDENT_RE+")*>)?";var KEYWORDS="false synchronized int abstract float private char boolean var static null if const "+"for true while long strictfp finally protected import native final void "+"enum else break transient catch instanceof byte super volatile case assert short "+"package default double public try this switch continue throws protected public private "+"module requires exports do";var JAVA_NUMBER_RE="\\b"+"("+"0[bB]([01]+[01_]+[01]+|[01]+)"+"|"+"0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)"+"|"+"("+"([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?"+"|"+"\\.([\\d]+[\\d_]+[\\d]+|[\\d]+)"+")"+"([eE][-+]?\\d+)?"+")"+"[lLfF]?";var JAVA_NUMBER_MODE={className:"number",begin:JAVA_NUMBER_RE,relevance:0};return{aliases:["jsp"],keywords:KEYWORDS,illegal:/<\/|#/,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"class",beginKeywords:"class interface",end:/[{;=]/,excludeEnd:true,keywords:"class interface",illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"new throw return else",relevance:0},{className:"function",begin:"("+GENERIC_IDENT_RE+"\\s+)+"+hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,end:/[{;=]/,excludeEnd:true,keywords:KEYWORDS,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,relevance:0,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},JAVA_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("csp",function(hljs){return{case_insensitive:false,lexemes:"[a-zA-Z][a-zA-Z0-9_-]*",keywords:{keyword:"base-uri child-src connect-src default-src font-src form-action"+" frame-ancestors frame-src img-src media-src object-src plugin-types"+" report-uri sandbox script-src style-src"},contains:[{className:"string",begin:"'",end:"'"},{className:"attribute",begin:"^Content",end:":",excludeEnd:true}]}});hljs.registerLanguage("ruby",function(hljs){var RUBY_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?";var RUBY_KEYWORDS={keyword:"and then defined module in return redo if BEGIN retry end for self when "+"next until do begin unless END rescue else break undef not super class case "+"require yield alias while ensure elsif or include attr_reader attr_writer attr_accessor",literal:"true false nil"};var YARDOCTAG={className:"doctag",begin:"@[A-Za-z]+"};var IRB_OBJECT={begin:"#<",end:">"};var COMMENT_MODES=[hljs.COMMENT("#","$",{contains:[YARDOCTAG]}),hljs.COMMENT("^\\=begin","^\\=end",{contains:[YARDOCTAG],relevance:10}),hljs.COMMENT("^__END__","\\n$")];var SUBST={className:"subst",begin:"#\\{",end:"}",keywords:RUBY_KEYWORDS};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{begin:"%[qQwWx]?\\(",end:"\\)"},{begin:"%[qQwWx]?\\[",end:"\\]"},{begin:"%[qQwWx]?{",end:"}"},{begin:"%[qQwWx]?<",end:">"},{begin:"%[qQwWx]?/",end:"/"},{begin:"%[qQwWx]?%",end:"%"},{begin:"%[qQwWx]?-",end:"-"},{begin:"%[qQwWx]?\\|",end:"\\|"},{begin:/\B\?(\\\d{1,3}|\\x[A-Fa-f0-9]{1,2}|\\u[A-Fa-f0-9]{4}|\\?\S)\b/},{begin:/<<[-~]?'?(\w+)(?:.|\n)*?\n\s*\1\b/,returnBegin:true,contains:[{begin:/<<[-~]?'?/},{begin:/\w+/,endSameAsBegin:true,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]}]};var PARAMS={className:"params",begin:"\\(",end:"\\)",endsParent:true,keywords:RUBY_KEYWORDS};var RUBY_DEFAULT_CONTAINS=[STRING,IRB_OBJECT,{className:"class",beginKeywords:"class module",end:"$|;",illegal:/=/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?"}),{begin:"<\\s*",contains:[{begin:"("+hljs.IDENT_RE+"::)?"+hljs.IDENT_RE}]}].concat(COMMENT_MODES)},{className:"function",beginKeywords:"def",end:"$|;",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:RUBY_METHOD_RE}),PARAMS].concat(COMMENT_MODES)},{begin:hljs.IDENT_RE+"::"},{className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"(\\!|\\?)?:",relevance:0},{className:"symbol",begin:":(?!\\s)",contains:[STRING,{begin:RUBY_METHOD_RE}],relevance:0},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},{begin:"(\\$\\W)|((\\$|\\@\\@?)(\\w+))"},{className:"params",begin:/\|/,end:/\|/,keywords:RUBY_KEYWORDS},{begin:"("+hljs.RE_STARTERS_RE+"|unless)\\s*",keywords:"unless",contains:[IRB_OBJECT,{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],illegal:/\n/,variants:[{begin:"/",end:"/[a-z]*"},{begin:"%r{",end:"}[a-z]*"},{begin:"%r\\(",end:"\\)[a-z]*"},{begin:"%r!",end:"![a-z]*"},{begin:"%r\\[",end:"\\][a-z]*"}]}].concat(COMMENT_MODES),relevance:0}].concat(COMMENT_MODES);SUBST.contains=RUBY_DEFAULT_CONTAINS;PARAMS.contains=RUBY_DEFAULT_CONTAINS;var SIMPLE_PROMPT="[>?]>";var DEFAULT_PROMPT="[\\w#]+\\(\\w+\\):\\d+:\\d+>";var RVM_PROMPT="(\\w+-)?\\d+\\.\\d+\\.\\d(p\\d+)?[^>]+>";var IRB_DEFAULT=[{begin:/^\s*=>/,starts:{end:"$",contains:RUBY_DEFAULT_CONTAINS}},{className:"meta",begin:"^("+SIMPLE_PROMPT+"|"+DEFAULT_PROMPT+"|"+RVM_PROMPT+")",starts:{end:"$",contains:RUBY_DEFAULT_CONTAINS}}];return{aliases:["rb","gemspec","podspec","thor","irb"],keywords:RUBY_KEYWORDS,illegal:/\/\*/,contains:COMMENT_MODES.concat(IRB_DEFAULT).concat(RUBY_DEFAULT_CONTAINS)}});hljs.registerLanguage("cs",function(hljs){var KEYWORDS={keyword:"abstract as base bool break byte case catch char checked const continue decimal "+"default delegate do double enum event explicit extern finally fixed float "+"for foreach goto if implicit in int interface internal is lock long "+"object operator out override params private protected public readonly ref sbyte "+"sealed short sizeof stackalloc static string struct switch this try typeof "+"uint ulong unchecked unsafe ushort using virtual void volatile while "+"add alias ascending async await by descending dynamic equals from get global group into join "+"let nameof on orderby partial remove select set value var when where yield",literal:"null false true"};var NUMBERS={className:"number",variants:[{begin:"\\b(0b[01']+)"},{begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],relevance:0};var VERBATIM_STRING={className:"string",begin:'@"',end:'"',contains:[{begin:'""'}]};var VERBATIM_STRING_NO_LF=hljs.inherit(VERBATIM_STRING,{illegal:/\n/});var SUBST={className:"subst",begin:"{",end:"}",keywords:KEYWORDS};var SUBST_NO_LF=hljs.inherit(SUBST,{illegal:/\n/});var INTERPOLATED_STRING={className:"string",begin:/\$"/,end:'"',illegal:/\n/,contains:[{begin:"{{"},{begin:"}}"},hljs.BACKSLASH_ESCAPE,SUBST_NO_LF]};var INTERPOLATED_VERBATIM_STRING={className:"string",begin:/\$@"/,end:'"',contains:[{begin:"{{"},{begin:"}}"},{begin:'""'},SUBST]};var INTERPOLATED_VERBATIM_STRING_NO_LF=hljs.inherit(INTERPOLATED_VERBATIM_STRING,{illegal:/\n/,contains:[{begin:"{{"},{begin:"}}"},{begin:'""'},SUBST_NO_LF]});SUBST.contains=[INTERPOLATED_VERBATIM_STRING,INTERPOLATED_STRING,VERBATIM_STRING,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,NUMBERS,hljs.C_BLOCK_COMMENT_MODE];SUBST_NO_LF.contains=[INTERPOLATED_VERBATIM_STRING_NO_LF,INTERPOLATED_STRING,VERBATIM_STRING_NO_LF,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,NUMBERS,hljs.inherit(hljs.C_BLOCK_COMMENT_MODE,{illegal:/\n/})];var STRING={variants:[INTERPOLATED_VERBATIM_STRING,INTERPOLATED_STRING,VERBATIM_STRING,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]};var TYPE_IDENT_RE=hljs.IDENT_RE+"(<"+hljs.IDENT_RE+"(\\s*,\\s*"+hljs.IDENT_RE+")*>)?(\\[\\])?";return{aliases:["csharp","c#"],keywords:KEYWORDS,illegal:/::/,contains:[hljs.COMMENT("///","$",{returnBegin:true,contains:[{className:"doctag",variants:[{begin:"///",relevance:0},{begin:"\x3c!--|--\x3e"},{begin:"</?",end:">"}]}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elif endif define undef warning error line region endregion pragma checksum"}},STRING,NUMBERS,{beginKeywords:"class interface",end:/[{;=]/,illegal:/[^\s:,]/,contains:[hljs.TITLE_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{beginKeywords:"namespace",end:/[{;=]/,illegal:/[^\s:]/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"[a-zA-Z](\\.?\\w)*"}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"meta",begin:"^\\s*\\[",excludeBegin:true,end:"\\]",excludeEnd:true,contains:[{className:"meta-string",begin:/"/,end:/"/}]},{beginKeywords:"new return throw await else",relevance:0},{className:"function",begin:"("+TYPE_IDENT_RE+"\\s+)+"+hljs.IDENT_RE+"\\s*\\(",returnBegin:true,end:/\s*[{;=]/,excludeEnd:true,keywords:KEYWORDS,contains:[{begin:hljs.IDENT_RE+"\\s*\\(",returnBegin:true,contains:[hljs.TITLE_MODE],relevance:0},{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,relevance:0,contains:[STRING,NUMBERS,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}]}});hljs.registerLanguage("x86asm",function(hljs){return{case_insensitive:true,lexemes:"[.%]?"+hljs.IDENT_RE,keywords:{keyword:"lock rep repe repz repne repnz xaquire xrelease bnd nobnd "+"aaa aad aam aas adc add and arpl bb0_reset bb1_reset bound bsf bsr bswap bt btc btr bts call cbw cdq cdqe clc cld cli clts cmc cmp cmpsb cmpsd cmpsq cmpsw cmpxchg cmpxchg486 cmpxchg8b cmpxchg16b cpuid cpu_read cpu_write cqo cwd cwde daa das dec div dmint emms enter equ f2xm1 fabs fadd faddp fbld fbstp fchs fclex fcmovb fcmovbe fcmove fcmovnb fcmovnbe fcmovne fcmovnu fcmovu fcom fcomi fcomip fcomp fcompp fcos fdecstp fdisi fdiv fdivp fdivr fdivrp femms feni ffree ffreep fiadd ficom ficomp fidiv fidivr fild fimul fincstp finit fist fistp fisttp fisub fisubr fld fld1 fldcw fldenv fldl2e fldl2t fldlg2 fldln2 fldpi fldz fmul fmulp fnclex fndisi fneni fninit fnop fnsave fnstcw fnstenv fnstsw fpatan fprem fprem1 fptan frndint frstor fsave fscale fsetpm fsin fsincos fsqrt fst fstcw fstenv fstp fstsw fsub fsubp fsubr fsubrp ftst fucom fucomi fucomip fucomp fucompp fxam fxch fxtract fyl2x fyl2xp1 hlt ibts icebp idiv imul in inc incbin insb insd insw int int01 int1 int03 int3 into invd invpcid invlpg invlpga iret iretd iretq iretw jcxz jecxz jrcxz jmp jmpe lahf lar lds lea leave les lfence lfs lgdt lgs lidt lldt lmsw loadall loadall286 lodsb lodsd lodsq lodsw loop loope loopne loopnz loopz lsl lss ltr mfence monitor mov movd movq movsb movsd movsq movsw movsx movsxd movzx mul mwait neg nop not or out outsb outsd outsw packssdw packsswb packuswb paddb paddd paddsb paddsiw paddsw paddusb paddusw paddw pand pandn pause paveb pavgusb pcmpeqb pcmpeqd pcmpeqw pcmpgtb pcmpgtd pcmpgtw pdistib pf2id pfacc pfadd pfcmpeq pfcmpge pfcmpgt pfmax pfmin pfmul pfrcp pfrcpit1 pfrcpit2 pfrsqit1 pfrsqrt pfsub pfsubr pi2fd pmachriw pmaddwd pmagw pmulhriw pmulhrwa pmulhrwc pmulhw pmullw pmvgezb pmvlzb pmvnzb pmvzb pop popa popad popaw popf popfd popfq popfw por prefetch prefetchw pslld psllq psllw psrad psraw psrld psrlq psrlw psubb psubd psubsb psubsiw psubsw psubusb psubusw psubw punpckhbw punpckhdq punpckhwd punpcklbw punpckldq punpcklwd push pusha pushad pushaw pushf pushfd pushfq pushfw pxor rcl rcr rdshr rdmsr rdpmc rdtsc rdtscp ret retf retn rol ror rdm rsdc rsldt rsm rsts sahf sal salc sar sbb scasb scasd scasq scasw sfence sgdt shl shld shr shrd sidt sldt skinit smi smint smintold smsw stc std sti stosb stosd stosq stosw str sub svdc svldt svts swapgs syscall sysenter sysexit sysret test ud0 ud1 ud2b ud2 ud2a umov verr verw fwait wbinvd wrshr wrmsr xadd xbts xchg xlatb xlat xor cmove cmovz cmovne cmovnz cmova cmovnbe cmovae cmovnb cmovb cmovnae cmovbe cmovna cmovg cmovnle cmovge cmovnl cmovl cmovnge cmovle cmovng cmovc cmovnc cmovo cmovno cmovs cmovns cmovp cmovpe cmovnp cmovpo je jz jne jnz ja jnbe jae jnb jb jnae jbe jna jg jnle jge jnl jl jnge jle jng jc jnc jo jno js jns jpo jnp jpe jp sete setz setne setnz seta setnbe setae setnb setnc setb setnae setcset setbe setna setg setnle setge setnl setl setnge setle setng sets setns seto setno setpe setp setpo setnp addps addss andnps andps cmpeqps cmpeqss cmpleps cmpless cmpltps cmpltss cmpneqps cmpneqss cmpnleps cmpnless cmpnltps cmpnltss cmpordps cmpordss cmpunordps cmpunordss cmpps cmpss comiss cvtpi2ps cvtps2pi cvtsi2ss cvtss2si cvttps2pi cvttss2si divps divss ldmxcsr maxps maxss minps minss movaps movhps movlhps movlps movhlps movmskps movntps movss movups mulps mulss orps rcpps rcpss rsqrtps rsqrtss shufps sqrtps sqrtss stmxcsr subps subss ucomiss unpckhps unpcklps xorps fxrstor fxrstor64 fxsave fxsave64 xgetbv xsetbv xsave xsave64 xsaveopt xsaveopt64 xrstor xrstor64 prefetchnta prefetcht0 prefetcht1 prefetcht2 maskmovq movntq pavgb pavgw pextrw pinsrw pmaxsw pmaxub pminsw pminub pmovmskb pmulhuw psadbw pshufw pf2iw pfnacc pfpnacc pi2fw pswapd maskmovdqu clflush movntdq movnti movntpd movdqa movdqu movdq2q movq2dq paddq pmuludq pshufd pshufhw pshuflw pslldq psrldq psubq punpckhqdq punpcklqdq addpd addsd andnpd andpd cmpeqpd cmpeqsd cmplepd cmplesd cmpltpd cmpltsd cmpneqpd cmpneqsd cmpnlepd cmpnlesd cmpnltpd cmpnltsd cmpordpd cmpordsd cmpunordpd cmpunordsd cmppd comisd cvtdq2pd cvtdq2ps cvtpd2dq cvtpd2pi cvtpd2ps cvtpi2pd cvtps2dq cvtps2pd cvtsd2si cvtsd2ss cvtsi2sd cvtss2sd cvttpd2pi cvttpd2dq cvttps2dq cvttsd2si divpd divsd maxpd maxsd minpd minsd movapd movhpd movlpd movmskpd movupd mulpd mulsd orpd shufpd sqrtpd sqrtsd subpd subsd ucomisd unpckhpd unpcklpd xorpd addsubpd addsubps haddpd haddps hsubpd hsubps lddqu movddup movshdup movsldup clgi stgi vmcall vmclear vmfunc vmlaunch vmload vmmcall vmptrld vmptrst vmread vmresume vmrun vmsave vmwrite vmxoff vmxon invept invvpid pabsb pabsw pabsd palignr phaddw phaddd phaddsw phsubw phsubd phsubsw pmaddubsw pmulhrsw pshufb psignb psignw psignd extrq insertq movntsd movntss lzcnt blendpd blendps blendvpd blendvps dppd dpps extractps insertps movntdqa mpsadbw packusdw pblendvb pblendw pcmpeqq pextrb pextrd pextrq phminposuw pinsrb pinsrd pinsrq pmaxsb pmaxsd pmaxud pmaxuw pminsb pminsd pminud pminuw pmovsxbw pmovsxbd pmovsxbq pmovsxwd pmovsxwq pmovsxdq pmovzxbw pmovzxbd pmovzxbq pmovzxwd pmovzxwq pmovzxdq pmuldq pmulld ptest roundpd roundps roundsd roundss crc32 pcmpestri pcmpestrm pcmpistri pcmpistrm pcmpgtq popcnt getsec pfrcpv pfrsqrtv movbe aesenc aesenclast aesdec aesdeclast aesimc aeskeygenassist vaesenc vaesenclast vaesdec vaesdeclast vaesimc vaeskeygenassist vaddpd vaddps vaddsd vaddss vaddsubpd vaddsubps vandpd vandps vandnpd vandnps vblendpd vblendps vblendvpd vblendvps vbroadcastss vbroadcastsd vbroadcastf128 vcmpeq_ospd vcmpeqpd vcmplt_ospd vcmpltpd vcmple_ospd vcmplepd vcmpunord_qpd vcmpunordpd vcmpneq_uqpd vcmpneqpd vcmpnlt_uspd vcmpnltpd vcmpnle_uspd vcmpnlepd vcmpord_qpd vcmpordpd vcmpeq_uqpd vcmpnge_uspd vcmpngepd vcmpngt_uspd vcmpngtpd vcmpfalse_oqpd vcmpfalsepd vcmpneq_oqpd vcmpge_ospd vcmpgepd vcmpgt_ospd vcmpgtpd vcmptrue_uqpd vcmptruepd vcmplt_oqpd vcmple_oqpd vcmpunord_spd vcmpneq_uspd vcmpnlt_uqpd vcmpnle_uqpd vcmpord_spd vcmpeq_uspd vcmpnge_uqpd vcmpngt_uqpd vcmpfalse_ospd vcmpneq_ospd vcmpge_oqpd vcmpgt_oqpd vcmptrue_uspd vcmppd vcmpeq_osps vcmpeqps vcmplt_osps vcmpltps vcmple_osps vcmpleps vcmpunord_qps vcmpunordps vcmpneq_uqps vcmpneqps vcmpnlt_usps vcmpnltps vcmpnle_usps vcmpnleps vcmpord_qps vcmpordps vcmpeq_uqps vcmpnge_usps vcmpngeps vcmpngt_usps vcmpngtps vcmpfalse_oqps vcmpfalseps vcmpneq_oqps vcmpge_osps vcmpgeps vcmpgt_osps vcmpgtps vcmptrue_uqps vcmptrueps vcmplt_oqps vcmple_oqps vcmpunord_sps vcmpneq_usps vcmpnlt_uqps vcmpnle_uqps vcmpord_sps vcmpeq_usps vcmpnge_uqps vcmpngt_uqps vcmpfalse_osps vcmpneq_osps vcmpge_oqps vcmpgt_oqps vcmptrue_usps vcmpps vcmpeq_ossd vcmpeqsd vcmplt_ossd vcmpltsd vcmple_ossd vcmplesd vcmpunord_qsd vcmpunordsd vcmpneq_uqsd vcmpneqsd vcmpnlt_ussd vcmpnltsd vcmpnle_ussd vcmpnlesd vcmpord_qsd vcmpordsd vcmpeq_uqsd vcmpnge_ussd vcmpngesd vcmpngt_ussd vcmpngtsd vcmpfalse_oqsd vcmpfalsesd vcmpneq_oqsd vcmpge_ossd vcmpgesd vcmpgt_ossd vcmpgtsd vcmptrue_uqsd vcmptruesd vcmplt_oqsd vcmple_oqsd vcmpunord_ssd vcmpneq_ussd vcmpnlt_uqsd vcmpnle_uqsd vcmpord_ssd vcmpeq_ussd vcmpnge_uqsd vcmpngt_uqsd vcmpfalse_ossd vcmpneq_ossd vcmpge_oqsd vcmpgt_oqsd vcmptrue_ussd vcmpsd vcmpeq_osss vcmpeqss vcmplt_osss vcmpltss vcmple_osss vcmpless vcmpunord_qss vcmpunordss vcmpneq_uqss vcmpneqss vcmpnlt_usss vcmpnltss vcmpnle_usss vcmpnless vcmpord_qss vcmpordss vcmpeq_uqss vcmpnge_usss vcmpngess vcmpngt_usss vcmpngtss vcmpfalse_oqss vcmpfalsess vcmpneq_oqss vcmpge_osss vcmpgess vcmpgt_osss vcmpgtss vcmptrue_uqss vcmptruess vcmplt_oqss vcmple_oqss vcmpunord_sss vcmpneq_usss vcmpnlt_uqss vcmpnle_uqss vcmpord_sss vcmpeq_usss vcmpnge_uqss vcmpngt_uqss vcmpfalse_osss vcmpneq_osss vcmpge_oqss vcmpgt_oqss vcmptrue_usss vcmpss vcomisd vcomiss vcvtdq2pd vcvtdq2ps vcvtpd2dq vcvtpd2ps vcvtps2dq vcvtps2pd vcvtsd2si vcvtsd2ss vcvtsi2sd vcvtsi2ss vcvtss2sd vcvtss2si vcvttpd2dq vcvttps2dq vcvttsd2si vcvttss2si vdivpd vdivps vdivsd vdivss vdppd vdpps vextractf128 vextractps vhaddpd vhaddps vhsubpd vhsubps vinsertf128 vinsertps vlddqu vldqqu vldmxcsr vmaskmovdqu vmaskmovps vmaskmovpd vmaxpd vmaxps vmaxsd vmaxss vminpd vminps vminsd vminss vmovapd vmovaps vmovd vmovq vmovddup vmovdqa vmovqqa vmovdqu vmovqqu vmovhlps vmovhpd vmovhps vmovlhps vmovlpd vmovlps vmovmskpd vmovmskps vmovntdq vmovntqq vmovntdqa vmovntpd vmovntps vmovsd vmovshdup vmovsldup vmovss vmovupd vmovups vmpsadbw vmulpd vmulps vmulsd vmulss vorpd vorps vpabsb vpabsw vpabsd vpacksswb vpackssdw vpackuswb vpackusdw vpaddb vpaddw vpaddd vpaddq vpaddsb vpaddsw vpaddusb vpaddusw vpalignr vpand vpandn vpavgb vpavgw vpblendvb vpblendw vpcmpestri vpcmpestrm vpcmpistri vpcmpistrm vpcmpeqb vpcmpeqw vpcmpeqd vpcmpeqq vpcmpgtb vpcmpgtw vpcmpgtd vpcmpgtq vpermilpd vpermilps vperm2f128 vpextrb vpextrw vpextrd vpextrq vphaddw vphaddd vphaddsw vphminposuw vphsubw vphsubd vphsubsw vpinsrb vpinsrw vpinsrd vpinsrq vpmaddwd vpmaddubsw vpmaxsb vpmaxsw vpmaxsd vpmaxub vpmaxuw vpmaxud vpminsb vpminsw vpminsd vpminub vpminuw vpminud vpmovmskb vpmovsxbw vpmovsxbd vpmovsxbq vpmovsxwd vpmovsxwq vpmovsxdq vpmovzxbw vpmovzxbd vpmovzxbq vpmovzxwd vpmovzxwq vpmovzxdq vpmulhuw vpmulhrsw vpmulhw vpmullw vpmulld vpmuludq vpmuldq vpor vpsadbw vpshufb vpshufd vpshufhw vpshuflw vpsignb vpsignw vpsignd vpslldq vpsrldq vpsllw vpslld vpsllq vpsraw vpsrad vpsrlw vpsrld vpsrlq vptest vpsubb vpsubw vpsubd vpsubq vpsubsb vpsubsw vpsubusb vpsubusw vpunpckhbw vpunpckhwd vpunpckhdq vpunpckhqdq vpunpcklbw vpunpcklwd vpunpckldq vpunpcklqdq vpxor vrcpps vrcpss vrsqrtps vrsqrtss vroundpd vroundps vroundsd vroundss vshufpd vshufps vsqrtpd vsqrtps vsqrtsd vsqrtss vstmxcsr vsubpd vsubps vsubsd vsubss vtestps vtestpd vucomisd vucomiss vunpckhpd vunpckhps vunpcklpd vunpcklps vxorpd vxorps vzeroall vzeroupper pclmullqlqdq pclmulhqlqdq pclmullqhqdq pclmulhqhqdq pclmulqdq vpclmullqlqdq vpclmulhqlqdq vpclmullqhqdq vpclmulhqhqdq vpclmulqdq vfmadd132ps vfmadd132pd vfmadd312ps vfmadd312pd vfmadd213ps vfmadd213pd vfmadd123ps vfmadd123pd vfmadd231ps vfmadd231pd vfmadd321ps vfmadd321pd vfmaddsub132ps vfmaddsub132pd vfmaddsub312ps vfmaddsub312pd vfmaddsub213ps vfmaddsub213pd vfmaddsub123ps vfmaddsub123pd vfmaddsub231ps vfmaddsub231pd vfmaddsub321ps vfmaddsub321pd vfmsub132ps vfmsub132pd vfmsub312ps vfmsub312pd vfmsub213ps vfmsub213pd vfmsub123ps vfmsub123pd vfmsub231ps vfmsub231pd vfmsub321ps vfmsub321pd vfmsubadd132ps vfmsubadd132pd vfmsubadd312ps vfmsubadd312pd vfmsubadd213ps vfmsubadd213pd vfmsubadd123ps vfmsubadd123pd vfmsubadd231ps vfmsubadd231pd vfmsubadd321ps vfmsubadd321pd vfnmadd132ps vfnmadd132pd vfnmadd312ps vfnmadd312pd vfnmadd213ps vfnmadd213pd vfnmadd123ps vfnmadd123pd vfnmadd231ps vfnmadd231pd vfnmadd321ps vfnmadd321pd vfnmsub132ps vfnmsub132pd vfnmsub312ps vfnmsub312pd vfnmsub213ps vfnmsub213pd vfnmsub123ps vfnmsub123pd vfnmsub231ps vfnmsub231pd vfnmsub321ps vfnmsub321pd vfmadd132ss vfmadd132sd vfmadd312ss vfmadd312sd vfmadd213ss vfmadd213sd vfmadd123ss vfmadd123sd vfmadd231ss vfmadd231sd vfmadd321ss vfmadd321sd vfmsub132ss vfmsub132sd vfmsub312ss vfmsub312sd vfmsub213ss vfmsub213sd vfmsub123ss vfmsub123sd vfmsub231ss vfmsub231sd vfmsub321ss vfmsub321sd vfnmadd132ss vfnmadd132sd vfnmadd312ss vfnmadd312sd vfnmadd213ss vfnmadd213sd vfnmadd123ss vfnmadd123sd vfnmadd231ss vfnmadd231sd vfnmadd321ss vfnmadd321sd vfnmsub132ss vfnmsub132sd vfnmsub312ss vfnmsub312sd vfnmsub213ss vfnmsub213sd vfnmsub123ss vfnmsub123sd vfnmsub231ss vfnmsub231sd vfnmsub321ss vfnmsub321sd rdfsbase rdgsbase rdrand wrfsbase wrgsbase vcvtph2ps vcvtps2ph adcx adox rdseed clac stac xstore xcryptecb xcryptcbc xcryptctr xcryptcfb xcryptofb montmul xsha1 xsha256 llwpcb slwpcb lwpval lwpins vfmaddpd vfmaddps vfmaddsd vfmaddss vfmaddsubpd vfmaddsubps vfmsubaddpd vfmsubaddps vfmsubpd vfmsubps vfmsubsd vfmsubss vfnmaddpd vfnmaddps vfnmaddsd vfnmaddss vfnmsubpd vfnmsubps vfnmsubsd vfnmsubss vfrczpd vfrczps vfrczsd vfrczss vpcmov vpcomb vpcomd vpcomq vpcomub vpcomud vpcomuq vpcomuw vpcomw vphaddbd vphaddbq vphaddbw vphadddq vphaddubd vphaddubq vphaddubw vphaddudq vphadduwd vphadduwq vphaddwd vphaddwq vphsubbw vphsubdq vphsubwd vpmacsdd vpmacsdqh vpmacsdql vpmacssdd vpmacssdqh vpmacssdql vpmacsswd vpmacssww vpmacswd vpmacsww vpmadcsswd vpmadcswd vpperm vprotb vprotd vprotq vprotw vpshab vpshad vpshaq vpshaw vpshlb vpshld vpshlq vpshlw vbroadcasti128 vpblendd vpbroadcastb vpbroadcastw vpbroadcastd vpbroadcastq vpermd vpermpd vpermps vpermq vperm2i128 vextracti128 vinserti128 vpmaskmovd vpmaskmovq vpsllvd vpsllvq vpsravd vpsrlvd vpsrlvq vgatherdpd vgatherqpd vgatherdps vgatherqps vpgatherdd vpgatherqd vpgatherdq vpgatherqq xabort xbegin xend xtest andn bextr blci blcic blsi blsic blcfill blsfill blcmsk blsmsk blsr blcs bzhi mulx pdep pext rorx sarx shlx shrx tzcnt tzmsk t1mskc valignd valignq vblendmpd vblendmps vbroadcastf32x4 vbroadcastf64x4 vbroadcasti32x4 vbroadcasti64x4 vcompresspd vcompressps vcvtpd2udq vcvtps2udq vcvtsd2usi vcvtss2usi vcvttpd2udq vcvttps2udq vcvttsd2usi vcvttss2usi vcvtudq2pd vcvtudq2ps vcvtusi2sd vcvtusi2ss vexpandpd vexpandps vextractf32x4 vextractf64x4 vextracti32x4 vextracti64x4 vfixupimmpd vfixupimmps vfixupimmsd vfixupimmss vgetexppd vgetexpps vgetexpsd vgetexpss vgetmantpd vgetmantps vgetmantsd vgetmantss vinsertf32x4 vinsertf64x4 vinserti32x4 vinserti64x4 vmovdqa32 vmovdqa64 vmovdqu32 vmovdqu64 vpabsq vpandd vpandnd vpandnq vpandq vpblendmd vpblendmq vpcmpltd vpcmpled vpcmpneqd vpcmpnltd vpcmpnled vpcmpd vpcmpltq vpcmpleq vpcmpneqq vpcmpnltq vpcmpnleq vpcmpq vpcmpequd vpcmpltud vpcmpleud vpcmpnequd vpcmpnltud vpcmpnleud vpcmpud vpcmpequq vpcmpltuq vpcmpleuq vpcmpnequq vpcmpnltuq vpcmpnleuq vpcmpuq vpcompressd vpcompressq vpermi2d vpermi2pd vpermi2ps vpermi2q vpermt2d vpermt2pd vpermt2ps vpermt2q vpexpandd vpexpandq vpmaxsq vpmaxuq vpminsq vpminuq vpmovdb vpmovdw vpmovqb vpmovqd vpmovqw vpmovsdb vpmovsdw vpmovsqb vpmovsqd vpmovsqw vpmovusdb vpmovusdw vpmovusqb vpmovusqd vpmovusqw vpord vporq vprold vprolq vprolvd vprolvq vprord vprorq vprorvd vprorvq vpscatterdd vpscatterdq vpscatterqd vpscatterqq vpsraq vpsravq vpternlogd vpternlogq vptestmd vptestmq vptestnmd vptestnmq vpxord vpxorq vrcp14pd vrcp14ps vrcp14sd vrcp14ss vrndscalepd vrndscaleps vrndscalesd vrndscaless vrsqrt14pd vrsqrt14ps vrsqrt14sd vrsqrt14ss vscalefpd vscalefps vscalefsd vscalefss vscatterdpd vscatterdps vscatterqpd vscatterqps vshuff32x4 vshuff64x2 vshufi32x4 vshufi64x2 kandnw kandw kmovw knotw kortestw korw kshiftlw kshiftrw kunpckbw kxnorw kxorw vpbroadcastmb2q vpbroadcastmw2d vpconflictd vpconflictq vplzcntd vplzcntq vexp2pd vexp2ps vrcp28pd vrcp28ps vrcp28sd vrcp28ss vrsqrt28pd vrsqrt28ps vrsqrt28sd vrsqrt28ss vgatherpf0dpd vgatherpf0dps vgatherpf0qpd vgatherpf0qps vgatherpf1dpd vgatherpf1dps vgatherpf1qpd vgatherpf1qps vscatterpf0dpd vscatterpf0dps vscatterpf0qpd vscatterpf0qps vscatterpf1dpd vscatterpf1dps vscatterpf1qpd vscatterpf1qps prefetchwt1 bndmk bndcl bndcu bndcn bndmov bndldx bndstx sha1rnds4 sha1nexte sha1msg1 sha1msg2 sha256rnds2 sha256msg1 sha256msg2 hint_nop0 hint_nop1 hint_nop2 hint_nop3 hint_nop4 hint_nop5 hint_nop6 hint_nop7 hint_nop8 hint_nop9 hint_nop10 hint_nop11 hint_nop12 hint_nop13 hint_nop14 hint_nop15 hint_nop16 hint_nop17 hint_nop18 hint_nop19 hint_nop20 hint_nop21 hint_nop22 hint_nop23 hint_nop24 hint_nop25 hint_nop26 hint_nop27 hint_nop28 hint_nop29 hint_nop30 hint_nop31 hint_nop32 hint_nop33 hint_nop34 hint_nop35 hint_nop36 hint_nop37 hint_nop38 hint_nop39 hint_nop40 hint_nop41 hint_nop42 hint_nop43 hint_nop44 hint_nop45 hint_nop46 hint_nop47 hint_nop48 hint_nop49 hint_nop50 hint_nop51 hint_nop52 hint_nop53 hint_nop54 hint_nop55 hint_nop56 hint_nop57 hint_nop58 hint_nop59 hint_nop60 hint_nop61 hint_nop62 hint_nop63",built_in:"ip eip rip "+"al ah bl bh cl ch dl dh sil dil bpl spl r8b r9b r10b r11b r12b r13b r14b r15b "+"ax bx cx dx si di bp sp r8w r9w r10w r11w r12w r13w r14w r15w "+"eax ebx ecx edx esi edi ebp esp eip r8d r9d r10d r11d r12d r13d r14d r15d "+"rax rbx rcx rdx rsi rdi rbp rsp r8 r9 r10 r11 r12 r13 r14 r15 "+"cs ds es fs gs ss "+"st st0 st1 st2 st3 st4 st5 st6 st7 "+"mm0 mm1 mm2 mm3 mm4 mm5 mm6 mm7 "+"xmm0  xmm1  xmm2  xmm3  xmm4  xmm5  xmm6  xmm7  xmm8  xmm9 xmm10  xmm11 xmm12 xmm13 xmm14 xmm15 "+"xmm16 xmm17 xmm18 xmm19 xmm20 xmm21 xmm22 xmm23 xmm24 xmm25 xmm26 xmm27 xmm28 xmm29 xmm30 xmm31 "+"ymm0  ymm1  ymm2  ymm3  ymm4  ymm5  ymm6  ymm7  ymm8  ymm9 ymm10  ymm11 ymm12 ymm13 ymm14 ymm15 "+"ymm16 ymm17 ymm18 ymm19 ymm20 ymm21 ymm22 ymm23 ymm24 ymm25 ymm26 ymm27 ymm28 ymm29 ymm30 ymm31 "+"zmm0  zmm1  zmm2  zmm3  zmm4  zmm5  zmm6  zmm7  zmm8  zmm9 zmm10  zmm11 zmm12 zmm13 zmm14 zmm15 "+"zmm16 zmm17 zmm18 zmm19 zmm20 zmm21 zmm22 zmm23 zmm24 zmm25 zmm26 zmm27 zmm28 zmm29 zmm30 zmm31 "+"k0 k1 k2 k3 k4 k5 k6 k7 "+"bnd0 bnd1 bnd2 bnd3 "+"cr0 cr1 cr2 cr3 cr4 cr8 dr0 dr1 dr2 dr3 dr8 tr3 tr4 tr5 tr6 tr7 "+"r0 r1 r2 r3 r4 r5 r6 r7 r0b r1b r2b r3b r4b r5b r6b r7b "+"r0w r1w r2w r3w r4w r5w r6w r7w r0d r1d r2d r3d r4d r5d r6d r7d "+"r0h r1h r2h r3h "+"r0l r1l r2l r3l r4l r5l r6l r7l r8l r9l r10l r11l r12l r13l r14l r15l "+"db dw dd dq dt ddq do dy dz "+"resb resw resd resq rest resdq reso resy resz "+"incbin equ times "+"byte word dword qword nosplit rel abs seg wrt strict near far a32 ptr",meta:"%define %xdefine %+ %undef %defstr %deftok %assign %strcat %strlen %substr %rotate %elif %else %endif "+"%if %ifmacro %ifctx %ifidn %ifidni %ifid %ifnum %ifstr %iftoken %ifempty %ifenv %error %warning %fatal %rep "+"%endrep %include %push %pop %repl %pathsearch %depend %use %arg %stacksize %local %line %comment %endcomment "+".nolist "+"__FILE__ __LINE__ __SECT__  __BITS__ __OUTPUT_FORMAT__ __DATE__ __TIME__ __DATE_NUM__ __TIME_NUM__ "+"__UTC_DATE__ __UTC_TIME__ __UTC_DATE_NUM__ __UTC_TIME_NUM__  __PASS__ struc endstruc istruc at iend "+"align alignb sectalign daz nodaz up down zero default option assume public "+"bits use16 use32 use64 default section segment absolute extern global common cpu float "+"__utf16__ __utf16le__ __utf16be__ __utf32__ __utf32le__ __utf32be__ "+"__float8__ __float16__ __float32__ __float64__ __float80m__ __float80e__ __float128l__ __float128h__ "+"__Infinity__ __QNaN__ __SNaN__ Inf NaN QNaN SNaN float8 float16 float32 float64 float80m float80e "+"float128l float128h __FLOAT_DAZ__ __FLOAT_ROUND__ __FLOAT__"},contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"number",variants:[{begin:"\\b(?:([0-9][0-9_]*)?\\.[0-9_]*(?:[eE][+-]?[0-9_]+)?|"+"(0[Xx])?[0-9][0-9_]*\\.?[0-9_]*(?:[pP](?:[+-]?[0-9_]+)?)?)\\b",relevance:0},{begin:"\\$[0-9][0-9A-Fa-f]*",relevance:0},{begin:"\\b(?:[0-9A-Fa-f][0-9A-Fa-f_]*[Hh]|[0-9][0-9_]*[DdTt]?|[0-7][0-7_]*[QqOo]|[0-1][0-1_]*[BbYy])\\b"},{begin:"\\b(?:0[Xx][0-9A-Fa-f_]+|0[DdTt][0-9_]+|0[QqOo][0-7_]+|0[BbYy][0-1_]+)\\b"}]},hljs.QUOTE_STRING_MODE,{className:"string",variants:[{begin:"'",end:"[^\\\\]'"},{begin:"`",end:"[^\\\\]`"}],relevance:0},{className:"symbol",variants:[{begin:"^\\s*[A-Za-z._?][A-Za-z0-9_$#@~.?]*(:|\\s+label)"},{begin:"^\\s*%%[A-Za-z0-9_$#@~.?]*:"}],relevance:0},{className:"subst",begin:"%[0-9]+",relevance:0},{className:"subst",begin:"%!S+",relevance:0},{className:"meta",begin:/^\s*\.[\w_-]+/}]}});hljs.registerLanguage("mipsasm",function(hljs){return{case_insensitive:true,aliases:["mips"],lexemes:"\\.?"+hljs.IDENT_RE,keywords:{meta:".2byte .4byte .align .ascii .asciz .balign .byte .code .data .else .end .endif .endm .endr .equ .err .exitm .extern .global .hword .if .ifdef .ifndef .include .irp .long .macro .rept .req .section .set .skip .space .text .word .ltorg ",built_in:"$0 $1 $2 $3 $4 $5 $6 $7 $8 $9 $10 $11 $12 $13 $14 $15 "+"$16 $17 $18 $19 $20 $21 $22 $23 $24 $25 $26 $27 $28 $29 $30 $31 "+"zero at v0 v1 a0 a1 a2 a3 a4 a5 a6 a7 "+"t0 t1 t2 t3 t4 t5 t6 t7 t8 t9 s0 s1 s2 s3 s4 s5 s6 s7 s8 "+"k0 k1 gp sp fp ra "+"$f0 $f1 $f2 $f2 $f4 $f5 $f6 $f7 $f8 $f9 $f10 $f11 $f12 $f13 $f14 $f15 "+"$f16 $f17 $f18 $f19 $f20 $f21 $f22 $f23 $f24 $f25 $f26 $f27 $f28 $f29 $f30 $f31 "+"Context Random EntryLo0 EntryLo1 Context PageMask Wired EntryHi "+"HWREna BadVAddr Count Compare SR IntCtl SRSCtl SRSMap Cause EPC PRId "+"EBase Config Config1 Config2 Config3 LLAddr Debug DEPC DESAVE CacheErr "+"ECC ErrorEPC TagLo DataLo TagHi DataHi WatchLo WatchHi PerfCtl PerfCnt "},contains:[{className:"keyword",begin:"\\b("+"addi?u?|andi?|b(al)?|beql?|bgez(al)?l?|bgtzl?|blezl?|bltz(al)?l?|"+"bnel?|cl[oz]|divu?|ext|ins|j(al)?|jalr(.hb)?|jr(.hb)?|lbu?|lhu?|"+"ll|lui|lw[lr]?|maddu?|mfhi|mflo|movn|movz|move|msubu?|mthi|mtlo|mul|"+"multu?|nop|nor|ori?|rotrv?|sb|sc|se[bh]|sh|sllv?|slti?u?|srav?|"+"srlv?|subu?|sw[lr]?|xori?|wsbh|"+"abs.[sd]|add.[sd]|alnv.ps|bc1[ft]l?|"+"c.(s?f|un|u?eq|[ou]lt|[ou]le|ngle?|seq|l[et]|ng[et]).[sd]|"+"(ceil|floor|round|trunc).[lw].[sd]|cfc1|cvt.d.[lsw]|"+"cvt.l.[dsw]|cvt.ps.s|cvt.s.[dlw]|cvt.s.p[lu]|cvt.w.[dls]|"+"div.[ds]|ldx?c1|luxc1|lwx?c1|madd.[sd]|mfc1|mov[fntz]?.[ds]|"+"msub.[sd]|mth?c1|mul.[ds]|neg.[ds]|nmadd.[ds]|nmsub.[ds]|"+"p[lu][lu].ps|recip.fmt|r?sqrt.[ds]|sdx?c1|sub.[ds]|suxc1|"+"swx?c1|"+"break|cache|d?eret|[de]i|ehb|mfc0|mtc0|pause|prefx?|rdhwr|"+"rdpgpr|sdbbp|ssnop|synci?|syscall|teqi?|tgei?u?|tlb(p|r|w[ir])|"+"tlti?u?|tnei?|wait|wrpgpr"+")",end:"\\s"},hljs.COMMENT("[;#](?!s*$)","$"),hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",relevance:0},{className:"title",begin:"\\|",end:"\\|",illegal:"\\n",relevance:0},{className:"number",variants:[{begin:"0x[0-9a-f]+"},{begin:"\\b-?\\d+"}],relevance:0},{className:"symbol",variants:[{begin:"^\\s*[a-z_\\.\\$][a-z0-9_\\.\\$]+:"},{begin:"^\\s*[0-9]+:"},{begin:"[0-9]+[bf]"}],relevance:0}],illegal:"/"}});hljs.registerLanguage("gams",function(hljs){var KEYWORDS={keyword:"abort acronym acronyms alias all and assign binary card diag display "+"else eq file files for free ge gt if integer le loop lt maximizing "+"minimizing model models ne negative no not option options or ord "+"positive prod put putpage puttl repeat sameas semicont semiint smax "+"smin solve sos1 sos2 sum system table then until using while xor yes",literal:"eps inf na","built-in":"abs arccos arcsin arctan arctan2 Beta betaReg binomial ceil centropy "+"cos cosh cvPower div div0 eDist entropy errorf execSeed exp fact "+"floor frac gamma gammaReg log logBeta logGamma log10 log2 mapVal max "+"min mod ncpCM ncpF ncpVUpow ncpVUsin normal pi poly power "+"randBinomial randLinear randTriangle round rPower sigmoid sign "+"signPower sin sinh slexp sllog10 slrec sqexp sqlog10 sqr sqrec sqrt "+"tan tanh trunc uniform uniformInt vcPower bool_and bool_eqv bool_imp "+"bool_not bool_or bool_xor ifThen rel_eq rel_ge rel_gt rel_le rel_lt "+"rel_ne gday gdow ghour gleap gmillisec gminute gmonth gsecond gyear "+"jdate jnow jstart jtime errorLevel execError gamsRelease gamsVersion "+"handleCollect handleDelete handleStatus handleSubmit heapFree "+"heapLimit heapSize jobHandle jobKill jobStatus jobTerminate "+"licenseLevel licenseStatus maxExecError sleep timeClose timeComp "+"timeElapsed timeExec timeStart"};var PARAMS={className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true};var SYMBOLS={className:"symbol",variants:[{begin:/\=[lgenxc]=/},{begin:/\$/}]};var QSTR={className:"comment",variants:[{begin:"'",end:"'"},{begin:'"',end:'"'}],illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]};var ASSIGNMENT={begin:"/",end:"/",keywords:KEYWORDS,contains:[QSTR,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE]};var DESCTEXT={begin:/[a-z][a-z0-9_]*(\([a-z0-9_, ]*\))?[ \t]+/,excludeBegin:true,end:"$",endsWithParent:true,contains:[QSTR,ASSIGNMENT,{className:"comment",begin:/([ ]*[a-z0-9&#*=?@>\\<:\-,()$\[\]_.{}!+%^]+)+/,relevance:0}]};return{aliases:["gms"],case_insensitive:true,keywords:KEYWORDS,contains:[hljs.COMMENT(/^\$ontext/,/^\$offtext/),{className:"meta",begin:"^\\$[a-z0-9]+",end:"$",returnBegin:true,contains:[{className:"meta-keyword",begin:"^\\$[a-z0-9]+"}]},hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{beginKeywords:"set sets parameter parameters variable variables "+"scalar scalars equation equations",end:";",contains:[hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,ASSIGNMENT,DESCTEXT]},{beginKeywords:"table",end:";",returnBegin:true,contains:[{beginKeywords:"table",end:"$",contains:[DESCTEXT]},hljs.COMMENT("^\\*","$"),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE]},{className:"function",begin:/^[a-z][a-z0-9_,\-+' ()$]+\.{2}/,returnBegin:true,contains:[{className:"title",begin:/^[a-z0-9_]+/},PARAMS,SYMBOLS]},hljs.C_NUMBER_MODE,SYMBOLS]}});hljs.registerLanguage("nimrod",function(hljs){return{aliases:["nim"],keywords:{keyword:"addr and as asm bind block break case cast const continue converter "+"discard distinct div do elif else end enum except export finally "+"for from generic if import in include interface is isnot iterator "+"let macro method mixin mod nil not notin object of or out proc ptr "+"raise ref return shl shr static template try tuple type using var "+"when while with without xor yield",literal:"shared guarded stdin stdout stderr result true false",built_in:"int int8 int16 int32 int64 uint uint8 uint16 uint32 uint64 float "+"float32 float64 bool char string cstring pointer expr stmt void "+"auto any range array openarray varargs seq set clong culong cchar "+"cschar cshort cint csize clonglong cfloat cdouble clongdouble "+"cuchar cushort cuint culonglong cstringarray semistatic"},contains:[{className:"meta",begin:/{\./,end:/\.}/,relevance:10},{className:"string",begin:/[a-zA-Z]\w*"/,end:/"/,contains:[{begin:/""/}]},{className:"string",begin:/([a-zA-Z]\w*)?"""/,end:/"""/},hljs.QUOTE_STRING_MODE,{className:"type",begin:/\b[A-Z]\w+\b/,relevance:0},{className:"number",relevance:0,variants:[{begin:/\b(0[xX][0-9a-fA-F][_0-9a-fA-F]*)('?[iIuU](8|16|32|64))?/},{begin:/\b(0o[0-7][_0-7]*)('?[iIuUfF](8|16|32|64))?/},{begin:/\b(0(b|B)[01][_01]*)('?[iIuUfF](8|16|32|64))?/},{begin:/\b(\d[_\d]*)('?[iIuUfF](8|16|32|64))?/}]},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("dts",function(hljs){var STRINGS={className:"string",variants:[hljs.inherit(hljs.QUOTE_STRING_MODE,{begin:'((u8?|U)|L)?"'}),{begin:'(u8?|U)?R"',end:'"',contains:[hljs.BACKSLASH_ESCAPE]},{begin:"'\\\\?.",end:"'",illegal:"."}]};var NUMBERS={className:"number",variants:[{begin:"\\b(\\d+(\\.\\d*)?|\\.\\d+)(u|U|l|L|ul|UL|f|F)"},{begin:hljs.C_NUMBER_RE}],relevance:0};var PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elif endif define undef ifdef ifndef"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",end:"$",keywords:{"meta-keyword":"include"},contains:[hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:"<",end:">",illegal:"\\n"}]},STRINGS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};var DTS_REFERENCE={className:"variable",begin:"\\&[a-z\\d_]*\\b"};var DTS_KEYWORD={className:"meta-keyword",begin:"/[a-z][a-z\\d-]*/"};var DTS_LABEL={className:"symbol",begin:"^\\s*[a-zA-Z_][a-zA-Z\\d_]*:"};var DTS_CELL_PROPERTY={className:"params",begin:"<",end:">",contains:[NUMBERS,DTS_REFERENCE]};var DTS_NODE={className:"class",begin:/[a-zA-Z_][a-zA-Z\d_@]*\s{/,end:/[{;=]/,returnBegin:true,excludeEnd:true};var DTS_ROOT_NODE={className:"class",begin:"/\\s*{",end:"};",relevance:10,contains:[DTS_REFERENCE,DTS_KEYWORD,DTS_LABEL,DTS_NODE,DTS_CELL_PROPERTY,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS]};return{keywords:"",contains:[DTS_ROOT_NODE,DTS_REFERENCE,DTS_KEYWORD,DTS_LABEL,DTS_NODE,DTS_CELL_PROPERTY,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,STRINGS,PREPROCESSOR,{begin:hljs.IDENT_RE+"::",keywords:""}]}});hljs.registerLanguage("step21",function(hljs){var STEP21_IDENT_RE="[A-Z_][A-Z0-9_.]*";var STEP21_KEYWORDS={keyword:"HEADER ENDSEC DATA"};var STEP21_START={className:"meta",begin:"ISO-10303-21;",relevance:10};var STEP21_CLOSE={className:"meta",begin:"END-ISO-10303-21;",relevance:10};return{aliases:["p21","step","stp"],case_insensitive:true,lexemes:STEP21_IDENT_RE,keywords:STEP21_KEYWORDS,contains:[STEP21_START,STEP21_CLOSE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT("/\\*\\*!","\\*/"),hljs.C_NUMBER_MODE,hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"string",begin:"'",end:"'"},{className:"symbol",variants:[{begin:"#",end:"\\d+",illegal:"\\W"}]}]}});hljs.registerLanguage("sml",function(hljs){return{aliases:["ml"],keywords:{keyword:"abstype and andalso as case datatype do else end eqtype "+"exception fn fun functor handle if in include infix infixr "+"let local nonfix of op open orelse raise rec sharing sig "+"signature struct structure then type val with withtype where while",built_in:"array bool char exn int list option order real ref string substring vector unit word",literal:"true false NONE SOME LESS EQUAL GREATER nil"},illegal:/\/\/|>>/,lexemes:"[a-z_]\\w*!?",contains:[{className:"literal",begin:/\[(\|\|)?\]|\(\)/,relevance:0},hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),{className:"symbol",begin:"'[A-Za-z_](?!')[\\w']*"},{className:"type",begin:"`[A-Z][\\w']*"},{className:"type",begin:"\\b[A-Z][\\w']*",relevance:0},{begin:"[a-z_]\\w*'[\\w']*"},hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"number",begin:"\\b(0[xX][a-fA-F0-9_]+[Lln]?|"+"0[oO][0-7_]+[Lln]?|"+"0[bB][01_]+[Lln]?|"+"[0-9][0-9_]*([Lln]|(\\.[0-9_]*)?([eE][-+]?[0-9_]+)?)?)",relevance:0},{begin:/[-=]>/}]}});hljs.registerLanguage("fsharp",function(hljs){var TYPEPARAM={begin:"<",end:">",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/'[a-zA-Z0-9_]+/})]};return{aliases:["fs"],keywords:"abstract and as assert base begin class default delegate do done "+"downcast downto elif else end exception extern false finally for "+"fun function global if in inherit inline interface internal lazy let "+"match member module mutable namespace new null of open or "+"override private public rec return sig static struct then to "+"true try type upcast use val void when while with yield",illegal:/\/\*/,contains:[{className:"keyword",begin:/\b(yield|return|let|do)!/},{className:"string",begin:'@"',end:'"',contains:[{begin:'""'}]},{className:"string",begin:'"""',end:'"""'},hljs.COMMENT("\\(\\*","\\*\\)"),{className:"class",beginKeywords:"type",end:"\\(|=|$",excludeEnd:true,contains:[hljs.UNDERSCORE_TITLE_MODE,TYPEPARAM]},{className:"meta",begin:"\\[<",end:">\\]",relevance:10},{className:"symbol",begin:"\\B('[A-Za-z])\\b",contains:[hljs.BACKSLASH_ESCAPE]},hljs.C_LINE_COMMENT_MODE,hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),hljs.C_NUMBER_MODE]}});hljs.registerLanguage("julia",function(hljs){var KEYWORDS={keyword:"in isa where "+"baremodule begin break catch ccall const continue do else elseif end export false finally for function "+"global if import importall let local macro module quote return true try using while "+"type immutable abstract bitstype typealias ",literal:"true false "+"ARGS C_NULL DevNull ENDIAN_BOM ENV I Inf Inf16 Inf32 Inf64 InsertionSort JULIA_HOME LOAD_PATH MergeSort "+"NaN NaN16 NaN32 NaN64 PROGRAM_FILE QuickSort RoundDown RoundFromZero RoundNearest RoundNearestTiesAway "+"RoundNearestTiesUp RoundToZero RoundUp STDERR STDIN STDOUT VERSION catalan e|0 eu|0 eulergamma golden im "+"nothing pi γ π φ ",built_in:"ANY AbstractArray AbstractChannel AbstractFloat AbstractMatrix AbstractRNG AbstractSerializer AbstractSet "+"AbstractSparseArray AbstractSparseMatrix AbstractSparseVector AbstractString AbstractUnitRange AbstractVecOrMat "+"AbstractVector Any ArgumentError Array AssertionError Associative Base64DecodePipe Base64EncodePipe Bidiagonal "+"BigFloat BigInt BitArray BitMatrix BitVector Bool BoundsError BufferStream CachingPool CapturedException "+"CartesianIndex CartesianRange Cchar Cdouble Cfloat Channel Char Cint Cintmax_t Clong Clonglong ClusterManager "+"Cmd CodeInfo Colon Complex Complex128 Complex32 Complex64 CompositeException Condition ConjArray ConjMatrix "+"ConjVector Cptrdiff_t Cshort Csize_t Cssize_t Cstring Cuchar Cuint Cuintmax_t Culong Culonglong Cushort Cwchar_t "+"Cwstring DataType Date DateFormat DateTime DenseArray DenseMatrix DenseVecOrMat DenseVector Diagonal Dict "+"DimensionMismatch Dims DirectIndexString Display DivideError DomainError EOFError EachLine Enum Enumerate "+"ErrorException Exception ExponentialBackOff Expr Factorization FileMonitor Float16 Float32 Float64 Function "+"Future GlobalRef GotoNode HTML Hermitian IO IOBuffer IOContext IOStream IPAddr IPv4 IPv6 IndexCartesian IndexLinear "+"IndexStyle InexactError InitError Int Int128 Int16 Int32 Int64 Int8 IntSet Integer InterruptException "+"InvalidStateException Irrational KeyError LabelNode LinSpace LineNumberNode LoadError LowerTriangular MIME Matrix "+"MersenneTwister Method MethodError MethodTable Module NTuple NewvarNode NullException Nullable Number ObjectIdDict "+"OrdinalRange OutOfMemoryError OverflowError Pair ParseError PartialQuickSort PermutedDimsArray Pipe "+"PollingFileWatcher ProcessExitedException Ptr QuoteNode RandomDevice Range RangeIndex Rational RawFD "+"ReadOnlyMemoryError Real ReentrantLock Ref Regex RegexMatch RemoteChannel RemoteException RevString RoundingMode "+"RowVector SSAValue SegmentationFault SerializationState Set SharedArray SharedMatrix SharedVector Signed "+"SimpleVector Slot SlotNumber SparseMatrixCSC SparseVector StackFrame StackOverflowError StackTrace StepRange "+"StepRangeLen StridedArray StridedMatrix StridedVecOrMat StridedVector String SubArray SubString SymTridiagonal "+"Symbol Symmetric SystemError TCPSocket Task Text TextDisplay Timer Tridiagonal Tuple Type TypeError TypeMapEntry "+"TypeMapLevel TypeName TypeVar TypedSlot UDPSocket UInt UInt128 UInt16 UInt32 UInt64 UInt8 UndefRefError UndefVarError "+"UnicodeError UniformScaling Union UnionAll UnitRange Unsigned UpperTriangular Val Vararg VecElement VecOrMat Vector "+"VersionNumber Void WeakKeyDict WeakRef WorkerConfig WorkerPool "};var VARIABLE_NAME_RE="[A-Za-z_\\u00A1-\\uFFFF][A-Za-z_0-9\\u00A1-\\uFFFF]*";var DEFAULT={lexemes:VARIABLE_NAME_RE,keywords:KEYWORDS,illegal:/<\//};var NUMBER={className:"number",begin:/(\b0x[\d_]*(\.[\d_]*)?|0x\.\d[\d_]*)p[-+]?\d+|\b0[box][a-fA-F0-9][a-fA-F0-9_]*|(\b\d[\d_]*(\.[\d_]*)?|\.\d[\d_]*)([eEfF][-+]?\d+)?/,relevance:0};var CHAR={className:"string",begin:/'(.|\\[xXuU][a-zA-Z0-9]+)'/};var INTERPOLATION={className:"subst",begin:/\$\(/,end:/\)/,keywords:KEYWORDS};var INTERPOLATED_VARIABLE={className:"variable",begin:"\\$"+VARIABLE_NAME_RE};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,INTERPOLATION,INTERPOLATED_VARIABLE],variants:[{begin:/\w*"""/,end:/"""\w*/,relevance:10},{begin:/\w*"/,end:/"\w*/}]};var COMMAND={className:"string",contains:[hljs.BACKSLASH_ESCAPE,INTERPOLATION,INTERPOLATED_VARIABLE],begin:"`",end:"`"};var MACROCALL={className:"meta",begin:"@"+VARIABLE_NAME_RE};var COMMENT={className:"comment",variants:[{begin:"#=",end:"=#",relevance:10},{begin:"#",end:"$"}]};DEFAULT.contains=[NUMBER,CHAR,STRING,COMMAND,MACROCALL,COMMENT,hljs.HASH_COMMENT_MODE,{className:"keyword",begin:"\\b(((abstract|primitive)\\s+)type|(mutable\\s+)?struct)\\b"},{begin:/<:/}];INTERPOLATION.contains=DEFAULT.contains;return DEFAULT});hljs.registerLanguage("julia-repl",function(hljs){return{contains:[{className:"meta",begin:/^julia>/,relevance:10,starts:{end:/^(?![ ]{6})/,subLanguage:"julia"},aliases:["jldoctest"]}]}});hljs.registerLanguage("fortran",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var F_KEYWORDS={literal:".False. .True.",keyword:"kind do while private call intrinsic where elsewhere "+"type endtype endmodule endselect endinterface end enddo endif if forall endforall only contains default return stop then block endblock "+"public subroutine|10 function program .and. .or. .not. .le. .eq. .ge. .gt. .lt. "+"goto save else use module select case "+"access blank direct exist file fmt form formatted iostat name named nextrec number opened rec recl sequential status unformatted unit "+"continue format pause cycle exit "+"c_null_char c_alert c_backspace c_form_feed flush wait decimal round iomsg "+"synchronous nopass non_overridable pass protected volatile abstract extends import "+"non_intrinsic value deferred generic final enumerator class associate bind enum "+"c_int c_short c_long c_long_long c_signed_char c_size_t c_int8_t c_int16_t c_int32_t c_int64_t c_int_least8_t c_int_least16_t "+"c_int_least32_t c_int_least64_t c_int_fast8_t c_int_fast16_t c_int_fast32_t c_int_fast64_t c_intmax_t C_intptr_t c_float c_double "+"c_long_double c_float_complex c_double_complex c_long_double_complex c_bool c_char c_null_ptr c_null_funptr "+"c_new_line c_carriage_return c_horizontal_tab c_vertical_tab iso_c_binding c_loc c_funloc c_associated  c_f_pointer "+"c_ptr c_funptr iso_fortran_env character_storage_size error_unit file_storage_size input_unit iostat_end iostat_eor "+"numeric_storage_size output_unit c_f_procpointer ieee_arithmetic ieee_support_underflow_control "+"ieee_get_underflow_mode ieee_set_underflow_mode newunit contiguous recursive "+"pad position action delim readwrite eor advance nml interface procedure namelist include sequence elemental pure "+"integer real character complex logical dimension allocatable|10 parameter "+"external implicit|10 none double precision assign intent optional pointer "+"target in out common equivalence data",built_in:"alog alog10 amax0 amax1 amin0 amin1 amod cabs ccos cexp clog csin csqrt dabs dacos dasin datan datan2 dcos dcosh ddim dexp dint "+"dlog dlog10 dmax1 dmin1 dmod dnint dsign dsin dsinh dsqrt dtan dtanh float iabs idim idint idnint ifix isign max0 max1 min0 min1 sngl "+"algama cdabs cdcos cdexp cdlog cdsin cdsqrt cqabs cqcos cqexp cqlog cqsin cqsqrt dcmplx dconjg derf derfc dfloat dgamma dimag dlgama "+"iqint qabs qacos qasin qatan qatan2 qcmplx qconjg qcos qcosh qdim qerf qerfc qexp qgamma qimag qlgama qlog qlog10 qmax1 qmin1 qmod "+"qnint qsign qsin qsinh qsqrt qtan qtanh abs acos aimag aint anint asin atan atan2 char cmplx conjg cos cosh exp ichar index int log "+"log10 max min nint sign sin sinh sqrt tan tanh print write dim lge lgt lle llt mod nullify allocate deallocate "+"adjustl adjustr all allocated any associated bit_size btest ceiling count cshift date_and_time digits dot_product "+"eoshift epsilon exponent floor fraction huge iand ibclr ibits ibset ieor ior ishft ishftc lbound len_trim matmul "+"maxexponent maxloc maxval merge minexponent minloc minval modulo mvbits nearest pack present product "+"radix random_number random_seed range repeat reshape rrspacing scale scan selected_int_kind selected_real_kind "+"set_exponent shape size spacing spread sum system_clock tiny transpose trim ubound unpack verify achar iachar transfer "+"dble entry dprod cpu_time command_argument_count get_command get_command_argument get_environment_variable is_iostat_end "+"ieee_arithmetic ieee_support_underflow_control ieee_get_underflow_mode ieee_set_underflow_mode "+"is_iostat_eor move_alloc new_line selected_char_kind same_type_as extends_type_of"+"acosh asinh atanh bessel_j0 bessel_j1 bessel_jn bessel_y0 bessel_y1 bessel_yn erf erfc erfc_scaled gamma log_gamma hypot norm2 "+"atomic_define atomic_ref execute_command_line leadz trailz storage_size merge_bits "+"bge bgt ble blt dshiftl dshiftr findloc iall iany iparity image_index lcobound ucobound maskl maskr "+"num_images parity popcnt poppar shifta shiftl shiftr this_image"};return{case_insensitive:true,aliases:["f90","f95"],keywords:F_KEYWORDS,illegal:/\/\*/,contains:[hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"string",relevance:0}),{className:"function",beginKeywords:"subroutine function program",illegal:"[${=\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]},hljs.COMMENT("!","$",{relevance:0}),{className:"number",begin:"(?=\\b|\\+|\\-|\\.)(?=\\.\\d|\\d)(?:\\d+)?(?:\\.?\\d*)(?:[de][+-]?\\d+)?\\b\\.?",relevance:0}]}});hljs.registerLanguage("basic",function(hljs){return{case_insensitive:true,illegal:"^.",lexemes:"[a-zA-Z][a-zA-Z0-9_$%!#]*",keywords:{keyword:"ABS ASC AND ATN AUTO|0 BEEP BLOAD|10 BSAVE|10 CALL CALLS CDBL CHAIN CHDIR CHR$|10 CINT CIRCLE "+"CLEAR CLOSE CLS COLOR COM COMMON CONT COS CSNG CSRLIN CVD CVI CVS DATA DATE$ "+"DEFDBL DEFINT DEFSNG DEFSTR DEF|0 SEG USR DELETE DIM DRAW EDIT END ENVIRON ENVIRON$ "+"EOF EQV ERASE ERDEV ERDEV$ ERL ERR ERROR EXP FIELD FILES FIX FOR|0 FRE GET GOSUB|10 GOTO "+"HEX$ IF THEN ELSE|0 INKEY$ INP INPUT INPUT# INPUT$ INSTR IMP INT IOCTL IOCTL$ KEY ON "+"OFF LIST KILL LEFT$ LEN LET LINE LLIST LOAD LOC LOCATE LOF LOG LPRINT USING LSET "+"MERGE MID$ MKDIR MKD$ MKI$ MKS$ MOD NAME NEW NEXT NOISE NOT OCT$ ON OR PEN PLAY STRIG OPEN OPTION "+"BASE OUT PAINT PALETTE PCOPY PEEK PMAP POINT POKE POS PRINT PRINT] PSET PRESET "+"PUT RANDOMIZE READ REM RENUM RESET|0 RESTORE RESUME RETURN|0 RIGHT$ RMDIR RND RSET "+"RUN SAVE SCREEN SGN SHELL SIN SOUND SPACE$ SPC SQR STEP STICK STOP STR$ STRING$ SWAP "+"SYSTEM TAB TAN TIME$ TIMER TROFF TRON TO USR VAL VARPTR VARPTR$ VIEW WAIT WHILE "+"WEND WIDTH WINDOW WRITE XOR"},contains:[hljs.QUOTE_STRING_MODE,hljs.COMMENT("REM","$",{relevance:10}),hljs.COMMENT("'","$",{relevance:0}),{className:"symbol",begin:"^[0-9]+ ",relevance:10},{className:"number",begin:"\\b([0-9]+[0-9edED.]*[#!]?)",relevance:0},{className:"number",begin:"(&[hH][0-9a-fA-F]{1,4})"},{className:"number",begin:"(&[oO][0-7]{1,6})"}]}});hljs.registerLanguage("ada",function(hljs){var INTEGER_RE="\\d(_|\\d)*";var EXPONENT_RE="[eE][-+]?"+INTEGER_RE;var DECIMAL_LITERAL_RE=INTEGER_RE+"(\\."+INTEGER_RE+")?"+"("+EXPONENT_RE+")?";var BASED_INTEGER_RE="\\w+";var BASED_LITERAL_RE=INTEGER_RE+"#"+BASED_INTEGER_RE+"(\\."+BASED_INTEGER_RE+")?"+"#"+"("+EXPONENT_RE+")?";var NUMBER_RE="\\b("+BASED_LITERAL_RE+"|"+DECIMAL_LITERAL_RE+")";var ID_REGEX="[A-Za-z](_?[A-Za-z0-9.])*";var BAD_CHARS="[]{}%#'\"";var COMMENTS=hljs.COMMENT("--","$");var VAR_DECLS={begin:"\\s+:\\s+",end:"\\s*(:=|;|\\)|=>|$)",illegal:BAD_CHARS,contains:[{beginKeywords:"loop for declare others",endsParent:true},{className:"keyword",beginKeywords:"not null constant access function procedure in out aliased exception"},{className:"type",begin:ID_REGEX,endsParent:true,relevance:0}]};return{case_insensitive:true,keywords:{keyword:"abort else new return abs elsif not reverse abstract end "+"accept entry select access exception of separate aliased exit or some "+"all others subtype and for out synchronized array function overriding "+"at tagged generic package task begin goto pragma terminate "+"body private then if procedure type case in protected constant interface "+"is raise use declare range delay limited record when delta loop rem while "+"digits renames with do mod requeue xor",literal:"True False"},contains:[COMMENTS,{className:"string",begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{className:"string",begin:/'.'/},{className:"number",begin:NUMBER_RE,relevance:0},{className:"symbol",begin:"'"+ID_REGEX},{className:"title",begin:"(\\bwith\\s+)?(\\bprivate\\s+)?\\bpackage\\s+(\\bbody\\s+)?",end:"(is|$)",keywords:"package body",excludeBegin:true,excludeEnd:true,illegal:BAD_CHARS},{begin:"(\\b(with|overriding)\\s+)?\\b(function|procedure)\\s+",end:"(\\bis|\\bwith|\\brenames|\\)\\s*;)",keywords:"overriding function procedure with is renames return",returnBegin:true,contains:[COMMENTS,{className:"title",begin:"(\\bwith\\s+)?\\b(function|procedure)\\s+",end:"(\\(|\\s+|$)",excludeBegin:true,excludeEnd:true,illegal:BAD_CHARS},VAR_DECLS,{className:"type",begin:"\\breturn\\s+",end:"(\\s+|;|$)",keywords:"return",excludeBegin:true,excludeEnd:true,endsParent:true,illegal:BAD_CHARS}]},{className:"type",begin:"\\b(sub)?type\\s+",end:"\\s+",keywords:"type",excludeBegin:true,illegal:BAD_CHARS},VAR_DECLS]}});hljs.registerLanguage("puppet",function(hljs){var PUPPET_KEYWORDS={keyword:"and case default else elsif false if in import enherits node or true undef unless main settings $string ",literal:"alias audit before loglevel noop require subscribe tag "+"owner ensure group mode name|0 changes context force incl lens load_path onlyif provider returns root show_diff type_check "+"en_address ip_address realname command environment hour monute month monthday special target weekday "+"creates cwd ogoutput refresh refreshonly tries try_sleep umask backup checksum content ctime force ignore "+"links mtime purge recurse recurselimit replace selinux_ignore_defaults selrange selrole seltype seluser source "+"souirce_permissions sourceselect validate_cmd validate_replacement allowdupe attribute_membership auth_membership forcelocal gid "+"ia_load_module members system host_aliases ip allowed_trunk_vlans description device_url duplex encapsulation etherchannel "+"native_vlan speed principals allow_root auth_class auth_type authenticate_user k_of_n mechanisms rule session_owner shared options "+"device fstype enable hasrestart directory present absent link atboot blockdevice device dump pass remounts poller_tag use "+"message withpath adminfile allow_virtual allowcdrom category configfiles flavor install_options instance package_settings platform "+"responsefile status uninstall_options vendor unless_system_user unless_uid binary control flags hasstatus manifest pattern restart running "+"start stop allowdupe auths expiry gid groups home iterations key_membership keys managehome membership password password_max_age "+"password_min_age profile_membership profiles project purge_ssh_keys role_membership roles salt shell uid baseurl cost descr enabled "+"enablegroups exclude failovermethod gpgcheck gpgkey http_caching include includepkgs keepalive metadata_expire metalink mirrorlist "+"priority protect proxy proxy_password proxy_username repo_gpgcheck s3_enabled skip_if_unavailable sslcacert sslclientcert sslclientkey "+"sslverify mounted",built_in:"architecture augeasversion blockdevices boardmanufacturer boardproductname boardserialnumber cfkey dhcp_servers "+"domain ec2_ ec2_userdata facterversion filesystems ldom fqdn gid hardwareisa hardwaremodel hostname id|0 interfaces "+"ipaddress ipaddress_ ipaddress6 ipaddress6_ iphostnumber is_virtual kernel kernelmajversion kernelrelease kernelversion "+"kernelrelease kernelversion lsbdistcodename lsbdistdescription lsbdistid lsbdistrelease lsbmajdistrelease lsbminordistrelease "+"lsbrelease macaddress macaddress_ macosx_buildversion macosx_productname macosx_productversion macosx_productverson_major "+"macosx_productversion_minor manufacturer memoryfree memorysize netmask metmask_ network_ operatingsystem operatingsystemmajrelease "+"operatingsystemrelease osfamily partitions path physicalprocessorcount processor processorcount productname ps puppetversion "+"rubysitedir rubyversion selinux selinux_config_mode selinux_config_policy selinux_current_mode selinux_current_mode selinux_enforced "+"selinux_policyversion serialnumber sp_ sshdsakey sshecdsakey sshrsakey swapencrypted swapfree swapsize timezone type uniqueid uptime "+"uptime_days uptime_hours uptime_seconds uuid virtual vlans xendomains zfs_version zonenae zones zpool_version"};var COMMENT=hljs.COMMENT("#","$");var IDENT_RE="([A-Za-z_]|::)(\\w|::)*";var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE});var VARIABLE={className:"variable",begin:"\\$"+IDENT_RE};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,VARIABLE],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/}]};return{aliases:["pp"],contains:[COMMENT,VARIABLE,STRING,{beginKeywords:"class",end:"\\{|;",illegal:/=/,contains:[TITLE,COMMENT]},{beginKeywords:"define",end:/\{/,contains:[{className:"section",begin:hljs.IDENT_RE,endsParent:true}]},{begin:hljs.IDENT_RE+"\\s+\\{",returnBegin:true,end:/\S/,contains:[{className:"keyword",begin:hljs.IDENT_RE},{begin:/\{/,end:/\}/,keywords:PUPPET_KEYWORDS,relevance:0,contains:[STRING,COMMENT,{begin:"[a-zA-Z_]+\\s*=>",returnBegin:true,end:"=>",contains:[{className:"attr",begin:hljs.IDENT_RE}]},{className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0},VARIABLE]}],relevance:0}]}});hljs.registerLanguage("vim",function(hljs){return{lexemes:/[!#@\w]+/,keywords:{keyword:"N|0 P|0 X|0 a|0 ab abc abo al am an|0 ar arga argd arge argdo argg argl argu as au aug aun b|0 bN ba bad bd be bel bf bl bm bn bo bp br brea breaka breakd breakl bro bufdo buffers bun bw c|0 cN cNf ca cabc caddb cad caddf cal cat cb cc ccl cd ce cex cf cfir cgetb cgete cg changes chd che checkt cl cla clo cm cmapc cme cn cnew cnf cno cnorea cnoreme co col colo com comc comp con conf cope "+"cp cpf cq cr cs cst cu cuna cunme cw delm deb debugg delc delf dif diffg diffo diffp diffpu diffs diffthis dig di dl dell dj dli do doautoa dp dr ds dsp e|0 ea ec echoe echoh echom echon el elsei em en endfo endf endt endw ene ex exe exi exu f|0 files filet fin fina fini fir fix fo foldc foldd folddoc foldo for fu go gr grepa gu gv ha helpf helpg helpt hi hid his ia iabc if ij il im imapc "+"ime ino inorea inoreme int is isp iu iuna iunme j|0 ju k|0 keepa kee keepj lN lNf l|0 lad laddb laddf la lan lat lb lc lch lcl lcs le lefta let lex lf lfir lgetb lgete lg lgr lgrepa lh ll lla lli lmak lm lmapc lne lnew lnf ln loadk lo loc lockv lol lope lp lpf lr ls lt lu lua luad luaf lv lvimgrepa lw m|0 ma mak map mapc marks mat me menut mes mk mks mksp mkv mkvie mod mz mzf nbc nb nbs new nm nmapc nme nn nnoreme noa no noh norea noreme norm nu nun nunme ol o|0 om omapc ome on ono onoreme opt ou ounme ow p|0 "+"profd prof pro promptr pc ped pe perld po popu pp pre prev ps pt ptN ptf ptj ptl ptn ptp ptr pts pu pw py3 python3 py3d py3f py pyd pyf quita qa rec red redi redr redraws reg res ret retu rew ri rightb rub rubyd rubyf rund ru rv sN san sa sal sav sb sbN sba sbf sbl sbm sbn sbp sbr scrip scripte scs se setf setg setl sf sfir sh sim sig sil sl sla sm smap smapc sme sn sni sno snor snoreme sor "+"so spelld spe spelli spellr spellu spellw sp spr sre st sta startg startr star stopi stj sts sun sunm sunme sus sv sw sy synti sync tN tabN tabc tabdo tabe tabf tabfir tabl tabm tabnew "+"tabn tabo tabp tabr tabs tab ta tags tc tcld tclf te tf th tj tl tm tn to tp tr try ts tu u|0 undoj undol una unh unl unlo unm unme uns up ve verb vert vim vimgrepa vi viu vie vm vmapc vme vne vn vnoreme vs vu vunme windo w|0 wN wa wh wi winc winp wn wp wq wqa ws wu wv x|0 xa xmapc xm xme xn xnoreme xu xunme y|0 z|0 ~ "+"Next Print append abbreviate abclear aboveleft all amenu anoremenu args argadd argdelete argedit argglobal arglocal argument ascii autocmd augroup aunmenu buffer bNext ball badd bdelete behave belowright bfirst blast bmodified bnext botright bprevious brewind break breakadd breakdel breaklist browse bunload "+"bwipeout change cNext cNfile cabbrev cabclear caddbuffer caddexpr caddfile call catch cbuffer cclose center cexpr cfile cfirst cgetbuffer cgetexpr cgetfile chdir checkpath checktime clist clast close cmap cmapclear cmenu cnext cnewer cnfile cnoremap cnoreabbrev cnoremenu copy colder colorscheme command comclear compiler continue confirm copen cprevious cpfile cquit crewind cscope cstag cunmap "+"cunabbrev cunmenu cwindow delete delmarks debug debuggreedy delcommand delfunction diffupdate diffget diffoff diffpatch diffput diffsplit digraphs display deletel djump dlist doautocmd doautoall deletep drop dsearch dsplit edit earlier echo echoerr echohl echomsg else elseif emenu endif endfor "+"endfunction endtry endwhile enew execute exit exusage file filetype find finally finish first fixdel fold foldclose folddoopen folddoclosed foldopen function global goto grep grepadd gui gvim hardcopy help helpfind helpgrep helptags highlight hide history insert iabbrev iabclear ijump ilist imap "+"imapclear imenu inoremap inoreabbrev inoremenu intro isearch isplit iunmap iunabbrev iunmenu join jumps keepalt keepmarks keepjumps lNext lNfile list laddexpr laddbuffer laddfile last language later lbuffer lcd lchdir lclose lcscope left leftabove lexpr lfile lfirst lgetbuffer lgetexpr lgetfile lgrep lgrepadd lhelpgrep llast llist lmake lmap lmapclear lnext lnewer lnfile lnoremap loadkeymap loadview "+"lockmarks lockvar lolder lopen lprevious lpfile lrewind ltag lunmap luado luafile lvimgrep lvimgrepadd lwindow move mark make mapclear match menu menutranslate messages mkexrc mksession mkspell mkvimrc mkview mode mzscheme mzfile nbclose nbkey nbsart next nmap nmapclear nmenu nnoremap "+"nnoremenu noautocmd noremap nohlsearch noreabbrev noremenu normal number nunmap nunmenu oldfiles open omap omapclear omenu only onoremap onoremenu options ounmap ounmenu ownsyntax print profdel profile promptfind promptrepl pclose pedit perl perldo pop popup ppop preserve previous psearch ptag ptNext "+"ptfirst ptjump ptlast ptnext ptprevious ptrewind ptselect put pwd py3do py3file python pydo pyfile quit quitall qall read recover redo redir redraw redrawstatus registers resize retab return rewind right rightbelow ruby rubydo rubyfile rundo runtime rviminfo substitute sNext sandbox sargument sall saveas sbuffer sbNext sball sbfirst sblast sbmodified sbnext sbprevious sbrewind scriptnames scriptencoding "+"scscope set setfiletype setglobal setlocal sfind sfirst shell simalt sign silent sleep slast smagic smapclear smenu snext sniff snomagic snoremap snoremenu sort source spelldump spellgood spellinfo spellrepall spellundo spellwrong split sprevious srewind stop stag startgreplace startreplace "+"startinsert stopinsert stjump stselect sunhide sunmap sunmenu suspend sview swapname syntax syntime syncbind tNext tabNext tabclose tabedit tabfind tabfirst tablast tabmove tabnext tabonly tabprevious tabrewind tag tcl tcldo tclfile tearoff tfirst throw tjump tlast tmenu tnext topleft tprevious "+"trewind tselect tunmenu undo undojoin undolist unabbreviate unhide unlet unlockvar unmap unmenu unsilent update vglobal version verbose vertical vimgrep vimgrepadd visual viusage view vmap vmapclear vmenu vnew "+"vnoremap vnoremenu vsplit vunmap vunmenu write wNext wall while winsize wincmd winpos wnext wprevious wqall wsverb wundo wviminfo xit xall xmapclear xmap xmenu xnoremap xnoremenu xunmap xunmenu yank",built_in:"synIDtrans atan2 range matcharg did_filetype asin feedkeys xor argv "+"complete_check add getwinposx getqflist getwinposy screencol "+"clearmatches empty extend getcmdpos mzeval garbagecollect setreg "+"ceil sqrt diff_hlID inputsecret get getfperm getpid filewritable "+"shiftwidth max sinh isdirectory synID system inputrestore winline "+"atan visualmode inputlist tabpagewinnr round getregtype mapcheck "+"hasmapto histdel argidx findfile sha256 exists toupper getcmdline "+"taglist string getmatches bufnr strftime winwidth bufexists "+"strtrans tabpagebuflist setcmdpos remote_read printf setloclist "+"getpos getline bufwinnr float2nr len getcmdtype diff_filler luaeval "+"resolve libcallnr foldclosedend reverse filter has_key bufname "+"str2float strlen setline getcharmod setbufvar index searchpos "+"shellescape undofile foldclosed setqflist buflisted strchars str2nr "+"virtcol floor remove undotree remote_expr winheight gettabwinvar "+"reltime cursor tabpagenr finddir localtime acos getloclist search "+"tanh matchend rename gettabvar strdisplaywidth type abs py3eval "+"setwinvar tolower wildmenumode log10 spellsuggest bufloaded "+"synconcealed nextnonblank server2client complete settabwinvar "+"executable input wincol setmatches getftype hlID inputsave "+"searchpair or screenrow line settabvar histadd deepcopy strpart "+"remote_peek and eval getftime submatch screenchar winsaveview "+"matchadd mkdir screenattr getfontname libcall reltimestr getfsize "+"winnr invert pow getbufline byte2line soundfold repeat fnameescape "+"tagfiles sin strwidth spellbadword trunc maparg log lispindent "+"hostname setpos globpath remote_foreground getchar synIDattr "+"fnamemodify cscope_connection stridx winbufnr indent min "+"complete_add nr2char searchpairpos inputdialog values matchlist "+"items hlexists strridx browsedir expand fmod pathshorten line2byte "+"argc count getwinvar glob foldtextresult getreg foreground cosh "+"matchdelete has char2nr simplify histget searchdecl iconv "+"winrestcmd pumvisible writefile foldlevel haslocaldir keys cos "+"matchstr foldtext histnr tan tempname getcwd byteidx getbufvar "+"islocked escape eventhandler remote_send serverlist winrestview "+"synstack pyeval prevnonblank readfile cindent filereadable changenr "+"exp"},illegal:/;/,contains:[hljs.NUMBER_MODE,{className:"string",begin:"'",end:"'",illegal:"\\n"},{className:"string",begin:/"(\\"|\n\\|[^"\n])*"/},hljs.COMMENT('"',"$"),{className:"variable",begin:/[bwtglsav]:[\w\d_]*/},{className:"function",beginKeywords:"function function!",end:"$",relevance:0,contains:[hljs.TITLE_MODE,{className:"params",begin:"\\(",end:"\\)"}]},{className:"symbol",begin:/<[\w-]+>/}]}});hljs.registerLanguage("dos",function(hljs){var COMMENT=hljs.COMMENT(/^\s*@?rem\b/,/$/,{relevance:10});var LABEL={className:"symbol",begin:"^\\s*[A-Za-z._?][A-Za-z0-9_$#@~.?]*(:|\\s+label)",relevance:0};return{aliases:["bat","cmd"],case_insensitive:true,illegal:/\/\*/,keywords:{keyword:"if else goto for in do call exit not exist errorlevel defined "+"equ neq lss leq gtr geq",built_in:"prn nul lpt3 lpt2 lpt1 con com4 com3 com2 com1 aux "+"shift cd dir echo setlocal endlocal set pause copy "+"append assoc at attrib break cacls cd chcp chdir chkdsk chkntfs cls cmd color "+"comp compact convert date dir diskcomp diskcopy doskey erase fs "+"find findstr format ftype graftabl help keyb label md mkdir mode more move path "+"pause print popd pushd promt rd recover rem rename replace restore rmdir shift"+"sort start subst time title tree type ver verify vol "+"ping net ipconfig taskkill xcopy ren del"},contains:[{className:"variable",begin:/%%[^ ]|%[^ ]+?%|![^ ]+?!/},{className:"function",begin:LABEL.begin,end:"goto:eof",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),COMMENT]},{className:"number",begin:"\\b\\d+",relevance:0},COMMENT]}});hljs.registerLanguage("ceylon",function(hljs){var KEYWORDS="assembly module package import alias class interface object given value "+"assign void function new of extends satisfies abstracts in out return "+"break continue throw assert dynamic if else switch case for while try "+"catch finally then let this outer super is exists nonempty";var DECLARATION_MODIFIERS="shared abstract formal default actual variable late native deprecated"+"final sealed annotation suppressWarnings small";var DOCUMENTATION="doc by license see throws tagged";var SUBST={className:"subst",excludeBegin:true,excludeEnd:true,begin:/``/,end:/``/,keywords:KEYWORDS,relevance:10};var EXPRESSIONS=[{className:"string",begin:'"""',end:'"""',relevance:10},{className:"string",begin:'"',end:'"',contains:[SUBST]},{className:"string",begin:"'",end:"'"},{className:"number",begin:"#[0-9a-fA-F_]+|\\$[01_]+|[0-9_]+(?:\\.[0-9_](?:[eE][+-]?\\d+)?)?[kMGTPmunpf]?",relevance:0}];SUBST.contains=EXPRESSIONS;return{keywords:{keyword:KEYWORDS+" "+DECLARATION_MODIFIERS,meta:DOCUMENTATION},illegal:"\\$[^01]|#[^0-9a-fA-F]",contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:["self"]}),{className:"meta",begin:'@[a-z]\\w*(?:\\:"[^"]*")?'}].concat(EXPRESSIONS)}});hljs.registerLanguage("nsis",function(hljs){var CONSTANTS={className:"variable",begin:/\$(ADMINTOOLS|APPDATA|CDBURN_AREA|CMDLINE|COMMONFILES32|COMMONFILES64|COMMONFILES|COOKIES|DESKTOP|DOCUMENTS|EXEDIR|EXEFILE|EXEPATH|FAVORITES|FONTS|HISTORY|HWNDPARENT|INSTDIR|INTERNET_CACHE|LANGUAGE|LOCALAPPDATA|MUSIC|NETHOOD|OUTDIR|PICTURES|PLUGINSDIR|PRINTHOOD|PROFILE|PROGRAMFILES32|PROGRAMFILES64|PROGRAMFILES|QUICKLAUNCH|RECENT|RESOURCES_LOCALIZED|RESOURCES|SENDTO|SMPROGRAMS|SMSTARTUP|STARTMENU|SYSDIR|TEMP|TEMPLATES|VIDEOS|WINDIR)/};var DEFINES={className:"variable",begin:/\$+{[\w\.:-]+}/};var VARIABLES={className:"variable",begin:/\$+\w+/,illegal:/\(\){}/};var LANGUAGES={className:"variable",begin:/\$+\([\w\^\.:-]+\)/};var PARAMETERS={className:"params",begin:"(ARCHIVE|FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_NORMAL|FILE_ATTRIBUTE_OFFLINE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_TEMPORARY|HKCR|HKCU|HKDD|HKEY_CLASSES_ROOT|HKEY_CURRENT_CONFIG|HKEY_CURRENT_USER|HKEY_DYN_DATA|HKEY_LOCAL_MACHINE|HKEY_PERFORMANCE_DATA|HKEY_USERS|HKLM|HKPD|HKU|IDABORT|IDCANCEL|IDIGNORE|IDNO|IDOK|IDRETRY|IDYES|MB_ABORTRETRYIGNORE|MB_DEFBUTTON1|MB_DEFBUTTON2|MB_DEFBUTTON3|MB_DEFBUTTON4|MB_ICONEXCLAMATION|MB_ICONINFORMATION|MB_ICONQUESTION|MB_ICONSTOP|MB_OK|MB_OKCANCEL|MB_RETRYCANCEL|MB_RIGHT|MB_RTLREADING|MB_SETFOREGROUND|MB_TOPMOST|MB_USERICON|MB_YESNO|NORMAL|OFFLINE|READONLY|SHCTX|SHELL_CONTEXT|SYSTEM|TEMPORARY)"};var COMPILER={className:"keyword",begin:/\!(addincludedir|addplugindir|appendfile|cd|define|delfile|echo|else|endif|error|execute|finalize|getdllversion|gettlbversion|if|ifdef|ifmacrodef|ifmacrondef|ifndef|include|insertmacro|macro|macroend|makensis|packhdr|searchparse|searchreplace|system|tempfile|undef|verbose|warning)/};var METACHARS={className:"meta",begin:/\$(\\[nrt]|\$)/};var PLUGINS={className:"class",begin:/\w+\:\:\w+/};var STRING={className:"string",variants:[{begin:'"',end:'"'},{begin:"'",end:"'"},{begin:"`",end:"`"}],illegal:/\n/,contains:[METACHARS,CONSTANTS,DEFINES,VARIABLES,LANGUAGES]};return{case_insensitive:false,keywords:{keyword:"Abort AddBrandingImage AddSize AllowRootDirInstall AllowSkipFiles AutoCloseWindow BGFont BGGradient BrandingText BringToFront Call CallInstDLL Caption ChangeUI CheckBitmap ClearErrors CompletedText ComponentText CopyFiles CRCCheck CreateDirectory CreateFont CreateShortCut Delete DeleteINISec DeleteINIStr DeleteRegKey DeleteRegValue DetailPrint DetailsButtonText DirText DirVar DirVerify EnableWindow EnumRegKey EnumRegValue Exch Exec ExecShell ExecShellWait ExecWait ExpandEnvStrings File FileBufSize FileClose FileErrorText FileOpen FileRead FileReadByte FileReadUTF16LE FileReadWord FileSeek FileWrite FileWriteByte FileWriteUTF16LE FileWriteWord FindClose FindFirst FindNext FindWindow FlushINI FunctionEnd GetCurInstType GetCurrentAddress GetDlgItem GetDLLVersion GetDLLVersionLocal GetErrorLevel GetFileTime GetFileTimeLocal GetFullPathName GetFunctionAddress GetInstDirError GetLabelAddress GetTempFileName Goto HideWindow Icon IfAbort IfErrors IfFileExists IfRebootFlag IfSilent InitPluginsDir InstallButtonText InstallColors InstallDir InstallDirRegKey InstProgressFlags InstType InstTypeGetText InstTypeSetText Int64Cmp Int64CmpU Int64Fmt IntCmp IntCmpU IntFmt IntOp IntPtrCmp IntPtrCmpU IntPtrOp IsWindow LangString LicenseBkColor LicenseData LicenseForceSelection LicenseLangString LicenseText LoadLanguageFile LockWindow LogSet LogText ManifestDPIAware ManifestSupportedOS MessageBox MiscButtonText Name Nop OutFile Page PageCallbacks PageExEnd Pop Push Quit ReadEnvStr ReadINIStr ReadRegDWORD ReadRegStr Reboot RegDLL Rename RequestExecutionLevel ReserveFile Return RMDir SearchPath SectionEnd SectionGetFlags SectionGetInstTypes SectionGetSize SectionGetText SectionGroupEnd SectionIn SectionSetFlags SectionSetInstTypes SectionSetSize SectionSetText SendMessage SetAutoClose SetBrandingImage SetCompress SetCompressor SetCompressorDictSize SetCtlColors SetCurInstType SetDatablockOptimize SetDateSave SetDetailsPrint SetDetailsView SetErrorLevel SetErrors SetFileAttributes SetFont SetOutPath SetOverwrite SetRebootFlag SetRegView SetShellVarContext SetSilent ShowInstDetails ShowUninstDetails ShowWindow SilentInstall SilentUnInstall Sleep SpaceTexts StrCmp StrCmpS StrCpy StrLen SubCaption Unicode UninstallButtonText UninstallCaption UninstallIcon UninstallSubCaption UninstallText UninstPage UnRegDLL Var VIAddVersionKey VIFileVersion VIProductVersion WindowIcon WriteINIStr WriteRegBin WriteRegDWORD WriteRegExpandStr WriteRegMultiStr WriteRegNone WriteRegStr WriteUninstaller XPStyle",literal:"admin all auto both bottom bzip2 colored components current custom directory false force hide highest ifdiff ifnewer instfiles lastused leave left license listonly lzma nevershow none normal notset off on open print right show silent silentlog smooth textonly top true try un.components un.custom un.directory un.instfiles un.license uninstConfirm user Win10 Win7 Win8 WinVista zlib"},contains:[hljs.HASH_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(";","$",{relevance:0}),{className:"function",beginKeywords:"Function PageEx Section SectionGroup",end:"$"},STRING,COMPILER,DEFINES,VARIABLES,LANGUAGES,PARAMETERS,PLUGINS,hljs.NUMBER_MODE]}});hljs.registerLanguage("go",function(hljs){var GO_KEYWORDS={keyword:"break default func interface select case map struct chan else goto package switch "+"const fallthrough if range type continue for import return var go defer "+"bool byte complex64 complex128 float32 float64 int8 int16 int32 int64 string uint8 "+"uint16 uint32 uint64 int uint uintptr rune",literal:"true false iota nil",built_in:"append cap close complex copy imag len make new panic print println real recover delete"};return{aliases:["golang"],keywords:GO_KEYWORDS,illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",variants:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{begin:"`",end:"`"}]},{className:"number",variants:[{begin:hljs.C_NUMBER_RE+"[i]",relevance:1},hljs.C_NUMBER_MODE]},{begin:/:=/},{className:"function",beginKeywords:"func",end:"\\s*(\\{|$)",excludeEnd:true,contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:GO_KEYWORDS,illegal:/["']/}]}]}});hljs.registerLanguage("nix",function(hljs){var NIX_KEYWORDS={keyword:"rec with let in inherit assert if else then",literal:"true false or and null",built_in:"import abort baseNameOf dirOf isNull builtins map removeAttrs throw "+"toString derivation"};var ANTIQUOTE={className:"subst",begin:/\$\{/,end:/}/,keywords:NIX_KEYWORDS};var ATTRS={begin:/[a-zA-Z0-9-_]+(\s*=)/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/\S+/}]};var STRING={className:"string",contains:[ANTIQUOTE],variants:[{begin:"''",end:"''"},{begin:'"',end:'"'}]};var EXPRESSIONS=[hljs.NUMBER_MODE,hljs.HASH_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING,ATTRS];ANTIQUOTE.contains=EXPRESSIONS;return{aliases:["nixos"],keywords:NIX_KEYWORDS,contains:EXPRESSIONS}});hljs.registerLanguage("mel",function(hljs){return{keywords:"int float string vector matrix if else switch case default while do for in break "+"continue global proc return about abs addAttr addAttributeEditorNodeHelp addDynamic "+"addNewShelfTab addPP addPanelCategory addPrefixToName advanceToNextDrivenKey "+"affectedNet affects aimConstraint air alias aliasAttr align alignCtx alignCurve "+"alignSurface allViewFit ambientLight angle angleBetween animCone animCurveEditor "+"animDisplay animView annotate appendStringArray applicationName applyAttrPreset "+"applyTake arcLenDimContext arcLengthDimension arclen arrayMapper art3dPaintCtx "+"artAttrCtx artAttrPaintVertexCtx artAttrSkinPaintCtx artAttrTool artBuildPaintMenu "+"artFluidAttrCtx artPuttyCtx artSelectCtx artSetPaintCtx artUserPaintCtx assignCommand "+"assignInputDevice assignViewportFactories attachCurve attachDeviceAttr attachSurface "+"attrColorSliderGrp attrCompatibility attrControlGrp attrEnumOptionMenu "+"attrEnumOptionMenuGrp attrFieldGrp attrFieldSliderGrp attrNavigationControlGrp "+"attrPresetEditWin attributeExists attributeInfo attributeMenu attributeQuery "+"autoKeyframe autoPlace bakeClip bakeFluidShading bakePartialHistory bakeResults "+"bakeSimulation basename basenameEx batchRender bessel bevel bevelPlus binMembership "+"bindSkin blend2 blendShape blendShapeEditor blendShapePanel blendTwoAttr blindDataType "+"boneLattice boundary boxDollyCtx boxZoomCtx bufferCurve buildBookmarkMenu "+"buildKeyframeMenu button buttonManip CBG cacheFile cacheFileCombine cacheFileMerge "+"cacheFileTrack camera cameraView canCreateManip canvas capitalizeString catch "+"catchQuiet ceil changeSubdivComponentDisplayLevel changeSubdivRegion channelBox "+"character characterMap characterOutlineEditor characterize chdir checkBox checkBoxGrp "+"checkDefaultRenderGlobals choice circle circularFillet clamp clear clearCache clip "+"clipEditor clipEditorCurrentTimeCtx clipSchedule clipSchedulerOutliner clipTrimBefore "+"closeCurve closeSurface cluster cmdFileOutput cmdScrollFieldExecuter "+"cmdScrollFieldReporter cmdShell coarsenSubdivSelectionList collision color "+"colorAtPoint colorEditor colorIndex colorIndexSliderGrp colorSliderButtonGrp "+"colorSliderGrp columnLayout commandEcho commandLine commandPort compactHairSystem "+"componentEditor compositingInterop computePolysetVolume condition cone confirmDialog "+"connectAttr connectControl connectDynamic connectJoint connectionInfo constrain "+"constrainValue constructionHistory container containsMultibyte contextInfo control "+"convertFromOldLayers convertIffToPsd convertLightmap convertSolidTx convertTessellation "+"convertUnit copyArray copyFlexor copyKey copySkinWeights cos cpButton cpCache "+"cpClothSet cpCollision cpConstraint cpConvClothToMesh cpForces cpGetSolverAttr cpPanel "+"cpProperty cpRigidCollisionFilter cpSeam cpSetEdit cpSetSolverAttr cpSolver "+"cpSolverTypes cpTool cpUpdateClothUVs createDisplayLayer createDrawCtx createEditor "+"createLayeredPsdFile createMotionField createNewShelf createNode createRenderLayer "+"createSubdivRegion cross crossProduct ctxAbort ctxCompletion ctxEditMode ctxTraverse "+"currentCtx currentTime currentTimeCtx currentUnit curve curveAddPtCtx "+"curveCVCtx curveEPCtx curveEditorCtx curveIntersect curveMoveEPCtx curveOnSurface "+"curveSketchCtx cutKey cycleCheck cylinder dagPose date defaultLightListCheckBox "+"defaultNavigation defineDataServer defineVirtualDevice deformer deg_to_rad delete "+"deleteAttr deleteShadingGroupsAndMaterials deleteShelfTab deleteUI deleteUnusedBrushes "+"delrandstr detachCurve detachDeviceAttr detachSurface deviceEditor devicePanel dgInfo "+"dgdirty dgeval dgtimer dimWhen directKeyCtx directionalLight dirmap dirname disable "+"disconnectAttr disconnectJoint diskCache displacementToPoly displayAffected "+"displayColor displayCull displayLevelOfDetail displayPref displayRGBColor "+"displaySmoothness displayStats displayString displaySurface distanceDimContext "+"distanceDimension doBlur dolly dollyCtx dopeSheetEditor dot dotProduct "+"doubleProfileBirailSurface drag dragAttrContext draggerContext dropoffLocator "+"duplicate duplicateCurve duplicateSurface dynCache dynControl dynExport dynExpression "+"dynGlobals dynPaintEditor dynParticleCtx dynPref dynRelEdPanel dynRelEditor "+"dynamicLoad editAttrLimits editDisplayLayerGlobals editDisplayLayerMembers "+"editRenderLayerAdjustment editRenderLayerGlobals editRenderLayerMembers editor "+"editorTemplate effector emit emitter enableDevice encodeString endString endsWith env "+"equivalent equivalentTol erf error eval evalDeferred evalEcho event "+"exactWorldBoundingBox exclusiveLightCheckBox exec executeForEachObject exists exp "+"expression expressionEditorListen extendCurve extendSurface extrude fcheck fclose feof "+"fflush fgetline fgetword file fileBrowserDialog fileDialog fileExtension fileInfo "+"filetest filletCurve filter filterCurve filterExpand filterStudioImport "+"findAllIntersections findAnimCurves findKeyframe findMenuItem findRelatedSkinCluster "+"finder firstParentOf fitBspline flexor floatEq floatField floatFieldGrp floatScrollBar "+"floatSlider floatSlider2 floatSliderButtonGrp floatSliderGrp floor flow fluidCacheInfo "+"fluidEmitter fluidVoxelInfo flushUndo fmod fontDialog fopen formLayout format fprint "+"frameLayout fread freeFormFillet frewind fromNativePath fwrite gamma gauss "+"geometryConstraint getApplicationVersionAsFloat getAttr getClassification "+"getDefaultBrush getFileList getFluidAttr getInputDeviceRange getMayaPanelTypes "+"getModifiers getPanel getParticleAttr getPluginResource getenv getpid glRender "+"glRenderEditor globalStitch gmatch goal gotoBindPose grabColor gradientControl "+"gradientControlNoAttr graphDollyCtx graphSelectContext graphTrackCtx gravity grid "+"gridLayout group groupObjectsByName HfAddAttractorToAS HfAssignAS HfBuildEqualMap "+"HfBuildFurFiles HfBuildFurImages HfCancelAFR HfConnectASToHF HfCreateAttractor "+"HfDeleteAS HfEditAS HfPerformCreateAS HfRemoveAttractorFromAS HfSelectAttached "+"HfSelectAttractors HfUnAssignAS hardenPointCurve hardware hardwareRenderPanel "+"headsUpDisplay headsUpMessage help helpLine hermite hide hilite hitTest hotBox hotkey "+"hotkeyCheck hsv_to_rgb hudButton hudSlider hudSliderButton hwReflectionMap hwRender "+"hwRenderLoad hyperGraph hyperPanel hyperShade hypot iconTextButton iconTextCheckBox "+"iconTextRadioButton iconTextRadioCollection iconTextScrollList iconTextStaticLabel "+"ikHandle ikHandleCtx ikHandleDisplayScale ikSolver ikSplineHandleCtx ikSystem "+"ikSystemInfo ikfkDisplayMethod illustratorCurves image imfPlugins inheritTransform "+"insertJoint insertJointCtx insertKeyCtx insertKnotCurve insertKnotSurface instance "+"instanceable instancer intField intFieldGrp intScrollBar intSlider intSliderGrp "+"interToUI internalVar intersect iprEngine isAnimCurve isConnected isDirty isParentOf "+"isSameObject isTrue isValidObjectName isValidString isValidUiName isolateSelect "+"itemFilter itemFilterAttr itemFilterRender itemFilterType joint jointCluster jointCtx "+"jointDisplayScale jointLattice keyTangent keyframe keyframeOutliner "+"keyframeRegionCurrentTimeCtx keyframeRegionDirectKeyCtx keyframeRegionDollyCtx "+"keyframeRegionInsertKeyCtx keyframeRegionMoveKeyCtx keyframeRegionScaleKeyCtx "+"keyframeRegionSelectKeyCtx keyframeRegionSetKeyCtx keyframeRegionTrackCtx "+"keyframeStats lassoContext lattice latticeDeformKeyCtx launch launchImageEditor "+"layerButton layeredShaderPort layeredTexturePort layout layoutDialog lightList "+"lightListEditor lightListPanel lightlink lineIntersection linearPrecision linstep "+"listAnimatable listAttr listCameras listConnections listDeviceAttachments listHistory "+"listInputDeviceAxes listInputDeviceButtons listInputDevices listMenuAnnotation "+"listNodeTypes listPanelCategories listRelatives listSets listTransforms "+"listUnselected listerEditor loadFluid loadNewShelf loadPlugin "+"loadPluginLanguageResources loadPrefObjects localizedPanelLabel lockNode loft log "+"longNameOf lookThru ls lsThroughFilter lsType lsUI Mayatomr mag makeIdentity makeLive "+"makePaintable makeRoll makeSingleSurface makeTubeOn makebot manipMoveContext "+"manipMoveLimitsCtx manipOptions manipRotateContext manipRotateLimitsCtx "+"manipScaleContext manipScaleLimitsCtx marker match max memory menu menuBarLayout "+"menuEditor menuItem menuItemToShelf menuSet menuSetPref messageLine min minimizeApp "+"mirrorJoint modelCurrentTimeCtx modelEditor modelPanel mouse movIn movOut move "+"moveIKtoFK moveKeyCtx moveVertexAlongDirection multiProfileBirailSurface mute "+"nParticle nameCommand nameField namespace namespaceInfo newPanelItems newton nodeCast "+"nodeIconButton nodeOutliner nodePreset nodeType noise nonLinear normalConstraint "+"normalize nurbsBoolean nurbsCopyUVSet nurbsCube nurbsEditUV nurbsPlane nurbsSelect "+"nurbsSquare nurbsToPoly nurbsToPolygonsPref nurbsToSubdiv nurbsToSubdivPref "+"nurbsUVSet nurbsViewDirectionVector objExists objectCenter objectLayer objectType "+"objectTypeUI obsoleteProc oceanNurbsPreviewPlane offsetCurve offsetCurveOnSurface "+"offsetSurface openGLExtension openMayaPref optionMenu optionMenuGrp optionVar orbit "+"orbitCtx orientConstraint outlinerEditor outlinerPanel overrideModifier "+"paintEffectsDisplay pairBlend palettePort paneLayout panel panelConfiguration "+"panelHistory paramDimContext paramDimension paramLocator parent parentConstraint "+"particle particleExists particleInstancer particleRenderInfo partition pasteKey "+"pathAnimation pause pclose percent performanceOptions pfxstrokes pickWalk picture "+"pixelMove planarSrf plane play playbackOptions playblast plugAttr plugNode pluginInfo "+"pluginResourceUtil pointConstraint pointCurveConstraint pointLight pointMatrixMult "+"pointOnCurve pointOnSurface pointPosition poleVectorConstraint polyAppend "+"polyAppendFacetCtx polyAppendVertex polyAutoProjection polyAverageNormal "+"polyAverageVertex polyBevel polyBlendColor polyBlindData polyBoolOp polyBridgeEdge "+"polyCacheMonitor polyCheck polyChipOff polyClipboard polyCloseBorder polyCollapseEdge "+"polyCollapseFacet polyColorBlindData polyColorDel polyColorPerVertex polyColorSet "+"polyCompare polyCone polyCopyUV polyCrease polyCreaseCtx polyCreateFacet "+"polyCreateFacetCtx polyCube polyCut polyCutCtx polyCylinder polyCylindricalProjection "+"polyDelEdge polyDelFacet polyDelVertex polyDuplicateAndConnect polyDuplicateEdge "+"polyEditUV polyEditUVShell polyEvaluate polyExtrudeEdge polyExtrudeFacet "+"polyExtrudeVertex polyFlipEdge polyFlipUV polyForceUV polyGeoSampler polyHelix "+"polyInfo polyInstallAction polyLayoutUV polyListComponentConversion polyMapCut "+"polyMapDel polyMapSew polyMapSewMove polyMergeEdge polyMergeEdgeCtx polyMergeFacet "+"polyMergeFacetCtx polyMergeUV polyMergeVertex polyMirrorFace polyMoveEdge "+"polyMoveFacet polyMoveFacetUV polyMoveUV polyMoveVertex polyNormal polyNormalPerVertex "+"polyNormalizeUV polyOptUvs polyOptions polyOutput polyPipe polyPlanarProjection "+"polyPlane polyPlatonicSolid polyPoke polyPrimitive polyPrism polyProjection "+"polyPyramid polyQuad polyQueryBlindData polyReduce polySelect polySelectConstraint "+"polySelectConstraintMonitor polySelectCtx polySelectEditCtx polySeparate "+"polySetToFaceNormal polySewEdge polyShortestPathCtx polySmooth polySoftEdge "+"polySphere polySphericalProjection polySplit polySplitCtx polySplitEdge polySplitRing "+"polySplitVertex polyStraightenUVBorder polySubdivideEdge polySubdivideFacet "+"polyToSubdiv polyTorus polyTransfer polyTriangulate polyUVSet polyUnite polyWedgeFace "+"popen popupMenu pose pow preloadRefEd print progressBar progressWindow projFileViewer "+"projectCurve projectTangent projectionContext projectionManip promptDialog propModCtx "+"propMove psdChannelOutliner psdEditTextureFile psdExport psdTextureFile putenv pwd "+"python querySubdiv quit rad_to_deg radial radioButton radioButtonGrp radioCollection "+"radioMenuItemCollection rampColorPort rand randomizeFollicles randstate rangeControl "+"readTake rebuildCurve rebuildSurface recordAttr recordDevice redo reference "+"referenceEdit referenceQuery refineSubdivSelectionList refresh refreshAE "+"registerPluginResource rehash reloadImage removeJoint removeMultiInstance "+"removePanelCategory rename renameAttr renameSelectionList renameUI render "+"renderGlobalsNode renderInfo renderLayerButton renderLayerParent "+"renderLayerPostProcess renderLayerUnparent renderManip renderPartition "+"renderQualityNode renderSettings renderThumbnailUpdate renderWindowEditor "+"renderWindowSelectContext renderer reorder reorderDeformers requires reroot "+"resampleFluid resetAE resetPfxToPolyCamera resetTool resolutionNode retarget "+"reverseCurve reverseSurface revolve rgb_to_hsv rigidBody rigidSolver roll rollCtx "+"rootOf rot rotate rotationInterpolation roundConstantRadius rowColumnLayout rowLayout "+"runTimeCommand runup sampleImage saveAllShelves saveAttrPreset saveFluid saveImage "+"saveInitialState saveMenu savePrefObjects savePrefs saveShelf saveToolSettings scale "+"scaleBrushBrightness scaleComponents scaleConstraint scaleKey scaleKeyCtx sceneEditor "+"sceneUIReplacement scmh scriptCtx scriptEditorInfo scriptJob scriptNode scriptTable "+"scriptToShelf scriptedPanel scriptedPanelType scrollField scrollLayout sculpt "+"searchPathArray seed selLoadSettings select selectContext selectCurveCV selectKey "+"selectKeyCtx selectKeyframeRegionCtx selectMode selectPref selectPriority selectType "+"selectedNodes selectionConnection separator setAttr setAttrEnumResource "+"setAttrMapping setAttrNiceNameResource setConstraintRestPosition "+"setDefaultShadingGroup setDrivenKeyframe setDynamic setEditCtx setEditor setFluidAttr "+"setFocus setInfinity setInputDeviceMapping setKeyCtx setKeyPath setKeyframe "+"setKeyframeBlendshapeTargetWts setMenuMode setNodeNiceNameResource setNodeTypeFlag "+"setParent setParticleAttr setPfxToPolyCamera setPluginResource setProject "+"setStampDensity setStartupMessage setState setToolTo setUITemplate setXformManip sets "+"shadingConnection shadingGeometryRelCtx shadingLightRelCtx shadingNetworkCompare "+"shadingNode shapeCompare shelfButton shelfLayout shelfTabLayout shellField "+"shortNameOf showHelp showHidden showManipCtx showSelectionInTitle "+"showShadingGroupAttrEditor showWindow sign simplify sin singleProfileBirailSurface "+"size sizeBytes skinCluster skinPercent smoothCurve smoothTangentSurface smoothstep "+"snap2to2 snapKey snapMode snapTogetherCtx snapshot soft softMod softModCtx sort sound "+"soundControl source spaceLocator sphere sphrand spotLight spotLightPreviewPort "+"spreadSheetEditor spring sqrt squareSurface srtContext stackTrace startString "+"startsWith stitchAndExplodeShell stitchSurface stitchSurfacePoints strcmp "+"stringArrayCatenate stringArrayContains stringArrayCount stringArrayInsertAtIndex "+"stringArrayIntersector stringArrayRemove stringArrayRemoveAtIndex "+"stringArrayRemoveDuplicates stringArrayRemoveExact stringArrayToString "+"stringToStringArray strip stripPrefixFromName stroke subdAutoProjection "+"subdCleanTopology subdCollapse subdDuplicateAndConnect subdEditUV "+"subdListComponentConversion subdMapCut subdMapSewMove subdMatchTopology subdMirror "+"subdToBlind subdToPoly subdTransferUVsToCache subdiv subdivCrease "+"subdivDisplaySmoothness substitute substituteAllString substituteGeometry substring "+"surface surfaceSampler surfaceShaderList swatchDisplayPort switchTable symbolButton "+"symbolCheckBox sysFile system tabLayout tan tangentConstraint texLatticeDeformContext "+"texManipContext texMoveContext texMoveUVShellContext texRotateContext texScaleContext "+"texSelectContext texSelectShortestPathCtx texSmudgeUVContext texWinToolCtx text "+"textCurves textField textFieldButtonGrp textFieldGrp textManip textScrollList "+"textToShelf textureDisplacePlane textureHairColor texturePlacementContext "+"textureWindow threadCount threePointArcCtx timeControl timePort timerX toNativePath "+"toggle toggleAxis toggleWindowVisibility tokenize tokenizeList tolerance tolower "+"toolButton toolCollection toolDropped toolHasOptions toolPropertyWindow torus toupper "+"trace track trackCtx transferAttributes transformCompare transformLimits translator "+"trim trunc truncateFluidCache truncateHairCache tumble tumbleCtx turbulence "+"twoPointArcCtx uiRes uiTemplate unassignInputDevice undo undoInfo ungroup uniform unit "+"unloadPlugin untangleUV untitledFileName untrim upAxis updateAE userCtx uvLink "+"uvSnapshot validateShelfName vectorize view2dToolCtx viewCamera viewClipPlane "+"viewFit viewHeadOn viewLookAt viewManip viewPlace viewSet visor volumeAxis vortex "+"waitCursor warning webBrowser webBrowserPrefs whatIs window windowPref wire "+"wireContext workspace wrinkle wrinkleContext writeTake xbmLangPathList xform",illegal:"</",contains:[hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE]},{begin:"[\\$\\%\\@](\\^\\w\\b|#\\w+|[^\\s\\w{]|{\\w+}|\\w+)"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("autoit",function(hljs){var KEYWORDS="ByRef Case Const ContinueCase ContinueLoop "+"Default Dim Do Else ElseIf EndFunc EndIf EndSelect "+"EndSwitch EndWith Enum Exit ExitLoop For Func "+"Global If In Local Next ReDim Return Select Static "+"Step Switch Then To Until Volatile WEnd While With",LITERAL="True False And Null Not Or",BUILT_IN="Abs ACos AdlibRegister AdlibUnRegister Asc AscW ASin Assign ATan AutoItSetOption AutoItWinGetTitle AutoItWinSetTitle Beep Binary BinaryLen BinaryMid BinaryToString BitAND BitNOT BitOR BitRotate BitShift BitXOR BlockInput Break Call CDTray Ceiling Chr ChrW ClipGet ClipPut ConsoleRead ConsoleWrite ConsoleWriteError ControlClick ControlCommand ControlDisable ControlEnable ControlFocus ControlGetFocus ControlGetHandle ControlGetPos ControlGetText ControlHide ControlListView ControlMove ControlSend ControlSetText ControlShow ControlTreeView Cos Dec DirCopy DirCreate DirGetSize DirMove DirRemove DllCall DllCallAddress DllCallbackFree DllCallbackGetPtr DllCallbackRegister DllClose DllOpen DllStructCreate DllStructGetData DllStructGetPtr DllStructGetSize DllStructSetData DriveGetDrive DriveGetFileSystem DriveGetLabel DriveGetSerial DriveGetType DriveMapAdd DriveMapDel DriveMapGet DriveSetLabel DriveSpaceFree DriveSpaceTotal DriveStatus EnvGet EnvSet EnvUpdate Eval Execute Exp FileChangeDir FileClose FileCopy FileCreateNTFSLink FileCreateShortcut FileDelete FileExists FileFindFirstFile FileFindNextFile FileFlush FileGetAttrib FileGetEncoding FileGetLongName FileGetPos FileGetShortcut FileGetShortName FileGetSize FileGetTime FileGetVersion FileInstall FileMove FileOpen FileOpenDialog FileRead FileReadLine FileReadToArray FileRecycle FileRecycleEmpty FileSaveDialog FileSelectFolder FileSetAttrib FileSetEnd FileSetPos FileSetTime FileWrite FileWriteLine Floor FtpSetProxy FuncName GUICreate GUICtrlCreateAvi GUICtrlCreateButton GUICtrlCreateCheckbox GUICtrlCreateCombo GUICtrlCreateContextMenu GUICtrlCreateDate GUICtrlCreateDummy GUICtrlCreateEdit GUICtrlCreateGraphic GUICtrlCreateGroup GUICtrlCreateIcon GUICtrlCreateInput GUICtrlCreateLabel GUICtrlCreateList GUICtrlCreateListView GUICtrlCreateListViewItem GUICtrlCreateMenu GUICtrlCreateMenuItem GUICtrlCreateMonthCal GUICtrlCreateObj GUICtrlCreatePic GUICtrlCreateProgress GUICtrlCreateRadio GUICtrlCreateSlider GUICtrlCreateTab GUICtrlCreateTabItem GUICtrlCreateTreeView GUICtrlCreateTreeViewItem GUICtrlCreateUpdown GUICtrlDelete GUICtrlGetHandle GUICtrlGetState GUICtrlRead GUICtrlRecvMsg GUICtrlRegisterListViewSort GUICtrlSendMsg GUICtrlSendToDummy GUICtrlSetBkColor GUICtrlSetColor GUICtrlSetCursor GUICtrlSetData GUICtrlSetDefBkColor GUICtrlSetDefColor GUICtrlSetFont GUICtrlSetGraphic GUICtrlSetImage GUICtrlSetLimit GUICtrlSetOnEvent GUICtrlSetPos GUICtrlSetResizing GUICtrlSetState GUICtrlSetStyle GUICtrlSetTip GUIDelete GUIGetCursorInfo GUIGetMsg GUIGetStyle GUIRegisterMsg GUISetAccelerators GUISetBkColor GUISetCoord GUISetCursor GUISetFont GUISetHelp GUISetIcon GUISetOnEvent GUISetState GUISetStyle GUIStartGroup GUISwitch Hex HotKeySet HttpSetProxy HttpSetUserAgent HWnd InetClose InetGet InetGetInfo InetGetSize InetRead IniDelete IniRead IniReadSection IniReadSectionNames IniRenameSection IniWrite IniWriteSection InputBox Int IsAdmin IsArray IsBinary IsBool IsDeclared IsDllStruct IsFloat IsFunc IsHWnd IsInt IsKeyword IsNumber IsObj IsPtr IsString Log MemGetStats Mod MouseClick MouseClickDrag MouseDown MouseGetCursor MouseGetPos MouseMove MouseUp MouseWheel MsgBox Number ObjCreate ObjCreateInterface ObjEvent ObjGet ObjName OnAutoItExitRegister OnAutoItExitUnRegister Ping PixelChecksum PixelGetColor PixelSearch ProcessClose ProcessExists ProcessGetStats ProcessList ProcessSetPriority ProcessWait ProcessWaitClose ProgressOff ProgressOn ProgressSet Ptr Random RegDelete RegEnumKey RegEnumVal RegRead RegWrite Round Run RunAs RunAsWait RunWait Send SendKeepActive SetError SetExtended ShellExecute ShellExecuteWait Shutdown Sin Sleep SoundPlay SoundSetWaveVolume SplashImageOn SplashOff SplashTextOn Sqrt SRandom StatusbarGetText StderrRead StdinWrite StdioClose StdoutRead String StringAddCR StringCompare StringFormat StringFromASCIIArray StringInStr StringIsAlNum StringIsAlpha StringIsASCII StringIsDigit StringIsFloat StringIsInt StringIsLower StringIsSpace StringIsUpper StringIsXDigit StringLeft StringLen StringLower StringMid StringRegExp StringRegExpReplace StringReplace StringReverse StringRight StringSplit StringStripCR StringStripWS StringToASCIIArray StringToBinary StringTrimLeft StringTrimRight StringUpper Tan TCPAccept TCPCloseSocket TCPConnect TCPListen TCPNameToIP TCPRecv TCPSend TCPShutdown, UDPShutdown TCPStartup, UDPStartup TimerDiff TimerInit ToolTip TrayCreateItem TrayCreateMenu TrayGetMsg TrayItemDelete TrayItemGetHandle TrayItemGetState TrayItemGetText TrayItemSetOnEvent TrayItemSetState TrayItemSetText TraySetClick TraySetIcon TraySetOnEvent TraySetPauseIcon TraySetState TraySetToolTip TrayTip UBound UDPBind UDPCloseSocket UDPOpen UDPRecv UDPSend VarGetType WinActivate WinActive WinClose WinExists WinFlash WinGetCaretPos WinGetClassList WinGetClientSize WinGetHandle WinGetPos WinGetProcess WinGetState WinGetText WinGetTitle WinKill WinList WinMenuSelectItem WinMinimizeAll WinMinimizeAllUndo WinMove WinSetOnTop WinSetState WinSetTitle WinSetTrans WinWait",COMMENT={variants:[hljs.COMMENT(";","$",{relevance:0}),hljs.COMMENT("#cs","#ce"),hljs.COMMENT("#comments-start","#comments-end")]},VARIABLE={begin:"\\$[A-z0-9_]+"},STRING={className:"string",variants:[{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]},NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]},PREPROCESSOR={className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"comments include include-once NoTrayIcon OnAutoItStartRegister pragma compile RequireAdmin"},contains:[{begin:/\\\n/,relevance:0},{beginKeywords:"include",keywords:{"meta-keyword":"include"},end:"$",contains:[STRING,{className:"meta-string",variants:[{begin:"<",end:">"},{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]}]},STRING,COMMENT]},CONSTANT={className:"symbol",begin:"@[A-z0-9_]+"},FUNCTION={className:"function",beginKeywords:"Func",end:"$",illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:[VARIABLE,STRING,NUMBER]}]};return{case_insensitive:true,illegal:/\/\*/,keywords:{keyword:KEYWORDS,built_in:BUILT_IN,literal:LITERAL},contains:[COMMENT,VARIABLE,STRING,NUMBER,PREPROCESSOR,CONSTANT,FUNCTION]}});hljs.registerLanguage("less",function(hljs){var IDENT_RE="[\\w-]+";var INTERP_IDENT_RE="("+IDENT_RE+"|@{"+IDENT_RE+"})";var RULES=[],VALUE=[];var STRING_MODE=function(c){return{className:"string",begin:"~?"+c+".*?"+c}};var IDENT_MODE=function(name,begin,relevance){return{className:name,begin:begin,relevance:relevance}};var PARENS_MODE={begin:"\\(",end:"\\)",contains:VALUE,relevance:0};VALUE.push(hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING_MODE("'"),STRING_MODE('"'),hljs.CSS_NUMBER_MODE,{begin:"(url|data-uri)\\(",starts:{className:"string",end:"[\\)\\n]",excludeEnd:true}},IDENT_MODE("number","#[0-9A-Fa-f]+\\b"),PARENS_MODE,IDENT_MODE("variable","@@?"+IDENT_RE,10),IDENT_MODE("variable","@{"+IDENT_RE+"}"),IDENT_MODE("built_in","~?`[^`]*?`"),{className:"attribute",begin:IDENT_RE+"\\s*:",end:":",returnBegin:true,excludeEnd:true},{className:"meta",begin:"!important"});var VALUE_WITH_RULESETS=VALUE.concat({begin:"{",end:"}",contains:RULES});var MIXIN_GUARD_MODE={beginKeywords:"when",endsWithParent:true,contains:[{beginKeywords:"and not"}].concat(VALUE)};var RULE_MODE={begin:INTERP_IDENT_RE+"\\s*:",returnBegin:true,end:"[;}]",relevance:0,contains:[{className:"attribute",begin:INTERP_IDENT_RE,end:":",excludeEnd:true,starts:{endsWithParent:true,illegal:"[<=$]",relevance:0,contains:VALUE}}]};var AT_RULE_MODE={className:"keyword",begin:"@(import|media|charset|font-face|(-[a-z]+-)?keyframes|supports|document|namespace|page|viewport|host)\\b",starts:{end:"[;{}]",returnEnd:true,contains:VALUE,relevance:0}};var VAR_RULE_MODE={className:"variable",variants:[{begin:"@"+IDENT_RE+"\\s*:",relevance:15},{begin:"@"+IDENT_RE}],starts:{end:"[;}]",returnEnd:true,contains:VALUE_WITH_RULESETS}};var SELECTOR_MODE={variants:[{begin:"[\\.#:&\\[>]",end:"[;{}]"},{begin:INTERP_IDENT_RE,end:"{"}],returnBegin:true,returnEnd:true,illegal:"[<='$\"]",relevance:0,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,MIXIN_GUARD_MODE,IDENT_MODE("keyword","all\\b"),IDENT_MODE("variable","@{"+IDENT_RE+"}"),IDENT_MODE("selector-tag",INTERP_IDENT_RE+"%?",0),IDENT_MODE("selector-id","#"+INTERP_IDENT_RE),IDENT_MODE("selector-class","\\."+INTERP_IDENT_RE,0),IDENT_MODE("selector-tag","&",0),{className:"selector-attr",begin:"\\[",end:"\\]"},{className:"selector-pseudo",begin:/:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/},{begin:"\\(",end:"\\)",contains:VALUE_WITH_RULESETS},{begin:"!important"}]};RULES.push(hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,AT_RULE_MODE,VAR_RULE_MODE,RULE_MODE,SELECTOR_MODE);return{case_insensitive:true,illegal:"[=>'/<($\"]",contains:RULES}});hljs.registerLanguage("tp",function(hljs){var TPID={className:"number",begin:"[1-9][0-9]*",relevance:0};var TPLABEL={className:"symbol",begin:":[^\\]]+"};var TPDATA={className:"built_in",begin:"(AR|P|PAYLOAD|PR|R|SR|RSR|LBL|VR|UALM|MESSAGE|UTOOL|UFRAME|TIMER|"+"TIMER_OVERFLOW|JOINT_MAX_SPEED|RESUME_PROG|DIAG_REC)\\[",end:"\\]",contains:["self",TPID,TPLABEL]};var TPIO={className:"built_in",begin:"(AI|AO|DI|DO|F|RI|RO|UI|UO|GI|GO|SI|SO)\\[",end:"\\]",contains:["self",TPID,hljs.QUOTE_STRING_MODE,TPLABEL]};return{keywords:{keyword:"ABORT ACC ADJUST AND AP_LD BREAK CALL CNT COL CONDITION CONFIG DA DB "+"DIV DETECT ELSE END ENDFOR ERR_NUM ERROR_PROG FINE FOR GP GUARD INC "+"IF JMP LINEAR_MAX_SPEED LOCK MOD MONITOR OFFSET Offset OR OVERRIDE "+"PAUSE PREG PTH RT_LD RUN SELECT SKIP Skip TA TB TO TOOL_OFFSET "+"Tool_Offset UF UT UFRAME_NUM UTOOL_NUM UNLOCK WAIT X Y Z W P R STRLEN "+"SUBSTR FINDSTR VOFFSET PROG ATTR MN POS",literal:"ON OFF max_speed LPOS JPOS ENABLE DISABLE START STOP RESET"},contains:[TPDATA,TPIO,{className:"keyword",begin:"/(PROG|ATTR|MN|POS|END)\\b"},{className:"keyword",begin:"(CALL|RUN|POINT_LOGIC|LBL)\\b"},{className:"keyword",begin:"\\b(ACC|CNT|Skip|Offset|PSPD|RT_LD|AP_LD|Tool_Offset)"},{className:"number",begin:"\\d+(sec|msec|mm/sec|cm/min|inch/min|deg/sec|mm|in|cm)?\\b",relevance:0},hljs.COMMENT("//","[;$]"),hljs.COMMENT("!","[;$]"),hljs.COMMENT("--eg:","$"),hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"'"},hljs.C_NUMBER_MODE,{className:"variable",begin:"\\$[A-Za-z0-9_]+"}]}});hljs.registerLanguage("cos",function cos(hljs){var STRINGS={className:"string",variants:[{begin:'"',end:'"',contains:[{begin:'""',relevance:0}]}]};var NUMBERS={className:"number",begin:"\\b(\\d+(\\.\\d*)?|\\.\\d+)",relevance:0};var COS_KEYWORDS="property parameter class classmethod clientmethod extends as break "+"catch close continue do d|0 else elseif for goto halt hang h|0 if job "+"j|0 kill k|0 lock l|0 merge new open quit q|0 read r|0 return set s|0 "+"tcommit throw trollback try tstart use view while write w|0 xecute x|0 "+"zkill znspace zn ztrap zwrite zw zzdump zzwrite print zbreak zinsert "+"zload zprint zremove zsave zzprint mv mvcall mvcrt mvdim mvprint zquit "+"zsync ascii";return{case_insensitive:true,aliases:["cos","cls"],keywords:COS_KEYWORDS,contains:[NUMBERS,STRINGS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"comment",begin:/;/,end:"$",relevance:0},{className:"built_in",begin:/(?:\$\$?|\.\.)\^?[a-zA-Z]+/},{className:"built_in",begin:/\$\$\$[a-zA-Z]+/},{className:"built_in",begin:/%[a-z]+(?:\.[a-z]+)*/},{className:"symbol",begin:/\^%?[a-zA-Z][\w]*/},{className:"keyword",begin:/##class|##super|#define|#dim/},{begin:/&sql\(/,end:/\)/,excludeBegin:true,excludeEnd:true,subLanguage:"sql"},{begin:/&(js|jscript|javascript)</,end:/>/,excludeBegin:true,excludeEnd:true,subLanguage:"javascript"},{begin:/&html<\s*</,end:/>\s*>/,subLanguage:"xml"}]}});hljs.registerLanguage("erlang-repl",function(hljs){return{keywords:{built_in:"spawn spawn_link self",keyword:"after and andalso|10 band begin bnot bor bsl bsr bxor case catch cond div end fun if "+"let not of or orelse|10 query receive rem try when xor"},contains:[{className:"meta",begin:"^[0-9]+> ",relevance:10},hljs.COMMENT("%","$"),{className:"number",begin:"\\b(\\d+#[a-fA-F0-9]+|\\d+(\\.\\d+)?([eE][-+]?\\d+)?)",relevance:0},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\?(::)?([A-Z]\\w*(::)?)+"},{begin:"->"},{begin:"ok"},{begin:"!"},{begin:"(\\b[a-z'][a-zA-Z0-9_']*:[a-z'][a-zA-Z0-9_']*)|(\\b[a-z'][a-zA-Z0-9_']*)",relevance:0},{begin:"[A-Z][a-zA-Z0-9_']*",relevance:0}]}});hljs.registerLanguage("vbscript",function(hljs){return{aliases:["vbs"],case_insensitive:true,keywords:{keyword:"call class const dim do loop erase execute executeglobal exit for each next function "+"if then else on error option explicit new private property let get public randomize "+"redim rem select case set stop sub while wend with end to elseif is or xor and not "+"class_initialize class_terminate default preserve in me byval byref step resume goto",built_in:"lcase month vartype instrrev ubound setlocale getobject rgb getref string "+"weekdayname rnd dateadd monthname now day minute isarray cbool round formatcurrency "+"conversions csng timevalue second year space abs clng timeserial fixs len asc "+"isempty maths dateserial atn timer isobject filter weekday datevalue ccur isdate "+"instr datediff formatdatetime replace isnull right sgn array snumeric log cdbl hex "+"chr lbound msgbox ucase getlocale cos cdate cbyte rtrim join hour oct typename trim "+"strcomp int createobject loadpicture tan formatnumber mid scriptenginebuildversion "+"scriptengine split scriptengineminorversion cint sin datepart ltrim sqr "+"scriptenginemajorversion time derived eval date formatpercent exp inputbox left ascw "+"chrw regexp server response request cstr err",literal:"true false null nothing empty"},illegal:"//",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[{begin:'""'}]}),hljs.COMMENT(/'/,/$/,{relevance:0}),hljs.C_NUMBER_MODE]}});hljs.registerLanguage("vbscript-html",function(hljs){return{subLanguage:"xml",contains:[{begin:"<%",end:"%>",subLanguage:"vbscript"}]}});hljs.registerLanguage("twig",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var FUNCTION_NAMES="attribute block constant cycle date dump include "+"max min parent random range source template_from_string";var FUNCTIONS={beginKeywords:FUNCTION_NAMES,keywords:{name:FUNCTION_NAMES},relevance:0,contains:[PARAMS]};var FILTER={begin:/\|[A-Za-z_]+:?/,keywords:"abs batch capitalize column convert_encoding date date_modify default "+"escape filter first format inky_to_html inline_css join json_encode keys last "+"length lower map markdown merge nl2br number_format raw reduce replace "+"reverse round slice sort spaceless split striptags title trim upper url_encode",contains:[FUNCTIONS]};var TAGS="apply autoescape block deprecated do embed extends filter flush for from "+"if import include macro sandbox set use verbatim with";TAGS=TAGS+" "+TAGS.split(" ").map(function(t){return"end"+t}).join(" ");return{aliases:["craftcms"],case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT(/\{#/,/#}/),{className:"template-tag",begin:/\{%/,end:/%}/,contains:[{className:"name",begin:/\w+/,keywords:TAGS,starts:{endsWithParent:true,contains:[FILTER,FUNCTIONS],relevance:0}}]},{className:"template-variable",begin:/\{\{/,end:/}}/,contains:["self",FILTER,FUNCTIONS]}]}});hljs.registerLanguage("cmake",function(hljs){return{aliases:["cmake.in"],case_insensitive:true,keywords:{keyword:"break cmake_host_system_information cmake_minimum_required cmake_parse_arguments "+"cmake_policy configure_file continue elseif else endforeach endfunction endif endmacro "+"endwhile execute_process file find_file find_library find_package find_path "+"find_program foreach function get_cmake_property get_directory_property "+"get_filename_component get_property if include include_guard list macro "+"mark_as_advanced math message option return separate_arguments "+"set_directory_properties set_property set site_name string unset variable_watch while "+"add_compile_definitions add_compile_options add_custom_command add_custom_target "+"add_definitions add_dependencies add_executable add_library add_link_options "+"add_subdirectory add_test aux_source_directory build_command create_test_sourcelist "+"define_property enable_language enable_testing export fltk_wrap_ui "+"get_source_file_property get_target_property get_test_property include_directories "+"include_external_msproject include_regular_expression install link_directories "+"link_libraries load_cache project qt_wrap_cpp qt_wrap_ui remove_definitions "+"set_source_files_properties set_target_properties set_tests_properties source_group "+"target_compile_definitions target_compile_features target_compile_options "+"target_include_directories target_link_directories target_link_libraries "+"target_link_options target_sources try_compile try_run "+"ctest_build ctest_configure ctest_coverage ctest_empty_binary_directory ctest_memcheck "+"ctest_read_custom_files ctest_run_script ctest_sleep ctest_start ctest_submit "+"ctest_test ctest_update ctest_upload "+"build_name exec_program export_library_dependencies install_files install_programs "+"install_targets load_command make_directory output_required_files remove "+"subdir_depends subdirs use_mangled_mesa utility_source variable_requires write_file "+"qt5_use_modules qt5_use_package qt5_wrap_cpp "+"on off true false and or not command policy target test exists is_newer_than "+"is_directory is_symlink is_absolute matches less greater equal less_equal "+"greater_equal strless strgreater strequal strless_equal strgreater_equal version_less "+"version_greater version_equal version_less_equal version_greater_equal in_list defined"},contains:[{className:"variable",begin:"\\${",end:"}"},hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE]}});hljs.registerLanguage("handlebars",function(hljs){var BUILT_INS={"builtin-name":"each in with if else unless bindattr action collection debugger log outlet template unbound view yield lookup"};var IDENTIFIER_PLAIN_OR_QUOTED={begin:/".*?"|'.*?'|\[.*?\]|\w+/};var EXPRESSION_OR_HELPER_CALL=hljs.inherit(IDENTIFIER_PLAIN_OR_QUOTED,{keywords:BUILT_INS,starts:{endsWithParent:true,relevance:0,contains:[hljs.inherit(IDENTIFIER_PLAIN_OR_QUOTED,{relevance:0})]}});var BLOCK_MUSTACHE_CONTENTS=hljs.inherit(EXPRESSION_OR_HELPER_CALL,{className:"name"});var BASIC_MUSTACHE_CONTENTS=hljs.inherit(EXPRESSION_OR_HELPER_CALL,{relevance:0});var ESCAPE_MUSTACHE_WITH_PRECEEDING_BACKSLASH={begin:/\\\{\{/,skip:true};var PREVENT_ESCAPE_WITH_ANOTHER_PRECEEDING_BACKSLASH={begin:/\\\\(?=\{\{)/,skip:true};return{aliases:["hbs","html.hbs","html.handlebars"],case_insensitive:true,subLanguage:"xml",contains:[ESCAPE_MUSTACHE_WITH_PRECEEDING_BACKSLASH,PREVENT_ESCAPE_WITH_ANOTHER_PRECEEDING_BACKSLASH,hljs.COMMENT(/\{\{!--/,/--\}\}/),hljs.COMMENT(/\{\{!/,/\}\}/),{className:"template-tag",begin:/\{\{\{\{(?!\/)/,end:/\}\}\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS],starts:{end:/\{\{\{\{\//,returnEnd:true,subLanguage:"xml"}},{className:"template-tag",begin:/\{\{\{\{\//,end:/\}\}\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS]},{className:"template-tag",begin:/\{\{[#\/]/,end:/\}\}/,contains:[BLOCK_MUSTACHE_CONTENTS]},{className:"template-variable",begin:/\{\{\{/,end:/\}\}\}/,keywords:BUILT_INS,contains:[BASIC_MUSTACHE_CONTENTS]},{className:"template-variable",begin:/\{\{/,end:/\}\}/,keywords:BUILT_INS,contains:[BASIC_MUSTACHE_CONTENTS]}]}});hljs.registerLanguage("delphi",function(hljs){var KEYWORDS="exports register file shl array record property for mod while set ally label uses raise not "+"stored class safecall var interface or private static exit index inherited to else stdcall "+"override shr asm far resourcestring finalization packed virtual out and protected library do "+"xorwrite goto near function end div overload object unit begin string on inline repeat until "+"destructor write message program with read initialization except default nil if case cdecl in "+"downto threadvar of try pascal const external constructor type public then implementation "+"finally published procedure absolute reintroduce operator as is abstract alias assembler "+"bitpacked break continue cppdecl cvar enumerator experimental platform deprecated "+"unimplemented dynamic export far16 forward generic helper implements interrupt iochecks "+"local name nodefault noreturn nostackframe oldfpccall otherwise saveregisters softfloat "+"specialize strict unaligned varargs ";var COMMENT_MODES=[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/\{/,/\}/,{relevance:0}),hljs.COMMENT(/\(\*/,/\*\)/,{relevance:10})];var DIRECTIVE={className:"meta",variants:[{begin:/\{\$/,end:/\}/},{begin:/\(\*\$/,end:/\*\)/}]};var STRING={className:"string",begin:/'/,end:/'/,contains:[{begin:/''/}]};var CHAR_STRING={className:"string",begin:/(#\d+)+/};var CLASS={begin:hljs.IDENT_RE+"\\s*=\\s*class\\s*\\(",returnBegin:true,contains:[hljs.TITLE_MODE]};var FUNCTION={className:"function",beginKeywords:"function constructor destructor procedure",end:/[:;]/,keywords:"function constructor|10 destructor|10 procedure|10",contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:[STRING,CHAR_STRING,DIRECTIVE].concat(COMMENT_MODES)},DIRECTIVE].concat(COMMENT_MODES)};return{aliases:["dpr","dfm","pas","pascal","freepascal","lazarus","lpr","lfm"],case_insensitive:true,keywords:KEYWORDS,illegal:/"|\$[G-Zg-z]|\/\*|<\/|\|/,contains:[STRING,CHAR_STRING,hljs.NUMBER_MODE,CLASS,FUNCTION,DIRECTIVE].concat(COMMENT_MODES)}});hljs.registerLanguage("javascript",function(hljs){var FRAGMENT={begin:"<>",end:"</>"};var XML_TAG={begin:/<[A-Za-z0-9\\._:-]+/,end:/\/[A-Za-z0-9\\._:-]+>|\/>/};var IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var KEYWORDS={keyword:"in of if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const export super debugger as async await static "+"import from as",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect "+"Promise"};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)n?"},{begin:"\\b(0[oO][0-7]+)n?"},{begin:hljs.C_NUMBER_RE+"n?"}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var HTML_TEMPLATE={begin:"html`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"xml"}};var CSS_TEMPLATE={begin:"css`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"css"}};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];var PARAMS_CONTAINS=SUBST.contains.concat([hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE]);return{aliases:["js","jsx","mjs","cjs"],keywords:KEYWORDS,contains:[{className:"meta",relevance:10,begin:/^\s*['"]use (strict|asm)['"]/},{className:"meta",begin:/^#!/,end:/$/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag",begin:"@[A-Za-z]+",contains:[{className:"type",begin:"\\{",end:"\\}",relevance:0},{className:"variable",begin:IDENT_RE+"(?=\\s*(-)|$)",endsParent:true,relevance:0},{begin:/(?=[^\n])\s/,relevance:0}]}]}),hljs.C_BLOCK_COMMENT_MODE,NUMBER,{begin:/[{,\n]\s*/,relevance:0,contains:[{begin:IDENT_RE+"\\s*:",returnBegin:true,relevance:0,contains:[{className:"attr",begin:IDENT_RE,relevance:0}]}]},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:PARAMS_CONTAINS}]}]},{className:"",begin:/\s/,end:/\s*/,skip:true},{variants:[{begin:FRAGMENT.begin,end:FRAGMENT.end},{begin:XML_TAG.begin,end:XML_TAG.end}],subLanguage:"xml",contains:[{begin:XML_TAG.begin,end:XML_TAG.end,skip:true,contains:["self"]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:PARAMS_CONTAINS}],illegal:/\[|%/},{begin:/\$[(.]/},hljs.METHOD_GUARD,{className:"class",beginKeywords:"class",end:/[{;=]/,excludeEnd:true,illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"constructor get set",end:/\{/,excludeEnd:true}],illegal:/#(?!!)/}});hljs.registerLanguage("rust",function(hljs){var NUM_SUFFIX="([ui](8|16|32|64|128|size)|f(32|64))?";var KEYWORDS="abstract as async await become box break const continue crate do dyn "+"else enum extern false final fn for if impl in let loop macro match mod "+"move mut override priv pub ref return self Self static struct super "+"trait true try type typeof unsafe unsized use virtual where while yield";var BUILTINS="drop "+"i8 i16 i32 i64 i128 isize "+"u8 u16 u32 u64 u128 usize "+"f32 f64 "+"str char bool "+"Box Option Result String Vec "+"Copy Send Sized Sync Drop Fn FnMut FnOnce ToOwned Clone Debug "+"PartialEq PartialOrd Eq Ord AsRef AsMut Into From Default Iterator "+"Extend IntoIterator DoubleEndedIterator ExactSizeIterator "+"SliceConcatExt ToString "+"assert! assert_eq! bitflags! bytes! cfg! col! concat! concat_idents! "+"debug_assert! debug_assert_eq! env! panic! file! format! format_args! "+"include_bin! include_str! line! local_data_key! module_path! "+"option_env! print! println! select! stringify! try! unimplemented! "+"unreachable! vec! write! writeln! macro_rules! assert_ne! debug_assert_ne!";return{aliases:["rs"],keywords:{keyword:KEYWORDS,literal:"true false Some None Ok Err",built_in:BUILTINS},lexemes:hljs.IDENT_RE+"!?",illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT("/\\*","\\*/",{contains:["self"]}),hljs.inherit(hljs.QUOTE_STRING_MODE,{begin:/b?"/,illegal:null}),{className:"string",variants:[{begin:/r(#*)"(.|\n)*?"\1(?!#)/},{begin:/b?'\\?(x\w{2}|u\w{4}|U\w{8}|.)'/}]},{className:"symbol",begin:/'[a-zA-Z_][a-zA-Z0-9_]*/},{className:"number",variants:[{begin:"\\b0b([01_]+)"+NUM_SUFFIX},{begin:"\\b0o([0-7_]+)"+NUM_SUFFIX},{begin:"\\b0x([A-Fa-f0-9_]+)"+NUM_SUFFIX},{begin:"\\b(\\d[\\d_]*(\\.[0-9_]+)?([eE][+-]?[0-9_]+)?)"+NUM_SUFFIX}],relevance:0},{className:"function",beginKeywords:"fn",end:"(\\(|<)",excludeEnd:true,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"meta",begin:"#\\!?\\[",end:"\\]",contains:[{className:"meta-string",begin:/"/,end:/"/}]},{className:"class",beginKeywords:"type",end:";",contains:[hljs.inherit(hljs.UNDERSCORE_TITLE_MODE,{endsParent:true})],illegal:"\\S"},{className:"class",beginKeywords:"trait enum struct union",end:"{",contains:[hljs.inherit(hljs.UNDERSCORE_TITLE_MODE,{endsParent:true})],illegal:"[\\w\\d]"},{begin:hljs.IDENT_RE+"::",keywords:{built_in:BUILTINS}},{begin:"->"}]}});hljs.registerLanguage("diff",function(hljs){return{aliases:["patch"],contains:[{className:"meta",relevance:10,variants:[{begin:/^@@ +\-\d+,\d+ +\+\d+,\d+ +@@$/},{begin:/^\*\*\* +\d+,\d+ +\*\*\*\*$/},{begin:/^\-\-\- +\d+,\d+ +\-\-\-\-$/}]},{className:"comment",variants:[{begin:/Index: /,end:/$/},{begin:/={3,}/,end:/$/},{begin:/^\-{3}/,end:/$/},{begin:/^\*{3} /,end:/$/},{begin:/^\+{3}/,end:/$/},{begin:/^\*{15}$/}]},{className:"addition",begin:"^\\+",end:"$"},{className:"deletion",begin:"^\\-",end:"$"},{className:"addition",begin:"^\\!",end:"$"}]}});hljs.registerLanguage("thrift",function(hljs){var BUILT_IN_TYPES="bool byte i16 i32 i64 double string binary";return{keywords:{keyword:"namespace const typedef struct enum service exception void oneway set list map required optional",built_in:BUILT_IN_TYPES,literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"class",beginKeywords:"struct enum service exception",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{begin:"\\b(set|list|map)\\s*<",end:">",keywords:BUILT_IN_TYPES,contains:["self"]}]}});hljs.registerLanguage("accesslog",function(hljs){var HTTP_VERBS=["GET","POST","HEAD","PUT","DELETE","CONNECT","OPTIONS","PATCH","TRACE"];return{contains:[{className:"number",begin:"^\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b",relevance:5},{className:"number",begin:"\\b\\d+\\b",relevance:0},{className:"string",begin:'"('+HTTP_VERBS.join("|")+")",end:'"',keywords:HTTP_VERBS.join(" "),illegal:"\\n",relevance:5,contains:[{begin:"HTTP/[12]\\.\\d",relevance:5}]},{className:"string",begin:/\[\d[^\]\n]{8,}\]/,illegal:"\\n",relevance:1},{className:"string",begin:/\[/,end:/\]/,illegal:"\\n",relevance:0},{className:"string",begin:'"Mozilla/\\d\\.\\d \\(',end:'"',illegal:"\\n",relevance:3},{className:"string",begin:'"',end:'"',illegal:"\\n",relevance:0}]}});hljs.registerLanguage("tex",function(hljs){var COMMAND={className:"tag",begin:/\\/,relevance:0,contains:[{className:"name",variants:[{begin:/[a-zA-Z\u0430-\u044f\u0410-\u042f]+[*]?/},{begin:/[^a-zA-Z\u0430-\u044f\u0410-\u042f0-9]/}],starts:{endsWithParent:true,relevance:0,contains:[{className:"string",variants:[{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/}]},{begin:/\s*=\s*/,endsWithParent:true,relevance:0,contains:[{className:"number",begin:/-?\d*\.?\d+(pt|pc|mm|cm|in|dd|cc|ex|em)?/}]}]}}]};return{contains:[COMMAND,{className:"formula",contains:[COMMAND],relevance:0,variants:[{begin:/\$\$/,end:/\$\$/},{begin:/\$/,end:/\$/}]},hljs.COMMENT("%","$",{relevance:0})]}});hljs.registerLanguage("livescript",function(hljs){var KEYWORDS={keyword:"in if for while finally new do return else break catch instanceof throw try this "+"switch continue typeof delete debugger case default function var with "+"then unless until loop of by when and or is isnt not it that otherwise from to til fallthrough super "+"case default function var void const let enum export import native list map "+"__hasProp __extends __slice __bind __indexOf",literal:"true false null undefined "+"yes no on off it that void",built_in:"npm require console print module global window document"};var JS_IDENT_RE="[A-Za-z$_](?:-[0-9A-Za-z$_]|[0-9A-Za-z$_])*";var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var SUBST_SIMPLE={className:"subst",begin:/#[A-Za-z$_]/,end:/(?:\-[0-9A-Za-z$_]|[0-9A-Za-z$_])*/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.BINARY_NUMBER_MODE,{className:"number",begin:"(\\b0[xX][a-fA-F0-9_]+)|(\\b\\d(\\d|_\\d)*(\\.(\\d(\\d|_\\d)*)?)?(_*[eE]([-+]\\d(_\\d|\\d)*)?)?[_a-z]*)",relevance:0,starts:{end:"(\\s*/)?",relevance:0}},{className:"string",variants:[{begin:/'''/,end:/'''/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"""/,end:/"""/,contains:[hljs.BACKSLASH_ESCAPE,SUBST,SUBST_SIMPLE]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST,SUBST_SIMPLE]},{begin:/\\/,end:/(\s|$)/,excludeEnd:true}]},{className:"regexp",variants:[{begin:"//",end:"//[gim]*",contains:[SUBST,hljs.HASH_COMMENT_MODE]},{begin:/\/(?![ *])(\\.|[^\\\n])*?\/[gim]*(?=\W)/}]},{begin:"@"+JS_IDENT_RE},{begin:"``",end:"``",excludeBegin:true,excludeEnd:true,subLanguage:"javascript"}];SUBST.contains=EXPRESSIONS;var PARAMS={className:"params",begin:"\\(",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};var SYMBOLS={begin:"(#=>|=>|\\|>>|-?->|\\!->)"};return{aliases:["ls"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("\\/\\*","\\*\\/"),hljs.HASH_COMMENT_MODE,SYMBOLS,{className:"function",contains:[TITLE,PARAMS],returnBegin:true,variants:[{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?(\\(.*\\))?\\s*\\B\\->\\*?",end:"\\->\\*?"},{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?!?(\\(.*\\))?\\s*\\B[-~]{1,2}>\\*?",end:"[-~]{1,2}>\\*?"},{begin:"("+JS_IDENT_RE+"\\s*(?:=|:=)\\s*)?(\\(.*\\))?\\s*\\B!?[-~]{1,2}>\\*?",end:"!?[-~]{1,2}>\\*?"}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("apache",function(hljs){var NUMBER={className:"number",begin:"[\\$%]\\d+"};return{aliases:["apacheconf"],case_insensitive:true,contains:[hljs.HASH_COMMENT_MODE,{className:"section",begin:"</?",end:">"},{className:"attribute",begin:/\w+/,relevance:0,keywords:{nomarkup:"order deny allow setenv rewriterule rewriteengine rewritecond documentroot "+"sethandler errordocument loadmodule options header listen serverroot "+"servername"},starts:{end:/$/,relevance:0,keywords:{literal:"on off all"},contains:[{className:"meta",begin:"\\s\\[",end:"\\]$"},{className:"variable",begin:"[\\$%]\\{",end:"\\}",contains:["self",NUMBER]},NUMBER,hljs.QUOTE_STRING_MODE]}}],illegal:/\S/}});hljs.registerLanguage("typescript",function(hljs){var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var KEYWORDS={keyword:"in if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const class public private protected get set super "+"static implements enum export import declare type namespace abstract "+"as from extends async await",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document any number boolean string void Promise"};var DECORATOR={className:"meta",begin:"@"+JS_IDENT_RE};var ARGS={begin:"\\(",end:/\)/,keywords:KEYWORDS,contains:["self",hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.NUMBER_MODE]};var PARAMS={className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,DECORATOR,ARGS]};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)n?"},{begin:"\\b(0[oO][0-7]+)n?"},{begin:hljs.C_NUMBER_RE+"n?"}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var HTML_TEMPLATE={begin:"html`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"xml"}};var CSS_TEMPLATE={begin:"css`",end:"",starts:{end:"`",returnEnd:false,contains:[hljs.BACKSLASH_ESCAPE,SUBST],subLanguage:"css"}};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];return{aliases:["ts"],keywords:KEYWORDS,contains:[{className:"meta",begin:/^\s*['"]use strict['"]/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,HTML_TEMPLATE,CSS_TEMPLATE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBER,{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+hljs.IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:hljs.IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:["self",hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/[\{;]/,excludeEnd:true,keywords:KEYWORDS,contains:["self",hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE}),PARAMS],illegal:/%/,relevance:0},{beginKeywords:"constructor",end:/[\{;]/,excludeEnd:true,contains:["self",PARAMS]},{begin:/module\./,keywords:{built_in:"module"},relevance:0},{beginKeywords:"module",end:/\{/,excludeEnd:true},{beginKeywords:"interface",end:/\{/,excludeEnd:true,keywords:"interface extends"},{begin:/\$[(.]/},{begin:"\\."+hljs.IDENT_RE,relevance:0},DECORATOR,ARGS]}});hljs.registerLanguage("roboconf",function(hljs){var IDENTIFIER="[a-zA-Z-_][^\\n{]+\\{";var PROPERTY={className:"attribute",begin:/[a-zA-Z-_]+/,end:/\s*:/,excludeEnd:true,starts:{end:";",relevance:0,contains:[{className:"variable",begin:/\.[a-zA-Z-_]+/},{className:"keyword",begin:/\(optional\)/}]}};return{aliases:["graph","instances"],case_insensitive:true,keywords:"import",contains:[{begin:"^facet "+IDENTIFIER,end:"}",keywords:"facet",contains:[PROPERTY,hljs.HASH_COMMENT_MODE]},{begin:"^\\s*instance of "+IDENTIFIER,end:"}",keywords:"name count channels instance-data instance-state instance of",illegal:/\S/,contains:["self",PROPERTY,hljs.HASH_COMMENT_MODE]},{begin:"^"+IDENTIFIER,end:"}",contains:[PROPERTY,hljs.HASH_COMMENT_MODE]},hljs.HASH_COMMENT_MODE]}});hljs.registerLanguage("htmlbars",function(hljs){var BUILT_INS="action collection component concat debugger each each-in else get hash if input link-to loc log mut outlet partial query-params render textarea unbound unless with yield view";var ATTR_ASSIGNMENT={illegal:/\}\}/,begin:/[a-zA-Z0-9_]+=/,returnBegin:true,relevance:0,contains:[{className:"attr",begin:/[a-zA-Z0-9_]+/}]};var SUB_EXPR={illegal:/\}\}/,begin:/\)/,end:/\)/,contains:[{begin:/[a-zA-Z\.\-]+/,keywords:{built_in:BUILT_INS},starts:{endsWithParent:true,relevance:0,contains:[hljs.QUOTE_STRING_MODE]}}]};var TAG_INNARDS={endsWithParent:true,relevance:0,keywords:{keyword:"as",built_in:BUILT_INS},contains:[hljs.QUOTE_STRING_MODE,ATTR_ASSIGNMENT,hljs.NUMBER_MODE]};return{case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT("{{!(--)?","(--)?}}"),{className:"template-tag",begin:/\{\{[#\/]/,end:/\}\}/,contains:[{className:"name",begin:/[a-zA-Z\.\-]+/,keywords:{"builtin-name":BUILT_INS},starts:TAG_INNARDS}]},{className:"template-variable",begin:/\{\{[a-zA-Z][a-zA-Z\-]+/,end:/\}\}/,keywords:{keyword:"as",built_in:BUILT_INS},contains:[hljs.QUOTE_STRING_MODE]}]}});hljs.registerLanguage("routeros",function(hljs){var STATEMENTS="foreach do while for if from to step else on-error and or not in";var GLOBAL_COMMANDS="global local beep delay put len typeof pick log time set find environment terminal error execute parse resolve toarray tobool toid toip toip6 tonum tostr totime";var COMMON_COMMANDS="add remove enable disable set get print export edit find run debug error info warning";var LITERALS="true false yes no nothing nil null";var OBJECTS="traffic-flow traffic-generator firewall scheduler aaa accounting address-list address align area bandwidth-server bfd bgp bridge client clock community config connection console customer default dhcp-client dhcp-server discovery dns e-mail ethernet filter firewall firmware gps graphing group hardware health hotspot identity igmp-proxy incoming instance interface ip ipsec ipv6 irq l2tp-server lcd ldp logging mac-server mac-winbox mangle manual mirror mme mpls nat nd neighbor network note ntp ospf ospf-v3 ovpn-server page peer pim ping policy pool port ppp pppoe-client pptp-server prefix profile proposal proxy queue radius resource rip ripng route routing screen script security-profiles server service service-port settings shares smb sms sniffer snmp snooper socks sstp-server system tool tracking type upgrade upnp user-manager users user vlan secret vrrp watchdog web-access wireless pptp pppoe lan wan layer7-protocol lease simple raw";var VAR_PREFIX="global local set for foreach";var VAR={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VAR,{className:"variable",begin:/\$\(/,end:/\)/,contains:[hljs.BACKSLASH_ESCAPE]}]};var APOS_STRING={className:"string",begin:/'/,end:/'/};var IPADDR="((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]).){3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\\b";var IPADDR_wBITMASK=IPADDR+"/(3[0-2]|[1-2][0-9]|\\d)";return{aliases:["routeros","mikrotik"],case_insensitive:true,lexemes:/:?[\w-]+/,keywords:{literal:LITERALS,keyword:STATEMENTS+" :"+STATEMENTS.split(" ").join(" :")+" :"+GLOBAL_COMMANDS.split(" ").join(" :")},contains:[{variants:[{begin:/^@/,end:/$/},{begin:/\/\*/,end:/\*\//},{begin:/%%/,end:/$/},{begin:/^'/,end:/$/},{begin:/^\s*\/[\w-]+=/,end:/$/},{begin:/\/\//,end:/$/},{begin:/^\[\</,end:/\>\]$/},{begin:/<\//,end:/>/},{begin:/^facet /,end:/\}/},{begin:"^1\\.\\.(\\d+)$",end:/$/}],illegal:/./},hljs.COMMENT("^#","$"),QUOTE_STRING,APOS_STRING,VAR,{begin:/[\w-]+\=([^\s\{\}\[\]\(\)]+)/,relevance:0,returnBegin:true,contains:[{className:"attribute",begin:/[^=]+/},{begin:/=/,endsWithParent:true,relevance:0,contains:[QUOTE_STRING,APOS_STRING,VAR,{className:"literal",begin:"\\b("+LITERALS.split(" ").join("|")+")\\b"},{begin:/("[^"]*"|[^\s\{\}\[\]]+)/}]}]},{className:"number",begin:/\*[0-9a-fA-F]+/},{begin:"\\b("+COMMON_COMMANDS.split(" ").join("|")+")([\\s[(]|])",returnBegin:true,contains:[{className:"builtin-name",begin:/\w+/}]},{className:"built_in",variants:[{begin:"(\\.\\./|/|\\s)(("+OBJECTS.split(" ").join("|")+");?\\s)+",relevance:10},{begin:/\.\./}]}]}});hljs.registerLanguage("mercury",function(hljs){var KEYWORDS={keyword:"module use_module import_module include_module end_module initialise "+"mutable initialize finalize finalise interface implementation pred "+"mode func type inst solver any_pred any_func is semidet det nondet "+"multi erroneous failure cc_nondet cc_multi typeclass instance where "+"pragma promise external trace atomic or_else require_complete_switch "+"require_det require_semidet require_multi require_nondet "+"require_cc_multi require_cc_nondet require_erroneous require_failure",meta:"inline no_inline type_spec source_file fact_table obsolete memo "+"loop_check minimal_model terminates does_not_terminate "+"check_termination promise_equivalent_clauses "+"foreign_proc foreign_decl foreign_code foreign_type "+"foreign_import_module foreign_export_enum foreign_export "+"foreign_enum may_call_mercury will_not_call_mercury thread_safe "+"not_thread_safe maybe_thread_safe promise_pure promise_semipure "+"tabled_for_io local untrailed trailed attach_to_io_state "+"can_pass_as_mercury_type stable will_not_throw_exception "+"may_modify_trail will_not_modify_trail may_duplicate "+"may_not_duplicate affects_liveness does_not_affect_liveness "+"doesnt_affect_liveness no_sharing unknown_sharing sharing",built_in:"some all not if then else true fail false try catch catch_any "+"semidet_true semidet_false semidet_fail impure_true impure semipure"};var COMMENT=hljs.COMMENT("%","$");var NUMCODE={className:"number",begin:"0'.\\|0[box][0-9a-fA-F]*"};var ATOM=hljs.inherit(hljs.APOS_STRING_MODE,{relevance:0});var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{relevance:0});var STRING_FMT={className:"subst",begin:"\\\\[abfnrtv]\\|\\\\x[0-9a-fA-F]*\\\\\\|%[-+# *.0-9]*[dioxXucsfeEgGp]",relevance:0};STRING.contains=STRING.contains.slice();STRING.contains.push(STRING_FMT);var IMPLICATION={className:"built_in",variants:[{begin:"<=>"},{begin:"<=",relevance:0},{begin:"=>",relevance:0},{begin:"/\\\\"},{begin:"\\\\/"}]};var HEAD_BODY_CONJUNCTION={className:"built_in",variants:[{begin:":-\\|--\x3e"},{begin:"=",relevance:0}]};return{aliases:["m","moo"],keywords:KEYWORDS,contains:[IMPLICATION,HEAD_BODY_CONJUNCTION,COMMENT,hljs.C_BLOCK_COMMENT_MODE,NUMCODE,hljs.NUMBER_MODE,ATOM,STRING,{begin:/:-/},{begin:/\.$/}]}});hljs.registerLanguage("inform7",function(hljs){var START_BRACKET="\\[";var END_BRACKET="\\]";return{aliases:["i7"],case_insensitive:true,keywords:{keyword:"thing room person man woman animal container "+"supporter backdrop door "+"scenery open closed locked inside gender "+"is are say understand "+"kind of rule"},contains:[{className:"string",begin:'"',end:'"',relevance:0,contains:[{className:"subst",begin:START_BRACKET,end:END_BRACKET}]},{className:"section",begin:/^(Volume|Book|Part|Chapter|Section|Table)\b/,end:"$"},{begin:/^(Check|Carry out|Report|Instead of|To|Rule|When|Before|After)\b/,end:":",contains:[{begin:"\\(This",end:"\\)"}]},{className:"comment",begin:START_BRACKET,end:END_BRACKET,contains:["self"]}]}});hljs.registerLanguage("lua",function(hljs){var OPENING_LONG_BRACKET="\\[=*\\[";var CLOSING_LONG_BRACKET="\\]=*\\]";var LONG_BRACKETS={begin:OPENING_LONG_BRACKET,end:CLOSING_LONG_BRACKET,contains:["self"]};var COMMENTS=[hljs.COMMENT("--(?!"+OPENING_LONG_BRACKET+")","$"),hljs.COMMENT("--"+OPENING_LONG_BRACKET,CLOSING_LONG_BRACKET,{contains:[LONG_BRACKETS],relevance:10})];return{lexemes:hljs.UNDERSCORE_IDENT_RE,keywords:{literal:"true false nil",keyword:"and break do else elseif end for goto if in local not or repeat return then until while",built_in:"_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len "+"__gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert "+"collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstring"+"module next pairs pcall print rawequal rawget rawset require select setfenv"+"setmetatable tonumber tostring type unpack xpcall arg self"+"coroutine resume yield status wrap create running debug getupvalue "+"debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv "+"io lines write close flush open output type read stderr stdin input stdout popen tmpfile "+"math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan "+"os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall "+"string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower "+"table setn insert getn foreachi maxn foreach concat sort remove"},contains:COMMENTS.concat([{className:"function",beginKeywords:"function",end:"\\)",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),{className:"params",begin:"\\(",endsWithParent:true,contains:COMMENTS}].concat(COMMENTS)},hljs.C_NUMBER_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:OPENING_LONG_BRACKET,end:CLOSING_LONG_BRACKET,contains:[LONG_BRACKETS],relevance:5}])}});hljs.registerLanguage("crmsh",function(hljs){var RESOURCES="primitive rsc_template";var COMMANDS="group clone ms master location colocation order fencing_topology "+"rsc_ticket acl_target acl_group user role "+"tag xml";var PROPERTY_SETS="property rsc_defaults op_defaults";var KEYWORDS="params meta operations op rule attributes utilization";var OPERATORS="read write deny defined not_defined in_range date spec in "+"ref reference attribute type xpath version and or lt gt tag "+"lte gte eq ne \\";var TYPES="number string";var LITERALS="Master Started Slave Stopped start promote demote stop monitor true false";return{aliases:["crm","pcmk"],case_insensitive:true,keywords:{keyword:KEYWORDS+" "+OPERATORS+" "+TYPES,literal:LITERALS},contains:[hljs.HASH_COMMENT_MODE,{beginKeywords:"node",starts:{end:"\\s*([\\w_-]+:)?",starts:{className:"title",end:"\\s*[\\$\\w_][\\w_-]*"}}},{beginKeywords:RESOURCES,starts:{className:"title",end:"\\s*[\\$\\w_][\\w_-]*",starts:{end:"\\s*@?[\\w_][\\w_\\.:-]*"}}},{begin:"\\b("+COMMANDS.split(" ").join("|")+")\\s+",keywords:COMMANDS,starts:{className:"title",end:"[\\$\\w_][\\w_-]*"}},{beginKeywords:PROPERTY_SETS,starts:{className:"title",end:"\\s*([\\w_-]+:)?"}},hljs.QUOTE_STRING_MODE,{className:"meta",begin:"(ocf|systemd|service|lsb):[\\w_:-]+",relevance:0},{className:"number",begin:"\\b\\d+(\\.\\d+)?(ms|s|h|m)?",relevance:0},{className:"literal",begin:"[-]?(infinity|inf)",relevance:0},{className:"attr",begin:/([A-Za-z\$_\#][\w_-]+)=/,relevance:0},{className:"tag",begin:"</?",end:"/?>",relevance:0}]}});hljs.registerLanguage("scheme",function(hljs){var SCHEME_IDENT_RE="[^\\(\\)\\[\\]\\{\\}\",'`;#|\\\\\\s]+";var SCHEME_SIMPLE_NUMBER_RE="(\\-|\\+)?\\d+([./]\\d+)?";var SCHEME_COMPLEX_NUMBER_RE=SCHEME_SIMPLE_NUMBER_RE+"[+\\-]"+SCHEME_SIMPLE_NUMBER_RE+"i";var BUILTINS={"builtin-name":"case-lambda call/cc class define-class exit-handler field import "+"inherit init-field interface let*-values let-values let/ec mixin "+"opt-lambda override protect provide public rename require "+"require-for-syntax syntax syntax-case syntax-error unit/sig unless "+"when with-syntax and begin call-with-current-continuation "+"call-with-input-file call-with-output-file case cond define "+"define-syntax delay do dynamic-wind else for-each if lambda let let* "+"let-syntax letrec letrec-syntax map or syntax-rules ' * + , ,@ - ... / "+"; < <= = => > >= ` abs acos angle append apply asin assoc assq assv atan "+"boolean? caar cadr call-with-input-file call-with-output-file "+"call-with-values car cdddar cddddr cdr ceiling char->integer "+"char-alphabetic? char-ci<=? char-ci<? char-ci=? char-ci>=? char-ci>? "+"char-downcase char-lower-case? char-numeric? char-ready? char-upcase "+"char-upper-case? char-whitespace? char<=? char<? char=? char>=? char>? "+"char? close-input-port close-output-port complex? cons cos "+"current-input-port current-output-port denominator display eof-object? "+"eq? equal? eqv? eval even? exact->inexact exact? exp expt floor "+"force gcd imag-part inexact->exact inexact? input-port? integer->char "+"integer? interaction-environment lcm length list list->string "+"list->vector list-ref list-tail list? load log magnitude make-polar "+"make-rectangular make-string make-vector max member memq memv min "+"modulo negative? newline not null-environment null? number->string "+"number? numerator odd? open-input-file open-output-file output-port? "+"pair? peek-char port? positive? procedure? quasiquote quote quotient "+"rational? rationalize read read-char real-part real? remainder reverse "+"round scheme-report-environment set! set-car! set-cdr! sin sqrt string "+"string->list string->number string->symbol string-append string-ci<=? "+"string-ci<? string-ci=? string-ci>=? string-ci>? string-copy "+"string-fill! string-length string-ref string-set! string<=? string<? "+"string=? string>=? string>? string? substring symbol->string symbol? "+"tan transcript-off transcript-on truncate values vector "+"vector->list vector-fill! vector-length vector-ref vector-set! "+"with-input-from-file with-output-to-file write write-char zero?"};var SHEBANG={className:"meta",begin:"^#!",end:"$"};var LITERAL={className:"literal",begin:"(#t|#f|#\\\\"+SCHEME_IDENT_RE+"|#\\\\.)"};var NUMBER={className:"number",variants:[{begin:SCHEME_SIMPLE_NUMBER_RE,relevance:0},{begin:SCHEME_COMPLEX_NUMBER_RE,relevance:0},{begin:"#b[0-1]+(/[0-1]+)?"},{begin:"#o[0-7]+(/[0-7]+)?"},{begin:"#x[0-9a-f]+(/[0-9a-f]+)?"}]};var STRING=hljs.QUOTE_STRING_MODE;var REGULAR_EXPRESSION={className:"regexp",begin:'#[pr]x"',end:'[^\\\\]"'};var COMMENT_MODES=[hljs.COMMENT(";","$",{relevance:0}),hljs.COMMENT("#\\|","\\|#")];var IDENT={begin:SCHEME_IDENT_RE,relevance:0};var QUOTED_IDENT={className:"symbol",begin:"'"+SCHEME_IDENT_RE};var BODY={endsWithParent:true,relevance:0};var QUOTED_LIST={variants:[{begin:/'/},{begin:"`"}],contains:[{begin:"\\(",end:"\\)",contains:["self",LITERAL,STRING,NUMBER,IDENT,QUOTED_IDENT]}]};var NAME={className:"name",begin:SCHEME_IDENT_RE,lexemes:SCHEME_IDENT_RE,keywords:BUILTINS};var LAMBDA={begin:/lambda/,endsWithParent:true,returnBegin:true,contains:[NAME,{begin:/\(/,end:/\)/,endsParent:true,contains:[IDENT]}]};var LIST={variants:[{begin:"\\(",end:"\\)"},{begin:"\\[",end:"\\]"}],contains:[LAMBDA,NAME,BODY]};BODY.contains=[LITERAL,NUMBER,STRING,IDENT,QUOTED_IDENT,QUOTED_LIST,LIST].concat(COMMENT_MODES);return{illegal:/\S/,contains:[SHEBANG,NUMBER,STRING,QUOTED_IDENT,QUOTED_LIST,LIST].concat(COMMENT_MODES)}});hljs.registerLanguage("bash",function(hljs){var VAR={className:"variable",variants:[{begin:/\$[\w\d#@][\w\d_]*/},{begin:/\$\{(.*?)}/}]};var QUOTE_STRING={className:"string",begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,VAR,{className:"variable",begin:/\$\(/,end:/\)/,contains:[hljs.BACKSLASH_ESCAPE]}]};var ESCAPED_QUOTE={className:"",begin:/\\"/};var APOS_STRING={className:"string",begin:/'/,end:/'/};return{aliases:["sh","zsh"],lexemes:/\b-?[a-z\._]+\b/,keywords:{keyword:"if then else elif fi for while in do done case esac function",literal:"true false",built_in:"break cd continue eval exec exit export getopts hash pwd readonly return shift test times "+"trap umask unset "+"alias bind builtin caller command declare echo enable help let local logout mapfile printf "+"read readarray source type typeset ulimit unalias "+"set shopt "+"autoload bg bindkey bye cap chdir clone comparguments compcall compctl compdescribe compfiles "+"compgroups compquote comptags comptry compvalues dirs disable disown echotc echoti emulate "+"fc fg float functions getcap getln history integer jobs kill limit log noglob popd print "+"pushd pushln rehash sched setcap setopt stat suspend ttyctl unfunction unhash unlimit "+"unsetopt vared wait whence where which zcompile zformat zftp zle zmodload zparseopts zprof "+"zpty zregexparse zsocket zstyle ztcp",_:"-ne -eq -lt -gt -f -d -e -s -l -a"},contains:[{className:"meta",begin:/^#![^\n]+sh\s*$/,relevance:10},{className:"function",begin:/\w[\w\d_]*\s*\(\s*\)\s*\{/,returnBegin:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/\w[\w\d_]*/})],relevance:0},hljs.HASH_COMMENT_MODE,QUOTE_STRING,ESCAPED_QUOTE,APOS_STRING,VAR]}});hljs.registerLanguage("dockerfile",function(hljs){return{aliases:["docker"],case_insensitive:true,keywords:"from maintainer expose env arg user onbuild stopsignal",contains:[hljs.HASH_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,{beginKeywords:"run cmd entrypoint volume add copy workdir label healthcheck shell",starts:{end:/[^\\]$/,subLanguage:"bash"}}],illegal:"</"}});hljs.registerLanguage("properties",function(hljs){var WS0="[ \\t\\f]*";var WS1="[ \\t\\f]+";var DELIM="("+WS0+"[:=]"+WS0+"|"+WS1+")";var KEY_ALPHANUM="([^\\\\\\W:= \\t\\f\\n]|\\\\.)+";var KEY_OTHER="([^\\\\:= \\t\\f\\n]|\\\\.)+";var DELIM_AND_VALUE={end:DELIM,relevance:0,starts:{className:"string",end:/$/,relevance:0,contains:[{begin:"\\\\\\n"}]}};return{case_insensitive:true,illegal:/\S/,contains:[hljs.COMMENT("^\\s*[!#]","$"),{begin:KEY_ALPHANUM+DELIM,returnBegin:true,contains:[{className:"attr",begin:KEY_ALPHANUM,endsParent:true,relevance:0}],starts:DELIM_AND_VALUE},{begin:KEY_OTHER+DELIM,returnBegin:true,relevance:0,contains:[{className:"meta",begin:KEY_OTHER,endsParent:true,relevance:0}],starts:DELIM_AND_VALUE},{className:"attr",relevance:0,begin:KEY_OTHER+WS0+"$"}]}});hljs.registerLanguage("dns",function(hljs){return{aliases:["bind","zone"],keywords:{keyword:"IN A AAAA AFSDB APL CAA CDNSKEY CDS CERT CNAME DHCID DLV DNAME DNSKEY DS HIP IPSECKEY KEY KX "+"LOC MX NAPTR NS NSEC NSEC3 NSEC3PARAM PTR RRSIG RP SIG SOA SRV SSHFP TA TKEY TLSA TSIG TXT"},contains:[hljs.COMMENT(";","$",{relevance:0}),{className:"meta",begin:/^\$(TTL|GENERATE|INCLUDE|ORIGIN)\b/},{className:"number",begin:"((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)(\\.(25[0-5]|2[0-4]\\d|1\\d\\d|[1-9]?\\d)){3}))|:)))\\b"},{className:"number",begin:"((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]).){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\\b"},hljs.inherit(hljs.NUMBER_MODE,{begin:/\b\d+[dhwm]?/})]}});hljs.registerLanguage("monkey",function(hljs){var NUMBER={className:"number",relevance:0,variants:[{begin:"[$][a-fA-F0-9]+"},hljs.NUMBER_MODE]};return{case_insensitive:true,keywords:{keyword:"public private property continue exit extern new try catch "+"eachin not abstract final select case default const local global field "+"end if then else elseif endif while wend repeat until forever for "+"to step next return module inline throw import",built_in:"DebugLog DebugStop Error Print ACos ACosr ASin ASinr ATan ATan2 ATan2r ATanr Abs Abs Ceil "+"Clamp Clamp Cos Cosr Exp Floor Log Max Max Min Min Pow Sgn Sgn Sin Sinr Sqrt Tan Tanr Seed PI HALFPI TWOPI",literal:"true false null and or shl shr mod"},illegal:/\/\*/,contains:[hljs.COMMENT("#rem","#end"),hljs.COMMENT("'","$",{relevance:0}),{className:"function",beginKeywords:"function method",end:"[(=:]|$",illegal:/\n/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"class",beginKeywords:"class interface",end:"$",contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{className:"built_in",begin:"\\b(self|super)\\b"},{className:"meta",begin:"\\s*#",end:"$",keywords:{"meta-keyword":"if else elseif endif end then"}},{className:"meta",begin:"^\\s*strict\\b"},{beginKeywords:"alias",end:"=",contains:[hljs.UNDERSCORE_TITLE_MODE]},hljs.QUOTE_STRING_MODE,NUMBER]}});hljs.registerLanguage("golo",function(hljs){return{keywords:{keyword:"println readln print import module function local return let var "+"while for foreach times in case when match with break continue "+"augment augmentation each find filter reduce "+"if then else otherwise try catch finally raise throw orIfNull "+"DynamicObject|10 DynamicVariable struct Observable map set vector list array",literal:"true false null"},contains:[hljs.HASH_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("livecodeserver",function(hljs){var VARIABLE={className:"variable",variants:[{begin:"\\b([gtps][A-Z]{1}[a-zA-Z0-9]*)(\\[.+\\])?(?:\\s*?)"},{begin:"\\$_[A-Z]+"}],relevance:0};var COMMENT_MODES=[hljs.C_BLOCK_COMMENT_MODE,hljs.HASH_COMMENT_MODE,hljs.COMMENT("--","$"),hljs.COMMENT("[^:]//","$")];var TITLE1=hljs.inherit(hljs.TITLE_MODE,{variants:[{begin:"\\b_*rig[A-Z]+[A-Za-z0-9_\\-]*"},{begin:"\\b_[a-z0-9\\-]+"}]});var TITLE2=hljs.inherit(hljs.TITLE_MODE,{begin:"\\b([A-Za-z0-9_\\-]+)\\b"});return{case_insensitive:false,keywords:{keyword:"$_COOKIE $_FILES $_GET $_GET_BINARY $_GET_RAW $_POST $_POST_BINARY $_POST_RAW $_SESSION $_SERVER "+"codepoint codepoints segment segments codeunit codeunits sentence sentences trueWord trueWords paragraph "+"after byte bytes english the until http forever descending using line real8 with seventh "+"for stdout finally element word words fourth before black ninth sixth characters chars stderr "+"uInt1 uInt1s uInt2 uInt2s stdin string lines relative rel any fifth items from middle mid "+"at else of catch then third it file milliseconds seconds second secs sec int1 int1s int4 "+"int4s internet int2 int2s normal text item last long detailed effective uInt4 uInt4s repeat "+"end repeat URL in try into switch to words https token binfile each tenth as ticks tick "+"system real4 by dateItems without char character ascending eighth whole dateTime numeric short "+"first ftp integer abbreviated abbr abbrev private case while if "+"div mod wrap and or bitAnd bitNot bitOr bitXor among not in a an within "+"contains ends with begins the keys of keys",literal:"SIX TEN FORMFEED NINE ZERO NONE SPACE FOUR FALSE COLON CRLF PI COMMA ENDOFFILE EOF EIGHT FIVE "+"QUOTE EMPTY ONE TRUE RETURN CR LINEFEED RIGHT BACKSLASH NULL SEVEN TAB THREE TWO "+"six ten formfeed nine zero none space four false colon crlf pi comma endoffile eof eight five "+"quote empty one true return cr linefeed right backslash null seven tab three two "+"RIVERSION RISTATE FILE_READ_MODE FILE_WRITE_MODE FILE_WRITE_MODE DIR_WRITE_MODE FILE_READ_UMASK "+"FILE_WRITE_UMASK DIR_READ_UMASK DIR_WRITE_UMASK",built_in:"put abs acos aliasReference annuity arrayDecode arrayEncode asin atan atan2 average avg avgDev base64Decode "+"base64Encode baseConvert binaryDecode binaryEncode byteOffset byteToNum cachedURL cachedURLs charToNum "+"cipherNames codepointOffset codepointProperty codepointToNum codeunitOffset commandNames compound compress "+"constantNames cos date dateFormat decompress difference directories "+"diskSpace DNSServers exp exp1 exp2 exp10 extents files flushEvents folders format functionNames geometricMean global "+"globals hasMemory harmonicMean hostAddress hostAddressToName hostName hostNameToAddress isNumber ISOToMac itemOffset "+"keys len length libURLErrorData libUrlFormData libURLftpCommand libURLLastHTTPHeaders libURLLastRHHeaders "+"libUrlMultipartFormAddPart libUrlMultipartFormData libURLVersion lineOffset ln ln1 localNames log log2 log10 "+"longFilePath lower macToISO matchChunk matchText matrixMultiply max md5Digest median merge messageAuthenticationCode messageDigest millisec "+"millisecs millisecond milliseconds min monthNames nativeCharToNum normalizeText num number numToByte numToChar "+"numToCodepoint numToNativeChar offset open openfiles openProcesses openProcessIDs openSockets "+"paragraphOffset paramCount param params peerAddress pendingMessages platform popStdDev populationStandardDeviation "+"populationVariance popVariance processID random randomBytes replaceText result revCreateXMLTree revCreateXMLTreeFromFile "+"revCurrentRecord revCurrentRecordIsFirst revCurrentRecordIsLast revDatabaseColumnCount revDatabaseColumnIsNull "+"revDatabaseColumnLengths revDatabaseColumnNames revDatabaseColumnNamed revDatabaseColumnNumbered "+"revDatabaseColumnTypes revDatabaseConnectResult revDatabaseCursors revDatabaseID revDatabaseTableNames "+"revDatabaseType revDataFromQuery revdb_closeCursor revdb_columnbynumber revdb_columncount revdb_columnisnull "+"revdb_columnlengths revdb_columnnames revdb_columntypes revdb_commit revdb_connect revdb_connections "+"revdb_connectionerr revdb_currentrecord revdb_cursorconnection revdb_cursorerr revdb_cursors revdb_dbtype "+"revdb_disconnect revdb_execute revdb_iseof revdb_isbof revdb_movefirst revdb_movelast revdb_movenext "+"revdb_moveprev revdb_query revdb_querylist revdb_recordcount revdb_rollback revdb_tablenames "+"revGetDatabaseDriverPath revNumberOfRecords revOpenDatabase revOpenDatabases revQueryDatabase "+"revQueryDatabaseBlob revQueryResult revQueryIsAtStart revQueryIsAtEnd revUnixFromMacPath revXMLAttribute "+"revXMLAttributes revXMLAttributeValues revXMLChildContents revXMLChildNames revXMLCreateTreeFromFileWithNamespaces "+"revXMLCreateTreeWithNamespaces revXMLDataFromXPathQuery revXMLEvaluateXPath revXMLFirstChild revXMLMatchingNode "+"revXMLNextSibling revXMLNodeContents revXMLNumberOfChildren revXMLParent revXMLPreviousSibling "+"revXMLRootNode revXMLRPC_CreateRequest revXMLRPC_Documents revXMLRPC_Error "+"revXMLRPC_GetHost revXMLRPC_GetMethod revXMLRPC_GetParam revXMLText revXMLRPC_Execute "+"revXMLRPC_GetParamCount revXMLRPC_GetParamNode revXMLRPC_GetParamType revXMLRPC_GetPath revXMLRPC_GetPort "+"revXMLRPC_GetProtocol revXMLRPC_GetRequest revXMLRPC_GetResponse revXMLRPC_GetSocket revXMLTree "+"revXMLTrees revXMLValidateDTD revZipDescribeItem revZipEnumerateItems revZipOpenArchives round sampVariance "+"sec secs seconds sentenceOffset sha1Digest shell shortFilePath sin specialFolderPath sqrt standardDeviation statRound "+"stdDev sum sysError systemVersion tan tempName textDecode textEncode tick ticks time to tokenOffset toLower toUpper "+"transpose truewordOffset trunc uniDecode uniEncode upper URLDecode URLEncode URLStatus uuid value variableNames "+"variance version waitDepth weekdayNames wordOffset xsltApplyStylesheet xsltApplyStylesheetFromFile xsltLoadStylesheet "+"xsltLoadStylesheetFromFile add breakpoint cancel clear local variable file word line folder directory URL close socket process "+"combine constant convert create new alias folder directory decrypt delete variable word line folder "+"directory URL dispatch divide do encrypt filter get include intersect kill libURLDownloadToFile "+"libURLFollowHttpRedirects libURLftpUpload libURLftpUploadFile libURLresetAll libUrlSetAuthCallback libURLSetDriver "+"libURLSetCustomHTTPHeaders libUrlSetExpect100 libURLSetFTPListCommand libURLSetFTPMode libURLSetFTPStopTime "+"libURLSetStatusCallback load extension loadedExtensions multiply socket prepare process post seek rel relative read from process rename "+"replace require resetAll resolve revAddXMLNode revAppendXML revCloseCursor revCloseDatabase revCommitDatabase "+"revCopyFile revCopyFolder revCopyXMLNode revDeleteFolder revDeleteXMLNode revDeleteAllXMLTrees "+"revDeleteXMLTree revExecuteSQL revGoURL revInsertXMLNode revMoveFolder revMoveToFirstRecord revMoveToLastRecord "+"revMoveToNextRecord revMoveToPreviousRecord revMoveToRecord revMoveXMLNode revPutIntoXMLNode revRollBackDatabase "+"revSetDatabaseDriverPath revSetXMLAttribute revXMLRPC_AddParam revXMLRPC_DeleteAllDocuments revXMLAddDTD "+"revXMLRPC_Free revXMLRPC_FreeAll revXMLRPC_DeleteDocument revXMLRPC_DeleteParam revXMLRPC_SetHost "+"revXMLRPC_SetMethod revXMLRPC_SetPort revXMLRPC_SetProtocol revXMLRPC_SetSocket revZipAddItemWithData "+"revZipAddItemWithFile revZipAddUncompressedItemWithData revZipAddUncompressedItemWithFile revZipCancel "+"revZipCloseArchive revZipDeleteItem revZipExtractItemToFile revZipExtractItemToVariable revZipSetProgressCallback "+"revZipRenameItem revZipReplaceItemWithData revZipReplaceItemWithFile revZipOpenArchive send set sort split start stop "+"subtract symmetric union unload vectorDotProduct wait write"},contains:[VARIABLE,{className:"keyword",begin:"\\bend\\sif\\b"},{className:"function",beginKeywords:"function",end:"$",contains:[VARIABLE,TITLE2,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1]},{className:"function",begin:"\\bend\\s+",end:"$",keywords:"end",contains:[TITLE2,TITLE1],relevance:0},{beginKeywords:"command on",end:"$",contains:[VARIABLE,TITLE2,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1]},{className:"meta",variants:[{begin:"<\\?(rev|lc|livecode)",relevance:10},{begin:"<\\?"},{begin:"\\?>"}]},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE,TITLE1].concat(COMMENT_MODES),illegal:";$|^\\[|^=|&|{"}});hljs.registerLanguage("haml",function(hljs){return{case_insensitive:true,contains:[{className:"meta",begin:"^!!!( (5|1\\.1|Strict|Frameset|Basic|Mobile|RDFa|XML\\b.*))?$",relevance:10},hljs.COMMENT("^\\s*(!=#|=#|-#|/).*$",false,{relevance:0}),{begin:"^\\s*(-|=|!=)(?!#)",starts:{end:"\\n",subLanguage:"ruby"}},{className:"tag",begin:"^\\s*%",contains:[{className:"selector-tag",begin:"\\w+"},{className:"selector-id",begin:"#[\\w-]+"},{className:"selector-class",begin:"\\.[\\w-]+"},{begin:"{\\s*",end:"\\s*}",contains:[{begin:":\\w+\\s*=>",end:",\\s+",returnBegin:true,endsWithParent:true,contains:[{className:"attr",begin:":\\w+"},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\w+",relevance:0}]}]},{begin:"\\(\\s*",end:"\\s*\\)",excludeEnd:true,contains:[{begin:"\\w+\\s*=",end:"\\s+",returnBegin:true,endsWithParent:true,contains:[{className:"attr",begin:"\\w+",relevance:0},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{begin:"\\w+",relevance:0}]}]}]},{begin:"^\\s*[=~]\\s*"},{begin:"#{",starts:{end:"}",subLanguage:"ruby"}}]}});hljs.registerLanguage("gcode",function(hljs){var GCODE_IDENT_RE="[A-Z_][A-Z0-9_.]*";var GCODE_CLOSE_RE="\\%";var GCODE_KEYWORDS="IF DO WHILE ENDWHILE CALL ENDIF SUB ENDSUB GOTO REPEAT ENDREPEAT "+"EQ LT GT NE GE LE OR XOR";var GCODE_START={className:"meta",begin:"([O])([0-9]+)"};var GCODE_CODE=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(/\(/,/\)/),hljs.inherit(hljs.C_NUMBER_MODE,{begin:"([-+]?([0-9]*\\.?[0-9]+\\.?))|"+hljs.C_NUMBER_RE}),hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"name",begin:"([G])([0-9]+\\.?[0-9]?)"},{className:"name",begin:"([M])([0-9]+\\.?[0-9]?)"},{className:"attr",begin:"(VC|VS|#)",end:"(\\d+)"},{className:"attr",begin:"(VZOFX|VZOFY|VZOFZ)"},{className:"built_in",begin:"(ATAN|ABS|ACOS|ASIN|SIN|COS|EXP|FIX|FUP|ROUND|LN|TAN)(\\[)",end:"([-+]?([0-9]*\\.?[0-9]+\\.?))(\\])"},{className:"symbol",variants:[{begin:"N",end:"\\d+",illegal:"\\W"}]}];return{aliases:["nc"],case_insensitive:true,lexemes:GCODE_IDENT_RE,keywords:GCODE_KEYWORDS,contains:[{className:"meta",begin:GCODE_CLOSE_RE},GCODE_START].concat(GCODE_CODE)}});hljs.registerLanguage("erlang",function(hljs){var BASIC_ATOM_RE="[a-z'][a-zA-Z0-9_']*";var FUNCTION_NAME_RE="("+BASIC_ATOM_RE+":"+BASIC_ATOM_RE+"|"+BASIC_ATOM_RE+")";var ERLANG_RESERVED={keyword:"after and andalso|10 band begin bnot bor bsl bzr bxor case catch cond div end fun if "+"let not of orelse|10 query receive rem try when xor",literal:"false true"};var COMMENT=hljs.COMMENT("%","$");var NUMBER={className:"number",begin:"\\b(\\d+#[a-fA-F0-9]+|\\d+(\\.\\d+)?([eE][-+]?\\d+)?)",relevance:0};var NAMED_FUN={begin:"fun\\s+"+BASIC_ATOM_RE+"/\\d+"};var FUNCTION_CALL={begin:FUNCTION_NAME_RE+"\\(",end:"\\)",returnBegin:true,relevance:0,contains:[{begin:FUNCTION_NAME_RE,relevance:0},{begin:"\\(",end:"\\)",endsWithParent:true,returnEnd:true,relevance:0}]};var TUPLE={begin:"{",end:"}",relevance:0};var VAR1={begin:"\\b_([A-Z][A-Za-z0-9_]*)?",relevance:0};var VAR2={begin:"[A-Z][a-zA-Z0-9_]*",relevance:0};var RECORD_ACCESS={begin:"#"+hljs.UNDERSCORE_IDENT_RE,relevance:0,returnBegin:true,contains:[{begin:"#"+hljs.UNDERSCORE_IDENT_RE,relevance:0},{begin:"{",end:"}",relevance:0}]};var BLOCK_STATEMENTS={beginKeywords:"fun receive if try case",end:"end",keywords:ERLANG_RESERVED};BLOCK_STATEMENTS.contains=[COMMENT,NAMED_FUN,hljs.inherit(hljs.APOS_STRING_MODE,{className:""}),BLOCK_STATEMENTS,FUNCTION_CALL,hljs.QUOTE_STRING_MODE,NUMBER,TUPLE,VAR1,VAR2,RECORD_ACCESS];var BASIC_MODES=[COMMENT,NAMED_FUN,BLOCK_STATEMENTS,FUNCTION_CALL,hljs.QUOTE_STRING_MODE,NUMBER,TUPLE,VAR1,VAR2,RECORD_ACCESS];FUNCTION_CALL.contains[1].contains=BASIC_MODES;TUPLE.contains=BASIC_MODES;RECORD_ACCESS.contains[1].contains=BASIC_MODES;var PARAMS={className:"params",begin:"\\(",end:"\\)",contains:BASIC_MODES};return{aliases:["erl"],keywords:ERLANG_RESERVED,illegal:"(</|\\*=|\\+=|-=|/\\*|\\*/|\\(\\*|\\*\\))",contains:[{className:"function",begin:"^"+BASIC_ATOM_RE+"\\s*\\(",end:"->",returnBegin:true,illegal:"\\(|#|//|/\\*|\\\\|:|;",contains:[PARAMS,hljs.inherit(hljs.TITLE_MODE,{begin:BASIC_ATOM_RE})],starts:{end:";|\\.",keywords:ERLANG_RESERVED,contains:BASIC_MODES}},COMMENT,{begin:"^-",end:"\\.",relevance:0,excludeEnd:true,returnBegin:true,lexemes:"-"+hljs.IDENT_RE,keywords:"-module -record -undef -export -ifdef -ifndef -author -copyright -doc -vsn "+"-import -include -include_lib -compile -define -else -endif -file -behaviour "+"-behavior -spec",contains:[PARAMS]},NUMBER,hljs.QUOTE_STRING_MODE,RECORD_ACCESS,VAR1,VAR2,TUPLE,{begin:/\.$/}]}});hljs.registerLanguage("dust",function(hljs){var EXPRESSION_KEYWORDS="if eq ne lt lte gt gte select default math sep";return{aliases:["dst"],case_insensitive:true,subLanguage:"xml",contains:[{className:"template-tag",begin:/\{[#\/]/,end:/\}/,illegal:/;/,contains:[{className:"name",begin:/[a-zA-Z\.-]+/,starts:{endsWithParent:true,relevance:0,contains:[hljs.QUOTE_STRING_MODE]}}]},{className:"template-variable",begin:/\{/,end:/\}/,illegal:/;/,keywords:EXPRESSION_KEYWORDS}]}});hljs.registerLanguage("mizar",function(hljs){return{keywords:"environ vocabularies notations constructors definitions "+"registrations theorems schemes requirements begin end definition "+"registration cluster existence pred func defpred deffunc theorem "+"proof let take assume then thus hence ex for st holds consider "+"reconsider such that and in provided of as from be being by means "+"equals implies iff redefine define now not or attr is mode "+"suppose per cases set thesis contradiction scheme reserve struct "+"correctness compatibility coherence symmetry assymetry "+"reflexivity irreflexivity connectedness uniqueness commutativity "+"idempotence involutiveness projectivity",contains:[hljs.COMMENT("::","$")]}});hljs.registerLanguage("arcade",function(hljs){var IDENT_RE="[A-Za-z_][0-9A-Za-z_]*";var KEYWORDS={keyword:"if for while var new function do return void else break",literal:"BackSlash DoubleQuote false ForwardSlash Infinity NaN NewLine null PI SingleQuote Tab TextFormatting true undefined",built_in:"Abs Acos Angle Attachments Area AreaGeodetic Asin Atan Atan2 Average Bearing Boolean Buffer BufferGeodetic "+"Ceil Centroid Clip Console Constrain Contains Cos Count Crosses Cut Date DateAdd "+"DateDiff Day Decode DefaultValue Dictionary Difference Disjoint Distance DistanceGeodetic Distinct "+"DomainCode DomainName Equals Exp Extent Feature FeatureSet FeatureSetByAssociation FeatureSetById FeatureSetByPortalItem "+"FeatureSetByRelationshipName FeatureSetByTitle FeatureSetByUrl Filter First Floor Geometry GroupBy Guid HasKey Hour IIf IndexOf "+"Intersection Intersects IsEmpty IsNan IsSelfIntersecting Length LengthGeodetic Log Max Mean Millisecond Min Minute Month "+"MultiPartToSinglePart Multipoint NextSequenceValue Now Number OrderBy Overlaps Point Polygon "+"Polyline Portal Pow Random Relate Reverse RingIsClockWise Round Second SetGeometry Sin Sort Sqrt Stdev Sum "+"SymmetricDifference Tan Text Timestamp Today ToLocal Top Touches ToUTC TrackCurrentTime "+"TrackGeometryWindow TrackIndex TrackStartTime TrackWindow TypeOf Union UrlEncode Variance "+"Weekday When Within Year "};var EXPRESSIONS;var SYMBOL={className:"symbol",begin:"\\$[datastore|feature|layer|map|measure|sourcefeature|sourcelayer|targetfeature|targetlayer|value|view]+"};var NUMBER={className:"number",variants:[{begin:"\\b(0[bB][01]+)"},{begin:"\\b(0[oO][0-7]+)"},{begin:hljs.C_NUMBER_RE}],relevance:0};var SUBST={className:"subst",begin:"\\$\\{",end:"\\}",keywords:KEYWORDS,contains:[]};var TEMPLATE_STRING={className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,SUBST]};SUBST.contains=[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,TEMPLATE_STRING,NUMBER,hljs.REGEXP_MODE];var PARAMS_CONTAINS=SUBST.contains.concat([hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE]);return{aliases:["arcade"],keywords:KEYWORDS,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,TEMPLATE_STRING,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,SYMBOL,NUMBER,{begin:/[{,]\s*/,relevance:0,contains:[{begin:IDENT_RE+"\\s*:",returnBegin:true,relevance:0,contains:[{className:"attr",begin:IDENT_RE,relevance:0}]}]},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(return)\\b)\\s*",keywords:"return",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{className:"function",begin:"(\\(.*?\\)|"+IDENT_RE+")\\s*=>",returnBegin:true,end:"\\s*=>",contains:[{className:"params",variants:[{begin:IDENT_RE},{begin:/\(\s*\)/},{begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,keywords:KEYWORDS,contains:PARAMS_CONTAINS}]}]}],relevance:0},{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:IDENT_RE}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:PARAMS_CONTAINS}],illegal:/\[|%/},{begin:/\$[(.]/}],illegal:/#(?!!)/}});hljs.registerLanguage("php",function(hljs){var VARIABLE={begin:"\\$+[a-zA-Z_-ÿ][a-zA-Z0-9_-ÿ]*"};var PREPROCESSOR={className:"meta",begin:/<\?(php)?|\?>/};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,PREPROCESSOR],variants:[{begin:'b"',end:'"'},{begin:"b'",end:"'"},hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]};var NUMBER={variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]};return{aliases:["php","php3","php4","php5","php6","php7"],case_insensitive:true,keywords:"and include_once list abstract global private echo interface as static endswitch "+"array null if endwhile or const for endforeach self var while isset public "+"protected exit foreach throw elseif include __FILE__ empty require_once do xor "+"return parent clone use __CLASS__ __LINE__ else break print eval new "+"catch __METHOD__ case exception default die require __FUNCTION__ "+"enddeclare final try switch continue endfor endif declare unset true false "+"trait goto instanceof insteadof __DIR__ __NAMESPACE__ "+"yield finally",contains:[hljs.HASH_COMMENT_MODE,hljs.COMMENT("//","$",{contains:[PREPROCESSOR]}),hljs.COMMENT("/\\*","\\*/",{contains:[{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.COMMENT("__halt_compiler.+?;",false,{endsWithParent:true,keywords:"__halt_compiler",lexemes:hljs.UNDERSCORE_IDENT_RE}),{className:"string",begin:/<<<['"]?\w+['"]?$/,end:/^\w+;?$/,contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",variants:[{begin:/\$\w+/},{begin:/\{\$/,end:/\}/}]}]},PREPROCESSOR,{className:"keyword",begin:/\$this\b/},VARIABLE,{begin:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{className:"function",beginKeywords:"function",end:/[;{]/,excludeEnd:true,illegal:"\\$|\\[|%",contains:[hljs.UNDERSCORE_TITLE_MODE,{className:"params",begin:"\\(",end:"\\)",contains:["self",VARIABLE,hljs.C_BLOCK_COMMENT_MODE,STRING,NUMBER]}]},{className:"class",beginKeywords:"class interface",end:"{",excludeEnd:true,illegal:/[:\(\$"]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"namespace",end:";",illegal:/[\.']/,contains:[hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"use",end:";",contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"=>"},STRING,NUMBER]}});hljs.registerLanguage("haxe",function(hljs){var IDENT_RE="[a-zA-Z_$][a-zA-Z0-9_$]*";var IDENT_FUNC_RETURN_TYPE_RE="([*]|[a-zA-Z_$][a-zA-Z0-9_$]*)";var HAXE_BASIC_TYPES="Int Float String Bool Dynamic Void Array ";return{aliases:["hx"],keywords:{keyword:"break case cast catch continue default do dynamic else enum extern "+"for function here if import in inline never new override package private get set "+"public return static super switch this throw trace try typedef untyped using var while "+HAXE_BASIC_TYPES,built_in:"trace this",literal:"true false null _"},contains:[{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",begin:"\\$\\{",end:"\\}"},{className:"subst",begin:"\\$",end:"\\W}"}]},hljs.QUOTE_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"@:",end:"$"},{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elseif end error"}},{className:"type",begin:":[ \t]*",end:"[^A-Za-z0-9_ \t\\->]",excludeBegin:true,excludeEnd:true,relevance:0},{className:"type",begin:":[ \t]*",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"type",begin:"new *",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"class",beginKeywords:"enum",end:"\\{",contains:[hljs.TITLE_MODE]},{className:"class",beginKeywords:"abstract",end:"[\\{$]",contains:[{className:"type",begin:"\\(",end:"\\)",excludeBegin:true,excludeEnd:true},{className:"type",begin:"from +",end:"\\W",excludeBegin:true,excludeEnd:true},{className:"type",begin:"to +",end:"\\W",excludeBegin:true,excludeEnd:true},hljs.TITLE_MODE],keywords:{keyword:"abstract from to"}},{className:"class",begin:"\\b(class|interface) +",end:"[\\{$]",excludeEnd:true,keywords:"class interface",contains:[{className:"keyword",begin:"\\b(extends|implements) +",keywords:"extends implements",contains:[{className:"type",begin:hljs.IDENT_RE,relevance:0}]},hljs.TITLE_MODE]},{className:"function",beginKeywords:"function",end:"\\(",excludeEnd:true,illegal:"\\S",contains:[hljs.TITLE_MODE]}],illegal:/<\//}});hljs.registerLanguage("autohotkey",function(hljs){var BACKTICK_ESCAPE={begin:"`[\\s\\S]"};return{case_insensitive:true,aliases:["ahk"],keywords:{keyword:"Break Continue Critical Exit ExitApp Gosub Goto New OnExit Pause return SetBatchLines SetTimer Suspend Thread Throw Until ahk_id ahk_class ahk_pid ahk_exe ahk_group",literal:"true false NOT AND OR",built_in:"ComSpec Clipboard ClipboardAll ErrorLevel"},contains:[BACKTICK_ESCAPE,hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[BACKTICK_ESCAPE]}),hljs.COMMENT(";","$",{relevance:0}),hljs.C_BLOCK_COMMENT_MODE,{className:"number",begin:hljs.NUMBER_RE,relevance:0},{className:"variable",begin:"%[a-zA-Z0-9#_$@]+%"},{className:"built_in",begin:"^\\s*\\w+\\s*(,|%)"},{className:"title",variants:[{begin:'^[^\\n";]+::(?!=)'},{begin:'^[^\\n";]+:(?!=)',relevance:0}]},{className:"meta",begin:"^\\s*#\\w+",end:"$",relevance:0},{className:"built_in",begin:"A_[a-zA-Z0-9]+"},{begin:",\\s*,"}]}});hljs.registerLanguage("protobuf",function(hljs){return{keywords:{keyword:"package import option optional required repeated group oneof",built_in:"double float int32 int64 uint32 uint64 sint32 sint64 "+"fixed32 fixed64 sfixed32 sfixed64 bool string bytes",literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.C_LINE_COMMENT_MODE,{className:"class",beginKeywords:"message enum service",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{className:"function",beginKeywords:"rpc",end:/;/,excludeEnd:true,keywords:"rpc returns"},{begin:/^\s*[A-Z_]+/,end:/\s*=/,excludeEnd:true}]}});hljs.registerLanguage("capnproto",function(hljs){return{aliases:["capnp"],keywords:{keyword:"struct enum interface union group import using const annotation extends in of on as with from fixed",built_in:"Void Bool Int8 Int16 Int32 Int64 UInt8 UInt16 UInt32 UInt64 Float32 Float64 "+"Text Data AnyPointer AnyStruct Capability List",literal:"true false"},contains:[hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.HASH_COMMENT_MODE,{className:"meta",begin:/@0x[\w\d]{16};/,illegal:/\n/},{className:"symbol",begin:/@\d+\b/},{className:"class",beginKeywords:"struct enum",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]},{className:"class",beginKeywords:"interface",end:/\{/,illegal:/\n/,contains:[hljs.inherit(hljs.TITLE_MODE,{starts:{endsWithParent:true,excludeEnd:true}})]}]}});hljs.registerLanguage("scala",function(hljs){var ANNOTATION={className:"meta",begin:"@[A-Za-z]+"};var SUBST={className:"subst",variants:[{begin:"\\$[A-Za-z0-9_]+"},{begin:"\\${",end:"}"}]};var STRING={className:"string",variants:[{begin:'"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]},{begin:'"""',end:'"""',relevance:10},{begin:'[a-z]+"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST]},{className:"string",begin:'[a-z]+"""',end:'"""',contains:[SUBST],relevance:10}]};var SYMBOL={className:"symbol",begin:"'\\w[\\w\\d_]*(?!')"};var TYPE={className:"type",begin:"\\b[A-Z][A-Za-z0-9_]*",relevance:0};var NAME={className:"title",begin:/[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/,relevance:0};var CLASS={className:"class",beginKeywords:"class object trait type",end:/[:={\[\n;]/,excludeEnd:true,contains:[{beginKeywords:"extends with",relevance:10},{begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[TYPE]},{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[TYPE]},NAME]};var METHOD={className:"function",beginKeywords:"def",end:/[:={\[(\n;]/,excludeEnd:true,contains:[NAME]};return{keywords:{literal:"true false null",keyword:"type yield lazy override def with val var sealed abstract private trait object if forSome for while throw finally protected extends import final return else break new catch super class case package default try this match continue throws implicit"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,STRING,SYMBOL,TYPE,METHOD,CLASS,hljs.C_NUMBER_MODE,ANNOTATION]}});hljs.registerLanguage("django",function(hljs){var FILTER={begin:/\|[A-Za-z]+:?/,keywords:{name:"truncatewords removetags linebreaksbr yesno get_digit timesince random striptags "+"filesizeformat escape linebreaks length_is ljust rjust cut urlize fix_ampersands "+"title floatformat capfirst pprint divisibleby add make_list unordered_list urlencode "+"timeuntil urlizetrunc wordcount stringformat linenumbers slice date dictsort "+"dictsortreversed default_if_none pluralize lower join center default "+"truncatewords_html upper length phone2numeric wordwrap time addslashes slugify first "+"escapejs force_escape iriencode last safe safeseq truncatechars localize unlocalize "+"localtime utc timezone"},contains:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE]};return{aliases:["jinja"],case_insensitive:true,subLanguage:"xml",contains:[hljs.COMMENT(/\{%\s*comment\s*%}/,/\{%\s*endcomment\s*%}/),hljs.COMMENT(/\{#/,/#}/),{className:"template-tag",begin:/\{%/,end:/%}/,contains:[{className:"name",begin:/\w+/,keywords:{name:"comment endcomment load templatetag ifchanged endifchanged if endif firstof for "+"endfor ifnotequal endifnotequal widthratio extends include spaceless "+"endspaceless regroup ifequal endifequal ssi now with cycle url filter "+"endfilter debug block endblock else autoescape endautoescape csrf_token empty elif "+"endwith static trans blocktrans endblocktrans get_static_prefix get_media_prefix "+"plural get_current_language language get_available_languages "+"get_current_language_bidi get_language_info get_language_info_list localize "+"endlocalize localtime endlocaltime timezone endtimezone get_current_timezone "+"verbatim"},starts:{endsWithParent:true,keywords:"in by as",contains:[FILTER],relevance:0}}]},{className:"template-variable",begin:/\{\{/,end:/}}/,contains:[FILTER]}]}});hljs.registerLanguage("shell",function(hljs){return{aliases:["console"],contains:[{className:"meta",begin:"^\\s{0,3}[/\\w\\d\\[\\]()@-]*[>%$#]",starts:{end:"$",subLanguage:"bash"}}]}});hljs.registerLanguage("powershell",function(hljs){var TYPES=["string","char","byte","int","long","bool","decimal","single","double","DateTime","xml","array","hashtable","void"];var VALID_VERBS="Add|Clear|Close|Copy|Enter|Exit|Find|Format|Get|Hide|Join|Lock|"+"Move|New|Open|Optimize|Pop|Push|Redo|Remove|Rename|Reset|Resize|"+"Search|Select|Set|Show|Skip|Split|Step|Switch|Undo|Unlock|"+"Watch|Backup|Checkpoint|Compare|Compress|Convert|ConvertFrom|"+"ConvertTo|Dismount|Edit|Expand|Export|Group|Import|Initialize|"+"Limit|Merge|New|Out|Publish|Restore|Save|Sync|Unpublish|Update|"+"Approve|Assert|Complete|Confirm|Deny|Disable|Enable|Install|Invoke|Register|"+"Request|Restart|Resume|Start|Stop|Submit|Suspend|Uninstall|"+"Unregister|Wait|Debug|Measure|Ping|Repair|Resolve|Test|Trace|Connect|"+"Disconnect|Read|Receive|Send|Write|Block|Grant|Protect|Revoke|Unblock|"+"Unprotect|Use|ForEach|Sort|Tee|Where";var COMPARISON_OPERATORS="-and|-as|-band|-bnot|-bor|-bxor|-casesensitive|-ccontains|-ceq|-cge|-cgt|"+"-cle|-clike|-clt|-cmatch|-cne|-cnotcontains|-cnotlike|-cnotmatch|-contains|"+"-creplace|-csplit|-eq|-exact|-f|-file|-ge|-gt|-icontains|-ieq|-ige|-igt|"+"-ile|-ilike|-ilt|-imatch|-in|-ine|-inotcontains|-inotlike|-inotmatch|"+"-ireplace|-is|-isnot|-isplit|-join|-le|-like|-lt|-match|-ne|-not|"+"-notcontains|-notin|-notlike|-notmatch|-or|-regex|-replace|-shl|-shr|"+"-split|-wildcard|-xor";var KEYWORDS={keyword:"if else foreach return do while until elseif begin for trap data dynamicparam "+"end break throw param continue finally in switch exit filter try process catch "+"hidden static parameter"};var TITLE_NAME_RE=/\w[\w\d]*((-)[\w\d]+)*/;var BACKTICK_ESCAPE={begin:"`[\\s\\S]",relevance:0};var VAR={className:"variable",variants:[{begin:/\$\B/},{className:"keyword",begin:/\$this/},{begin:/\$[\w\d][\w\d_:]*/}]};var LITERAL={className:"literal",begin:/\$(null|true|false)\b/};var QUOTE_STRING={className:"string",variants:[{begin:/"/,end:/"/},{begin:/@"/,end:/^"@/}],contains:[BACKTICK_ESCAPE,VAR,{className:"variable",begin:/\$[A-z]/,end:/[^A-z]/}]};var APOS_STRING={className:"string",variants:[{begin:/'/,end:/'/},{begin:/@'/,end:/^'@/}]};var PS_HELPTAGS={className:"doctag",variants:[{begin:/\.(synopsis|description|example|inputs|outputs|notes|link|component|role|functionality)/},{begin:/\.(parameter|forwardhelptargetname|forwardhelpcategory|remotehelprunspace|externalhelp)\s+\S+/}]};var PS_COMMENT=hljs.inherit(hljs.COMMENT(null,null),{variants:[{begin:/#/,end:/$/},{begin:/<#/,end:/#>/}],contains:[PS_HELPTAGS]});var CMDLETS={className:"built_in",variants:[{begin:"(".concat(VALID_VERBS,")+(-)[\\w\\d]+")}]};var PS_CLASS={className:"class",beginKeywords:"class enum",end:/\s*[{]/,excludeEnd:true,relevance:0,contains:[hljs.TITLE_MODE]};var PS_FUNCTION={className:"function",begin:/function\s+/,end:/\s*\{|$/,excludeEnd:true,returnBegin:true,relevance:0,contains:[{begin:"function",relevance:0,className:"keyword"},{className:"title",begin:TITLE_NAME_RE,relevance:0},{begin:/\(/,end:/\)/,className:"params",relevance:0,contains:[VAR]}]};var PS_USING={begin:/using\s/,end:/$/,returnBegin:true,contains:[QUOTE_STRING,APOS_STRING,{className:"keyword",begin:/(using|assembly|command|module|namespace|type)/}]};var PS_ARGUMENTS={variants:[{className:"operator",begin:"(".concat(COMPARISON_OPERATORS,")\\b")},{className:"literal",begin:/(-)[\w\d]+/,relevance:0}]};var STATIC_MEMBER={className:"selector-tag",begin:/::\w+\b/,end:/$/,returnBegin:true,contains:[{className:"attribute",begin:/\w+/,endsParent:true}]};var HASH_SIGNS={className:"selector-tag",begin:/\@\B/,relevance:0};var PS_NEW_OBJECT_TYPE={className:"built_in",begin:/New-Object\s+\w/,end:/$/,returnBegin:true,contains:[{begin:/New-Object\s+/,relevance:0},{className:"meta",begin:/([\w\.])+/,endsParent:true}]};var PS_METHODS={className:"function",begin:/\[.*\]\s*[\w]+[ ]??\(/,end:/$/,returnBegin:true,relevance:0,contains:[{className:"keyword",begin:"(".concat(KEYWORDS.keyword.toString().replace(/\s/g,"|"),")\\b"),endsParent:true,relevance:0},hljs.inherit(hljs.TITLE_MODE,{endsParent:true})]};var GENTLEMANS_SET=[PS_METHODS,PS_COMMENT,BACKTICK_ESCAPE,hljs.NUMBER_MODE,QUOTE_STRING,APOS_STRING,CMDLETS,VAR,LITERAL,HASH_SIGNS];var PS_TYPE={begin:/\[/,end:/\]/,excludeBegin:true,excludeEnd:true,relevance:0,contains:[].concat("self",GENTLEMANS_SET,{begin:"("+TYPES.join("|")+")",className:"built_in",relevance:0},{className:"type",begin:/[\.\w\d]+/,relevance:0})};PS_METHODS.contains.unshift(PS_TYPE);return{aliases:["ps","ps1"],lexemes:/-?[A-z\.\-]+/,case_insensitive:true,keywords:KEYWORDS,contains:GENTLEMANS_SET.concat(PS_CLASS,PS_FUNCTION,PS_USING,PS_ARGUMENTS,PS_TYPE)}});hljs.registerLanguage("swift",function(hljs){var SWIFT_KEYWORDS={keyword:"#available #colorLiteral #column #else #elseif #endif #file "+"#fileLiteral #function #if #imageLiteral #line #selector #sourceLocation "+"_ __COLUMN__ __FILE__ __FUNCTION__ __LINE__ Any as as! as? associatedtype "+"associativity break case catch class continue convenience default defer deinit didSet do "+"dynamic dynamicType else enum extension fallthrough false fileprivate final for func "+"get guard if import in indirect infix init inout internal is lazy left let "+"mutating nil none nonmutating open operator optional override postfix precedence "+"prefix private protocol Protocol public repeat required rethrows return "+"right self Self set static struct subscript super switch throw throws true "+"try try! try? Type typealias unowned var weak where while willSet",literal:"true false nil",built_in:"abs advance alignof alignofValue anyGenerator assert assertionFailure "+"bridgeFromObjectiveC bridgeFromObjectiveCUnconditional bridgeToObjectiveC "+"bridgeToObjectiveCUnconditional c contains count countElements countLeadingZeros "+"debugPrint debugPrintln distance dropFirst dropLast dump encodeBitsAsWords "+"enumerate equal fatalError filter find getBridgedObjectiveCType getVaList "+"indices insertionSort isBridgedToObjectiveC isBridgedVerbatimToObjectiveC "+"isUniquelyReferenced isUniquelyReferencedNonObjC join lazy lexicographicalCompare "+"map max maxElement min minElement numericCast overlaps partition posix "+"precondition preconditionFailure print println quickSort readLine reduce reflect "+"reinterpretCast reverse roundUpToAlignment sizeof sizeofValue sort split "+"startsWith stride strideof strideofValue swap toString transcode "+"underestimateCount unsafeAddressOf unsafeBitCast unsafeDowncast unsafeUnwrap "+"unsafeReflect withExtendedLifetime withObjectAtPlusZero withUnsafePointer "+"withUnsafePointerToObject withUnsafeMutablePointer withUnsafeMutablePointers "+"withUnsafePointer withUnsafePointers withVaList zip"};var TYPE={className:"type",begin:"\\b[A-Z][\\wÀ-ʸ']*",relevance:0};var OPTIONAL_USING_TYPE={className:"type",begin:"\\b[A-Z][\\wÀ-ʸ']*[!?]"};var BLOCK_COMMENT=hljs.COMMENT("/\\*","\\*/",{contains:["self"]});var SUBST={className:"subst",begin:/\\\(/,end:"\\)",keywords:SWIFT_KEYWORDS,contains:[]};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"""/,end:/"""/},{begin:/"/,end:/"/}]};var NUMBERS={className:"number",begin:"\\b([\\d_]+(\\.[\\deE_]+)?|0x[a-fA-F0-9_]+(\\.[a-fA-F0-9p_]+)?|0b[01_]+|0o[0-7_]+)\\b",relevance:0};SUBST.contains=[NUMBERS];return{keywords:SWIFT_KEYWORDS,contains:[STRING,hljs.C_LINE_COMMENT_MODE,BLOCK_COMMENT,OPTIONAL_USING_TYPE,TYPE,NUMBERS,{className:"function",beginKeywords:"func",end:"{",excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][0-9A-Za-z$_]*/}),{begin:/</,end:/>/},{className:"params",begin:/\(/,end:/\)/,endsParent:true,keywords:SWIFT_KEYWORDS,contains:["self",NUMBERS,STRING,hljs.C_BLOCK_COMMENT_MODE,{begin:":"}],illegal:/["']/}],illegal:/\[|%/},{className:"class",beginKeywords:"struct protocol class extension enum",keywords:SWIFT_KEYWORDS,end:"\\{",excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/})]},{className:"meta",begin:"(@discardableResult|@warn_unused_result|@exported|@lazy|@noescape|"+"@NSCopying|@NSManaged|@objc|@objcMembers|@convention|@required|"+"@noreturn|@IBAction|@IBDesignable|@IBInspectable|@IBOutlet|"+"@infix|@prefix|@postfix|@autoclosure|@testable|@available|"+"@nonobjc|@NSApplicationMain|@UIApplicationMain|@dynamicMemberLookup|"+"@propertyWrapper)"},{beginKeywords:"import",end:/$/,contains:[hljs.C_LINE_COMMENT_MODE,BLOCK_COMMENT]}]}});hljs.registerLanguage("sas",function(hljs){var SAS_KEYWORDS=""+"do if then else end until while "+""+"abort array attrib by call cards cards4 catname continue "+"datalines datalines4 delete delim delimiter display dm drop "+"endsas error file filename footnote format goto in infile "+"informat input keep label leave length libname link list "+"lostcard merge missing modify options output out page put "+"redirect remove rename replace retain return select set skip "+"startsas stop title update waitsas where window x systask "+""+"add and alter as cascade check create delete describe "+"distinct drop foreign from group having index insert into in "+"key like message modify msgtype not null on or order primary "+"references reset restrict select set table unique update "+"validate view where";var SAS_FUN=""+"abs|addr|airy|arcos|arsin|atan|attrc|attrn|band|"+"betainv|blshift|bnot|bor|brshift|bxor|byte|cdf|ceil|"+"cexist|cinv|close|cnonct|collate|compbl|compound|"+"compress|cos|cosh|css|curobs|cv|daccdb|daccdbsl|"+"daccsl|daccsyd|dacctab|dairy|date|datejul|datepart|"+"datetime|day|dclose|depdb|depdbsl|depdbsl|depsl|"+"depsl|depsyd|depsyd|deptab|deptab|dequote|dhms|dif|"+"digamma|dim|dinfo|dnum|dopen|doptname|doptnum|dread|"+"dropnote|dsname|erf|erfc|exist|exp|fappend|fclose|"+"fcol|fdelete|fetch|fetchobs|fexist|fget|fileexist|"+"filename|fileref|finfo|finv|fipname|fipnamel|"+"fipstate|floor|fnonct|fnote|fopen|foptname|foptnum|"+"fpoint|fpos|fput|fread|frewind|frlen|fsep|fuzz|"+"fwrite|gaminv|gamma|getoption|getvarc|getvarn|hbound|"+"hms|hosthelp|hour|ibessel|index|indexc|indexw|input|"+"inputc|inputn|int|intck|intnx|intrr|irr|jbessel|"+"juldate|kurtosis|lag|lbound|left|length|lgamma|"+"libname|libref|log|log10|log2|logpdf|logpmf|logsdf|"+"lowcase|max|mdy|mean|min|minute|mod|month|mopen|"+"mort|n|netpv|nmiss|normal|note|npv|open|ordinal|"+"pathname|pdf|peek|peekc|pmf|point|poisson|poke|"+"probbeta|probbnml|probchi|probf|probgam|probhypr|"+"probit|probnegb|probnorm|probt|put|putc|putn|qtr|"+"quote|ranbin|rancau|ranexp|rangam|range|rank|rannor|"+"ranpoi|rantbl|rantri|ranuni|repeat|resolve|reverse|"+"rewind|right|round|saving|scan|sdf|second|sign|"+"sin|sinh|skewness|soundex|spedis|sqrt|std|stderr|"+"stfips|stname|stnamel|substr|sum|symget|sysget|"+"sysmsg|sysprod|sysrc|system|tan|tanh|time|timepart|"+"tinv|tnonct|today|translate|tranwrd|trigamma|"+"trim|trimn|trunc|uniform|upcase|uss|var|varfmt|"+"varinfmt|varlabel|varlen|varname|varnum|varray|"+"varrayx|vartype|verify|vformat|vformatd|vformatdx|"+"vformatn|vformatnx|vformatw|vformatwx|vformatx|"+"vinarray|vinarrayx|vinformat|vinformatd|vinformatdx|"+"vinformatn|vinformatnx|vinformatw|vinformatwx|"+"vinformatx|vlabel|vlabelx|vlength|vlengthx|vname|"+"vnamex|vtype|vtypex|weekday|year|yyq|zipfips|zipname|"+"zipnamel|zipstate";var SAS_MACRO_FUN="bquote|nrbquote|cmpres|qcmpres|compstor|"+"datatyp|display|do|else|end|eval|global|goto|"+"if|index|input|keydef|label|left|length|let|"+"local|lowcase|macro|mend|nrbquote|nrquote|"+"nrstr|put|qcmpres|qleft|qlowcase|qscan|"+"qsubstr|qsysfunc|qtrim|quote|qupcase|scan|str|"+"substr|superq|syscall|sysevalf|sysexec|sysfunc|"+"sysget|syslput|sysprod|sysrc|sysrput|then|to|"+"trim|unquote|until|upcase|verify|while|window";return{aliases:["sas","SAS"],case_insensitive:true,keywords:{literal:"null missing _all_ _automatic_ _character_ _infile_ "+"_n_ _name_ _null_ _numeric_ _user_ _webout_",meta:SAS_KEYWORDS},contains:[{className:"keyword",begin:/^\s*(proc [\w\d_]+|data|run|quit)[\s\;]/},{className:"variable",begin:/\&[a-zA-Z_\&][a-zA-Z0-9_]*\.?/},{className:"emphasis",begin:/^\s*datalines|cards.*;/,end:/^\s*;\s*$/},{className:"built_in",begin:"%("+SAS_MACRO_FUN+")"},{className:"name",begin:/%[a-zA-Z_][a-zA-Z_0-9]*/},{className:"meta",begin:"[^%]("+SAS_FUN+")[(]"},{className:"string",variants:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE]},hljs.COMMENT("\\*",";"),hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("gradle",function(hljs){return{case_insensitive:true,keywords:{keyword:"task project allprojects subprojects artifacts buildscript configurations "+"dependencies repositories sourceSets description delete from into include "+"exclude source classpath destinationDir includes options sourceCompatibility "+"targetCompatibility group flatDir doLast doFirst flatten todir fromdir ant "+"def abstract break case catch continue default do else extends final finally "+"for if implements instanceof native new private protected public return static "+"switch synchronized throw throws transient try volatile while strictfp package "+"import false null super this true antlrtask checkstyle codenarc copy boolean "+"byte char class double float int interface long short void compile runTime "+"file fileTree abs any append asList asWritable call collect compareTo count "+"div dump each eachByte eachFile eachLine every find findAll flatten getAt "+"getErr getIn getOut getText grep immutable inject inspect intersect invokeMethods "+"isCase join leftShift minus multiply newInputStream newOutputStream newPrintWriter "+"newReader newWriter next plus pop power previous print println push putAt read "+"readBytes readLines reverse reverseEach round size sort splitEachLine step subMap "+"times toInteger toList tokenize upto waitForOrKill withPrintWriter withReader "+"withStream withWriter withWriterAppend write writeLine"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.NUMBER_MODE,hljs.REGEXP_MODE]}});hljs.registerLanguage("stylus",function(hljs){var VARIABLE={className:"variable",begin:"\\$"+hljs.IDENT_RE};var HEX_COLOR={className:"number",begin:"#([a-fA-F0-9]{6}|[a-fA-F0-9]{3})"};var AT_KEYWORDS=["charset","css","debug","extend","font-face","for","import","include","media","mixin","page","warn","while"];var PSEUDO_SELECTORS=["after","before","first-letter","first-line","active","first-child","focus","hover","lang","link","visited"];var TAGS=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"];var LOOKAHEAD_TAG_END="(?=[\\.\\s\\n\\[\\:,])";var ATTRIBUTES=["align-content","align-items","align-self","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","auto","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","clip-path","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","font","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-stretch","font-style","font-variant","font-variant-ligatures","font-weight","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inherit","initial","justify-content","left","letter-spacing","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marks","mask","max-height","max-width","min-height","min-width","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","perspective","perspective-origin","pointer-events","position","quotes","resize","right","tab-size","table-layout","text-align","text-align-last","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-indent","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","white-space","widows","width","word-break","word-spacing","word-wrap","z-index"];var ILLEGAL=["\\?","(\\bReturn\\b)","(\\bEnd\\b)","(\\bend\\b)","(\\bdef\\b)",";","#\\s","\\*\\s","===\\s","\\|","%"];return{aliases:["styl"],case_insensitive:false,keywords:"if else for in",illegal:"("+ILLEGAL.join("|")+")",contains:[hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,HEX_COLOR,{begin:"\\.[a-zA-Z][a-zA-Z0-9_-]*"+LOOKAHEAD_TAG_END,className:"selector-class"},{begin:"\\#[a-zA-Z][a-zA-Z0-9_-]*"+LOOKAHEAD_TAG_END,className:"selector-id"},{begin:"\\b("+TAGS.join("|")+")"+LOOKAHEAD_TAG_END,className:"selector-tag"},{begin:"&?:?:\\b("+PSEUDO_SELECTORS.join("|")+")"+LOOKAHEAD_TAG_END},{begin:"@("+AT_KEYWORDS.join("|")+")\\b"},VARIABLE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,{className:"function",begin:"^[a-zA-Z][a-zA-Z0-9_-]*\\(.*\\)",illegal:"[\\n]",returnBegin:true,contains:[{className:"title",begin:"\\b[a-zA-Z][a-zA-Z0-9_-]*"},{className:"params",begin:/\(/,end:/\)/,contains:[HEX_COLOR,VARIABLE,hljs.APOS_STRING_MODE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,hljs.QUOTE_STRING_MODE]}]},{className:"attribute",begin:"\\b("+ATTRIBUTES.reverse().join("|")+")\\b",starts:{end:/;|$/,contains:[HEX_COLOR,VARIABLE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.CSS_NUMBER_MODE,hljs.NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE],illegal:/\./,relevance:0}}]}});hljs.registerLanguage("profile",function(hljs){return{contains:[hljs.C_NUMBER_MODE,{begin:"[a-zA-Z_][\\da-zA-Z_]+\\.[\\da-zA-Z_]{1,3}",end:":",excludeEnd:true},{begin:"(ncalls|tottime|cumtime)",end:"$",keywords:"ncalls tottime|10 cumtime|10 filename",relevance:10},{begin:"function calls",end:"$",contains:[hljs.C_NUMBER_MODE],relevance:10},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"\\(",end:"\\)$",excludeBegin:true,excludeEnd:true,relevance:0}]}});hljs.registerLanguage("sqf",function(hljs){var VARIABLE={className:"variable",begin:/\b_+[a-zA-Z_]\w*/};var FUNCTION={className:"title",begin:/[a-zA-Z][a-zA-Z0-9]+_fnc_\w*/};var STRINGS={className:"string",variants:[{begin:'"',end:'"',contains:[{begin:'""',relevance:0}]},{begin:"'",end:"'",contains:[{begin:"''",relevance:0}]}]};var PREPROCESSOR={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"define undef ifdef ifndef else endif include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(STRINGS,{className:"meta-string"}),{className:"meta-string",begin:/<[^\n>]*>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]};return{aliases:["sqf"],case_insensitive:true,keywords:{keyword:"case catch default do else exit exitWith for forEach from if "+"private switch then throw to try waitUntil while with",built_in:"abs accTime acos action actionIDs actionKeys actionKeysImages actionKeysNames "+"actionKeysNamesArray actionName actionParams activateAddons activatedAddons activateKey "+"add3DENConnection add3DENEventHandler add3DENLayer addAction addBackpack addBackpackCargo "+"addBackpackCargoGlobal addBackpackGlobal addCamShake addCuratorAddons addCuratorCameraArea "+"addCuratorEditableObjects addCuratorEditingArea addCuratorPoints addEditorObject addEventHandler "+"addForce addGoggles addGroupIcon addHandgunItem addHeadgear addItem addItemCargo "+"addItemCargoGlobal addItemPool addItemToBackpack addItemToUniform addItemToVest addLiveStats "+"addMagazine addMagazineAmmoCargo addMagazineCargo addMagazineCargoGlobal addMagazineGlobal "+"addMagazinePool addMagazines addMagazineTurret addMenu addMenuItem addMissionEventHandler "+"addMPEventHandler addMusicEventHandler addOwnedMine addPlayerScores addPrimaryWeaponItem "+"addPublicVariableEventHandler addRating addResources addScore addScoreSide addSecondaryWeaponItem "+"addSwitchableUnit addTeamMember addToRemainsCollector addTorque addUniform addVehicle addVest "+"addWaypoint addWeapon addWeaponCargo addWeaponCargoGlobal addWeaponGlobal addWeaponItem "+"addWeaponPool addWeaponTurret admin agent agents AGLToASL aimedAtTarget aimPos airDensityRTD "+"airplaneThrottle airportSide AISFinishHeal alive all3DENEntities allAirports allControls "+"allCurators allCutLayers allDead allDeadMen allDisplays allGroups allMapMarkers allMines "+"allMissionObjects allow3DMode allowCrewInImmobile allowCuratorLogicIgnoreAreas allowDamage "+"allowDammage allowFileOperations allowFleeing allowGetIn allowSprint allPlayers allSimpleObjects "+"allSites allTurrets allUnits allUnitsUAV allVariables ammo ammoOnPylon and animate animateBay "+"animateDoor animatePylon animateSource animationNames animationPhase animationSourcePhase "+"animationState append apply armoryPoints arrayIntersect asin ASLToAGL ASLToATL assert "+"assignAsCargo assignAsCargoIndex assignAsCommander assignAsDriver assignAsGunner assignAsTurret "+"assignCurator assignedCargo assignedCommander assignedDriver assignedGunner assignedItems "+"assignedTarget assignedTeam assignedVehicle assignedVehicleRole assignItem assignTeam "+"assignToAirport atan atan2 atg ATLToASL attachedObject attachedObjects attachedTo attachObject "+"attachTo attackEnabled backpack backpackCargo backpackContainer backpackItems backpackMagazines "+"backpackSpaceFor behaviour benchmark binocular boundingBox boundingBoxReal boundingCenter "+"breakOut breakTo briefingName buildingExit buildingPos buttonAction buttonSetAction cadetMode "+"call callExtension camCommand camCommit camCommitPrepared camCommitted camConstuctionSetParams "+"camCreate camDestroy cameraEffect cameraEffectEnableHUD cameraInterest cameraOn cameraView "+"campaignConfigFile camPreload camPreloaded camPrepareBank camPrepareDir camPrepareDive "+"camPrepareFocus camPrepareFov camPrepareFovRange camPreparePos camPrepareRelPos camPrepareTarget "+"camSetBank camSetDir camSetDive camSetFocus camSetFov camSetFovRange camSetPos camSetRelPos "+"camSetTarget camTarget camUseNVG canAdd canAddItemToBackpack canAddItemToUniform canAddItemToVest "+"cancelSimpleTaskDestination canFire canMove canSlingLoad canStand canSuspend "+"canTriggerDynamicSimulation canUnloadInCombat canVehicleCargo captive captiveNum cbChecked "+"cbSetChecked ceil channelEnabled cheatsEnabled checkAIFeature checkVisibility className "+"clearAllItemsFromBackpack clearBackpackCargo clearBackpackCargoGlobal clearGroupIcons "+"clearItemCargo clearItemCargoGlobal clearItemPool clearMagazineCargo clearMagazineCargoGlobal "+"clearMagazinePool clearOverlay clearRadio clearWeaponCargo clearWeaponCargoGlobal clearWeaponPool "+"clientOwner closeDialog closeDisplay closeOverlay collapseObjectTree collect3DENHistory "+"collectiveRTD combatMode commandArtilleryFire commandChat commander commandFire commandFollow "+"commandFSM commandGetOut commandingMenu commandMove commandRadio commandStop "+"commandSuppressiveFire commandTarget commandWatch comment commitOverlay compile compileFinal "+"completedFSM composeText configClasses configFile configHierarchy configName configProperties "+"configSourceAddonList configSourceMod configSourceModList confirmSensorTarget "+"connectTerminalToUAV controlsGroupCtrl copyFromClipboard copyToClipboard copyWaypoints cos count "+"countEnemy countFriendly countSide countType countUnknown create3DENComposition create3DENEntity "+"createAgent createCenter createDialog createDiaryLink createDiaryRecord createDiarySubject "+"createDisplay createGearDialog createGroup createGuardedPoint createLocation createMarker "+"createMarkerLocal createMenu createMine createMissionDisplay createMPCampaignDisplay "+"createSimpleObject createSimpleTask createSite createSoundSource createTask createTeam "+"createTrigger createUnit createVehicle createVehicleCrew createVehicleLocal crew ctAddHeader "+"ctAddRow ctClear ctCurSel ctData ctFindHeaderRows ctFindRowHeader ctHeaderControls ctHeaderCount "+"ctRemoveHeaders ctRemoveRows ctrlActivate ctrlAddEventHandler ctrlAngle ctrlAutoScrollDelay "+"ctrlAutoScrollRewind ctrlAutoScrollSpeed ctrlChecked ctrlClassName ctrlCommit ctrlCommitted "+"ctrlCreate ctrlDelete ctrlEnable ctrlEnabled ctrlFade ctrlHTMLLoaded ctrlIDC ctrlIDD "+"ctrlMapAnimAdd ctrlMapAnimClear ctrlMapAnimCommit ctrlMapAnimDone ctrlMapCursor ctrlMapMouseOver "+"ctrlMapScale ctrlMapScreenToWorld ctrlMapWorldToScreen ctrlModel ctrlModelDirAndUp ctrlModelScale "+"ctrlParent ctrlParentControlsGroup ctrlPosition ctrlRemoveAllEventHandlers ctrlRemoveEventHandler "+"ctrlScale ctrlSetActiveColor ctrlSetAngle ctrlSetAutoScrollDelay ctrlSetAutoScrollRewind "+"ctrlSetAutoScrollSpeed ctrlSetBackgroundColor ctrlSetChecked ctrlSetEventHandler ctrlSetFade "+"ctrlSetFocus ctrlSetFont ctrlSetFontH1 ctrlSetFontH1B ctrlSetFontH2 ctrlSetFontH2B ctrlSetFontH3 "+"ctrlSetFontH3B ctrlSetFontH4 ctrlSetFontH4B ctrlSetFontH5 ctrlSetFontH5B ctrlSetFontH6 "+"ctrlSetFontH6B ctrlSetFontHeight ctrlSetFontHeightH1 ctrlSetFontHeightH2 ctrlSetFontHeightH3 "+"ctrlSetFontHeightH4 ctrlSetFontHeightH5 ctrlSetFontHeightH6 ctrlSetFontHeightSecondary "+"ctrlSetFontP ctrlSetFontPB ctrlSetFontSecondary ctrlSetForegroundColor ctrlSetModel "+"ctrlSetModelDirAndUp ctrlSetModelScale ctrlSetPixelPrecision ctrlSetPosition ctrlSetScale "+"ctrlSetStructuredText ctrlSetText ctrlSetTextColor ctrlSetTooltip ctrlSetTooltipColorBox "+"ctrlSetTooltipColorShade ctrlSetTooltipColorText ctrlShow ctrlShown ctrlText ctrlTextHeight "+"ctrlTextWidth ctrlType ctrlVisible ctRowControls ctRowCount ctSetCurSel ctSetData "+"ctSetHeaderTemplate ctSetRowTemplate ctSetValue ctValue curatorAddons curatorCamera "+"curatorCameraArea curatorCameraAreaCeiling curatorCoef curatorEditableObjects curatorEditingArea "+"curatorEditingAreaType curatorMouseOver curatorPoints curatorRegisteredObjects curatorSelected "+"curatorWaypointCost current3DENOperation currentChannel currentCommand currentMagazine "+"currentMagazineDetail currentMagazineDetailTurret currentMagazineTurret currentMuzzle "+"currentNamespace currentTask currentTasks currentThrowable currentVisionMode currentWaypoint "+"currentWeapon currentWeaponMode currentWeaponTurret currentZeroing cursorObject cursorTarget "+"customChat customRadio cutFadeOut cutObj cutRsc cutText damage date dateToNumber daytime "+"deActivateKey debriefingText debugFSM debugLog deg delete3DENEntities deleteAt deleteCenter "+"deleteCollection deleteEditorObject deleteGroup deleteGroupWhenEmpty deleteIdentity "+"deleteLocation deleteMarker deleteMarkerLocal deleteRange deleteResources deleteSite deleteStatus "+"deleteTeam deleteVehicle deleteVehicleCrew deleteWaypoint detach detectedMines "+"diag_activeMissionFSMs diag_activeScripts diag_activeSQFScripts diag_activeSQSScripts "+"diag_captureFrame diag_captureFrameToFile diag_captureSlowFrame diag_codePerformance "+"diag_drawMode diag_enable diag_enabled diag_fps diag_fpsMin diag_frameNo diag_lightNewLoad "+"diag_list diag_log diag_logSlowFrame diag_mergeConfigFile diag_recordTurretLimits "+"diag_setLightNew diag_tickTime diag_toggle dialog diarySubjectExists didJIP didJIPOwner "+"difficulty difficultyEnabled difficultyEnabledRTD difficultyOption direction directSay disableAI "+"disableCollisionWith disableConversation disableDebriefingStats disableMapIndicators "+"disableNVGEquipment disableRemoteSensors disableSerialization disableTIEquipment "+"disableUAVConnectability disableUserInput displayAddEventHandler displayCtrl displayParent "+"displayRemoveAllEventHandlers displayRemoveEventHandler displaySetEventHandler dissolveTeam "+"distance distance2D distanceSqr distributionRegion do3DENAction doArtilleryFire doFire doFollow "+"doFSM doGetOut doMove doorPhase doStop doSuppressiveFire doTarget doWatch drawArrow drawEllipse "+"drawIcon drawIcon3D drawLine drawLine3D drawLink drawLocation drawPolygon drawRectangle "+"drawTriangle driver drop dynamicSimulationDistance dynamicSimulationDistanceCoef "+"dynamicSimulationEnabled dynamicSimulationSystemEnabled echo edit3DENMissionAttributes editObject "+"editorSetEventHandler effectiveCommander emptyPositions enableAI enableAIFeature "+"enableAimPrecision enableAttack enableAudioFeature enableAutoStartUpRTD enableAutoTrimRTD "+"enableCamShake enableCaustics enableChannel enableCollisionWith enableCopilot "+"enableDebriefingStats enableDiagLegend enableDynamicSimulation enableDynamicSimulationSystem "+"enableEndDialog enableEngineArtillery enableEnvironment enableFatigue enableGunLights "+"enableInfoPanelComponent enableIRLasers enableMimics enablePersonTurret enableRadio enableReload "+"enableRopeAttach enableSatNormalOnDetail enableSaving enableSentences enableSimulation "+"enableSimulationGlobal enableStamina enableTeamSwitch enableTraffic enableUAVConnectability "+"enableUAVWaypoints enableVehicleCargo enableVehicleSensor enableWeaponDisassembly "+"endLoadingScreen endMission engineOn enginesIsOnRTD enginesRpmRTD enginesTorqueRTD entities "+"environmentEnabled estimatedEndServerTime estimatedTimeLeft evalObjectArgument everyBackpack "+"everyContainer exec execEditorScript execFSM execVM exp expectedDestination exportJIPMessages "+"eyeDirection eyePos face faction fadeMusic fadeRadio fadeSound fadeSpeech failMission "+"fillWeaponsFromPool find findCover findDisplay findEditorObject findEmptyPosition "+"findEmptyPositionReady findIf findNearestEnemy finishMissionInit finite fire fireAtTarget "+"firstBackpack flag flagAnimationPhase flagOwner flagSide flagTexture fleeing floor flyInHeight "+"flyInHeightASL fog fogForecast fogParams forceAddUniform forcedMap forceEnd forceFlagTexture "+"forceFollowRoad forceMap forceRespawn forceSpeed forceWalk forceWeaponFire forceWeatherChange "+"forEachMember forEachMemberAgent forEachMemberTeam forgetTarget format formation "+"formationDirection formationLeader formationMembers formationPosition formationTask formatText "+"formLeader freeLook fromEditor fuel fullCrew gearIDCAmmoCount gearSlotAmmoCount gearSlotData "+"get3DENActionState get3DENAttribute get3DENCamera get3DENConnections get3DENEntity "+"get3DENEntityID get3DENGrid get3DENIconsVisible get3DENLayerEntities get3DENLinesVisible "+"get3DENMissionAttribute get3DENMouseOver get3DENSelected getAimingCoef getAllEnvSoundControllers "+"getAllHitPointsDamage getAllOwnedMines getAllSoundControllers getAmmoCargo getAnimAimPrecision "+"getAnimSpeedCoef getArray getArtilleryAmmo getArtilleryComputerSettings getArtilleryETA "+"getAssignedCuratorLogic getAssignedCuratorUnit getBackpackCargo getBleedingRemaining "+"getBurningValue getCameraViewDirection getCargoIndex getCenterOfMass getClientState "+"getClientStateNumber getCompatiblePylonMagazines getConnectedUAV getContainerMaxLoad "+"getCursorObjectParams getCustomAimCoef getDammage getDescription getDir getDirVisual "+"getDLCAssetsUsage getDLCAssetsUsageByName getDLCs getEditorCamera getEditorMode "+"getEditorObjectScope getElevationOffset getEnvSoundController getFatigue getForcedFlagTexture "+"getFriend getFSMVariable getFuelCargo getGroupIcon getGroupIconParams getGroupIcons getHideFrom "+"getHit getHitIndex getHitPointDamage getItemCargo getMagazineCargo getMarkerColor getMarkerPos "+"getMarkerSize getMarkerType getMass getMissionConfig getMissionConfigValue getMissionDLCs "+"getMissionLayerEntities getModelInfo getMousePosition getMusicPlayedTime getNumber "+"getObjectArgument getObjectChildren getObjectDLC getObjectMaterials getObjectProxy "+"getObjectTextures getObjectType getObjectViewDistance getOxygenRemaining getPersonUsedDLCs "+"getPilotCameraDirection getPilotCameraPosition getPilotCameraRotation getPilotCameraTarget "+"getPlateNumber getPlayerChannel getPlayerScores getPlayerUID getPos getPosASL getPosASLVisual "+"getPosASLW getPosATL getPosATLVisual getPosVisual getPosWorld getPylonMagazines getRelDir "+"getRelPos getRemoteSensorsDisabled getRepairCargo getResolution getShadowDistance getShotParents "+"getSlingLoad getSoundController getSoundControllerResult getSpeed getStamina getStatValue "+"getSuppression getTerrainGrid getTerrainHeightASL getText getTotalDLCUsageTime getUnitLoadout "+"getUnitTrait getUserMFDText getUserMFDvalue getVariable getVehicleCargo getWeaponCargo "+"getWeaponSway getWingsOrientationRTD getWingsPositionRTD getWPPos glanceAt globalChat globalRadio "+"goggles goto group groupChat groupFromNetId groupIconSelectable groupIconsVisible groupId "+"groupOwner groupRadio groupSelectedUnits groupSelectUnit gunner gusts halt handgunItems "+"handgunMagazine handgunWeapon handsHit hasInterface hasPilotCamera hasWeapon hcAllGroups "+"hcGroupParams hcLeader hcRemoveAllGroups hcRemoveGroup hcSelected hcSelectGroup hcSetGroup "+"hcShowBar hcShownBar headgear hideBody hideObject hideObjectGlobal hideSelection hint hintC "+"hintCadet hintSilent hmd hostMission htmlLoad HUDMovementLevels humidity image importAllGroups "+"importance in inArea inAreaArray incapacitatedState inflame inflamed infoPanel "+"infoPanelComponentEnabled infoPanelComponents infoPanels inGameUISetEventHandler inheritsFrom "+"initAmbientLife inPolygon inputAction inRangeOfArtillery insertEditorObject intersect is3DEN "+"is3DENMultiplayer isAbleToBreathe isAgent isArray isAutoHoverOn isAutonomous isAutotest "+"isBleeding isBurning isClass isCollisionLightOn isCopilotEnabled isDamageAllowed isDedicated "+"isDLCAvailable isEngineOn isEqualTo isEqualType isEqualTypeAll isEqualTypeAny isEqualTypeArray "+"isEqualTypeParams isFilePatchingEnabled isFlashlightOn isFlatEmpty isForcedWalk isFormationLeader "+"isGroupDeletedWhenEmpty isHidden isInRemainsCollector isInstructorFigureEnabled isIRLaserOn "+"isKeyActive isKindOf isLaserOn isLightOn isLocalized isManualFire isMarkedForCollection "+"isMultiplayer isMultiplayerSolo isNil isNull isNumber isObjectHidden isObjectRTD isOnRoad "+"isPipEnabled isPlayer isRealTime isRemoteExecuted isRemoteExecutedJIP isServer isShowing3DIcons "+"isSimpleObject isSprintAllowed isStaminaEnabled isSteamMission isStreamFriendlyUIEnabled isText "+"isTouchingGround isTurnedOut isTutHintsEnabled isUAVConnectable isUAVConnected isUIContext "+"isUniformAllowed isVehicleCargo isVehicleRadarOn isVehicleSensorEnabled isWalking "+"isWeaponDeployed isWeaponRested itemCargo items itemsWithMagazines join joinAs joinAsSilent "+"joinSilent joinString kbAddDatabase kbAddDatabaseTargets kbAddTopic kbHasTopic kbReact "+"kbRemoveTopic kbTell kbWasSaid keyImage keyName knowsAbout land landAt landResult language "+"laserTarget lbAdd lbClear lbColor lbColorRight lbCurSel lbData lbDelete lbIsSelected lbPicture "+"lbPictureRight lbSelection lbSetColor lbSetColorRight lbSetCurSel lbSetData lbSetPicture "+"lbSetPictureColor lbSetPictureColorDisabled lbSetPictureColorSelected lbSetPictureRight "+"lbSetPictureRightColor lbSetPictureRightColorDisabled lbSetPictureRightColorSelected "+"lbSetSelectColor lbSetSelectColorRight lbSetSelected lbSetText lbSetTextRight lbSetTooltip "+"lbSetValue lbSize lbSort lbSortByValue lbText lbTextRight lbValue leader leaderboardDeInit "+"leaderboardGetRows leaderboardInit leaderboardRequestRowsFriends leaderboardsRequestUploadScore "+"leaderboardsRequestUploadScoreKeepBest leaderboardState leaveVehicle libraryCredits "+"libraryDisclaimers lifeState lightAttachObject lightDetachObject lightIsOn lightnings limitSpeed "+"linearConversion lineIntersects lineIntersectsObjs lineIntersectsSurfaces lineIntersectsWith "+"linkItem list listObjects listRemoteTargets listVehicleSensors ln lnbAddArray lnbAddColumn "+"lnbAddRow lnbClear lnbColor lnbCurSelRow lnbData lnbDeleteColumn lnbDeleteRow "+"lnbGetColumnsPosition lnbPicture lnbSetColor lnbSetColumnsPos lnbSetCurSelRow lnbSetData "+"lnbSetPicture lnbSetText lnbSetValue lnbSize lnbSort lnbSortByValue lnbText lnbValue load loadAbs "+"loadBackpack loadFile loadGame loadIdentity loadMagazine loadOverlay loadStatus loadUniform "+"loadVest local localize locationPosition lock lockCameraTo lockCargo lockDriver locked "+"lockedCargo lockedDriver lockedTurret lockIdentity lockTurret lockWP log logEntities logNetwork "+"logNetworkTerminate lookAt lookAtPos magazineCargo magazines magazinesAllTurrets magazinesAmmo "+"magazinesAmmoCargo magazinesAmmoFull magazinesDetail magazinesDetailBackpack "+"magazinesDetailUniform magazinesDetailVest magazinesTurret magazineTurretAmmo mapAnimAdd "+"mapAnimClear mapAnimCommit mapAnimDone mapCenterOnCamera mapGridPosition markAsFinishedOnSteam "+"markerAlpha markerBrush markerColor markerDir markerPos markerShape markerSize markerText "+"markerType max members menuAction menuAdd menuChecked menuClear menuCollapse menuData menuDelete "+"menuEnable menuEnabled menuExpand menuHover menuPicture menuSetAction menuSetCheck menuSetData "+"menuSetPicture menuSetValue menuShortcut menuShortcutText menuSize menuSort menuText menuURL "+"menuValue min mineActive mineDetectedBy missionConfigFile missionDifficulty missionName "+"missionNamespace missionStart missionVersion mod modelToWorld modelToWorldVisual "+"modelToWorldVisualWorld modelToWorldWorld modParams moonIntensity moonPhase morale move "+"move3DENCamera moveInAny moveInCargo moveInCommander moveInDriver moveInGunner moveInTurret "+"moveObjectToEnd moveOut moveTime moveTo moveToCompleted moveToFailed musicVolume name nameSound "+"nearEntities nearestBuilding nearestLocation nearestLocations nearestLocationWithDubbing "+"nearestObject nearestObjects nearestTerrainObjects nearObjects nearObjectsReady nearRoads "+"nearSupplies nearTargets needReload netId netObjNull newOverlay nextMenuItemIndex "+"nextWeatherChange nMenuItems not numberOfEnginesRTD numberToDate objectCurators objectFromNetId "+"objectParent objStatus onBriefingGroup onBriefingNotes onBriefingPlan onBriefingTeamSwitch "+"onCommandModeChanged onDoubleClick onEachFrame onGroupIconClick onGroupIconOverEnter "+"onGroupIconOverLeave onHCGroupSelectionChanged onMapSingleClick onPlayerConnected "+"onPlayerDisconnected onPreloadFinished onPreloadStarted onShowNewObject onTeamSwitch "+"openCuratorInterface openDLCPage openMap openSteamApp openYoutubeVideo or orderGetIn overcast "+"overcastForecast owner param params parseNumber parseSimpleArray parseText parsingNamespace "+"particlesQuality pickWeaponPool pitch pixelGrid pixelGridBase pixelGridNoUIScale pixelH pixelW "+"playableSlotsNumber playableUnits playAction playActionNow player playerRespawnTime playerSide "+"playersNumber playGesture playMission playMove playMoveNow playMusic playScriptedMission "+"playSound playSound3D position positionCameraToWorld posScreenToWorld posWorldToScreen "+"ppEffectAdjust ppEffectCommit ppEffectCommitted ppEffectCreate ppEffectDestroy ppEffectEnable "+"ppEffectEnabled ppEffectForceInNVG precision preloadCamera preloadObject preloadSound "+"preloadTitleObj preloadTitleRsc preprocessFile preprocessFileLineNumbers primaryWeapon "+"primaryWeaponItems primaryWeaponMagazine priority processDiaryLink productVersion profileName "+"profileNamespace profileNameSteam progressLoadingScreen progressPosition progressSetPosition "+"publicVariable publicVariableClient publicVariableServer pushBack pushBackUnique putWeaponPool "+"queryItemsPool queryMagazinePool queryWeaponPool rad radioChannelAdd radioChannelCreate "+"radioChannelRemove radioChannelSetCallSign radioChannelSetLabel radioVolume rain rainbow random "+"rank rankId rating rectangular registeredTasks registerTask reload reloadEnabled remoteControl "+"remoteExec remoteExecCall remoteExecutedOwner remove3DENConnection remove3DENEventHandler "+"remove3DENLayer removeAction removeAll3DENEventHandlers removeAllActions removeAllAssignedItems "+"removeAllContainers removeAllCuratorAddons removeAllCuratorCameraAreas "+"removeAllCuratorEditingAreas removeAllEventHandlers removeAllHandgunItems removeAllItems "+"removeAllItemsWithMagazines removeAllMissionEventHandlers removeAllMPEventHandlers "+"removeAllMusicEventHandlers removeAllOwnedMines removeAllPrimaryWeaponItems removeAllWeapons "+"removeBackpack removeBackpackGlobal removeCuratorAddons removeCuratorCameraArea "+"removeCuratorEditableObjects removeCuratorEditingArea removeDrawIcon removeDrawLinks "+"removeEventHandler removeFromRemainsCollector removeGoggles removeGroupIcon removeHandgunItem "+"removeHeadgear removeItem removeItemFromBackpack removeItemFromUniform removeItemFromVest "+"removeItems removeMagazine removeMagazineGlobal removeMagazines removeMagazinesTurret "+"removeMagazineTurret removeMenuItem removeMissionEventHandler removeMPEventHandler "+"removeMusicEventHandler removeOwnedMine removePrimaryWeaponItem removeSecondaryWeaponItem "+"removeSimpleTask removeSwitchableUnit removeTeamMember removeUniform removeVest removeWeapon "+"removeWeaponAttachmentCargo removeWeaponCargo removeWeaponGlobal removeWeaponTurret "+"reportRemoteTarget requiredVersion resetCamShake resetSubgroupDirection resize resources "+"respawnVehicle restartEditorCamera reveal revealMine reverse reversedMouseY roadAt "+"roadsConnectedTo roleDescription ropeAttachedObjects ropeAttachedTo ropeAttachEnabled "+"ropeAttachTo ropeCreate ropeCut ropeDestroy ropeDetach ropeEndPosition ropeLength ropes "+"ropeUnwind ropeUnwound rotorsForcesRTD rotorsRpmRTD round runInitScript safeZoneH safeZoneW "+"safeZoneWAbs safeZoneX safeZoneXAbs safeZoneY save3DENInventory saveGame saveIdentity "+"saveJoysticks saveOverlay saveProfileNamespace saveStatus saveVar savingEnabled say say2D say3D "+"scopeName score scoreSide screenshot screenToWorld scriptDone scriptName scudState "+"secondaryWeapon secondaryWeaponItems secondaryWeaponMagazine select selectBestPlaces "+"selectDiarySubject selectedEditorObjects selectEditorObject selectionNames selectionPosition "+"selectLeader selectMax selectMin selectNoPlayer selectPlayer selectRandom selectRandomWeighted "+"selectWeapon selectWeaponTurret sendAUMessage sendSimpleCommand sendTask sendTaskResult "+"sendUDPMessage serverCommand serverCommandAvailable serverCommandExecutable serverName serverTime "+"set set3DENAttribute set3DENAttributes set3DENGrid set3DENIconsVisible set3DENLayer "+"set3DENLinesVisible set3DENLogicType set3DENMissionAttribute set3DENMissionAttributes "+"set3DENModelsVisible set3DENObjectType set3DENSelected setAccTime setActualCollectiveRTD "+"setAirplaneThrottle setAirportSide setAmmo setAmmoCargo setAmmoOnPylon setAnimSpeedCoef "+"setAperture setApertureNew setArmoryPoints setAttributes setAutonomous setBehaviour "+"setBleedingRemaining setBrakesRTD setCameraInterest setCamShakeDefParams setCamShakeParams "+"setCamUseTI setCaptive setCenterOfMass setCollisionLight setCombatMode setCompassOscillation "+"setConvoySeparation setCuratorCameraAreaCeiling setCuratorCoef setCuratorEditingAreaType "+"setCuratorWaypointCost setCurrentChannel setCurrentTask setCurrentWaypoint setCustomAimCoef "+"setCustomWeightRTD setDamage setDammage setDate setDebriefingText setDefaultCamera setDestination "+"setDetailMapBlendPars setDir setDirection setDrawIcon setDriveOnPath setDropInterval "+"setDynamicSimulationDistance setDynamicSimulationDistanceCoef setEditorMode setEditorObjectScope "+"setEffectCondition setEngineRPMRTD setFace setFaceAnimation setFatigue setFeatureType "+"setFlagAnimationPhase setFlagOwner setFlagSide setFlagTexture setFog setFormation "+"setFormationTask setFormDir setFriend setFromEditor setFSMVariable setFuel setFuelCargo "+"setGroupIcon setGroupIconParams setGroupIconsSelectable setGroupIconsVisible setGroupId "+"setGroupIdGlobal setGroupOwner setGusts setHideBehind setHit setHitIndex setHitPointDamage "+"setHorizonParallaxCoef setHUDMovementLevels setIdentity setImportance setInfoPanel setLeader "+"setLightAmbient setLightAttenuation setLightBrightness setLightColor setLightDayLight "+"setLightFlareMaxDistance setLightFlareSize setLightIntensity setLightnings setLightUseFlare "+"setLocalWindParams setMagazineTurretAmmo setMarkerAlpha setMarkerAlphaLocal setMarkerBrush "+"setMarkerBrushLocal setMarkerColor setMarkerColorLocal setMarkerDir setMarkerDirLocal "+"setMarkerPos setMarkerPosLocal setMarkerShape setMarkerShapeLocal setMarkerSize "+"setMarkerSizeLocal setMarkerText setMarkerTextLocal setMarkerType setMarkerTypeLocal setMass "+"setMimic setMousePosition setMusicEffect setMusicEventHandler setName setNameSound "+"setObjectArguments setObjectMaterial setObjectMaterialGlobal setObjectProxy setObjectTexture "+"setObjectTextureGlobal setObjectViewDistance setOvercast setOwner setOxygenRemaining "+"setParticleCircle setParticleClass setParticleFire setParticleParams setParticleRandom "+"setPilotCameraDirection setPilotCameraRotation setPilotCameraTarget setPilotLight setPiPEffect "+"setPitch setPlateNumber setPlayable setPlayerRespawnTime setPos setPosASL setPosASL2 setPosASLW "+"setPosATL setPosition setPosWorld setPylonLoadOut setPylonsPriority setRadioMsg setRain "+"setRainbow setRandomLip setRank setRectangular setRepairCargo setRotorBrakeRTD setShadowDistance "+"setShotParents setSide setSimpleTaskAlwaysVisible setSimpleTaskCustomData "+"setSimpleTaskDescription setSimpleTaskDestination setSimpleTaskTarget setSimpleTaskType "+"setSimulWeatherLayers setSize setSkill setSlingLoad setSoundEffect setSpeaker setSpeech "+"setSpeedMode setStamina setStaminaScheme setStatValue setSuppression setSystemOfUnits "+"setTargetAge setTaskMarkerOffset setTaskResult setTaskState setTerrainGrid setText "+"setTimeMultiplier setTitleEffect setTrafficDensity setTrafficDistance setTrafficGap "+"setTrafficSpeed setTriggerActivation setTriggerArea setTriggerStatements setTriggerText "+"setTriggerTimeout setTriggerType setType setUnconscious setUnitAbility setUnitLoadout setUnitPos "+"setUnitPosWeak setUnitRank setUnitRecoilCoefficient setUnitTrait setUnloadInCombat "+"setUserActionText setUserMFDText setUserMFDvalue setVariable setVectorDir setVectorDirAndUp "+"setVectorUp setVehicleAmmo setVehicleAmmoDef setVehicleArmor setVehicleCargo setVehicleId "+"setVehicleLock setVehiclePosition setVehicleRadar setVehicleReceiveRemoteTargets "+"setVehicleReportOwnPosition setVehicleReportRemoteTargets setVehicleTIPars setVehicleVarName "+"setVelocity setVelocityModelSpace setVelocityTransformation setViewDistance "+"setVisibleIfTreeCollapsed setWantedRPMRTD setWaves setWaypointBehaviour setWaypointCombatMode "+"setWaypointCompletionRadius setWaypointDescription setWaypointForceBehaviour setWaypointFormation "+"setWaypointHousePosition setWaypointLoiterRadius setWaypointLoiterType setWaypointName "+"setWaypointPosition setWaypointScript setWaypointSpeed setWaypointStatements setWaypointTimeout "+"setWaypointType setWaypointVisible setWeaponReloadingTime setWind setWindDir setWindForce "+"setWindStr setWingForceScaleRTD setWPPos show3DIcons showChat showCinemaBorder showCommandingMenu "+"showCompass showCuratorCompass showGPS showHUD showLegend showMap shownArtilleryComputer "+"shownChat shownCompass shownCuratorCompass showNewEditorObject shownGPS shownHUD shownMap "+"shownPad shownRadio shownScoretable shownUAVFeed shownWarrant shownWatch showPad showRadio "+"showScoretable showSubtitles showUAVFeed showWarrant showWatch showWaypoint showWaypoints side "+"sideChat sideEnemy sideFriendly sideRadio simpleTasks simulationEnabled simulCloudDensity "+"simulCloudOcclusion simulInClouds simulWeatherSync sin size sizeOf skill skillFinal skipTime "+"sleep sliderPosition sliderRange sliderSetPosition sliderSetRange sliderSetSpeed sliderSpeed "+"slingLoadAssistantShown soldierMagazines someAmmo sort soundVolume spawn speaker speed speedMode "+"splitString sqrt squadParams stance startLoadingScreen step stop stopEngineRTD stopped str "+"sunOrMoon supportInfo suppressFor surfaceIsWater surfaceNormal surfaceType swimInDepth "+"switchableUnits switchAction switchCamera switchGesture switchLight switchMove "+"synchronizedObjects synchronizedTriggers synchronizedWaypoints synchronizeObjectsAdd "+"synchronizeObjectsRemove synchronizeTrigger synchronizeWaypoint systemChat systemOfUnits tan "+"targetKnowledge targets targetsAggregate targetsQuery taskAlwaysVisible taskChildren "+"taskCompleted taskCustomData taskDescription taskDestination taskHint taskMarkerOffset taskParent "+"taskResult taskState taskType teamMember teamName teams teamSwitch teamSwitchEnabled teamType "+"terminate terrainIntersect terrainIntersectASL terrainIntersectAtASL text textLog textLogFormat "+"tg time timeMultiplier titleCut titleFadeOut titleObj titleRsc titleText toArray toFixed toLower "+"toString toUpper triggerActivated triggerActivation triggerArea triggerAttachedVehicle "+"triggerAttachObject triggerAttachVehicle triggerDynamicSimulation triggerStatements triggerText "+"triggerTimeout triggerTimeoutCurrent triggerType turretLocal turretOwner turretUnit tvAdd tvClear "+"tvCollapse tvCollapseAll tvCount tvCurSel tvData tvDelete tvExpand tvExpandAll tvPicture "+"tvSetColor tvSetCurSel tvSetData tvSetPicture tvSetPictureColor tvSetPictureColorDisabled "+"tvSetPictureColorSelected tvSetPictureRight tvSetPictureRightColor tvSetPictureRightColorDisabled "+"tvSetPictureRightColorSelected tvSetText tvSetTooltip tvSetValue tvSort tvSortByValue tvText "+"tvTooltip tvValue type typeName typeOf UAVControl uiNamespace uiSleep unassignCurator "+"unassignItem unassignTeam unassignVehicle underwater uniform uniformContainer uniformItems "+"uniformMagazines unitAddons unitAimPosition unitAimPositionVisual unitBackpack unitIsUAV unitPos "+"unitReady unitRecoilCoefficient units unitsBelowHeight unlinkItem unlockAchievement "+"unregisterTask updateDrawIcon updateMenuItem updateObjectTree useAISteeringComponent "+"useAudioTimeForMoves userInputDisabled vectorAdd vectorCos vectorCrossProduct vectorDiff "+"vectorDir vectorDirVisual vectorDistance vectorDistanceSqr vectorDotProduct vectorFromTo "+"vectorMagnitude vectorMagnitudeSqr vectorModelToWorld vectorModelToWorldVisual vectorMultiply "+"vectorNormalized vectorUp vectorUpVisual vectorWorldToModel vectorWorldToModelVisual vehicle "+"vehicleCargoEnabled vehicleChat vehicleRadio vehicleReceiveRemoteTargets vehicleReportOwnPosition "+"vehicleReportRemoteTargets vehicles vehicleVarName velocity velocityModelSpace verifySignature "+"vest vestContainer vestItems vestMagazines viewDistance visibleCompass visibleGPS visibleMap "+"visiblePosition visiblePositionASL visibleScoretable visibleWatch waves waypointAttachedObject "+"waypointAttachedVehicle waypointAttachObject waypointAttachVehicle waypointBehaviour "+"waypointCombatMode waypointCompletionRadius waypointDescription waypointForceBehaviour "+"waypointFormation waypointHousePosition waypointLoiterRadius waypointLoiterType waypointName "+"waypointPosition waypoints waypointScript waypointsEnabledUAV waypointShow waypointSpeed "+"waypointStatements waypointTimeout waypointTimeoutCurrent waypointType waypointVisible "+"weaponAccessories weaponAccessoriesCargo weaponCargo weaponDirection weaponInertia weaponLowered "+"weapons weaponsItems weaponsItemsCargo weaponState weaponsTurret weightRTD WFSideText wind ",literal:"blufor civilian configNull controlNull displayNull east endl false grpNull independent lineBreak "+"locationNull nil objNull opfor pi resistance scriptNull sideAmbientLife sideEmpty sideLogic "+"sideUnknown taskNull teamMemberNull true west"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.NUMBER_MODE,VARIABLE,FUNCTION,STRINGS,PREPROCESSOR],illegal:/#|^\$ /}});hljs.registerLanguage("qml",function(hljs){var KEYWORDS={keyword:"in of on if for while finally var new function do return void else break catch "+"instanceof with throw case default try this switch continue typeof delete "+"let yield const export super debugger as async await import",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent "+"encodeURI encodeURIComponent escape unescape Object Function Boolean Error "+"EvalError InternalError RangeError ReferenceError StopIteration SyntaxError "+"TypeError URIError Number Math Date String RegExp Array Float32Array "+"Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array "+"Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require "+"module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect "+"Behavior bool color coordinate date double enumeration font geocircle georectangle "+"geoshape int list matrix4x4 parent point quaternion real rect "+"size string url variant vector2d vector3d vector4d"+"Promise"};var QML_IDENT_RE="[a-zA-Z_][a-zA-Z0-9\\._]*";var PROPERTY={className:"keyword",begin:"\\bproperty\\b",starts:{className:"string",end:"(:|=|;|,|//|/\\*|$)",returnEnd:true}};var SIGNAL={className:"keyword",begin:"\\bsignal\\b",starts:{className:"string",end:"(\\(|:|=|;|,|//|/\\*|$)",returnEnd:true}};var ID_ID={className:"attribute",begin:"\\bid\\s*:",starts:{className:"string",end:QML_IDENT_RE,returnEnd:false}};var QML_ATTRIBUTE={begin:QML_IDENT_RE+"\\s*:",returnBegin:true,contains:[{className:"attribute",begin:QML_IDENT_RE,end:"\\s*:",excludeEnd:true,relevance:0}],relevance:0};var QML_OBJECT={begin:QML_IDENT_RE+"\\s*{",end:"{",returnBegin:true,relevance:0,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:QML_IDENT_RE})]};return{aliases:["qt"],case_insensitive:false,keywords:KEYWORDS,contains:[{className:"meta",begin:/^\s*['"]use (strict|asm)['"]/},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"string",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE,{className:"subst",begin:"\\$\\{",end:"\\}"}]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"number",variants:[{begin:"\\b(0[bB][01]+)"},{begin:"\\b(0[oO][0-7]+)"},{begin:hljs.C_NUMBER_RE}],relevance:0},{begin:"("+hljs.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*",keywords:"return throw case",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.REGEXP_MODE,{begin:/</,end:/>\s*[);\]]/,relevance:0,subLanguage:"xml"}],relevance:0},SIGNAL,PROPERTY,{className:"function",beginKeywords:"function",end:/\{/,excludeEnd:true,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:/[A-Za-z$_][0-9A-Za-z$_]*/}),{className:"params",begin:/\(/,end:/\)/,excludeBegin:true,excludeEnd:true,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}],illegal:/\[|%/},{begin:"\\."+hljs.IDENT_RE,relevance:0},ID_ID,QML_ATTRIBUTE,QML_OBJECT],illegal:/#/}});hljs.registerLanguage("erb",function(hljs){return{subLanguage:"xml",contains:[hljs.COMMENT("<%#","%>"),{begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:true,excludeEnd:true}]}});hljs.registerLanguage("openscad",function(hljs){var SPECIAL_VARS={className:"keyword",begin:"\\$(f[asn]|t|vp[rtd]|children)"},LITERALS={className:"literal",begin:"false|true|PI|undef"},NUMBERS={className:"number",begin:"\\b\\d+(\\.\\d+)?(e-?\\d+)?",relevance:0},STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),PREPRO={className:"meta",keywords:{"meta-keyword":"include use"},begin:"include|use <",end:">"},PARAMS={className:"params",begin:"\\(",end:"\\)",contains:["self",NUMBERS,STRING,SPECIAL_VARS,LITERALS]},MODIFIERS={begin:"[*!#%]",relevance:0},FUNCTIONS={className:"function",beginKeywords:"module function",end:"\\=|\\{",contains:[PARAMS,hljs.UNDERSCORE_TITLE_MODE]};return{aliases:["scad"],keywords:{keyword:"function module include use for intersection_for if else \\%",literal:"false true PI undef",built_in:"circle square polygon text sphere cube cylinder polyhedron translate rotate scale resize mirror multmatrix color offset hull minkowski union difference intersection abs sign sin cos tan acos asin atan atan2 floor round ceil ln log pow sqrt exp rands min max concat lookup str chr search version version_num norm cross parent_module echo import import_dxf dxf_linear_extrude linear_extrude rotate_extrude surface projection render children dxf_cross dxf_dim let assign"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,NUMBERS,PREPRO,STRING,SPECIAL_VARS,MODIFIERS,FUNCTIONS]}});hljs.registerLanguage("elixir",function(hljs){var ELIXIR_IDENT_RE="[a-zA-Z_][a-zA-Z0-9_.]*(\\!|\\?)?";var ELIXIR_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?";var ELIXIR_KEYWORDS="and false then defined module in return redo retry end for true self when "+"next until do begin unless nil break not case cond alias while ensure or "+"include use alias fn quote require import with|0";var SUBST={className:"subst",begin:"#\\{",end:"}",lexemes:ELIXIR_IDENT_RE,keywords:ELIXIR_KEYWORDS};var SIGIL_DELIMITERS="[/|([{<\"']";var LOWERCASE_SIGIL={className:"string",begin:"~[a-z]"+"(?="+SIGIL_DELIMITERS+")",contains:[{endsParent:true,contains:[{contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"/,end:/"/},{begin:/'/,end:/'/},{begin:/\//,end:/\//},{begin:/\|/,end:/\|/},{begin:/\(/,end:/\)/},{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/},{begin:/</,end:/>/}]}]}]};var UPCASE_SIGIL={className:"string",begin:"~[A-Z]"+"(?="+SIGIL_DELIMITERS+")",contains:[{begin:/"/,end:/"/},{begin:/'/,end:/'/},{begin:/\//,end:/\//},{begin:/\|/,end:/\|/},{begin:/\(/,end:/\)/},{begin:/\[/,end:/\]/},{begin:/\{/,end:/\}/},{begin:/\</,end:/\>/}]};var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/"""/,end:/"""/},{begin:/'''/,end:/'''/},{begin:/~S"""/,end:/"""/,contains:[]},{begin:/~S"/,end:/"/,contains:[]},{begin:/~S'''/,end:/'''/,contains:[]},{begin:/~S'/,end:/'/,contains:[]},{begin:/'/,end:/'/},{begin:/"/,end:/"/}]};var FUNCTION={className:"function",beginKeywords:"def defp defmacro",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:ELIXIR_IDENT_RE,endsParent:true})]};var CLASS=hljs.inherit(FUNCTION,{className:"class",beginKeywords:"defimpl defmodule defprotocol defrecord",end:/\bdo\b|$|;/});var ELIXIR_DEFAULT_CONTAINS=[STRING,UPCASE_SIGIL,LOWERCASE_SIGIL,hljs.HASH_COMMENT_MODE,CLASS,FUNCTION,{begin:"::"},{className:"symbol",begin:":(?![\\s:])",contains:[STRING,{begin:ELIXIR_METHOD_RE}],relevance:0},{className:"symbol",begin:ELIXIR_IDENT_RE+":(?!:)",relevance:0},{className:"number",begin:"(\\b0o[0-7_]+)|(\\b0b[01_]+)|(\\b0x[0-9a-fA-F_]+)|(-?\\b[1-9][0-9_]*(.[0-9_]+([eE][-+]?[0-9]+)?)?)",relevance:0},{className:"variable",begin:"(\\$\\W)|((\\$|\\@\\@?)(\\w+))"},{begin:"->"},{begin:"("+hljs.RE_STARTERS_RE+")\\s*",contains:[hljs.HASH_COMMENT_MODE,{className:"regexp",illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"/",end:"/[a-z]*"},{begin:"%r\\[",end:"\\][a-z]*"}]}],relevance:0}];SUBST.contains=ELIXIR_DEFAULT_CONTAINS;return{lexemes:ELIXIR_IDENT_RE,keywords:ELIXIR_KEYWORDS,contains:ELIXIR_DEFAULT_CONTAINS}});hljs.registerLanguage("vhdl",function(hljs){var INTEGER_RE="\\d(_|\\d)*";var EXPONENT_RE="[eE][-+]?"+INTEGER_RE;var DECIMAL_LITERAL_RE=INTEGER_RE+"(\\."+INTEGER_RE+")?"+"("+EXPONENT_RE+")?";var BASED_INTEGER_RE="\\w+";var BASED_LITERAL_RE=INTEGER_RE+"#"+BASED_INTEGER_RE+"(\\."+BASED_INTEGER_RE+")?"+"#"+"("+EXPONENT_RE+")?";var NUMBER_RE="\\b("+BASED_LITERAL_RE+"|"+DECIMAL_LITERAL_RE+")";return{case_insensitive:true,keywords:{keyword:"abs access after alias all and architecture array assert assume assume_guarantee attribute "+"begin block body buffer bus case component configuration constant context cover disconnect "+"downto default else elsif end entity exit fairness file for force function generate "+"generic group guarded if impure in inertial inout is label library linkage literal "+"loop map mod nand new next nor not null of on open or others out package parameter port "+"postponed procedure process property protected pure range record register reject "+"release rem report restrict restrict_guarantee return rol ror select sequence "+"severity shared signal sla sll sra srl strong subtype then to transport type "+"unaffected units until use variable view vmode vprop vunit wait when while with xnor xor",built_in:"boolean bit character "+"integer time delay_length natural positive "+"string bit_vector file_open_kind file_open_status "+"std_logic std_logic_vector unsigned signed boolean_vector integer_vector "+"std_ulogic std_ulogic_vector unresolved_unsigned u_unsigned unresolved_signed u_signed "+"real_vector time_vector",literal:"false true note warning error failure "+"line text side width"},illegal:"{",contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT("--","$"),hljs.QUOTE_STRING_MODE,{className:"number",begin:NUMBER_RE,relevance:0},{className:"string",begin:"'(U|X|0|1|Z|W|L|H|-)'",contains:[hljs.BACKSLASH_ESCAPE]},{className:"symbol",begin:"'[A-Za-z](_?[A-Za-z0-9])*",contains:[hljs.BACKSLASH_ESCAPE]}]}});hljs.registerLanguage("aspectj",function(hljs){var KEYWORDS="false synchronized int abstract float private char boolean static null if const "+"for true while long throw strictfp finally protected import native final return void "+"enum else extends implements break transient new catch instanceof byte super volatile case "+"assert short package default double public try this switch continue throws privileged "+"aspectOf adviceexecution proceed cflowbelow cflow initialization preinitialization "+"staticinitialization withincode target within execution getWithinTypeName handler "+"thisJoinPoint thisJoinPointStaticPart thisEnclosingJoinPointStaticPart declare parents "+"warning error soft precedence thisAspectInstance";var SHORTKEYS="get set args call";return{keywords:KEYWORDS,illegal:/<\/|#/,contains:[hljs.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/,relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,{className:"class",beginKeywords:"aspect",end:/[{;=]/,excludeEnd:true,illegal:/[:;"\[\]]/,contains:[{beginKeywords:"extends implements pertypewithin perthis pertarget percflowbelow percflow issingleton"},hljs.UNDERSCORE_TITLE_MODE,{begin:/\([^\)]*/,end:/[)]+/,keywords:KEYWORDS+" "+SHORTKEYS,excludeEnd:false}]},{className:"class",beginKeywords:"class interface",end:/[{;=]/,excludeEnd:true,relevance:0,keywords:"class interface",illegal:/[:"\[\]]/,contains:[{beginKeywords:"extends implements"},hljs.UNDERSCORE_TITLE_MODE]},{beginKeywords:"pointcut after before around throwing returning",end:/[)]/,excludeEnd:false,illegal:/["\[\]]/,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,contains:[hljs.UNDERSCORE_TITLE_MODE]}]},{begin:/[:]/,returnBegin:true,end:/[{;]/,relevance:0,excludeEnd:false,keywords:KEYWORDS,illegal:/["\[\]]/,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",keywords:KEYWORDS+" "+SHORTKEYS,relevance:0},hljs.QUOTE_STRING_MODE]},{beginKeywords:"new throw",relevance:0},{className:"function",begin:/\w+ +\w+(\.)?\w+\s*\([^\)]*\)\s*((throws)[\w\s,]+)?[\{;]/,returnBegin:true,end:/[{;=]/,keywords:KEYWORDS,excludeEnd:true,contains:[{begin:hljs.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:true,relevance:0,contains:[hljs.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/,relevance:0,keywords:KEYWORDS,contains:[hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_NUMBER_MODE,{className:"meta",begin:"@[A-Za-z]+"}]}});hljs.registerLanguage("plaintext",function(hljs){return{disableAutodetect:true}});hljs.registerLanguage("xquery",function(hljs){var KEYWORDS="module schema namespace boundary-space preserve no-preserve strip default collation base-uri ordering context decimal-format decimal-separator copy-namespaces empty-sequence except exponent-separator external grouping-separator inherit no-inherit lax minus-sign per-mille percent schema-attribute schema-element strict unordered zero-digit "+"declare import option function validate variable "+"for at in let where order group by return if then else "+"tumbling sliding window start when only end previous next stable "+"ascending descending allowing empty greatest least some every satisfies switch case typeswitch try catch "+"and or to union intersect instance of treat as castable cast map array "+"delete insert into replace value rename copy modify update";var TYPE="item document-node node attribute document element comment namespace namespace-node processing-instruction text construction "+"xs:anyAtomicType xs:untypedAtomic xs:duration xs:time xs:decimal xs:float xs:double xs:gYearMonth xs:gYear xs:gMonthDay xs:gMonth xs:gDay xs:boolean xs:base64Binary xs:hexBinary xs:anyURI xs:QName xs:NOTATION xs:dateTime xs:dateTimeStamp xs:date xs:string xs:normalizedString xs:token xs:language xs:NMTOKEN xs:Name xs:NCName xs:ID xs:IDREF xs:ENTITY xs:integer xs:nonPositiveInteger xs:negativeInteger xs:long xs:int xs:short xs:byte xs:nonNegativeInteger xs:unisignedLong xs:unsignedInt xs:unsignedShort xs:unsignedByte xs:positiveInteger xs:yearMonthDuration xs:dayTimeDuration";var LITERAL="eq ne lt le gt ge is "+"self:: child:: descendant:: descendant-or-self:: attribute:: following:: following-sibling:: parent:: ancestor:: ancestor-or-self:: preceding:: preceding-sibling:: "+"NaN";var BUILT_IN={className:"built_in",variants:[{begin:/\barray\:/,end:/(?:append|filter|flatten|fold\-(?:left|right)|for-each(?:\-pair)?|get|head|insert\-before|join|put|remove|reverse|size|sort|subarray|tail)\b/},{begin:/\bmap\:/,end:/(?:contains|entry|find|for\-each|get|keys|merge|put|remove|size)\b/},{begin:/\bmath\:/,end:/(?:a(?:cos|sin|tan[2]?)|cos|exp(?:10)?|log(?:10)?|pi|pow|sin|sqrt|tan)\b/},{begin:/\bop\:/,end:/\(/,excludeEnd:true},{begin:/\bfn\:/,end:/\(/,excludeEnd:true},{begin:/[^<\/\$\:'"-]\b(?:abs|accumulator\-(?:after|before)|adjust\-(?:date(?:Time)?|time)\-to\-timezone|analyze\-string|apply|available\-(?:environment\-variables|system\-properties)|avg|base\-uri|boolean|ceiling|codepoints?\-(?:equal|to\-string)|collation\-key|collection|compare|concat|contains(?:\-token)?|copy\-of|count|current(?:\-)?(?:date(?:Time)?|time|group(?:ing\-key)?|output\-uri|merge\-(?:group|key))?data|dateTime|days?\-from\-(?:date(?:Time)?|duration)|deep\-equal|default\-(?:collation|language)|distinct\-values|document(?:\-uri)?|doc(?:\-available)?|element\-(?:available|with\-id)|empty|encode\-for\-uri|ends\-with|environment\-variable|error|escape\-html\-uri|exactly\-one|exists|false|filter|floor|fold\-(?:left|right)|for\-each(?:\-pair)?|format\-(?:date(?:Time)?|time|integer|number)|function\-(?:arity|available|lookup|name)|generate\-id|has\-children|head|hours\-from\-(?:dateTime|duration|time)|id(?:ref)?|implicit\-timezone|in\-scope\-prefixes|index\-of|innermost|insert\-before|iri\-to\-uri|json\-(?:doc|to\-xml)|key|lang|last|load\-xquery\-module|local\-name(?:\-from\-QName)?|(?:lower|upper)\-case|matches|max|minutes\-from\-(?:dateTime|duration|time)|min|months?\-from\-(?:date(?:Time)?|duration)|name(?:space\-uri\-?(?:for\-prefix|from\-QName)?)?|nilled|node\-name|normalize\-(?:space|unicode)|not|number|one\-or\-more|outermost|parse\-(?:ietf\-date|json)|path|position|(?:prefix\-from\-)?QName|random\-number\-generator|regex\-group|remove|replace|resolve\-(?:QName|uri)|reverse|root|round(?:\-half\-to\-even)?|seconds\-from\-(?:dateTime|duration|time)|snapshot|sort|starts\-with|static\-base\-uri|stream\-available|string\-?(?:join|length|to\-codepoints)?|subsequence|substring\-?(?:after|before)?|sum|system\-property|tail|timezone\-from\-(?:date(?:Time)?|time)|tokenize|trace|trans(?:form|late)|true|type\-available|unordered|unparsed\-(?:entity|text)?\-?(?:public\-id|uri|available|lines)?|uri\-collection|xml\-to\-json|years?\-from\-(?:date(?:Time)?|duration)|zero\-or\-one)\b/},{begin:/\blocal\:/,end:/\(/,excludeEnd:true},{begin:/\bzip\:/,end:/(?:zip\-file|(?:xml|html|text|binary)\-entry| (?:update\-)?entries)\b/},{begin:/\b(?:util|db|functx|app|xdmp|xmldb)\:/,end:/\(/,excludeEnd:true}]};var TITLE={className:"title",begin:/\bxquery version "[13]\.[01]"\s?(?:encoding ".+")?/,end:/;/};var VAR={className:"variable",begin:/[\$][\w-:]+/};var NUMBER={className:"number",begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",relevance:0};var STRING={className:"string",variants:[{begin:/"/,end:/"/,contains:[{begin:/""/,relevance:0}]},{begin:/'/,end:/'/,contains:[{begin:/''/,relevance:0}]}]};var ANNOTATION={className:"meta",begin:/%[\w-:]+/};var COMMENT={className:"comment",begin:"\\(:",end:":\\)",relevance:10,contains:[{className:"doctag",begin:"@\\w+"}]};var COMPUTED={beginKeywords:"element attribute comment document processing-instruction",end:"{",excludeEnd:true};var DIRECT={begin:/<([\w\._:\-]+)((\s*.*)=('|").*('|"))?>/,end:/(\/[\w\._:\-]+>)/,subLanguage:"xml",contains:[{begin:"{",end:"}",subLanguage:"xquery"},"self"]};var CONTAINS=[VAR,BUILT_IN,STRING,NUMBER,COMMENT,ANNOTATION,TITLE,COMPUTED,DIRECT];var METHOD={begin:"{",end:"}",contains:CONTAINS};return{aliases:["xpath","xq"],case_insensitive:false,lexemes:/[a-zA-Z\$][a-zA-Z0-9_:\-]*/,illegal:/(proc)|(abstract)|(extends)|(until)|(#)/,keywords:{keyword:KEYWORDS,type:TYPE,literal:LITERAL},contains:CONTAINS}});hljs.registerLanguage("irpf90",function(hljs){var PARAMS={className:"params",begin:"\\(",end:"\\)"};var F_KEYWORDS={literal:".False. .True.",keyword:"kind do while private call intrinsic where elsewhere "+"type endtype endmodule endselect endinterface end enddo endif if forall endforall only contains default return stop then "+"public subroutine|10 function program .and. .or. .not. .le. .eq. .ge. .gt. .lt. "+"goto save else use module select case "+"access blank direct exist file fmt form formatted iostat name named nextrec number opened rec recl sequential status unformatted unit "+"continue format pause cycle exit "+"c_null_char c_alert c_backspace c_form_feed flush wait decimal round iomsg "+"synchronous nopass non_overridable pass protected volatile abstract extends import "+"non_intrinsic value deferred generic final enumerator class associate bind enum "+"c_int c_short c_long c_long_long c_signed_char c_size_t c_int8_t c_int16_t c_int32_t c_int64_t c_int_least8_t c_int_least16_t "+"c_int_least32_t c_int_least64_t c_int_fast8_t c_int_fast16_t c_int_fast32_t c_int_fast64_t c_intmax_t C_intptr_t c_float c_double "+"c_long_double c_float_complex c_double_complex c_long_double_complex c_bool c_char c_null_ptr c_null_funptr "+"c_new_line c_carriage_return c_horizontal_tab c_vertical_tab iso_c_binding c_loc c_funloc c_associated  c_f_pointer "+"c_ptr c_funptr iso_fortran_env character_storage_size error_unit file_storage_size input_unit iostat_end iostat_eor "+"numeric_storage_size output_unit c_f_procpointer ieee_arithmetic ieee_support_underflow_control "+"ieee_get_underflow_mode ieee_set_underflow_mode newunit contiguous recursive "+"pad position action delim readwrite eor advance nml interface procedure namelist include sequence elemental pure "+"integer real character complex logical dimension allocatable|10 parameter "+"external implicit|10 none double precision assign intent optional pointer "+"target in out common equivalence data "+"begin_provider &begin_provider end_provider begin_shell end_shell begin_template end_template subst assert touch "+"soft_touch provide no_dep free irp_if irp_else irp_endif irp_write irp_read",built_in:"alog alog10 amax0 amax1 amin0 amin1 amod cabs ccos cexp clog csin csqrt dabs dacos dasin datan datan2 dcos dcosh ddim dexp dint "+"dlog dlog10 dmax1 dmin1 dmod dnint dsign dsin dsinh dsqrt dtan dtanh float iabs idim idint idnint ifix isign max0 max1 min0 min1 sngl "+"algama cdabs cdcos cdexp cdlog cdsin cdsqrt cqabs cqcos cqexp cqlog cqsin cqsqrt dcmplx dconjg derf derfc dfloat dgamma dimag dlgama "+"iqint qabs qacos qasin qatan qatan2 qcmplx qconjg qcos qcosh qdim qerf qerfc qexp qgamma qimag qlgama qlog qlog10 qmax1 qmin1 qmod "+"qnint qsign qsin qsinh qsqrt qtan qtanh abs acos aimag aint anint asin atan atan2 char cmplx conjg cos cosh exp ichar index int log "+"log10 max min nint sign sin sinh sqrt tan tanh print write dim lge lgt lle llt mod nullify allocate deallocate "+"adjustl adjustr all allocated any associated bit_size btest ceiling count cshift date_and_time digits dot_product "+"eoshift epsilon exponent floor fraction huge iand ibclr ibits ibset ieor ior ishft ishftc lbound len_trim matmul "+"maxexponent maxloc maxval merge minexponent minloc minval modulo mvbits nearest pack present product "+"radix random_number random_seed range repeat reshape rrspacing scale scan selected_int_kind selected_real_kind "+"set_exponent shape size spacing spread sum system_clock tiny transpose trim ubound unpack verify achar iachar transfer "+"dble entry dprod cpu_time command_argument_count get_command get_command_argument get_environment_variable is_iostat_end "+"ieee_arithmetic ieee_support_underflow_control ieee_get_underflow_mode ieee_set_underflow_mode "+"is_iostat_eor move_alloc new_line selected_char_kind same_type_as extends_type_of"+"acosh asinh atanh bessel_j0 bessel_j1 bessel_jn bessel_y0 bessel_y1 bessel_yn erf erfc erfc_scaled gamma log_gamma hypot norm2 "+"atomic_define atomic_ref execute_command_line leadz trailz storage_size merge_bits "+"bge bgt ble blt dshiftl dshiftr findloc iall iany iparity image_index lcobound ucobound maskl maskr "+"num_images parity popcnt poppar shifta shiftl shiftr this_image "+"IRP_ALIGN irp_here"};return{case_insensitive:true,keywords:F_KEYWORDS,illegal:/\/\*/,contains:[hljs.inherit(hljs.APOS_STRING_MODE,{className:"string",relevance:0}),hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"string",relevance:0}),{className:"function",beginKeywords:"subroutine function program",illegal:"[${=\\n]",contains:[hljs.UNDERSCORE_TITLE_MODE,PARAMS]},hljs.COMMENT("!","$",{relevance:0}),hljs.COMMENT("begin_doc","end_doc",{relevance:10}),{className:"number",begin:"(?=\\b|\\+|\\-|\\.)(?=\\.\\d|\\d)(?:\\d+)?(?:\\.?\\d*)(?:[de][+-]?\\d+)?\\b\\.?",relevance:0}]}});hljs.registerLanguage("cal",function(hljs){var KEYWORDS="div mod in and or not xor asserterror begin case do downto else end exit for if of repeat then to "+"until while with var";var LITERALS="false true";var COMMENT_MODES=[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/\{/,/\}/,{relevance:0}),hljs.COMMENT(/\(\*/,/\*\)/,{relevance:10})];var STRING={className:"string",begin:/'/,end:/'/,contains:[{begin:/''/}]};var CHAR_STRING={className:"string",begin:/(#\d+)+/};var DATE={className:"number",begin:"\\b\\d+(\\.\\d+)?(DT|D|T)",relevance:0};var DBL_QUOTED_VARIABLE={className:"string",begin:'"',end:'"'};var PROCEDURE={className:"function",beginKeywords:"procedure",end:/[:;]/,keywords:"procedure|10",contains:[hljs.TITLE_MODE,{className:"params",begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:[STRING,CHAR_STRING]}].concat(COMMENT_MODES)};var OBJECT={className:"class",begin:"OBJECT (Table|Form|Report|Dataport|Codeunit|XMLport|MenuSuite|Page|Query) (\\d+) ([^\\r\\n]+)",returnBegin:true,contains:[hljs.TITLE_MODE,PROCEDURE]};return{case_insensitive:true,keywords:{keyword:KEYWORDS,literal:LITERALS},illegal:/\/\*/,contains:[STRING,CHAR_STRING,DATE,DBL_QUOTED_VARIABLE,hljs.NUMBER_MODE,OBJECT,PROCEDURE]}});hljs.registerLanguage("gherkin",function(hljs){return{aliases:["feature"],keywords:"Feature Background Ability Business Need Scenario Scenarios Scenario Outline Scenario Template Examples Given And Then But When",contains:[{className:"symbol",begin:"\\*",relevance:0},{className:"meta",begin:"@[^@\\s]+"},{begin:"\\|",end:"\\|\\w*$",contains:[{className:"string",begin:"[^|]+"}]},{className:"variable",begin:"<",end:">"},hljs.HASH_COMMENT_MODE,{className:"string",begin:'"""',end:'"""'},hljs.QUOTE_STRING_MODE]}});hljs.registerLanguage("subunit",function(hljs){var DETAILS={className:"string",begin:"\\[\n(multipart)?",end:"\\]\n"};var TIME={className:"string",begin:"\\d{4}-\\d{2}-\\d{2}(\\s+)\\d{2}:\\d{2}:\\d{2}.\\d+Z"};var PROGRESSVALUE={className:"string",begin:"(\\+|-)\\d+"};var KEYWORDS={className:"keyword",relevance:10,variants:[{begin:"^(test|testing|success|successful|failure|error|skip|xfail|uxsuccess)(:?)\\s+(test)?"},{begin:"^progress(:?)(\\s+)?(pop|push)?"},{begin:"^tags:"},{begin:"^time:"}]};return{case_insensitive:true,contains:[DETAILS,TIME,PROGRESSVALUE,KEYWORDS]}});hljs.registerLanguage("stata",function(hljs){return{aliases:["do","ado"],case_insensitive:true,keywords:"if else in foreach for forv forva forval forvalu forvalue forvalues by bys bysort xi quietly qui capture about ac ac_7 acprplot acprplot_7 adjust ado adopath adoupdate alpha ameans an ano anov anova anova_estat anova_terms anovadef aorder ap app appe appen append arch arch_dr arch_estat arch_p archlm areg areg_p args arima arima_dr arima_estat arima_p as asmprobit asmprobit_estat asmprobit_lf asmprobit_mfx__dlg asmprobit_p ass asse asser assert avplot avplot_7 avplots avplots_7 bcskew0 bgodfrey bias binreg bip0_lf biplot bipp_lf bipr_lf bipr_p biprobit bitest bitesti bitowt blogit bmemsize boot bootsamp bootstrap bootstrap_8 boxco_l boxco_p boxcox boxcox_6 boxcox_p bprobit br break brier bro brow brows browse brr brrstat bs bs_7 bsampl_w bsample bsample_7 bsqreg bstat bstat_7 bstat_8 bstrap bstrap_7 bubble bubbleplot ca ca_estat ca_p cabiplot camat canon canon_8 canon_8_p canon_estat canon_p cap caprojection capt captu captur capture cat cc cchart cchart_7 cci cd censobs_table centile cf char chdir checkdlgfiles checkestimationsample checkhlpfiles checksum chelp ci cii cl class classutil clear cli clis clist clo clog clog_lf clog_p clogi clogi_sw clogit clogit_lf clogit_p clogitp clogl_sw cloglog clonevar clslistarray cluster cluster_measures cluster_stop cluster_tree cluster_tree_8 clustermat cmdlog cnr cnre cnreg cnreg_p cnreg_sw cnsreg codebook collaps4 collapse colormult_nb colormult_nw compare compress conf confi confir confirm conren cons const constr constra constrai constrain constraint continue contract copy copyright copysource cor corc corr corr2data corr_anti corr_kmo corr_smc corre correl correla correlat correlate corrgram cou coun count cox cox_p cox_sw coxbase coxhaz coxvar cprplot cprplot_7 crc cret cretu cretur creturn cross cs cscript cscript_log csi ct ct_is ctset ctst_5 ctst_st cttost cumsp cumsp_7 cumul cusum cusum_7 cutil d|0 datasig datasign datasigna datasignat datasignatu datasignatur datasignature datetof db dbeta de dec deco decod decode deff des desc descr descri describ describe destring dfbeta dfgls dfuller di di_g dir dirstats dis discard disp disp_res disp_s displ displa display distinct do doe doed doedi doedit dotplot dotplot_7 dprobit drawnorm drop ds ds_util dstdize duplicates durbina dwstat dydx e|0 ed edi edit egen eivreg emdef en enc enco encod encode eq erase ereg ereg_lf ereg_p ereg_sw ereghet ereghet_glf ereghet_glf_sh ereghet_gp ereghet_ilf ereghet_ilf_sh ereghet_ip eret eretu eretur ereturn err erro error esize est est_cfexist est_cfname est_clickable est_expand est_hold est_table est_unhold est_unholdok estat estat_default estat_summ estat_vce_only esti estimates etodow etof etomdy ex exi exit expand expandcl fac fact facto factor factor_estat factor_p factor_pca_rotated factor_rotate factormat fcast fcast_compute fcast_graph fdades fdadesc fdadescr fdadescri fdadescrib fdadescribe fdasav fdasave fdause fh_st file open file read file close file filefilter fillin find_hlp_file findfile findit findit_7 fit fl fli flis flist for5_0 forest forestplot form forma format fpredict frac_154 frac_adj frac_chk frac_cox frac_ddp frac_dis frac_dv frac_in frac_mun frac_pp frac_pq frac_pv frac_wgt frac_xo fracgen fracplot fracplot_7 fracpoly fracpred fron_ex fron_hn fron_p fron_tn fron_tn2 frontier ftodate ftoe ftomdy ftowdate funnel funnelplot g|0 gamhet_glf gamhet_gp gamhet_ilf gamhet_ip gamma gamma_d2 gamma_p gamma_sw gammahet gdi_hexagon gdi_spokes ge gen gene gener genera generat generate genrank genstd genvmean gettoken gl gladder gladder_7 glim_l01 glim_l02 glim_l03 glim_l04 glim_l05 glim_l06 glim_l07 glim_l08 glim_l09 glim_l10 glim_l11 glim_l12 glim_lf glim_mu glim_nw1 glim_nw2 glim_nw3 glim_p glim_v1 glim_v2 glim_v3 glim_v4 glim_v5 glim_v6 glim_v7 glm glm_6 glm_p glm_sw glmpred glo glob globa global glogit glogit_8 glogit_p gmeans gnbre_lf gnbreg gnbreg_5 gnbreg_p gomp_lf gompe_sw gomper_p gompertz gompertzhet gomphet_glf gomphet_glf_sh gomphet_gp gomphet_ilf gomphet_ilf_sh gomphet_ip gphdot gphpen gphprint gprefs gprobi_p gprobit gprobit_8 gr gr7 gr_copy gr_current gr_db gr_describe gr_dir gr_draw gr_draw_replay gr_drop gr_edit gr_editviewopts gr_example gr_example2 gr_export gr_print gr_qscheme gr_query gr_read gr_rename gr_replay gr_save gr_set gr_setscheme gr_table gr_undo gr_use graph graph7 grebar greigen greigen_7 greigen_8 grmeanby grmeanby_7 gs_fileinfo gs_filetype gs_graphinfo gs_stat gsort gwood h|0 hadimvo hareg hausman haver he heck_d2 heckma_p heckman heckp_lf heckpr_p heckprob hel help hereg hetpr_lf hetpr_p hetprob hettest hexdump hilite hist hist_7 histogram hlogit hlu hmeans hotel hotelling hprobit hreg hsearch icd9 icd9_ff icd9p iis impute imtest inbase include inf infi infil infile infix inp inpu input ins insheet insp inspe inspec inspect integ inten intreg intreg_7 intreg_p intrg2_ll intrg_ll intrg_ll2 ipolate iqreg ir irf irf_create irfm iri is_svy is_svysum isid istdize ivprob_1_lf ivprob_lf ivprobit ivprobit_p ivreg ivreg_footnote ivtob_1_lf ivtob_lf ivtobit ivtobit_p jackknife jacknife jknife jknife_6 jknife_8 jkstat joinby kalarma1 kap kap_3 kapmeier kappa kapwgt kdensity kdensity_7 keep ksm ksmirnov ktau kwallis l|0 la lab labbe labbeplot labe label labelbook ladder levels levelsof leverage lfit lfit_p li lincom line linktest lis list lloghet_glf lloghet_glf_sh lloghet_gp lloghet_ilf lloghet_ilf_sh lloghet_ip llogi_sw llogis_p llogist llogistic llogistichet lnorm_lf lnorm_sw lnorma_p lnormal lnormalhet lnormhet_glf lnormhet_glf_sh lnormhet_gp lnormhet_ilf lnormhet_ilf_sh lnormhet_ip lnskew0 loadingplot loc loca local log logi logis_lf logistic logistic_p logit logit_estat logit_p loglogs logrank loneway lookfor lookup lowess lowess_7 lpredict lrecomp lroc lroc_7 lrtest ls lsens lsens_7 lsens_x lstat ltable ltable_7 ltriang lv lvr2plot lvr2plot_7 m|0 ma mac macr macro makecns man manova manova_estat manova_p manovatest mantel mark markin markout marksample mat mat_capp mat_order mat_put_rr mat_rapp mata mata_clear mata_describe mata_drop mata_matdescribe mata_matsave mata_matuse mata_memory mata_mlib mata_mosave mata_rename mata_which matalabel matcproc matlist matname matr matri matrix matrix_input__dlg matstrik mcc mcci md0_ md1_ md1debug_ md2_ md2debug_ mds mds_estat mds_p mdsconfig mdslong mdsmat mdsshepard mdytoe mdytof me_derd mean means median memory memsize menl meqparse mer merg merge meta mfp mfx mhelp mhodds minbound mixed_ll mixed_ll_reparm mkassert mkdir mkmat mkspline ml ml_5 ml_adjs ml_bhhhs ml_c_d ml_check ml_clear ml_cnt ml_debug ml_defd ml_e0 ml_e0_bfgs ml_e0_cycle ml_e0_dfp ml_e0i ml_e1 ml_e1_bfgs ml_e1_bhhh ml_e1_cycle ml_e1_dfp ml_e2 ml_e2_cycle ml_ebfg0 ml_ebfr0 ml_ebfr1 ml_ebh0q ml_ebhh0 ml_ebhr0 ml_ebr0i ml_ecr0i ml_edfp0 ml_edfr0 ml_edfr1 ml_edr0i ml_eds ml_eer0i ml_egr0i ml_elf ml_elf_bfgs ml_elf_bhhh ml_elf_cycle ml_elf_dfp ml_elfi ml_elfs ml_enr0i ml_enrr0 ml_erdu0 ml_erdu0_bfgs ml_erdu0_bhhh ml_erdu0_bhhhq ml_erdu0_cycle ml_erdu0_dfp ml_erdu0_nrbfgs ml_exde ml_footnote ml_geqnr ml_grad0 ml_graph ml_hbhhh ml_hd0 ml_hold ml_init ml_inv ml_log ml_max ml_mlout ml_mlout_8 ml_model ml_nb0 ml_opt ml_p ml_plot ml_query ml_rdgrd ml_repor ml_s_e ml_score ml_searc ml_technique ml_unhold mleval mlf_ mlmatbysum mlmatsum mlog mlogi mlogit mlogit_footnote mlogit_p mlopts mlsum mlvecsum mnl0_ mor more mov move mprobit mprobit_lf mprobit_p mrdu0_ mrdu1_ mvdecode mvencode mvreg mvreg_estat n|0 nbreg nbreg_al nbreg_lf nbreg_p nbreg_sw nestreg net newey newey_7 newey_p news nl nl_7 nl_9 nl_9_p nl_p nl_p_7 nlcom nlcom_p nlexp2 nlexp2_7 nlexp2a nlexp2a_7 nlexp3 nlexp3_7 nlgom3 nlgom3_7 nlgom4 nlgom4_7 nlinit nllog3 nllog3_7 nllog4 nllog4_7 nlog_rd nlogit nlogit_p nlogitgen nlogittree nlpred no nobreak noi nois noisi noisil noisily note notes notes_dlg nptrend numlabel numlist odbc old_ver olo olog ologi ologi_sw ologit ologit_p ologitp on one onew onewa oneway op_colnm op_comp op_diff op_inv op_str opr opro oprob oprob_sw oprobi oprobi_p oprobit oprobitp opts_exclusive order orthog orthpoly ou out outf outfi outfil outfile outs outsh outshe outshee outsheet ovtest pac pac_7 palette parse parse_dissim pause pca pca_8 pca_display pca_estat pca_p pca_rotate pcamat pchart pchart_7 pchi pchi_7 pcorr pctile pentium pergram pergram_7 permute permute_8 personal peto_st pkcollapse pkcross pkequiv pkexamine pkexamine_7 pkshape pksumm pksumm_7 pl plo plot plugin pnorm pnorm_7 poisgof poiss_lf poiss_sw poisso_p poisson poisson_estat post postclose postfile postutil pperron pr prais prais_e prais_e2 prais_p predict predictnl preserve print pro prob probi probit probit_estat probit_p proc_time procoverlay procrustes procrustes_estat procrustes_p profiler prog progr progra program prop proportion prtest prtesti pwcorr pwd q\\s qby qbys qchi qchi_7 qladder qladder_7 qnorm qnorm_7 qqplot qqplot_7 qreg qreg_c qreg_p qreg_sw qu quadchk quantile quantile_7 que quer query range ranksum ratio rchart rchart_7 rcof recast reclink recode reg reg3 reg3_p regdw regr regre regre_p2 regres regres_p regress regress_estat regriv_p remap ren rena renam rename renpfix repeat replace report reshape restore ret retu retur return rm rmdir robvar roccomp roccomp_7 roccomp_8 rocf_lf rocfit rocfit_8 rocgold rocplot rocplot_7 roctab roctab_7 rolling rologit rologit_p rot rota rotat rotate rotatemat rreg rreg_p ru run runtest rvfplot rvfplot_7 rvpplot rvpplot_7 sa safesum sample sampsi sav save savedresults saveold sc sca scal scala scalar scatter scm_mine sco scob_lf scob_p scobi_sw scobit scor score scoreplot scoreplot_help scree screeplot screeplot_help sdtest sdtesti se search separate seperate serrbar serrbar_7 serset set set_defaults sfrancia sh she shel shell shewhart shewhart_7 signestimationsample signrank signtest simul simul_7 simulate simulate_8 sktest sleep slogit slogit_d2 slogit_p smooth snapspan so sor sort spearman spikeplot spikeplot_7 spikeplt spline_x split sqreg sqreg_p sret sretu sretur sreturn ssc st st_ct st_hc st_hcd st_hcd_sh st_is st_issys st_note st_promo st_set st_show st_smpl st_subid stack statsby statsby_8 stbase stci stci_7 stcox stcox_estat stcox_fr stcox_fr_ll stcox_p stcox_sw stcoxkm stcoxkm_7 stcstat stcurv stcurve stcurve_7 stdes stem stepwise stereg stfill stgen stir stjoin stmc stmh stphplot stphplot_7 stphtest stphtest_7 stptime strate strate_7 streg streg_sw streset sts sts_7 stset stsplit stsum sttocc sttoct stvary stweib su suest suest_8 sum summ summa summar summari summariz summarize sunflower sureg survcurv survsum svar svar_p svmat svy svy_disp svy_dreg svy_est svy_est_7 svy_estat svy_get svy_gnbreg_p svy_head svy_header svy_heckman_p svy_heckprob_p svy_intreg_p svy_ivreg_p svy_logistic_p svy_logit_p svy_mlogit_p svy_nbreg_p svy_ologit_p svy_oprobit_p svy_poisson_p svy_probit_p svy_regress_p svy_sub svy_sub_7 svy_x svy_x_7 svy_x_p svydes svydes_8 svygen svygnbreg svyheckman svyheckprob svyintreg svyintreg_7 svyintrg svyivreg svylc svylog_p svylogit svymarkout svymarkout_8 svymean svymlog svymlogit svynbreg svyolog svyologit svyoprob svyoprobit svyopts svypois svypois_7 svypoisson svyprobit svyprobt svyprop svyprop_7 svyratio svyreg svyreg_p svyregress svyset svyset_7 svyset_8 svytab svytab_7 svytest svytotal sw sw_8 swcnreg swcox swereg swilk swlogis swlogit swologit swoprbt swpois swprobit swqreg swtobit swweib symmetry symmi symplot symplot_7 syntax sysdescribe sysdir sysuse szroeter ta tab tab1 tab2 tab_or tabd tabdi tabdis tabdisp tabi table tabodds tabodds_7 tabstat tabu tabul tabula tabulat tabulate te tempfile tempname tempvar tes test testnl testparm teststd tetrachoric time_it timer tis tob tobi tobit tobit_p tobit_sw token tokeni tokeniz tokenize tostring total translate translator transmap treat_ll treatr_p treatreg trim trimfill trnb_cons trnb_mean trpoiss_d2 trunc_ll truncr_p truncreg tsappend tset tsfill tsline tsline_ex tsreport tsrevar tsrline tsset tssmooth tsunab ttest ttesti tut_chk tut_wait tutorial tw tware_st two twoway twoway__fpfit_serset twoway__function_gen twoway__histogram_gen twoway__ipoint_serset twoway__ipoints_serset twoway__kdensity_gen twoway__lfit_serset twoway__normgen_gen twoway__pci_serset twoway__qfit_serset twoway__scatteri_serset twoway__sunflower_gen twoway_ksm_serset ty typ type typeof u|0 unab unabbrev unabcmd update us use uselabel var var_mkcompanion var_p varbasic varfcast vargranger varirf varirf_add varirf_cgraph varirf_create varirf_ctable varirf_describe varirf_dir varirf_drop varirf_erase varirf_graph varirf_ograph varirf_rename varirf_set varirf_table varlist varlmar varnorm varsoc varstable varstable_w varstable_w2 varwle vce vec vec_fevd vec_mkphi vec_p vec_p_w vecirf_create veclmar veclmar_w vecnorm vecnorm_w vecrank vecstable verinst vers versi versio version view viewsource vif vwls wdatetof webdescribe webseek webuse weib1_lf weib2_lf weib_lf weib_lf0 weibhet_glf weibhet_glf_sh weibhet_glfa weibhet_glfa_sh weibhet_gp weibhet_ilf weibhet_ilf_sh weibhet_ilfa weibhet_ilfa_sh weibhet_ip weibu_sw weibul_p weibull weibull_c weibull_s weibullhet wh whelp whi which whil while wilc_st wilcoxon win wind windo window winexec wntestb wntestb_7 wntestq xchart xchart_7 xcorr xcorr_7 xi xi_6 xmlsav xmlsave xmluse xpose xsh xshe xshel xshell xt_iis xt_tis xtab_p xtabond xtbin_p xtclog xtcloglog xtcloglog_8 xtcloglog_d2 xtcloglog_pa_p xtcloglog_re_p xtcnt_p xtcorr xtdata xtdes xtfront_p xtfrontier xtgee xtgee_elink xtgee_estat xtgee_makeivar xtgee_p xtgee_plink xtgls xtgls_p xthaus xthausman xtht_p xthtaylor xtile xtint_p xtintreg xtintreg_8 xtintreg_d2 xtintreg_p xtivp_1 xtivp_2 xtivreg xtline xtline_ex xtlogit xtlogit_8 xtlogit_d2 xtlogit_fe_p xtlogit_pa_p xtlogit_re_p xtmixed xtmixed_estat xtmixed_p xtnb_fe xtnb_lf xtnbreg xtnbreg_pa_p xtnbreg_refe_p xtpcse xtpcse_p xtpois xtpoisson xtpoisson_d2 xtpoisson_pa_p xtpoisson_refe_p xtpred xtprobit xtprobit_8 xtprobit_d2 xtprobit_re_p xtps_fe xtps_lf xtps_ren xtps_ren_8 xtrar_p xtrc xtrc_p xtrchh xtrefe_p xtreg xtreg_be xtreg_fe xtreg_ml xtreg_pa_p xtreg_re xtregar xtrere_p xtset xtsf_ll xtsf_llti xtsum xttab xttest0 xttobit xttobit_8 xttobit_p xttrans yx yxview__barlike_draw yxview_area_draw yxview_bar_draw yxview_dot_draw yxview_dropline_draw yxview_function_draw yxview_iarrow_draw yxview_ilabels_draw yxview_normal_draw yxview_pcarrow_draw yxview_pcbarrow_draw yxview_pccapsym_draw yxview_pcscatter_draw yxview_pcspike_draw yxview_rarea_draw yxview_rbar_draw yxview_rbarm_draw yxview_rcap_draw yxview_rcapsym_draw yxview_rconnected_draw yxview_rline_draw yxview_rscatter_draw yxview_rspike_draw yxview_spike_draw yxview_sunflower_draw zap_s zinb zinb_llf zinb_plf zip zip_llf zip_p zip_plf zt_ct_5 zt_hc_5 zt_hcd_5 zt_is_5 zt_iss_5 zt_sho_5 zt_smp_5 ztbase_5 ztcox_5 ztdes_5 ztereg_5 ztfill_5 ztgen_5 ztir_5 ztjoin_5 ztnb ztnb_p ztp ztp_p zts_5 ztset_5 ztspli_5 ztsum_5 zttoct_5 ztvary_5 ztweib_5",contains:[{className:"symbol",begin:/`[a-zA-Z0-9_]+'/},{className:"variable",begin:/\$\{?[a-zA-Z0-9_]+\}?/},{className:"string",variants:[{begin:'`"[^\r\n]*?"\''},{begin:'"[^\r\n"]*"'}]},{className:"built_in",variants:[{begin:"\\b(abs|acos|asin|atan|atan2|atanh|ceil|cloglog|comb|cos|digamma|exp|floor|invcloglog|invlogit|ln|lnfact|lnfactorial|lngamma|log|log10|max|min|mod|reldif|round|sign|sin|sqrt|sum|tan|tanh|trigamma|trunc|betaden|Binomial|binorm|binormal|chi2|chi2tail|dgammapda|dgammapdada|dgammapdadx|dgammapdx|dgammapdxdx|F|Fden|Ftail|gammaden|gammap|ibeta|invbinomial|invchi2|invchi2tail|invF|invFtail|invgammap|invibeta|invnchi2|invnFtail|invnibeta|invnorm|invnormal|invttail|nbetaden|nchi2|nFden|nFtail|nibeta|norm|normal|normalden|normd|npnchi2|tden|ttail|uniform|abbrev|char|index|indexnot|length|lower|ltrim|match|plural|proper|real|regexm|regexr|regexs|reverse|rtrim|string|strlen|strlower|strltrim|strmatch|strofreal|strpos|strproper|strreverse|strrtrim|strtrim|strupper|subinstr|subinword|substr|trim|upper|word|wordcount|_caller|autocode|byteorder|chop|clip|cond|e|epsdouble|epsfloat|group|inlist|inrange|irecode|matrix|maxbyte|maxdouble|maxfloat|maxint|maxlong|mi|minbyte|mindouble|minfloat|minint|minlong|missing|r|recode|replay|return|s|scalar|d|date|day|dow|doy|halfyear|mdy|month|quarter|week|year|d|daily|dofd|dofh|dofm|dofq|dofw|dofy|h|halfyearly|hofd|m|mofd|monthly|q|qofd|quarterly|tin|twithin|w|weekly|wofd|y|yearly|yh|ym|yofd|yq|yw|cholesky|colnumb|colsof|corr|det|diag|diag0cnt|el|get|hadamard|I|inv|invsym|issym|issymmetric|J|matmissing|matuniform|mreldif|nullmat|rownumb|rowsof|sweep|syminv|trace|vec|vecdiag)(?=\\()"}]},hljs.COMMENT("^[ \t]*\\*.*$",false),hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("scss",function(hljs){var AT_IDENTIFIER="@[a-z-]+";var AT_MODIFIERS="and or not only";var IDENT_RE="[a-zA-Z-][a-zA-Z0-9_-]*";var VARIABLE={className:"variable",begin:"(\\$"+IDENT_RE+")\\b"};var HEXCOLOR={className:"number",begin:"#[0-9A-Fa-f]+"};var DEF_INTERNALS={className:"attribute",begin:"[A-Z\\_\\.\\-]+",end:":",excludeEnd:true,illegal:"[^\\s]",starts:{endsWithParent:true,excludeEnd:true,contains:[HEXCOLOR,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"!important"}]}};return{case_insensitive:true,illegal:"[=/|']",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"selector-id",begin:"\\#[A-Za-z0-9_-]+",relevance:0},{className:"selector-class",begin:"\\.[A-Za-z0-9_-]+",relevance:0},{className:"selector-attr",begin:"\\[",end:"\\]",illegal:"$"},{className:"selector-tag",begin:"\\b(a|abbr|acronym|address|area|article|aside|audio|b|base|big|blockquote|body|br|button|canvas|caption|cite|code|col|colgroup|command|datalist|dd|del|details|dfn|div|dl|dt|em|embed|fieldset|figcaption|figure|footer|form|frame|frameset|(h[1-6])|head|header|hgroup|hr|html|i|iframe|img|input|ins|kbd|keygen|label|legend|li|link|map|mark|meta|meter|nav|noframes|noscript|object|ol|optgroup|option|output|p|param|pre|progress|q|rp|rt|ruby|samp|script|section|select|small|span|strike|strong|style|sub|sup|table|tbody|td|textarea|tfoot|th|thead|time|title|tr|tt|ul|var|video)\\b",relevance:0},{className:"selector-pseudo",begin:":(visited|valid|root|right|required|read-write|read-only|out-range|optional|only-of-type|only-child|nth-of-type|nth-last-of-type|nth-last-child|nth-child|not|link|left|last-of-type|last-child|lang|invalid|indeterminate|in-range|hover|focus|first-of-type|first-line|first-letter|first-child|first|enabled|empty|disabled|default|checked|before|after|active)"},{className:"selector-pseudo",begin:"::(after|before|choices|first-letter|first-line|repeat-index|repeat-item|selection|value)"},VARIABLE,{className:"attribute",begin:"\\b(src|z-index|word-wrap|word-spacing|word-break|width|widows|white-space|visibility|vertical-align|unicode-bidi|transition-timing-function|transition-property|transition-duration|transition-delay|transition|transform-style|transform-origin|transform|top|text-underline-position|text-transform|text-shadow|text-rendering|text-overflow|text-indent|text-decoration-style|text-decoration-line|text-decoration-color|text-decoration|text-align-last|text-align|tab-size|table-layout|right|resize|quotes|position|pointer-events|perspective-origin|perspective|page-break-inside|page-break-before|page-break-after|padding-top|padding-right|padding-left|padding-bottom|padding|overflow-y|overflow-x|overflow-wrap|overflow|outline-width|outline-style|outline-offset|outline-color|outline|orphans|order|opacity|object-position|object-fit|normal|none|nav-up|nav-right|nav-left|nav-index|nav-down|min-width|min-height|max-width|max-height|mask|marks|margin-top|margin-right|margin-left|margin-bottom|margin|list-style-type|list-style-position|list-style-image|list-style|line-height|letter-spacing|left|justify-content|initial|inherit|ime-mode|image-orientation|image-resolution|image-rendering|icon|hyphens|height|font-weight|font-variant-ligatures|font-variant|font-style|font-stretch|font-size-adjust|font-size|font-language-override|font-kerning|font-feature-settings|font-family|font|float|flex-wrap|flex-shrink|flex-grow|flex-flow|flex-direction|flex-basis|flex|filter|empty-cells|display|direction|cursor|counter-reset|counter-increment|content|column-width|column-span|column-rule-width|column-rule-style|column-rule-color|column-rule|column-gap|column-fill|column-count|columns|color|clip-path|clip|clear|caption-side|break-inside|break-before|break-after|box-sizing|box-shadow|box-decoration-break|bottom|border-width|border-top-width|border-top-style|border-top-right-radius|border-top-left-radius|border-top-color|border-top|border-style|border-spacing|border-right-width|border-right-style|border-right-color|border-right|border-radius|border-left-width|border-left-style|border-left-color|border-left|border-image-width|border-image-source|border-image-slice|border-image-repeat|border-image-outset|border-image|border-color|border-collapse|border-bottom-width|border-bottom-style|border-bottom-right-radius|border-bottom-left-radius|border-bottom-color|border-bottom|border|background-size|background-repeat|background-position|background-origin|background-image|background-color|background-clip|background-attachment|background-blend-mode|background|backface-visibility|auto|animation-timing-function|animation-play-state|animation-name|animation-iteration-count|animation-fill-mode|animation-duration|animation-direction|animation-delay|animation|align-self|align-items|align-content)\\b",illegal:"[^\\s]"},{begin:"\\b(whitespace|wait|w-resize|visible|vertical-text|vertical-ideographic|uppercase|upper-roman|upper-alpha|underline|transparent|top|thin|thick|text|text-top|text-bottom|tb-rl|table-header-group|table-footer-group|sw-resize|super|strict|static|square|solid|small-caps|separate|se-resize|scroll|s-resize|rtl|row-resize|ridge|right|repeat|repeat-y|repeat-x|relative|progress|pointer|overline|outside|outset|oblique|nowrap|not-allowed|normal|none|nw-resize|no-repeat|no-drop|newspaper|ne-resize|n-resize|move|middle|medium|ltr|lr-tb|lowercase|lower-roman|lower-alpha|loose|list-item|line|line-through|line-edge|lighter|left|keep-all|justify|italic|inter-word|inter-ideograph|inside|inset|inline|inline-block|inherit|inactive|ideograph-space|ideograph-parenthesis|ideograph-numeric|ideograph-alpha|horizontal|hidden|help|hand|groove|fixed|ellipsis|e-resize|double|dotted|distribute|distribute-space|distribute-letter|distribute-all-lines|disc|disabled|default|decimal|dashed|crosshair|collapse|col-resize|circle|char|center|capitalize|break-word|break-all|bottom|both|bolder|bold|block|bidi-override|below|baseline|auto|always|all-scroll|absolute|table|table-cell)\\b"},{begin:":",end:";",contains:[VARIABLE,HEXCOLOR,hljs.CSS_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"meta",begin:"!important"}]},{begin:"@(page|font-face)",lexemes:AT_IDENTIFIER,keywords:"@page @font-face"},{begin:"@",end:"[{;]",returnBegin:true,keywords:AT_MODIFIERS,contains:[{begin:AT_IDENTIFIER,className:"keyword"},VARIABLE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,HEXCOLOR,hljs.CSS_NUMBER_MODE]}]}});hljs.registerLanguage("tcl",function(hljs){return{aliases:["tk"],keywords:"after append apply array auto_execok auto_import auto_load auto_mkindex "+"auto_mkindex_old auto_qualify auto_reset bgerror binary break catch cd chan clock "+"close concat continue dde dict encoding eof error eval exec exit expr fblocked "+"fconfigure fcopy file fileevent filename flush for foreach format gets glob global "+"history http if incr info interp join lappend|10 lassign|10 lindex|10 linsert|10 list "+"llength|10 load lrange|10 lrepeat|10 lreplace|10 lreverse|10 lsearch|10 lset|10 lsort|10 "+"mathfunc mathop memory msgcat namespace open package parray pid pkg::create pkg_mkIndex "+"platform platform::shell proc puts pwd read refchan regexp registry regsub|10 rename "+"return safe scan seek set socket source split string subst switch tcl_endOfWord "+"tcl_findLibrary tcl_startOfNextWord tcl_startOfPreviousWord tcl_wordBreakAfter "+"tcl_wordBreakBefore tcltest tclvars tell time tm trace unknown unload unset update "+"uplevel upvar variable vwait while",contains:[hljs.COMMENT(";[ \\t]*#","$"),hljs.COMMENT("^[ \\t]*#","$"),{beginKeywords:"proc",end:"[\\{]",excludeEnd:true,contains:[{className:"title",begin:"[ \\t\\n\\r]+(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*",end:"[ \\t\\n\\r]",endsWithParent:true,excludeEnd:true}]},{excludeEnd:true,variants:[{begin:"\\$(\\{)?(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*\\(([a-zA-Z0-9_])*\\)",end:"[^a-zA-Z0-9_\\}\\$]"},{begin:"\\$(\\{)?(::)?[a-zA-Z_]((::)?[a-zA-Z0-9_])*",end:"(\\))?[^a-zA-Z0-9_\\}\\$]"}]},{className:"string",contains:[hljs.BACKSLASH_ESCAPE],variants:[hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null})]},{className:"number",variants:[hljs.BINARY_NUMBER_MODE,hljs.C_NUMBER_MODE]}]}});hljs.registerLanguage("pgsql",function(hljs){var COMMENT_MODE=hljs.COMMENT("--","$");var UNQUOTED_IDENT="[a-zA-Z_][a-zA-Z_0-9$]*";var DOLLAR_STRING="\\$([a-zA-Z_]?|[a-zA-Z_][a-zA-Z_0-9]*)\\$";var LABEL="<<\\s*"+UNQUOTED_IDENT+"\\s*>>";var SQL_KW="ABORT ALTER ANALYZE BEGIN CALL CHECKPOINT|10 CLOSE CLUSTER COMMENT COMMIT COPY CREATE DEALLOCATE DECLARE "+"DELETE DISCARD DO DROP END EXECUTE EXPLAIN FETCH GRANT IMPORT INSERT LISTEN LOAD LOCK MOVE NOTIFY "+"PREPARE REASSIGN|10 REFRESH REINDEX RELEASE RESET REVOKE ROLLBACK SAVEPOINT SECURITY SELECT SET SHOW "+"START TRUNCATE UNLISTEN|10 UPDATE VACUUM|10 VALUES "+"AGGREGATE COLLATION CONVERSION|10 DATABASE DEFAULT PRIVILEGES DOMAIN TRIGGER EXTENSION FOREIGN "+"WRAPPER|10 TABLE FUNCTION GROUP LANGUAGE LARGE OBJECT MATERIALIZED VIEW OPERATOR CLASS "+"FAMILY POLICY PUBLICATION|10 ROLE RULE SCHEMA SEQUENCE SERVER STATISTICS SUBSCRIPTION SYSTEM "+"TABLESPACE CONFIGURATION DICTIONARY PARSER TEMPLATE TYPE USER MAPPING PREPARED ACCESS "+"METHOD CAST AS TRANSFORM TRANSACTION OWNED TO INTO SESSION AUTHORIZATION "+"INDEX PROCEDURE ASSERTION "+"ALL ANALYSE AND ANY ARRAY ASC ASYMMETRIC|10 BOTH CASE CHECK "+"COLLATE COLUMN CONCURRENTLY|10 CONSTRAINT CROSS "+"DEFERRABLE RANGE "+"DESC DISTINCT ELSE EXCEPT FOR FREEZE|10 FROM FULL HAVING "+"ILIKE IN INITIALLY INNER INTERSECT IS ISNULL JOIN LATERAL LEADING LIKE LIMIT "+"NATURAL NOT NOTNULL NULL OFFSET ON ONLY OR ORDER OUTER OVERLAPS PLACING PRIMARY "+"REFERENCES RETURNING SIMILAR SOME SYMMETRIC TABLESAMPLE THEN "+"TRAILING UNION UNIQUE USING VARIADIC|10 VERBOSE WHEN WHERE WINDOW WITH "+"BY RETURNS INOUT OUT SETOF|10 IF STRICT CURRENT CONTINUE OWNER LOCATION OVER PARTITION WITHIN "+"BETWEEN ESCAPE EXTERNAL INVOKER DEFINER WORK RENAME VERSION CONNECTION CONNECT "+"TABLES TEMP TEMPORARY FUNCTIONS SEQUENCES TYPES SCHEMAS OPTION CASCADE RESTRICT ADD ADMIN "+"EXISTS VALID VALIDATE ENABLE DISABLE REPLICA|10 ALWAYS PASSING COLUMNS PATH "+"REF VALUE OVERRIDING IMMUTABLE STABLE VOLATILE BEFORE AFTER EACH ROW PROCEDURAL "+"ROUTINE NO HANDLER VALIDATOR OPTIONS STORAGE OIDS|10 WITHOUT INHERIT DEPENDS CALLED "+"INPUT LEAKPROOF|10 COST ROWS NOWAIT SEARCH UNTIL ENCRYPTED|10 PASSWORD CONFLICT|10 "+"INSTEAD INHERITS CHARACTERISTICS WRITE CURSOR ALSO STATEMENT SHARE EXCLUSIVE INLINE "+"ISOLATION REPEATABLE READ COMMITTED SERIALIZABLE UNCOMMITTED LOCAL GLOBAL SQL PROCEDURES "+"RECURSIVE SNAPSHOT ROLLUP CUBE TRUSTED|10 INCLUDE FOLLOWING PRECEDING UNBOUNDED RANGE GROUPS "+"UNENCRYPTED|10 SYSID FORMAT DELIMITER HEADER QUOTE ENCODING FILTER OFF "+"FORCE_QUOTE FORCE_NOT_NULL FORCE_NULL COSTS BUFFERS TIMING SUMMARY DISABLE_PAGE_SKIPPING "+"RESTART CYCLE GENERATED IDENTITY DEFERRED IMMEDIATE LEVEL LOGGED UNLOGGED "+"OF NOTHING NONE EXCLUDE ATTRIBUTE "+"USAGE ROUTINES "+"TRUE FALSE NAN INFINITY ";var ROLE_ATTRS="SUPERUSER NOSUPERUSER CREATEDB NOCREATEDB CREATEROLE NOCREATEROLE INHERIT NOINHERIT "+"LOGIN NOLOGIN REPLICATION NOREPLICATION BYPASSRLS NOBYPASSRLS ";var PLPGSQL_KW="ALIAS BEGIN CONSTANT DECLARE END EXCEPTION RETURN PERFORM|10 RAISE GET DIAGNOSTICS "+"STACKED|10 FOREACH LOOP ELSIF EXIT WHILE REVERSE SLICE DEBUG LOG INFO NOTICE WARNING ASSERT "+"OPEN ";var TYPES="BIGINT INT8 BIGSERIAL SERIAL8 BIT VARYING VARBIT BOOLEAN BOOL BOX BYTEA CHARACTER CHAR VARCHAR "+"CIDR CIRCLE DATE DOUBLE PRECISION FLOAT8 FLOAT INET INTEGER INT INT4 INTERVAL JSON JSONB LINE LSEG|10 "+"MACADDR MACADDR8 MONEY NUMERIC DEC DECIMAL PATH POINT POLYGON REAL FLOAT4 SMALLINT INT2 "+"SMALLSERIAL|10 SERIAL2|10 SERIAL|10 SERIAL4|10 TEXT TIME ZONE TIMETZ|10 TIMESTAMP TIMESTAMPTZ|10 TSQUERY|10 TSVECTOR|10 "+"TXID_SNAPSHOT|10 UUID XML NATIONAL NCHAR "+"INT4RANGE|10 INT8RANGE|10 NUMRANGE|10 TSRANGE|10 TSTZRANGE|10 DATERANGE|10 "+"ANYELEMENT ANYARRAY ANYNONARRAY ANYENUM ANYRANGE CSTRING INTERNAL "+"RECORD PG_DDL_COMMAND VOID UNKNOWN OPAQUE REFCURSOR "+"NAME "+"OID REGPROC|10 REGPROCEDURE|10 REGOPER|10 REGOPERATOR|10 REGCLASS|10 REGTYPE|10 REGROLE|10 "+"REGNAMESPACE|10 REGCONFIG|10 REGDICTIONARY|10 ";"HSTORE|10 LO LTREE|10 ";var TYPES_RE=TYPES.trim().split(" ").map(function(val){return val.split("|")[0]}).join("|");var SQL_BI="CURRENT_TIME CURRENT_TIMESTAMP CURRENT_USER CURRENT_CATALOG|10 CURRENT_DATE LOCALTIME LOCALTIMESTAMP "+"CURRENT_ROLE|10 CURRENT_SCHEMA|10 SESSION_USER PUBLIC ";var PLPGSQL_BI="FOUND NEW OLD TG_NAME|10 TG_WHEN|10 TG_LEVEL|10 TG_OP|10 TG_RELID|10 TG_RELNAME|10 "+"TG_TABLE_NAME|10 TG_TABLE_SCHEMA|10 TG_NARGS|10 TG_ARGV|10 TG_EVENT|10 TG_TAG|10 "+"ROW_COUNT RESULT_OID|10 PG_CONTEXT|10 RETURNED_SQLSTATE COLUMN_NAME CONSTRAINT_NAME "+"PG_DATATYPE_NAME|10 MESSAGE_TEXT TABLE_NAME SCHEMA_NAME PG_EXCEPTION_DETAIL|10 "+"PG_EXCEPTION_HINT|10 PG_EXCEPTION_CONTEXT|10 ";var PLPGSQL_EXCEPTIONS="SQLSTATE SQLERRM|10 "+"SUCCESSFUL_COMPLETION WARNING DYNAMIC_RESULT_SETS_RETURNED IMPLICIT_ZERO_BIT_PADDING "+"NULL_VALUE_ELIMINATED_IN_SET_FUNCTION PRIVILEGE_NOT_GRANTED PRIVILEGE_NOT_REVOKED "+"STRING_DATA_RIGHT_TRUNCATION DEPRECATED_FEATURE NO_DATA NO_ADDITIONAL_DYNAMIC_RESULT_SETS_RETURNED "+"SQL_STATEMENT_NOT_YET_COMPLETE CONNECTION_EXCEPTION CONNECTION_DOES_NOT_EXIST CONNECTION_FAILURE "+"SQLCLIENT_UNABLE_TO_ESTABLISH_SQLCONNECTION SQLSERVER_REJECTED_ESTABLISHMENT_OF_SQLCONNECTION "+"TRANSACTION_RESOLUTION_UNKNOWN PROTOCOL_VIOLATION TRIGGERED_ACTION_EXCEPTION FEATURE_NOT_SUPPORTED "+"INVALID_TRANSACTION_INITIATION LOCATOR_EXCEPTION INVALID_LOCATOR_SPECIFICATION INVALID_GRANTOR "+"INVALID_GRANT_OPERATION INVALID_ROLE_SPECIFICATION DIAGNOSTICS_EXCEPTION "+"STACKED_DIAGNOSTICS_ACCESSED_WITHOUT_ACTIVE_HANDLER CASE_NOT_FOUND CARDINALITY_VIOLATION "+"DATA_EXCEPTION ARRAY_SUBSCRIPT_ERROR CHARACTER_NOT_IN_REPERTOIRE DATETIME_FIELD_OVERFLOW "+"DIVISION_BY_ZERO ERROR_IN_ASSIGNMENT ESCAPE_CHARACTER_CONFLICT INDICATOR_OVERFLOW "+"INTERVAL_FIELD_OVERFLOW INVALID_ARGUMENT_FOR_LOGARITHM INVALID_ARGUMENT_FOR_NTILE_FUNCTION "+"INVALID_ARGUMENT_FOR_NTH_VALUE_FUNCTION INVALID_ARGUMENT_FOR_POWER_FUNCTION "+"INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION INVALID_CHARACTER_VALUE_FOR_CAST "+"INVALID_DATETIME_FORMAT INVALID_ESCAPE_CHARACTER INVALID_ESCAPE_OCTET INVALID_ESCAPE_SEQUENCE "+"NONSTANDARD_USE_OF_ESCAPE_CHARACTER INVALID_INDICATOR_PARAMETER_VALUE INVALID_PARAMETER_VALUE "+"INVALID_REGULAR_EXPRESSION INVALID_ROW_COUNT_IN_LIMIT_CLAUSE "+"INVALID_ROW_COUNT_IN_RESULT_OFFSET_CLAUSE INVALID_TABLESAMPLE_ARGUMENT INVALID_TABLESAMPLE_REPEAT "+"INVALID_TIME_ZONE_DISPLACEMENT_VALUE INVALID_USE_OF_ESCAPE_CHARACTER MOST_SPECIFIC_TYPE_MISMATCH "+"NULL_VALUE_NOT_ALLOWED NULL_VALUE_NO_INDICATOR_PARAMETER NUMERIC_VALUE_OUT_OF_RANGE "+"SEQUENCE_GENERATOR_LIMIT_EXCEEDED STRING_DATA_LENGTH_MISMATCH STRING_DATA_RIGHT_TRUNCATION "+"SUBSTRING_ERROR TRIM_ERROR UNTERMINATED_C_STRING ZERO_LENGTH_CHARACTER_STRING "+"FLOATING_POINT_EXCEPTION INVALID_TEXT_REPRESENTATION INVALID_BINARY_REPRESENTATION "+"BAD_COPY_FILE_FORMAT UNTRANSLATABLE_CHARACTER NOT_AN_XML_DOCUMENT INVALID_XML_DOCUMENT "+"INVALID_XML_CONTENT INVALID_XML_COMMENT INVALID_XML_PROCESSING_INSTRUCTION "+"INTEGRITY_CONSTRAINT_VIOLATION RESTRICT_VIOLATION NOT_NULL_VIOLATION FOREIGN_KEY_VIOLATION "+"UNIQUE_VIOLATION CHECK_VIOLATION EXCLUSION_VIOLATION INVALID_CURSOR_STATE "+"INVALID_TRANSACTION_STATE ACTIVE_SQL_TRANSACTION BRANCH_TRANSACTION_ALREADY_ACTIVE "+"HELD_CURSOR_REQUIRES_SAME_ISOLATION_LEVEL INAPPROPRIATE_ACCESS_MODE_FOR_BRANCH_TRANSACTION "+"INAPPROPRIATE_ISOLATION_LEVEL_FOR_BRANCH_TRANSACTION "+"NO_ACTIVE_SQL_TRANSACTION_FOR_BRANCH_TRANSACTION READ_ONLY_SQL_TRANSACTION "+"SCHEMA_AND_DATA_STATEMENT_MIXING_NOT_SUPPORTED NO_ACTIVE_SQL_TRANSACTION "+"IN_FAILED_SQL_TRANSACTION IDLE_IN_TRANSACTION_SESSION_TIMEOUT INVALID_SQL_STATEMENT_NAME "+"TRIGGERED_DATA_CHANGE_VIOLATION INVALID_AUTHORIZATION_SPECIFICATION INVALID_PASSWORD "+"DEPENDENT_PRIVILEGE_DESCRIPTORS_STILL_EXIST DEPENDENT_OBJECTS_STILL_EXIST "+"INVALID_TRANSACTION_TERMINATION SQL_ROUTINE_EXCEPTION FUNCTION_EXECUTED_NO_RETURN_STATEMENT "+"MODIFYING_SQL_DATA_NOT_PERMITTED PROHIBITED_SQL_STATEMENT_ATTEMPTED "+"READING_SQL_DATA_NOT_PERMITTED INVALID_CURSOR_NAME EXTERNAL_ROUTINE_EXCEPTION "+"CONTAINING_SQL_NOT_PERMITTED MODIFYING_SQL_DATA_NOT_PERMITTED "+"PROHIBITED_SQL_STATEMENT_ATTEMPTED READING_SQL_DATA_NOT_PERMITTED "+"EXTERNAL_ROUTINE_INVOCATION_EXCEPTION INVALID_SQLSTATE_RETURNED NULL_VALUE_NOT_ALLOWED "+"TRIGGER_PROTOCOL_VIOLATED SRF_PROTOCOL_VIOLATED EVENT_TRIGGER_PROTOCOL_VIOLATED "+"SAVEPOINT_EXCEPTION INVALID_SAVEPOINT_SPECIFICATION INVALID_CATALOG_NAME "+"INVALID_SCHEMA_NAME TRANSACTION_ROLLBACK TRANSACTION_INTEGRITY_CONSTRAINT_VIOLATION "+"SERIALIZATION_FAILURE STATEMENT_COMPLETION_UNKNOWN DEADLOCK_DETECTED "+"SYNTAX_ERROR_OR_ACCESS_RULE_VIOLATION SYNTAX_ERROR INSUFFICIENT_PRIVILEGE CANNOT_COERCE "+"GROUPING_ERROR WINDOWING_ERROR INVALID_RECURSION INVALID_FOREIGN_KEY INVALID_NAME "+"NAME_TOO_LONG RESERVED_NAME DATATYPE_MISMATCH INDETERMINATE_DATATYPE COLLATION_MISMATCH "+"INDETERMINATE_COLLATION WRONG_OBJECT_TYPE GENERATED_ALWAYS UNDEFINED_COLUMN "+"UNDEFINED_FUNCTION UNDEFINED_TABLE UNDEFINED_PARAMETER UNDEFINED_OBJECT "+"DUPLICATE_COLUMN DUPLICATE_CURSOR DUPLICATE_DATABASE DUPLICATE_FUNCTION "+"DUPLICATE_PREPARED_STATEMENT DUPLICATE_SCHEMA DUPLICATE_TABLE DUPLICATE_ALIAS "+"DUPLICATE_OBJECT AMBIGUOUS_COLUMN AMBIGUOUS_FUNCTION AMBIGUOUS_PARAMETER AMBIGUOUS_ALIAS "+"INVALID_COLUMN_REFERENCE INVALID_COLUMN_DEFINITION INVALID_CURSOR_DEFINITION "+"INVALID_DATABASE_DEFINITION INVALID_FUNCTION_DEFINITION "+"INVALID_PREPARED_STATEMENT_DEFINITION INVALID_SCHEMA_DEFINITION INVALID_TABLE_DEFINITION "+"INVALID_OBJECT_DEFINITION WITH_CHECK_OPTION_VIOLATION INSUFFICIENT_RESOURCES DISK_FULL "+"OUT_OF_MEMORY TOO_MANY_CONNECTIONS CONFIGURATION_LIMIT_EXCEEDED PROGRAM_LIMIT_EXCEEDED "+"STATEMENT_TOO_COMPLEX TOO_MANY_COLUMNS TOO_MANY_ARGUMENTS OBJECT_NOT_IN_PREREQUISITE_STATE "+"OBJECT_IN_USE CANT_CHANGE_RUNTIME_PARAM LOCK_NOT_AVAILABLE OPERATOR_INTERVENTION "+"QUERY_CANCELED ADMIN_SHUTDOWN CRASH_SHUTDOWN CANNOT_CONNECT_NOW DATABASE_DROPPED "+"SYSTEM_ERROR IO_ERROR UNDEFINED_FILE DUPLICATE_FILE SNAPSHOT_TOO_OLD CONFIG_FILE_ERROR "+"LOCK_FILE_EXISTS FDW_ERROR FDW_COLUMN_NAME_NOT_FOUND FDW_DYNAMIC_PARAMETER_VALUE_NEEDED "+"FDW_FUNCTION_SEQUENCE_ERROR FDW_INCONSISTENT_DESCRIPTOR_INFORMATION "+"FDW_INVALID_ATTRIBUTE_VALUE FDW_INVALID_COLUMN_NAME FDW_INVALID_COLUMN_NUMBER "+"FDW_INVALID_DATA_TYPE FDW_INVALID_DATA_TYPE_DESCRIPTORS "+"FDW_INVALID_DESCRIPTOR_FIELD_IDENTIFIER FDW_INVALID_HANDLE FDW_INVALID_OPTION_INDEX "+"FDW_INVALID_OPTION_NAME FDW_INVALID_STRING_LENGTH_OR_BUFFER_LENGTH "+"FDW_INVALID_STRING_FORMAT FDW_INVALID_USE_OF_NULL_POINTER FDW_TOO_MANY_HANDLES "+"FDW_OUT_OF_MEMORY FDW_NO_SCHEMAS FDW_OPTION_NAME_NOT_FOUND FDW_REPLY_HANDLE "+"FDW_SCHEMA_NOT_FOUND FDW_TABLE_NOT_FOUND FDW_UNABLE_TO_CREATE_EXECUTION "+"FDW_UNABLE_TO_CREATE_REPLY FDW_UNABLE_TO_ESTABLISH_CONNECTION PLPGSQL_ERROR "+"RAISE_EXCEPTION NO_DATA_FOUND TOO_MANY_ROWS ASSERT_FAILURE INTERNAL_ERROR DATA_CORRUPTED "+"INDEX_CORRUPTED ";var FUNCTIONS="ARRAY_AGG AVG BIT_AND BIT_OR BOOL_AND BOOL_OR COUNT EVERY JSON_AGG JSONB_AGG JSON_OBJECT_AGG "+"JSONB_OBJECT_AGG MAX MIN MODE STRING_AGG SUM XMLAGG "+"CORR COVAR_POP COVAR_SAMP REGR_AVGX REGR_AVGY REGR_COUNT REGR_INTERCEPT REGR_R2 REGR_SLOPE "+"REGR_SXX REGR_SXY REGR_SYY STDDEV STDDEV_POP STDDEV_SAMP VARIANCE VAR_POP VAR_SAMP "+"PERCENTILE_CONT PERCENTILE_DISC "+"ROW_NUMBER RANK DENSE_RANK PERCENT_RANK CUME_DIST NTILE LAG LEAD FIRST_VALUE LAST_VALUE NTH_VALUE "+"NUM_NONNULLS NUM_NULLS "+"ABS CBRT CEIL CEILING DEGREES DIV EXP FLOOR LN LOG MOD PI POWER RADIANS ROUND SCALE SIGN SQRT "+"TRUNC WIDTH_BUCKET "+"RANDOM SETSEED "+"ACOS ACOSD ASIN ASIND ATAN ATAND ATAN2 ATAN2D COS COSD COT COTD SIN SIND TAN TAND "+"BIT_LENGTH CHAR_LENGTH CHARACTER_LENGTH LOWER OCTET_LENGTH OVERLAY POSITION SUBSTRING TREAT TRIM UPPER "+"ASCII BTRIM CHR CONCAT CONCAT_WS CONVERT CONVERT_FROM CONVERT_TO DECODE ENCODE INITCAP"+"LEFT LENGTH LPAD LTRIM MD5 PARSE_IDENT PG_CLIENT_ENCODING QUOTE_IDENT|10 QUOTE_LITERAL|10 "+"QUOTE_NULLABLE|10 REGEXP_MATCH REGEXP_MATCHES REGEXP_REPLACE REGEXP_SPLIT_TO_ARRAY "+"REGEXP_SPLIT_TO_TABLE REPEAT REPLACE REVERSE RIGHT RPAD RTRIM SPLIT_PART STRPOS SUBSTR "+"TO_ASCII TO_HEX TRANSLATE "+"OCTET_LENGTH GET_BIT GET_BYTE SET_BIT SET_BYTE "+"TO_CHAR TO_DATE TO_NUMBER TO_TIMESTAMP "+"AGE CLOCK_TIMESTAMP|10 DATE_PART DATE_TRUNC ISFINITE JUSTIFY_DAYS JUSTIFY_HOURS JUSTIFY_INTERVAL "+"MAKE_DATE MAKE_INTERVAL|10 MAKE_TIME MAKE_TIMESTAMP|10 MAKE_TIMESTAMPTZ|10 NOW STATEMENT_TIMESTAMP|10 "+"TIMEOFDAY TRANSACTION_TIMESTAMP|10 "+"ENUM_FIRST ENUM_LAST ENUM_RANGE "+"AREA CENTER DIAMETER HEIGHT ISCLOSED ISOPEN NPOINTS PCLOSE POPEN RADIUS WIDTH "+"BOX BOUND_BOX CIRCLE LINE LSEG PATH POLYGON "+"ABBREV BROADCAST HOST HOSTMASK MASKLEN NETMASK NETWORK SET_MASKLEN TEXT INET_SAME_FAMILY"+"INET_MERGE MACADDR8_SET7BIT "+"ARRAY_TO_TSVECTOR GET_CURRENT_TS_CONFIG NUMNODE PLAINTO_TSQUERY PHRASETO_TSQUERY WEBSEARCH_TO_TSQUERY "+"QUERYTREE SETWEIGHT STRIP TO_TSQUERY TO_TSVECTOR JSON_TO_TSVECTOR JSONB_TO_TSVECTOR TS_DELETE "+"TS_FILTER TS_HEADLINE TS_RANK TS_RANK_CD TS_REWRITE TSQUERY_PHRASE TSVECTOR_TO_ARRAY "+"TSVECTOR_UPDATE_TRIGGER TSVECTOR_UPDATE_TRIGGER_COLUMN "+"XMLCOMMENT XMLCONCAT XMLELEMENT XMLFOREST XMLPI XMLROOT "+"XMLEXISTS XML_IS_WELL_FORMED XML_IS_WELL_FORMED_DOCUMENT XML_IS_WELL_FORMED_CONTENT "+"XPATH XPATH_EXISTS XMLTABLE XMLNAMESPACES "+"TABLE_TO_XML TABLE_TO_XMLSCHEMA TABLE_TO_XML_AND_XMLSCHEMA "+"QUERY_TO_XML QUERY_TO_XMLSCHEMA QUERY_TO_XML_AND_XMLSCHEMA "+"CURSOR_TO_XML CURSOR_TO_XMLSCHEMA "+"SCHEMA_TO_XML SCHEMA_TO_XMLSCHEMA SCHEMA_TO_XML_AND_XMLSCHEMA "+"DATABASE_TO_XML DATABASE_TO_XMLSCHEMA DATABASE_TO_XML_AND_XMLSCHEMA "+"XMLATTRIBUTES "+"TO_JSON TO_JSONB ARRAY_TO_JSON ROW_TO_JSON JSON_BUILD_ARRAY JSONB_BUILD_ARRAY JSON_BUILD_OBJECT "+"JSONB_BUILD_OBJECT JSON_OBJECT JSONB_OBJECT JSON_ARRAY_LENGTH JSONB_ARRAY_LENGTH JSON_EACH "+"JSONB_EACH JSON_EACH_TEXT JSONB_EACH_TEXT JSON_EXTRACT_PATH JSONB_EXTRACT_PATH "+"JSON_OBJECT_KEYS JSONB_OBJECT_KEYS JSON_POPULATE_RECORD JSONB_POPULATE_RECORD JSON_POPULATE_RECORDSET "+"JSONB_POPULATE_RECORDSET JSON_ARRAY_ELEMENTS JSONB_ARRAY_ELEMENTS JSON_ARRAY_ELEMENTS_TEXT "+"JSONB_ARRAY_ELEMENTS_TEXT JSON_TYPEOF JSONB_TYPEOF JSON_TO_RECORD JSONB_TO_RECORD JSON_TO_RECORDSET "+"JSONB_TO_RECORDSET JSON_STRIP_NULLS JSONB_STRIP_NULLS JSONB_SET JSONB_INSERT JSONB_PRETTY "+"CURRVAL LASTVAL NEXTVAL SETVAL "+"COALESCE NULLIF GREATEST LEAST "+"ARRAY_APPEND ARRAY_CAT ARRAY_NDIMS ARRAY_DIMS ARRAY_FILL ARRAY_LENGTH ARRAY_LOWER ARRAY_POSITION "+"ARRAY_POSITIONS ARRAY_PREPEND ARRAY_REMOVE ARRAY_REPLACE ARRAY_TO_STRING ARRAY_UPPER CARDINALITY "+"STRING_TO_ARRAY UNNEST "+"ISEMPTY LOWER_INC UPPER_INC LOWER_INF UPPER_INF RANGE_MERGE "+"GENERATE_SERIES GENERATE_SUBSCRIPTS "+"CURRENT_DATABASE CURRENT_QUERY CURRENT_SCHEMA|10 CURRENT_SCHEMAS|10 INET_CLIENT_ADDR INET_CLIENT_PORT "+"INET_SERVER_ADDR INET_SERVER_PORT ROW_SECURITY_ACTIVE FORMAT_TYPE "+"TO_REGCLASS TO_REGPROC TO_REGPROCEDURE TO_REGOPER TO_REGOPERATOR TO_REGTYPE TO_REGNAMESPACE TO_REGROLE "+"COL_DESCRIPTION OBJ_DESCRIPTION SHOBJ_DESCRIPTION "+"TXID_CURRENT TXID_CURRENT_IF_ASSIGNED TXID_CURRENT_SNAPSHOT TXID_SNAPSHOT_XIP TXID_SNAPSHOT_XMAX "+"TXID_SNAPSHOT_XMIN TXID_VISIBLE_IN_SNAPSHOT TXID_STATUS "+"CURRENT_SETTING SET_CONFIG BRIN_SUMMARIZE_NEW_VALUES BRIN_SUMMARIZE_RANGE BRIN_DESUMMARIZE_RANGE "+"GIN_CLEAN_PENDING_LIST "+"SUPPRESS_REDUNDANT_UPDATES_TRIGGER "+"LO_FROM_BYTEA LO_PUT LO_GET LO_CREAT LO_CREATE LO_UNLINK LO_IMPORT LO_EXPORT LOREAD LOWRITE "+"GROUPING CAST ";var FUNCTIONS_RE=FUNCTIONS.trim().split(" ").map(function(val){return val.split("|")[0]}).join("|");return{aliases:["postgres","postgresql"],case_insensitive:true,keywords:{keyword:SQL_KW+PLPGSQL_KW+ROLE_ATTRS,built_in:SQL_BI+PLPGSQL_BI+PLPGSQL_EXCEPTIONS},illegal:/:==|\W\s*\(\*|(^|\s)\$[a-z]|{{|[a-z]:\s*$|\.\.\.|TO:|DO:/,contains:[{className:"keyword",variants:[{begin:/\bTEXT\s*SEARCH\b/},{begin:/\b(PRIMARY|FOREIGN|FOR(\s+NO)?)\s+KEY\b/},{begin:/\bPARALLEL\s+(UNSAFE|RESTRICTED|SAFE)\b/},{begin:/\bSTORAGE\s+(PLAIN|EXTERNAL|EXTENDED|MAIN)\b/},{begin:/\bMATCH\s+(FULL|PARTIAL|SIMPLE)\b/},{begin:/\bNULLS\s+(FIRST|LAST)\b/},{begin:/\bEVENT\s+TRIGGER\b/},{begin:/\b(MAPPING|OR)\s+REPLACE\b/},{begin:/\b(FROM|TO)\s+(PROGRAM|STDIN|STDOUT)\b/},{begin:/\b(SHARE|EXCLUSIVE)\s+MODE\b/},{begin:/\b(LEFT|RIGHT)\s+(OUTER\s+)?JOIN\b/},{begin:/\b(FETCH|MOVE)\s+(NEXT|PRIOR|FIRST|LAST|ABSOLUTE|RELATIVE|FORWARD|BACKWARD)\b/},{begin:/\bPRESERVE\s+ROWS\b/},{begin:/\bDISCARD\s+PLANS\b/},{begin:/\bREFERENCING\s+(OLD|NEW)\b/},{begin:/\bSKIP\s+LOCKED\b/},{begin:/\bGROUPING\s+SETS\b/},{begin:/\b(BINARY|INSENSITIVE|SCROLL|NO\s+SCROLL)\s+(CURSOR|FOR)\b/},{begin:/\b(WITH|WITHOUT)\s+HOLD\b/},{begin:/\bWITH\s+(CASCADED|LOCAL)\s+CHECK\s+OPTION\b/},{begin:/\bEXCLUDE\s+(TIES|NO\s+OTHERS)\b/},{begin:/\bFORMAT\s+(TEXT|XML|JSON|YAML)\b/},{begin:/\bSET\s+((SESSION|LOCAL)\s+)?NAMES\b/},{begin:/\bIS\s+(NOT\s+)?UNKNOWN\b/},{begin:/\bSECURITY\s+LABEL\b/},{begin:/\bSTANDALONE\s+(YES|NO|NO\s+VALUE)\b/},{begin:/\bWITH\s+(NO\s+)?DATA\b/},{begin:/\b(FOREIGN|SET)\s+DATA\b/},{begin:/\bSET\s+(CATALOG|CONSTRAINTS)\b/},{begin:/\b(WITH|FOR)\s+ORDINALITY\b/},{begin:/\bIS\s+(NOT\s+)?DOCUMENT\b/},{begin:/\bXML\s+OPTION\s+(DOCUMENT|CONTENT)\b/},{begin:/\b(STRIP|PRESERVE)\s+WHITESPACE\b/},{begin:/\bNO\s+(ACTION|MAXVALUE|MINVALUE)\b/},{begin:/\bPARTITION\s+BY\s+(RANGE|LIST|HASH)\b/},{begin:/\bAT\s+TIME\s+ZONE\b/},{begin:/\bGRANTED\s+BY\b/},{begin:/\bRETURN\s+(QUERY|NEXT)\b/},{begin:/\b(ATTACH|DETACH)\s+PARTITION\b/},{begin:/\bFORCE\s+ROW\s+LEVEL\s+SECURITY\b/},{begin:/\b(INCLUDING|EXCLUDING)\s+(COMMENTS|CONSTRAINTS|DEFAULTS|IDENTITY|INDEXES|STATISTICS|STORAGE|ALL)\b/},{begin:/\bAS\s+(ASSIGNMENT|IMPLICIT|PERMISSIVE|RESTRICTIVE|ENUM|RANGE)\b/}]},{begin:/\b(FORMAT|FAMILY|VERSION)\s*\(/},{begin:/\bINCLUDE\s*\(/,keywords:"INCLUDE"},{begin:/\bRANGE(?!\s*(BETWEEN|UNBOUNDED|CURRENT|[-0-9]+))/},{begin:/\b(VERSION|OWNER|TEMPLATE|TABLESPACE|CONNECTION\s+LIMIT|PROCEDURE|RESTRICT|JOIN|PARSER|COPY|START|END|COLLATION|INPUT|ANALYZE|STORAGE|LIKE|DEFAULT|DELIMITER|ENCODING|COLUMN|CONSTRAINT|TABLE|SCHEMA)\s*=/},{begin:/\b(PG_\w+?|HAS_[A-Z_]+_PRIVILEGE)\b/,relevance:10},{begin:/\bEXTRACT\s*\(/,end:/\bFROM\b/,returnEnd:true,keywords:{type:"CENTURY DAY DECADE DOW DOY EPOCH HOUR ISODOW ISOYEAR MICROSECONDS "+"MILLENNIUM MILLISECONDS MINUTE MONTH QUARTER SECOND TIMEZONE TIMEZONE_HOUR "+"TIMEZONE_MINUTE WEEK YEAR"}},{begin:/\b(XMLELEMENT|XMLPI)\s*\(\s*NAME/,keywords:{keyword:"NAME"}},{begin:/\b(XMLPARSE|XMLSERIALIZE)\s*\(\s*(DOCUMENT|CONTENT)/,keywords:{keyword:"DOCUMENT CONTENT"}},{beginKeywords:"CACHE INCREMENT MAXVALUE MINVALUE",end:hljs.C_NUMBER_RE,returnEnd:true,keywords:"BY CACHE INCREMENT MAXVALUE MINVALUE"},{className:"type",begin:/\b(WITH|WITHOUT)\s+TIME\s+ZONE\b/},{className:"type",begin:/\bINTERVAL\s+(YEAR|MONTH|DAY|HOUR|MINUTE|SECOND)(\s+TO\s+(MONTH|HOUR|MINUTE|SECOND))?\b/},{begin:/\bRETURNS\s+(LANGUAGE_HANDLER|TRIGGER|EVENT_TRIGGER|FDW_HANDLER|INDEX_AM_HANDLER|TSM_HANDLER)\b/,keywords:{keyword:"RETURNS",type:"LANGUAGE_HANDLER TRIGGER EVENT_TRIGGER FDW_HANDLER INDEX_AM_HANDLER TSM_HANDLER"}},{begin:"\\b("+FUNCTIONS_RE+")\\s*\\("},{begin:"\\.("+TYPES_RE+")\\b"},{begin:"\\b("+TYPES_RE+")\\s+PATH\\b",keywords:{keyword:"PATH",type:TYPES.replace("PATH ","")}},{className:"type",begin:"\\b("+TYPES_RE+")\\b"},{className:"string",begin:"'",end:"'",contains:[{begin:"''"}]},{className:"string",begin:"(e|E|u&|U&)'",end:"'",contains:[{begin:"\\\\."}],relevance:10},{begin:DOLLAR_STRING,endSameAsBegin:true,contains:[{subLanguage:["pgsql","perl","python","tcl","r","lua","java","php","ruby","bash","scheme","xml","json"],endsWithParent:true}]},{begin:'"',end:'"',contains:[{begin:'""'}]},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE,COMMENT_MODE,{className:"meta",variants:[{begin:"%(ROW)?TYPE",relevance:10},{begin:"\\$\\d+"},{begin:"^#\\w",end:"$"}]},{className:"symbol",begin:LABEL,relevance:10}]}});hljs.registerLanguage("brainfuck",function(hljs){var LITERAL={className:"literal",begin:"[\\+\\-]",relevance:0};return{aliases:["bf"],contains:[hljs.COMMENT("[^\\[\\]\\.,\\+\\-<> \r\n]","[\\[\\]\\.,\\+\\-<> \r\n]",{returnEnd:true,relevance:0}),{className:"title",begin:"[\\[\\]]",relevance:0},{className:"string",begin:"[\\.,]",relevance:0},{begin:/(?:\+\+|\-\-)/,contains:[LITERAL]},LITERAL]}});hljs.registerLanguage("rsl",function(hljs){return{keywords:{keyword:"float color point normal vector matrix while for if do return else break extern continue",built_in:"abs acos ambient area asin atan atmosphere attribute calculatenormal ceil cellnoise "+"clamp comp concat cos degrees depth Deriv diffuse distance Du Dv environment exp "+"faceforward filterstep floor format fresnel incident length lightsource log match "+"max min mod noise normalize ntransform opposite option phong pnoise pow printf "+"ptlined radians random reflect refract renderinfo round setcomp setxcomp setycomp "+"setzcomp shadow sign sin smoothstep specular specularbrdf spline sqrt step tan "+"texture textureinfo trace transform vtransform xcomp ycomp zcomp"},illegal:"</",contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$"},{className:"class",beginKeywords:"surface displacement light volume imager",end:"\\("},{beginKeywords:"illuminate illuminance gather",end:"\\("}]}});hljs.registerLanguage("crystal",function(hljs){var INT_SUFFIX="(_*[ui](8|16|32|64|128))?";var FLOAT_SUFFIX="(_*f(32|64))?";var CRYSTAL_IDENT_RE="[a-zA-Z_]\\w*[!?=]?";var CRYSTAL_METHOD_RE="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|[=!]~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~|]|//|//=|&[-+*]=?|&\\*\\*|\\[\\][=?]?";var CRYSTAL_PATH_RE="[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?";var CRYSTAL_KEYWORDS={keyword:"abstract alias annotation as as? asm begin break case class def do else elsif end ensure enum extend for fun if "+"include instance_sizeof is_a? lib macro module next nil? of out pointerof private protected rescue responds_to? "+"return require select self sizeof struct super then type typeof union uninitialized unless until verbatim when while with yield "+"__DIR__ __END_LINE__ __FILE__ __LINE__",literal:"false nil true"};var SUBST={className:"subst",begin:"#{",end:"}",keywords:CRYSTAL_KEYWORDS};var EXPANSION={className:"template-variable",variants:[{begin:"\\{\\{",end:"\\}\\}"},{begin:"\\{%",end:"%\\}"}],keywords:CRYSTAL_KEYWORDS};function recursiveParen(begin,end){var contains=[{begin:begin,end:end}];contains[0].contains=contains;return contains}var STRING={className:"string",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{begin:"%[Qwi]?\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%[Qwi]?\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%[Qwi]?{",end:"}",contains:recursiveParen("{","}")},{begin:"%[Qwi]?<",end:">",contains:recursiveParen("<",">")},{begin:"%[Qwi]?\\|",end:"\\|"},{begin:/<<-\w+$/,end:/^\s*\w+$/}],relevance:0};var Q_STRING={className:"string",variants:[{begin:"%q\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%q\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%q{",end:"}",contains:recursiveParen("{","}")},{begin:"%q<",end:">",contains:recursiveParen("<",">")},{begin:"%q\\|",end:"\\|"},{begin:/<<-'\w+'$/,end:/^\s*\w+$/}],relevance:0};var REGEXP={begin:"(?!%})("+hljs.RE_STARTERS_RE+"|\\n|\\b(case|if|select|unless|until|when|while)\\b)\\s*",keywords:"case if select unless until when while",contains:[{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"//[a-z]*",relevance:0},{begin:"/(?!\\/)",end:"/[a-z]*"}]}],relevance:0};var REGEXP2={className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,SUBST],variants:[{begin:"%r\\(",end:"\\)",contains:recursiveParen("\\(","\\)")},{begin:"%r\\[",end:"\\]",contains:recursiveParen("\\[","\\]")},{begin:"%r{",end:"}",contains:recursiveParen("{","}")},{begin:"%r<",end:">",contains:recursiveParen("<",">")},{begin:"%r\\|",end:"\\|"}],relevance:0};var ATTRIBUTE={className:"meta",begin:"@\\[",end:"\\]",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"})]};var CRYSTAL_DEFAULT_CONTAINS=[EXPANSION,STRING,Q_STRING,REGEXP2,REGEXP,ATTRIBUTE,hljs.HASH_COMMENT_MODE,{className:"class",beginKeywords:"class module struct",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE}),{begin:"<"}]},{className:"class",beginKeywords:"lib enum union",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE})],relevance:10},{beginKeywords:"annotation",end:"$|;",illegal:/=/,contains:[hljs.HASH_COMMENT_MODE,hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_PATH_RE})],relevance:10},{className:"function",beginKeywords:"def",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_METHOD_RE,endsParent:true})]},{className:"function",beginKeywords:"fun macro",end:/\B\b/,contains:[hljs.inherit(hljs.TITLE_MODE,{begin:CRYSTAL_METHOD_RE,endsParent:true})],relevance:5},{className:"symbol",begin:hljs.UNDERSCORE_IDENT_RE+"(\\!|\\?)?:",relevance:0},{className:"symbol",begin:":",contains:[STRING,{begin:CRYSTAL_METHOD_RE}],relevance:0},{className:"number",variants:[{begin:"\\b0b([01_]+)"+INT_SUFFIX},{begin:"\\b0o([0-7_]+)"+INT_SUFFIX},{begin:"\\b0x([A-Fa-f0-9_]+)"+INT_SUFFIX},{begin:"\\b([1-9][0-9_]*[0-9]|[0-9])(\\.[0-9][0-9_]*)?([eE]_*[-+]?[0-9_]*)?"+FLOAT_SUFFIX+"(?!_)"},{begin:"\\b([1-9][0-9_]*|0)"+INT_SUFFIX}],relevance:0}];SUBST.contains=CRYSTAL_DEFAULT_CONTAINS;EXPANSION.contains=CRYSTAL_DEFAULT_CONTAINS.slice(1);return{aliases:["cr"],lexemes:CRYSTAL_IDENT_RE,keywords:CRYSTAL_KEYWORDS,contains:CRYSTAL_DEFAULT_CONTAINS}});hljs.registerLanguage("smalltalk",function(hljs){var VAR_IDENT_RE="[a-z][a-zA-Z0-9_]*";var CHAR={className:"string",begin:"\\$.{1}"};var SYMBOL={className:"symbol",begin:"#"+hljs.UNDERSCORE_IDENT_RE};return{aliases:["st"],keywords:"self super nil true false thisContext",contains:[hljs.COMMENT('"','"'),hljs.APOS_STRING_MODE,{className:"type",begin:"\\b[A-Z][A-Za-z0-9_]*",relevance:0},{begin:VAR_IDENT_RE+":",relevance:0},hljs.C_NUMBER_MODE,SYMBOL,CHAR,{begin:"\\|[ ]*"+VAR_IDENT_RE+"([ ]+"+VAR_IDENT_RE+")*[ ]*\\|",returnBegin:true,end:/\|/,illegal:/\S/,contains:[{begin:"(\\|[ ]*)?"+VAR_IDENT_RE}]},{begin:"\\#\\(",end:"\\)",contains:[hljs.APOS_STRING_MODE,CHAR,hljs.C_NUMBER_MODE,SYMBOL]}]}});hljs.registerLanguage("vbnet",function(hljs){return{aliases:["vb"],case_insensitive:true,keywords:{keyword:"addhandler addressof alias and andalso aggregate ansi as async assembly auto await binary by byref byval "+"call case catch class compare const continue custom declare default delegate dim distinct do "+"each equals else elseif end enum erase error event exit explicit finally for friend from function "+"get global goto group handles if implements imports in inherits interface into is isfalse isnot istrue iterator "+"join key let lib like loop me mid mod module mustinherit mustoverride mybase myclass "+"nameof namespace narrowing new next not notinheritable notoverridable "+"of off on operator option optional or order orelse overloads overridable overrides "+"paramarray partial preserve private property protected public "+"raiseevent readonly redim rem removehandler resume return "+"select set shadows shared skip static step stop structure strict sub synclock "+"take text then throw to try unicode until using when where while widening with withevents writeonly xor yield",built_in:"boolean byte cbool cbyte cchar cdate cdec cdbl char cint clng cobj csbyte cshort csng cstr ctype "+"date decimal directcast double gettype getxmlnamespace iif integer long object "+"sbyte short single string trycast typeof uinteger ulong ushort",literal:"true false nothing"},illegal:"//|{|}|endif|gosub|variant|wend|^\\$ ",contains:[hljs.inherit(hljs.QUOTE_STRING_MODE,{contains:[{begin:'""'}]}),hljs.COMMENT("'","$",{returnBegin:true,contains:[{className:"doctag",begin:"'''|\x3c!--|--\x3e",contains:[hljs.PHRASAL_WORDS_MODE]},{className:"doctag",begin:"</?",end:">",contains:[hljs.PHRASAL_WORDS_MODE]}]}),hljs.C_NUMBER_MODE,{className:"meta",begin:"#",end:"$",keywords:{"meta-keyword":"if else elseif end region externalsource"}}]}});hljs.registerLanguage("hy",function(hljs){var keywords={"builtin-name":"!= % %= & &= * ** **= *= *map "+"+ += , --build-class-- --import-- -= . / // //= "+"/= < << <<= <= = > >= >> >>= "+"@ @= ^ ^= abs accumulate all and any ap-compose "+"ap-dotimes ap-each ap-each-while ap-filter ap-first ap-if ap-last ap-map ap-map-when ap-pipe "+"ap-reduce ap-reject apply as-> ascii assert assoc bin break butlast "+"callable calling-module-name car case cdr chain chr coll? combinations compile "+"compress cond cons cons? continue count curry cut cycle dec "+"def default-method defclass defmacro defmacro-alias defmacro/g! defmain defmethod defmulti defn "+"defn-alias defnc defnr defreader defseq del delattr delete-route dict-comp dir "+"disassemble dispatch-reader-macro distinct divmod do doto drop drop-last drop-while empty? "+"end-sequence eval eval-and-compile eval-when-compile even? every? except exec filter first "+"flatten float? fn fnc fnr for for* format fraction genexpr "+"gensym get getattr global globals group-by hasattr hash hex id "+"identity if if* if-not if-python2 import in inc input instance? "+"integer integer-char? integer? interleave interpose is is-coll is-cons is-empty is-even "+"is-every is-float is-instance is-integer is-integer-char is-iterable is-iterator is-keyword is-neg is-none "+"is-not is-numeric is-odd is-pos is-string is-symbol is-zero isinstance islice issubclass "+"iter iterable? iterate iterator? keyword keyword? lambda last len let "+"lif lif-not list* list-comp locals loop macro-error macroexpand macroexpand-1 macroexpand-all "+"map max merge-with method-decorator min multi-decorator multicombinations name neg? next "+"none? nonlocal not not-in not? nth numeric? oct odd? open "+"or ord partition permutations pos? post-route postwalk pow prewalk print "+"product profile/calls profile/cpu put-route quasiquote quote raise range read read-str "+"recursive-replace reduce remove repeat repeatedly repr require rest round route "+"route-with-methods rwm second seq set-comp setattr setv some sorted string "+"string? sum switch symbol? take take-nth take-while tee try unless "+"unquote unquote-splicing vars walk when while with with* with-decorator with-gensyms "+"xi xor yield yield-from zero? zip zip-longest | |= ~"};var SYMBOLSTART="a-zA-Z_\\-!.?+*=<>&#'";var SYMBOL_RE="["+SYMBOLSTART+"]["+SYMBOLSTART+"0-9/;:]*";var SIMPLE_NUMBER_RE="[-+]?\\d+(\\.\\d+)?";var SHEBANG={className:"meta",begin:"^#!",end:"$"};var SYMBOL={begin:SYMBOL_RE,relevance:0};var NUMBER={className:"number",begin:SIMPLE_NUMBER_RE,relevance:0};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var LITERAL={className:"literal",begin:/\b([Tt]rue|[Ff]alse|nil|None)\b/};var COLLECTION={begin:"[\\[\\{]",end:"[\\]\\}]"};var HINT={className:"comment",begin:"\\^"+SYMBOL_RE};var HINT_COL=hljs.COMMENT("\\^\\{","\\}");var KEY={className:"symbol",begin:"[:]{1,2}"+SYMBOL_RE};var LIST={begin:"\\(",end:"\\)"};var BODY={endsWithParent:true,relevance:0};var NAME={keywords:keywords,lexemes:SYMBOL_RE,className:"name",begin:SYMBOL_RE,starts:BODY};var DEFAULT_CONTAINS=[LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL,SYMBOL];LIST.contains=[hljs.COMMENT("comment",""),NAME,BODY];BODY.contains=DEFAULT_CONTAINS;COLLECTION.contains=DEFAULT_CONTAINS;return{aliases:["hylang"],illegal:/\S/,contains:[SHEBANG,LIST,STRING,HINT,HINT_COL,COMMENT,KEY,COLLECTION,NUMBER,LITERAL]}});hljs.registerLanguage("objectivec",function(hljs){var API_CLASS={className:"built_in",begin:"\\b(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)\\w+"};var OBJC_KEYWORDS={keyword:"int float while char export sizeof typedef const struct for union "+"unsigned long volatile static bool mutable if do return goto void "+"enum else break extern asm case short default double register explicit "+"signed typename this switch continue wchar_t inline readonly assign "+"readwrite self @synchronized id typeof "+"nonatomic super unichar IBOutlet IBAction strong weak copy "+"in out inout bycopy byref oneway __strong __weak __block __autoreleasing "+"@private @protected @public @try @property @end @throw @catch @finally "+"@autoreleasepool @synthesize @dynamic @selector @optional @required "+"@encode @package @import @defs @compatibility_alias "+"__bridge __bridge_transfer __bridge_retained __bridge_retain "+"__covariant __contravariant __kindof "+"_Nonnull _Nullable _Null_unspecified "+"__FUNCTION__ __PRETTY_FUNCTION__ __attribute__ "+"getter setter retain unsafe_unretained "+"nonnull nullable null_unspecified null_resettable class instancetype "+"NS_DESIGNATED_INITIALIZER NS_UNAVAILABLE NS_REQUIRES_SUPER "+"NS_RETURNS_INNER_POINTER NS_INLINE NS_AVAILABLE NS_DEPRECATED "+"NS_ENUM NS_OPTIONS NS_SWIFT_UNAVAILABLE "+"NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END "+"NS_REFINED_FOR_SWIFT NS_SWIFT_NAME NS_SWIFT_NOTHROW "+"NS_DURING NS_HANDLER NS_ENDHANDLER NS_VALUERETURN NS_VOIDRETURN",literal:"false true FALSE TRUE nil YES NO NULL",built_in:"BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once"};var LEXEMES=/[a-zA-Z@][a-zA-Z0-9_]*/;var CLASS_KEYWORDS="@interface @class @protocol @implementation";return{aliases:["mm","objc","obj-c"],keywords:OBJC_KEYWORDS,lexemes:LEXEMES,illegal:"</",contains:[API_CLASS,hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.C_NUMBER_MODE,hljs.QUOTE_STRING_MODE,hljs.APOS_STRING_MODE,{className:"string",variants:[{begin:'@"',end:'"',illegal:"\\n",contains:[hljs.BACKSLASH_ESCAPE]}]},{className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{"meta-keyword":"if else elif endif define undef warning error line "+"pragma ifdef ifndef include"},contains:[{begin:/\\\n/,relevance:0},hljs.inherit(hljs.QUOTE_STRING_MODE,{className:"meta-string"}),{className:"meta-string",begin:/<.*?>/,end:/$/,illegal:"\\n"},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE]},{className:"class",begin:"("+CLASS_KEYWORDS.split(" ").join("|")+")\\b",end:"({|$)",excludeEnd:true,keywords:CLASS_KEYWORDS,lexemes:LEXEMES,contains:[hljs.UNDERSCORE_TITLE_MODE]},{begin:"\\."+hljs.UNDERSCORE_IDENT_RE,relevance:0}]}});hljs.registerLanguage("stan",function(hljs){var BLOCKS=["functions","model","data","parameters","quantities","transformed","generated"];var STATEMENTS=["for","in","if","else","while","break","continue","return"];var SPECIAL_FUNCTIONS=["print","reject","increment_log_prob|10","integrate_ode|10","integrate_ode_rk45|10","integrate_ode_bdf|10","algebra_solver"];var VAR_TYPES=["int","real","vector","ordered","positive_ordered","simplex","unit_vector","row_vector","matrix","cholesky_factor_corr|10","cholesky_factor_cov|10","corr_matrix|10","cov_matrix|10","void"];var FUNCTIONS=["Phi","Phi_approx","abs","acos","acosh","algebra_solver","append_array","append_col","append_row","asin","asinh","atan","atan2","atanh","bernoulli_cdf","bernoulli_lccdf","bernoulli_lcdf","bernoulli_logit_lpmf","bernoulli_logit_rng","bernoulli_lpmf","bernoulli_rng","bessel_first_kind","bessel_second_kind","beta_binomial_cdf","beta_binomial_lccdf","beta_binomial_lcdf","beta_binomial_lpmf","beta_binomial_rng","beta_cdf","beta_lccdf","beta_lcdf","beta_lpdf","beta_rng","binary_log_loss","binomial_cdf","binomial_coefficient_log","binomial_lccdf","binomial_lcdf","binomial_logit_lpmf","binomial_lpmf","binomial_rng","block","categorical_logit_lpmf","categorical_logit_rng","categorical_lpmf","categorical_rng","cauchy_cdf","cauchy_lccdf","cauchy_lcdf","cauchy_lpdf","cauchy_rng","cbrt","ceil","chi_square_cdf","chi_square_lccdf","chi_square_lcdf","chi_square_lpdf","chi_square_rng","cholesky_decompose","choose","col","cols","columns_dot_product","columns_dot_self","cos","cosh","cov_exp_quad","crossprod","csr_extract_u","csr_extract_v","csr_extract_w","csr_matrix_times_vector","csr_to_dense_matrix","cumulative_sum","determinant","diag_matrix","diag_post_multiply","diag_pre_multiply","diagonal","digamma","dims","dirichlet_lpdf","dirichlet_rng","distance","dot_product","dot_self","double_exponential_cdf","double_exponential_lccdf","double_exponential_lcdf","double_exponential_lpdf","double_exponential_rng","e","eigenvalues_sym","eigenvectors_sym","erf","erfc","exp","exp2","exp_mod_normal_cdf","exp_mod_normal_lccdf","exp_mod_normal_lcdf","exp_mod_normal_lpdf","exp_mod_normal_rng","expm1","exponential_cdf","exponential_lccdf","exponential_lcdf","exponential_lpdf","exponential_rng","fabs","falling_factorial","fdim","floor","fma","fmax","fmin","fmod","frechet_cdf","frechet_lccdf","frechet_lcdf","frechet_lpdf","frechet_rng","gamma_cdf","gamma_lccdf","gamma_lcdf","gamma_lpdf","gamma_p","gamma_q","gamma_rng","gaussian_dlm_obs_lpdf","get_lp","gumbel_cdf","gumbel_lccdf","gumbel_lcdf","gumbel_lpdf","gumbel_rng","head","hypergeometric_lpmf","hypergeometric_rng","hypot","inc_beta","int_step","integrate_ode","integrate_ode_bdf","integrate_ode_rk45","inv","inv_Phi","inv_chi_square_cdf","inv_chi_square_lccdf","inv_chi_square_lcdf","inv_chi_square_lpdf","inv_chi_square_rng","inv_cloglog","inv_gamma_cdf","inv_gamma_lccdf","inv_gamma_lcdf","inv_gamma_lpdf","inv_gamma_rng","inv_logit","inv_sqrt","inv_square","inv_wishart_lpdf","inv_wishart_rng","inverse","inverse_spd","is_inf","is_nan","lbeta","lchoose","lgamma","lkj_corr_cholesky_lpdf","lkj_corr_cholesky_rng","lkj_corr_lpdf","lkj_corr_rng","lmgamma","lmultiply","log","log10","log1m","log1m_exp","log1m_inv_logit","log1p","log1p_exp","log2","log_determinant","log_diff_exp","log_falling_factorial","log_inv_logit","log_mix","log_rising_factorial","log_softmax","log_sum_exp","logistic_cdf","logistic_lccdf","logistic_lcdf","logistic_lpdf","logistic_rng","logit","lognormal_cdf","lognormal_lccdf","lognormal_lcdf","lognormal_lpdf","lognormal_rng","machine_precision","matrix_exp","max","mdivide_left_spd","mdivide_left_tri_low","mdivide_right_spd","mdivide_right_tri_low","mean","min","modified_bessel_first_kind","modified_bessel_second_kind","multi_gp_cholesky_lpdf","multi_gp_lpdf","multi_normal_cholesky_lpdf","multi_normal_cholesky_rng","multi_normal_lpdf","multi_normal_prec_lpdf","multi_normal_rng","multi_student_t_lpdf","multi_student_t_rng","multinomial_lpmf","multinomial_rng","multiply_log","multiply_lower_tri_self_transpose","neg_binomial_2_cdf","neg_binomial_2_lccdf","neg_binomial_2_lcdf","neg_binomial_2_log_lpmf","neg_binomial_2_log_rng","neg_binomial_2_lpmf","neg_binomial_2_rng","neg_binomial_cdf","neg_binomial_lccdf","neg_binomial_lcdf","neg_binomial_lpmf","neg_binomial_rng","negative_infinity","normal_cdf","normal_lccdf","normal_lcdf","normal_lpdf","normal_rng","not_a_number","num_elements","ordered_logistic_lpmf","ordered_logistic_rng","owens_t","pareto_cdf","pareto_lccdf","pareto_lcdf","pareto_lpdf","pareto_rng","pareto_type_2_cdf","pareto_type_2_lccdf","pareto_type_2_lcdf","pareto_type_2_lpdf","pareto_type_2_rng","pi","poisson_cdf","poisson_lccdf","poisson_lcdf","poisson_log_lpmf","poisson_log_rng","poisson_lpmf","poisson_rng","positive_infinity","pow","print","prod","qr_Q","qr_R","quad_form","quad_form_diag","quad_form_sym","rank","rayleigh_cdf","rayleigh_lccdf","rayleigh_lcdf","rayleigh_lpdf","rayleigh_rng","reject","rep_array","rep_matrix","rep_row_vector","rep_vector","rising_factorial","round","row","rows","rows_dot_product","rows_dot_self","scaled_inv_chi_square_cdf","scaled_inv_chi_square_lccdf","scaled_inv_chi_square_lcdf","scaled_inv_chi_square_lpdf","scaled_inv_chi_square_rng","sd","segment","sin","singular_values","sinh","size","skew_normal_cdf","skew_normal_lccdf","skew_normal_lcdf","skew_normal_lpdf","skew_normal_rng","softmax","sort_asc","sort_desc","sort_indices_asc","sort_indices_desc","sqrt","sqrt2","square","squared_distance","step","student_t_cdf","student_t_lccdf","student_t_lcdf","student_t_lpdf","student_t_rng","sub_col","sub_row","sum","tail","tan","tanh","target","tcrossprod","tgamma","to_array_1d","to_array_2d","to_matrix","to_row_vector","to_vector","trace","trace_gen_quad_form","trace_quad_form","trigamma","trunc","uniform_cdf","uniform_lccdf","uniform_lcdf","uniform_lpdf","uniform_rng","variance","von_mises_lpdf","von_mises_rng","weibull_cdf","weibull_lccdf","weibull_lcdf","weibull_lpdf","weibull_rng","wiener_lpdf","wishart_lpdf","wishart_rng"];var DISTRIBUTIONS=["bernoulli","bernoulli_logit","beta","beta_binomial","binomial","binomial_logit","categorical","categorical_logit","cauchy","chi_square","dirichlet","double_exponential","exp_mod_normal","exponential","frechet","gamma","gaussian_dlm_obs","gumbel","hypergeometric","inv_chi_square","inv_gamma","inv_wishart","lkj_corr","lkj_corr_cholesky","logistic","lognormal","multi_gp","multi_gp_cholesky","multi_normal","multi_normal_cholesky","multi_normal_prec","multi_student_t","multinomial","neg_binomial","neg_binomial_2","neg_binomial_2_log","normal","ordered_logistic","pareto","pareto_type_2","poisson","poisson_log","rayleigh","scaled_inv_chi_square","skew_normal","student_t","uniform","von_mises","weibull","wiener","wishart"];return{aliases:["stanfuncs"],keywords:{title:BLOCKS.join(" "),keyword:STATEMENTS.concat(VAR_TYPES).concat(SPECIAL_FUNCTIONS).join(" "),built_in:FUNCTIONS.join(" ")},lexemes:hljs.IDENT_RE,contains:[hljs.C_LINE_COMMENT_MODE,hljs.COMMENT(/#/,/$/,{relevance:0,keywords:{"meta-keyword":"include"}}),hljs.COMMENT(/\/\*/,/\*\//,{relevance:0,contains:[{className:"doctag",begin:/@(return|param)/}]}),{begin:/<\s*lower\s*=/,keywords:"lower"},{begin:/[<,]*upper\s*=/,keywords:"upper"},{className:"keyword",begin:/\btarget\s*\+=/,relevance:10},{begin:"~\\s*("+hljs.IDENT_RE+")\\s*\\(",keywords:DISTRIBUTIONS.join(" ")},{className:"number",variants:[{begin:/\b\d+(?:\.\d*)?(?:[eE][+-]?\d+)?/},{begin:/\.\d+(?:[eE][+-]?\d+)?\b/}],relevance:0},{className:"string",begin:'"',end:'"',relevance:0}]}});hljs.registerLanguage("lasso",function(hljs){var LASSO_IDENT_RE="[a-zA-Z_][\\w.]*";var LASSO_ANGLE_RE="<\\?(lasso(script)?|=)";var LASSO_CLOSE_RE="\\]|\\?>";var LASSO_KEYWORDS={literal:"true false none minimal full all void and or not "+"bw nbw ew new cn ncn lt lte gt gte eq neq rx nrx ft",built_in:"array date decimal duration integer map pair string tag xml null "+"boolean bytes keyword list locale queue set stack staticarray "+"local var variable global data self inherited currentcapture givenblock",keyword:"cache database_names database_schemanames database_tablenames "+"define_tag define_type email_batch encode_set html_comment handle "+"handle_error header if inline iterate ljax_target link "+"link_currentaction link_currentgroup link_currentrecord link_detail "+"link_firstgroup link_firstrecord link_lastgroup link_lastrecord "+"link_nextgroup link_nextrecord link_prevgroup link_prevrecord log "+"loop namespace_using output_none portal private protect records "+"referer referrer repeating resultset rows search_args "+"search_arguments select sort_args sort_arguments thread_atomic "+"value_list while abort case else fail_if fail_ifnot fail if_empty "+"if_false if_null if_true loop_abort loop_continue loop_count params "+"params_up return return_value run_children soap_definetag "+"soap_lastrequest soap_lastresponse tag_name ascending average by "+"define descending do equals frozen group handle_failure import in "+"into join let match max min on order parent protected provide public "+"require returnhome skip split_thread sum take thread to trait type "+"where with yield yieldhome"};var HTML_COMMENT=hljs.COMMENT("\x3c!--","--\x3e",{relevance:0});var LASSO_NOPROCESS={className:"meta",begin:"\\[noprocess\\]",starts:{end:"\\[/noprocess\\]",returnEnd:true,contains:[HTML_COMMENT]}};var LASSO_START={className:"meta",begin:"\\[/noprocess|"+LASSO_ANGLE_RE};var LASSO_DATAMEMBER={className:"symbol",begin:"'"+LASSO_IDENT_RE+"'"};var LASSO_CODE=[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.inherit(hljs.C_NUMBER_MODE,{begin:hljs.C_NUMBER_RE+"|(-?infinity|NaN)\\b"}),hljs.inherit(hljs.APOS_STRING_MODE,{illegal:null}),hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null}),{className:"string",begin:"`",end:"`"},{variants:[{begin:"[#$]"+LASSO_IDENT_RE},{begin:"#",end:"\\d+",illegal:"\\W"}]},{className:"type",begin:"::\\s*",end:LASSO_IDENT_RE,illegal:"\\W"},{className:"params",variants:[{begin:"-(?!infinity)"+LASSO_IDENT_RE,relevance:0},{begin:"(\\.\\.\\.)"}]},{begin:/(->|\.)\s*/,relevance:0,contains:[LASSO_DATAMEMBER]},{className:"class",beginKeywords:"define",returnEnd:true,end:"\\(|=>",contains:[hljs.inherit(hljs.TITLE_MODE,{begin:LASSO_IDENT_RE+"(=(?!>))?|[-+*/%](?!>)"})]}];return{aliases:["ls","lassoscript"],case_insensitive:true,lexemes:LASSO_IDENT_RE+"|&[lg]t;",keywords:LASSO_KEYWORDS,contains:[{className:"meta",begin:LASSO_CLOSE_RE,relevance:0,starts:{end:"\\[|"+LASSO_ANGLE_RE,returnEnd:true,relevance:0,contains:[HTML_COMMENT]}},LASSO_NOPROCESS,LASSO_START,{className:"meta",begin:"\\[no_square_brackets",starts:{end:"\\[/no_square_brackets\\]",lexemes:LASSO_IDENT_RE+"|&[lg]t;",keywords:LASSO_KEYWORDS,contains:[{className:"meta",begin:LASSO_CLOSE_RE,relevance:0,starts:{end:"\\[noprocess\\]|"+LASSO_ANGLE_RE,returnEnd:true,contains:[HTML_COMMENT]}},LASSO_NOPROCESS,LASSO_START].concat(LASSO_CODE)}},{className:"meta",begin:"\\[",relevance:0},{className:"meta",begin:"^#!",end:"lasso9$",relevance:10}].concat(LASSO_CODE)}});hljs.registerLanguage("lisp",function(hljs){var LISP_IDENT_RE="[a-zA-Z_\\-\\+\\*\\/\\<\\=\\>\\&\\#][a-zA-Z0-9_\\-\\+\\*\\/\\<\\=\\>\\&\\#!]*";var MEC_RE="\\|[^]*?\\|";var LISP_SIMPLE_NUMBER_RE="(\\-|\\+)?\\d+(\\.\\d+|\\/\\d+)?((d|e|f|l|s|D|E|F|L|S)(\\+|\\-)?\\d+)?";var SHEBANG={className:"meta",begin:"^#!",end:"$"};var LITERAL={className:"literal",begin:"\\b(t{1}|nil)\\b"};var NUMBER={className:"number",variants:[{begin:LISP_SIMPLE_NUMBER_RE,relevance:0},{begin:"#(b|B)[0-1]+(/[0-1]+)?"},{begin:"#(o|O)[0-7]+(/[0-7]+)?"},{begin:"#(x|X)[0-9a-fA-F]+(/[0-9a-fA-F]+)?"},{begin:"#(c|C)\\("+LISP_SIMPLE_NUMBER_RE+" +"+LISP_SIMPLE_NUMBER_RE,end:"\\)"}]};var STRING=hljs.inherit(hljs.QUOTE_STRING_MODE,{illegal:null});var COMMENT=hljs.COMMENT(";","$",{relevance:0});var VARIABLE={begin:"\\*",end:"\\*"};var KEYWORD={className:"symbol",begin:"[:&]"+LISP_IDENT_RE};var IDENT={begin:LISP_IDENT_RE,relevance:0};var MEC={begin:MEC_RE};var QUOTED_LIST={begin:"\\(",end:"\\)",contains:["self",LITERAL,STRING,NUMBER,IDENT]};var QUOTED={contains:[NUMBER,STRING,VARIABLE,KEYWORD,QUOTED_LIST,IDENT],variants:[{begin:"['`]\\(",end:"\\)"},{begin:"\\(quote ",end:"\\)",keywords:{name:"quote"}},{begin:"'"+MEC_RE}]};var QUOTED_ATOM={variants:[{begin:"'"+LISP_IDENT_RE},{begin:"#'"+LISP_IDENT_RE+"(::"+LISP_IDENT_RE+")*"}]};var LIST={begin:"\\(\\s*",end:"\\)"};var BODY={endsWithParent:true,relevance:0};LIST.contains=[{className:"name",variants:[{begin:LISP_IDENT_RE},{begin:MEC_RE}]},BODY];BODY.contains=[QUOTED,QUOTED_ATOM,LIST,LITERAL,NUMBER,STRING,COMMENT,VARIABLE,KEYWORD,MEC,IDENT];return{illegal:/\S/,contains:[NUMBER,SHEBANG,LITERAL,STRING,COMMENT,QUOTED,QUOTED_ATOM,LIST,IDENT]}});hljs.registerLanguage("mojolicious",function(hljs){return{subLanguage:"xml",contains:[{className:"meta",begin:"^__(END|DATA)__$"},{begin:"^\\s*%{1,2}={0,2}",end:"$",subLanguage:"perl"},{begin:"<%{1,2}={0,2}",end:"={0,1}%>",subLanguage:"perl",excludeBegin:true,excludeEnd:true}]}});hljs.registerLanguage("processing",function(hljs){return{keywords:{keyword:"BufferedReader PVector PFont PImage PGraphics HashMap boolean byte char color "+"double float int long String Array FloatDict FloatList IntDict IntList JSONArray JSONObject "+"Object StringDict StringList Table TableRow XML "+"false synchronized int abstract float private char boolean static null if const "+"for true while long throw strictfp finally protected import native final return void "+"enum else break transient new catch instanceof byte super volatile case assert short "+"package default double public try this switch continue throws protected public private",literal:"P2D P3D HALF_PI PI QUARTER_PI TAU TWO_PI",title:"setup draw",built_in:"displayHeight displayWidth mouseY mouseX mousePressed pmouseX pmouseY key "+"keyCode pixels focused frameCount frameRate height width "+"size createGraphics beginDraw createShape loadShape PShape arc ellipse line point "+"quad rect triangle bezier bezierDetail bezierPoint bezierTangent curve curveDetail curvePoint "+"curveTangent curveTightness shape shapeMode beginContour beginShape bezierVertex curveVertex "+"endContour endShape quadraticVertex vertex ellipseMode noSmooth rectMode smooth strokeCap "+"strokeJoin strokeWeight mouseClicked mouseDragged mouseMoved mousePressed mouseReleased "+"mouseWheel keyPressed keyPressedkeyReleased keyTyped print println save saveFrame day hour "+"millis minute month second year background clear colorMode fill noFill noStroke stroke alpha "+"blue brightness color green hue lerpColor red saturation modelX modelY modelZ screenX screenY "+"screenZ ambient emissive shininess specular add createImage beginCamera camera endCamera frustum "+"ortho perspective printCamera printProjection cursor frameRate noCursor exit loop noLoop popStyle "+"pushStyle redraw binary boolean byte char float hex int str unbinary unhex join match matchAll nf "+"nfc nfp nfs split splitTokens trim append arrayCopy concat expand reverse shorten sort splice subset "+"box sphere sphereDetail createInput createReader loadBytes loadJSONArray loadJSONObject loadStrings "+"loadTable loadXML open parseXML saveTable selectFolder selectInput beginRaw beginRecord createOutput "+"createWriter endRaw endRecord PrintWritersaveBytes saveJSONArray saveJSONObject saveStream saveStrings "+"saveXML selectOutput popMatrix printMatrix pushMatrix resetMatrix rotate rotateX rotateY rotateZ scale "+"shearX shearY translate ambientLight directionalLight lightFalloff lights lightSpecular noLights normal "+"pointLight spotLight image imageMode loadImage noTint requestImage tint texture textureMode textureWrap "+"blend copy filter get loadPixels set updatePixels blendMode loadShader PShaderresetShader shader createFont "+"loadFont text textFont textAlign textLeading textMode textSize textWidth textAscent textDescent abs ceil "+"constrain dist exp floor lerp log mag map max min norm pow round sq sqrt acos asin atan atan2 cos degrees "+"radians sin tan noise noiseDetail noiseSeed random randomGaussian randomSeed"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("clean",function(hljs){return{aliases:["clean","icl","dcl"],keywords:{keyword:"if let in with where case of class instance otherwise "+"implementation definition system module from import qualified as "+"special code inline foreign export ccall stdcall generic derive "+"infix infixl infixr",built_in:"Int Real Char Bool",literal:"True False"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{begin:"->|<-[|:]?|#!?|>>=|\\{\\||\\|\\}|:==|=:|<>"}]}});hljs.registerLanguage("d",function(hljs){var D_KEYWORDS={keyword:"abstract alias align asm assert auto body break byte case cast catch class "+"const continue debug default delete deprecated do else enum export extern final "+"finally for foreach foreach_reverse|10 goto if immutable import in inout int "+"interface invariant is lazy macro mixin module new nothrow out override package "+"pragma private protected public pure ref return scope shared static struct "+"super switch synchronized template this throw try typedef typeid typeof union "+"unittest version void volatile while with __FILE__ __LINE__ __gshared|10 "+"__thread __traits __DATE__ __EOF__ __TIME__ __TIMESTAMP__ __VENDOR__ __VERSION__",built_in:"bool cdouble cent cfloat char creal dchar delegate double dstring float function "+"idouble ifloat ireal long real short string ubyte ucent uint ulong ushort wchar "+"wstring",literal:"false null true"};var decimal_integer_re="(0|[1-9][\\d_]*)",decimal_integer_nosus_re="(0|[1-9][\\d_]*|\\d[\\d_]*|[\\d_]+?\\d)",binary_integer_re="0[bB][01_]+",hexadecimal_digits_re="([\\da-fA-F][\\da-fA-F_]*|_[\\da-fA-F][\\da-fA-F_]*)",hexadecimal_integer_re="0[xX]"+hexadecimal_digits_re,decimal_exponent_re="([eE][+-]?"+decimal_integer_nosus_re+")",decimal_float_re="("+decimal_integer_nosus_re+"(\\.\\d*|"+decimal_exponent_re+")|"+"\\d+\\."+decimal_integer_nosus_re+decimal_integer_nosus_re+"|"+"\\."+decimal_integer_re+decimal_exponent_re+"?"+")",hexadecimal_float_re="(0[xX]("+hexadecimal_digits_re+"\\."+hexadecimal_digits_re+"|"+"\\.?"+hexadecimal_digits_re+")[pP][+-]?"+decimal_integer_nosus_re+")",integer_re="("+decimal_integer_re+"|"+binary_integer_re+"|"+hexadecimal_integer_re+")",float_re="("+hexadecimal_float_re+"|"+decimal_float_re+")";var escape_sequence_re="\\\\("+"['\"\\?\\\\abfnrtv]|"+"u[\\dA-Fa-f]{4}|"+"[0-7]{1,3}|"+"x[\\dA-Fa-f]{2}|"+"U[\\dA-Fa-f]{8}"+")|"+"&[a-zA-Z\\d]{2,};";var D_INTEGER_MODE={className:"number",begin:"\\b"+integer_re+"(L|u|U|Lu|LU|uL|UL)?",relevance:0};var D_FLOAT_MODE={className:"number",begin:"\\b("+float_re+"([fF]|L|i|[fF]i|Li)?|"+integer_re+"(i|[fF]i|Li)"+")",relevance:0};var D_CHARACTER_MODE={className:"string",begin:"'("+escape_sequence_re+"|.)",end:"'",illegal:"."};var D_ESCAPE_SEQUENCE={begin:escape_sequence_re,relevance:0};var D_STRING_MODE={className:"string",begin:'"',contains:[D_ESCAPE_SEQUENCE],end:'"[cwd]?'};var D_WYSIWYG_DELIMITED_STRING_MODE={className:"string",begin:'[rq]"',end:'"[cwd]?',relevance:5};var D_ALTERNATE_WYSIWYG_STRING_MODE={className:"string",begin:"`",end:"`[cwd]?"};var D_HEX_STRING_MODE={className:"string",begin:'x"[\\da-fA-F\\s\\n\\r]*"[cwd]?',relevance:10};var D_TOKEN_STRING_MODE={className:"string",begin:'q"\\{',end:'\\}"'};var D_HASHBANG_MODE={className:"meta",begin:"^#!",end:"$",relevance:5};var D_SPECIAL_TOKEN_SEQUENCE_MODE={className:"meta",begin:"#(line)",end:"$",relevance:5};var D_ATTRIBUTE_MODE={className:"keyword",begin:"@[a-zA-Z_][a-zA-Z_\\d]*"};var D_NESTING_COMMENT_MODE=hljs.COMMENT("\\/\\+","\\+\\/",{contains:["self"],relevance:10});return{lexemes:hljs.UNDERSCORE_IDENT_RE,keywords:D_KEYWORDS,contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,D_NESTING_COMMENT_MODE,D_HEX_STRING_MODE,D_STRING_MODE,D_WYSIWYG_DELIMITED_STRING_MODE,D_ALTERNATE_WYSIWYG_STRING_MODE,D_TOKEN_STRING_MODE,D_FLOAT_MODE,D_INTEGER_MODE,D_CHARACTER_MODE,D_HASHBANG_MODE,D_SPECIAL_TOKEN_SEQUENCE_MODE,D_ATTRIBUTE_MODE]}});hljs.registerLanguage("excel",function(hljs){return{aliases:["xlsx","xls"],case_insensitive:true,lexemes:/[a-zA-Z][\w\.]*/,keywords:{built_in:"ABS ACCRINT ACCRINTM ACOS ACOSH ACOT ACOTH AGGREGATE ADDRESS AMORDEGRC AMORLINC AND ARABIC AREAS ASC ASIN ASINH ATAN ATAN2 ATANH AVEDEV AVERAGE AVERAGEA AVERAGEIF AVERAGEIFS BAHTTEXT BASE BESSELI BESSELJ BESSELK BESSELY BETADIST BETA.DIST BETAINV BETA.INV BIN2DEC BIN2HEX BIN2OCT BINOMDIST BINOM.DIST BINOM.DIST.RANGE BINOM.INV BITAND BITLSHIFT BITOR BITRSHIFT BITXOR CALL CEILING CEILING.MATH CEILING.PRECISE CELL CHAR CHIDIST CHIINV CHITEST CHISQ.DIST CHISQ.DIST.RT CHISQ.INV CHISQ.INV.RT CHISQ.TEST CHOOSE CLEAN CODE COLUMN COLUMNS COMBIN COMBINA COMPLEX CONCAT CONCATENATE CONFIDENCE CONFIDENCE.NORM CONFIDENCE.T CONVERT CORREL COS COSH COT COTH COUNT COUNTA COUNTBLANK COUNTIF COUNTIFS COUPDAYBS COUPDAYS COUPDAYSNC COUPNCD COUPNUM COUPPCD COVAR COVARIANCE.P COVARIANCE.S CRITBINOM CSC CSCH CUBEKPIMEMBER CUBEMEMBER CUBEMEMBERPROPERTY CUBERANKEDMEMBER CUBESET CUBESETCOUNT CUBEVALUE CUMIPMT CUMPRINC DATE DATEDIF DATEVALUE DAVERAGE DAY DAYS DAYS360 DB DBCS DCOUNT DCOUNTA DDB DEC2BIN DEC2HEX DEC2OCT DECIMAL DEGREES DELTA DEVSQ DGET DISC DMAX DMIN DOLLAR DOLLARDE DOLLARFR DPRODUCT DSTDEV DSTDEVP DSUM DURATION DVAR DVARP EDATE EFFECT ENCODEURL EOMONTH ERF ERF.PRECISE ERFC ERFC.PRECISE ERROR.TYPE EUROCONVERT EVEN EXACT EXP EXPON.DIST EXPONDIST FACT FACTDOUBLE FALSE|0 F.DIST FDIST F.DIST.RT FILTERXML FIND FINDB F.INV F.INV.RT FINV FISHER FISHERINV FIXED FLOOR FLOOR.MATH FLOOR.PRECISE FORECAST FORECAST.ETS FORECAST.ETS.CONFINT FORECAST.ETS.SEASONALITY FORECAST.ETS.STAT FORECAST.LINEAR FORMULATEXT FREQUENCY F.TEST FTEST FV FVSCHEDULE GAMMA GAMMA.DIST GAMMADIST GAMMA.INV GAMMAINV GAMMALN GAMMALN.PRECISE GAUSS GCD GEOMEAN GESTEP GETPIVOTDATA GROWTH HARMEAN HEX2BIN HEX2DEC HEX2OCT HLOOKUP HOUR HYPERLINK HYPGEOM.DIST HYPGEOMDIST IF IFERROR IFNA IFS IMABS IMAGINARY IMARGUMENT IMCONJUGATE IMCOS IMCOSH IMCOT IMCSC IMCSCH IMDIV IMEXP IMLN IMLOG10 IMLOG2 IMPOWER IMPRODUCT IMREAL IMSEC IMSECH IMSIN IMSINH IMSQRT IMSUB IMSUM IMTAN INDEX INDIRECT INFO INT INTERCEPT INTRATE IPMT IRR ISBLANK ISERR ISERROR ISEVEN ISFORMULA ISLOGICAL ISNA ISNONTEXT ISNUMBER ISODD ISREF ISTEXT ISO.CEILING ISOWEEKNUM ISPMT JIS KURT LARGE LCM LEFT LEFTB LEN LENB LINEST LN LOG LOG10 LOGEST LOGINV LOGNORM.DIST LOGNORMDIST LOGNORM.INV LOOKUP LOWER MATCH MAX MAXA MAXIFS MDETERM MDURATION MEDIAN MID MIDBs MIN MINIFS MINA MINUTE MINVERSE MIRR MMULT MOD MODE MODE.MULT MODE.SNGL MONTH MROUND MULTINOMIAL MUNIT N NA NEGBINOM.DIST NEGBINOMDIST NETWORKDAYS NETWORKDAYS.INTL NOMINAL NORM.DIST NORMDIST NORMINV NORM.INV NORM.S.DIST NORMSDIST NORM.S.INV NORMSINV NOT NOW NPER NPV NUMBERVALUE OCT2BIN OCT2DEC OCT2HEX ODD ODDFPRICE ODDFYIELD ODDLPRICE ODDLYIELD OFFSET OR PDURATION PEARSON PERCENTILE.EXC PERCENTILE.INC PERCENTILE PERCENTRANK.EXC PERCENTRANK.INC PERCENTRANK PERMUT PERMUTATIONA PHI PHONETIC PI PMT POISSON.DIST POISSON POWER PPMT PRICE PRICEDISC PRICEMAT PROB PRODUCT PROPER PV QUARTILE QUARTILE.EXC QUARTILE.INC QUOTIENT RADIANS RAND RANDBETWEEN RANK.AVG RANK.EQ RANK RATE RECEIVED REGISTER.ID REPLACE REPLACEB REPT RIGHT RIGHTB ROMAN ROUND ROUNDDOWN ROUNDUP ROW ROWS RRI RSQ RTD SEARCH SEARCHB SEC SECH SECOND SERIESSUM SHEET SHEETS SIGN SIN SINH SKEW SKEW.P SLN SLOPE SMALL SQL.REQUEST SQRT SQRTPI STANDARDIZE STDEV STDEV.P STDEV.S STDEVA STDEVP STDEVPA STEYX SUBSTITUTE SUBTOTAL SUM SUMIF SUMIFS SUMPRODUCT SUMSQ SUMX2MY2 SUMX2PY2 SUMXMY2 SWITCH SYD T TAN TANH TBILLEQ TBILLPRICE TBILLYIELD T.DIST T.DIST.2T T.DIST.RT TDIST TEXT TEXTJOIN TIME TIMEVALUE T.INV T.INV.2T TINV TODAY TRANSPOSE TREND TRIM TRIMMEAN TRUE|0 TRUNC T.TEST TTEST TYPE UNICHAR UNICODE UPPER VALUE VAR VAR.P VAR.S VARA VARP VARPA VDB VLOOKUP WEBSERVICE WEEKDAY WEEKNUM WEIBULL WEIBULL.DIST WORKDAY WORKDAY.INTL XIRR XNPV XOR YEAR YEARFRAC YIELD YIELDDISC YIELDMAT Z.TEST ZTEST"},contains:[{begin:/^=/,end:/[^=]/,returnEnd:true,illegal:/=/,relevance:10},{className:"symbol",begin:/\b[A-Z]{1,2}\d+\b/,end:/[^\d]/,excludeEnd:true,relevance:0},{className:"symbol",begin:/[A-Z]{0,2}\d*:[A-Z]{0,2}\d*/,relevance:0},hljs.BACKSLASH_ESCAPE,hljs.QUOTE_STRING_MODE,{className:"number",begin:hljs.NUMBER_RE+"(%)?",relevance:0},hljs.COMMENT(/\bN\(/,/\)/,{excludeBegin:true,excludeEnd:true,illegal:/\n/})]}});hljs.registerLanguage("moonscript",function(hljs){var KEYWORDS={keyword:"if then not for in while do return else elseif break continue switch and or "+"unless when class extends super local import export from using",literal:"true false nil",built_in:"_G _VERSION assert collectgarbage dofile error getfenv getmetatable ipairs load "+"loadfile loadstring module next pairs pcall print rawequal rawget rawset require "+"select setfenv setmetatable tonumber tostring type unpack xpcall coroutine debug "+"io math os package string table"};var JS_IDENT_RE="[A-Za-z$_][0-9A-Za-z$_]*";var SUBST={className:"subst",begin:/#\{/,end:/}/,keywords:KEYWORDS};var EXPRESSIONS=[hljs.inherit(hljs.C_NUMBER_MODE,{starts:{end:"(\\s*/)?",relevance:0}}),{className:"string",variants:[{begin:/'/,end:/'/,contains:[hljs.BACKSLASH_ESCAPE]},{begin:/"/,end:/"/,contains:[hljs.BACKSLASH_ESCAPE,SUBST]}]},{className:"built_in",begin:"@__"+hljs.IDENT_RE},{begin:"@"+hljs.IDENT_RE},{begin:hljs.IDENT_RE+"\\\\"+hljs.IDENT_RE}];SUBST.contains=EXPRESSIONS;var TITLE=hljs.inherit(hljs.TITLE_MODE,{begin:JS_IDENT_RE});var PARAMS_RE="(\\(.*\\))?\\s*\\B[-=]>";var PARAMS={className:"params",begin:"\\([^\\(]",returnBegin:true,contains:[{begin:/\(/,end:/\)/,keywords:KEYWORDS,contains:["self"].concat(EXPRESSIONS)}]};return{aliases:["moon"],keywords:KEYWORDS,illegal:/\/\*/,contains:EXPRESSIONS.concat([hljs.COMMENT("--","$"),{className:"function",begin:"^\\s*"+JS_IDENT_RE+"\\s*=\\s*"+PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[TITLE,PARAMS]},{begin:/[\(,:=]\s*/,relevance:0,contains:[{className:"function",begin:PARAMS_RE,end:"[-=]>",returnBegin:true,contains:[PARAMS]}]},{className:"class",beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{beginKeywords:"extends",endsWithParent:true,illegal:/[:="\[\]]/,contains:[TITLE]},TITLE]},{className:"name",begin:JS_IDENT_RE+":",end:":",returnBegin:true,returnEnd:true,relevance:0}])}});hljs.registerLanguage("vala",function(hljs){return{keywords:{keyword:"char uchar unichar int uint long ulong short ushort int8 int16 int32 int64 uint8 "+"uint16 uint32 uint64 float double bool struct enum string void "+"weak unowned owned "+"async signal static abstract interface override virtual delegate "+"if while do for foreach else switch case break default return try catch "+"public private protected internal "+"using new this get set const stdout stdin stderr var",built_in:"DBus GLib CCode Gee Object Gtk Posix",literal:"false true null"},contains:[{className:"class",beginKeywords:"class interface namespace",end:"{",excludeEnd:true,illegal:"[^,:\\n\\s\\.]",contains:[hljs.UNDERSCORE_TITLE_MODE]},hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,{className:"string",begin:'"""',end:'"""',relevance:5},hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"meta",begin:"^#",end:"$",relevance:2}]}});hljs.registerLanguage("ruleslanguage",function(hljs){return{keywords:{keyword:"BILL_PERIOD BILL_START BILL_STOP RS_EFFECTIVE_START RS_EFFECTIVE_STOP RS_JURIS_CODE RS_OPCO_CODE "+"INTDADDATTRIBUTE|5 INTDADDVMSG|5 INTDBLOCKOP|5 INTDBLOCKOPNA|5 INTDCLOSE|5 INTDCOUNT|5 "+"INTDCOUNTSTATUSCODE|5 INTDCREATEMASK|5 INTDCREATEDAYMASK|5 INTDCREATEFACTORMASK|5 "+"INTDCREATEHANDLE|5 INTDCREATEOVERRIDEDAYMASK|5 INTDCREATEOVERRIDEMASK|5 "+"INTDCREATESTATUSCODEMASK|5 INTDCREATETOUPERIOD|5 INTDDELETE|5 INTDDIPTEST|5 INTDEXPORT|5 "+"INTDGETERRORCODE|5 INTDGETERRORMESSAGE|5 INTDISEQUAL|5 INTDJOIN|5 INTDLOAD|5 INTDLOADACTUALCUT|5 "+"INTDLOADDATES|5 INTDLOADHIST|5 INTDLOADLIST|5 INTDLOADLISTDATES|5 INTDLOADLISTENERGY|5 "+"INTDLOADLISTHIST|5 INTDLOADRELATEDCHANNEL|5 INTDLOADSP|5 INTDLOADSTAGING|5 INTDLOADUOM|5 "+"INTDLOADUOMDATES|5 INTDLOADUOMHIST|5 INTDLOADVERSION|5 INTDOPEN|5 INTDREADFIRST|5 INTDREADNEXT|5 "+"INTDRECCOUNT|5 INTDRELEASE|5 INTDREPLACE|5 INTDROLLAVG|5 INTDROLLPEAK|5 INTDSCALAROP|5 INTDSCALE|5 "+"INTDSETATTRIBUTE|5 INTDSETDSTPARTICIPANT|5 INTDSETSTRING|5 INTDSETVALUE|5 INTDSETVALUESTATUS|5 "+"INTDSHIFTSTARTTIME|5 INTDSMOOTH|5 INTDSORT|5 INTDSPIKETEST|5 INTDSUBSET|5 INTDTOU|5 "+"INTDTOURELEASE|5 INTDTOUVALUE|5 INTDUPDATESTATS|5 INTDVALUE|5 STDEV INTDDELETEEX|5 "+"INTDLOADEXACTUAL|5 INTDLOADEXCUT|5 INTDLOADEXDATES|5 INTDLOADEX|5 INTDLOADEXRELATEDCHANNEL|5 "+"INTDSAVEEX|5 MVLOAD|5 MVLOADACCT|5 MVLOADACCTDATES|5 MVLOADACCTHIST|5 MVLOADDATES|5 MVLOADHIST|5 "+"MVLOADLIST|5 MVLOADLISTDATES|5 MVLOADLISTHIST|5 IF FOR NEXT DONE SELECT END CALL ABORT CLEAR CHANNEL FACTOR LIST NUMBER "+"OVERRIDE SET WEEK DISTRIBUTIONNODE ELSE WHEN THEN OTHERWISE IENUM CSV INCLUDE LEAVE RIDER SAVE DELETE "+"NOVALUE SECTION WARN SAVE_UPDATE DETERMINANT LABEL REPORT REVENUE EACH "+"IN FROM TOTAL CHARGE BLOCK AND OR CSV_FILE RATE_CODE AUXILIARY_DEMAND "+"UIDACCOUNT RS BILL_PERIOD_SELECT HOURS_PER_MONTH INTD_ERROR_STOP SEASON_SCHEDULE_NAME "+"ACCOUNTFACTOR ARRAYUPPERBOUND CALLSTOREDPROC GETADOCONNECTION GETCONNECT GETDATASOURCE "+"GETQUALIFIER GETUSERID HASVALUE LISTCOUNT LISTOP LISTUPDATE LISTVALUE PRORATEFACTOR RSPRORATE "+"SETBINPATH SETDBMONITOR WQ_OPEN BILLINGHOURS DATE DATEFROMFLOAT DATETIMEFROMSTRING "+"DATETIMETOSTRING DATETOFLOAT DAY DAYDIFF DAYNAME DBDATETIME HOUR MINUTE MONTH MONTHDIFF "+"MONTHHOURS MONTHNAME ROUNDDATE SAMEWEEKDAYLASTYEAR SECOND WEEKDAY WEEKDIFF YEAR YEARDAY "+"YEARSTR COMPSUM HISTCOUNT HISTMAX HISTMIN HISTMINNZ HISTVALUE MAXNRANGE MAXRANGE MINRANGE "+"COMPIKVA COMPKVA COMPKVARFROMKQKW COMPLF IDATTR FLAG LF2KW LF2KWH MAXKW POWERFACTOR "+"READING2USAGE AVGSEASON MAXSEASON MONTHLYMERGE SEASONVALUE SUMSEASON ACCTREADDATES "+"ACCTTABLELOAD CONFIGADD CONFIGGET CREATEOBJECT CREATEREPORT EMAILCLIENT EXPBLKMDMUSAGE "+"EXPMDMUSAGE EXPORT_USAGE FACTORINEFFECT GETUSERSPECIFIEDSTOP INEFFECT ISHOLIDAY RUNRATE "+"SAVE_PROFILE SETREPORTTITLE USEREXIT WATFORRUNRATE TO TABLE ACOS ASIN ATAN ATAN2 BITAND CEIL "+"COS COSECANT COSH COTANGENT DIVQUOT DIVREM EXP FABS FLOOR FMOD FREPM FREXPN LOG LOG10 MAX MAXN "+"MIN MINNZ MODF POW ROUND ROUND2VALUE ROUNDINT SECANT SIN SINH SQROOT TAN TANH FLOAT2STRING "+"FLOAT2STRINGNC INSTR LEFT LEN LTRIM MID RIGHT RTRIM STRING STRINGNC TOLOWER TOUPPER TRIM "+"NUMDAYS READ_DATE STAGING",built_in:"IDENTIFIER OPTIONS XML_ELEMENT XML_OP XML_ELEMENT_OF DOMDOCCREATE DOMDOCLOADFILE DOMDOCLOADXML "+"DOMDOCSAVEFILE DOMDOCGETROOT DOMDOCADDPI DOMNODEGETNAME DOMNODEGETTYPE DOMNODEGETVALUE DOMNODEGETCHILDCT "+"DOMNODEGETFIRSTCHILD DOMNODEGETSIBLING DOMNODECREATECHILDELEMENT DOMNODESETATTRIBUTE "+"DOMNODEGETCHILDELEMENTCT DOMNODEGETFIRSTCHILDELEMENT DOMNODEGETSIBLINGELEMENT DOMNODEGETATTRIBUTECT "+"DOMNODEGETATTRIBUTEI DOMNODEGETATTRIBUTEBYNAME DOMNODEGETBYNAME"},contains:[hljs.C_LINE_COMMENT_MODE,hljs.C_BLOCK_COMMENT_MODE,hljs.APOS_STRING_MODE,hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE,{className:"literal",variants:[{begin:"#\\s+[a-zA-Z\\ \\.]*",relevance:0},{begin:"#[a-zA-Z\\ \\.]+"}]}]}});hljs.registerLanguage("verilog",function(hljs){var SV_KEYWORDS={keyword:"accept_on alias always always_comb always_ff always_latch and assert assign "+"assume automatic before begin bind bins binsof bit break buf|0 bufif0 bufif1 "+"byte case casex casez cell chandle checker class clocking cmos config const "+"constraint context continue cover covergroup coverpoint cross deassign default "+"defparam design disable dist do edge else end endcase endchecker endclass "+"endclocking endconfig endfunction endgenerate endgroup endinterface endmodule "+"endpackage endprimitive endprogram endproperty endspecify endsequence endtable "+"endtask enum event eventually expect export extends extern final first_match for "+"force foreach forever fork forkjoin function generate|5 genvar global highz0 highz1 "+"if iff ifnone ignore_bins illegal_bins implements implies import incdir include "+"initial inout input inside instance int integer interconnect interface intersect "+"join join_any join_none large let liblist library local localparam logic longint "+"macromodule matches medium modport module nand negedge nettype new nexttime nmos "+"nor noshowcancelled not notif0 notif1 or output package packed parameter pmos "+"posedge primitive priority program property protected pull0 pull1 pulldown pullup "+"pulsestyle_ondetect pulsestyle_onevent pure rand randc randcase randsequence rcmos "+"real realtime ref reg reject_on release repeat restrict return rnmos rpmos rtran "+"rtranif0 rtranif1 s_always s_eventually s_nexttime s_until s_until_with scalared "+"sequence shortint shortreal showcancelled signed small soft solve specify specparam "+"static string strong strong0 strong1 struct super supply0 supply1 sync_accept_on "+"sync_reject_on table tagged task this throughout time timeprecision timeunit tran "+"tranif0 tranif1 tri tri0 tri1 triand trior trireg type typedef union unique unique0 "+"unsigned until until_with untyped use uwire var vectored virtual void wait wait_order "+"wand weak weak0 weak1 while wildcard wire with within wor xnor xor",literal:"null",built_in:"$finish $stop $exit $fatal $error $warning $info $realtime $time $printtimescale "+"$bitstoreal $bitstoshortreal $itor $signed $cast $bits $stime $timeformat "+"$realtobits $shortrealtobits $rtoi $unsigned $asserton $assertkill $assertpasson "+"$assertfailon $assertnonvacuouson $assertoff $assertcontrol $assertpassoff "+"$assertfailoff $assertvacuousoff $isunbounded $sampled $fell $changed $past_gclk "+"$fell_gclk $changed_gclk $rising_gclk $steady_gclk $coverage_control "+"$coverage_get $coverage_save $set_coverage_db_name $rose $stable $past "+"$rose_gclk $stable_gclk $future_gclk $falling_gclk $changing_gclk $display "+"$coverage_get_max $coverage_merge $get_coverage $load_coverage_db $typename "+"$unpacked_dimensions $left $low $increment $clog2 $ln $log10 $exp $sqrt $pow "+"$floor $ceil $sin $cos $tan $countbits $onehot $isunknown $fatal $warning "+"$dimensions $right $high $size $asin $acos $atan $atan2 $hypot $sinh $cosh "+"$tanh $asinh $acosh $atanh $countones $onehot0 $error $info $random "+"$dist_chi_square $dist_erlang $dist_exponential $dist_normal $dist_poisson "+"$dist_t $dist_uniform $q_initialize $q_remove $q_exam $async$and$array "+"$async$nand$array $async$or$array $async$nor$array $sync$and$array "+"$sync$nand$array $sync$or$array $sync$nor$array $q_add $q_full $psprintf "+"$async$and$plane $async$nand$plane $async$or$plane $async$nor$plane "+"$sync$and$plane $sync$nand$plane $sync$or$plane $sync$nor$plane $system "+"$display $displayb $displayh $displayo $strobe $strobeb $strobeh $strobeo "+"$write $readmemb $readmemh $writememh $value$plusargs "+"$dumpvars $dumpon $dumplimit $dumpports $dumpportson $dumpportslimit "+"$writeb $writeh $writeo $monitor $monitorb $monitorh $monitoro $writememb "+"$dumpfile $dumpoff $dumpall $dumpflush $dumpportsoff $dumpportsall "+"$dumpportsflush $fclose $fdisplay $fdisplayb $fdisplayh $fdisplayo "+"$fstrobe $fstrobeb $fstrobeh $fstrobeo $swrite $swriteb $swriteh "+"$swriteo $fscanf $fread $fseek $fflush $feof $fopen $fwrite $fwriteb "+"$fwriteh $fwriteo $fmonitor $fmonitorb $fmonitorh $fmonitoro $sformat "+"$sformatf $fgetc $ungetc $fgets $sscanf $rewind $ftell $ferror"};return{aliases:["v","sv","svh"],case_insensitive:false,keywords:SV_KEYWORDS,lexemes:/[\w\$]+/,contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.C_LINE_COMMENT_MODE,hljs.QUOTE_STRING_MODE,{className:"number",contains:[hljs.BACKSLASH_ESCAPE],variants:[{begin:"\\b((\\d+'(b|h|o|d|B|H|O|D))[0-9xzXZa-fA-F_]+)"},{begin:"\\B(('(b|h|o|d|B|H|O|D))[0-9xzXZa-fA-F_]+)"},{begin:"\\b([0-9_])+",relevance:0}]},{className:"variable",variants:[{begin:"#\\((?!parameter).+\\)"},{begin:"\\.\\w+",relevance:0}]},{className:"meta",begin:"`",end:"$",keywords:{"meta-keyword":"define __FILE__ "+"__LINE__ begin_keywords celldefine default_nettype define "+"else elsif end_keywords endcelldefine endif ifdef ifndef "+"include line nounconnected_drive pragma resetall timescale "+"unconnected_drive undef undefineall"},relevance:0}]}});hljs.registerLanguage("mathematica",function(hljs){return{aliases:["mma","wl"],lexemes:"(\\$|\\b)"+hljs.IDENT_RE+"\\b",keywords:"AASTriangle AbelianGroup Abort AbortKernels AbortProtect AbortScheduledTask Above Abs AbsArg AbsArgPlot Absolute AbsoluteCorrelation AbsoluteCorrelationFunction AbsoluteCurrentValue AbsoluteDashing AbsoluteFileName AbsoluteOptions AbsolutePointSize AbsoluteThickness AbsoluteTime AbsoluteTiming AcceptanceThreshold AccountingForm Accumulate Accuracy AccuracyGoal ActionDelay ActionMenu ActionMenuBox ActionMenuBoxOptions Activate Active ActiveClassification ActiveClassificationObject ActiveItem ActivePrediction ActivePredictionObject ActiveStyle AcyclicGraphQ AddOnHelpPath AddSides AddTo AddToSearchIndex AddUsers AdjacencyGraph AdjacencyList AdjacencyMatrix AdjustmentBox AdjustmentBoxOptions AdjustTimeSeriesForecast AdministrativeDivisionData AffineHalfSpace AffineSpace AffineStateSpaceModel AffineTransform After AggregatedEntityClass AggregationLayer AircraftData AirportData AirPressureData AirTemperatureData AiryAi AiryAiPrime AiryAiZero AiryBi AiryBiPrime AiryBiZero AlgebraicIntegerQ AlgebraicNumber AlgebraicNumberDenominator AlgebraicNumberNorm AlgebraicNumberPolynomial AlgebraicNumberTrace AlgebraicRules AlgebraicRulesData Algebraics AlgebraicUnitQ Alignment AlignmentMarker AlignmentPoint All AllowAdultContent AllowedCloudExtraParameters AllowedCloudParameterExtensions AllowedDimensions AllowedFrequencyRange AllowedHeads AllowGroupClose AllowIncomplete AllowInlineCells AllowKernelInitialization AllowLooseGrammar AllowReverseGroupClose AllowScriptLevelChange AllTrue Alphabet AlphabeticOrder AlphabeticSort AlphaChannel AlternateImage AlternatingFactorial AlternatingGroup AlternativeHypothesis Alternatives AltitudeMethod AmbientLight AmbiguityFunction AmbiguityList Analytic AnatomyData AnatomyForm AnatomyPlot3D AnatomySkinStyle AnatomyStyling AnchoredSearch And AndersonDarlingTest AngerJ AngleBisector AngleBracket AnglePath AnglePath3D AngleVector AngularGauge Animate AnimationCycleOffset AnimationCycleRepetitions AnimationDirection AnimationDisplayTime AnimationRate AnimationRepetitions AnimationRunning AnimationRunTime AnimationTimeIndex Animator AnimatorBox AnimatorBoxOptions AnimatorElements Annotate Annotation AnnotationDelete AnnotationNames AnnotationRules AnnotationValue Annuity AnnuityDue Annulus AnomalyDetection AnomalyDetectorFunction Anonymous Antialiasing AntihermitianMatrixQ Antisymmetric AntisymmetricMatrixQ Antonyms AnyOrder AnySubset AnyTrue Apart ApartSquareFree APIFunction Appearance AppearanceElements AppearanceRules AppellF1 Append AppendCheck AppendLayer AppendTo ApplicationIdentificationKey Apply ApplySides ArcCos ArcCosh ArcCot ArcCoth ArcCsc ArcCsch ArcCurvature ARCHProcess ArcLength ArcSec ArcSech ArcSin ArcSinDistribution ArcSinh ArcTan ArcTanh Area Arg ArgMax ArgMin ArgumentCountQ ARIMAProcess ArithmeticGeometricMean ARMAProcess Around AroundReplace ARProcess Array ArrayComponents ArrayDepth ArrayFilter ArrayFlatten ArrayMesh ArrayPad ArrayPlot ArrayQ ArrayResample ArrayReshape ArrayRules Arrays Arrow Arrow3DBox ArrowBox Arrowheads ASATriangle Ask AskAppend AskConfirm AskDisplay AskedQ AskedValue AskFunction AskState AskTemplateDisplay AspectRatio AspectRatioFixed Assert AssociateTo Association AssociationFormat AssociationMap AssociationQ AssociationThread AssumeDeterministic Assuming Assumptions AstronomicalData AsymptoticDSolveValue AsymptoticEqual AsymptoticEquivalent AsymptoticGreater AsymptoticGreaterEqual AsymptoticIntegrate AsymptoticLess AsymptoticLessEqual AsymptoticOutputTracker AsymptoticRSolveValue AsymptoticSolve AsymptoticSum Asynchronous AsynchronousTaskObject AsynchronousTasks Atom AtomCoordinates AtomCount AtomDiagramCoordinates AtomList AtomQ AttentionLayer Attributes Audio AudioAmplify AudioAnnotate AudioAnnotationLookup AudioBlockMap AudioCapture AudioChannelAssignment AudioChannelCombine AudioChannelMix AudioChannels AudioChannelSeparate AudioData AudioDelay AudioDelete AudioDevice AudioDistance AudioFade AudioFrequencyShift AudioGenerator AudioIdentify AudioInputDevice AudioInsert AudioIntervals AudioJoin AudioLabel AudioLength AudioLocalMeasurements AudioLooping AudioLoudness AudioMeasurements AudioNormalize AudioOutputDevice AudioOverlay AudioPad AudioPan AudioPartition AudioPause AudioPitchShift AudioPlay AudioPlot AudioQ AudioRecord AudioReplace AudioResample AudioReverb AudioSampleRate AudioSpectralMap AudioSpectralTransformation AudioSplit AudioStop AudioStream AudioStreams AudioTimeStretch AudioTrim AudioType AugmentedPolyhedron AugmentedSymmetricPolynomial Authenticate Authentication AuthenticationDialog AutoAction Autocomplete AutocompletionFunction AutoCopy AutocorrelationTest AutoDelete AutoEvaluateEvents AutoGeneratedPackage AutoIndent AutoIndentSpacings AutoItalicWords AutoloadPath AutoMatch Automatic AutomaticImageSize AutoMultiplicationSymbol AutoNumberFormatting AutoOpenNotebooks AutoOpenPalettes AutoQuoteCharacters AutoRefreshed AutoRemove AutorunSequencing AutoScaling AutoScroll AutoSpacing AutoStyleOptions AutoStyleWords AutoSubmitting Axes AxesEdge AxesLabel AxesOrigin AxesStyle AxiomaticTheory Axis"+"BabyMonsterGroupB Back Background BackgroundAppearance BackgroundTasksSettings Backslash Backsubstitution Backward Ball Band BandpassFilter BandstopFilter BarabasiAlbertGraphDistribution BarChart BarChart3D BarcodeImage BarcodeRecognize BaringhausHenzeTest BarLegend BarlowProschanImportance BarnesG BarOrigin BarSpacing BartlettHannWindow BartlettWindow BaseDecode BaseEncode BaseForm Baseline BaselinePosition BaseStyle BasicRecurrentLayer BatchNormalizationLayer BatchSize BatesDistribution BattleLemarieWavelet BayesianMaximization BayesianMaximizationObject BayesianMinimization BayesianMinimizationObject Because BeckmannDistribution Beep Before Begin BeginDialogPacket BeginFrontEndInteractionPacket BeginPackage BellB BellY Below BenfordDistribution BeniniDistribution BenktanderGibratDistribution BenktanderWeibullDistribution BernoulliB BernoulliDistribution BernoulliGraphDistribution BernoulliProcess BernsteinBasis BesselFilterModel BesselI BesselJ BesselJZero BesselK BesselY BesselYZero Beta BetaBinomialDistribution BetaDistribution BetaNegativeBinomialDistribution BetaPrimeDistribution BetaRegularized Between BetweennessCentrality BeveledPolyhedron BezierCurve BezierCurve3DBox BezierCurve3DBoxOptions BezierCurveBox BezierCurveBoxOptions BezierFunction BilateralFilter Binarize BinaryDeserialize BinaryDistance BinaryFormat BinaryImageQ BinaryRead BinaryReadList BinarySerialize BinaryWrite BinCounts BinLists Binomial BinomialDistribution BinomialProcess BinormalDistribution BiorthogonalSplineWavelet BipartiteGraphQ BiquadraticFilterModel BirnbaumImportance BirnbaumSaundersDistribution BitAnd BitClear BitGet BitLength BitNot BitOr BitSet BitShiftLeft BitShiftRight BitXor BiweightLocation BiweightMidvariance Black BlackmanHarrisWindow BlackmanNuttallWindow BlackmanWindow Blank BlankForm BlankNullSequence BlankSequence Blend Block BlockchainAddressData BlockchainBase BlockchainBlockData BlockchainContractValue BlockchainData BlockchainGet BlockchainKeyEncode BlockchainPut BlockchainTokenData BlockchainTransaction BlockchainTransactionData BlockchainTransactionSign BlockchainTransactionSubmit BlockMap BlockRandom BlomqvistBeta BlomqvistBetaTest Blue Blur BodePlot BohmanWindow Bold Bond BondCount BondList BondQ Bookmarks Boole BooleanConsecutiveFunction BooleanConvert BooleanCountingFunction BooleanFunction BooleanGraph BooleanMaxterms BooleanMinimize BooleanMinterms BooleanQ BooleanRegion Booleans BooleanStrings BooleanTable BooleanVariables BorderDimensions BorelTannerDistribution Bottom BottomHatTransform BoundaryDiscretizeGraphics BoundaryDiscretizeRegion BoundaryMesh BoundaryMeshRegion BoundaryMeshRegionQ BoundaryStyle BoundedRegionQ BoundingRegion Bounds Box BoxBaselineShift BoxData BoxDimensions Boxed Boxes BoxForm BoxFormFormatTypes BoxFrame BoxID BoxMargins BoxMatrix BoxObject BoxRatios BoxRotation BoxRotationPoint BoxStyle BoxWhiskerChart Bra BracketingBar BraKet BrayCurtisDistance BreadthFirstScan Break BridgeData BrightnessEqualize BroadcastStationData Brown BrownForsytheTest BrownianBridgeProcess BrowserCategory BSplineBasis BSplineCurve BSplineCurve3DBox BSplineCurve3DBoxOptions BSplineCurveBox BSplineCurveBoxOptions BSplineFunction BSplineSurface BSplineSurface3DBox BSplineSurface3DBoxOptions BubbleChart BubbleChart3D BubbleScale BubbleSizes BuildingData BulletGauge BusinessDayQ ButterflyGraph ButterworthFilterModel Button ButtonBar ButtonBox ButtonBoxOptions ButtonCell ButtonContents ButtonData ButtonEvaluator ButtonExpandable ButtonFrame ButtonFunction ButtonMargins ButtonMinHeight ButtonNote ButtonNotebook ButtonSource ButtonStyle ButtonStyleMenuListing Byte ByteArray ByteArrayFormat ByteArrayQ ByteArrayToString ByteCount ByteOrdering"+"C CachedValue CacheGraphics CachePersistence CalendarConvert CalendarData CalendarType Callout CalloutMarker CalloutStyle CallPacket CanberraDistance Cancel CancelButton CandlestickChart CanonicalGraph CanonicalizePolygon CanonicalizePolyhedron CanonicalName CanonicalWarpingCorrespondence CanonicalWarpingDistance CantorMesh CantorStaircase Cap CapForm CapitalDifferentialD Capitalize CapsuleShape CaptureRunning CardinalBSplineBasis CarlemanLinearize CarmichaelLambda CaseOrdering Cases CaseSensitive Cashflow Casoratian Catalan CatalanNumber Catch Catenate CatenateLayer CauchyDistribution CauchyWindow CayleyGraph CDF CDFDeploy CDFInformation CDFWavelet Ceiling CelestialSystem Cell CellAutoOverwrite CellBaseline CellBoundingBox CellBracketOptions CellChangeTimes CellContents CellContext CellDingbat CellDynamicExpression CellEditDuplicate CellElementsBoundingBox CellElementSpacings CellEpilog CellEvaluationDuplicate CellEvaluationFunction CellEvaluationLanguage CellEventActions CellFrame CellFrameColor CellFrameLabelMargins CellFrameLabels CellFrameMargins CellGroup CellGroupData CellGrouping CellGroupingRules CellHorizontalScrolling CellID CellLabel CellLabelAutoDelete CellLabelMargins CellLabelPositioning CellLabelStyle CellLabelTemplate CellMargins CellObject CellOpen CellPrint CellProlog Cells CellSize CellStyle CellTags CellularAutomaton CensoredDistribution Censoring Center CenterArray CenterDot CentralFeature CentralMoment CentralMomentGeneratingFunction Cepstrogram CepstrogramArray CepstrumArray CForm ChampernowneNumber ChangeOptions ChannelBase ChannelBrokerAction ChannelDatabin ChannelHistoryLength ChannelListen ChannelListener ChannelListeners ChannelListenerWait ChannelObject ChannelPreSendFunction ChannelReceiverFunction ChannelSend ChannelSubscribers ChanVeseBinarize Character CharacterCounts CharacterEncoding CharacterEncodingsPath CharacteristicFunction CharacteristicPolynomial CharacterName CharacterRange Characters ChartBaseStyle ChartElementData ChartElementDataFunction ChartElementFunction ChartElements ChartLabels ChartLayout ChartLegends ChartStyle Chebyshev1FilterModel Chebyshev2FilterModel ChebyshevDistance ChebyshevT ChebyshevU Check CheckAbort CheckAll Checkbox CheckboxBar CheckboxBox CheckboxBoxOptions ChemicalData ChessboardDistance ChiDistribution ChineseRemainder ChiSquareDistribution ChoiceButtons ChoiceDialog CholeskyDecomposition Chop ChromaticityPlot ChromaticityPlot3D ChromaticPolynomial Circle CircleBox CircleDot CircleMinus CirclePlus CirclePoints CircleThrough CircleTimes CirculantGraph CircularOrthogonalMatrixDistribution CircularQuaternionMatrixDistribution CircularRealMatrixDistribution CircularSymplecticMatrixDistribution CircularUnitaryMatrixDistribution Circumsphere CityData ClassifierFunction ClassifierInformation ClassifierMeasurements ClassifierMeasurementsObject Classify ClassPriors Clear ClearAll ClearAttributes ClearCookies ClearPermissions ClearSystemCache ClebschGordan ClickPane Clip ClipboardNotebook ClipFill ClippingStyle ClipPlanes ClipPlanesStyle ClipRange Clock ClockGauge ClockwiseContourIntegral Close Closed CloseKernels ClosenessCentrality Closing ClosingAutoSave ClosingEvent CloudAccountData CloudBase CloudConnect CloudDeploy CloudDirectory CloudDisconnect CloudEvaluate CloudExport CloudExpression CloudExpressions CloudFunction CloudGet CloudImport CloudLoggingData CloudObject CloudObjectInformation CloudObjectInformationData CloudObjectNameFormat CloudObjects CloudObjectURLType CloudPublish CloudPut CloudRenderingMethod CloudSave CloudShare CloudSubmit CloudSymbol CloudUnshare ClusterClassify ClusterDissimilarityFunction ClusteringComponents ClusteringTree CMYKColor Coarse CodeAssistOptions Coefficient CoefficientArrays CoefficientDomain CoefficientList CoefficientRules CoifletWavelet Collect Colon ColonForm ColorBalance ColorCombine ColorConvert ColorCoverage ColorData ColorDataFunction ColorDetect ColorDistance ColorFunction ColorFunctionScaling Colorize ColorNegate ColorOutput ColorProfileData ColorQ ColorQuantize ColorReplace ColorRules ColorSelectorSettings ColorSeparate ColorSetter ColorSetterBox ColorSetterBoxOptions ColorSlider ColorsNear ColorSpace ColorToneMapping Column ColumnAlignments ColumnBackgrounds ColumnForm ColumnLines ColumnsEqual ColumnSpacings ColumnWidths CombinedEntityClass CombinerFunction CometData CommonDefaultFormatTypes Commonest CommonestFilter CommonName CommonUnits CommunityBoundaryStyle CommunityGraphPlot CommunityLabels CommunityRegionStyle CompanyData CompatibleUnitQ CompilationOptions CompilationTarget Compile Compiled CompiledCodeFunction CompiledFunction CompilerOptions Complement CompleteGraph CompleteGraphQ CompleteKaryTree CompletionsListPacket Complex Complexes ComplexExpand ComplexInfinity ComplexityFunction ComplexListPlot ComplexPlot ComplexPlot3D ComponentMeasurements ComponentwiseContextMenu Compose ComposeList ComposeSeries CompositeQ Composition CompoundElement CompoundExpression CompoundPoissonDistribution CompoundPoissonProcess CompoundRenewalProcess Compress CompressedData ComputeUncertainty Condition ConditionalExpression Conditioned Cone ConeBox ConfidenceLevel ConfidenceRange ConfidenceTransform ConfigurationPath ConformAudio ConformImages Congruent ConicHullRegion ConicHullRegion3DBox ConicHullRegionBox ConicOptimization Conjugate ConjugateTranspose Conjunction Connect ConnectedComponents ConnectedGraphComponents ConnectedGraphQ ConnectedMeshComponents ConnectedMoleculeComponents ConnectedMoleculeQ ConnectionSettings ConnectLibraryCallbackFunction ConnectSystemModelComponents ConnesWindow ConoverTest ConsoleMessage ConsoleMessagePacket ConsolePrint Constant ConstantArray ConstantArrayLayer ConstantImage ConstantPlusLayer ConstantRegionQ Constants ConstantTimesLayer ConstellationData ConstrainedMax ConstrainedMin Construct Containing ContainsAll ContainsAny ContainsExactly ContainsNone ContainsOnly ContentFieldOptions ContentLocationFunction ContentObject ContentPadding ContentsBoundingBox ContentSelectable ContentSize Context ContextMenu Contexts ContextToFileName Continuation Continue ContinuedFraction ContinuedFractionK ContinuousAction ContinuousMarkovProcess ContinuousTask ContinuousTimeModelQ ContinuousWaveletData ContinuousWaveletTransform ContourDetect ContourGraphics ContourIntegral ContourLabels ContourLines ContourPlot ContourPlot3D Contours ContourShading ContourSmoothing ContourStyle ContraharmonicMean ContrastiveLossLayer Control ControlActive ControlAlignment ControlGroupContentsBox ControllabilityGramian ControllabilityMatrix ControllableDecomposition ControllableModelQ ControllerDuration ControllerInformation ControllerInformationData ControllerLinking ControllerManipulate ControllerMethod ControllerPath ControllerState ControlPlacement ControlsRendering ControlType Convergents ConversionOptions ConversionRules ConvertToBitmapPacket ConvertToPostScript ConvertToPostScriptPacket ConvexHullMesh ConvexPolygonQ ConvexPolyhedronQ ConvolutionLayer Convolve ConwayGroupCo1 ConwayGroupCo2 ConwayGroupCo3 CookieFunction Cookies CoordinateBoundingBox CoordinateBoundingBoxArray CoordinateBounds CoordinateBoundsArray CoordinateChartData CoordinatesToolOptions CoordinateTransform CoordinateTransformData CoprimeQ Coproduct CopulaDistribution Copyable CopyDatabin CopyDirectory CopyFile CopyTag CopyToClipboard CornerFilter CornerNeighbors Correlation CorrelationDistance CorrelationFunction CorrelationTest Cos Cosh CoshIntegral CosineDistance CosineWindow CosIntegral Cot Coth Count CountDistinct CountDistinctBy CounterAssignments CounterBox CounterBoxOptions CounterClockwiseContourIntegral CounterEvaluator CounterFunction CounterIncrements CounterStyle CounterStyleMenuListing CountRoots CountryData Counts CountsBy Covariance CovarianceEstimatorFunction CovarianceFunction CoxianDistribution CoxIngersollRossProcess CoxModel CoxModelFit CramerVonMisesTest CreateArchive CreateCellID CreateChannel CreateCloudExpression CreateDatabin CreateDataSystemModel CreateDialog CreateDirectory CreateDocument CreateFile CreateIntermediateDirectories CreateManagedLibraryExpression CreateNotebook CreatePalette CreatePalettePacket CreatePermissionsGroup CreateScheduledTask CreateSearchIndex CreateSystemModel CreateTemporary CreateUUID CreateWindow CriterionFunction CriticalityFailureImportance CriticalitySuccessImportance CriticalSection Cross CrossEntropyLossLayer CrossingCount CrossingDetect CrossingPolygon CrossMatrix Csc Csch CTCLossLayer Cube CubeRoot Cubics Cuboid CuboidBox Cumulant CumulantGeneratingFunction Cup CupCap Curl CurlyDoubleQuote CurlyQuote CurrencyConvert CurrentDate CurrentImage CurrentlySpeakingPacket CurrentNotebookImage CurrentScreenImage CurrentValue Curry CurvatureFlowFilter CurveClosed Cyan CycleGraph CycleIndexPolynomial Cycles CyclicGroup Cyclotomic Cylinder CylinderBox CylindricalDecomposition"+"D DagumDistribution DamData DamerauLevenshteinDistance DampingFactor Darker Dashed Dashing DatabaseConnect DatabaseDisconnect DatabaseReference Databin DatabinAdd DatabinRemove Databins DatabinUpload DataCompression DataDistribution DataRange DataReversed Dataset Date DateBounds Dated DateDelimiters DateDifference DatedUnit DateFormat DateFunction DateHistogram DateList DateListLogPlot DateListPlot DateListStepPlot DateObject DateObjectQ DateOverlapsQ DatePattern DatePlus DateRange DateReduction DateString DateTicksFormat DateValue DateWithinQ DaubechiesWavelet DavisDistribution DawsonF DayCount DayCountConvention DayHemisphere DaylightQ DayMatchQ DayName DayNightTerminator DayPlus DayRange DayRound DeBruijnGraph DeBruijnSequence Debug DebugTag Decapitalize Decimal DecimalForm DeclareKnownSymbols DeclarePackage Decompose DeconvolutionLayer Decrement Decrypt DecryptFile DedekindEta DeepSpaceProbeData Default DefaultAxesStyle DefaultBaseStyle DefaultBoxStyle DefaultButton DefaultColor DefaultControlPlacement DefaultDuplicateCellStyle DefaultDuration DefaultElement DefaultFaceGridsStyle DefaultFieldHintStyle DefaultFont DefaultFontProperties DefaultFormatType DefaultFormatTypeForStyle DefaultFrameStyle DefaultFrameTicksStyle DefaultGridLinesStyle DefaultInlineFormatType DefaultInputFormatType DefaultLabelStyle DefaultMenuStyle DefaultNaturalLanguage DefaultNewCellStyle DefaultNewInlineCellStyle DefaultNotebook DefaultOptions DefaultOutputFormatType DefaultPrintPrecision DefaultStyle DefaultStyleDefinitions DefaultTextFormatType DefaultTextInlineFormatType DefaultTicksStyle DefaultTooltipStyle DefaultValue DefaultValues Defer DefineExternal DefineInputStreamMethod DefineOutputStreamMethod DefineResourceFunction Definition Degree DegreeCentrality DegreeGraphDistribution DegreeLexicographic DegreeReverseLexicographic DEigensystem DEigenvalues Deinitialization Del DelaunayMesh Delayed Deletable Delete DeleteAnomalies DeleteBorderComponents DeleteCases DeleteChannel DeleteCloudExpression DeleteContents DeleteDirectory DeleteDuplicates DeleteDuplicatesBy DeleteFile DeleteMissing DeleteObject DeletePermissionsKey DeleteSearchIndex DeleteSmallComponents DeleteStopwords DeleteWithContents DeletionWarning DelimitedArray DelimitedSequence Delimiter DelimiterFlashTime DelimiterMatching Delimiters DeliveryFunction Dendrogram Denominator DensityGraphics DensityHistogram DensityPlot DensityPlot3D DependentVariables Deploy Deployed Depth DepthFirstScan Derivative DerivativeFilter DerivedKey DescriptorStateSpace DesignMatrix DestroyAfterEvaluation Det DeviceClose DeviceConfigure DeviceExecute DeviceExecuteAsynchronous DeviceObject DeviceOpen DeviceOpenQ DeviceRead DeviceReadBuffer DeviceReadLatest DeviceReadList DeviceReadTimeSeries Devices DeviceStreams DeviceWrite DeviceWriteBuffer DGaussianWavelet DiacriticalPositioning Diagonal DiagonalizableMatrixQ DiagonalMatrix DiagonalMatrixQ Dialog DialogIndent DialogInput DialogLevel DialogNotebook DialogProlog DialogReturn DialogSymbols Diamond DiamondMatrix DiceDissimilarity DictionaryLookup DictionaryWordQ DifferenceDelta DifferenceOrder DifferenceQuotient DifferenceRoot DifferenceRootReduce Differences DifferentialD DifferentialRoot DifferentialRootReduce DifferentiatorFilter DigitalSignature DigitBlock DigitBlockMinimum DigitCharacter DigitCount DigitQ DihedralAngle DihedralGroup Dilation DimensionalCombinations DimensionalMeshComponents DimensionReduce DimensionReducerFunction DimensionReduction Dimensions DiracComb DiracDelta DirectedEdge DirectedEdges DirectedGraph DirectedGraphQ DirectedInfinity Direction Directive Directory DirectoryName DirectoryQ DirectoryStack DirichletBeta DirichletCharacter DirichletCondition DirichletConvolve DirichletDistribution DirichletEta DirichletL DirichletLambda DirichletTransform DirichletWindow DisableConsolePrintPacket DisableFormatting DiscreteChirpZTransform DiscreteConvolve DiscreteDelta DiscreteHadamardTransform DiscreteIndicator DiscreteLimit DiscreteLQEstimatorGains DiscreteLQRegulatorGains DiscreteLyapunovSolve DiscreteMarkovProcess DiscreteMaxLimit DiscreteMinLimit DiscretePlot DiscretePlot3D DiscreteRatio DiscreteRiccatiSolve DiscreteShift DiscreteTimeModelQ DiscreteUniformDistribution DiscreteVariables DiscreteWaveletData DiscreteWaveletPacketTransform DiscreteWaveletTransform DiscretizeGraphics DiscretizeRegion Discriminant DisjointQ Disjunction Disk DiskBox DiskMatrix DiskSegment Dispatch DispatchQ DispersionEstimatorFunction Display DisplayAllSteps DisplayEndPacket DisplayFlushImagePacket DisplayForm DisplayFunction DisplayPacket DisplayRules DisplaySetSizePacket DisplayString DisplayTemporary DisplayWith DisplayWithRef DisplayWithVariable DistanceFunction DistanceMatrix DistanceTransform Distribute Distributed DistributedContexts DistributeDefinitions DistributionChart DistributionDomain DistributionFitTest DistributionParameterAssumptions DistributionParameterQ Dithering Div Divergence Divide DivideBy Dividers DivideSides Divisible Divisors DivisorSigma DivisorSum DMSList DMSString Do DockedCells DocumentGenerator DocumentGeneratorInformation DocumentGeneratorInformationData DocumentGenerators DocumentNotebook DocumentWeightingRules Dodecahedron DomainRegistrationInformation DominantColors DOSTextFormat Dot DotDashed DotEqual DotLayer DotPlusLayer Dotted DoubleBracketingBar DoubleContourIntegral DoubleDownArrow DoubleLeftArrow DoubleLeftRightArrow DoubleLeftTee DoubleLongLeftArrow DoubleLongLeftRightArrow DoubleLongRightArrow DoubleRightArrow DoubleRightTee DoubleUpArrow DoubleUpDownArrow DoubleVerticalBar DoublyInfinite Down DownArrow DownArrowBar DownArrowUpArrow DownLeftRightVector DownLeftTeeVector DownLeftVector DownLeftVectorBar DownRightTeeVector DownRightVector DownRightVectorBar Downsample DownTee DownTeeArrow DownValues DragAndDrop DrawEdges DrawFrontFaces DrawHighlighted Drop DropoutLayer DSolve DSolveValue Dt DualLinearProgramming DualPolyhedron DualSystemsModel DumpGet DumpSave DuplicateFreeQ Duration Dynamic DynamicBox DynamicBoxOptions DynamicEvaluationTimeout DynamicGeoGraphics DynamicImage DynamicLocation DynamicModule DynamicModuleBox DynamicModuleBoxOptions DynamicModuleParent DynamicModuleValues DynamicName DynamicNamespace DynamicReference DynamicSetting DynamicUpdating DynamicWrapper DynamicWrapperBox DynamicWrapperBoxOptions"+"E EarthImpactData EarthquakeData EccentricityCentrality Echo EchoFunction EclipseType EdgeAdd EdgeBetweennessCentrality EdgeCapacity EdgeCapForm EdgeColor EdgeConnectivity EdgeContract EdgeCost EdgeCount EdgeCoverQ EdgeCycleMatrix EdgeDashing EdgeDelete EdgeDetect EdgeForm EdgeIndex EdgeJoinForm EdgeLabeling EdgeLabels EdgeLabelStyle EdgeList EdgeOpacity EdgeQ EdgeRenderingFunction EdgeRules EdgeShapeFunction EdgeStyle EdgeThickness EdgeWeight EdgeWeightedGraphQ Editable EditButtonSettings EditCellTagsSettings EditDistance EffectiveInterest Eigensystem Eigenvalues EigenvectorCentrality Eigenvectors Element ElementData ElementwiseLayer ElidedForms Eliminate EliminationOrder Ellipsoid EllipticE EllipticExp EllipticExpPrime EllipticF EllipticFilterModel EllipticK EllipticLog EllipticNomeQ EllipticPi EllipticReducedHalfPeriods EllipticTheta EllipticThetaPrime EmbedCode EmbeddedHTML EmbeddedService EmbeddingLayer EmbeddingObject EmitSound EmphasizeSyntaxErrors EmpiricalDistribution Empty EmptyGraphQ EmptyRegion EnableConsolePrintPacket Enabled Encode Encrypt EncryptedObject EncryptFile End EndAdd EndDialogPacket EndFrontEndInteractionPacket EndOfBuffer EndOfFile EndOfLine EndOfString EndPackage EngineEnvironment EngineeringForm Enter EnterExpressionPacket EnterTextPacket Entity EntityClass EntityClassList EntityCopies EntityFunction EntityGroup EntityInstance EntityList EntityPrefetch EntityProperties EntityProperty EntityPropertyClass EntityRegister EntityStore EntityStores EntityTypeName EntityUnregister EntityValue Entropy EntropyFilter Environment Epilog EpilogFunction Equal EqualColumns EqualRows EqualTilde EqualTo EquatedTo Equilibrium EquirippleFilterKernel Equivalent Erf Erfc Erfi ErlangB ErlangC ErlangDistribution Erosion ErrorBox ErrorBoxOptions ErrorNorm ErrorPacket ErrorsDialogSettings EscapeRadius EstimatedBackground EstimatedDistribution EstimatedProcess EstimatorGains EstimatorRegulator EuclideanDistance EulerAngles EulerCharacteristic EulerE EulerGamma EulerianGraphQ EulerMatrix EulerPhi Evaluatable Evaluate Evaluated EvaluatePacket EvaluateScheduledTask EvaluationBox EvaluationCell EvaluationCompletionAction EvaluationData EvaluationElements EvaluationEnvironment EvaluationMode EvaluationMonitor EvaluationNotebook EvaluationObject EvaluationOrder Evaluator EvaluatorNames EvenQ EventData EventEvaluator EventHandler EventHandlerTag EventLabels EventSeries ExactBlackmanWindow ExactNumberQ ExactRootIsolation ExampleData Except ExcludedForms ExcludedLines ExcludedPhysicalQuantities ExcludePods Exclusions ExclusionsStyle Exists Exit ExitDialog ExoplanetData Exp Expand ExpandAll ExpandDenominator ExpandFileName ExpandNumerator Expectation ExpectationE ExpectedValue ExpGammaDistribution ExpIntegralE ExpIntegralEi ExpirationDate Exponent ExponentFunction ExponentialDistribution ExponentialFamily ExponentialGeneratingFunction ExponentialMovingAverage ExponentialPowerDistribution ExponentPosition ExponentStep Export ExportAutoReplacements ExportByteArray ExportForm ExportPacket ExportString Expression ExpressionCell ExpressionPacket ExpressionUUID ExpToTrig ExtendedEntityClass ExtendedGCD Extension ExtentElementFunction ExtentMarkers ExtentSize ExternalBundle ExternalCall ExternalDataCharacterEncoding ExternalEvaluate ExternalFunction ExternalFunctionName ExternalObject ExternalOptions ExternalSessionObject ExternalSessions ExternalTypeSignature ExternalValue Extract ExtractArchive ExtractLayer ExtremeValueDistribution"+"FaceForm FaceGrids FaceGridsStyle FacialFeatures Factor FactorComplete Factorial Factorial2 FactorialMoment FactorialMomentGeneratingFunction FactorialPower FactorInteger FactorList FactorSquareFree FactorSquareFreeList FactorTerms FactorTermsList Fail Failure FailureAction FailureDistribution FailureQ False FareySequence FARIMAProcess FeatureDistance FeatureExtract FeatureExtraction FeatureExtractor FeatureExtractorFunction FeatureNames FeatureNearest FeatureSpacePlot FeatureSpacePlot3D FeatureTypes FEDisableConsolePrintPacket FeedbackLinearize FeedbackSector FeedbackSectorStyle FeedbackType FEEnableConsolePrintPacket FetalGrowthData Fibonacci Fibonorial FieldCompletionFunction FieldHint FieldHintStyle FieldMasked FieldSize File FileBaseName FileByteCount FileConvert FileDate FileExistsQ FileExtension FileFormat FileHandler FileHash FileInformation FileName FileNameDepth FileNameDialogSettings FileNameDrop FileNameForms FileNameJoin FileNames FileNameSetter FileNameSplit FileNameTake FilePrint FileSize FileSystemMap FileSystemScan FileTemplate FileTemplateApply FileType FilledCurve FilledCurveBox FilledCurveBoxOptions Filling FillingStyle FillingTransform FilteredEntityClass FilterRules FinancialBond FinancialData FinancialDerivative FinancialIndicator Find FindAnomalies FindArgMax FindArgMin FindChannels FindClique FindClusters FindCookies FindCurvePath FindCycle FindDevices FindDistribution FindDistributionParameters FindDivisions FindEdgeCover FindEdgeCut FindEdgeIndependentPaths FindEquationalProof FindEulerianCycle FindExternalEvaluators FindFaces FindFile FindFit FindFormula FindFundamentalCycles FindGeneratingFunction FindGeoLocation FindGeometricConjectures FindGeometricTransform FindGraphCommunities FindGraphIsomorphism FindGraphPartition FindHamiltonianCycle FindHamiltonianPath FindHiddenMarkovStates FindIndependentEdgeSet FindIndependentVertexSet FindInstance FindIntegerNullVector FindKClan FindKClique FindKClub FindKPlex FindLibrary FindLinearRecurrence FindList FindMatchingColor FindMaximum FindMaximumFlow FindMaxValue FindMeshDefects FindMinimum FindMinimumCostFlow FindMinimumCut FindMinValue FindMoleculeSubstructure FindPath FindPeaks FindPermutation FindPostmanTour FindProcessParameters FindRepeat FindRoot FindSequenceFunction FindSettings FindShortestPath FindShortestTour FindSpanningTree FindSystemModelEquilibrium FindTextualAnswer FindThreshold FindTransientRepeat FindVertexCover FindVertexCut FindVertexIndependentPaths Fine FinishDynamic FiniteAbelianGroupCount FiniteGroupCount FiniteGroupData First FirstCase FirstPassageTimeDistribution FirstPosition FischerGroupFi22 FischerGroupFi23 FischerGroupFi24Prime FisherHypergeometricDistribution FisherRatioTest FisherZDistribution Fit FitAll FitRegularization FittedModel FixedOrder FixedPoint FixedPointList FlashSelection Flat Flatten FlattenAt FlattenLayer FlatTopWindow FlipView Floor FlowPolynomial FlushPrintOutputPacket Fold FoldList FoldPair FoldPairList FollowRedirects Font FontColor FontFamily FontForm FontName FontOpacity FontPostScriptName FontProperties FontReencoding FontSize FontSlant FontSubstitutions FontTracking FontVariations FontWeight For ForAll Format FormatRules FormatType FormatTypeAutoConvert FormatValues FormBox FormBoxOptions FormControl FormFunction FormLayoutFunction FormObject FormPage FormTheme FormulaData FormulaLookup FortranForm Forward ForwardBackward Fourier FourierCoefficient FourierCosCoefficient FourierCosSeries FourierCosTransform FourierDCT FourierDCTFilter FourierDCTMatrix FourierDST FourierDSTMatrix FourierMatrix FourierParameters FourierSequenceTransform FourierSeries FourierSinCoefficient FourierSinSeries FourierSinTransform FourierTransform FourierTrigSeries FractionalBrownianMotionProcess FractionalGaussianNoiseProcess FractionalPart FractionBox FractionBoxOptions FractionLine Frame FrameBox FrameBoxOptions Framed FrameInset FrameLabel Frameless FrameMargins FrameRate FrameStyle FrameTicks FrameTicksStyle FRatioDistribution FrechetDistribution FreeQ FrenetSerretSystem FrequencySamplingFilterKernel FresnelC FresnelF FresnelG FresnelS Friday FrobeniusNumber FrobeniusSolve FromAbsoluteTime FromCharacterCode FromCoefficientRules FromContinuedFraction FromDate FromDigits FromDMS FromEntity FromJulianDate FromLetterNumber FromPolarCoordinates FromRomanNumeral FromSphericalCoordinates FromUnixTime Front FrontEndDynamicExpression FrontEndEventActions FrontEndExecute FrontEndObject FrontEndResource FrontEndResourceString FrontEndStackSize FrontEndToken FrontEndTokenExecute FrontEndValueCache FrontEndVersion FrontFaceColor FrontFaceOpacity Full FullAxes FullDefinition FullForm FullGraphics FullInformationOutputRegulator FullOptions FullRegion FullSimplify Function FunctionCompile FunctionCompileExport FunctionCompileExportByteArray FunctionCompileExportLibrary FunctionCompileExportString FunctionDomain FunctionExpand FunctionInterpolation FunctionPeriod FunctionRange FunctionSpace FussellVeselyImportance"+"GaborFilter GaborMatrix GaborWavelet GainMargins GainPhaseMargins GalaxyData GalleryView Gamma GammaDistribution GammaRegularized GapPenalty GARCHProcess GatedRecurrentLayer Gather GatherBy GaugeFaceElementFunction GaugeFaceStyle GaugeFrameElementFunction GaugeFrameSize GaugeFrameStyle GaugeLabels GaugeMarkers GaugeStyle GaussianFilter GaussianIntegers GaussianMatrix GaussianOrthogonalMatrixDistribution GaussianSymplecticMatrixDistribution GaussianUnitaryMatrixDistribution GaussianWindow GCD GegenbauerC General GeneralizedLinearModelFit GenerateAsymmetricKeyPair GenerateConditions GeneratedCell GeneratedDocumentBinding GenerateDerivedKey GenerateDigitalSignature GenerateDocument GeneratedParameters GeneratedQuantityMagnitudes GenerateHTTPResponse GenerateSecuredAuthenticationKey GenerateSymmetricKey GeneratingFunction GeneratorDescription GeneratorHistoryLength GeneratorOutputType Generic GenericCylindricalDecomposition GenomeData GenomeLookup GeoAntipode GeoArea GeoArraySize GeoBackground GeoBoundingBox GeoBounds GeoBoundsRegion GeoBubbleChart GeoCenter GeoCircle GeodesicClosing GeodesicDilation GeodesicErosion GeodesicOpening GeoDestination GeodesyData GeoDirection GeoDisk GeoDisplacement GeoDistance GeoDistanceList GeoElevationData GeoEntities GeoGraphics GeogravityModelData GeoGridDirectionDifference GeoGridLines GeoGridLinesStyle GeoGridPosition GeoGridRange GeoGridRangePadding GeoGridUnitArea GeoGridUnitDistance GeoGridVector GeoGroup GeoHemisphere GeoHemisphereBoundary GeoHistogram GeoIdentify GeoImage GeoLabels GeoLength GeoListPlot GeoLocation GeologicalPeriodData GeomagneticModelData GeoMarker GeometricAssertion GeometricBrownianMotionProcess GeometricDistribution GeometricMean GeometricMeanFilter GeometricScene GeometricTransformation GeometricTransformation3DBox GeometricTransformation3DBoxOptions GeometricTransformationBox GeometricTransformationBoxOptions GeoModel GeoNearest GeoPath GeoPosition GeoPositionENU GeoPositionXYZ GeoProjection GeoProjectionData GeoRange GeoRangePadding GeoRegionValuePlot GeoResolution GeoScaleBar GeoServer GeoSmoothHistogram GeoStreamPlot GeoStyling GeoStylingImageFunction GeoVariant GeoVector GeoVectorENU GeoVectorPlot GeoVectorXYZ GeoVisibleRegion GeoVisibleRegionBoundary GeoWithinQ GeoZoomLevel GestureHandler GestureHandlerTag Get GetBoundingBoxSizePacket GetContext GetEnvironment GetFileName GetFrontEndOptionsDataPacket GetLinebreakInformationPacket GetMenusPacket GetPageBreakInformationPacket Glaisher GlobalClusteringCoefficient GlobalPreferences GlobalSession Glow GoldenAngle GoldenRatio GompertzMakehamDistribution GoodmanKruskalGamma GoodmanKruskalGammaTest Goto Grad Gradient GradientFilter GradientOrientationFilter GrammarApply GrammarRules GrammarToken Graph Graph3D GraphAssortativity GraphAutomorphismGroup GraphCenter GraphComplement GraphData GraphDensity GraphDiameter GraphDifference GraphDisjointUnion GraphDistance GraphDistanceMatrix GraphElementData GraphEmbedding GraphHighlight GraphHighlightStyle GraphHub Graphics Graphics3D Graphics3DBox Graphics3DBoxOptions GraphicsArray GraphicsBaseline GraphicsBox GraphicsBoxOptions GraphicsColor GraphicsColumn GraphicsComplex GraphicsComplex3DBox GraphicsComplex3DBoxOptions GraphicsComplexBox GraphicsComplexBoxOptions GraphicsContents GraphicsData GraphicsGrid GraphicsGridBox GraphicsGroup GraphicsGroup3DBox GraphicsGroup3DBoxOptions GraphicsGroupBox GraphicsGroupBoxOptions GraphicsGrouping GraphicsHighlightColor GraphicsRow GraphicsSpacing GraphicsStyle GraphIntersection GraphLayout GraphLinkEfficiency GraphPeriphery GraphPlot GraphPlot3D GraphPower GraphPropertyDistribution GraphQ GraphRadius GraphReciprocity GraphRoot GraphStyle GraphUnion Gray GrayLevel Greater GreaterEqual GreaterEqualLess GreaterEqualThan GreaterFullEqual GreaterGreater GreaterLess GreaterSlantEqual GreaterThan GreaterTilde Green GreenFunction Grid GridBaseline GridBox GridBoxAlignment GridBoxBackground GridBoxDividers GridBoxFrame GridBoxItemSize GridBoxItemStyle GridBoxOptions GridBoxSpacings GridCreationSettings GridDefaultElement GridElementStyleOptions GridFrame GridFrameMargins GridGraph GridLines GridLinesStyle GroebnerBasis GroupActionBase GroupBy GroupCentralizer GroupElementFromWord GroupElementPosition GroupElementQ GroupElements GroupElementToWord GroupGenerators Groupings GroupMultiplicationTable GroupOrbits GroupOrder GroupPageBreakWithin GroupSetwiseStabilizer GroupStabilizer GroupStabilizerChain GroupTogetherGrouping GroupTogetherNestedGrouping GrowCutComponents Gudermannian GuidedFilter GumbelDistribution"+"HaarWavelet HadamardMatrix HalfLine HalfNormalDistribution HalfPlane HalfSpace HamiltonianGraphQ HammingDistance HammingWindow HandlerFunctions HandlerFunctionsKeys HankelH1 HankelH2 HankelMatrix HankelTransform HannPoissonWindow HannWindow HaradaNortonGroupHN HararyGraph HarmonicMean HarmonicMeanFilter HarmonicNumber Hash Haversine HazardFunction Head HeadCompose HeaderLines Heads HeavisideLambda HeavisidePi HeavisideTheta HeldGroupHe HeldPart HelpBrowserLookup HelpBrowserNotebook HelpBrowserSettings Here HermiteDecomposition HermiteH HermitianMatrixQ HessenbergDecomposition Hessian HexadecimalCharacter Hexahedron HexahedronBox HexahedronBoxOptions HiddenMarkovProcess HiddenSurface Highlighted HighlightGraph HighlightImage HighlightMesh HighpassFilter HigmanSimsGroupHS HilbertCurve HilbertFilter HilbertMatrix Histogram Histogram3D HistogramDistribution HistogramList HistogramTransform HistogramTransformInterpolation HistoricalPeriodData HitMissTransform HITSCentrality HjorthDistribution HodgeDual HoeffdingD HoeffdingDTest Hold HoldAll HoldAllComplete HoldComplete HoldFirst HoldForm HoldPattern HoldRest HolidayCalendar HomeDirectory HomePage Horizontal HorizontalForm HorizontalGauge HorizontalScrollPosition HornerForm HostLookup HotellingTSquareDistribution HoytDistribution HTMLSave HTTPErrorResponse HTTPRedirect HTTPRequest HTTPRequestData HTTPResponse Hue HumanGrowthData HumpDownHump HumpEqual HurwitzLerchPhi HurwitzZeta HyperbolicDistribution HypercubeGraph HyperexponentialDistribution Hyperfactorial Hypergeometric0F1 Hypergeometric0F1Regularized Hypergeometric1F1 Hypergeometric1F1Regularized Hypergeometric2F1 Hypergeometric2F1Regularized HypergeometricDistribution HypergeometricPFQ HypergeometricPFQRegularized HypergeometricU Hyperlink HyperlinkCreationSettings Hyperplane Hyphenation HyphenationOptions HypoexponentialDistribution HypothesisTestData"+"I IconData Iconize IconizedObject IconRules Icosahedron Identity IdentityMatrix If IgnoreCase IgnoreDiacritics IgnorePunctuation IgnoreSpellCheck IgnoringInactive Im Image Image3D Image3DProjection Image3DSlices ImageAccumulate ImageAdd ImageAdjust ImageAlign ImageApply ImageApplyIndexed ImageAspectRatio ImageAssemble ImageAugmentationLayer ImageBoundingBoxes ImageCache ImageCacheValid ImageCapture ImageCaptureFunction ImageCases ImageChannels ImageClip ImageCollage ImageColorSpace ImageCompose ImageContainsQ ImageContents ImageConvolve ImageCooccurrence ImageCorners ImageCorrelate ImageCorrespondingPoints ImageCrop ImageData ImageDeconvolve ImageDemosaic ImageDifference ImageDimensions ImageDisplacements ImageDistance ImageEffect ImageExposureCombine ImageFeatureTrack ImageFileApply ImageFileFilter ImageFileScan ImageFilter ImageFocusCombine ImageForestingComponents ImageFormattingWidth ImageForwardTransformation ImageGraphics ImageHistogram ImageIdentify ImageInstanceQ ImageKeypoints ImageLevels ImageLines ImageMargins ImageMarker ImageMarkers ImageMeasurements ImageMesh ImageMultiply ImageOffset ImagePad ImagePadding ImagePartition ImagePeriodogram ImagePerspectiveTransformation ImagePosition ImagePreviewFunction ImagePyramid ImagePyramidApply ImageQ ImageRangeCache ImageRecolor ImageReflect ImageRegion ImageResize ImageResolution ImageRestyle ImageRotate ImageRotated ImageSaliencyFilter ImageScaled ImageScan ImageSize ImageSizeAction ImageSizeCache ImageSizeMultipliers ImageSizeRaw ImageSubtract ImageTake ImageTransformation ImageTrim ImageType ImageValue ImageValuePositions ImagingDevice ImplicitRegion Implies Import ImportAutoReplacements ImportByteArray ImportOptions ImportString ImprovementImportance In Inactivate Inactive IncidenceGraph IncidenceList IncidenceMatrix IncludeAromaticBonds IncludeConstantBasis IncludeDefinitions IncludeDirectories IncludeFileExtension IncludeGeneratorTasks IncludeHydrogens IncludeInflections IncludeMetaInformation IncludePods IncludeQuantities IncludeRelatedTables IncludeSingularTerm IncludeWindowTimes Increment IndefiniteMatrixQ Indent IndentingNewlineSpacings IndentMaxFraction IndependenceTest IndependentEdgeSetQ IndependentPhysicalQuantity IndependentUnit IndependentUnitDimension IndependentVertexSetQ Indeterminate IndeterminateThreshold IndexCreationOptions Indexed IndexGraph IndexTag Inequality InexactNumberQ InexactNumbers InfiniteLine InfinitePlane Infinity Infix InflationAdjust InflationMethod Information InformationData InformationDataGrid Inherited InheritScope InhomogeneousPoissonProcess InitialEvaluationHistory Initialization InitializationCell InitializationCellEvaluation InitializationCellWarning InitializationObjects InitializationValue Initialize InitialSeeding InlineCounterAssignments InlineCounterIncrements InlineRules Inner InnerPolygon InnerPolyhedron Inpaint Input InputAliases InputAssumptions InputAutoReplacements InputField InputFieldBox InputFieldBoxOptions InputForm InputGrouping InputNamePacket InputNotebook InputPacket InputSettings InputStream InputString InputStringPacket InputToBoxFormPacket Insert InsertionFunction InsertionPointObject InsertLinebreaks InsertResults Inset Inset3DBox Inset3DBoxOptions InsetBox InsetBoxOptions Insphere Install InstallService InstanceNormalizationLayer InString Integer IntegerDigits IntegerExponent IntegerLength IntegerName IntegerPart IntegerPartitions IntegerQ IntegerReverse Integers IntegerString Integral Integrate Interactive InteractiveTradingChart Interlaced Interleaving InternallyBalancedDecomposition InterpolatingFunction InterpolatingPolynomial Interpolation InterpolationOrder InterpolationPoints InterpolationPrecision Interpretation InterpretationBox InterpretationBoxOptions InterpretationFunction Interpreter InterpretTemplate InterquartileRange Interrupt InterruptSettings IntersectingQ Intersection Interval IntervalIntersection IntervalMarkers IntervalMarkersStyle IntervalMemberQ IntervalSlider IntervalUnion Into Inverse InverseBetaRegularized InverseCDF InverseChiSquareDistribution InverseContinuousWaveletTransform InverseDistanceTransform InverseEllipticNomeQ InverseErf InverseErfc InverseFourier InverseFourierCosTransform InverseFourierSequenceTransform InverseFourierSinTransform InverseFourierTransform InverseFunction InverseFunctions InverseGammaDistribution InverseGammaRegularized InverseGaussianDistribution InverseGudermannian InverseHankelTransform InverseHaversine InverseImagePyramid InverseJacobiCD InverseJacobiCN InverseJacobiCS InverseJacobiDC InverseJacobiDN InverseJacobiDS InverseJacobiNC InverseJacobiND InverseJacobiNS InverseJacobiSC InverseJacobiSD InverseJacobiSN InverseLaplaceTransform InverseMellinTransform InversePermutation InverseRadon InverseRadonTransform InverseSeries InverseShortTimeFourier InverseSpectrogram InverseSurvivalFunction InverseTransformedRegion InverseWaveletTransform InverseWeierstrassP InverseWishartMatrixDistribution InverseZTransform Invisible InvisibleApplication InvisibleTimes IPAddress IrreduciblePolynomialQ IslandData IsolatingInterval IsomorphicGraphQ IsotopeData Italic Item ItemAspectRatio ItemBox ItemBoxOptions ItemSize ItemStyle ItoProcess"+"JaccardDissimilarity JacobiAmplitude Jacobian JacobiCD JacobiCN JacobiCS JacobiDC JacobiDN JacobiDS JacobiNC JacobiND JacobiNS JacobiP JacobiSC JacobiSD JacobiSN JacobiSymbol JacobiZeta JankoGroupJ1 JankoGroupJ2 JankoGroupJ3 JankoGroupJ4 JarqueBeraALMTest JohnsonDistribution Join JoinAcross Joined JoinedCurve JoinedCurveBox JoinedCurveBoxOptions JoinForm JordanDecomposition JordanModelDecomposition JulianDate JuliaSetBoettcher JuliaSetIterationCount JuliaSetPlot JuliaSetPoints"+"K KagiChart KaiserBesselWindow KaiserWindow KalmanEstimator KalmanFilter KarhunenLoeveDecomposition KaryTree KatzCentrality KCoreComponents KDistribution KEdgeConnectedComponents KEdgeConnectedGraphQ KelvinBei KelvinBer KelvinKei KelvinKer KendallTau KendallTauTest KernelExecute KernelFunction KernelMixtureDistribution Kernels Ket Key KeyCollisionFunction KeyComplement KeyDrop KeyDropFrom KeyExistsQ KeyFreeQ KeyIntersection KeyMap KeyMemberQ KeypointStrength Keys KeySelect KeySort KeySortBy KeyTake KeyUnion KeyValueMap KeyValuePattern Khinchin KillProcess KirchhoffGraph KirchhoffMatrix KleinInvariantJ KnapsackSolve KnightTourGraph KnotData KnownUnitQ KochCurve KolmogorovSmirnovTest KroneckerDelta KroneckerModelDecomposition KroneckerProduct KroneckerSymbol KuiperTest KumaraswamyDistribution Kurtosis KuwaharaFilter KVertexConnectedComponents KVertexConnectedGraphQ"+"LABColor Label Labeled LabeledSlider LabelingFunction LabelingSize LabelStyle LabelVisibility LaguerreL LakeData LambdaComponents LambertW LaminaData LanczosWindow LandauDistribution Language LanguageCategory LanguageData LanguageIdentify LanguageOptions LaplaceDistribution LaplaceTransform Laplacian LaplacianFilter LaplacianGaussianFilter Large Larger Last Latitude LatitudeLongitude LatticeData LatticeReduce Launch LaunchKernels LayeredGraphPlot LayerSizeFunction LayoutInformation LCHColor LCM LeaderSize LeafCount LeapYearQ LearnDistribution LearnedDistribution LearningRate LearningRateMultipliers LeastSquares LeastSquaresFilterKernel Left LeftArrow LeftArrowBar LeftArrowRightArrow LeftDownTeeVector LeftDownVector LeftDownVectorBar LeftRightArrow LeftRightVector LeftTee LeftTeeArrow LeftTeeVector LeftTriangle LeftTriangleBar LeftTriangleEqual LeftUpDownVector LeftUpTeeVector LeftUpVector LeftUpVectorBar LeftVector LeftVectorBar LegendAppearance Legended LegendFunction LegendLabel LegendLayout LegendMargins LegendMarkers LegendMarkerSize LegendreP LegendreQ LegendreType Length LengthWhile LerchPhi Less LessEqual LessEqualGreater LessEqualThan LessFullEqual LessGreater LessLess LessSlantEqual LessThan LessTilde LetterCharacter LetterCounts LetterNumber LetterQ Level LeveneTest LeviCivitaTensor LevyDistribution Lexicographic LibraryDataType LibraryFunction LibraryFunctionError LibraryFunctionInformation LibraryFunctionLoad LibraryFunctionUnload LibraryLoad LibraryUnload LicenseID LiftingFilterData LiftingWaveletTransform LightBlue LightBrown LightCyan Lighter LightGray LightGreen Lighting LightingAngle LightMagenta LightOrange LightPink LightPurple LightRed LightSources LightYellow Likelihood Limit LimitsPositioning LimitsPositioningTokens LindleyDistribution Line Line3DBox Line3DBoxOptions LinearFilter LinearFractionalOptimization LinearFractionalTransform LinearGradientImage LinearizingTransformationData LinearLayer LinearModelFit LinearOffsetFunction LinearOptimization LinearProgramming LinearRecurrence LinearSolve LinearSolveFunction LineBox LineBoxOptions LineBreak LinebreakAdjustments LineBreakChart LinebreakSemicolonWeighting LineBreakWithin LineColor LineGraph LineIndent LineIndentMaxFraction LineIntegralConvolutionPlot LineIntegralConvolutionScale LineLegend LineOpacity LineSpacing LineWrapParts LinkActivate LinkClose LinkConnect LinkConnectedQ LinkCreate LinkError LinkFlush LinkFunction LinkHost LinkInterrupt LinkLaunch LinkMode LinkObject LinkOpen LinkOptions LinkPatterns LinkProtocol LinkRankCentrality LinkRead LinkReadHeld LinkReadyQ Links LinkService LinkWrite LinkWriteHeld LiouvilleLambda List Listable ListAnimate ListContourPlot ListContourPlot3D ListConvolve ListCorrelate ListCurvePathPlot ListDeconvolve ListDensityPlot ListDensityPlot3D Listen ListFormat ListFourierSequenceTransform ListInterpolation ListLineIntegralConvolutionPlot ListLinePlot ListLogLinearPlot ListLogLogPlot ListLogPlot ListPicker ListPickerBox ListPickerBoxBackground ListPickerBoxOptions ListPlay ListPlot ListPlot3D ListPointPlot3D ListPolarPlot ListQ ListSliceContourPlot3D ListSliceDensityPlot3D ListSliceVectorPlot3D ListStepPlot ListStreamDensityPlot ListStreamPlot ListSurfacePlot3D ListVectorDensityPlot ListVectorPlot ListVectorPlot3D ListZTransform Literal LiteralSearch LocalAdaptiveBinarize LocalCache LocalClusteringCoefficient LocalizeDefinitions LocalizeVariables LocalObject LocalObjects LocalResponseNormalizationLayer LocalSubmit LocalSymbol LocalTime LocalTimeZone LocationEquivalenceTest LocationTest Locator LocatorAutoCreate LocatorBox LocatorBoxOptions LocatorCentering LocatorPane LocatorPaneBox LocatorPaneBoxOptions LocatorRegion Locked Log Log10 Log2 LogBarnesG LogGamma LogGammaDistribution LogicalExpand LogIntegral LogisticDistribution LogisticSigmoid LogitModelFit LogLikelihood LogLinearPlot LogLogisticDistribution LogLogPlot LogMultinormalDistribution LogNormalDistribution LogPlot LogRankTest LogSeriesDistribution LongEqual Longest LongestCommonSequence LongestCommonSequencePositions LongestCommonSubsequence LongestCommonSubsequencePositions LongestMatch LongestOrderedSequence LongForm Longitude LongLeftArrow LongLeftRightArrow LongRightArrow LongShortTermMemoryLayer Lookup Loopback LoopFreeGraphQ LossFunction LowerCaseQ LowerLeftArrow LowerRightArrow LowerTriangularize LowerTriangularMatrixQ LowpassFilter LQEstimatorGains LQGRegulator LQOutputRegulatorGains LQRegulatorGains LUBackSubstitution LucasL LuccioSamiComponents LUDecomposition LunarEclipse LUVColor LyapunovSolve LyonsGroupLy"+"MachineID MachineName MachineNumberQ MachinePrecision MacintoshSystemPageSetup Magenta Magnification Magnify MailAddressValidation MailExecute MailFolder MailItem MailReceiverFunction MailResponseFunction MailSearch MailServerConnect MailServerConnection MailSettings MainSolve MaintainDynamicCaches Majority MakeBoxes MakeExpression MakeRules ManagedLibraryExpressionID ManagedLibraryExpressionQ MandelbrotSetBoettcher MandelbrotSetDistance MandelbrotSetIterationCount MandelbrotSetMemberQ MandelbrotSetPlot MangoldtLambda ManhattanDistance Manipulate Manipulator MannedSpaceMissionData MannWhitneyTest MantissaExponent Manual Map MapAll MapAt MapIndexed MAProcess MapThread MarchenkoPasturDistribution MarcumQ MardiaCombinedTest MardiaKurtosisTest MardiaSkewnessTest MarginalDistribution MarkovProcessProperties Masking MatchingDissimilarity MatchLocalNameQ MatchLocalNames MatchQ Material MathematicalFunctionData MathematicaNotation MathieuC MathieuCharacteristicA MathieuCharacteristicB MathieuCharacteristicExponent MathieuCPrime MathieuGroupM11 MathieuGroupM12 MathieuGroupM22 MathieuGroupM23 MathieuGroupM24 MathieuS MathieuSPrime MathMLForm MathMLText Matrices MatrixExp MatrixForm MatrixFunction MatrixLog MatrixNormalDistribution MatrixPlot MatrixPower MatrixPropertyDistribution MatrixQ MatrixRank MatrixTDistribution Max MaxBend MaxCellMeasure MaxColorDistance MaxDetect MaxDuration MaxExtraBandwidths MaxExtraConditions MaxFeatureDisplacement MaxFeatures MaxFilter MaximalBy Maximize MaxItems MaxIterations MaxLimit MaxMemoryUsed MaxMixtureKernels MaxOverlapFraction MaxPlotPoints MaxPoints MaxRecursion MaxStableDistribution MaxStepFraction MaxSteps MaxStepSize MaxTrainingRounds MaxValue MaxwellDistribution MaxWordGap McLaughlinGroupMcL Mean MeanAbsoluteLossLayer MeanAround MeanClusteringCoefficient MeanDegreeConnectivity MeanDeviation MeanFilter MeanGraphDistance MeanNeighborDegree MeanShift MeanShiftFilter MeanSquaredLossLayer Median MedianDeviation MedianFilter MedicalTestData Medium MeijerG MeijerGReduce MeixnerDistribution MellinConvolve MellinTransform MemberQ MemoryAvailable MemoryConstrained MemoryConstraint MemoryInUse MengerMesh Menu MenuAppearance MenuCommandKey MenuEvaluator MenuItem MenuList MenuPacket MenuSortingValue MenuStyle MenuView Merge MergeDifferences MergingFunction MersennePrimeExponent MersennePrimeExponentQ Mesh MeshCellCentroid MeshCellCount MeshCellHighlight MeshCellIndex MeshCellLabel MeshCellMarker MeshCellMeasure MeshCellQuality MeshCells MeshCellShapeFunction MeshCellStyle MeshCoordinates MeshFunctions MeshPrimitives MeshQualityGoal MeshRange MeshRefinementFunction MeshRegion MeshRegionQ MeshShading MeshStyle Message MessageDialog MessageList MessageName MessageObject MessageOptions MessagePacket Messages MessagesNotebook MetaCharacters MetaInformation MeteorShowerData Method MethodOptions MexicanHatWavelet MeyerWavelet Midpoint Min MinColorDistance MinDetect MineralData MinFilter MinimalBy MinimalPolynomial MinimalStateSpaceModel Minimize MinimumTimeIncrement MinIntervalSize MinkowskiQuestionMark MinLimit MinMax MinorPlanetData Minors MinRecursion MinSize MinStableDistribution Minus MinusPlus MinValue Missing MissingBehavior MissingDataMethod MissingDataRules MissingQ MissingString MissingStyle MissingValuePattern MittagLefflerE MixedFractionParts MixedGraphQ MixedMagnitude MixedRadix MixedRadixQuantity MixedUnit MixtureDistribution Mod Modal Mode Modular ModularInverse ModularLambda Module Modulus MoebiusMu Molecule MoleculeContainsQ MoleculeEquivalentQ MoleculeGraph MoleculeModify MoleculePattern MoleculePlot MoleculePlot3D MoleculeProperty MoleculeQ MoleculeValue Moment Momentary MomentConvert MomentEvaluate MomentGeneratingFunction MomentOfInertia Monday Monitor MonomialList MonomialOrder MonsterGroupM MoonPhase MoonPosition MorletWavelet MorphologicalBinarize MorphologicalBranchPoints MorphologicalComponents MorphologicalEulerNumber MorphologicalGraph MorphologicalPerimeter MorphologicalTransform MortalityData Most MountainData MouseAnnotation MouseAppearance MouseAppearanceTag MouseButtons Mouseover MousePointerNote MousePosition MovieData MovingAverage MovingMap MovingMedian MoyalDistribution Multicolumn MultiedgeStyle MultigraphQ MultilaunchWarning MultiLetterItalics MultiLetterStyle MultilineFunction Multinomial MultinomialDistribution MultinormalDistribution MultiplicativeOrder Multiplicity MultiplySides Multiselection MultivariateHypergeometricDistribution MultivariatePoissonDistribution MultivariateTDistribution"+"N NakagamiDistribution NameQ Names NamespaceBox NamespaceBoxOptions Nand NArgMax NArgMin NBernoulliB NBodySimulation NBodySimulationData NCache NDEigensystem NDEigenvalues NDSolve NDSolveValue Nearest NearestFunction NearestNeighborGraph NearestTo NebulaData NeedCurrentFrontEndPackagePacket NeedCurrentFrontEndSymbolsPacket NeedlemanWunschSimilarity Needs Negative NegativeBinomialDistribution NegativeDefiniteMatrixQ NegativeIntegers NegativeMultinomialDistribution NegativeRationals NegativeReals NegativeSemidefiniteMatrixQ NeighborhoodData NeighborhoodGraph Nest NestedGreaterGreater NestedLessLess NestedScriptRules NestGraph NestList NestWhile NestWhileList NetAppend NetBidirectionalOperator NetChain NetDecoder NetDelete NetDrop NetEncoder NetEvaluationMode NetExtract NetFlatten NetFoldOperator NetGraph NetInformation NetInitialize NetInsert NetInsertSharedArrays NetJoin NetMapOperator NetMapThreadOperator NetMeasurements NetModel NetNestOperator NetPairEmbeddingOperator NetPort NetPortGradient NetPrepend NetRename NetReplace NetReplacePart NetSharedArray NetStateObject NetTake NetTrain NetTrainResultsObject NetworkPacketCapture NetworkPacketRecording NetworkPacketRecordingDuring NetworkPacketTrace NeumannValue NevilleThetaC NevilleThetaD NevilleThetaN NevilleThetaS NewPrimitiveStyle NExpectation Next NextCell NextDate NextPrime NextScheduledTaskTime NHoldAll NHoldFirst NHoldRest NicholsGridLines NicholsPlot NightHemisphere NIntegrate NMaximize NMaxValue NMinimize NMinValue NominalVariables NonAssociative NoncentralBetaDistribution NoncentralChiSquareDistribution NoncentralFRatioDistribution NoncentralStudentTDistribution NonCommutativeMultiply NonConstants NondimensionalizationTransform None NoneTrue NonlinearModelFit NonlinearStateSpaceModel NonlocalMeansFilter NonNegative NonNegativeIntegers NonNegativeRationals NonNegativeReals NonPositive NonPositiveIntegers NonPositiveRationals NonPositiveReals Nor NorlundB Norm Normal NormalDistribution NormalGrouping NormalizationLayer Normalize Normalized NormalizedSquaredEuclideanDistance NormalMatrixQ NormalsFunction NormFunction Not NotCongruent NotCupCap NotDoubleVerticalBar Notebook NotebookApply NotebookAutoSave NotebookClose NotebookConvertSettings NotebookCreate NotebookCreateReturnObject NotebookDefault NotebookDelete NotebookDirectory NotebookDynamicExpression NotebookEvaluate NotebookEventActions NotebookFileName NotebookFind NotebookFindReturnObject NotebookGet NotebookGetLayoutInformationPacket NotebookGetMisspellingsPacket NotebookImport NotebookInformation NotebookInterfaceObject NotebookLocate NotebookObject NotebookOpen NotebookOpenReturnObject NotebookPath NotebookPrint NotebookPut NotebookPutReturnObject NotebookRead NotebookResetGeneratedCells Notebooks NotebookSave NotebookSaveAs NotebookSelection NotebookSetupLayoutInformationPacket NotebooksMenu NotebookTemplate NotebookWrite NotElement NotEqualTilde NotExists NotGreater NotGreaterEqual NotGreaterFullEqual NotGreaterGreater NotGreaterLess NotGreaterSlantEqual NotGreaterTilde Nothing NotHumpDownHump NotHumpEqual NotificationFunction NotLeftTriangle NotLeftTriangleBar NotLeftTriangleEqual NotLess NotLessEqual NotLessFullEqual NotLessGreater NotLessLess NotLessSlantEqual NotLessTilde NotNestedGreaterGreater NotNestedLessLess NotPrecedes NotPrecedesEqual NotPrecedesSlantEqual NotPrecedesTilde NotReverseElement NotRightTriangle NotRightTriangleBar NotRightTriangleEqual NotSquareSubset NotSquareSubsetEqual NotSquareSuperset NotSquareSupersetEqual NotSubset NotSubsetEqual NotSucceeds NotSucceedsEqual NotSucceedsSlantEqual NotSucceedsTilde NotSuperset NotSupersetEqual NotTilde NotTildeEqual NotTildeFullEqual NotTildeTilde NotVerticalBar Now NoWhitespace NProbability NProduct NProductFactors NRoots NSolve NSum NSumTerms NuclearExplosionData NuclearReactorData Null NullRecords NullSpace NullWords Number NumberCompose NumberDecompose NumberExpand NumberFieldClassNumber NumberFieldDiscriminant NumberFieldFundamentalUnits NumberFieldIntegralBasis NumberFieldNormRepresentatives NumberFieldRegulator NumberFieldRootsOfUnity NumberFieldSignature NumberForm NumberFormat NumberLinePlot NumberMarks NumberMultiplier NumberPadding NumberPoint NumberQ NumberSeparator NumberSigns NumberString Numerator NumeratorDenominator NumericalOrder NumericalSort NumericArray NumericArrayQ NumericArrayType NumericFunction NumericQ NuttallWindow NValues NyquistGridLines NyquistPlot"+"O ObservabilityGramian ObservabilityMatrix ObservableDecomposition ObservableModelQ OceanData Octahedron OddQ Off Offset OLEData On ONanGroupON Once OneIdentity Opacity OpacityFunction OpacityFunctionScaling Open OpenAppend Opener OpenerBox OpenerBoxOptions OpenerView OpenFunctionInspectorPacket Opening OpenRead OpenSpecialOptions OpenTemporary OpenWrite Operate OperatingSystem OptimumFlowData Optional OptionalElement OptionInspectorSettings OptionQ Options OptionsPacket OptionsPattern OptionValue OptionValueBox OptionValueBoxOptions Or Orange Order OrderDistribution OrderedQ Ordering OrderingBy OrderingLayer Orderless OrderlessPatternSequence OrnsteinUhlenbeckProcess Orthogonalize OrthogonalMatrixQ Out Outer OuterPolygon OuterPolyhedron OutputAutoOverwrite OutputControllabilityMatrix OutputControllableModelQ OutputForm OutputFormData OutputGrouping OutputMathEditExpression OutputNamePacket OutputResponse OutputSizeLimit OutputStream Over OverBar OverDot Overflow OverHat Overlaps Overlay OverlayBox OverlayBoxOptions Overscript OverscriptBox OverscriptBoxOptions OverTilde OverVector OverwriteTarget OwenT OwnValues"+"Package PackingMethod PaddedForm Padding PaddingLayer PaddingSize PadeApproximant PadLeft PadRight PageBreakAbove PageBreakBelow PageBreakWithin PageFooterLines PageFooters PageHeaderLines PageHeaders PageHeight PageRankCentrality PageTheme PageWidth Pagination PairedBarChart PairedHistogram PairedSmoothHistogram PairedTTest PairedZTest PaletteNotebook PalettePath PalindromeQ Pane PaneBox PaneBoxOptions Panel PanelBox PanelBoxOptions Paneled PaneSelector PaneSelectorBox PaneSelectorBoxOptions PaperWidth ParabolicCylinderD ParagraphIndent ParagraphSpacing ParallelArray ParallelCombine ParallelDo Parallelepiped ParallelEvaluate Parallelization Parallelize ParallelMap ParallelNeeds Parallelogram ParallelProduct ParallelSubmit ParallelSum ParallelTable ParallelTry Parameter ParameterEstimator ParameterMixtureDistribution ParameterVariables ParametricFunction ParametricNDSolve ParametricNDSolveValue ParametricPlot ParametricPlot3D ParametricRegion ParentBox ParentCell ParentConnect ParentDirectory ParentForm Parenthesize ParentList ParentNotebook ParetoDistribution ParetoPickandsDistribution ParkData Part PartBehavior PartialCorrelationFunction PartialD ParticleAcceleratorData ParticleData Partition PartitionGranularity PartitionsP PartitionsQ PartLayer PartOfSpeech PartProtection ParzenWindow PascalDistribution PassEventsDown PassEventsUp Paste PasteAutoQuoteCharacters PasteBoxFormInlineCells PasteButton Path PathGraph PathGraphQ Pattern PatternSequence PatternTest PauliMatrix PaulWavelet Pause PausedTime PDF PeakDetect PeanoCurve PearsonChiSquareTest PearsonCorrelationTest PearsonDistribution PercentForm PerfectNumber PerfectNumberQ PerformanceGoal Perimeter PeriodicBoundaryCondition PeriodicInterpolation Periodogram PeriodogramArray Permanent Permissions PermissionsGroup PermissionsGroupMemberQ PermissionsGroups PermissionsKey PermissionsKeys PermutationCycles PermutationCyclesQ PermutationGroup PermutationLength PermutationList PermutationListQ PermutationMax PermutationMin PermutationOrder PermutationPower PermutationProduct PermutationReplace Permutations PermutationSupport Permute PeronaMalikFilter Perpendicular PerpendicularBisector PersistenceLocation PersistenceTime PersistentObject PersistentObjects PersistentValue PersonData PERTDistribution PetersenGraph PhaseMargins PhaseRange PhysicalSystemData Pi Pick PIDData PIDDerivativeFilter PIDFeedforward PIDTune Piecewise PiecewiseExpand PieChart PieChart3D PillaiTrace PillaiTraceTest PingTime Pink PitchRecognize Pivoting PixelConstrained PixelValue PixelValuePositions Placed Placeholder PlaceholderReplace Plain PlanarAngle PlanarGraph PlanarGraphQ PlanckRadiationLaw PlaneCurveData PlanetaryMoonData PlanetData PlantData Play PlayRange Plot Plot3D Plot3Matrix PlotDivision PlotJoined PlotLabel PlotLabels PlotLayout PlotLegends PlotMarkers PlotPoints PlotRange PlotRangeClipping PlotRangeClipPlanesStyle PlotRangePadding PlotRegion PlotStyle PlotTheme Pluralize Plus PlusMinus Pochhammer PodStates PodWidth Point Point3DBox Point3DBoxOptions PointBox PointBoxOptions PointFigureChart PointLegend PointSize PoissonConsulDistribution PoissonDistribution PoissonProcess PoissonWindow PolarAxes PolarAxesOrigin PolarGridLines PolarPlot PolarTicks PoleZeroMarkers PolyaAeppliDistribution PolyGamma Polygon Polygon3DBox Polygon3DBoxOptions PolygonalNumber PolygonAngle PolygonBox PolygonBoxOptions PolygonCoordinates PolygonDecomposition PolygonHoleScale PolygonIntersections PolygonScale Polyhedron PolyhedronAngle PolyhedronCoordinates PolyhedronData PolyhedronDecomposition PolyhedronGenus PolyLog PolynomialExtendedGCD PolynomialForm PolynomialGCD PolynomialLCM PolynomialMod PolynomialQ PolynomialQuotient PolynomialQuotientRemainder PolynomialReduce PolynomialRemainder Polynomials PoolingLayer PopupMenu PopupMenuBox PopupMenuBoxOptions PopupView PopupWindow Position PositionIndex Positive PositiveDefiniteMatrixQ PositiveIntegers PositiveRationals PositiveReals PositiveSemidefiniteMatrixQ PossibleZeroQ Postfix PostScript Power PowerDistribution PowerExpand PowerMod PowerModList PowerRange PowerSpectralDensity PowersRepresentations PowerSymmetricPolynomial Precedence PrecedenceForm Precedes PrecedesEqual PrecedesSlantEqual PrecedesTilde Precision PrecisionGoal PreDecrement Predict PredictionRoot PredictorFunction PredictorInformation PredictorMeasurements PredictorMeasurementsObject PreemptProtect PreferencesPath Prefix PreIncrement Prepend PrependLayer PrependTo PreprocessingRules PreserveColor PreserveImageOptions Previous PreviousCell PreviousDate PriceGraphDistribution PrimaryPlaceholder Prime PrimeNu PrimeOmega PrimePi PrimePowerQ PrimeQ Primes PrimeZetaP PrimitivePolynomialQ PrimitiveRoot PrimitiveRootList PrincipalComponents PrincipalValue Print PrintableASCIIQ PrintAction PrintForm PrintingCopies PrintingOptions PrintingPageRange PrintingStartingPageNumber PrintingStyleEnvironment Printout3D Printout3DPreviewer PrintPrecision PrintTemporary Prism PrismBox PrismBoxOptions PrivateCellOptions PrivateEvaluationOptions PrivateFontOptions PrivateFrontEndOptions PrivateKey PrivateNotebookOptions PrivatePaths Probability ProbabilityDistribution ProbabilityPlot ProbabilityPr ProbabilityScalePlot ProbitModelFit ProcessConnection ProcessDirectory ProcessEnvironment Processes ProcessEstimator ProcessInformation ProcessObject ProcessParameterAssumptions ProcessParameterQ ProcessStateDomain ProcessStatus ProcessTimeDomain Product ProductDistribution ProductLog ProgressIndicator ProgressIndicatorBox ProgressIndicatorBoxOptions Projection Prolog PromptForm ProofObject Properties Property PropertyList PropertyValue Proportion Proportional Protect Protected ProteinData Pruning PseudoInverse PsychrometricPropertyData PublicKey PublisherID PulsarData PunctuationCharacter Purple Put PutAppend Pyramid PyramidBox PyramidBoxOptions"+"QBinomial QFactorial QGamma QHypergeometricPFQ QnDispersion QPochhammer QPolyGamma QRDecomposition QuadraticIrrationalQ QuadraticOptimization Quantile QuantilePlot Quantity QuantityArray QuantityDistribution QuantityForm QuantityMagnitude QuantityQ QuantityUnit QuantityVariable QuantityVariableCanonicalUnit QuantityVariableDimensions QuantityVariableIdentifier QuantityVariablePhysicalQuantity Quartics QuartileDeviation Quartiles QuartileSkewness Query QueueingNetworkProcess QueueingProcess QueueProperties Quiet Quit Quotient QuotientRemainder"+"RadialGradientImage RadialityCentrality RadicalBox RadicalBoxOptions RadioButton RadioButtonBar RadioButtonBox RadioButtonBoxOptions Radon RadonTransform RamanujanTau RamanujanTauL RamanujanTauTheta RamanujanTauZ Ramp Random RandomChoice RandomColor RandomComplex RandomEntity RandomFunction RandomGeoPosition RandomGraph RandomImage RandomInstance RandomInteger RandomPermutation RandomPoint RandomPolygon RandomPolyhedron RandomPrime RandomReal RandomSample RandomSeed RandomSeeding RandomVariate RandomWalkProcess RandomWord Range RangeFilter RangeSpecification RankedMax RankedMin RarerProbability Raster Raster3D Raster3DBox Raster3DBoxOptions RasterArray RasterBox RasterBoxOptions Rasterize RasterSize Rational RationalFunctions Rationalize Rationals Ratios RawArray RawBoxes RawData RawMedium RayleighDistribution Re Read ReadByteArray ReadLine ReadList ReadProtected ReadString Real RealAbs RealBlockDiagonalForm RealDigits RealExponent Reals RealSign Reap RecognitionPrior RecognitionThreshold Record RecordLists RecordSeparators Rectangle RectangleBox RectangleBoxOptions RectangleChart RectangleChart3D RectangularRepeatingElement RecurrenceFilter RecurrenceTable RecurringDigitsForm Red Reduce RefBox ReferenceLineStyle ReferenceMarkers ReferenceMarkerStyle Refine ReflectionMatrix ReflectionTransform Refresh RefreshRate Region RegionBinarize RegionBoundary RegionBounds RegionCentroid RegionDifference RegionDimension RegionDisjoint RegionDistance RegionDistanceFunction RegionEmbeddingDimension RegionEqual RegionFunction RegionImage RegionIntersection RegionMeasure RegionMember RegionMemberFunction RegionMoment RegionNearest RegionNearestFunction RegionPlot RegionPlot3D RegionProduct RegionQ RegionResize RegionSize RegionSymmetricDifference RegionUnion RegionWithin RegisterExternalEvaluator RegularExpression Regularization RegularlySampledQ RegularPolygon ReIm ReImLabels ReImPlot ReImStyle Reinstall RelationalDatabase RelationGraph Release ReleaseHold ReliabilityDistribution ReliefImage ReliefPlot RemoteAuthorizationCaching RemoteConnect RemoteConnectionObject RemoteFile RemoteRun RemoteRunProcess Remove RemoveAlphaChannel RemoveAsynchronousTask RemoveAudioStream RemoveBackground RemoveChannelListener RemoveChannelSubscribers Removed RemoveDiacritics RemoveInputStreamMethod RemoveOutputStreamMethod RemoveProperty RemoveScheduledTask RemoveUsers RenameDirectory RenameFile RenderAll RenderingOptions RenewalProcess RenkoChart RepairMesh Repeated RepeatedNull RepeatedString RepeatedTiming RepeatingElement Replace ReplaceAll ReplaceHeldPart ReplaceImageValue ReplaceList ReplacePart ReplacePixelValue ReplaceRepeated ReplicateLayer RequiredPhysicalQuantities Resampling ResamplingAlgorithmData ResamplingMethod Rescale RescalingTransform ResetDirectory ResetMenusPacket ResetScheduledTask ReshapeLayer Residue ResizeLayer Resolve ResourceAcquire ResourceData ResourceFunction ResourceObject ResourceRegister ResourceRemove ResourceSearch ResourceSubmissionObject ResourceSubmit ResourceSystemBase ResourceUpdate ResponseForm Rest RestartInterval Restricted Resultant ResumePacket Return ReturnEntersInput ReturnExpressionPacket ReturnInputFormPacket ReturnPacket ReturnReceiptFunction ReturnTextPacket Reverse ReverseBiorthogonalSplineWavelet ReverseElement ReverseEquilibrium ReverseGraph ReverseSort ReverseSortBy ReverseUpEquilibrium RevolutionAxis RevolutionPlot3D RGBColor RiccatiSolve RiceDistribution RidgeFilter RiemannR RiemannSiegelTheta RiemannSiegelZ RiemannXi Riffle Right RightArrow RightArrowBar RightArrowLeftArrow RightComposition RightCosetRepresentative RightDownTeeVector RightDownVector RightDownVectorBar RightTee RightTeeArrow RightTeeVector RightTriangle RightTriangleBar RightTriangleEqual RightUpDownVector RightUpTeeVector RightUpVector RightUpVectorBar RightVector RightVectorBar RiskAchievementImportance RiskReductionImportance RogersTanimotoDissimilarity RollPitchYawAngles RollPitchYawMatrix RomanNumeral Root RootApproximant RootIntervals RootLocusPlot RootMeanSquare RootOfUnityQ RootReduce Roots RootSum Rotate RotateLabel RotateLeft RotateRight RotationAction RotationBox RotationBoxOptions RotationMatrix RotationTransform Round RoundImplies RoundingRadius Row RowAlignments RowBackgrounds RowBox RowHeights RowLines RowMinHeight RowReduce RowsEqual RowSpacings RSolve RSolveValue RudinShapiro RudvalisGroupRu Rule RuleCondition RuleDelayed RuleForm RulePlot RulerUnits Run RunProcess RunScheduledTask RunThrough RuntimeAttributes RuntimeOptions RussellRaoDissimilarity"+"SameQ SameTest SampledEntityClass SampleDepth SampledSoundFunction SampledSoundList SampleRate SamplingPeriod SARIMAProcess SARMAProcess SASTriangle SatelliteData SatisfiabilityCount SatisfiabilityInstances SatisfiableQ Saturday Save Saveable SaveAutoDelete SaveConnection SaveDefinitions SavitzkyGolayMatrix SawtoothWave Scale Scaled ScaleDivisions ScaledMousePosition ScaleOrigin ScalePadding ScaleRanges ScaleRangeStyle ScalingFunctions ScalingMatrix ScalingTransform Scan ScheduledTask ScheduledTaskActiveQ ScheduledTaskInformation ScheduledTaskInformationData ScheduledTaskObject ScheduledTasks SchurDecomposition ScientificForm ScientificNotationThreshold ScorerGi ScorerGiPrime ScorerHi ScorerHiPrime ScreenRectangle ScreenStyleEnvironment ScriptBaselineShifts ScriptForm ScriptLevel ScriptMinSize ScriptRules ScriptSizeMultipliers Scrollbars ScrollingOptions ScrollPosition SearchAdjustment SearchIndexObject SearchIndices SearchQueryString SearchResultObject Sec Sech SechDistribution SecondOrderConeOptimization SectionGrouping SectorChart SectorChart3D SectorOrigin SectorSpacing SecuredAuthenticationKey SecuredAuthenticationKeys SeedRandom Select Selectable SelectComponents SelectedCells SelectedNotebook SelectFirst Selection SelectionAnimate SelectionCell SelectionCellCreateCell SelectionCellDefaultStyle SelectionCellParentStyle SelectionCreateCell SelectionDebuggerTag SelectionDuplicateCell SelectionEvaluate SelectionEvaluateCreateCell SelectionMove SelectionPlaceholder SelectionSetStyle SelectWithContents SelfLoops SelfLoopStyle SemanticImport SemanticImportString SemanticInterpretation SemialgebraicComponentInstances SemidefiniteOptimization SendMail SendMessage Sequence SequenceAlignment SequenceAttentionLayer SequenceCases SequenceCount SequenceFold SequenceFoldList SequenceForm SequenceHold SequenceLastLayer SequenceMostLayer SequencePosition SequencePredict SequencePredictorFunction SequenceReplace SequenceRestLayer SequenceReverseLayer SequenceSplit Series SeriesCoefficient SeriesData ServiceConnect ServiceDisconnect ServiceExecute ServiceObject ServiceRequest ServiceResponse ServiceSubmit SessionSubmit SessionTime Set SetAccuracy SetAlphaChannel SetAttributes Setbacks SetBoxFormNamesPacket SetCloudDirectory SetCookies SetDelayed SetDirectory SetEnvironment SetEvaluationNotebook SetFileDate SetFileLoadingContext SetNotebookStatusLine SetOptions SetOptionsPacket SetPermissions SetPrecision SetProperty SetSecuredAuthenticationKey SetSelectedNotebook SetSharedFunction SetSharedVariable SetSpeechParametersPacket SetStreamPosition SetSystemModel SetSystemOptions Setter SetterBar SetterBox SetterBoxOptions Setting SetUsers SetValue Shading Shallow ShannonWavelet ShapiroWilkTest Share SharingList Sharpen ShearingMatrix ShearingTransform ShellRegion ShenCastanMatrix ShiftedGompertzDistribution ShiftRegisterSequence Short ShortDownArrow Shortest ShortestMatch ShortestPathFunction ShortLeftArrow ShortRightArrow ShortTimeFourier ShortTimeFourierData ShortUpArrow Show ShowAutoConvert ShowAutoSpellCheck ShowAutoStyles ShowCellBracket ShowCellLabel ShowCellTags ShowClosedCellArea ShowCodeAssist ShowContents ShowControls ShowCursorTracker ShowGroupOpenCloseIcon ShowGroupOpener ShowInvisibleCharacters ShowPageBreaks ShowPredictiveInterface ShowSelection ShowShortBoxForm ShowSpecialCharacters ShowStringCharacters ShowSyntaxStyles ShrinkingDelay ShrinkWrapBoundingBox SiderealTime SiegelTheta SiegelTukeyTest SierpinskiCurve SierpinskiMesh Sign Signature SignedRankTest SignedRegionDistance SignificanceLevel SignPadding SignTest SimilarityRules SimpleGraph SimpleGraphQ SimplePolygonQ SimplePolyhedronQ Simplex Simplify Sin Sinc SinghMaddalaDistribution SingleEvaluation SingleLetterItalics SingleLetterStyle SingularValueDecomposition SingularValueList SingularValuePlot SingularValues Sinh SinhIntegral SinIntegral SixJSymbol Skeleton SkeletonTransform SkellamDistribution Skewness SkewNormalDistribution SkinStyle Skip SliceContourPlot3D SliceDensityPlot3D SliceDistribution SliceVectorPlot3D Slider Slider2D Slider2DBox Slider2DBoxOptions SliderBox SliderBoxOptions SlideView Slot SlotSequence Small SmallCircle Smaller SmithDecomposition SmithDelayCompensator SmithWatermanSimilarity SmoothDensityHistogram SmoothHistogram SmoothHistogram3D SmoothKernelDistribution SnDispersion Snippet SnubPolyhedron SocialMediaData Socket SocketConnect SocketListen SocketListener SocketObject SocketOpen SocketReadMessage SocketReadyQ Sockets SocketWaitAll SocketWaitNext SoftmaxLayer SokalSneathDissimilarity SolarEclipse SolarSystemFeatureData SolidAngle SolidData SolidRegionQ Solve SolveAlways SolveDelayed Sort SortBy SortedBy SortedEntityClass Sound SoundAndGraphics SoundNote SoundVolume SourceLink Sow Space SpaceCurveData SpaceForm Spacer Spacings Span SpanAdjustments SpanCharacterRounding SpanFromAbove SpanFromBoth SpanFromLeft SpanLineThickness SpanMaxSize SpanMinSize SpanningCharacters SpanSymmetric SparseArray SpatialGraphDistribution SpatialMedian SpatialTransformationLayer Speak SpeakTextPacket SpearmanRankTest SpearmanRho SpeciesData SpecificityGoal SpectralLineData Spectrogram SpectrogramArray Specularity SpeechRecognize SpeechSynthesize SpellingCorrection SpellingCorrectionList SpellingDictionaries SpellingDictionariesPath SpellingOptions SpellingSuggestionsPacket Sphere SphereBox SpherePoints SphericalBesselJ SphericalBesselY SphericalHankelH1 SphericalHankelH2 SphericalHarmonicY SphericalPlot3D SphericalRegion SphericalShell SpheroidalEigenvalue SpheroidalJoiningFactor SpheroidalPS SpheroidalPSPrime SpheroidalQS SpheroidalQSPrime SpheroidalRadialFactor SpheroidalS1 SpheroidalS1Prime SpheroidalS2 SpheroidalS2Prime Splice SplicedDistribution SplineClosed SplineDegree SplineKnots SplineWeights Split SplitBy SpokenString Sqrt SqrtBox SqrtBoxOptions Square SquaredEuclideanDistance SquareFreeQ SquareIntersection SquareMatrixQ SquareRepeatingElement SquaresR SquareSubset SquareSubsetEqual SquareSuperset SquareSupersetEqual SquareUnion SquareWave SSSTriangle StabilityMargins StabilityMarginsStyle StableDistribution Stack StackBegin StackComplete StackedDateListPlot StackedListPlot StackInhibit StadiumShape StandardAtmosphereData StandardDeviation StandardDeviationFilter StandardForm Standardize Standardized StandardOceanData StandbyDistribution Star StarClusterData StarData StarGraph StartAsynchronousTask StartExternalSession StartingStepSize StartOfLine StartOfString StartProcess StartScheduledTask StartupSound StartWebSession StateDimensions StateFeedbackGains StateOutputEstimator StateResponse StateSpaceModel StateSpaceRealization StateSpaceTransform StateTransformationLinearize StationaryDistribution StationaryWaveletPacketTransform StationaryWaveletTransform StatusArea StatusCentrality StepMonitor StereochemistryElements StieltjesGamma StirlingS1 StirlingS2 StopAsynchronousTask StoppingPowerData StopScheduledTask StrataVariables StratonovichProcess StreamColorFunction StreamColorFunctionScaling StreamDensityPlot StreamMarkers StreamPlot StreamPoints StreamPosition Streams StreamScale StreamStyle String StringBreak StringByteCount StringCases StringContainsQ StringCount StringDelete StringDrop StringEndsQ StringExpression StringExtract StringForm StringFormat StringFreeQ StringInsert StringJoin StringLength StringMatchQ StringPadLeft StringPadRight StringPart StringPartition StringPosition StringQ StringRepeat StringReplace StringReplaceList StringReplacePart StringReverse StringRiffle StringRotateLeft StringRotateRight StringSkeleton StringSplit StringStartsQ StringTake StringTemplate StringToByteArray StringToStream StringTrim StripBoxes StripOnInput StripWrapperBoxes StrokeForm StructuralImportance StructuredArray StructuredSelection StruveH StruveL Stub StudentTDistribution Style StyleBox StyleBoxAutoDelete StyleData StyleDefinitions StyleForm StyleHints StyleKeyMapping StyleMenuListing StyleNameDialogSettings StyleNames StylePrint StyleSheetPath Subdivide Subfactorial Subgraph SubMinus SubPlus SubresultantPolynomialRemainders SubresultantPolynomials Subresultants Subscript SubscriptBox SubscriptBoxOptions Subscripted Subsequences Subset SubsetEqual SubsetMap SubsetQ Subsets SubStar SubstitutionSystem Subsuperscript SubsuperscriptBox SubsuperscriptBoxOptions Subtract SubtractFrom SubtractSides SubValues Succeeds SucceedsEqual SucceedsSlantEqual SucceedsTilde Success SuchThat Sum SumConvergence SummationLayer Sunday SunPosition Sunrise Sunset SuperDagger SuperMinus SupernovaData SuperPlus Superscript SuperscriptBox SuperscriptBoxOptions Superset SupersetEqual SuperStar Surd SurdForm SurfaceArea SurfaceColor SurfaceData SurfaceGraphics SurvivalDistribution SurvivalFunction SurvivalModel SurvivalModelFit SuspendPacket SuzukiDistribution SuzukiGroupSuz SwatchLegend Switch Symbol SymbolName SymletWavelet Symmetric SymmetricGroup SymmetricKey SymmetricMatrixQ SymmetricPolynomial SymmetricReduction Symmetrize SymmetrizedArray SymmetrizedArrayRules SymmetrizedDependentComponents SymmetrizedIndependentComponents SymmetrizedReplacePart SynchronousInitialization SynchronousUpdating Synonyms Syntax SyntaxForm SyntaxInformation SyntaxLength SyntaxPacket SyntaxQ SynthesizeMissingValues SystemDialogInput SystemException SystemGet SystemHelpPath SystemInformation SystemInformationData SystemInstall SystemModel SystemModeler SystemModelExamples SystemModelLinearize SystemModelParametricSimulate SystemModelPlot SystemModelProgressReporting SystemModelReliability SystemModels SystemModelSimulate SystemModelSimulateSensitivity SystemModelSimulationData SystemOpen SystemOptions SystemProcessData SystemProcesses SystemsConnectionsModel SystemsModelDelay SystemsModelDelayApproximate SystemsModelDelete SystemsModelDimensions SystemsModelExtract SystemsModelFeedbackConnect SystemsModelLabels SystemsModelLinearity SystemsModelMerge SystemsModelOrder SystemsModelParallelConnect SystemsModelSeriesConnect SystemsModelStateFeedbackConnect SystemsModelVectorRelativeOrders SystemStub SystemTest"+"Tab TabFilling Table TableAlignments TableDepth TableDirections TableForm TableHeadings TableSpacing TableView TableViewBox TableViewBoxBackground TableViewBoxOptions TabSpacings TabView TabViewBox TabViewBoxOptions TagBox TagBoxNote TagBoxOptions TaggingRules TagSet TagSetDelayed TagStyle TagUnset Take TakeDrop TakeLargest TakeLargestBy TakeList TakeSmallest TakeSmallestBy TakeWhile Tally Tan Tanh TargetDevice TargetFunctions TargetSystem TargetUnits TaskAbort TaskExecute TaskObject TaskRemove TaskResume Tasks TaskSuspend TaskWait TautologyQ TelegraphProcess TemplateApply TemplateArgBox TemplateBox TemplateBoxOptions TemplateEvaluate TemplateExpression TemplateIf TemplateObject TemplateSequence TemplateSlot TemplateSlotSequence TemplateUnevaluated TemplateVerbatim TemplateWith TemporalData TemporalRegularity Temporary TemporaryVariable TensorContract TensorDimensions TensorExpand TensorProduct TensorQ TensorRank TensorReduce TensorSymmetry TensorTranspose TensorWedge TestID TestReport TestReportObject TestResultObject Tetrahedron TetrahedronBox TetrahedronBoxOptions TeXForm TeXSave Text Text3DBox Text3DBoxOptions TextAlignment TextBand TextBoundingBox TextBox TextCases TextCell TextClipboardType TextContents TextData TextElement TextForm TextGrid TextJustification TextLine TextPacket TextParagraph TextPosition TextRecognize TextSearch TextSearchReport TextSentences TextString TextStructure TextStyle TextTranslation Texture TextureCoordinateFunction TextureCoordinateScaling TextWords Therefore ThermodynamicData ThermometerGauge Thick Thickness Thin Thinning ThisLink ThompsonGroupTh Thread ThreadingLayer ThreeJSymbol Threshold Through Throw ThueMorse Thumbnail Thursday Ticks TicksStyle TideData Tilde TildeEqual TildeFullEqual TildeTilde TimeConstrained TimeConstraint TimeDirection TimeFormat TimeGoal TimelinePlot TimeObject TimeObjectQ Times TimesBy TimeSeries TimeSeriesAggregate TimeSeriesForecast TimeSeriesInsert TimeSeriesInvertibility TimeSeriesMap TimeSeriesMapThread TimeSeriesModel TimeSeriesModelFit TimeSeriesResample TimeSeriesRescale TimeSeriesShift TimeSeriesThread TimeSeriesWindow TimeUsed TimeValue TimeWarpingCorrespondence TimeWarpingDistance TimeZone TimeZoneConvert TimeZoneOffset Timing Tiny TitleGrouping TitsGroupT ToBoxes ToCharacterCode ToColor ToContinuousTimeModel ToDate Today ToDiscreteTimeModel ToEntity ToeplitzMatrix ToExpression ToFileName Together Toggle ToggleFalse Toggler TogglerBar TogglerBox TogglerBoxOptions ToHeldExpression ToInvertibleTimeSeries TokenWords Tolerance ToLowerCase Tomorrow ToNumberField TooBig Tooltip TooltipBox TooltipBoxOptions TooltipDelay TooltipStyle Top TopHatTransform ToPolarCoordinates TopologicalSort ToRadicals ToRules ToSphericalCoordinates ToString Total TotalHeight TotalLayer TotalVariationFilter TotalWidth TouchPosition TouchscreenAutoZoom TouchscreenControlPlacement ToUpperCase Tr Trace TraceAbove TraceAction TraceBackward TraceDepth TraceDialog TraceForward TraceInternal TraceLevel TraceOff TraceOn TraceOriginal TracePrint TraceScan TrackedSymbols TrackingFunction TracyWidomDistribution TradingChart TraditionalForm TraditionalFunctionNotation TraditionalNotation TraditionalOrder TrainingProgressCheckpointing TrainingProgressFunction TrainingProgressMeasurements TrainingProgressReporting TrainingStoppingCriterion TransferFunctionCancel TransferFunctionExpand TransferFunctionFactor TransferFunctionModel TransferFunctionPoles TransferFunctionTransform TransferFunctionZeros TransformationClass TransformationFunction TransformationFunctions TransformationMatrix TransformedDistribution TransformedField TransformedProcess TransformedRegion TransitionDirection TransitionDuration TransitionEffect TransitiveClosureGraph TransitiveReductionGraph Translate TranslationOptions TranslationTransform Transliterate Transparent TransparentColor Transpose TransposeLayer TrapSelection TravelDirections TravelDirectionsData TravelDistance TravelDistanceList TravelMethod TravelTime TreeForm TreeGraph TreeGraphQ TreePlot TrendStyle Triangle TriangleCenter TriangleConstruct TriangleMeasurement TriangleWave TriangularDistribution TriangulateMesh Trig TrigExpand TrigFactor TrigFactorList Trigger TrigReduce TrigToExp TrimmedMean TrimmedVariance TropicalStormData True TrueQ TruncatedDistribution TruncatedPolyhedron TsallisQExponentialDistribution TsallisQGaussianDistribution TTest Tube TubeBezierCurveBox TubeBezierCurveBoxOptions TubeBox TubeBoxOptions TubeBSplineCurveBox TubeBSplineCurveBoxOptions Tuesday TukeyLambdaDistribution TukeyWindow TunnelData Tuples TuranGraph TuringMachine TuttePolynomial TwoWayRule Typed TypeSpecifier"+"UnateQ Uncompress UnconstrainedParameters Undefined UnderBar Underflow Underlined Underoverscript UnderoverscriptBox UnderoverscriptBoxOptions Underscript UnderscriptBox UnderscriptBoxOptions UnderseaFeatureData UndirectedEdge UndirectedGraph UndirectedGraphQ UndoOptions UndoTrackedVariables Unequal UnequalTo Unevaluated UniformDistribution UniformGraphDistribution UniformPolyhedron UniformSumDistribution Uninstall Union UnionPlus Unique UnitaryMatrixQ UnitBox UnitConvert UnitDimensions Unitize UnitRootTest UnitSimplify UnitStep UnitSystem UnitTriangle UnitVector UnitVectorLayer UnityDimensions UniverseModelData UniversityData UnixTime Unprotect UnregisterExternalEvaluator UnsameQ UnsavedVariables Unset UnsetShared UntrackedVariables Up UpArrow UpArrowBar UpArrowDownArrow Update UpdateDynamicObjects UpdateDynamicObjectsSynchronous UpdateInterval UpdateSearchIndex UpDownArrow UpEquilibrium UpperCaseQ UpperLeftArrow UpperRightArrow UpperTriangularize UpperTriangularMatrixQ Upsample UpSet UpSetDelayed UpTee UpTeeArrow UpTo UpValues URL URLBuild URLDecode URLDispatcher URLDownload URLDownloadSubmit URLEncode URLExecute URLExpand URLFetch URLFetchAsynchronous URLParse URLQueryDecode URLQueryEncode URLRead URLResponseTime URLSave URLSaveAsynchronous URLShorten URLSubmit UseGraphicsRange UserDefinedWavelet Using UsingFrontEnd UtilityFunction"+"V2Get ValenceErrorHandling ValidationLength ValidationSet Value ValueBox ValueBoxOptions ValueDimensions ValueForm ValuePreprocessingFunction ValueQ Values ValuesData Variables Variance VarianceEquivalenceTest VarianceEstimatorFunction VarianceGammaDistribution VarianceTest VectorAngle VectorAround VectorColorFunction VectorColorFunctionScaling VectorDensityPlot VectorGlyphData VectorGreater VectorGreaterEqual VectorLess VectorLessEqual VectorMarkers VectorPlot VectorPlot3D VectorPoints VectorQ Vectors VectorScale VectorStyle Vee Verbatim Verbose VerboseConvertToPostScriptPacket VerificationTest VerifyConvergence VerifyDerivedKey VerifyDigitalSignature VerifyInterpretation VerifySecurityCertificates VerifySolutions VerifyTestAssumptions Version VersionNumber VertexAdd VertexCapacity VertexColors VertexComponent VertexConnectivity VertexContract VertexCoordinateRules VertexCoordinates VertexCorrelationSimilarity VertexCosineSimilarity VertexCount VertexCoverQ VertexDataCoordinates VertexDegree VertexDelete VertexDiceSimilarity VertexEccentricity VertexInComponent VertexInDegree VertexIndex VertexJaccardSimilarity VertexLabeling VertexLabels VertexLabelStyle VertexList VertexNormals VertexOutComponent VertexOutDegree VertexQ VertexRenderingFunction VertexReplace VertexShape VertexShapeFunction VertexSize VertexStyle VertexTextureCoordinates VertexWeight VertexWeightedGraphQ Vertical VerticalBar VerticalForm VerticalGauge VerticalSeparator VerticalSlider VerticalTilde ViewAngle ViewCenter ViewMatrix ViewPoint ViewPointSelectorSettings ViewPort ViewProjection ViewRange ViewVector ViewVertical VirtualGroupData Visible VisibleCell VoiceStyleData VoigtDistribution VolcanoData Volume VonMisesDistribution VoronoiMesh"+"WaitAll WaitAsynchronousTask WaitNext WaitUntil WakebyDistribution WalleniusHypergeometricDistribution WaringYuleDistribution WarpingCorrespondence WarpingDistance WatershedComponents WatsonUSquareTest WattsStrogatzGraphDistribution WaveletBestBasis WaveletFilterCoefficients WaveletImagePlot WaveletListPlot WaveletMapIndexed WaveletMatrixPlot WaveletPhi WaveletPsi WaveletScale WaveletScalogram WaveletThreshold WeaklyConnectedComponents WeaklyConnectedGraphComponents WeaklyConnectedGraphQ WeakStationarity WeatherData WeatherForecastData WebAudioSearch WebElementObject WeberE WebExecute WebImage WebImageSearch WebSearch WebSessionObject WebSessions WebWindowObject Wedge Wednesday WeibullDistribution WeierstrassE1 WeierstrassE2 WeierstrassE3 WeierstrassEta1 WeierstrassEta2 WeierstrassEta3 WeierstrassHalfPeriods WeierstrassHalfPeriodW1 WeierstrassHalfPeriodW2 WeierstrassHalfPeriodW3 WeierstrassInvariantG2 WeierstrassInvariantG3 WeierstrassInvariants WeierstrassP WeierstrassPPrime WeierstrassSigma WeierstrassZeta WeightedAdjacencyGraph WeightedAdjacencyMatrix WeightedData WeightedGraphQ Weights WelchWindow WheelGraph WhenEvent Which While White WhiteNoiseProcess WhitePoint Whitespace WhitespaceCharacter WhittakerM WhittakerW WienerFilter WienerProcess WignerD WignerSemicircleDistribution WikipediaData WikipediaSearch WilksW WilksWTest WindDirectionData WindingCount WindingPolygon WindowClickSelect WindowElements WindowFloating WindowFrame WindowFrameElements WindowMargins WindowMovable WindowOpacity WindowPersistentStyles WindowSelected WindowSize WindowStatusArea WindowTitle WindowToolbars WindowWidth WindSpeedData WindVectorData WinsorizedMean WinsorizedVariance WishartMatrixDistribution With WolframAlpha WolframAlphaDate WolframAlphaQuantity WolframAlphaResult WolframLanguageData Word WordBoundary WordCharacter WordCloud WordCount WordCounts WordData WordDefinition WordFrequency WordFrequencyData WordList WordOrientation WordSearch WordSelectionFunction WordSeparators WordSpacings WordStem WordTranslation WorkingPrecision WrapAround Write WriteLine WriteString Wronskian"+"XMLElement XMLObject XMLTemplate Xnor Xor XYZColor"+"Yellow Yesterday YuleDissimilarity"+"ZernikeR ZeroSymmetric ZeroTest ZeroWidthTimes Zeta ZetaZero ZIPCodeData ZipfDistribution ZoomCenter ZoomFactor ZTest ZTransform"+"$Aborted $ActivationGroupID $ActivationKey $ActivationUserRegistered $AddOnsDirectory $AllowExternalChannelFunctions $AssertFunction $Assumptions $AsynchronousTask $AudioInputDevices $AudioOutputDevices $BaseDirectory $BatchInput $BatchOutput $BlockchainBase $BoxForms $ByteOrdering $CacheBaseDirectory $Canceled $ChannelBase $CharacterEncoding $CharacterEncodings $CloudBase $CloudConnected $CloudCreditsAvailable $CloudEvaluation $CloudExpressionBase $CloudObjectNameFormat $CloudObjectURLType $CloudRootDirectory $CloudSymbolBase $CloudUserID $CloudUserUUID $CloudVersion $CloudVersionNumber $CloudWolframEngineVersionNumber $CommandLine $CompilationTarget $ConditionHold $ConfiguredKernels $Context $ContextPath $ControlActiveSetting $Cookies $CookieStore $CreationDate $CurrentLink $CurrentTask $CurrentWebSession $DateStringFormat $DefaultAudioInputDevice $DefaultAudioOutputDevice $DefaultFont $DefaultFrontEnd $DefaultImagingDevice $DefaultLocalBase $DefaultMailbox $DefaultNetworkInterface $DefaultPath $Display $DisplayFunction $DistributedContexts $DynamicEvaluation $Echo $EmbedCodeEnvironments $EmbeddableServices $EntityStores $Epilog $EvaluationCloudBase $EvaluationCloudObject $EvaluationEnvironment $ExportFormats $Failed $FinancialDataSource $FontFamilies $FormatType $FrontEnd $FrontEndSession $GeoEntityTypes $GeoLocation $GeoLocationCity $GeoLocationCountry $GeoLocationPrecision $GeoLocationSource $HistoryLength $HomeDirectory $HTMLExportRules $HTTPCookies $HTTPRequest $IgnoreEOF $ImageFormattingWidth $ImagingDevice $ImagingDevices $ImportFormats $IncomingMailSettings $InitialDirectory $Initialization $InitializationContexts $Input $InputFileName $InputStreamMethods $Inspector $InstallationDate $InstallationDirectory $InterfaceEnvironment $InterpreterTypes $IterationLimit $KernelCount $KernelID $Language $LaunchDirectory $LibraryPath $LicenseExpirationDate $LicenseID $LicenseProcesses $LicenseServer $LicenseSubprocesses $LicenseType $Line $Linked $LinkSupported $LoadedFiles $LocalBase $LocalSymbolBase $MachineAddresses $MachineDomain $MachineDomains $MachineEpsilon $MachineID $MachineName $MachinePrecision $MachineType $MaxExtraPrecision $MaxLicenseProcesses $MaxLicenseSubprocesses $MaxMachineNumber $MaxNumber $MaxPiecewiseCases $MaxPrecision $MaxRootDegree $MessageGroups $MessageList $MessagePrePrint $Messages $MinMachineNumber $MinNumber $MinorReleaseNumber $MinPrecision $MobilePhone $ModuleNumber $NetworkConnected $NetworkInterfaces $NetworkLicense $NewMessage $NewSymbol $Notebooks $NoValue $NumberMarks $Off $OperatingSystem $Output $OutputForms $OutputSizeLimit $OutputStreamMethods $Packages $ParentLink $ParentProcessID $PasswordFile $PatchLevelID $Path $PathnameSeparator $PerformanceGoal $Permissions $PermissionsGroupBase $PersistenceBase $PersistencePath $PipeSupported $PlotTheme $Post $Pre $PreferencesDirectory $PreInitialization $PrePrint $PreRead $PrintForms $PrintLiteral $Printout3DPreviewer $ProcessID $ProcessorCount $ProcessorType $ProductInformation $ProgramName $PublisherID $RandomState $RecursionLimit $RegisteredDeviceClasses $RegisteredUserName $ReleaseNumber $RequesterAddress $RequesterWolframID $RequesterWolframUUID $ResourceSystemBase $RootDirectory $ScheduledTask $ScriptCommandLine $ScriptInputString $SecuredAuthenticationKeyTokens $ServiceCreditsAvailable $Services $SessionID $SetParentLink $SharedFunctions $SharedVariables $SoundDisplay $SoundDisplayFunction $SourceLink $SSHAuthentication $SummaryBoxDataSizeLimit $SuppressInputFormHeads $SynchronousEvaluation $SyntaxHandler $System $SystemCharacterEncoding $SystemID $SystemMemory $SystemShell $SystemTimeZone $SystemWordLength $TemplatePath $TemporaryDirectory $TemporaryPrefix $TestFileName $TextStyle $TimedOut $TimeUnit $TimeZone $TimeZoneEntity $TopDirectory $TraceOff $TraceOn $TracePattern $TracePostAction $TracePreAction $UnitSystem $Urgent $UserAddOnsDirectory $UserAgentLanguages $UserAgentMachine $UserAgentName $UserAgentOperatingSystem $UserAgentString $UserAgentVersion $UserBaseDirectory $UserDocumentsDirectory $Username $UserName $UserURLBase $Version $VersionNumber $VoiceStyles $WolframID $WolframUUID",contains:[hljs.COMMENT("\\(\\*","\\*\\)",{contains:["self"]}),hljs.QUOTE_STRING_MODE,hljs.C_NUMBER_MODE]}});hljs.registerLanguage("nginx",function(hljs){var VAR={className:"variable",variants:[{begin:/\$\d+/},{begin:/\$\{/,end:/}/},{begin:"[\\$\\@]"+hljs.UNDERSCORE_IDENT_RE}]};var DEFAULT={endsWithParent:true,lexemes:"[a-z/_]+",keywords:{literal:"on off yes no true false none blocked debug info notice warn error crit "+"select break last permanent redirect kqueue rtsig epoll poll /dev/poll"},relevance:0,illegal:"=>",contains:[hljs.HASH_COMMENT_MODE,{className:"string",contains:[hljs.BACKSLASH_ESCAPE,VAR],variants:[{begin:/"/,end:/"/},{begin:/'/,end:/'/}]},{begin:"([a-z]+):/",end:"\\s",endsWithParent:true,excludeEnd:true,contains:[VAR]},{className:"regexp",contains:[hljs.BACKSLASH_ESCAPE,VAR],variants:[{begin:"\\s\\^",end:"\\s|{|;",returnEnd:true},{begin:"~\\*?\\s+",end:"\\s|{|;",returnEnd:true},{begin:"\\*(\\.[a-z\\-]+)+"},{begin:"([a-z\\-]+\\.)+\\*"}]},{className:"number",begin:"\\b\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b"},{className:"number",begin:"\\b\\d+[kKmMgGdshdwy]*\\b",relevance:0},VAR]};return{aliases:["nginxconf"],contains:[hljs.HASH_COMMENT_MODE,{begin:hljs.UNDERSCORE_IDENT_RE+"\\s+{",returnBegin:true,end:"{",contains:[{className:"section",begin:hljs.UNDERSCORE_IDENT_RE}],relevance:0},{begin:hljs.UNDERSCORE_IDENT_RE+"\\s",end:";|{",returnBegin:true,contains:[{className:"attribute",begin:hljs.UNDERSCORE_IDENT_RE,starts:DEFAULT}],relevance:0}],illegal:"[^\\s\\}]"}});hljs.registerLanguage("n1ql",function(hljs){return{case_insensitive:true,contains:[{beginKeywords:"build create index delete drop explain infer|10 insert merge prepare select update upsert|10",end:/;/,endsWithParent:true,keywords:{keyword:"all alter analyze and any array as asc begin between binary boolean break bucket build by call "+"case cast cluster collate collection commit connect continue correlate cover create database "+"dataset datastore declare decrement delete derived desc describe distinct do drop each element "+"else end every except exclude execute exists explain fetch first flatten for force from "+"function grant group gsi having if ignore ilike in include increment index infer inline inner "+"insert intersect into is join key keys keyspace known last left let letting like limit lsm map "+"mapping matched materialized merge minus namespace nest not number object offset on "+"option or order outer over parse partition password path pool prepare primary private privilege "+"procedure public raw realm reduce rename return returning revoke right role rollback satisfies "+"schema select self semi set show some start statistics string system then to transaction trigger "+"truncate under union unique unknown unnest unset update upsert use user using validate value "+"valued values via view when where while with within work xor",literal:"true false null missing|5",built_in:"array_agg array_append array_concat array_contains array_count array_distinct array_ifnull array_length "+"array_max array_min array_position array_prepend array_put array_range array_remove array_repeat array_replace "+"array_reverse array_sort array_sum avg count max min sum greatest least ifmissing ifmissingornull ifnull "+"missingif nullif ifinf ifnan ifnanorinf naninf neginfif posinfif clock_millis clock_str date_add_millis "+"date_add_str date_diff_millis date_diff_str date_part_millis date_part_str date_trunc_millis date_trunc_str "+"duration_to_str millis str_to_millis millis_to_str millis_to_utc millis_to_zone_name now_millis now_str "+"str_to_duration str_to_utc str_to_zone_name decode_json encode_json encoded_size poly_length base64 base64_encode "+"base64_decode meta uuid abs acos asin atan atan2 ceil cos degrees e exp ln log floor pi power radians random "+"round sign sin sqrt tan trunc object_length object_names object_pairs object_inner_pairs object_values "+"object_inner_values object_add object_put object_remove object_unwrap regexp_contains regexp_like regexp_position "+"regexp_replace contains initcap length lower ltrim position repeat replace rtrim split substr title trim upper "+"isarray isatom isboolean isnumber isobject isstring type toarray toatom toboolean tonumber toobject tostring"},contains:[{className:"string",begin:"'",end:"'",contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"string",begin:'"',end:'"',contains:[hljs.BACKSLASH_ESCAPE],relevance:0},{className:"symbol",begin:"`",end:"`",contains:[hljs.BACKSLASH_ESCAPE],relevance:2},hljs.C_NUMBER_MODE,hljs.C_BLOCK_COMMENT_MODE]},hljs.C_BLOCK_COMMENT_MODE]}});hljs.registerLanguage("pf",function(hljs){var MACRO={className:"variable",begin:/\$[\w\d#@][\w\d_]*/};var TABLE={className:"variable",begin:/<(?!\/)/,end:/>/};var QUOTE_STRING={className:"string",begin:/"/,end:/"/};return{aliases:["pf.conf"],lexemes:/[a-z0-9_<>-]+/,keywords:{built_in:"block match pass load anchor|5 antispoof|10 set table",keyword:"in out log quick on rdomain inet inet6 proto from port os to route"+"allow-opts divert-packet divert-reply divert-to flags group icmp-type"+"icmp6-type label once probability recieved-on rtable prio queue"+"tos tag tagged user keep fragment for os drop"+"af-to|10 binat-to|10 nat-to|10 rdr-to|10 bitmask least-stats random round-robin"+"source-hash static-port"+"dup-to reply-to route-to"+"parent bandwidth default min max qlimit"+"block-policy debug fingerprints hostid limit loginterface optimization"+"reassemble ruleset-optimization basic none profile skip state-defaults"+"state-policy timeout"+"const counters persist"+"no modulate synproxy state|5 floating if-bound no-sync pflow|10 sloppy"+"source-track global rule max-src-nodes max-src-states max-src-conn"+"max-src-conn-rate overload flush"+"scrub|5 max-mss min-ttl no-df|10 random-id",literal:"all any no-route self urpf-failed egress|5 unknown"},contains:[hljs.HASH_COMMENT_MODE,hljs.NUMBER_MODE,hljs.QUOTE_STRING_MODE,MACRO,TABLE]}});hljs.registerLanguage("llvm",function(hljs){var identifier="([-a-zA-Z$._][\\w\\-$.]*)";return{keywords:"begin end true false declare define global "+"constant private linker_private internal "+"available_externally linkonce linkonce_odr weak "+"weak_odr appending dllimport dllexport common "+"default hidden protected extern_weak external "+"thread_local zeroinitializer undef null to tail "+"target triple datalayout volatile nuw nsw nnan "+"ninf nsz arcp fast exact inbounds align "+"addrspace section alias module asm sideeffect "+"gc dbg linker_private_weak attributes blockaddress "+"initialexec localdynamic localexec prefix unnamed_addr "+"ccc fastcc coldcc x86_stdcallcc x86_fastcallcc "+"arm_apcscc arm_aapcscc arm_aapcs_vfpcc ptx_device "+"ptx_kernel intel_ocl_bicc msp430_intrcc spir_func "+"spir_kernel x86_64_sysvcc x86_64_win64cc x86_thiscallcc "+"cc c signext zeroext inreg sret nounwind "+"noreturn noalias nocapture byval nest readnone "+"readonly inlinehint noinline alwaysinline optsize ssp "+"sspreq noredzone noimplicitfloat naked builtin cold "+"nobuiltin noduplicate nonlazybind optnone returns_twice "+"sanitize_address sanitize_memory sanitize_thread sspstrong "+"uwtable returned type opaque eq ne slt sgt "+"sle sge ult ugt ule uge oeq one olt ogt "+"ole oge ord uno ueq une x acq_rel acquire "+"alignstack atomic catch cleanup filter inteldialect "+"max min monotonic nand personality release seq_cst "+"singlethread umax umin unordered xchg add fadd "+"sub fsub mul fmul udiv sdiv fdiv urem srem "+"frem shl lshr ashr and or xor icmp fcmp "+"phi call trunc zext sext fptrunc fpext uitofp "+"sitofp fptoui fptosi inttoptr ptrtoint bitcast "+"addrspacecast select va_arg ret br switch invoke "+"unwind unreachable indirectbr landingpad resume "+"malloc alloca free load store getelementptr "+"extractelement insertelement shufflevector getresult "+"extractvalue insertvalue atomicrmw cmpxchg fence "+"argmemonly double",contains:[{className:"keyword",begin:"i\\d+"},hljs.COMMENT(";","\\n",{relevance:0}),hljs.QUOTE_STRING_MODE,{className:"string",variants:[{begin:'"',end:'[^\\\\]"'}],relevance:0},{className:"title",variants:[{begin:"@"+identifier},{begin:"@\\d+"},{begin:"!"+identifier},{begin:"!\\d+"+identifier}]},{className:"symbol",variants:[{begin:"%"+identifier},{begin:"%\\d+"},{begin:"#\\d+"}]},{className:"number",variants:[{begin:"0[xX][a-fA-F0-9]+"},{begin:"-?\\d+(?:[.]\\d+)?(?:[eE][-+]?\\d+(?:[.]\\d+)?)?"}],relevance:0}]}});hljs.registerLanguage("avrasm",function(hljs){return{case_insensitive:true,lexemes:"\\.?"+hljs.IDENT_RE,keywords:{keyword:"adc add adiw and andi asr bclr bld brbc brbs brcc brcs break breq brge brhc brhs "+"brid brie brlo brlt brmi brne brpl brsh brtc brts brvc brvs bset bst call cbi cbr "+"clc clh cli cln clr cls clt clv clz com cp cpc cpi cpse dec eicall eijmp elpm eor "+"fmul fmuls fmulsu icall ijmp in inc jmp ld ldd ldi lds lpm lsl lsr mov movw mul "+"muls mulsu neg nop or ori out pop push rcall ret reti rjmp rol ror sbc sbr sbrc sbrs "+"sec seh sbi sbci sbic sbis sbiw sei sen ser ses set sev sez sleep spm st std sts sub "+"subi swap tst wdr",built_in:"r0 r1 r2 r3 r4 r5 r6 r7 r8 r9 r10 r11 r12 r13 r14 r15 r16 r17 r18 r19 r20 r21 r22 "+"r23 r24 r25 r26 r27 r28 r29 r30 r31 x|0 xh xl y|0 yh yl z|0 zh zl "+"ucsr1c udr1 ucsr1a ucsr1b ubrr1l ubrr1h ucsr0c ubrr0h tccr3c tccr3a tccr3b tcnt3h "+"tcnt3l ocr3ah ocr3al ocr3bh ocr3bl ocr3ch ocr3cl icr3h icr3l etimsk etifr tccr1c "+"ocr1ch ocr1cl twcr twdr twar twsr twbr osccal xmcra xmcrb eicra spmcsr spmcr portg "+"ddrg ping portf ddrf sreg sph spl xdiv rampz eicrb eimsk gimsk gicr eifr gifr timsk "+"tifr mcucr mcucsr tccr0 tcnt0 ocr0 assr tccr1a tccr1b tcnt1h tcnt1l ocr1ah ocr1al "+"ocr1bh ocr1bl icr1h icr1l tccr2 tcnt2 ocr2 ocdr wdtcr sfior eearh eearl eedr eecr "+"porta ddra pina portb ddrb pinb portc ddrc pinc portd ddrd pind spdr spsr spcr udr0 "+"ucsr0a ucsr0b ubrr0l acsr admux adcsr adch adcl porte ddre pine pinf",meta:".byte .cseg .db .def .device .dseg .dw .endmacro .equ .eseg .exit .include .list "+".listmac .macro .nolist .org .set"},contains:[hljs.C_BLOCK_COMMENT_MODE,hljs.COMMENT(";","$",{relevance:0}),hljs.C_NUMBER_MODE,hljs.BINARY_NUMBER_MODE,{className:"number",begin:"\\b(\\$[a-zA-Z0-9]+|0o[0-7]+)"},hljs.QUOTE_STRING_MODE,{className:"string",begin:"'",end:"[^\\\\]'",illegal:"[^\\\\][^']"},{className:"symbol",begin:"^[A-Za-z0-9_.$]+:"},{className:"meta",begin:"#",end:"$"},{className:"subst",begin:"@[0-9]+"}]}});</script>
+
+<style type="text/css">
+  code{white-space: pre-wrap;}
+  span.smallcaps{font-variant: small-caps;}
+  span.underline{text-decoration: underline;}
+  div.column{display: inline-block; vertical-align: top; width: 50%;}
+  div.hanging-indent{margin-left: 1.5em; text-indent: -1.5em;}
+  ul.task-list{list-style: none;}
+    </style>
+
+<style type="text/css">code{white-space: pre;}</style>
+<script type="text/javascript">
+if (window.hljs) {
+  hljs.configure({languages: []});
+  hljs.initHighlightingOnLoad();
+  if (document.readyState && document.readyState === "complete") {
+    window.setTimeout(function() { hljs.initHighlighting(); }, 0);
+  }
+}
+</script>
+
+
+
+
+
+
+
+
+
+<style type="text/css">
+.main-container {
+  max-width: 940px;
+  margin-left: auto;
+  margin-right: auto;
+}
+img {
+  max-width:100%;
+}
+.tabbed-pane {
+  padding-top: 12px;
+}
+.html-widget {
+  margin-bottom: 20px;
+}
+button.code-folding-btn:focus {
+  outline: none;
+}
+summary {
+  display: list-item;
+}
+details > summary > p:only-child {
+  display: inline;
+}
+pre code {
+  padding: 0;
+}
+</style>
+
+
+
+<!-- tabsets -->
+
+<style type="text/css">
+.tabset-dropdown > .nav-tabs {
+  display: inline-table;
+  max-height: 500px;
+  min-height: 44px;
+  overflow-y: auto;
+  border: 1px solid #ddd;
+  border-radius: 4px;
+}
+
+.tabset-dropdown > .nav-tabs > li.active:before, .tabset-dropdown > .nav-tabs.nav-tabs-open:before {
+  content: "\e259";
+  font-family: 'Glyphicons Halflings';
+  display: inline-block;
+  padding: 10px;
+  border-right: 1px solid #ddd;
+}
+
+.tabset-dropdown > .nav-tabs.nav-tabs-open > li.active:before {
+  content: "\e258";
+  font-family: 'Glyphicons Halflings';
+  border: none;
+}
+
+.tabset-dropdown > .nav-tabs > li.active {
+  display: block;
+}
+
+.tabset-dropdown > .nav-tabs > li > a,
+.tabset-dropdown > .nav-tabs > li > a:focus,
+.tabset-dropdown > .nav-tabs > li > a:hover {
+  border: none;
+  display: inline-block;
+  border-radius: 4px;
+  background-color: transparent;
+}
+
+.tabset-dropdown > .nav-tabs.nav-tabs-open > li {
+  display: block;
+  float: none;
+}
+
+.tabset-dropdown > .nav-tabs > li {
+  display: none;
+}
+</style>
+
+<!-- code folding -->
+
+
+
+
+</head>
+
+<body>
+
+
+<div class="container-fluid main-container">
+
+
+
+
+<div id="header">
+
+
+
+<h1 class="title toc-ignore">Examples of using survtab</h1>
+<h4 class="author">Joonas Miettinen</h4>
+<h4 class="date">2023-01-19</h4>
+
+</div>
+
+<div id="TOC">
+<ul>
+<li><a href="#overview">Overview</a></li>
+<li><a href="#using-survtab">Using <code>survtab</code></a>
+<ul>
+<li><a href="#relativenet-survival">Relative/net survival</a></li>
+<li><a href="#adjusting-estimates">Adjusting estimates</a></li>
+<li><a href="#other-survival-time-functions">Other survival time
+functions</a></li>
+</ul></li>
+<li><a href="#using-survtab_ag">Using <code>survtab_ag</code></a></li>
+</ul>
+</div>
+
+<div id="overview" class="section level1">
+<h1>Overview</h1>
+<p>This vignette aims to clarify the usage of the
+<code>survtab_ag</code> and <code>survtab</code> functions included in
+this package. <code>survtab_ag</code> estimates various survival
+functions and cumulative incidence functions (CIFs) non-parametrically
+using aggregated data, and <code>survtab</code> is a wrapper for
+<code>survtab_ag</code>, to which <code>Lexis</code> data is
+supplied.</p>
+<p>Two methods (<code>surv.method</code>) are currently supported: The
+<code>&quot;lifetable&quot;</code> (actuarial) method only makes use of counts
+when estimating any of the supported survival time functions. The
+default method (<code>&quot;hazard&quot;</code>}) estimates appropriate hazards
+and transforms them into survival function or CIF estimates.</p>
+<p>For relative survival estimation we need also to enumerate the
+expected hazard levels for the subjects in the data. This is done by
+merging expected hazards to individuals’ subintervals (which divide
+their survival time lines to a number of small intervals). For
+Pohar-Perme-weighted analyses one must additionally compute various
+weighted figures at the level of split subject data.</p>
+<p>If one has subject-level data, the simplest way of computing survival
+function estimates with <code>popEpi</code> is by defining a
+<code>Lexis</code> object and using <code>survtab</code>, which will do
+the rest. For pre-aggregated data one may use the
+<code>survtab_ag</code> function instead. One can also use the
+<code>lexpand</code> function to split, merge population hazards, and
+aggregate in a single function call and then use <code>survtab_ag</code>
+if that is convenient.</p>
+</div>
+<div id="using-survtab" class="section level1">
+<h1>Using <code>survtab</code></h1>
+<p>It is straightforward to estimate various survival time functions
+with <code>survtab</code> once a <code>Lexis</code> object has been
+defined (see <code>?Lexis</code> in package <code>Epi</code> for
+details):</p>
+<pre class="r"><code>library(popEpi)
+library(Epi)</code></pre>
+<pre class="r"><code>data(sire)
+
+## NOTE: recommended to use factor status variable
+x &lt;- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date &lt; sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c(&quot;alive&quot;, &quot;canD&quot;, &quot;othD&quot;)), 
+           merge = TRUE)</code></pre>
+<pre><code>## NOTE: entry.status has been set to &quot;alive&quot; for all.</code></pre>
+<pre class="r"><code>## pretend some are male
+set.seed(1L)
+x$sex &lt;- rbinom(nrow(x), 1, 0.5)
+
+## observed survival - explicit method
+st &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+              surv.type = &quot;surv.obs&quot;,
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## observed survival - easy method (assumes lex.Xst in x is the status variable)
+st &lt;- survtab(FUT ~ sex, data = x, 
+              surv.type = &quot;surv.obs&quot;,
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## printing gives the used settings and 
+## estimates at the middle and end of the estimated
+## curves; more information available using summary()
+st</code></pre>
+<pre><code>## 
+## Call: 
+##  survtab(formula = FUT ~ sex, data = x, breaks = list(FUT = seq(0, 5, 1/12)), surv.type = &quot;surv.obs&quot;) 
+## 
+## Type arguments: 
+##  surv.type: surv.obs --- surv.method: hazard
+##  
+## Confidence interval arguments: 
+##  level: 95 % --- transformation: log-log
+##  
+## Totals:
+##  person-time:23993 --- events: 3636
+##  
+## Stratified by: &#39;sex&#39;
+##    sex Tstop surv.obs.lo surv.obs surv.obs.hi SE.surv.obs
+## 1:   0   2.5      0.6174   0.6328      0.6478    0.007751
+## 2:   0   5.0      0.4962   0.5126      0.5288    0.008321
+## 3:   1   2.5      0.6235   0.6389      0.6539    0.007748
+## 4:   1   5.0      0.5006   0.5171      0.5334    0.008370</code></pre>
+<p>Plotting by strata (men = blue, women = red):</p>
+<pre class="r"><code>plot(st, col = c(&quot;blue&quot;, &quot;red&quot;))</code></pre>
+<p><img src="" width="576" /></p>
+<p>Note that the correct usage of the <code>formula</code> argument in
+<code>survtab</code> specifies the time scale in the <code>Lexis</code>
+object over which survival is computed (here <code>&quot;FUT&quot;</code> for
+follow-up time). This is used to identify the appropriate time scale in
+the data. When only supplying the survival time scale as the
+right-hand-side of the formula, the column <code>lex.Xst</code> in the
+supplied <code>Lexis</code> object is assumed to be the (correctly
+formatted!) status variable. When using <code>Surv()</code> to be
+explicit, we effectively (and exceptionally) pass the starting times to
+the <code>time</code> argument in <code>Surv()</code>, and
+<code>time2</code> is ignored entirely. The function will fail if
+<code>time</code> does not match exactly with a time scale in data.</p>
+<p>When using <code>Surv()</code>, one must also pass the status
+variable, which can be something other than the <code>lex.Xst</code>
+variable created by <code>Lexis()</code>, though usually
+`<code>lex.Xst</code> is what you want to use (especially if the data
+has already been split using e.g. <code>splitLexis</code> or
+<code>splitMulti</code>, which is allowed). It is recommended to use a
+factor status variable to pass to <code>Surv()</code>, though a numeric
+variable will work in simple cases (0 = alive, 1 = dead; also
+<code>FALSE</code> = alive, <code>TRUE</code> = dead). Using
+<code>Surv()</code> also allows easy passing of transformations of
+<code>lex.Xst</code>, e.g. <code>Surv(FUT, lex.Xst %in% 1:2)</code>.</p>
+<p>The argument <code>breaks</code> must be a named list of breaks by
+which to split the <code>Lexis</code> data (see
+<code>?splitMulti</code>). It is mandatory to assign breaks at least to
+the survival time scale (<code>&quot;FUT&quot;</code> in our example) so that
+<code>survtab</code> knows what intervals to use to estimate the
+requested survival time function(s). The breaks also determine the
+window used: It is therefore easy to compute so called period estimates
+by defining the roof and floor along the calendar time scale, e.g. </p>
+<p><code>breaks = list(FUT = seq(0, 5, 1/12), CAL = c(2000, 2005))</code></p>
+<p>would cause <code>survtab</code> to compute period estimates for
+2000-2004 (breaks given here as fractional years, so 2005 is effectively
+2004.99999…).</p>
+<div id="relativenet-survival" class="section level2">
+<h2>Relative/net survival</h2>
+<p>Relative/net survival estimation requires knowledge of the expected
+hazard levels for the individuals in the data. In <code>survtab</code>
+this is accomplished by passing a long-format <code>data.frame</code> of
+population hazards via the <code>pophaz</code> argument. E.g. the
+<code>popmort</code> dataset included in <code>popEpi</code> (Finnish
+overall mortality rates for men and women).</p>
+<pre class="r"><code>data(popmort)
+pm &lt;- data.frame(popmort)
+names(pm) &lt;- c(&quot;sex&quot;, &quot;CAL&quot;, &quot;AGE&quot;, &quot;haz&quot;)
+head(pm)</code></pre>
+<pre><code>##   sex  CAL AGE         haz
+## 1   0 1951   0 0.036363176
+## 2   0 1951   1 0.003616547
+## 3   0 1951   2 0.002172384
+## 4   0 1951   3 0.001581249
+## 5   0 1951   4 0.001180690
+## 6   0 1951   5 0.001070595</code></pre>
+<p>The <code>data.frame</code> should contain a variable named
+<code>&quot;haz&quot;</code> indicating the population hazard at the level of one
+subject-year. Any other variables are considered to be variables, by
+which to merge population hazards to the (split) subject-level data
+within <code>survtab</code>. These merging variables may correspond to
+the time scales in the used <code>Lexis</code> object. This allows for
+e.g. merging in different population hazards for the same subject as
+they get older.</p>
+<p>The following causes <code>survtab</code> to estimate EdererII
+relative survival:</p>
+<pre class="r"><code>st.e2 &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = &quot;surv.rel&quot;, relsurv.method = &quot;e2&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)</code></pre>
+<pre class="r"><code>plot(st.e2, y = &quot;r.e2&quot;, col = c(&quot;blue&quot;, &quot;red&quot;))</code></pre>
+<p><img src="" width="576" /></p>
+<p>Note that the curves diverge due to merging in the “wrong” population
+hazards for some individuals which we randomized earlier to be male
+though all the individuals in data are actually female.
+Pohar-Perme-weighted estimates can be computed by</p>
+<pre class="r"><code>st.pp &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = &quot;surv.rel&quot;, relsurv.method = &quot;pp&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)</code></pre>
+<p>Compare with EdererII estimates:</p>
+<pre class="r"><code>plot(st.e2, y = &quot;r.e2&quot;, col = c(&quot;blue&quot;, &quot;red&quot;), lty = 1)
+lines(st.pp, y = &quot;r.pp&quot;, col = c(&quot;blue&quot;, &quot;red&quot;), lty = 2)</code></pre>
+<p><img src="" width="576" /></p>
+</div>
+<div id="adjusting-estimates" class="section level2">
+<h2>Adjusting estimates</h2>
+<p><code>survtab</code> also allows for adjusting the survival curves by
+categorical variables — typically by age groups. The following
+demonstrates how:</p>
+<pre class="r"><code>## an age group variable
+x$agegr &lt;- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
+
+## using &quot;internal weights&quot; - see ?ICSS for international weights standards
+w &lt;- table(x$agegr)
+w</code></pre>
+<pre><code>## 
+##   [0,60)  [60,70)  [70,80) [80,Inf) 
+##     1781     1889     2428     2129</code></pre>
+<pre class="r"><code>w &lt;- list(agegr = as.numeric(w))</code></pre>
+<pre class="r"><code>st.as &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
+                 data = x, weights = w,
+                 surv.type = &quot;surv.rel&quot;, relsurv.method = &quot;e2&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)</code></pre>
+<pre class="r"><code>plot(st.as, y = &quot;r.e2.as&quot;, col = c(&quot;blue&quot;, &quot;red&quot;))</code></pre>
+<p><img src="" width="576" /></p>
+<p>We now have age-adjusted EdererII relative/net survival estimates.
+The <code>weights</code> argument allows for either a list of weights
+(with one or multiple variables to adjust by) or a
+<code>data.frame</code> of weights. Examples:</p>
+<pre class="r"><code>list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))</code></pre>
+<pre><code>## $sex
+## [1] 0.4 0.6
+## 
+## $agegr
+## [1] 0.2 0.2 0.4 0.2</code></pre>
+<pre class="r"><code>wdf &lt;- merge(0:1, 1:4)
+names(wdf) &lt;- c(&quot;sex&quot;, &quot;agegr&quot;)
+wdf$weights &lt;- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
+wdf</code></pre>
+<pre><code>##   sex agegr weights
+## 1   0     1     0.1
+## 2   1     1     0.1
+## 3   0     2     0.1
+## 4   1     2     0.1
+## 5   0     3     0.2
+## 6   1     3     0.2
+## 7   0     4     0.1
+## 8   1     4     0.1</code></pre>
+<p>The weights do not have to sum to one when supplied as they are
+internally forced to do so within each stratum. In the
+<code>data.frame</code> of weights, the column of actual weights to use
+must be named “weights”. When there are more than one variable to adjust
+by, and a list of weights has been supplied, the variable-specific
+weights are first multiplied together (cumulatively) and then scaled to
+sum to one.</p>
+<p>This adjusting can be done to any survival time function that
+<code>survtab</code> (and <code>survtab_ag</code>) estimates. One can
+also supply adjusting variables via the <code>adjust</code> argument if
+convenient:</p>
+<pre class="r"><code>st.as &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
+                 adjust = &quot;agegr&quot;,
+                 data = x, weights = w,
+                 surv.type = &quot;surv.rel&quot;, relsurv.method = &quot;e2&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)</code></pre>
+<p>Where <code>adjust</code> could also be <code>adjust = agegr</code>,
+<code>adjust = list(agegr)</code> or</p>
+<p><code>adjust = list(agegr = cut(dg_age, c(0, 60, 70, 80, Inf), right = FALSE))</code></p>
+<p>for exactly the same results. When adjusting by multiple variables,
+one must supply a vector of variable names in data or a list of multiple
+elements (as in the base function <code>aggregate</code>).</p>
+</div>
+<div id="other-survival-time-functions" class="section level2">
+<h2>Other survival time functions</h2>
+<p>One can also estimate cause-specific survival functions, cumulative
+incidence functions (CIFs, a.k.a. crude risk a.k.a. absolute risk
+functions), and CIFs based on the excess numbers of events.
+Cause-specific survival is close to net survival as they are
+philosophically highly similar concepts:</p>
+<pre class="r"><code>st.ca &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                 data = x, 
+                 surv.type = &quot;surv.cause&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)))
+
+st.pp &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
+                 surv.type = &quot;surv.rel&quot;, relsurv.method = &quot;pp&quot;,
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+plot(st.ca, y = &quot;surv.obs.canD&quot;, col = &quot;blue&quot;)
+lines(st.pp, y = &quot;r.pp&quot;, col = &quot;red&quot;)</code></pre>
+<p><img src="" width="576" /></p>
+<p>Absolute risk:</p>
+<pre class="r"><code>st.cif &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = &quot;cif.obs&quot;,
+                  breaks = list(FUT = seq(0, 5, 1/12)))
+
+plot(st.cif, y = &quot;CIF_canD&quot;, conf.int = FALSE)
+lines(st.cif, y = &quot;CIF_othD&quot;, conf.int = FALSE, col = &quot;red&quot;)</code></pre>
+<p><img src="" width="576" /></p>
+<p>The “relative CIF” attempts to be close to the true CIF without using
+knowledge about the types of events, e.g. causes of death:</p>
+<pre class="r"><code>st.cir &lt;- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = &quot;cif.rel&quot;,
+                  breaks = list(FUT = seq(0, 5, 1/12)),
+                  pophaz = pm)
+plot(st.cif, y = &quot;CIF_canD&quot;, conf.int = FALSE, col = &quot;blue&quot;)
+lines(st.cir, y = &quot;CIF.rel&quot;, conf.int = FALSE, col = &quot;red&quot;)</code></pre>
+<p><img src="" width="576" /></p>
+</div>
+</div>
+<div id="using-survtab_ag" class="section level1">
+<h1>Using <code>survtab_ag</code></h1>
+<p>Arguments concerning the types and methods of estimating of survival
+time functions work the same in <code>survtab_ag</code> as in
+<code>survtab</code> (the latter uses the former). However, with
+aggregated data one must explicitly supply the various count and
+person-time variables. Also, usage of the <code>formula</code> argument
+is different.</p>
+<p>For demonstration purposes we form an aggregated data set using
+<code>lexpand</code>; see <code>?lexpand</code> for more information on
+that function.</p>
+<pre class="r"><code>sire$sex &lt;- rbinom(nrow(sire), size = 1, prob = 0.5)
+ag &lt;- lexpand(sire, birth = &quot;bi_date&quot;, entry = &quot;dg_date&quot;, exit = &quot;ex_date&quot;,
+              status = &quot;status&quot;, breaks = list(fot = seq(0, 5, 1/12)), 
+              aggre = list(sex, fot))</code></pre>
+<pre><code>## dropped 16 rows where entry == exit</code></pre>
+<pre class="r"><code>head(ag)</code></pre>
+<pre><code>##    sex        fot     pyrs at.risk from0to0 from0to1 from0to2
+## 1:   0 0.00000000 336.9447    4126       12      143       12
+## 2:   0 0.08333333 323.8542    3959       19      101       17
+## 3:   0 0.16666667 314.1913    3822       10       86       15
+## 4:   0 0.25000000 305.0373    3711       15       77       13
+## 5:   0 0.33333333 296.7300    3606       16       72       13
+## 6:   0 0.41666667 289.3323    3505       13       50       14</code></pre>
+<p>Now simply do:</p>
+<pre class="r"><code>st &lt;- survtab_ag(fot ~ sex, data = ag, surv.type = &quot;surv.obs&quot;,
+                 surv.method = &quot;hazard&quot;,
+                 d = c(&quot;from0to1&quot;, &quot;from0to2&quot;), pyrs = &quot;pyrs&quot;)</code></pre>
+<p>Or:</p>
+<pre class="r"><code>st &lt;- survtab_ag(fot ~ sex, data = ag, surv.type = &quot;surv.obs&quot;,
+                 surv.method = &quot;lifetable&quot;,
+                 d = c(&quot;from0to1&quot;, &quot;from0to2&quot;), n = &quot;at.risk&quot;,
+                 n.cens = &quot;from0to0&quot;)</code></pre>
+<p>Note that e.g. argument <code>d</code> could also have been supplied
+as</p>
+<p><code>list(from0to1, from0to2)</code></p>
+<p>or</p>
+<p><code>list(canD = from0to1, othD = from0to2)</code></p>
+<p>for identical results. The last is convenient for
+e.g. <code>surv.cause</code> computations:</p>
+<pre class="r"><code>st.ca &lt;- survtab_ag(fot ~ sex, data = ag, surv.type = &quot;surv.cause&quot;,
+                    surv.method = &quot;hazard&quot;,
+                    d = list(canD = from0to1, othD = from0to2), pyrs = &quot;pyrs&quot;)
+plot(st.ca, y = &quot;surv.obs.canD&quot;, col = c(&quot;blue&quot;, &quot;red&quot;))</code></pre>
+<p><img src="" width="576" /></p>
+<p>One has to supply the most variables when computing Pohar-Perme
+estimates (though it is probably rare to have third-source aggregated
+data with Pohar-Perme weighted figures, it is implemented here to be
+used as a workhorse for <code>survtab</code>). For this we must
+aggregate again to get the Pohar-Perme weighted counts and
+subject-times:</p>
+<pre class="r"><code>ag &lt;- lexpand(sire, birth = &quot;bi_date&quot;, entry = &quot;dg_date&quot;, exit = &quot;ex_date&quot;,
+              status = &quot;status&quot;, breaks = list(fot = seq(0, 5, 1/12)), 
+              pophaz = popmort, pp = TRUE,
+              aggre = list(sex, fot))</code></pre>
+<pre><code>## dropped 16 rows where entry == exit</code></pre>
+<pre><code>## NOTE: 83 rows in split data had values of &#39;age&#39; higher than max of pophaz&#39;s &#39;agegroup&#39;; the hazard values at &#39;agegroup&#39; == 100 were used for these</code></pre>
+<pre class="r"><code>st.pp &lt;- survtab_ag(fot ~ sex, data = ag, surv.type = &quot;surv.rel&quot;,
+                    surv.method = &quot;hazard&quot;, relsurv.method = &quot;pp&quot;,
+                    d = list(from0to1 + from0to2), pyrs = &quot;pyrs&quot;,
+                    d.pp = list(from0to1.pp + from0to2.pp),
+                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
+                    pyrs.pp = &quot;ptime.pp&quot;, d.exp.pp = &quot;d.exp.pp&quot;)
+plot(st.pp, y = &quot;r.pp&quot;, col = c(&quot;blue&quot;, &quot;red&quot;))</code></pre>
+<p><img src="" width="576" /></p>
+<p>Here it is best to supply only one column to each argument since
+Pohar-Perme estimates will not be computed for several types of events
+at the same time.</p>
+</div>
+
+
+
+
+</div>
+
+<script>
+
+// add bootstrap table styles to pandoc tables
+function bootstrapStylePandocTables() {
+  $('tr.odd').parent('tbody').parent('table').addClass('table table-condensed');
+}
+$(document).ready(function () {
+  bootstrapStylePandocTables();
+});
+
+
+</script>
+
+<!-- tabsets -->
+
+<script>
+$(document).ready(function () {
+  window.buildTabsets("TOC");
+});
+
+$(document).ready(function () {
+  $('.tabset-dropdown > .nav-tabs > li').click(function () {
+    $(this).parent().toggleClass('nav-tabs-open');
+  });
+});
+</script>
+
+<!-- code folding -->
+
+
+<!-- dynamically load mathjax for compatibility with self-contained -->
+<script>
+  (function () {
+    var script = document.createElement("script");
+    script.type = "text/javascript";
+    script.src  = "https://mathjax.rstudio.com/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML";
+    document.getElementsByTagName("head")[0].appendChild(script);
+  })();
+</script>
+
+</body>
+</html>
diff --git a/man/ICSS.Rd b/man/ICSS.Rd
index 1e57ad7..7230d21 100644
--- a/man/ICSS.Rd
+++ b/man/ICSS.Rd
@@ -1,45 +1,45 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{ICSS}
-\alias{ICSS}
-\title{Age standardisation weights from the ICSS scheme.}
-\format{
-data.table with columns
-\itemize{
- \item age - lower bound of the age group
- \item ICSS1 - first set of weights, sums to 100 000
- \item ICSS2 - second set of weights, sums to 100 000
- \item ICSS3 - third set of weights, sums to 100 000
-}
-}
-\source{
-\href{https://seer.cancer.gov/stdpopulations/survival.html}{ICSS weights (US National Cancer Institute website)}
-
-Corazziari, Isabella, Mike Quinn, and Riccardo Capocaccia. "Standard cancer patient population for age standardising survival ratios." European Journal of Cancer 40.15 (2004): 2307-2316.
-}
-\description{
-Contains three sets age-standardisation weights for age-standardized survival (net, relative or observed).
-}
-\examples{
-## aggregate weights to a subset of age groups
-data(ICSS)
-cut <- c(0, 30, 50, 70, Inf)
-agegr <- cut(ICSS$age, cut, right = FALSE)
-aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-}
-\seealso{
-Other popEpi data: 
-\code{\link{meanpop_fi}},
-\code{\link{popmort}},
-\code{\link{sibr}},
-\code{\link{sire}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-
-Other weights: 
-\code{\link{direct_standardization}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-}
-\concept{popEpi data}
-\concept{weights}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{ICSS}
+\alias{ICSS}
+\title{Age standardisation weights from the ICSS scheme.}
+\format{
+data.table with columns
+\itemize{
+ \item age - lower bound of the age group
+ \item ICSS1 - first set of weights, sums to 100 000
+ \item ICSS2 - second set of weights, sums to 100 000
+ \item ICSS3 - third set of weights, sums to 100 000
+}
+}
+\source{
+\href{https://seer.cancer.gov/stdpopulations/survival.html}{ICSS weights (US National Cancer Institute website)}
+
+Corazziari, Isabella, Mike Quinn, and Riccardo Capocaccia. "Standard cancer patient population for age standardising survival ratios." European Journal of Cancer 40.15 (2004): 2307-2316.
+}
+\description{
+Contains three sets age-standardisation weights for age-standardized survival (net, relative or observed).
+}
+\examples{
+## aggregate weights to a subset of age groups
+data(ICSS)
+cut <- c(0, 30, 50, 70, Inf)
+agegr <- cut(ICSS$age, cut, right = FALSE)
+aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+}
+\seealso{
+Other popEpi data: 
+\code{\link{meanpop_fi}},
+\code{\link{popmort}},
+\code{\link{sibr}},
+\code{\link{sire}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+
+Other weights: 
+\code{\link{direct_standardization}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+}
+\concept{popEpi data}
+\concept{weights}
diff --git a/man/Lexis_fpa.Rd b/man/Lexis_fpa.Rd
index 796cf88..0a46365 100644
--- a/man/Lexis_fpa.Rd
+++ b/man/Lexis_fpa.Rd
@@ -1,90 +1,90 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{Lexis_fpa}
-\alias{Lexis_fpa}
-\title{Create a Lexis Object with Follow-up Time, Period, and Age
-Time Scales}
-\usage{
-Lexis_fpa(
-  data,
-  birth = NULL,
-  entry = NULL,
-  exit = NULL,
-  entry.status = NULL,
-  exit.status = NULL,
-  subset = NULL,
-  ...
-)
-}
-\arguments{
-\item{data}{a \code{data.frame}; mandatory}
-
-\item{birth}{the time of birth; A character string naming the variable in 
-data or an expression to evaluate - see 
-\link[=flexible_argument]{Flexible input}}
-
-\item{entry}{the time at entry to follow-up; supplied the 
-same way as \code{birth}}
-
-\item{exit}{the time at exit from follow-up; supplied the 
-same way as \code{birth}}
-
-\item{entry.status}{passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
-supplied the same way as \code{birth}}
-
-\item{exit.status}{passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
-supplied the same way as \code{birth}}
-
-\item{subset}{a logical condition to subset by before passing data
-and arguments to \code{\link[Epi]{Lexis}}}
-
-\item{...}{additional optional arguments passed on to 
-\code{\link[Epi]{Lexis}}}
-}
-\value{
-A \code{Lexis} object with the usual columns that \code{Lexis} objects
-have, with time scale columns \code{fot}, \code{per}, and \code{age}.
-They are calculated as
-
-\code{fot = entry - entry} (to ensure correct format, e.g. difftime)
-
-\code{per = entry}
-
-and 
-
-\code{age = entry - birth}
-}
-\description{
-This is a simple wrapper around \code{\link[Epi]{Lexis}} for creating
-a \code{Lexis} object with the time scales \code{fot}, \code{per},
-and \code{age}.
-}
-\examples{
-
-data("sire", package = "popEpi")
-
-lex <- Lexis_fpa(sire, 
-                 birth = "bi_date", 
-                 entry = dg_date, 
-                 exit = ex_date + 1L,
-                 exit.status = "status")
-
-## some special cases
-myVar <- "bi_date"
-l <- list(myVar = "bi_date")
-sire$l <- sire$myVar <- 1
-
-## conflict: myVar taken from data when "bi_date" was intended
-lex <- Lexis_fpa(sire, 
-                 birth = myVar, 
-                 entry = dg_date, 
-                 exit = ex_date + 1L,
-                 exit.status = "status")
-
-## no conflict with names in data
-lex <- Lexis_fpa(sire, 
-                 birth = l$myVar, 
-                 entry = dg_date, 
-                 exit = ex_date + 1L,
-                 exit.status = "status")
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{Lexis_fpa}
+\alias{Lexis_fpa}
+\title{Create a Lexis Object with Follow-up Time, Period, and Age
+Time Scales}
+\usage{
+Lexis_fpa(
+  data,
+  birth = NULL,
+  entry = NULL,
+  exit = NULL,
+  entry.status = NULL,
+  exit.status = NULL,
+  subset = NULL,
+  ...
+)
+}
+\arguments{
+\item{data}{a \code{data.frame}; mandatory}
+
+\item{birth}{the time of birth; A character string naming the variable in 
+data or an expression to evaluate - see 
+\link[=flexible_argument]{Flexible input}}
+
+\item{entry}{the time at entry to follow-up; supplied the 
+same way as \code{birth}}
+
+\item{exit}{the time at exit from follow-up; supplied the 
+same way as \code{birth}}
+
+\item{entry.status}{passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
+supplied the same way as \code{birth}}
+
+\item{exit.status}{passed on to \code{\link[Epi]{Lexis}} if not \code{NULL};
+supplied the same way as \code{birth}}
+
+\item{subset}{a logical condition to subset by before passing data
+and arguments to \code{\link[Epi]{Lexis}}}
+
+\item{...}{additional optional arguments passed on to 
+\code{\link[Epi]{Lexis}}}
+}
+\value{
+A \code{Lexis} object with the usual columns that \code{Lexis} objects
+have, with time scale columns \code{fot}, \code{per}, and \code{age}.
+They are calculated as
+
+\code{fot = entry - entry} (to ensure correct format, e.g. difftime)
+
+\code{per = entry}
+
+and 
+
+\code{age = entry - birth}
+}
+\description{
+This is a simple wrapper around \code{\link[Epi]{Lexis}} for creating
+a \code{Lexis} object with the time scales \code{fot}, \code{per},
+and \code{age}.
+}
+\examples{
+
+data("sire", package = "popEpi")
+
+lex <- Lexis_fpa(sire, 
+                 birth = "bi_date", 
+                 entry = dg_date, 
+                 exit = ex_date + 1L,
+                 exit.status = "status")
+
+## some special cases
+myVar <- "bi_date"
+l <- list(myVar = "bi_date")
+sire$l <- sire$myVar <- 1
+
+## conflict: myVar taken from data when "bi_date" was intended
+lex <- Lexis_fpa(sire, 
+                 birth = myVar, 
+                 entry = dg_date, 
+                 exit = ex_date + 1L,
+                 exit.status = "status")
+
+## no conflict with names in data
+lex <- Lexis_fpa(sire, 
+                 birth = l$myVar, 
+                 entry = dg_date, 
+                 exit = ex_date + 1L,
+                 exit.status = "status")
+}
diff --git a/man/RPL.Rd b/man/RPL.Rd
index 0dd7317..20c0bbf 100644
--- a/man/RPL.Rd
+++ b/man/RPL.Rd
@@ -1,26 +1,26 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/relative_poisson_net_survival.R
-\docType{data}
-\name{RPL}
-\alias{RPL}
-\title{Relative Poisson family object}
-\format{
-A list very similar to that created by \code{poisson()}.
-}
-\usage{
-RPL
-}
-\description{
-A family object for GLM fitting of relative Poisson models
-}
-\seealso{
-Other relpois functions: 
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{rpcurve}()}
-}
-\author{
-Karri Seppa
-}
-\concept{relpois functions}
-\keyword{datasets}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/relative_poisson_net_survival.R
+\docType{data}
+\name{RPL}
+\alias{RPL}
+\title{Relative Poisson family object}
+\format{
+A list very similar to that created by \code{poisson()}.
+}
+\usage{
+RPL
+}
+\description{
+A family object for GLM fitting of relative Poisson models
+}
+\seealso{
+Other relpois functions: 
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{rpcurve}()}
+}
+\author{
+Karri Seppa
+}
+\concept{relpois functions}
+\keyword{datasets}
diff --git a/man/Surv.Rd b/man/Surv.Rd
index 7d5257c..49b5195 100644
--- a/man/Surv.Rd
+++ b/man/Surv.Rd
@@ -1,72 +1,72 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/Surv.R
-\name{Surv}
-\alias{Surv}
-\title{Survival Objects}
-\usage{
-Surv(
-  time,
-  time2,
-  event,
-  type = c("right", "left", "interval", "counting", "interval2", "mstate"),
-  origin = 0
-)
-}
-\arguments{
-\item{time}{see  \link[survival:Surv]{survival::Surv}}
-
-\item{time2}{see  \link[survival:Surv]{survival::Surv}}
-
-\item{event}{see  \link[survival:Surv]{survival::Surv}}
-
-\item{type}{see  \link[survival:Surv]{survival::Surv}}
-
-\item{origin}{see \link[survival:Surv]{survival::Surv}}
-}
-\value{
-See \verb{[survival::Surv]}.
-}
-\description{
-Wrapper for \link[survival:Surv]{survival::Surv}.
-}
-\section{Surv in survival vs. in popEpi}{
-
-\code{popEpi::Surv} is a wrapper for \link[survival:Surv]{survival::Surv}.
-Therefore you don't need to to do \code{library("survival")} when using \code{Surv}
-with e.g.
-\code{\link{survtab}}. Remember that if you do \code{library("survival")} after
-\code{library("popEpi")}, the \code{Surv} from \pkg{survival} is used instead of
-from \pkg{popEpi} (\code{R} throws a warning about this) when an expression
-such as \code{Surv(my_times, my_events)} is evaluated. You can avoid such
-conflicts by writing e.g. \code{popEpi::Surv(my_times, my_events)} instead.
-However, \code{popEpi::Surv} is designed in such a way that this should not
-become a problem and you should be able to use the two interchangeably.
-}
-
-\seealso{
-Other main functions: 
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-
-Other survtab functions: 
-\code{\link{lines.survtab}()},
-\code{\link{plot.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-
-Other survmean functions: 
-\code{\link{lines.survmean}()},
-\code{\link{plot.survmean}()},
-\code{\link{survmean}()}
-}
-\concept{main functions}
-\concept{survmean functions}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/Surv.R
+\name{Surv}
+\alias{Surv}
+\title{Survival Objects}
+\usage{
+Surv(
+  time,
+  time2,
+  event,
+  type = c("right", "left", "interval", "counting", "interval2", "mstate"),
+  origin = 0
+)
+}
+\arguments{
+\item{time}{see  \link[survival:Surv]{survival::Surv}}
+
+\item{time2}{see  \link[survival:Surv]{survival::Surv}}
+
+\item{event}{see  \link[survival:Surv]{survival::Surv}}
+
+\item{type}{see  \link[survival:Surv]{survival::Surv}}
+
+\item{origin}{see \link[survival:Surv]{survival::Surv}}
+}
+\value{
+See \verb{[survival::Surv]}.
+}
+\description{
+Wrapper for \link[survival:Surv]{survival::Surv}.
+}
+\section{Surv in survival vs. in popEpi}{
+
+\code{popEpi::Surv} is a wrapper for \link[survival:Surv]{survival::Surv}.
+Therefore you don't need to to do \code{library("survival")} when using \code{Surv}
+with e.g.
+\code{\link{survtab}}. Remember that if you do \code{library("survival")} after
+\code{library("popEpi")}, the \code{Surv} from \pkg{survival} is used instead of
+from \pkg{popEpi} (\code{R} throws a warning about this) when an expression
+such as \code{Surv(my_times, my_events)} is evaluated. You can avoid such
+conflicts by writing e.g. \code{popEpi::Surv(my_times, my_events)} instead.
+However, \code{popEpi::Surv} is designed in such a way that this should not
+become a problem and you should be able to use the two interchangeably.
+}
+
+\seealso{
+Other main functions: 
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+
+Other survtab functions: 
+\code{\link{lines.survtab}()},
+\code{\link{plot.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+
+Other survmean functions: 
+\code{\link{lines.survmean}()},
+\code{\link{plot.survmean}()},
+\code{\link{survmean}()}
+}
+\concept{main functions}
+\concept{survmean functions}
+\concept{survtab functions}
diff --git a/man/adjust.Rd b/man/adjust.Rd
index 8ff8267..36ff61f 100644
--- a/man/adjust.Rd
+++ b/man/adjust.Rd
@@ -1,24 +1,24 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/evaluation.R
-\name{adjust}
-\alias{adjust}
-\title{Adjust Estimates by Categorical Variables}
-\usage{
-adjust(...)
-}
-\arguments{
-\item{...}{variables to adjust by, e.g. \code{adjust(factor(v1), v2, v3)}}
-}
-\value{
-Returns a list of promises of the variables supplied which can be
-evaluated.
-}
-\description{
-This function is only intended to be used within a formula
-when supplied to e.g. \code{\link{survtab_ag}} and should not be
-used elsewhere.
-}
-\examples{
-
-y ~ x + adjust(z)
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/evaluation.R
+\name{adjust}
+\alias{adjust}
+\title{Adjust Estimates by Categorical Variables}
+\usage{
+adjust(...)
+}
+\arguments{
+\item{...}{variables to adjust by, e.g. \code{adjust(factor(v1), v2, v3)}}
+}
+\value{
+Returns a list of promises of the variables supplied which can be
+evaluated.
+}
+\description{
+This function is only intended to be used within a formula
+when supplied to e.g. \code{\link{survtab_ag}} and should not be
+used elsewhere.
+}
+\examples{
+
+y ~ x + adjust(z)
+}
diff --git a/man/aggre.Rd b/man/aggre.Rd
index 50b7760..9d3c935 100644
--- a/man/aggre.Rd
+++ b/man/aggre.Rd
@@ -1,194 +1,194 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/aggregating.R
-\name{aggre}
-\alias{aggre}
-\title{Aggregation of split \code{Lexis} data}
-\usage{
-aggre(
-  lex,
-  by = NULL,
-  type = c("unique", "full"),
-  sum.values = NULL,
-  subset = NULL,
-  verbose = FALSE
-)
-}
-\arguments{
-\item{lex}{a \code{Lexis} object split with e.g. 
-\code{\link[Epi]{splitLexis}} or \code{\link{splitMulti}}}
-
-\item{by}{variables to tabulate (aggregate) by.
-\link[=flexible_argument]{Flexible input}, typically e.g.
-\code{by = c("V1", "V2")}. See Details and Examples.}
-
-\item{type}{determines output levels to which data is aggregated varying
-from returning only rows with \code{pyrs > 0} (\code{"unique"}) to
-returning all possible combinations of variables given in \code{aggre} even
-if those combinations are not represented in data (\code{"full"}); 
-see Details}
-
-\item{sum.values}{optional: additional variables to sum by argument
- \code{by}. \link[=flexible_argument]{Flexible input}, typically e.g.
-\code{sum.values = c("V1", "V2")}}
-
-\item{subset}{a logical condition to subset by before computations;
-e.g. \code{subset = area \%in\% c("A", "B")}}
-
-\item{verbose}{\code{logical}; if \code{TRUE}, the function returns timings
-and some information useful for debugging along the aggregation process}
-}
-\value{
-A long \code{data.frame} or \code{data.table} of aggregated person-years 
-(\code{pyrs}), numbers of subjects at risk (\code{at.risk}), and events
-formatted \code{fromXtoY}, where \code{X} and \code{X} are states 
-transitioning from and to or states at the end of each \code{lex.id}'s 
-follow-up (implying \code{X} = \code{Y}). Subjects at risk are computed 
-in the beginning of an interval defined by any Lexis time scales and 
-mentioned in \code{by}, but events occur at any point within an interval.
-
-When the data has been split along multiple time scales, the last
-time scale mentioned in \code{by} is considered to be the survival time 
-scale with regard to computing events. Time lines cut short by the
-extrema of non-survival-time-scales are considered to be censored
-("transitions" from the current state to the current state).
-}
-\description{
-Aggregates a split \code{Lexis} object by given variables 
-and / or expressions into a long-format table of person-years and 
-transitions / end-points. Automatic aggregation over time scales
-by which data has been split if the respective time scales are mentioned
-in the aggregation argument to e.g. intervals of calendar time, follow-up time
-and/or age.
-}
-\details{
-\strong{Basics}
-
-\code{aggre} is intended for aggregation of split \code{Lexis} data only.
-See \code{\link[Epi]{Lexis}} for forming \code{Lexis} objects by hand
-and e.g. \code{\link[Epi]{splitLexis}}, \code{\link{splitLexisDT}}, and
-\code{\link{splitMulti}} for splitting the data. \code{\link{lexpand}}
-may be used for simple data sets to do both steps as well as aggregation
-in the same function call.
-
-Here aggregation refers to computing person-years and the appropriate events
-(state transitions and end points in status) for the subjects in the data.
-Hence, it computes e.g. deaths (end-point and state transition) and 
-censorings (end-point) as well as events in a multi-state setting
-(state transitions).
-
-The result is a long-format \code{data.frame} or \code{data.table}
-(depending on \code{options("popEpi.datatable")}; see \code{?popEpi})
-with the columns \code{pyrs} and the appropriate transitions named as
-\code{fromXtoY}, e.g. \code{from0to0} and \code{from0to1} depending
-on the values of \code{lex.Cst} and \code{lex.Xst}.
-
-
-\strong{The by argument}
-
-The \code{by} argument determines the length of the table, i.e.
-the combinations of variables to which data is aggregated.  
-\code{by} is relatively flexible, as it can be supplied as
-
-\itemize{
- \item{a character string vector, e.g. \code{c("sex", "area")}, 
- naming variables existing in \code{lex}}
- \item{an expression, e.g. \code{factor(sex, 0:1, c("m", "f"))} 
- using any variable found in \code{lex}}
- \item{a list (fully or partially named) of expressions, e.g. 
- \code{list(gender = factor(sex, 0:1, c("m", "f"), area)}}
-}
-
-Note that expressions effectively allow a variable to be supplied simply as
-e.g. \code{by = sex} (as a symbol/name in R lingo).
-
-The data is then aggregated to the levels of the given variables 
-or expression(s). Variables defined to be time scales in the supplied 
-\code{Lexis} are processed in a special way: If any are mentioned in the
-\code{by} argument, intervals of them are formed based on the breaks
-used to split the data: e.g. if \code{age} was split using the breaks 
-\code{c(0, 50, Inf)}, mentioning \code{age} in \code{by} leads to
-creating the \code{age} intervals \code{[0, 50)} and \code{[50, Inf)}
-and aggregating to them. The intervals are identified in the output
-as the lower bounds of the appropriate intervals.
-
-The order of multiple time scales mentioned in \code{by} matters,
-as the last mentioned time scale is assumed to be a survival time scale
-for when computing event counts. E.g. when the data is split by the breaks
-\code{list(FUT = 0:5, CAL = c(2008,2010))}, time lines cut short at
-\code{CAL = 2010} are considered to be censored, but time lines cut short at
-\code{FUT = 5} are not. See Return.
-
-\strong{Aggregation types (styles)}
-
-It is almost always enough to aggregate the data to variable levels
-that are actually represented in the data 
-(default \code{aggre = "unique"}; alias \code{"non-empty"}). 
-For certain uses it may be useful
-to have also "empty" levels represented (resulting in some rows in output
-with zero person-years and events); in these cases supplying
-\code{aggre = "full"} (alias \code{"cartesian"}) causes \code{aggre}
-to determine the Cartesian product of all the levels of the supplied 
-\code{by} variables or expressions and aggregate to them. As an example
-of a Cartesian product, try
-
-\code{merge(1:2, 1:5)}.
-}
-\examples{
-
-## form a Lexis object
-library(Epi)
-data(sibr)
-x <- sibr[1:10,]
-x[1:5,]$sex <- 0 ## pretend some are male
-x <- Lexis(data = x,
-           entry = list(AGE = dg_age, CAL = get.yrs(dg_date)),
-           exit = list(CAL = get.yrs(ex_date)),
-           entry.status=0, exit.status = status)
-x <- splitMulti(x, breaks = list(CAL = seq(1993, 2013, 5), 
-                                 AGE = seq(0, 100, 50)))
-
-## these produce the same results (with differing ways of determining aggre)
-a1 <- aggre(x, by = list(gender = factor(sex, 0:1, c("m", "f")), 
-             agegroup = AGE, period = CAL))
-
-a2 <- aggre(x, by = c("sex", "AGE", "CAL"))
-
-a3 <- aggre(x, by = list(sex, agegroup = AGE, CAL))
-
-## returning also empty levels
-a4 <- aggre(x, by = c("sex", "AGE", "CAL"), type = "full")
-
-## computing also expected numbers of cases
-x <- lexpand(sibr[1:10,], birth = bi_date, entry = dg_date,
-             exit = ex_date, status = status \%in\% 1:2, 
-             pophaz = popmort, fot = 0:5, age = c(0, 50, 100))
-x$d.exp <- with(x, lex.dur*pop.haz)
-## these produce the same result
-a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = list(d.exp))
-a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = "d.exp")
-a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = d.exp)
-## same result here with custom name
-a5 <- aggre(x, by = c("sex", "age", "fot"), 
-             sum.values = list(expCases = d.exp))
-             
-## computing pohar-perme weighted figures
-x$d.exp.pp <- with(x, lex.dur*pop.haz*pp)
-a6 <- aggre(x, by = c("sex", "age", "fot"), 
-             sum.values = c("d.exp", "d.exp.pp"))
-## or equivalently e.g. sum.values = list(expCases = d.exp, expCases.p = d.exp.pp).
-}
-\seealso{
-\code{\link{aggregate}} for a similar base R solution,
-and \code{\link{ltable}} for a \code{data.table} based aggregator. Neither
-are directly applicable to split \code{Lexis} data.
-
-Other aggregation functions: 
-\code{\link{as.aggre}()},
-\code{\link{lexpand}()},
-\code{\link{setaggre}()},
-\code{\link{summary.aggre}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{aggregation functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/aggregating.R
+\name{aggre}
+\alias{aggre}
+\title{Aggregation of split \code{Lexis} data}
+\usage{
+aggre(
+  lex,
+  by = NULL,
+  type = c("unique", "full"),
+  sum.values = NULL,
+  subset = NULL,
+  verbose = FALSE
+)
+}
+\arguments{
+\item{lex}{a \code{Lexis} object split with e.g. 
+\code{\link[Epi]{splitLexis}} or \code{\link{splitMulti}}}
+
+\item{by}{variables to tabulate (aggregate) by.
+\link[=flexible_argument]{Flexible input}, typically e.g.
+\code{by = c("V1", "V2")}. See Details and Examples.}
+
+\item{type}{determines output levels to which data is aggregated varying
+from returning only rows with \code{pyrs > 0} (\code{"unique"}) to
+returning all possible combinations of variables given in \code{aggre} even
+if those combinations are not represented in data (\code{"full"}); 
+see Details}
+
+\item{sum.values}{optional: additional variables to sum by argument
+ \code{by}. \link[=flexible_argument]{Flexible input}, typically e.g.
+\code{sum.values = c("V1", "V2")}}
+
+\item{subset}{a logical condition to subset by before computations;
+e.g. \code{subset = area \%in\% c("A", "B")}}
+
+\item{verbose}{\code{logical}; if \code{TRUE}, the function returns timings
+and some information useful for debugging along the aggregation process}
+}
+\value{
+A long \code{data.frame} or \code{data.table} of aggregated person-years 
+(\code{pyrs}), numbers of subjects at risk (\code{at.risk}), and events
+formatted \code{fromXtoY}, where \code{X} and \code{X} are states 
+transitioning from and to or states at the end of each \code{lex.id}'s 
+follow-up (implying \code{X} = \code{Y}). Subjects at risk are computed 
+in the beginning of an interval defined by any Lexis time scales and 
+mentioned in \code{by}, but events occur at any point within an interval.
+
+When the data has been split along multiple time scales, the last
+time scale mentioned in \code{by} is considered to be the survival time 
+scale with regard to computing events. Time lines cut short by the
+extrema of non-survival-time-scales are considered to be censored
+("transitions" from the current state to the current state).
+}
+\description{
+Aggregates a split \code{Lexis} object by given variables 
+and / or expressions into a long-format table of person-years and 
+transitions / end-points. Automatic aggregation over time scales
+by which data has been split if the respective time scales are mentioned
+in the aggregation argument to e.g. intervals of calendar time, follow-up time
+and/or age.
+}
+\details{
+\strong{Basics}
+
+\code{aggre} is intended for aggregation of split \code{Lexis} data only.
+See \code{\link[Epi]{Lexis}} for forming \code{Lexis} objects by hand
+and e.g. \code{\link[Epi]{splitLexis}}, \code{\link{splitLexisDT}}, and
+\code{\link{splitMulti}} for splitting the data. \code{\link{lexpand}}
+may be used for simple data sets to do both steps as well as aggregation
+in the same function call.
+
+Here aggregation refers to computing person-years and the appropriate events
+(state transitions and end points in status) for the subjects in the data.
+Hence, it computes e.g. deaths (end-point and state transition) and 
+censorings (end-point) as well as events in a multi-state setting
+(state transitions).
+
+The result is a long-format \code{data.frame} or \code{data.table}
+(depending on \code{options("popEpi.datatable")}; see \code{?popEpi})
+with the columns \code{pyrs} and the appropriate transitions named as
+\code{fromXtoY}, e.g. \code{from0to0} and \code{from0to1} depending
+on the values of \code{lex.Cst} and \code{lex.Xst}.
+
+
+\strong{The by argument}
+
+The \code{by} argument determines the length of the table, i.e.
+the combinations of variables to which data is aggregated.  
+\code{by} is relatively flexible, as it can be supplied as
+
+\itemize{
+ \item{a character string vector, e.g. \code{c("sex", "area")}, 
+ naming variables existing in \code{lex}}
+ \item{an expression, e.g. \code{factor(sex, 0:1, c("m", "f"))} 
+ using any variable found in \code{lex}}
+ \item{a list (fully or partially named) of expressions, e.g. 
+ \code{list(gender = factor(sex, 0:1, c("m", "f"), area)}}
+}
+
+Note that expressions effectively allow a variable to be supplied simply as
+e.g. \code{by = sex} (as a symbol/name in R lingo).
+
+The data is then aggregated to the levels of the given variables 
+or expression(s). Variables defined to be time scales in the supplied 
+\code{Lexis} are processed in a special way: If any are mentioned in the
+\code{by} argument, intervals of them are formed based on the breaks
+used to split the data: e.g. if \code{age} was split using the breaks 
+\code{c(0, 50, Inf)}, mentioning \code{age} in \code{by} leads to
+creating the \code{age} intervals \code{[0, 50)} and \code{[50, Inf)}
+and aggregating to them. The intervals are identified in the output
+as the lower bounds of the appropriate intervals.
+
+The order of multiple time scales mentioned in \code{by} matters,
+as the last mentioned time scale is assumed to be a survival time scale
+for when computing event counts. E.g. when the data is split by the breaks
+\code{list(FUT = 0:5, CAL = c(2008,2010))}, time lines cut short at
+\code{CAL = 2010} are considered to be censored, but time lines cut short at
+\code{FUT = 5} are not. See Return.
+
+\strong{Aggregation types (styles)}
+
+It is almost always enough to aggregate the data to variable levels
+that are actually represented in the data 
+(default \code{aggre = "unique"}; alias \code{"non-empty"}). 
+For certain uses it may be useful
+to have also "empty" levels represented (resulting in some rows in output
+with zero person-years and events); in these cases supplying
+\code{aggre = "full"} (alias \code{"cartesian"}) causes \code{aggre}
+to determine the Cartesian product of all the levels of the supplied 
+\code{by} variables or expressions and aggregate to them. As an example
+of a Cartesian product, try
+
+\code{merge(1:2, 1:5)}.
+}
+\examples{
+
+## form a Lexis object
+library(Epi)
+data(sibr)
+x <- sibr[1:10,]
+x[1:5,]$sex <- 0 ## pretend some are male
+x <- Lexis(data = x,
+           entry = list(AGE = dg_age, CAL = get.yrs(dg_date)),
+           exit = list(CAL = get.yrs(ex_date)),
+           entry.status=0, exit.status = status)
+x <- splitMulti(x, breaks = list(CAL = seq(1993, 2013, 5), 
+                                 AGE = seq(0, 100, 50)))
+
+## these produce the same results (with differing ways of determining aggre)
+a1 <- aggre(x, by = list(gender = factor(sex, 0:1, c("m", "f")), 
+             agegroup = AGE, period = CAL))
+
+a2 <- aggre(x, by = c("sex", "AGE", "CAL"))
+
+a3 <- aggre(x, by = list(sex, agegroup = AGE, CAL))
+
+## returning also empty levels
+a4 <- aggre(x, by = c("sex", "AGE", "CAL"), type = "full")
+
+## computing also expected numbers of cases
+x <- lexpand(sibr[1:10,], birth = bi_date, entry = dg_date,
+             exit = ex_date, status = status \%in\% 1:2, 
+             pophaz = popmort, fot = 0:5, age = c(0, 50, 100))
+x$d.exp <- with(x, lex.dur*pop.haz)
+## these produce the same result
+a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = list(d.exp))
+a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = "d.exp")
+a5 <- aggre(x, by = c("sex", "age", "fot"), sum.values = d.exp)
+## same result here with custom name
+a5 <- aggre(x, by = c("sex", "age", "fot"), 
+             sum.values = list(expCases = d.exp))
+             
+## computing pohar-perme weighted figures
+x$d.exp.pp <- with(x, lex.dur*pop.haz*pp)
+a6 <- aggre(x, by = c("sex", "age", "fot"), 
+             sum.values = c("d.exp", "d.exp.pp"))
+## or equivalently e.g. sum.values = list(expCases = d.exp, expCases.p = d.exp.pp).
+}
+\seealso{
+\code{\link{aggregate}} for a similar base R solution,
+and \code{\link{ltable}} for a \code{data.table} based aggregator. Neither
+are directly applicable to split \code{Lexis} data.
+
+Other aggregation functions: 
+\code{\link{as.aggre}()},
+\code{\link{lexpand}()},
+\code{\link{setaggre}()},
+\code{\link{summary.aggre}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{aggregation functions}
diff --git a/man/all_names_present.Rd b/man/all_names_present.Rd
index 287ab1f..93ffc80 100644
--- a/man/all_names_present.Rd
+++ b/man/all_names_present.Rd
@@ -1,34 +1,34 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{all_names_present}
-\alias{all_names_present}
-\title{Check if all names are present in given data}
-\usage{
-all_names_present(data, var.names, stops = TRUE, msg = NULL)
-}
-\arguments{
-\item{data}{dataset where the variable names should be found}
-
-\item{var.names}{a character vector of variable names, e.g.
-\code{c("var1", "var2")}}
-
-\item{stops}{logical, stop returns exception}
-
-\item{msg}{Custom message to return instead of default message.
-Special: include \code{\%\%VARS\%\%} in message string and the missing 
-variable names will be inserted there (quoted, separated by comma, e.g. 
-\code{'var1'}, \code{'var2'} --- no leading or tracing white space).}
-}
-\value{
-`TRUE` if all `var.names` are in `data`, else `FALSE`,
-}
-\description{
-Given a character vector, checks if all names are present in \code{names(data)}.
-Throws error if \code{stops=TRUE}, else returns \code{FALSE} if some variable name is not present.
-}
-\seealso{
-\code{\link{robust_values}}
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{all_names_present}
+\alias{all_names_present}
+\title{Check if all names are present in given data}
+\usage{
+all_names_present(data, var.names, stops = TRUE, msg = NULL)
+}
+\arguments{
+\item{data}{dataset where the variable names should be found}
+
+\item{var.names}{a character vector of variable names, e.g.
+\code{c("var1", "var2")}}
+
+\item{stops}{logical, stop returns exception}
+
+\item{msg}{Custom message to return instead of default message.
+Special: include \code{\%\%VARS\%\%} in message string and the missing 
+variable names will be inserted there (quoted, separated by comma, e.g. 
+\code{'var1'}, \code{'var2'} --- no leading or tracing white space).}
+}
+\value{
+`TRUE` if all `var.names` are in `data`, else `FALSE`,
+}
+\description{
+Given a character vector, checks if all names are present in \code{names(data)}.
+Throws error if \code{stops=TRUE}, else returns \code{FALSE} if some variable name is not present.
+}
+\seealso{
+\code{\link{robust_values}}
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/array_df_ratetable_utils.Rd b/man/array_df_ratetable_utils.Rd
index 0bff50e..84d1e55 100644
--- a/man/array_df_ratetable_utils.Rd
+++ b/man/array_df_ratetable_utils.Rd
@@ -1,143 +1,143 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/long_df_and_array.R
-\name{array_df_ratetable_utils}
-\alias{array_df_ratetable_utils}
-\alias{long_df_to_array}
-\alias{long_df_to_ratetable}
-\alias{long_dt_to_array}
-\alias{long_dt_to_ratetable}
-\alias{array_to_long_df}
-\alias{array_to_long_dt}
-\alias{array_to_ratetable}
-\alias{ratetable_to_array}
-\alias{ratetable_to_long_df}
-\alias{ratetable_to_long_dt}
-\title{`array`s, `data.frame`s and `ratetable`s}
-\usage{
-long_df_to_array(x, stratum.col.nms, value.col.nm)
-
-long_df_to_ratetable(
-  x,
-  stratum.col.nms,
-  value.col.nm,
-  dim.types,
-  cut.points = NULL
-)
-
-long_dt_to_array(x, stratum.col.nms, value.col.nm)
-
-long_dt_to_ratetable(
-  x,
-  stratum.col.nms,
-  value.col.nm,
-  dim.types,
-  cut.points = NULL
-)
-
-array_to_long_df(x)
-
-array_to_long_dt(x)
-
-array_to_ratetable(x, dim.types, cut.points = NULL)
-
-ratetable_to_array(x)
-
-ratetable_to_long_df(x)
-
-ratetable_to_long_dt(x)
-}
-\arguments{
-\item{x}{`[data.frame, data.table, array, ratetable]` (mandatory, no default)
-
-- `long_df_to_array`: a `data.frame`
-- `long_df_to_ratetable`: a `data.frame`
-- `long_dt_to_array`: a `data.table`
-- `long_dt_to_ratetable`: a `data.table`
-- `array_to_long_df`: an `array`
-- `array_to_long_dt`: an `array`
-- `array_to_ratetable`: an `array`
-- `ratetable_to_array`: a [survival::ratetable]
-- `ratetable_to_long_df`: a [survival::ratetable]
-- `ratetable_to_long_dt`: a [survival::ratetable]}
-
-\item{stratum.col.nms}{`[character]` (mandatory, no default)
-
-a vector of column names in `x` by which values are stratified}
-
-\item{value.col.nm}{`[character]` (mandatory, no default)
-
-name of column in `x` containing values (these will be contents of the
-array)}
-
-\item{dim.types}{`[integer]` (mandatory, no default)
-
-see `type` under **Details** in [survival::ratetable]}
-
-\item{cut.points}{`[NULL, list]` (optional, default `NULL`)
-
-see `cutpoints` under **Details** in [survival::ratetable] 
-
-- `NULL`: automatically set using `dimnames(x)` and `dim.types`
-- `list`: one element for each dimensions of `x`}
-}
-\value{
-- `long_df_to_array`: an `array`
-- `long_df_to_ratetable`: a [survival::ratetable]
-- `long_dt_to_array`: an `array`
-- `long_dt_to_ratetable`: a [survival::ratetable]
-- `array_to_long_df`: an `data.frame`
-- `array_to_long_dt`: an `data.table`
-- `array_to_ratetable`: a [survival::ratetable]
-- `ratetable_to_array`: an `array`
-- `ratetable_to_long_df`: a `data.frame`
-- `ratetable_to_long_dt`: a `data.table`
-}
-\description{
-Utilities to transform objects between `array`, `data.frame`, and
-[survival::ratetable].
-}
-\details{
-- `long_df_to_array`: converts a long-format `data.frame` to an `array`
-  with one or more dimensions
-
-- `long_df_to_ratetable`: calls `long_df_to_array` and then 
-  `array_to_ratetable`
-
-- `long_dt_to_array`: simply asserts that `x` is a `data.table` and 
-calls `long_df_to_array`
-
-- `long_dt_to_ratetable`: calls `long_dt_to_array` and then 
-  `array_to_ratetable`
-
-- `array_to_long_df`: converts an array with one or more dimensions into
-  a long-format `data.frame`; any [dimnames] are used to name and fill the 
-  stratifying columns; for dimensions without a name, `".dX"` is used
-  for stratifying column number `X`; for each `k`, if there are no contents
-  in `dimnames(x)[[k]]`, the elements of `seq(dim(x)[k])` are used to fill 
-  the corresponding stratifying column; the value column always has the name
-  `"value"`
-
-- `array_to_long_dt`: calls `array_to_long_df` and converts result to a 
-  `data.table` for convenience
-
-- `array_to_ratetable`: converts an array to a [survival::ratetable]
-
-- `ratetable_to_array`: converts a [survival::ratetable] to an array
-
-- `ratetable_to_long_df`: calls `ratetable_to_array` and then
-  `array_to_long_df`
-
-- `ratetable_to_long_dt`: calls `ratetable_to_array` and then
-  `array_to_long_dt`
-}
-\examples{
-
-long_dt <- popEpi::popmort
-arr <- long_df_to_array(long_dt, c("agegroup", "year", "sex"), "haz") 
-rt <- array_to_ratetable(arr, dim.types = c(2L, 4L, 1L))
-
-arr2 <- ratetable_to_array(rt)
-long_df2 <- array_to_long_df(arr2)
-
-identical(sort(long_dt[["haz"]]), sort(long_df2[["value"]]))
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/long_df_and_array.R
+\name{array_df_ratetable_utils}
+\alias{array_df_ratetable_utils}
+\alias{long_df_to_array}
+\alias{long_df_to_ratetable}
+\alias{long_dt_to_array}
+\alias{long_dt_to_ratetable}
+\alias{array_to_long_df}
+\alias{array_to_long_dt}
+\alias{array_to_ratetable}
+\alias{ratetable_to_array}
+\alias{ratetable_to_long_df}
+\alias{ratetable_to_long_dt}
+\title{`array`s, `data.frame`s and `ratetable`s}
+\usage{
+long_df_to_array(x, stratum.col.nms, value.col.nm)
+
+long_df_to_ratetable(
+  x,
+  stratum.col.nms,
+  value.col.nm,
+  dim.types,
+  cut.points = NULL
+)
+
+long_dt_to_array(x, stratum.col.nms, value.col.nm)
+
+long_dt_to_ratetable(
+  x,
+  stratum.col.nms,
+  value.col.nm,
+  dim.types,
+  cut.points = NULL
+)
+
+array_to_long_df(x)
+
+array_to_long_dt(x)
+
+array_to_ratetable(x, dim.types, cut.points = NULL)
+
+ratetable_to_array(x)
+
+ratetable_to_long_df(x)
+
+ratetable_to_long_dt(x)
+}
+\arguments{
+\item{x}{`[data.frame, data.table, array, ratetable]` (mandatory, no default)
+
+- `long_df_to_array`: a `data.frame`
+- `long_df_to_ratetable`: a `data.frame`
+- `long_dt_to_array`: a `data.table`
+- `long_dt_to_ratetable`: a `data.table`
+- `array_to_long_df`: an `array`
+- `array_to_long_dt`: an `array`
+- `array_to_ratetable`: an `array`
+- `ratetable_to_array`: a [survival::ratetable]
+- `ratetable_to_long_df`: a [survival::ratetable]
+- `ratetable_to_long_dt`: a [survival::ratetable]}
+
+\item{stratum.col.nms}{`[character]` (mandatory, no default)
+
+a vector of column names in `x` by which values are stratified}
+
+\item{value.col.nm}{`[character]` (mandatory, no default)
+
+name of column in `x` containing values (these will be contents of the
+array)}
+
+\item{dim.types}{`[integer]` (mandatory, no default)
+
+see `type` under **Details** in [survival::ratetable]}
+
+\item{cut.points}{`[NULL, list]` (optional, default `NULL`)
+
+see `cutpoints` under **Details** in [survival::ratetable] 
+
+- `NULL`: automatically set using `dimnames(x)` and `dim.types`
+- `list`: one element for each dimensions of `x`}
+}
+\value{
+- `long_df_to_array`: an `array`
+- `long_df_to_ratetable`: a [survival::ratetable]
+- `long_dt_to_array`: an `array`
+- `long_dt_to_ratetable`: a [survival::ratetable]
+- `array_to_long_df`: an `data.frame`
+- `array_to_long_dt`: an `data.table`
+- `array_to_ratetable`: a [survival::ratetable]
+- `ratetable_to_array`: an `array`
+- `ratetable_to_long_df`: a `data.frame`
+- `ratetable_to_long_dt`: a `data.table`
+}
+\description{
+Utilities to transform objects between `array`, `data.frame`, and
+[survival::ratetable].
+}
+\details{
+- `long_df_to_array`: converts a long-format `data.frame` to an `array`
+  with one or more dimensions
+
+- `long_df_to_ratetable`: calls `long_df_to_array` and then 
+  `array_to_ratetable`
+
+- `long_dt_to_array`: simply asserts that `x` is a `data.table` and 
+calls `long_df_to_array`
+
+- `long_dt_to_ratetable`: calls `long_dt_to_array` and then 
+  `array_to_ratetable`
+
+- `array_to_long_df`: converts an array with one or more dimensions into
+  a long-format `data.frame`; any [dimnames] are used to name and fill the 
+  stratifying columns; for dimensions without a name, `".dX"` is used
+  for stratifying column number `X`; for each `k`, if there are no contents
+  in `dimnames(x)[[k]]`, the elements of `seq(dim(x)[k])` are used to fill 
+  the corresponding stratifying column; the value column always has the name
+  `"value"`
+
+- `array_to_long_dt`: calls `array_to_long_df` and converts result to a 
+  `data.table` for convenience
+
+- `array_to_ratetable`: converts an array to a [survival::ratetable]
+
+- `ratetable_to_array`: converts a [survival::ratetable] to an array
+
+- `ratetable_to_long_df`: calls `ratetable_to_array` and then
+  `array_to_long_df`
+
+- `ratetable_to_long_dt`: calls `ratetable_to_array` and then
+  `array_to_long_dt`
+}
+\examples{
+
+long_dt <- popEpi::popmort
+arr <- long_df_to_array(long_dt, c("agegroup", "year", "sex"), "haz") 
+rt <- array_to_ratetable(arr, dim.types = c(2L, 4L, 1L))
+
+arr2 <- ratetable_to_array(rt)
+long_df2 <- array_to_long_df(arr2)
+
+identical(sort(long_dt[["haz"]]), sort(long_df2[["value"]]))
+}
diff --git a/man/as.Date.yrs.Rd b/man/as.Date.yrs.Rd
index b39c48a..cff6b98 100644
--- a/man/as.Date.yrs.Rd
+++ b/man/as.Date.yrs.Rd
@@ -1,45 +1,45 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/fractional_years.R
-\name{as.Date.yrs}
-\alias{as.Date.yrs}
-\title{Coerce Fractional Year Values to Date Values}
-\usage{
-\method{as.Date}{yrs}(x, ...)
-}
-\arguments{
-\item{x}{an \code{yrs} object created by \code{get.yrs}}
-
-\item{...}{unused, included for compatibility with other \code{as.Date}
-methods}
-}
-\value{
-A vector of `Date` values based on the input fractional years.
-}
-\description{
-Coerces an \code{yrs} object to a \code{Date} object.
-Some loss of information comes if \code{year.length = "approx"} 
-was set when using \code{\link{get.yrs}}, so the transformation back
-to \code{Date} will not be perfect there. With \code{year.length = "actual"}
-the original values are perfectly retrieved.
-}
-\examples{
-data("sire", package = "popEpi")
-
-## approximate year lengths: here 20 \% have an extra day added
-sire$dg_yrs <- get.yrs(sire$dg_date)
-summary(sire$dg_yrs)
-dg_date2 <- as.Date(sire$dg_yrs)
-summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-
-## using actual year lengths
-sire$dg_yrs <- get.yrs(sire$dg_date, year.length = "actual")
-summary(sire$dg_yrs)
-dg_date2 <- as.Date(sire$dg_yrs)
-summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-}
-\seealso{
-\code{\link{get.yrs}}
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/fractional_years.R
+\name{as.Date.yrs}
+\alias{as.Date.yrs}
+\title{Coerce Fractional Year Values to Date Values}
+\usage{
+\method{as.Date}{yrs}(x, ...)
+}
+\arguments{
+\item{x}{an \code{yrs} object created by \code{get.yrs}}
+
+\item{...}{unused, included for compatibility with other \code{as.Date}
+methods}
+}
+\value{
+A vector of `Date` values based on the input fractional years.
+}
+\description{
+Coerces an \code{yrs} object to a \code{Date} object.
+Some loss of information comes if \code{year.length = "approx"} 
+was set when using \code{\link{get.yrs}}, so the transformation back
+to \code{Date} will not be perfect there. With \code{year.length = "actual"}
+the original values are perfectly retrieved.
+}
+\examples{
+data("sire", package = "popEpi")
+
+## approximate year lengths: here 20 \% have an extra day added
+sire$dg_yrs <- get.yrs(sire$dg_date)
+summary(sire$dg_yrs)
+dg_date2 <- as.Date(sire$dg_yrs)
+summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+
+## using actual year lengths
+sire$dg_yrs <- get.yrs(sire$dg_date, year.length = "actual")
+summary(sire$dg_yrs)
+dg_date2 <- as.Date(sire$dg_yrs)
+summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+}
+\seealso{
+\code{\link{get.yrs}}
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/as.aggre.Rd b/man/as.aggre.Rd
index 2e44e6f..dca0930 100644
--- a/man/as.aggre.Rd
+++ b/man/as.aggre.Rd
@@ -1,81 +1,81 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/aggregating.R
-\name{as.aggre}
-\alias{as.aggre}
-\alias{as.aggre.data.frame}
-\alias{as.aggre.data.table}
-\alias{as.aggre.default}
-\title{Coercion to Class \code{aggre}}
-\usage{
-as.aggre(x, values = NULL, by = NULL, breaks = NULL, ...)
-
-\method{as.aggre}{data.frame}(x, values = NULL, by = NULL, breaks = NULL, ...)
-
-\method{as.aggre}{data.table}(x, values = NULL, by = NULL, breaks = NULL, ...)
-
-\method{as.aggre}{default}(x, ...)
-}
-\arguments{
-\item{x}{a \code{data.frame} or \code{data.table}}
-
-\item{values}{a character string vector; the names of value variables}
-
-\item{by}{a character string vector; the names of variables by which 
-\code{values} have been tabulated}
-
-\item{breaks}{a list of breaks, where each element is a breaks vector
-as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
-fully named, with the names corresponding to time scales at the aggregate
-level in your data. Every unique value in a time scale variable in data must
-also exist in the corresponding vector in the breaks list.}
-
-\item{...}{arguments passed to or from methods}
-}
-\value{
-Returns a copy of `x` with attributes set to those of an object of class
-`"aggre"`.
-}
-\description{
-Coerces an R object to an \code{aggre} object, identifying
-the object as one containing aggregated counts, person-years and other
-information.
-}
-\section{Methods (by class)}{
-\itemize{
-\item \code{as.aggre(data.frame)}: Coerces a \code{data.frame} to an \code{aggre} object
-
-\item \code{as.aggre(data.table)}: Coerces a \code{data.table} to an \code{aggre} object
-
-\item \code{as.aggre(default)}: Default method for \code{as.aggre} (stops computations
-if no class-specific method found)
-
-}}
-\examples{
-library("data.table")
-df <- data.frame(sex = rep(c("male", "female"), each = 5), 
-                 obs = rpois(10, rep(7,5, each=5)), 
-                 pyrs = rpois(10, lambda = 10000))
-dt <- as.data.table(df)
-
-df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex")
-dt <- as.aggre(dt, values = c("pyrs", "obs"), by = "sex")
-
-class(df)
-class(dt)
-
-BL <- list(fot = 0:5)
-df <- data.frame(df)
-df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex", breaks = BL)
-
-}
-\seealso{
-Other aggregation functions: 
-\code{\link{aggre}()},
-\code{\link{lexpand}()},
-\code{\link{setaggre}()},
-\code{\link{summary.aggre}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{aggregation functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/aggregating.R
+\name{as.aggre}
+\alias{as.aggre}
+\alias{as.aggre.data.frame}
+\alias{as.aggre.data.table}
+\alias{as.aggre.default}
+\title{Coercion to Class \code{aggre}}
+\usage{
+as.aggre(x, values = NULL, by = NULL, breaks = NULL, ...)
+
+\method{as.aggre}{data.frame}(x, values = NULL, by = NULL, breaks = NULL, ...)
+
+\method{as.aggre}{data.table}(x, values = NULL, by = NULL, breaks = NULL, ...)
+
+\method{as.aggre}{default}(x, ...)
+}
+\arguments{
+\item{x}{a \code{data.frame} or \code{data.table}}
+
+\item{values}{a character string vector; the names of value variables}
+
+\item{by}{a character string vector; the names of variables by which 
+\code{values} have been tabulated}
+
+\item{breaks}{a list of breaks, where each element is a breaks vector
+as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
+fully named, with the names corresponding to time scales at the aggregate
+level in your data. Every unique value in a time scale variable in data must
+also exist in the corresponding vector in the breaks list.}
+
+\item{...}{arguments passed to or from methods}
+}
+\value{
+Returns a copy of `x` with attributes set to those of an object of class
+`"aggre"`.
+}
+\description{
+Coerces an R object to an \code{aggre} object, identifying
+the object as one containing aggregated counts, person-years and other
+information.
+}
+\section{Methods (by class)}{
+\itemize{
+\item \code{as.aggre(data.frame)}: Coerces a \code{data.frame} to an \code{aggre} object
+
+\item \code{as.aggre(data.table)}: Coerces a \code{data.table} to an \code{aggre} object
+
+\item \code{as.aggre(default)}: Default method for \code{as.aggre} (stops computations
+if no class-specific method found)
+
+}}
+\examples{
+library("data.table")
+df <- data.frame(sex = rep(c("male", "female"), each = 5), 
+                 obs = rpois(10, rep(7,5, each=5)), 
+                 pyrs = rpois(10, lambda = 10000))
+dt <- as.data.table(df)
+
+df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex")
+dt <- as.aggre(dt, values = c("pyrs", "obs"), by = "sex")
+
+class(df)
+class(dt)
+
+BL <- list(fot = 0:5)
+df <- data.frame(df)
+df <- as.aggre(df, values = c("pyrs", "obs"), by = "sex", breaks = BL)
+
+}
+\seealso{
+Other aggregation functions: 
+\code{\link{aggre}()},
+\code{\link{lexpand}()},
+\code{\link{setaggre}()},
+\code{\link{summary.aggre}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{aggregation functions}
diff --git a/man/cast_simple.Rd b/man/cast_simple.Rd
index 59bca33..d72d2f3 100644
--- a/man/cast_simple.Rd
+++ b/man/cast_simple.Rd
@@ -1,52 +1,52 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{cast_simple}
-\alias{cast_simple}
-\title{Cast \code{data.table}/\code{data.frame} from long format to wide format}
-\usage{
-cast_simple(data = NULL, columns = NULL, rows = NULL, values = NULL)
-}
-\arguments{
-\item{data}{a \code{data.table} or \code{data.frame}}
-
-\item{columns}{a character string vector; the (unique combinations of the) 
-levels of these variable will be different rows}
-
-\item{rows}{a character string vector; the (unique combinations of the) 
-levels of these variable will be different columns}
-
-\item{values}{a character string; the variable which will be represented
-on rows and columns as specified by \code{columns} and \code{rows}}
-}
-\value{
-A `data.table` just like `[data.table::dcast]`.
-}
-\description{
-Convenience function for using \code{\link[data.table]{dcast.data.table}};
-inputs are character strings (names of variables) instead of a formula.
-}
-\details{
-This function is just a small interface for \code{dcast} / 
-\code{dcast.data.table} and less flexible than the originals.
-
-Note that all \code{data.table} objects are also \code{data.frame} 
-objects, but that each have their own \code{dcast} method.
-\code{\link[data.table]{dcast.data.table}} is faster.
-
-If any values in \code{value.vars} need to be 
-aggregated, they are aggregated using \code{sum}.
-See \code{?dcast}.
-}
-\examples{
-library("data.table")
-## e.g. silly counts from a long-format table to a wide format
-test <- data.table::copy(popEpi::sire)
-test$dg_y <- year(test$dg_date)
-test$ex_y <- year(test$ex_date)
-tab <- ltable(test, c("dg_y","ex_y"))
-cast_simple(tab, columns='dg_y', rows="ex_y", values="obs")
-
-}
-\author{
-Matti Rantanen, Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{cast_simple}
+\alias{cast_simple}
+\title{Cast \code{data.table}/\code{data.frame} from long format to wide format}
+\usage{
+cast_simple(data = NULL, columns = NULL, rows = NULL, values = NULL)
+}
+\arguments{
+\item{data}{a \code{data.table} or \code{data.frame}}
+
+\item{columns}{a character string vector; the (unique combinations of the) 
+levels of these variable will be different rows}
+
+\item{rows}{a character string vector; the (unique combinations of the) 
+levels of these variable will be different columns}
+
+\item{values}{a character string; the variable which will be represented
+on rows and columns as specified by \code{columns} and \code{rows}}
+}
+\value{
+A `data.table` just like `[data.table::dcast]`.
+}
+\description{
+Convenience function for using \code{\link[data.table]{dcast.data.table}};
+inputs are character strings (names of variables) instead of a formula.
+}
+\details{
+This function is just a small interface for \code{dcast} / 
+\code{dcast.data.table} and less flexible than the originals.
+
+Note that all \code{data.table} objects are also \code{data.frame} 
+objects, but that each have their own \code{dcast} method.
+\code{\link[data.table]{dcast.data.table}} is faster.
+
+If any values in \code{value.vars} need to be 
+aggregated, they are aggregated using \code{sum}.
+See \code{?dcast}.
+}
+\examples{
+library("data.table")
+## e.g. silly counts from a long-format table to a wide format
+test <- data.table::copy(popEpi::sire)
+test$dg_y <- year(test$dg_date)
+test$ex_y <- year(test$ex_date)
+tab <- ltable(test, c("dg_y","ex_y"))
+cast_simple(tab, columns='dg_y', rows="ex_y", values="obs")
+
+}
+\author{
+Matti Rantanen, Joonas Miettinen
+}
diff --git a/man/cut_bound.Rd b/man/cut_bound.Rd
index 5c0ae65..762cbd7 100644
--- a/man/cut_bound.Rd
+++ b/man/cut_bound.Rd
@@ -1,32 +1,32 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{cut_bound}
-\alias{cut_bound}
-\title{Change output values from cut(..., labels = NULL) output}
-\usage{
-cut_bound(t, factor = TRUE)
-}
-\arguments{
-\item{t}{is a character vector of elements, e.g. "(20,60]"}
-
-\item{factor}{logical; TRUE returns informative character string, FALSE numeric (left value)}
-}
-\value{
-If `factor = TRUE`, returns a character vector; else returns a numeric 
-vector.
-}
-\description{
-Selects lowest values of each factor after cut() based
-on the assumption that the value starts from index 2 and end in comma ",".
-}
-\details{
-type = 'factor': "[50,52)" -> "50-51" OR "[50,51)" -> "50"
-
-type = 'numeric': lowest bound in numeric.
-}
-\examples{
-cut_bound("[1900, 1910)") ## "1900-1909"
-}
-\author{
-Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{cut_bound}
+\alias{cut_bound}
+\title{Change output values from cut(..., labels = NULL) output}
+\usage{
+cut_bound(t, factor = TRUE)
+}
+\arguments{
+\item{t}{is a character vector of elements, e.g. "(20,60]"}
+
+\item{factor}{logical; TRUE returns informative character string, FALSE numeric (left value)}
+}
+\value{
+If `factor = TRUE`, returns a character vector; else returns a numeric 
+vector.
+}
+\description{
+Selects lowest values of each factor after cut() based
+on the assumption that the value starts from index 2 and end in comma ",".
+}
+\details{
+type = 'factor': "[50,52)" -> "50-51" OR "[50,51)" -> "50"
+
+type = 'numeric': lowest bound in numeric.
+}
+\examples{
+cut_bound("[1900, 1910)") ## "1900-1909"
+}
+\author{
+Matti Rantanen
+}
diff --git a/man/direct_standardization.Rd b/man/direct_standardization.Rd
index af19c3e..3bb9fd8 100644
--- a/man/direct_standardization.Rd
+++ b/man/direct_standardization.Rd
@@ -1,157 +1,157 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/direct_adjusting.R
-\name{direct_standardization}
-\alias{direct_standardization}
-\alias{direct_adjusting}
-\title{Direct Adjusting in \pkg{popEpi} Using Weights}
-\description{
-Several functions in \pkg{popEpi} have support for direct standardization
-of estimates. This document explains the usage of weighting with those
-functions.
-}
-\details{
-Direct standardization is performed by computing estimates of
-\code{E}
-by the set of adjusting variables \code{A}, to which a set of weights 
-\code{W} is applicable. The weighted average over \code{A} is then the
-direct-adjusted estimate of \code{E} (\code{E*}).
-
-To enable both quick and easy as well as more rigorous usage of direct
-standardization with weights, the weights arguments in \pkg{popEpi}
-can be supplied in several ways. Ability to use the different
-ways depends on the number of adjusting variables.
-
-The weights are always handled internally to sum to 1, so they do not
-need to be scaled in this manner when they are supplied. E.g.
-counts of subjects in strata may be passed.
-}
-\section{Basic usage - one adjusting variable}{
-
-
-In the simple case where we are adjusting by only one variable 
-(e.g. by age group), one can simply supply a vector of weights:
-
-\code{FUN(weights = c(0.1, 0.25, 0.25, 0.2, 0.2))}
-
-which may be stored in advance:
-
-\code{w <- c(0.1, 0.25, 0.25, 0.2, 0.2)}
-
-\code{FUN(weights = w)}
-
-The order of the weights matters. \pkg{popEpi} functions with direct
-adjusting enabled match the supplied weights to the adjusting variables
-as follows: If the adjusting variable is a \code{factor}, the order
-of the levels is used. Otherwise, the alphabetic order of the unique
-values is used (try \code{sort} to see how it works). For clarity
-and certainty we recommend using \code{factor} or \code{numeric} variables
-when possible. \code{character} variables should be avoided: to see why,
-try \code{sort(15:9)} and \code{sort(as.character(15:9))}.
-
-It is also possible to supply a \code{character} string corresponding
-to one of the age group standardization schemes integrated into \pkg{popEpi}:
-
-\itemize{
-\item \code{'europe_1976_18of5'} - European std. population (1976), 18 age groups
-\item \code{'nordic_2000_18of5'} - Nordic std. population (2000), 18 age groups
-\item \code{'world_1966_18of5'} - world standard (1966), 18 age groups
-\item \code{'world_2000_18of5'} - world standard (2000), 18 age groups
-\item \code{'world_2000_20of5'} - world standard (2000), 20 age groups 
-\item \code{'world_2000_101of1'} - world standard (2000), 101 age groups
-}
-
-Additionally, \code{\link{ICSS}} contains international weights used in 
-cancer survival analysis, but they are not currently usable by passing
-a string to \code{weights} and must be supplied by hand.
-
-You may also supply \code{weights = "internal"} to use internally
-computed weights, i.e. usually simply the counts of subjects / person-time
-experienced in each stratum. E.g.
-
-\code{FUN(weights = "world_2000_18of5")}
-
-will use the world standard population from 2000 as 
-weights for 18 age groups, that your adjusting variable is 
-assumed to contain. The adjusting variable must be coded in this case as 
-a numeric variable containing \code{1:18} or as a \code{factor} with
-18 levels (coded from the youngest to the oldest age group).
-}
-
-\section{More than one adjusting variable}{
-
-
-In the case that you employ more than one adjusting variable, separate
-weights should be passed to match to the levels of the different adjusting
-variables. When supplied correctly, "grand" weights are formed based on
-the variable-specific weights by multiplying over the variable-specific
-weights (e.g. if men have \code{w = 0.5} and the age group 0-4 has 
-\code{w = 0.1}, the "grand" weight for men aged 0-4 is \code{0.5*0.1}).
-The "grand" weights are then used for adjusting after ensuring they
-sum to one.
-
-When using multiple adjusting variables, you
-are allowed to pass either a named \code{list} of 
-weights or a \code{data.frame} of weights. E.g.
-
-\code{WL <- list(agegroup = age_w, sex = sex_w)}
-
-\code{FUN(weights = WL)}
-
-where \code{age_w} and \code{sex_w} are numeric vectors. Given the 
-conditions explained in the previous section are satisfied, you may also do
-e.g.
-
-\code{WL <- list(agegroup = "world_2000_18of", sex = sex_w)}
-
-\code{FUN(weights = WL)}
-
-and the world standard pop is used as weights for the age groups as outlined
-in the previous section.
-
-Sometimes using a \code{data.frame} can be clearer (and it is fool-proof
-as well). To do this, form a \code{data.frame} that repeats the levels
-of your adjusting variables by each level of every other adjusting variable,
-and assign the weights as a column named \code{"weights"}. E.g.
-
-\code{wdf <- data.frame(sex = rep(0:1, each = 18), agegroup = rep(1:18, 2))}
-
-\code{wdf$weights <- rbinom(36, size = 100, prob = 0.25)}
-
-\code{FUN(weights = wdf)}
-
-If you want to use the counts of subjects in strata as the weights,
-one way to do this is by e.g.
-
-\code{wdf <- as.data.frame(x$V1, x$V2, x$V3)}
-\code{names(wdf) <- c("V1", "V2", "V3", "weights")}
-}
-
-\references{
-Source of the Nordic standard population in 5-year age groups 
-(also contains European & 1966 world standards):
-\url{https://www-dep.iarc.fr/NORDCAN/english/glossary.htm}
-
-Source of the 1976 European standard population: 
-
-Waterhouse, J.,Muir, C.S.,Correa, P.,Powell, J., eds (1976). 
-Cancer Incidence in Five Continents, Vol. III. 
-IARC Scientific Publications, No. 15, Lyon, IARC.
-ISBN: 9789283211150
-
-Source of 2000 world standard population in 1-year age groups:
-\url{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}
-}
-\seealso{
-Other weights: 
-\code{\link{ICSS}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-
-Other popEpi argument evaluation docs: 
-\code{\link{flexible_argument}}
-}
-\author{
-Joonas Miettinen
-}
-\concept{popEpi argument evaluation docs}
-\concept{weights}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/direct_adjusting.R
+\name{direct_standardization}
+\alias{direct_standardization}
+\alias{direct_adjusting}
+\title{Direct Adjusting in \pkg{popEpi} Using Weights}
+\description{
+Several functions in \pkg{popEpi} have support for direct standardization
+of estimates. This document explains the usage of weighting with those
+functions.
+}
+\details{
+Direct standardization is performed by computing estimates of
+\code{E}
+by the set of adjusting variables \code{A}, to which a set of weights 
+\code{W} is applicable. The weighted average over \code{A} is then the
+direct-adjusted estimate of \code{E} (\code{E*}).
+
+To enable both quick and easy as well as more rigorous usage of direct
+standardization with weights, the weights arguments in \pkg{popEpi}
+can be supplied in several ways. Ability to use the different
+ways depends on the number of adjusting variables.
+
+The weights are always handled internally to sum to 1, so they do not
+need to be scaled in this manner when they are supplied. E.g.
+counts of subjects in strata may be passed.
+}
+\section{Basic usage - one adjusting variable}{
+
+
+In the simple case where we are adjusting by only one variable 
+(e.g. by age group), one can simply supply a vector of weights:
+
+\code{FUN(weights = c(0.1, 0.25, 0.25, 0.2, 0.2))}
+
+which may be stored in advance:
+
+\code{w <- c(0.1, 0.25, 0.25, 0.2, 0.2)}
+
+\code{FUN(weights = w)}
+
+The order of the weights matters. \pkg{popEpi} functions with direct
+adjusting enabled match the supplied weights to the adjusting variables
+as follows: If the adjusting variable is a \code{factor}, the order
+of the levels is used. Otherwise, the alphabetic order of the unique
+values is used (try \code{sort} to see how it works). For clarity
+and certainty we recommend using \code{factor} or \code{numeric} variables
+when possible. \code{character} variables should be avoided: to see why,
+try \code{sort(15:9)} and \code{sort(as.character(15:9))}.
+
+It is also possible to supply a \code{character} string corresponding
+to one of the age group standardization schemes integrated into \pkg{popEpi}:
+
+\itemize{
+\item \code{'europe_1976_18of5'} - European std. population (1976), 18 age groups
+\item \code{'nordic_2000_18of5'} - Nordic std. population (2000), 18 age groups
+\item \code{'world_1966_18of5'} - world standard (1966), 18 age groups
+\item \code{'world_2000_18of5'} - world standard (2000), 18 age groups
+\item \code{'world_2000_20of5'} - world standard (2000), 20 age groups 
+\item \code{'world_2000_101of1'} - world standard (2000), 101 age groups
+}
+
+Additionally, \code{\link{ICSS}} contains international weights used in 
+cancer survival analysis, but they are not currently usable by passing
+a string to \code{weights} and must be supplied by hand.
+
+You may also supply \code{weights = "internal"} to use internally
+computed weights, i.e. usually simply the counts of subjects / person-time
+experienced in each stratum. E.g.
+
+\code{FUN(weights = "world_2000_18of5")}
+
+will use the world standard population from 2000 as 
+weights for 18 age groups, that your adjusting variable is 
+assumed to contain. The adjusting variable must be coded in this case as 
+a numeric variable containing \code{1:18} or as a \code{factor} with
+18 levels (coded from the youngest to the oldest age group).
+}
+
+\section{More than one adjusting variable}{
+
+
+In the case that you employ more than one adjusting variable, separate
+weights should be passed to match to the levels of the different adjusting
+variables. When supplied correctly, "grand" weights are formed based on
+the variable-specific weights by multiplying over the variable-specific
+weights (e.g. if men have \code{w = 0.5} and the age group 0-4 has 
+\code{w = 0.1}, the "grand" weight for men aged 0-4 is \code{0.5*0.1}).
+The "grand" weights are then used for adjusting after ensuring they
+sum to one.
+
+When using multiple adjusting variables, you
+are allowed to pass either a named \code{list} of 
+weights or a \code{data.frame} of weights. E.g.
+
+\code{WL <- list(agegroup = age_w, sex = sex_w)}
+
+\code{FUN(weights = WL)}
+
+where \code{age_w} and \code{sex_w} are numeric vectors. Given the 
+conditions explained in the previous section are satisfied, you may also do
+e.g.
+
+\code{WL <- list(agegroup = "world_2000_18of", sex = sex_w)}
+
+\code{FUN(weights = WL)}
+
+and the world standard pop is used as weights for the age groups as outlined
+in the previous section.
+
+Sometimes using a \code{data.frame} can be clearer (and it is fool-proof
+as well). To do this, form a \code{data.frame} that repeats the levels
+of your adjusting variables by each level of every other adjusting variable,
+and assign the weights as a column named \code{"weights"}. E.g.
+
+\code{wdf <- data.frame(sex = rep(0:1, each = 18), agegroup = rep(1:18, 2))}
+
+\code{wdf$weights <- rbinom(36, size = 100, prob = 0.25)}
+
+\code{FUN(weights = wdf)}
+
+If you want to use the counts of subjects in strata as the weights,
+one way to do this is by e.g.
+
+\code{wdf <- as.data.frame(x$V1, x$V2, x$V3)}
+\code{names(wdf) <- c("V1", "V2", "V3", "weights")}
+}
+
+\references{
+Source of the Nordic standard population in 5-year age groups 
+(also contains European & 1966 world standards):
+\url{https://www-dep.iarc.fr/NORDCAN/english/glossary.htm}
+
+Source of the 1976 European standard population: 
+
+Waterhouse, J.,Muir, C.S.,Correa, P.,Powell, J., eds (1976). 
+Cancer Incidence in Five Continents, Vol. III. 
+IARC Scientific Publications, No. 15, Lyon, IARC.
+ISBN: 9789283211150
+
+Source of 2000 world standard population in 1-year age groups:
+\url{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}
+}
+\seealso{
+Other weights: 
+\code{\link{ICSS}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+
+Other popEpi argument evaluation docs: 
+\code{\link{flexible_argument}}
+}
+\author{
+Joonas Miettinen
+}
+\concept{popEpi argument evaluation docs}
+\concept{weights}
diff --git a/man/fac2num.Rd b/man/fac2num.Rd
index b4441d4..b6199c1 100644
--- a/man/fac2num.Rd
+++ b/man/fac2num.Rd
@@ -1,42 +1,42 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{fac2num}
-\alias{fac2num}
-\title{Convert factor variable to numeric}
-\source{
-\href{https://stackoverflow.com/questions/3418128/how-to-convert-a-factor-to-an-integer-numeric-without-a-loss-of-information}{Stackoverflow thread}
-}
-\usage{
-fac2num(x)
-}
-\arguments{
-\item{x}{a factor variable with numbers as levels}
-}
-\value{
-A numeric vector based on the levels of `x`.
-}
-\description{
-Convert factor variable with numbers as levels into a numeric variable
-}
-\details{
-For example, a factor with levels \code{c("5","7")} is converted into 
-a numeric variable with values \code{c(5,7)}.
-}
-\examples{
-## this is often not intended
-as.numeric(factor(c(5,7))) ## result: c(1,2)
-## but this
-fac2num(factor(c(5,7))) ## result: c(5,7)
-
-## however
-as.numeric(factor(c("5","7","a"))) ## 1:3
-
-suppressWarnings(
-  fac2num(factor(c("5","7","a"))) ## c(5,7,NA)
-)
-
-
-}
-\seealso{
-\code{\link{robust_values}}
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{fac2num}
+\alias{fac2num}
+\title{Convert factor variable to numeric}
+\source{
+\href{https://stackoverflow.com/questions/3418128/how-to-convert-a-factor-to-an-integer-numeric-without-a-loss-of-information}{Stackoverflow thread}
+}
+\usage{
+fac2num(x)
+}
+\arguments{
+\item{x}{a factor variable with numbers as levels}
+}
+\value{
+A numeric vector based on the levels of `x`.
+}
+\description{
+Convert factor variable with numbers as levels into a numeric variable
+}
+\details{
+For example, a factor with levels \code{c("5","7")} is converted into 
+a numeric variable with values \code{c(5,7)}.
+}
+\examples{
+## this is often not intended
+as.numeric(factor(c(5,7))) ## result: c(1,2)
+## but this
+fac2num(factor(c(5,7))) ## result: c(5,7)
+
+## however
+as.numeric(factor(c("5","7","a"))) ## 1:3
+
+suppressWarnings(
+  fac2num(factor(c("5","7","a"))) ## c(5,7,NA)
+)
+
+
+}
+\seealso{
+\code{\link{robust_values}}
+}
diff --git a/man/flexible_argument.Rd b/man/flexible_argument.Rd
index a9cdab2..581a715 100644
--- a/man/flexible_argument.Rd
+++ b/man/flexible_argument.Rd
@@ -1,169 +1,169 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/flexyargs.R
-\name{flexible_argument}
-\alias{flexible_argument}
-\title{Flexible Variable Usage in \pkg{popEpi} Functions}
-\description{
-Certain arguments in \pkg{popEpi} can be passed in multiple 
-ways. This document shows the usage and a pitfall in the
-usage of such flexible arguments.
-}
-\details{
-Flexible arguments in \pkg{popEpi} are used to pass variables existing
-in your data or in the environment where the function is used 
-(for everyday users this is the global environment - in simple terms,
-where your data is / your work space). The flexible arguments
-are modelled after the \code{by} argument in \code{data.tables} - 
-see \code{?data.table}. There are many ways to supply the same information
-to certain functions in \pkg{popEpi}, but the possible ways listed below
-may be limited in some of them to only allow for using only a part of them.
-}
-\section{Everyday usage}{
-
-
-Most commonly you may pass
-variable names as character strings, e.g.
-
-\code{FUN(arg = c("V1", "V2"), data = x)}
-
-which may be stored in advance:
-
-\code{vars <- c("V1", "V2")}
-
-\code{FUN(arg = vars, data = x)}
-
-where \code{x} contains those variables. You may also supply variable
-names as symbols:
-
-\code{FUN(arg = V1, data = x)}
-
-Or as a list of symbols (similarly to as in \code{\link{aggregate}}):
-
-\code{FUN(arg = list(V1, V2), data = x)}
-
-Or as a list of expressions:
-
-\code{FUN(arg = list(V1 + 1, factor(V2)), data = x)}
-
-A formula without a left-hand-side specified is sometimes allowed as well:
-
-\code{FUN(arg = ~ I(V1 + 1) + factor(V2), data = x)}
-
-Using a symbol or a list of symbols/expressions typically
-causes the function to look for the variable(s)
-first in the supplied data (if any) and then where the function was called.
-For everyday users this means you might define e.g.
-
-\code{V3 <- factor(letters)}
-
-and do e.g.
-
-\code{FUN(arg = list(V1 + 1, factor(V2), V3), data = x)}
-
-provided \code{V1} and \code{V2} exist in \code{x} or in the function calling
-environment.
-}
-
-\section{A pitfall}{
-
-
-There is one way to use flexible arguments incorrectly: By supplying
-the name of a variable which exists both in the supplied data
-and the calling environment, and intending the latter to be used. E.g.
-
-\code{vars <- c("V2")}
-
-\code{FUN(arg = V3, data = x)}
-
-where \code{x} has a column named \code{vars}. This causes the function to
-use \code{x$vars} and NOT \code{x$V2}.
-}
-
-\section{Advanced}{
-
-
-Function programmers are advised to pass character strings
-whenever possible. To fool-proof against conflicts as described in the
-section above, refer to the calling environment explicitly when
-passing the variable containing the character strings:
-
-\code{TF <- environment() ## current env to refer to}
-
-\code{vars <- c("V1", "V2")}
-
-\code{FUN(arg = TF$vars, data = x)}
-
-Even if \code{x} has columns named \code{vars} and \code{TF}, 
-using \code{TF$vars} does not use those columns but only evaluates
-\code{TF$vars}
-in the calling environment. This is made possible by the fact
-that data is always passed as a \code{data.frame}, within which evaluation
-of expressions using the dollar operator is not possible. Therefore
-it is safe to assume the data should not be used. However, lists of 
-expressions will not be checked for dollar use and will fail in conflict
-situations:
-
-\code{TF <- environment() ## current env to refer to}
-
-\code{vars <- letters[1:5]}
-
-\code{x <- data.frame(vars = 1:5, TF = 5:1, V1 = 10:6)}
-
-\code{FUN(arg = list(TF$vars, V1), data = x)}
-
-On the other hand you may typically also pass quoted (\code{\link{quote}})
-or substituted \code{\link{substitute}} expressions etc., where
-the \code{env$object} trick will work as well:
-
-\code{q <- quote(list(vars, V1))}
-
-\code{FUN(arg = TF$q, data = x)}
-
-This works even with
-
-\code{a <- 1:5}
-
-\code{V1 <- quote(TF$a)}
-
-\code{FUN(arg = TF$V1, data = x)}
-
-So no conflicts should occur.
-}
-
-\examples{
-
-data(sire)
-## prepare data for e.g. 5-year "period analysis" for 2008-2012
-## note: sire is a simulated cohort integrated into popEpi.
-BL <- list(fot=seq(0, 5, by = 1/12))
-x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status \%in\% 1:2,
-             breaks = BL)
-              
-x <- aggre(x, by = fot)
-
-## silly example of referring to pyrs data by fixed character string;
-## its possible that the real name wont be fixed in a real-life application.
-pyrs <- "actual_pyrs"  
-TF <- environment()
-x$actual_pyrs <- as.numeric(x$pyrs)
-x$pyrs <- 1
-             
-## this works (uses actual_pyrs eventually)
-st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
-                 pyrs = TF$pyrs, d = from0to1, 
-                 surv.method = "hazard")
-## this would be wrong (sees expression 'pyrs' and uses that column,
-## which is not what is intended here)
-st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
-                 pyrs = pyrs, d = from0to1,
-                 surv.method = "hazard")
-}
-\seealso{
-Other popEpi argument evaluation docs: 
-\code{\link{direct_standardization}}
-}
-\author{
-Joonas Miettinen
-}
-\concept{popEpi argument evaluation docs}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/flexyargs.R
+\name{flexible_argument}
+\alias{flexible_argument}
+\title{Flexible Variable Usage in \pkg{popEpi} Functions}
+\description{
+Certain arguments in \pkg{popEpi} can be passed in multiple 
+ways. This document shows the usage and a pitfall in the
+usage of such flexible arguments.
+}
+\details{
+Flexible arguments in \pkg{popEpi} are used to pass variables existing
+in your data or in the environment where the function is used 
+(for everyday users this is the global environment - in simple terms,
+where your data is / your work space). The flexible arguments
+are modelled after the \code{by} argument in \code{data.tables} - 
+see \code{?data.table}. There are many ways to supply the same information
+to certain functions in \pkg{popEpi}, but the possible ways listed below
+may be limited in some of them to only allow for using only a part of them.
+}
+\section{Everyday usage}{
+
+
+Most commonly you may pass
+variable names as character strings, e.g.
+
+\code{FUN(arg = c("V1", "V2"), data = x)}
+
+which may be stored in advance:
+
+\code{vars <- c("V1", "V2")}
+
+\code{FUN(arg = vars, data = x)}
+
+where \code{x} contains those variables. You may also supply variable
+names as symbols:
+
+\code{FUN(arg = V1, data = x)}
+
+Or as a list of symbols (similarly to as in \code{\link{aggregate}}):
+
+\code{FUN(arg = list(V1, V2), data = x)}
+
+Or as a list of expressions:
+
+\code{FUN(arg = list(V1 + 1, factor(V2)), data = x)}
+
+A formula without a left-hand-side specified is sometimes allowed as well:
+
+\code{FUN(arg = ~ I(V1 + 1) + factor(V2), data = x)}
+
+Using a symbol or a list of symbols/expressions typically
+causes the function to look for the variable(s)
+first in the supplied data (if any) and then where the function was called.
+For everyday users this means you might define e.g.
+
+\code{V3 <- factor(letters)}
+
+and do e.g.
+
+\code{FUN(arg = list(V1 + 1, factor(V2), V3), data = x)}
+
+provided \code{V1} and \code{V2} exist in \code{x} or in the function calling
+environment.
+}
+
+\section{A pitfall}{
+
+
+There is one way to use flexible arguments incorrectly: By supplying
+the name of a variable which exists both in the supplied data
+and the calling environment, and intending the latter to be used. E.g.
+
+\code{vars <- c("V2")}
+
+\code{FUN(arg = V3, data = x)}
+
+where \code{x} has a column named \code{vars}. This causes the function to
+use \code{x$vars} and NOT \code{x$V2}.
+}
+
+\section{Advanced}{
+
+
+Function programmers are advised to pass character strings
+whenever possible. To fool-proof against conflicts as described in the
+section above, refer to the calling environment explicitly when
+passing the variable containing the character strings:
+
+\code{TF <- environment() ## current env to refer to}
+
+\code{vars <- c("V1", "V2")}
+
+\code{FUN(arg = TF$vars, data = x)}
+
+Even if \code{x} has columns named \code{vars} and \code{TF}, 
+using \code{TF$vars} does not use those columns but only evaluates
+\code{TF$vars}
+in the calling environment. This is made possible by the fact
+that data is always passed as a \code{data.frame}, within which evaluation
+of expressions using the dollar operator is not possible. Therefore
+it is safe to assume the data should not be used. However, lists of 
+expressions will not be checked for dollar use and will fail in conflict
+situations:
+
+\code{TF <- environment() ## current env to refer to}
+
+\code{vars <- letters[1:5]}
+
+\code{x <- data.frame(vars = 1:5, TF = 5:1, V1 = 10:6)}
+
+\code{FUN(arg = list(TF$vars, V1), data = x)}
+
+On the other hand you may typically also pass quoted (\code{\link{quote}})
+or substituted \code{\link{substitute}} expressions etc., where
+the \code{env$object} trick will work as well:
+
+\code{q <- quote(list(vars, V1))}
+
+\code{FUN(arg = TF$q, data = x)}
+
+This works even with
+
+\code{a <- 1:5}
+
+\code{V1 <- quote(TF$a)}
+
+\code{FUN(arg = TF$V1, data = x)}
+
+So no conflicts should occur.
+}
+
+\examples{
+
+data(sire)
+## prepare data for e.g. 5-year "period analysis" for 2008-2012
+## note: sire is a simulated cohort integrated into popEpi.
+BL <- list(fot=seq(0, 5, by = 1/12))
+x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status \%in\% 1:2,
+             breaks = BL)
+              
+x <- aggre(x, by = fot)
+
+## silly example of referring to pyrs data by fixed character string;
+## its possible that the real name wont be fixed in a real-life application.
+pyrs <- "actual_pyrs"  
+TF <- environment()
+x$actual_pyrs <- as.numeric(x$pyrs)
+x$pyrs <- 1
+             
+## this works (uses actual_pyrs eventually)
+st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
+                 pyrs = TF$pyrs, d = from0to1, 
+                 surv.method = "hazard")
+## this would be wrong (sees expression 'pyrs' and uses that column,
+## which is not what is intended here)
+st <- survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs",
+                 pyrs = pyrs, d = from0to1,
+                 surv.method = "hazard")
+}
+\seealso{
+Other popEpi argument evaluation docs: 
+\code{\link{direct_standardization}}
+}
+\author{
+Joonas Miettinen
+}
+\concept{popEpi argument evaluation docs}
diff --git a/man/get.yrs.Rd b/man/get.yrs.Rd
index c07e2c7..eea4b67 100644
--- a/man/get.yrs.Rd
+++ b/man/get.yrs.Rd
@@ -1,81 +1,81 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/fractional_years.R
-\name{get.yrs}
-\alias{get.yrs}
-\title{Convert date objects to fractional years}
-\usage{
-get.yrs(x, year.length = "approx", ...)
-}
-\arguments{
-\item{x}{a \code{Date} object, or anything that \code{link{as.Date}}
-accepts}
-
-\item{year.length}{character string, either \code{'actual'} or 
-\code{'approx'}; can be abbreviated; see Details}
-
-\item{...}{additional arguments passed on to \code{\link{as.Date}};
-typically \code{format} when \code{x} is a character string variable,
-and \code{origin} when \code{x} is numeric}
-}
-\value{
-A numeric vector of fractional years.
-}
-\description{
-Using Date objects, calculates given 
-dates as fractional years.
-}
-\details{
-\code{x} should preferably be a \code{date}, \code{Date} or \code{IDate} 
-object, although it can also be a character string variable 
-which is coerced internally to \code{Date} format 
-using \code{\link{as.Date.character}}.
-
-When \code{ year.length = 'actual' }, fractional years are calculated as 
-\code{ year + (day_in_year-1)/365 } for non-leap-years
-and as \code{ year + (day_in_year-1)/366 } for leap years. 
-If \code{ year.length = 'approx' }, fractional years are always
-calculated as in \code{ year + (day_in_year-1)/365.242199 }. 
-
-There is a slight difference, then, between the two methods
-when calculating durations between fractional years. For
-meticulous accuracy one might instead want to calculate durations using
-dates (days) and convert the results to fractional years.
- 
-Note that dates are effectively converted to fractional years at 
-\code{ 00:00:01 } o'clock:
-
-
-\code{ get.yrs("2000-01-01") = 2000 }, and
-\code{ get.yrs("2000-01-02") = 2000 + 1/365.242199 }.
-}
-\examples{
-
-data("sire")
-sire$dg_yrs <- get.yrs(sire$dg_date)
-summary(sire$dg_yrs)
-
-## see: ?as.Date.yrs
-dg_date2 <- as.Date(sire$dg_yrs)
-summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
-
-## Epi's cal.yr versus get.yrs
-d <- as.Date("2000-01-01")
-Epi::cal.yr(d) ## 1999.999
-get.yrs(d) ## 2000
-
-## "..." passed on to as.Date, so character / numeric also accepted as input
-## (and whatever else as.Date accepts)
-get.yrs("2000-06-01")
-get.yrs("20000601", format = "\%Y\%m\%d")
-get.yrs("1/6/00", format = "\%d/\%m/\%y")
-
-get.yrs(100, origin = "1970-01-01")
-
-
-}
-\seealso{
-\code{\link[Epi]{cal.yr}}, \code{\link{as.Date.yrs}}, \code{\link{as.Date}}
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/fractional_years.R
+\name{get.yrs}
+\alias{get.yrs}
+\title{Convert date objects to fractional years}
+\usage{
+get.yrs(x, year.length = "approx", ...)
+}
+\arguments{
+\item{x}{a \code{Date} object, or anything that \code{link{as.Date}}
+accepts}
+
+\item{year.length}{character string, either \code{'actual'} or 
+\code{'approx'}; can be abbreviated; see Details}
+
+\item{...}{additional arguments passed on to \code{\link{as.Date}};
+typically \code{format} when \code{x} is a character string variable,
+and \code{origin} when \code{x} is numeric}
+}
+\value{
+A numeric vector of fractional years.
+}
+\description{
+Using Date objects, calculates given 
+dates as fractional years.
+}
+\details{
+\code{x} should preferably be a \code{date}, \code{Date} or \code{IDate} 
+object, although it can also be a character string variable 
+which is coerced internally to \code{Date} format 
+using \code{\link{as.Date.character}}.
+
+When \code{ year.length = 'actual' }, fractional years are calculated as 
+\code{ year + (day_in_year-1)/365 } for non-leap-years
+and as \code{ year + (day_in_year-1)/366 } for leap years. 
+If \code{ year.length = 'approx' }, fractional years are always
+calculated as in \code{ year + (day_in_year-1)/365.242199 }. 
+
+There is a slight difference, then, between the two methods
+when calculating durations between fractional years. For
+meticulous accuracy one might instead want to calculate durations using
+dates (days) and convert the results to fractional years.
+ 
+Note that dates are effectively converted to fractional years at 
+\code{ 00:00:01 } o'clock:
+
+
+\code{ get.yrs("2000-01-01") = 2000 }, and
+\code{ get.yrs("2000-01-02") = 2000 + 1/365.242199 }.
+}
+\examples{
+
+data("sire")
+sire$dg_yrs <- get.yrs(sire$dg_date)
+summary(sire$dg_yrs)
+
+## see: ?as.Date.yrs
+dg_date2 <- as.Date(sire$dg_yrs)
+summary(as.numeric(dg_date2 - as.Date(sire$dg_date)))
+
+## Epi's cal.yr versus get.yrs
+d <- as.Date("2000-01-01")
+Epi::cal.yr(d) ## 1999.999
+get.yrs(d) ## 2000
+
+## "..." passed on to as.Date, so character / numeric also accepted as input
+## (and whatever else as.Date accepts)
+get.yrs("2000-06-01")
+get.yrs("20000601", format = "\%Y\%m\%d")
+get.yrs("1/6/00", format = "\%d/\%m/\%y")
+
+get.yrs(100, origin = "1970-01-01")
+
+
+}
+\seealso{
+\code{\link[Epi]{cal.yr}}, \code{\link{as.Date.yrs}}, \code{\link{as.Date}}
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/is.Date.Rd b/man/is.Date.Rd
index 950edc3..906ce90 100644
--- a/man/is.Date.Rd
+++ b/man/is.Date.Rd
@@ -1,26 +1,26 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{is.Date}
-\alias{is.Date}
-\title{Test if object is a \code{Date} object}
-\usage{
-is.Date(obj)
-}
-\arguments{
-\item{obj}{object to test on}
-}
-\value{
-`TRUE` if `obj` is of class `"Date"` or `"IDate"`.
-}
-\description{
-Tests if an object is a \code{Date} object and returns
-a logical vector of length 1. \code{IDate} objects are also 
-\code{Date} objects, but \code{date} objects from package \pkg{date}
-are not.
-}
-\seealso{
-\code{\link{get.yrs}}, \code{\link{is_leap_year}}, \code{\link{as.Date}}
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{is.Date}
+\alias{is.Date}
+\title{Test if object is a \code{Date} object}
+\usage{
+is.Date(obj)
+}
+\arguments{
+\item{obj}{object to test on}
+}
+\value{
+`TRUE` if `obj` is of class `"Date"` or `"IDate"`.
+}
+\description{
+Tests if an object is a \code{Date} object and returns
+a logical vector of length 1. \code{IDate} objects are also 
+\code{Date} objects, but \code{date} objects from package \pkg{date}
+are not.
+}
+\seealso{
+\code{\link{get.yrs}}, \code{\link{is_leap_year}}, \code{\link{as.Date}}
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/is_leap_year.Rd b/man/is_leap_year.Rd
index b6f71da..b592d93 100644
--- a/man/is_leap_year.Rd
+++ b/man/is_leap_year.Rd
@@ -1,31 +1,31 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{is_leap_year}
-\alias{is_leap_year}
-\title{Detect leap years}
-\usage{
-is_leap_year(years)
-}
-\arguments{
-\item{years}{a vector or column of year values (numeric or integer)}
-}
-\value{
-A `logical` vector where `TRUE` indicates a leap year.
-}
-\description{
-Given a vector or column of year values (numeric or integer), \code{\link{is_leap_year}} returns a vector of equal length
-of logical indicators, i.e. a vector where corresponding leap years have value TRUE, and FALSE otherwise.
-}
-\examples{
-## can be used to assign new columns easily, e.g. a dummy indicator column
-df <- data.frame(yrs=c(1900,1904,2005,1995))
-df$lyd <- as.integer(is_leap_year(df$yrs))
-
-## mostly it is useful as a condition or to indicate which rows have leap years
-which(is_leap_year(df$yrs)) # 2
-df[is_leap_year(df$yrs),] # 2nd row
-
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{is_leap_year}
+\alias{is_leap_year}
+\title{Detect leap years}
+\usage{
+is_leap_year(years)
+}
+\arguments{
+\item{years}{a vector or column of year values (numeric or integer)}
+}
+\value{
+A `logical` vector where `TRUE` indicates a leap year.
+}
+\description{
+Given a vector or column of year values (numeric or integer), \code{\link{is_leap_year}} returns a vector of equal length
+of logical indicators, i.e. a vector where corresponding leap years have value TRUE, and FALSE otherwise.
+}
+\examples{
+## can be used to assign new columns easily, e.g. a dummy indicator column
+df <- data.frame(yrs=c(1900,1904,2005,1995))
+df$lyd <- as.integer(is_leap_year(df$yrs))
+
+## mostly it is useful as a condition or to indicate which rows have leap years
+which(is_leap_year(df$yrs)) # 2
+df[is_leap_year(df$yrs),] # 2nd row
+
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/lexpand.Rd b/man/lexpand.Rd
index d73642e..9451118 100644
--- a/man/lexpand.Rd
+++ b/man/lexpand.Rd
@@ -1,389 +1,389 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/lexpand.R
-\name{lexpand}
-\alias{lexpand}
-\title{Split case-level observations}
-\usage{
-lexpand(
-  data,
-  birth = NULL,
-  entry = NULL,
-  exit = NULL,
-  event = NULL,
-  status = status != 0,
-  entry.status = NULL,
-  breaks = list(fot = c(0, Inf)),
-  id = NULL,
-  overlapping = TRUE,
-  aggre = NULL,
-  aggre.type = c("unique", "cartesian"),
-  drop = TRUE,
-  pophaz = NULL,
-  pp = TRUE,
-  subset = NULL,
-  merge = TRUE,
-  verbose = FALSE,
-  ...
-)
-}
-\arguments{
-\item{data}{dataset of e.g. cancer cases as rows}
-
-\item{birth}{birth time in date format 
-or fractional years; string, symbol or expression}
-
-\item{entry}{entry time in date format 
-or fractional years; string, symbol or expression}
-
-\item{exit}{exit from follow-up time in date 
-format or fractional years; string, symbol or expression}
-
-\item{event}{advanced: time of possible event differing from \code{exit};
-typically only used in certain SIR/SMR calculations - see Details; 
-string, symbol or expression}
-
-\item{status}{variable indicating type of event at \code{exit} or \code{event}; 
-e.g. \code{status = status != 0}; expression or quoted variable name}
-
-\item{entry.status}{input in the same way as \code{status}; 
-status at \code{entry}; see Details}
-
-\item{breaks}{a named list of vectors of time breaks; 
-e.g. \code{breaks = list(fot=0:5, age=c(0,45,65,Inf))}; see Details}
-
-\item{id}{optional; an id variable; e.g. \code{id = my_id};
-string, symbol or expression}
-
-\item{overlapping}{advanced, logical; if \code{FALSE} AND if \code{data} contains
-multiple rows per subject, 
-ensures that the timelines of \code{id}-specific rows do not overlap;
-this ensures e.g. that person-years are only computed once per subject 
-in a multi-state paradigm}
-
-\item{aggre}{e.g. \code{aggre = list(sex, fot)}; 
-a list of unquoted variables and/or expressions thereof,
-which are interpreted as factors; data events and person-years will
-be aggregated by the unique combinations of these; see Details}
-
-\item{aggre.type}{one of \code{c("unique","cartesian")};
-can be abbreviated; see Details}
-
-\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
-after splitting that reside outside
-the time window as defined by the given breaks (all time scales)}
-
-\item{pophaz}{a dataset of population hazards to merge
-with split data; see Details}
-
-\item{pp}{logical; if \code{TRUE}, computes Pohar-Perme weights using
-\code{pophaz}; adds variable with reserved name \code{pp}; 
-see Details for computing method}
-
-\item{subset}{a logical vector or any logical condition; data is subsetted
-before splitting accordingly}
-
-\item{merge}{logical; if \code{TRUE}, retains all 
-original variables from the data}
-
-\item{verbose}{logical; if \code{TRUE}, the function is chatty and 
-returns some messages along the way}
-
-\item{...}{e.g. \code{fot = 0:5}; instead of specifying a \code{breaks} list, 
-correctly named breaks vectors can be given 
-for \code{fot}, \code{age}, and \code{per}; these override any breaks in the
-\code{breaks} list; see Examples}
-}
-\value{
-If \code{aggre = NULL}, returns 
-a \code{data.table} or \code{data.frame} 
-(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-object expanded to accommodate split observations with time scales as
-fractional years and \code{pophaz} merged in if given. Population
-hazard levels in new variable \code{pop.haz}, and Pohar-Perme
-weights as new variable \code{pp} if requested.
-
-If \code{aggre} is defined, returns a long-format 
-\code{data.table}/\code{data.frame} with the variable \code{pyrs} (person-years),
-and variables for the counts of transitions in state or state at end of 
-follow-up formatted \code{fromXtoY}, where \code{X} and \code{Y} are 
-the states transitioned from and to, respectively. The data may also have
-the columns \code{d.exp} for expected numbers of cases and various
-Pohar-Perme weighted figures as identified by the suffix \code{.pp}; see 
-Details.
-}
-\description{
-Given subject-level data, data is split 
-by calendar time (\code{per}), \code{age}, and follow-up
-time (\code{fot}, from 0 to the end of follow-up) 
-into subject-time-interval rows according to 
-given \code{breaks} and additionally processed if requested.
-}
-\details{
-\strong{Basics}
-
-\code{\link{lexpand}} splits a given data set (with e.g. cancer diagnoses 
-as rows) to subintervals of time over 
-calendar time, age, and follow-up time with given time breaks 
-using \code{\link{splitMulti}}.
-
-The dataset must contain appropriate 
-\code{Date} / \code{IDate} / \code{date} format or
-other numeric variables that can be used
-as the time variables.
-
-You may take a look at a simulated cohort 
-\code{\link{sire}} as an example of the
-minimum required information for processing data with \code{lexpand}.
-
-Many arguments can be supplied as a character string naming the appropriate
-variable (e.g. \code{"sex"}), as a symbol (e.g. \code{sex}) or as an expression
-(e.g. \code{factor(sex, 0:1, c("m", "f"))}) for flexibility.
-
-\strong{Breaks}
-
-You should define all breaks as left inclusive and right exclusive 
-time points (e.g.\code{[a,b)} )
-for 1-3 time dimensions so that the last member of a breaks vector
-is a meaningful "final upper limit",
- e.g. \code{per = c(2002,2007,2012)} 
-to create a last subinterval of the form \code{[2007,2012)}. 
-
-All breaks are explicit, i.e. if \code{drop = TRUE},
-any data beyond the outermost breaks points are dropped. 
-If one wants to have unspecified upper / lower limits on one time scale,
-use \code{Inf}: e.g. \code{breaks = list(fot = 0:5, age = c(0,45,Inf))}.
-Breaks for \code{per} can also be given in 
-\code{Date}/\code{IDate}/\code{date} format, whereupon
-they are converted to fractional years before used in splitting.
-
-The \code{age} time scale can additionally 
-be automatically split into common age grouping schemes
-by naming the scheme with an appropriate character string:
-
-\itemize{
-  \item \code{"18of5"}: age groups 0-4, 5-9, 10-14, ..., 75-79, 80-84, 85+
-  \item \code{"20of5"}: age groups 0-4, 5-9, 10-14, ..., 85-89, 90-94, 95+
-  \item \code{"101of1"}: age groups 0, 1, 2, ..., 98, 99, 100+
-}
-
-\strong{Time variables}
-
-If any of the given time variables
-(\code{birth}, \code{entry}, \code{exit}, \code{event})
-is in any kind of date format, they are first coerced to 
-fractional years before splitting
-using \code{\link{get.yrs}} (with \code{year.length = "actual"}).
-
-Sometimes in e.g. SIR/SMR calculation one may want the event time to differ
-from the time of exit from follow-up, if the subject is still considered
-to be at risk of the event. If \code{event} is specified, the transition to
- \code{status} is moved to \code{event} from \code{exit} 
- using \code{\link[Epi]{cutLexis}}. See Examples.
- 
-\strong{The status variable}
-
-The statuses in the expanded output (\code{lex.Cst} and \code{lex.Xst})
-are determined by using either only \code{status} or both \code{status}
-and \code{entry.status}. If \code{entry.status = NULL}, the status at entry
-is guessed according to the type of variable supplied via \code{status}:
-For numeric variables it will be zero, for factors the first level
-(\code{levels(status)[1]}) and otherwise the first unique value in alphabetical
-order (\code{sort(unique(status))[1]}). 
-
-Using numeric or factor status
-variables is strongly recommended. Logical expressions are also allowed
-(e.g. \code{status = my_status != 0L}) and are converted to integer internally.
-
-\strong{Merging population hazard information}
-
-To enable computing relative/net survivals with \code{\link{survtab}}
-and \code{\link{relpois}}, \code{lexpand} merges an appropriate
-population hazard data (\code{pophaz}) to the expanded data 
-before dropping rows outside the specified
-time window (if \code{drop = TRUE}). \code{pophaz} must, for this reason, 
-contain at a minimum the variables named
-\code{agegroup}, \code{year}, and \code{haz}. \code{pophaz} may contain additional variables to specify
-different population hazard levels in different strata; e.g. \code{popmort} includes \code{sex}.
-All the strata-defining variables must be present in the supplied \code{data}. \code{lexpand} will
-automatically detect variables with common names in the two datasets and merge using them.
-
-Currently \code{year} must be an integer variable specifying the appropriate year. \code{agegroup}
-must currently also specify one-year age groups, e.g. \code{popmort} specifies 101 age groups
-of length 1 year. In both
-\code{year} and \code{agegroup} variables the values are interpreted as the lower bounds of intervals
-(and passed on to a \code{cut} call). The mandatory variable \code{haz}
-must specify the appropriate average rate at the person-year level;
-e.g. \code{haz = -log(survProb)} where \code{survProb} is a one-year conditional
-survival probability will be the correct hazard specification. 
-
-The corresponding \code{pophaz} population hazard value is merged by using the mid points
-of the records after splitting as reference values. E.g. if \code{age=89.9} at the start
-of a 1-year interval, then the reference age value is \code{90.4} for merging. 
-This way we get a "typical" population hazard level for each record.
-
-\strong{Computing Pohar-Perme weights}
-
-If \code{pp = TRUE}, Pohar-Perme weights 
-(the inverse of cumulative population survival) are computed. This will
-create the new \code{pp} variable in the expanded data. \code{pp} is a
-reserved name and \code{lexpand} throws exception if a variable with that name
-exists in \code{data}.
-
-When a survival interval contains one or several rows per subject
-(e.g. due to splitting by the \code{per} scale),
-\code{pp} is cumulated from the beginning of the first record in a survival
-interval for each subject to the mid-point of the remaining time within that
-survival interval, and  that value is given for every other record 
-that a given person has within the same survival interval. 
-
-E.g. with 5 rows of duration \code{1/5} within a survival interval 
-\code{[0,1)]}, \code{pp} is determined for all records by a cumulative 
-population survival from \code{0} to \code{0.5}. The existing accuracy is used,
-so that the weight is cumulated first up to the end of the second row
-and then over the remaining distance to the mid-point (first to 0.4, then to
-0.5). This ensures that more accurately merged population hazards are fully
-used.
-
-\strong{Event not at end of follow-up & overlapping time lines}
-
-\code{event} may be used if the event indicated by \code{status} should
-occur at a time differing from \code{exit}. If \code{event} is defined,
-\code{cutLexis} is used on the data set after coercing it to the \code{Lexis}
-format and before splitting. Note that some values of \code{event} are allowed
-to be \code{NA} as with \code{cutLexis} to accommodate observations
-without an event occurring.
-
-Additionally, setting \code{overlapping = FALSE} ensures that (irrespective
-of using \code{event}) the each subject defined by \code{id} only has one
-continuous time line instead of possibly overlapping time lines if
-there are multiple rows in \code{data} by \code{id}.
-
-
-\strong{Aggregating}
-
-Certain analyses such as SIR/SMR calculations require tables of events and
-person-years by the unique combinations (interactions) of several variables. 
-For this, \code{aggre} can be specified as a list of such variables 
-(preferably \code{factor} variables but not mandatory)
- and any arbitrary functions of the 
-variables at one's disposal. E.g. 
-
-\code{aggre = list(sex, agegr = cut(dg_age, 0:100))}
-
-would tabulate events and person-years by sex and an ad-hoc age group
-variable. Every ad-hoc-created variable should be named.
-
-\code{fot}, \code{per}, and \code{age} are special reserved variables which,
-when present in the \code{aggre} list, are output as categories of the
-corresponding time scale variables by using 
-e.g. 
-
-\code{cut(fot, breaks$fot, right=FALSE)}. 
-
-This only works if
-the corresponding breaks are defined in \code{breaks} or via "\code{...}".
-E.g. 
-
-\code{aggre = list(sex, fot.int = fot)} with 
-
-\code{breaks = list(fot=0:5)}.
-
-The output variable \code{fot.int} in the above example will have
-the lower limits of the appropriate intervals as values.
-
-\code{aggre} as a named list will output numbers of events and person-years
-with the given new names as categorizing variable names, e.g. 
-\code{aggre = list(follow_up = fot, gender = sex, agegroup = age)}.
-
-The output table has person-years (\code{pyrs}) and event counts
-(e.g. \code{from0to1}) as columns. Event counts are the numbers of transitions
-(\code{lex.Cst != lex.Xst}) or the \code{lex.Xst} value at a subject's 
-last record (subject possibly defined by \code{id}).
-
-If \code{aggre.type = "unique"} (alias \code{"non-empty"}), 
-the above results are computed for existing
-combinations of expressions given in \code{aggre}, but also for non-existing
-combinations if \code{aggre.type = "cartesian"} (alias \code{"full"}). E.g. if a
-factor variable has levels \code{"a", "b", "c"} but the data is limited
-to only have levels \code{"a", "b"} present 
-(more than zero rows have these level values), the former setting only
-computes results for \code{"a", "b"}, and the latter also for \code{"c"}
-and any combination with other variables or expression given in \code{aggre}.
-In essence, \code{"cartesian"} forces also combinations of variables used
-in \code{aggre} that have no match in data to be shown in the result.
-
-If \code{aggre} is not \code{NULL} and \code{pophaz} has been supplied,
-\code{lexpand} also aggregates the expected counts of events, which
-appears in the output data by the reserved name \code{d.exp}. Additionally,
-having \code{pp = TRUE} causes \code{lexpand} to also compute various
-Pohar-Perme weighted figures necessary for computing Pohar-Perme net survivals
-with \code{\link{survtab_ag}}. This can be slow, so consider what is really
-needed. The Pohar-Perme weighted figures have the suffix \code{.pp}.
-}
-\examples{
-\donttest{
-## prepare data for e.g. 5-year cohort survival calculation
-x <- lexpand(sire, breaks=list(fot=seq(0, 5, by = 1/12)), 
-             birth = bi_date, entry = dg_date, exit = ex_date,
-             status =  status != 0, pophaz=popmort)
-
-## prepare data for e.g. 5-year "period analysis" for 2008-2012
-BL <- list(fot = seq(0, 5, by = 1/12), per = c("2008-01-01", "2013-01-01"))
-x <- lexpand(sire, breaks = BL, 
-             birth = bi_date, entry = dg_date, exit = ex_date,
-             pophaz=popmort, status =  status != 0)
-
-## aggregating
-BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-ag <- lexpand(sire, breaks = BL, status = status != 0, 
-             birth = bi_date, entry = dg_date, exit = ex_date,
-              aggre=list(sex, period = per, surv.int = fot))
-
-## aggregating even more
-ag <- lexpand(sire, breaks = BL, status = status != 0, 
-              birth = bi_date, entry = dg_date, exit = ex_date,
-              aggre=list(sex, period = per, surv.int = fot),
-              pophaz = popmort, pp = TRUE)
-
-## using "..."
-x <- lexpand(sire, fot=0:5, status =  status != 0,
-             birth = bi_date, entry = dg_date, exit = ex_date,
-             pophaz=popmort) 
-
-x <- lexpand(sire, fot=0:5, status =  status != 0, 
-             birth = bi_date, entry = dg_date, exit = ex_date,
-             aggre=list(sex, surv.int = fot))
-             
-## using the "event" argument: it just places the transition to given "status"
-## at the "event" time instead of at the end, if possible using cutLexis
-x <- lexpand(sire, status = status, event = dg_date,
-             birth = bi_date, entry = dg_date, exit = ex_date,) 
-
-## aggregating with custom "event" time
-## (the transition to status is moved to the "event" time)
-x <- lexpand(sire, status = status, event = dg_date, 
-             birth = bi_date, entry = dg_date, exit = ex_date,
-             per = 1970:2014, age = c(0:100,Inf),
-             aggre = list(sex, year = per, agegroup = age)) 
-
-}
-
-}
-\seealso{
-\code{\link[Epi]{Lexis}}, \code{\link{popmort}}
-
-Other splitting functions: 
-\code{\link{splitLexisDT}()},
-\code{\link{splitMulti}()}
-
-Other aggregation functions: 
-\code{\link{aggre}()},
-\code{\link{as.aggre}()},
-\code{\link{setaggre}()},
-\code{\link{summary.aggre}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{aggregation functions}
-\concept{splitting functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/lexpand.R
+\name{lexpand}
+\alias{lexpand}
+\title{Split case-level observations}
+\usage{
+lexpand(
+  data,
+  birth = NULL,
+  entry = NULL,
+  exit = NULL,
+  event = NULL,
+  status = status != 0,
+  entry.status = NULL,
+  breaks = list(fot = c(0, Inf)),
+  id = NULL,
+  overlapping = TRUE,
+  aggre = NULL,
+  aggre.type = c("unique", "cartesian"),
+  drop = TRUE,
+  pophaz = NULL,
+  pp = TRUE,
+  subset = NULL,
+  merge = TRUE,
+  verbose = FALSE,
+  ...
+)
+}
+\arguments{
+\item{data}{dataset of e.g. cancer cases as rows}
+
+\item{birth}{birth time in date format 
+or fractional years; string, symbol or expression}
+
+\item{entry}{entry time in date format 
+or fractional years; string, symbol or expression}
+
+\item{exit}{exit from follow-up time in date 
+format or fractional years; string, symbol or expression}
+
+\item{event}{advanced: time of possible event differing from \code{exit};
+typically only used in certain SIR/SMR calculations - see Details; 
+string, symbol or expression}
+
+\item{status}{variable indicating type of event at \code{exit} or \code{event}; 
+e.g. \code{status = status != 0}; expression or quoted variable name}
+
+\item{entry.status}{input in the same way as \code{status}; 
+status at \code{entry}; see Details}
+
+\item{breaks}{a named list of vectors of time breaks; 
+e.g. \code{breaks = list(fot=0:5, age=c(0,45,65,Inf))}; see Details}
+
+\item{id}{optional; an id variable; e.g. \code{id = my_id};
+string, symbol or expression}
+
+\item{overlapping}{advanced, logical; if \code{FALSE} AND if \code{data} contains
+multiple rows per subject, 
+ensures that the timelines of \code{id}-specific rows do not overlap;
+this ensures e.g. that person-years are only computed once per subject 
+in a multi-state paradigm}
+
+\item{aggre}{e.g. \code{aggre = list(sex, fot)}; 
+a list of unquoted variables and/or expressions thereof,
+which are interpreted as factors; data events and person-years will
+be aggregated by the unique combinations of these; see Details}
+
+\item{aggre.type}{one of \code{c("unique","cartesian")};
+can be abbreviated; see Details}
+
+\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
+after splitting that reside outside
+the time window as defined by the given breaks (all time scales)}
+
+\item{pophaz}{a dataset of population hazards to merge
+with split data; see Details}
+
+\item{pp}{logical; if \code{TRUE}, computes Pohar-Perme weights using
+\code{pophaz}; adds variable with reserved name \code{pp}; 
+see Details for computing method}
+
+\item{subset}{a logical vector or any logical condition; data is subsetted
+before splitting accordingly}
+
+\item{merge}{logical; if \code{TRUE}, retains all 
+original variables from the data}
+
+\item{verbose}{logical; if \code{TRUE}, the function is chatty and 
+returns some messages along the way}
+
+\item{...}{e.g. \code{fot = 0:5}; instead of specifying a \code{breaks} list, 
+correctly named breaks vectors can be given 
+for \code{fot}, \code{age}, and \code{per}; these override any breaks in the
+\code{breaks} list; see Examples}
+}
+\value{
+If \code{aggre = NULL}, returns 
+a \code{data.table} or \code{data.frame} 
+(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+object expanded to accommodate split observations with time scales as
+fractional years and \code{pophaz} merged in if given. Population
+hazard levels in new variable \code{pop.haz}, and Pohar-Perme
+weights as new variable \code{pp} if requested.
+
+If \code{aggre} is defined, returns a long-format 
+\code{data.table}/\code{data.frame} with the variable \code{pyrs} (person-years),
+and variables for the counts of transitions in state or state at end of 
+follow-up formatted \code{fromXtoY}, where \code{X} and \code{Y} are 
+the states transitioned from and to, respectively. The data may also have
+the columns \code{d.exp} for expected numbers of cases and various
+Pohar-Perme weighted figures as identified by the suffix \code{.pp}; see 
+Details.
+}
+\description{
+Given subject-level data, data is split 
+by calendar time (\code{per}), \code{age}, and follow-up
+time (\code{fot}, from 0 to the end of follow-up) 
+into subject-time-interval rows according to 
+given \code{breaks} and additionally processed if requested.
+}
+\details{
+\strong{Basics}
+
+\code{\link{lexpand}} splits a given data set (with e.g. cancer diagnoses 
+as rows) to subintervals of time over 
+calendar time, age, and follow-up time with given time breaks 
+using \code{\link{splitMulti}}.
+
+The dataset must contain appropriate 
+\code{Date} / \code{IDate} / \code{date} format or
+other numeric variables that can be used
+as the time variables.
+
+You may take a look at a simulated cohort 
+\code{\link{sire}} as an example of the
+minimum required information for processing data with \code{lexpand}.
+
+Many arguments can be supplied as a character string naming the appropriate
+variable (e.g. \code{"sex"}), as a symbol (e.g. \code{sex}) or as an expression
+(e.g. \code{factor(sex, 0:1, c("m", "f"))}) for flexibility.
+
+\strong{Breaks}
+
+You should define all breaks as left inclusive and right exclusive 
+time points (e.g.\code{[a,b)} )
+for 1-3 time dimensions so that the last member of a breaks vector
+is a meaningful "final upper limit",
+ e.g. \code{per = c(2002,2007,2012)} 
+to create a last subinterval of the form \code{[2007,2012)}. 
+
+All breaks are explicit, i.e. if \code{drop = TRUE},
+any data beyond the outermost breaks points are dropped. 
+If one wants to have unspecified upper / lower limits on one time scale,
+use \code{Inf}: e.g. \code{breaks = list(fot = 0:5, age = c(0,45,Inf))}.
+Breaks for \code{per} can also be given in 
+\code{Date}/\code{IDate}/\code{date} format, whereupon
+they are converted to fractional years before used in splitting.
+
+The \code{age} time scale can additionally 
+be automatically split into common age grouping schemes
+by naming the scheme with an appropriate character string:
+
+\itemize{
+  \item \code{"18of5"}: age groups 0-4, 5-9, 10-14, ..., 75-79, 80-84, 85+
+  \item \code{"20of5"}: age groups 0-4, 5-9, 10-14, ..., 85-89, 90-94, 95+
+  \item \code{"101of1"}: age groups 0, 1, 2, ..., 98, 99, 100+
+}
+
+\strong{Time variables}
+
+If any of the given time variables
+(\code{birth}, \code{entry}, \code{exit}, \code{event})
+is in any kind of date format, they are first coerced to 
+fractional years before splitting
+using \code{\link{get.yrs}} (with \code{year.length = "actual"}).
+
+Sometimes in e.g. SIR/SMR calculation one may want the event time to differ
+from the time of exit from follow-up, if the subject is still considered
+to be at risk of the event. If \code{event} is specified, the transition to
+ \code{status} is moved to \code{event} from \code{exit} 
+ using \code{\link[Epi]{cutLexis}}. See Examples.
+ 
+\strong{The status variable}
+
+The statuses in the expanded output (\code{lex.Cst} and \code{lex.Xst})
+are determined by using either only \code{status} or both \code{status}
+and \code{entry.status}. If \code{entry.status = NULL}, the status at entry
+is guessed according to the type of variable supplied via \code{status}:
+For numeric variables it will be zero, for factors the first level
+(\code{levels(status)[1]}) and otherwise the first unique value in alphabetical
+order (\code{sort(unique(status))[1]}). 
+
+Using numeric or factor status
+variables is strongly recommended. Logical expressions are also allowed
+(e.g. \code{status = my_status != 0L}) and are converted to integer internally.
+
+\strong{Merging population hazard information}
+
+To enable computing relative/net survivals with \code{\link{survtab}}
+and \code{\link{relpois}}, \code{lexpand} merges an appropriate
+population hazard data (\code{pophaz}) to the expanded data 
+before dropping rows outside the specified
+time window (if \code{drop = TRUE}). \code{pophaz} must, for this reason, 
+contain at a minimum the variables named
+\code{agegroup}, \code{year}, and \code{haz}. \code{pophaz} may contain additional variables to specify
+different population hazard levels in different strata; e.g. \code{popmort} includes \code{sex}.
+All the strata-defining variables must be present in the supplied \code{data}. \code{lexpand} will
+automatically detect variables with common names in the two datasets and merge using them.
+
+Currently \code{year} must be an integer variable specifying the appropriate year. \code{agegroup}
+must currently also specify one-year age groups, e.g. \code{popmort} specifies 101 age groups
+of length 1 year. In both
+\code{year} and \code{agegroup} variables the values are interpreted as the lower bounds of intervals
+(and passed on to a \code{cut} call). The mandatory variable \code{haz}
+must specify the appropriate average rate at the person-year level;
+e.g. \code{haz = -log(survProb)} where \code{survProb} is a one-year conditional
+survival probability will be the correct hazard specification. 
+
+The corresponding \code{pophaz} population hazard value is merged by using the mid points
+of the records after splitting as reference values. E.g. if \code{age=89.9} at the start
+of a 1-year interval, then the reference age value is \code{90.4} for merging. 
+This way we get a "typical" population hazard level for each record.
+
+\strong{Computing Pohar-Perme weights}
+
+If \code{pp = TRUE}, Pohar-Perme weights 
+(the inverse of cumulative population survival) are computed. This will
+create the new \code{pp} variable in the expanded data. \code{pp} is a
+reserved name and \code{lexpand} throws exception if a variable with that name
+exists in \code{data}.
+
+When a survival interval contains one or several rows per subject
+(e.g. due to splitting by the \code{per} scale),
+\code{pp} is cumulated from the beginning of the first record in a survival
+interval for each subject to the mid-point of the remaining time within that
+survival interval, and  that value is given for every other record 
+that a given person has within the same survival interval. 
+
+E.g. with 5 rows of duration \code{1/5} within a survival interval 
+\code{[0,1)]}, \code{pp} is determined for all records by a cumulative 
+population survival from \code{0} to \code{0.5}. The existing accuracy is used,
+so that the weight is cumulated first up to the end of the second row
+and then over the remaining distance to the mid-point (first to 0.4, then to
+0.5). This ensures that more accurately merged population hazards are fully
+used.
+
+\strong{Event not at end of follow-up & overlapping time lines}
+
+\code{event} may be used if the event indicated by \code{status} should
+occur at a time differing from \code{exit}. If \code{event} is defined,
+\code{cutLexis} is used on the data set after coercing it to the \code{Lexis}
+format and before splitting. Note that some values of \code{event} are allowed
+to be \code{NA} as with \code{cutLexis} to accommodate observations
+without an event occurring.
+
+Additionally, setting \code{overlapping = FALSE} ensures that (irrespective
+of using \code{event}) the each subject defined by \code{id} only has one
+continuous time line instead of possibly overlapping time lines if
+there are multiple rows in \code{data} by \code{id}.
+
+
+\strong{Aggregating}
+
+Certain analyses such as SIR/SMR calculations require tables of events and
+person-years by the unique combinations (interactions) of several variables. 
+For this, \code{aggre} can be specified as a list of such variables 
+(preferably \code{factor} variables but not mandatory)
+ and any arbitrary functions of the 
+variables at one's disposal. E.g. 
+
+\code{aggre = list(sex, agegr = cut(dg_age, 0:100))}
+
+would tabulate events and person-years by sex and an ad-hoc age group
+variable. Every ad-hoc-created variable should be named.
+
+\code{fot}, \code{per}, and \code{age} are special reserved variables which,
+when present in the \code{aggre} list, are output as categories of the
+corresponding time scale variables by using 
+e.g. 
+
+\code{cut(fot, breaks$fot, right=FALSE)}. 
+
+This only works if
+the corresponding breaks are defined in \code{breaks} or via "\code{...}".
+E.g. 
+
+\code{aggre = list(sex, fot.int = fot)} with 
+
+\code{breaks = list(fot=0:5)}.
+
+The output variable \code{fot.int} in the above example will have
+the lower limits of the appropriate intervals as values.
+
+\code{aggre} as a named list will output numbers of events and person-years
+with the given new names as categorizing variable names, e.g. 
+\code{aggre = list(follow_up = fot, gender = sex, agegroup = age)}.
+
+The output table has person-years (\code{pyrs}) and event counts
+(e.g. \code{from0to1}) as columns. Event counts are the numbers of transitions
+(\code{lex.Cst != lex.Xst}) or the \code{lex.Xst} value at a subject's 
+last record (subject possibly defined by \code{id}).
+
+If \code{aggre.type = "unique"} (alias \code{"non-empty"}), 
+the above results are computed for existing
+combinations of expressions given in \code{aggre}, but also for non-existing
+combinations if \code{aggre.type = "cartesian"} (alias \code{"full"}). E.g. if a
+factor variable has levels \code{"a", "b", "c"} but the data is limited
+to only have levels \code{"a", "b"} present 
+(more than zero rows have these level values), the former setting only
+computes results for \code{"a", "b"}, and the latter also for \code{"c"}
+and any combination with other variables or expression given in \code{aggre}.
+In essence, \code{"cartesian"} forces also combinations of variables used
+in \code{aggre} that have no match in data to be shown in the result.
+
+If \code{aggre} is not \code{NULL} and \code{pophaz} has been supplied,
+\code{lexpand} also aggregates the expected counts of events, which
+appears in the output data by the reserved name \code{d.exp}. Additionally,
+having \code{pp = TRUE} causes \code{lexpand} to also compute various
+Pohar-Perme weighted figures necessary for computing Pohar-Perme net survivals
+with \code{\link{survtab_ag}}. This can be slow, so consider what is really
+needed. The Pohar-Perme weighted figures have the suffix \code{.pp}.
+}
+\examples{
+\donttest{
+## prepare data for e.g. 5-year cohort survival calculation
+x <- lexpand(sire, breaks=list(fot=seq(0, 5, by = 1/12)), 
+             birth = bi_date, entry = dg_date, exit = ex_date,
+             status =  status != 0, pophaz=popmort)
+
+## prepare data for e.g. 5-year "period analysis" for 2008-2012
+BL <- list(fot = seq(0, 5, by = 1/12), per = c("2008-01-01", "2013-01-01"))
+x <- lexpand(sire, breaks = BL, 
+             birth = bi_date, entry = dg_date, exit = ex_date,
+             pophaz=popmort, status =  status != 0)
+
+## aggregating
+BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+ag <- lexpand(sire, breaks = BL, status = status != 0, 
+             birth = bi_date, entry = dg_date, exit = ex_date,
+              aggre=list(sex, period = per, surv.int = fot))
+
+## aggregating even more
+ag <- lexpand(sire, breaks = BL, status = status != 0, 
+              birth = bi_date, entry = dg_date, exit = ex_date,
+              aggre=list(sex, period = per, surv.int = fot),
+              pophaz = popmort, pp = TRUE)
+
+## using "..."
+x <- lexpand(sire, fot=0:5, status =  status != 0,
+             birth = bi_date, entry = dg_date, exit = ex_date,
+             pophaz=popmort) 
+
+x <- lexpand(sire, fot=0:5, status =  status != 0, 
+             birth = bi_date, entry = dg_date, exit = ex_date,
+             aggre=list(sex, surv.int = fot))
+             
+## using the "event" argument: it just places the transition to given "status"
+## at the "event" time instead of at the end, if possible using cutLexis
+x <- lexpand(sire, status = status, event = dg_date,
+             birth = bi_date, entry = dg_date, exit = ex_date,) 
+
+## aggregating with custom "event" time
+## (the transition to status is moved to the "event" time)
+x <- lexpand(sire, status = status, event = dg_date, 
+             birth = bi_date, entry = dg_date, exit = ex_date,
+             per = 1970:2014, age = c(0:100,Inf),
+             aggre = list(sex, year = per, agegroup = age)) 
+
+}
+
+}
+\seealso{
+\code{\link[Epi]{Lexis}}, \code{\link{popmort}}
+
+Other splitting functions: 
+\code{\link{splitLexisDT}()},
+\code{\link{splitMulti}()}
+
+Other aggregation functions: 
+\code{\link{aggre}()},
+\code{\link{as.aggre}()},
+\code{\link{setaggre}()},
+\code{\link{summary.aggre}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{aggregation functions}
+\concept{splitting functions}
diff --git a/man/lines.sirspline.Rd b/man/lines.sirspline.Rd
index 9c397ca..dbd60ad 100644
--- a/man/lines.sirspline.Rd
+++ b/man/lines.sirspline.Rd
@@ -1,49 +1,49 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{lines.sirspline}
-\alias{lines.sirspline}
-\title{lines method for sirspline-object}
-\usage{
-\method{lines}{sirspline}(x, conf.int = TRUE, print.levels = NA, select.spline, ...)
-}
-\arguments{
-\item{x}{an object returned by function sirspline}
-
-\item{conf.int}{logical; default TRUE draws also the 95 confidence intervals}
-
-\item{print.levels}{name(s) to be plotted. Default plots all levels.}
-
-\item{select.spline}{select which spline variable (a number or a name) is plotted.}
-
-\item{...}{arguments passed on to lines()}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plot SIR spline lines with R base graphics
-}
-\details{
-In \code{lines.sirspline} most of graphical parameters is user 
-adjustable.
-Desired spline variable can be selected with \code{select.spline} and only one
-can be plotted at a time. The spline variable can include 
-several levels, e.g. gender (these are the levels of \code{print}
-from \code{sirspline}). All levels are printed by default, but a
-specific level can be selected using argument
-\code{print.levels}. Printing the levels separately enables  e.g. to
-give different colours for each level.
-}
-\seealso{
-Other sir functions: 
-\code{\link{plot.sirspline}()},
-\code{\link{sir_exp}()},
-\code{\link{sir_ratio}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()}
-}
-\author{
-Matti Rantanen
-}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{lines.sirspline}
+\alias{lines.sirspline}
+\title{lines method for sirspline-object}
+\usage{
+\method{lines}{sirspline}(x, conf.int = TRUE, print.levels = NA, select.spline, ...)
+}
+\arguments{
+\item{x}{an object returned by function sirspline}
+
+\item{conf.int}{logical; default TRUE draws also the 95 confidence intervals}
+
+\item{print.levels}{name(s) to be plotted. Default plots all levels.}
+
+\item{select.spline}{select which spline variable (a number or a name) is plotted.}
+
+\item{...}{arguments passed on to lines()}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plot SIR spline lines with R base graphics
+}
+\details{
+In \code{lines.sirspline} most of graphical parameters is user 
+adjustable.
+Desired spline variable can be selected with \code{select.spline} and only one
+can be plotted at a time. The spline variable can include 
+several levels, e.g. gender (these are the levels of \code{print}
+from \code{sirspline}). All levels are printed by default, but a
+specific level can be selected using argument
+\code{print.levels}. Printing the levels separately enables  e.g. to
+give different colours for each level.
+}
+\seealso{
+Other sir functions: 
+\code{\link{plot.sirspline}()},
+\code{\link{sir_exp}()},
+\code{\link{sir_ratio}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()}
+}
+\author{
+Matti Rantanen
+}
+\concept{sir functions}
diff --git a/man/lines.survmean.Rd b/man/lines.survmean.Rd
index 60682bf..e5173e9 100644
--- a/man/lines.survmean.Rd
+++ b/man/lines.survmean.Rd
@@ -1,42 +1,42 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{lines.survmean}
-\alias{lines.survmean}
-\title{Graphically Inspect Curves Used in Mean Survival Computation}
-\usage{
-\method{lines}{survmean}(x, ...)
-}
-\arguments{
-\item{x}{a \code{survmean} object}
-
-\item{...}{arguments passed (ultimately) to \code{matlines}; you
-may, therefore, supply e.g. \code{lwd} through this, though arguments
-such as \code{lty} and \code{col} will not work}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plots the observed (with extrapolation) and expected survival
-curves for all strata in an object created by \code{\link{survmean}}
-}
-\details{
-This function is intended to be a workhorse for \code{\link{plot.survmean}}.
-If you want finer control over the plotted curves, extract the curves from
-the \code{survmean} output using 
-
-\code{attr(x, "curves")}
-
-where \code{x} is a \code{survmean} object.
-}
-\seealso{
-Other survmean functions: 
-\code{\link{Surv}()},
-\code{\link{plot.survmean}()},
-\code{\link{survmean}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survmean functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{lines.survmean}
+\alias{lines.survmean}
+\title{Graphically Inspect Curves Used in Mean Survival Computation}
+\usage{
+\method{lines}{survmean}(x, ...)
+}
+\arguments{
+\item{x}{a \code{survmean} object}
+
+\item{...}{arguments passed (ultimately) to \code{matlines}; you
+may, therefore, supply e.g. \code{lwd} through this, though arguments
+such as \code{lty} and \code{col} will not work}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plots the observed (with extrapolation) and expected survival
+curves for all strata in an object created by \code{\link{survmean}}
+}
+\details{
+This function is intended to be a workhorse for \code{\link{plot.survmean}}.
+If you want finer control over the plotted curves, extract the curves from
+the \code{survmean} output using 
+
+\code{attr(x, "curves")}
+
+where \code{x} is a \code{survmean} object.
+}
+\seealso{
+Other survmean functions: 
+\code{\link{Surv}()},
+\code{\link{plot.survmean}()},
+\code{\link{survmean}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survmean functions}
diff --git a/man/lines.survtab.Rd b/man/lines.survtab.Rd
index 28e95bc..835d15f 100644
--- a/man/lines.survtab.Rd
+++ b/man/lines.survtab.Rd
@@ -1,67 +1,67 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{lines.survtab}
-\alias{lines.survtab}
-\title{\code{lines} method for survtab objects}
-\usage{
-\method{lines}{survtab}(x, y = NULL, subset = NULL, conf.int = TRUE, col = NULL, lty = NULL, ...)
-}
-\arguments{
-\item{x}{a \code{survtab} output object}
-
-\item{y}{a variable to plot; a quoted name of a variable
-in \code{x}; e.g. \code{y = "surv.obs"};
-if \code{NULL}, picks last survival variable column in order in \code{x}}
-
-\item{subset}{a logical condition; \code{obj} is subset accordingly 
-before plotting; use this for limiting to specific strata, 
-e.g. \code{subset = sex == "male"}}
-
-\item{conf.int}{logical; if \code{TRUE}, also plots any confidence intervals
-present in \code{obj} for variables in \code{y}}
-
-\item{col}{line colour passed to \code{matlines}}
-
-\item{lty}{line type passed to \code{matlines}}
-
-\item{...}{additional arguments passed on to to a \code{matlines} call;
-e.g. \code{lwd} can be defined this way}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plot \code{lines} from a \code{survtab} object
-}
-\examples{
-data(sire)
-data(sibr)
-si <- rbind(sire, sibr)
-si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
-si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
-x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
-             status = status \%in\% 1:2, 
-             fot = 0:5, aggre = list(cancer, period, fot))
-st <- survtab_ag(fot ~ cancer + period, data = x, 
-                 surv.method = "lifetable", surv.type = "surv.obs")
-
-plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
-lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
-
-## or
-plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
-}
-\seealso{
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{plot.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{lines.survtab}
+\alias{lines.survtab}
+\title{\code{lines} method for survtab objects}
+\usage{
+\method{lines}{survtab}(x, y = NULL, subset = NULL, conf.int = TRUE, col = NULL, lty = NULL, ...)
+}
+\arguments{
+\item{x}{a \code{survtab} output object}
+
+\item{y}{a variable to plot; a quoted name of a variable
+in \code{x}; e.g. \code{y = "surv.obs"};
+if \code{NULL}, picks last survival variable column in order in \code{x}}
+
+\item{subset}{a logical condition; \code{obj} is subset accordingly 
+before plotting; use this for limiting to specific strata, 
+e.g. \code{subset = sex == "male"}}
+
+\item{conf.int}{logical; if \code{TRUE}, also plots any confidence intervals
+present in \code{obj} for variables in \code{y}}
+
+\item{col}{line colour passed to \code{matlines}}
+
+\item{lty}{line type passed to \code{matlines}}
+
+\item{...}{additional arguments passed on to to a \code{matlines} call;
+e.g. \code{lwd} can be defined this way}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plot \code{lines} from a \code{survtab} object
+}
+\examples{
+data(sire)
+data(sibr)
+si <- rbind(sire, sibr)
+si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
+si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
+x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
+             status = status \%in\% 1:2, 
+             fot = 0:5, aggre = list(cancer, period, fot))
+st <- survtab_ag(fot ~ cancer + period, data = x, 
+                 surv.method = "lifetable", surv.type = "surv.obs")
+
+plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
+lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
+
+## or
+plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
+}
+\seealso{
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{plot.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survtab functions}
diff --git a/man/lower_bound.Rd b/man/lower_bound.Rd
index b433bed..cd30944 100644
--- a/man/lower_bound.Rd
+++ b/man/lower_bound.Rd
@@ -1,21 +1,21 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{lower_bound}
-\alias{lower_bound}
-\title{Return lower_bound value from char string (20,30]}
-\usage{
-lower_bound(cut)
-}
-\arguments{
-\item{cut}{is a character vector of elements "(20,60]"}
-}
-\value{
-A numeric vector.
-}
-\description{
-selects lowest values of each factor after cut() based
-on that the value starts from index 2 and end in comma ",".
-}
-\author{
-Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{lower_bound}
+\alias{lower_bound}
+\title{Return lower_bound value from char string (20,30]}
+\usage{
+lower_bound(cut)
+}
+\arguments{
+\item{cut}{is a character vector of elements "(20,60]"}
+}
+\value{
+A numeric vector.
+}
+\description{
+selects lowest values of each factor after cut() based
+on that the value starts from index 2 and end in comma ",".
+}
+\author{
+Matti Rantanen
+}
diff --git a/man/ltable.Rd b/man/ltable.Rd
index 5155cf5..388aba2 100644
--- a/man/ltable.Rd
+++ b/man/ltable.Rd
@@ -1,165 +1,165 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/ltable.R
-\name{ltable}
-\alias{ltable}
-\alias{expr.by.cj}
-\title{Tabulate Counts and Other Functions by Multiple Variables into a 
-Long-Format Table}
-\usage{
-ltable(
-  data,
-  by.vars = NULL,
-  expr = list(obs = .N),
-  subset = NULL,
-  use.levels = TRUE,
-  na.rm = FALSE,
-  robust = TRUE
-)
-
-expr.by.cj(
-  data,
-  by.vars = NULL,
-  expr = list(obs = .N),
-  subset = NULL,
-  use.levels = FALSE,
-  na.rm = FALSE,
-  robust = FALSE,
-  .SDcols = NULL,
-  enclos = parent.frame(1L),
-  ...
-)
-}
-\arguments{
-\item{data}{a \code{data.table}/\code{data.frame}}
-
-\item{by.vars}{names of variables that are used for categorization, 
-as a character vector, e.g. \code{c('sex','agegroup')}}
-
-\item{expr}{object or a list of objects where each object is a function 
-of a variable (see: details)}
-
-\item{subset}{a logical condition; data is limited accordingly before
-evaluating \code{expr} - but the result of \code{expr} is also
-returned as \code{NA} for levels not existing in the subset. See Examples.}
-
-\item{use.levels}{logical; if \code{TRUE}, uses factor levels of given 
-variables if present;  if you want e.g. counts for levels
-that actually have zero observations but are levels in a factor variable, 
-use this}
-
-\item{na.rm}{logical; if \code{TRUE}, drops rows in table that have 
-\code{NA} as values in any of \code{by.vars} columns}
-
-\item{robust}{logical; if \code{TRUE}, runs the output data's 
-\code{by.vars} columns through \code{robust_values} before outputting}
-
-\item{.SDcols}{advanced; a character vector of column names 
-passed to inside the data.table's brackets 
-\code{DT[, , ...]}; see \code{\link{data.table}}; if \code{NULL},
-uses all appropriate columns. See Examples for usage.}
-
-\item{enclos}{advanced; an environment; the enclosing
-environment of the data.}
-
-\item{...}{advanced; other arguments passed to inside the 
-data.table's brackets \code{DT[, , ...]}; see \code{\link{data.table}}}
-}
-\value{
-A `data.table` of statistics (e.g. counts) stratified by the columns defined
-in `by.vars`.
-}
-\description{
-\code{ltable} makes use of \code{data.table} 
-capabilities to tabulate frequencies or 
-arbitrary functions of given variables into a long format 
-\code{data.table}/\code{data.frame}. \code{expr.by.cj} is the 
-equivalent for more advanced users.
-}
-\details{
-Returns \code{expr} for each unique combination of given \code{by.vars}.
-
-By default makes use of any and all \code{\link{levels}} present for 
-each variable in  \code{by.vars}. This is useful,
-because even if a subset of the data does not contain observations 
-for e.g. a specific age group, those age groups are 
-nevertheless presented in the resulting table; e.g. with the default 
-\code{expr = list(obs = .N)} all age group levels
-are represented by a row and can have  \code{obs = 0}.
-
-The function differs from the
-vanilla \code{\link{table}} by giving a long format table of values
-regardless of the number of \code{by.vars} given.
-Make use of e.g. \code{\link{cast_simple}} if data needs to be 
-presented in a wide format (e.g. a two-way table).
-
-The rows of the long-format table are effectively Cartesian products 
-of the levels of each variable in  \code{by.vars},
-e.g. with  \code{by.vars = c("sex", "area")} all levels of  
-\code{area} are repeated for both levels of  \code{sex}
-in the table.
-
-The \code{expr} allows the user to apply any function(s) on all 
-levels defined by  \code{by.vars}. Here are some examples:
-\itemize{
-  \item .N or list(.N) is a function used inside a \code{data.table} to 
-  calculate counts in each group
-  \item list(obs = .N), same as above but user assigned variable name
-  \item list(sum(obs), sum(pyrs), mean(dg_age)), multiple objects in a list
-  \item list(obs = sum(obs), pyrs = sum(pyrs)), same as above with user 
-  defined variable names
-}
-
-If  \code{use.levels = FALSE}, no \code{levels} information will
- be used. This means that if e.g. the  \code{agegroup}
-variable is a factor and has 18 levels defined, but only 15 levels
- are present in the data, no rows for the missing
-levels will be shown in the table.
-
-\code{na.rm} simply drops any rows from the resulting table where 
-any of the  \code{by.vars} values was \code{NA}.
-}
-\section{Functions}{
-\itemize{
-\item \code{expr.by.cj()}: Somewhat more streamlined \code{ltable} with 
-defaults for speed. Explicit determination of enclosing environment
-of data.
-
-}}
-\examples{
-data("sire", package = "popEpi")
-sr <- sire
-sr$agegroup <- cut(sr$dg_age, breaks=c(0,45,60,75,85,Inf))
-## counts by default
-ltable(sr, "agegroup")
-
-## any expression can be given
-ltable(sr, "agegroup", list(mage = mean(dg_age)))
-ltable(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
-
-## also returns levels where there are zero rows (expressions as NA)
-ltable(sr, "agegroup", list(obs = .N, 
-                            minage = min(dg_age), 
-                            maxage = max(dg_age)), 
-       subset = dg_age < 85)
-       
-#### expr.by.cj
-expr.by.cj(sr, "agegroup")
-
-## any arbitrary expression can be given
-expr.by.cj(sr, "agegroup", list(mage = mean(dg_age)))
-expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
-
-## only uses levels of by.vars present in data
-expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)), 
-           subset = dg_age < 70)
-           
-## .SDcols trick
-expr.by.cj(sr, "agegroup", lapply(.SD, mean), 
-           subset = dg_age < 70, .SDcols = c("dg_age", "status"))
-}
-\seealso{
-\code{\link{table}}, \code{\link{cast_simple}}, \code{\link{melt}}
-}
-\author{
-Joonas Miettinen, Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/ltable.R
+\name{ltable}
+\alias{ltable}
+\alias{expr.by.cj}
+\title{Tabulate Counts and Other Functions by Multiple Variables into a 
+Long-Format Table}
+\usage{
+ltable(
+  data,
+  by.vars = NULL,
+  expr = list(obs = .N),
+  subset = NULL,
+  use.levels = TRUE,
+  na.rm = FALSE,
+  robust = TRUE
+)
+
+expr.by.cj(
+  data,
+  by.vars = NULL,
+  expr = list(obs = .N),
+  subset = NULL,
+  use.levels = FALSE,
+  na.rm = FALSE,
+  robust = FALSE,
+  .SDcols = NULL,
+  enclos = parent.frame(1L),
+  ...
+)
+}
+\arguments{
+\item{data}{a \code{data.table}/\code{data.frame}}
+
+\item{by.vars}{names of variables that are used for categorization, 
+as a character vector, e.g. \code{c('sex','agegroup')}}
+
+\item{expr}{object or a list of objects where each object is a function 
+of a variable (see: details)}
+
+\item{subset}{a logical condition; data is limited accordingly before
+evaluating \code{expr} - but the result of \code{expr} is also
+returned as \code{NA} for levels not existing in the subset. See Examples.}
+
+\item{use.levels}{logical; if \code{TRUE}, uses factor levels of given 
+variables if present;  if you want e.g. counts for levels
+that actually have zero observations but are levels in a factor variable, 
+use this}
+
+\item{na.rm}{logical; if \code{TRUE}, drops rows in table that have 
+\code{NA} as values in any of \code{by.vars} columns}
+
+\item{robust}{logical; if \code{TRUE}, runs the output data's 
+\code{by.vars} columns through \code{robust_values} before outputting}
+
+\item{.SDcols}{advanced; a character vector of column names 
+passed to inside the data.table's brackets 
+\code{DT[, , ...]}; see \code{\link{data.table}}; if \code{NULL},
+uses all appropriate columns. See Examples for usage.}
+
+\item{enclos}{advanced; an environment; the enclosing
+environment of the data.}
+
+\item{...}{advanced; other arguments passed to inside the 
+data.table's brackets \code{DT[, , ...]}; see \code{\link{data.table}}}
+}
+\value{
+A `data.table` of statistics (e.g. counts) stratified by the columns defined
+in `by.vars`.
+}
+\description{
+\code{ltable} makes use of \code{data.table} 
+capabilities to tabulate frequencies or 
+arbitrary functions of given variables into a long format 
+\code{data.table}/\code{data.frame}. \code{expr.by.cj} is the 
+equivalent for more advanced users.
+}
+\details{
+Returns \code{expr} for each unique combination of given \code{by.vars}.
+
+By default makes use of any and all \code{\link{levels}} present for 
+each variable in  \code{by.vars}. This is useful,
+because even if a subset of the data does not contain observations 
+for e.g. a specific age group, those age groups are 
+nevertheless presented in the resulting table; e.g. with the default 
+\code{expr = list(obs = .N)} all age group levels
+are represented by a row and can have  \code{obs = 0}.
+
+The function differs from the
+vanilla \code{\link{table}} by giving a long format table of values
+regardless of the number of \code{by.vars} given.
+Make use of e.g. \code{\link{cast_simple}} if data needs to be 
+presented in a wide format (e.g. a two-way table).
+
+The rows of the long-format table are effectively Cartesian products 
+of the levels of each variable in  \code{by.vars},
+e.g. with  \code{by.vars = c("sex", "area")} all levels of  
+\code{area} are repeated for both levels of  \code{sex}
+in the table.
+
+The \code{expr} allows the user to apply any function(s) on all 
+levels defined by  \code{by.vars}. Here are some examples:
+\itemize{
+  \item .N or list(.N) is a function used inside a \code{data.table} to 
+  calculate counts in each group
+  \item list(obs = .N), same as above but user assigned variable name
+  \item list(sum(obs), sum(pyrs), mean(dg_age)), multiple objects in a list
+  \item list(obs = sum(obs), pyrs = sum(pyrs)), same as above with user 
+  defined variable names
+}
+
+If  \code{use.levels = FALSE}, no \code{levels} information will
+ be used. This means that if e.g. the  \code{agegroup}
+variable is a factor and has 18 levels defined, but only 15 levels
+ are present in the data, no rows for the missing
+levels will be shown in the table.
+
+\code{na.rm} simply drops any rows from the resulting table where 
+any of the  \code{by.vars} values was \code{NA}.
+}
+\section{Functions}{
+\itemize{
+\item \code{expr.by.cj()}: Somewhat more streamlined \code{ltable} with 
+defaults for speed. Explicit determination of enclosing environment
+of data.
+
+}}
+\examples{
+data("sire", package = "popEpi")
+sr <- sire
+sr$agegroup <- cut(sr$dg_age, breaks=c(0,45,60,75,85,Inf))
+## counts by default
+ltable(sr, "agegroup")
+
+## any expression can be given
+ltable(sr, "agegroup", list(mage = mean(dg_age)))
+ltable(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
+
+## also returns levels where there are zero rows (expressions as NA)
+ltable(sr, "agegroup", list(obs = .N, 
+                            minage = min(dg_age), 
+                            maxage = max(dg_age)), 
+       subset = dg_age < 85)
+       
+#### expr.by.cj
+expr.by.cj(sr, "agegroup")
+
+## any arbitrary expression can be given
+expr.by.cj(sr, "agegroup", list(mage = mean(dg_age)))
+expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)))
+
+## only uses levels of by.vars present in data
+expr.by.cj(sr, "agegroup", list(mage = mean(dg_age), vage = var(dg_age)), 
+           subset = dg_age < 70)
+           
+## .SDcols trick
+expr.by.cj(sr, "agegroup", lapply(.SD, mean), 
+           subset = dg_age < 70, .SDcols = c("dg_age", "status"))
+}
+\seealso{
+\code{\link{table}}, \code{\link{cast_simple}}, \code{\link{melt}}
+}
+\author{
+Joonas Miettinen, Matti Rantanen
+}
diff --git a/man/meanpop_fi.Rd b/man/meanpop_fi.Rd
index 6ebda6f..cb6ddf9 100644
--- a/man/meanpop_fi.Rd
+++ b/man/meanpop_fi.Rd
@@ -1,33 +1,33 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{meanpop_fi}
-\alias{meanpop_fi}
-\title{Mean population counts in Finland year, sex, and age group.}
-\format{
-\code{data.table} with columns
-\itemize{
- \item \code{sex} gender coded as male, female (0, 1)
- \item \code{year} calendar year 1981-2016
- \item \code{agegroup} - coded 0 to 100; one-year age groups
- \item \code{meanpop} the mean population count; that is, the mean of the
- annual population counts of two consecutive years; e.g. for 1990 
- \code{meanpop} is the mean of population counts for 1990 and 1991 
- (counted at 1990-01-01 and 1991-01-01, respectively)
-}
-}
-\source{
-Statistics Finland
-}
-\description{
-Mean population counts in Finland year, sex, and age group.
-}
-\seealso{
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{popmort}},
-\code{\link{sibr}},
-\code{\link{sire}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-}
-\concept{popEpi data}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{meanpop_fi}
+\alias{meanpop_fi}
+\title{Mean population counts in Finland year, sex, and age group.}
+\format{
+\code{data.table} with columns
+\itemize{
+ \item \code{sex} gender coded as male, female (0, 1)
+ \item \code{year} calendar year 1981-2016
+ \item \code{agegroup} - coded 0 to 100; one-year age groups
+ \item \code{meanpop} the mean population count; that is, the mean of the
+ annual population counts of two consecutive years; e.g. for 1990 
+ \code{meanpop} is the mean of population counts for 1990 and 1991 
+ (counted at 1990-01-01 and 1991-01-01, respectively)
+}
+}
+\source{
+Statistics Finland
+}
+\description{
+Mean population counts in Finland year, sex, and age group.
+}
+\seealso{
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{popmort}},
+\code{\link{sibr}},
+\code{\link{sire}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+}
+\concept{popEpi data}
diff --git a/man/na2zero.Rd b/man/na2zero.Rd
index 4db3bea..1aa9373 100644
--- a/man/na2zero.Rd
+++ b/man/na2zero.Rd
@@ -1,30 +1,30 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{na2zero}
-\alias{na2zero}
-\title{Convert NA's to zero in data.table}
-\usage{
-na2zero(DT, vars = NULL)
-}
-\arguments{
-\item{DT}{\code{data.table} object}
-
-\item{vars}{a character string vector of variables names in \code{DT};
-if \code{NULL}, uses all variable names in \code{DT}}
-}
-\value{
-A copy of `DT` where `NA` values have been replaced with zero.
-}
-\description{
-Given a \code{data.table DT}, replaces any \code{NA} values
-in the variables given in \code{vars} in \code{DT}. Takes a copy of the 
-original data and returns the modified copy.
-}
-\details{
-Given a \code{data.table} object, converts \code{NA} values
-to numeric (double) zeros for all variables named in \code{vars} or
-all variables if \code{vars = NULL}.
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{na2zero}
+\alias{na2zero}
+\title{Convert NA's to zero in data.table}
+\usage{
+na2zero(DT, vars = NULL)
+}
+\arguments{
+\item{DT}{\code{data.table} object}
+
+\item{vars}{a character string vector of variables names in \code{DT};
+if \code{NULL}, uses all variable names in \code{DT}}
+}
+\value{
+A copy of `DT` where `NA` values have been replaced with zero.
+}
+\description{
+Given a \code{data.table DT}, replaces any \code{NA} values
+in the variables given in \code{vars} in \code{DT}. Takes a copy of the 
+original data and returns the modified copy.
+}
+\details{
+Given a \code{data.table} object, converts \code{NA} values
+to numeric (double) zeros for all variables named in \code{vars} or
+all variables if \code{vars = NULL}.
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/plot.rate.Rd b/man/plot.rate.Rd
index 84420af..2d21872 100644
--- a/man/plot.rate.Rd
+++ b/man/plot.rate.Rd
@@ -1,37 +1,37 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{plot.rate}
-\alias{plot.rate}
-\title{plot method for rate object}
-\usage{
-\method{plot}{rate}(x, conf.int = TRUE, eps = 0.2, left.margin, xlim, ...)
-}
-\arguments{
-\item{x}{a rate object (see \code{\link{rate}})}
-
-\item{conf.int}{logical; default TRUE draws the confidence intervals}
-
-\item{eps}{is the height of the ending of the error bars}
-
-\item{left.margin}{set a custom left margin for long variable names. Function
-tries to do it by default.}
-
-\item{xlim}{change the x-axis location}
-
-\item{...}{arguments passed on to graphical functions points and segment 
-(e.g. \code{col}, \code{lwd}, \code{pch} and \code{cex})}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plot rate estimates with confidence intervals lines using R base graphics
-}
-\details{
-This is limited explanatory tool but most graphical 
-parameters are user adjustable.
-}
-\author{
-Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{plot.rate}
+\alias{plot.rate}
+\title{plot method for rate object}
+\usage{
+\method{plot}{rate}(x, conf.int = TRUE, eps = 0.2, left.margin, xlim, ...)
+}
+\arguments{
+\item{x}{a rate object (see \code{\link{rate}})}
+
+\item{conf.int}{logical; default TRUE draws the confidence intervals}
+
+\item{eps}{is the height of the ending of the error bars}
+
+\item{left.margin}{set a custom left margin for long variable names. Function
+tries to do it by default.}
+
+\item{xlim}{change the x-axis location}
+
+\item{...}{arguments passed on to graphical functions points and segment 
+(e.g. \code{col}, \code{lwd}, \code{pch} and \code{cex})}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plot rate estimates with confidence intervals lines using R base graphics
+}
+\details{
+This is limited explanatory tool but most graphical 
+parameters are user adjustable.
+}
+\author{
+Matti Rantanen
+}
diff --git a/man/plot.sir.Rd b/man/plot.sir.Rd
index 87dd5ac..3144565 100644
--- a/man/plot.sir.Rd
+++ b/man/plot.sir.Rd
@@ -1,85 +1,85 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{plot.sir}
-\alias{plot.sir}
-\title{Plot method for sir-object}
-\usage{
-\method{plot}{sir}(
-  x,
-  conf.int = TRUE,
-  ylab,
-  xlab,
-  xlim,
-  main,
-  eps = 0.2,
-  abline = TRUE,
-  log = FALSE,
-  left.margin,
-  ...
-)
-}
-\arguments{
-\item{x}{an object returned by function \code{sir}}
-
-\item{conf.int}{default TRUE draws confidence intervals}
-
-\item{ylab}{overwrites default y-axis label}
-
-\item{xlab}{overwrites default x-axis label}
-
-\item{xlim}{x-axis minimum and maximum values}
-
-\item{main}{optional plot title}
-
-\item{eps}{error bar vertical bar height (works only in 'model' or 'univariate')}
-
-\item{abline}{logical; draws a grey line in SIR = 1}
-
-\item{log}{logical; SIR is not in log scale by default}
-
-\item{left.margin}{adjust left marginal of the plot to fit long variable names}
-
-\item{...}{arguments passed on to plot(), segment and lines()}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plot SIR estimates with error bars
-}
-\details{
-Plot SIR estimates and confidence intervals 
-\itemize{
- \item univariate - plots SIR with univariate confidence intervals
- \item model - plots SIR with Poisson modelled confidence intervals
-}
-
-\strong{Customize}
-Normal plot parameters can be passed to \code{plot}. These can be a vector when plotting error bars:
-\itemize{
- \item \code{pch} - point type
- \item \code{lty} - line type
- \item \code{col} - line/point colour 
- \item \code{lwd} - point/line size
-}
-
-\strong{Tips for plotting splines}
-It's possible to use \code{plot} to first draw the 
-confidence intervals using specific line type or colour and then plotting 
-again the estimate using \code{lines(... , conf.int = FALSE)} with different 
-settings. This works only when \code{plot.type} is 'splines'.
-}
-\examples{
-\donttest{
-# Plot SIR estimates
-# plot(sir.by.gender, col = c(4,2), log=FALSE, eps=0.2, lty=1, lwd=2, pch=19,  
-#      main = 'SIR by gender', abline=TRUE)
-}
-}
-\seealso{
-\code{\link{sir}},  \code{\link{sirspline}}
-}
-\author{
-Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{plot.sir}
+\alias{plot.sir}
+\title{Plot method for sir-object}
+\usage{
+\method{plot}{sir}(
+  x,
+  conf.int = TRUE,
+  ylab,
+  xlab,
+  xlim,
+  main,
+  eps = 0.2,
+  abline = TRUE,
+  log = FALSE,
+  left.margin,
+  ...
+)
+}
+\arguments{
+\item{x}{an object returned by function \code{sir}}
+
+\item{conf.int}{default TRUE draws confidence intervals}
+
+\item{ylab}{overwrites default y-axis label}
+
+\item{xlab}{overwrites default x-axis label}
+
+\item{xlim}{x-axis minimum and maximum values}
+
+\item{main}{optional plot title}
+
+\item{eps}{error bar vertical bar height (works only in 'model' or 'univariate')}
+
+\item{abline}{logical; draws a grey line in SIR = 1}
+
+\item{log}{logical; SIR is not in log scale by default}
+
+\item{left.margin}{adjust left marginal of the plot to fit long variable names}
+
+\item{...}{arguments passed on to plot(), segment and lines()}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plot SIR estimates with error bars
+}
+\details{
+Plot SIR estimates and confidence intervals 
+\itemize{
+ \item univariate - plots SIR with univariate confidence intervals
+ \item model - plots SIR with Poisson modelled confidence intervals
+}
+
+\strong{Customize}
+Normal plot parameters can be passed to \code{plot}. These can be a vector when plotting error bars:
+\itemize{
+ \item \code{pch} - point type
+ \item \code{lty} - line type
+ \item \code{col} - line/point colour 
+ \item \code{lwd} - point/line size
+}
+
+\strong{Tips for plotting splines}
+It's possible to use \code{plot} to first draw the 
+confidence intervals using specific line type or colour and then plotting 
+again the estimate using \code{lines(... , conf.int = FALSE)} with different 
+settings. This works only when \code{plot.type} is 'splines'.
+}
+\examples{
+\donttest{
+# Plot SIR estimates
+# plot(sir.by.gender, col = c(4,2), log=FALSE, eps=0.2, lty=1, lwd=2, pch=19,  
+#      main = 'SIR by gender', abline=TRUE)
+}
+}
+\seealso{
+\code{\link{sir}},  \code{\link{sirspline}}
+}
+\author{
+Matti Rantanen
+}
diff --git a/man/plot.sirspline.Rd b/man/plot.sirspline.Rd
index f108aef..48fedff 100644
--- a/man/plot.sirspline.Rd
+++ b/man/plot.sirspline.Rd
@@ -1,53 +1,53 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{plot.sirspline}
-\alias{plot.sirspline}
-\title{\code{plot} method for sirspline-object}
-\usage{
-\method{plot}{sirspline}(x, conf.int = TRUE, abline = TRUE, log = FALSE, type, ylab, xlab, ...)
-}
-\arguments{
-\item{x}{an object returned by function sirspline}
-
-\item{conf.int}{logical; default TRUE draws also the 95 confidence intervals}
-
-\item{abline}{logical; draws a reference line where SIR = 1}
-
-\item{log}{logical; default FALSE. Should the y-axis be in log scale}
-
-\item{type}{select \code{type = 'n'} to plot only figure frames}
-
-\item{ylab}{overwrites default y-axis label; can be a vector if multiple splines fitted}
-
-\item{xlab}{overwrites default x-axis label; can be a vector if multiple splines fitted}
-
-\item{...}{arguments passed on to plot()}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plot SIR splines using R base graphics.
-}
-\details{
-In \code{plot.sirspline} almost every graphical parameter are user
-adjustable, such as \code{ylim}, \code{xlim}.
-\code{plot.sirsplines} calls \code{lines.splines} to add lines.
-
-The plot axis without lines can be plotted using option \code{type = 'n'}. 
-On top of the frame it's then possible to add a \code{grid}, 
-\code{abline} or text before plotting the lines (see: \code{sirspline}).
-}
-\seealso{
-Other sir functions: 
-\code{\link{lines.sirspline}()},
-\code{\link{sir_exp}()},
-\code{\link{sir_ratio}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()}
-}
-\author{
-Matti Rantanen
-}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{plot.sirspline}
+\alias{plot.sirspline}
+\title{\code{plot} method for sirspline-object}
+\usage{
+\method{plot}{sirspline}(x, conf.int = TRUE, abline = TRUE, log = FALSE, type, ylab, xlab, ...)
+}
+\arguments{
+\item{x}{an object returned by function sirspline}
+
+\item{conf.int}{logical; default TRUE draws also the 95 confidence intervals}
+
+\item{abline}{logical; draws a reference line where SIR = 1}
+
+\item{log}{logical; default FALSE. Should the y-axis be in log scale}
+
+\item{type}{select \code{type = 'n'} to plot only figure frames}
+
+\item{ylab}{overwrites default y-axis label; can be a vector if multiple splines fitted}
+
+\item{xlab}{overwrites default x-axis label; can be a vector if multiple splines fitted}
+
+\item{...}{arguments passed on to plot()}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plot SIR splines using R base graphics.
+}
+\details{
+In \code{plot.sirspline} almost every graphical parameter are user
+adjustable, such as \code{ylim}, \code{xlim}.
+\code{plot.sirsplines} calls \code{lines.splines} to add lines.
+
+The plot axis without lines can be plotted using option \code{type = 'n'}. 
+On top of the frame it's then possible to add a \code{grid}, 
+\code{abline} or text before plotting the lines (see: \code{sirspline}).
+}
+\seealso{
+Other sir functions: 
+\code{\link{lines.sirspline}()},
+\code{\link{sir_exp}()},
+\code{\link{sir_ratio}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()}
+}
+\author{
+Matti Rantanen
+}
+\concept{sir functions}
diff --git a/man/plot.survmean.Rd b/man/plot.survmean.Rd
index f76a750..3de7f7b 100644
--- a/man/plot.survmean.Rd
+++ b/man/plot.survmean.Rd
@@ -1,45 +1,45 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{plot.survmean}
-\alias{plot.survmean}
-\title{Graphically Inspect Curves Used in Mean Survival Computation}
-\usage{
-\method{plot}{survmean}(x, ...)
-}
-\arguments{
-\item{x}{a \code{survmean} object}
-
-\item{...}{arguments passed (ultimately) to \code{matlines}; you
-may, therefore, supply e.g. \code{xlab} through this, though arguments
-such as \code{lty} and \code{col} will not work}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plots the observed (with extrapolation) and expected survival
-curves for all strata in an object created by \code{\link{survmean}}
-}
-\details{
-For examples see \code{\link{survmean}}. This function is intended only
-for graphically inspecting that the observed survival curves with extrapolation
-and the expected survival curves have been sensibly computed in \code{survmean}.
-
-If you want finer control over the plotted curves, extract the curves from
-the \code{survmean} output using 
-
-\code{attr(x, "curves")}
-
-where \code{x} is a \code{survmean} object.
-}
-\seealso{
-Other survmean functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survmean}()},
-\code{\link{survmean}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survmean functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{plot.survmean}
+\alias{plot.survmean}
+\title{Graphically Inspect Curves Used in Mean Survival Computation}
+\usage{
+\method{plot}{survmean}(x, ...)
+}
+\arguments{
+\item{x}{a \code{survmean} object}
+
+\item{...}{arguments passed (ultimately) to \code{matlines}; you
+may, therefore, supply e.g. \code{xlab} through this, though arguments
+such as \code{lty} and \code{col} will not work}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plots the observed (with extrapolation) and expected survival
+curves for all strata in an object created by \code{\link{survmean}}
+}
+\details{
+For examples see \code{\link{survmean}}. This function is intended only
+for graphically inspecting that the observed survival curves with extrapolation
+and the expected survival curves have been sensibly computed in \code{survmean}.
+
+If you want finer control over the plotted curves, extract the curves from
+the \code{survmean} output using 
+
+\code{attr(x, "curves")}
+
+where \code{x} is a \code{survmean} object.
+}
+\seealso{
+Other survmean functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survmean}()},
+\code{\link{survmean}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survmean functions}
diff --git a/man/plot.survtab.Rd b/man/plot.survtab.Rd
index 3367655..4784a4c 100644
--- a/man/plot.survtab.Rd
+++ b/man/plot.survtab.Rd
@@ -1,80 +1,80 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{plot.survtab}
-\alias{plot.survtab}
-\title{\code{plot} method for survtab objects}
-\usage{
-\method{plot}{survtab}(
-  x,
-  y = NULL,
-  subset = NULL,
-  conf.int = TRUE,
-  col = NULL,
-  lty = NULL,
-  ylab = NULL,
-  xlab = NULL,
-  ...
-)
-}
-\arguments{
-\item{x}{a \code{survtab} output object}
-
-\item{y}{survival a character vector of a variable names to plot;
-e.g. \code{y = "r.e2"}}
-
-\item{subset}{a logical condition; \code{obj} is subset accordingly 
-before plotting; use this for limiting to specific strata, 
-e.g. \code{subset = sex == "male"}}
-
-\item{conf.int}{logical; if \code{TRUE}, also plots any confidence intervals
-present in \code{obj} for variables in \code{y}}
-
-\item{col}{line colour; one value for each stratum; will be recycled}
-
-\item{lty}{line type; one value for each stratum; will be recycled}
-
-\item{ylab}{label for Y-axis}
-
-\item{xlab}{label for X-axis}
-
-\item{...}{additional arguments passed on to \code{plot} and 
-\code{lines.survtab}; e.g. \code{ylim} can be defined this way}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Plotting for \code{survtab} objects
-}
-\examples{
-data(sire)
-data(sibr)
-si <- rbind(sire, sibr)
-si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
-si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
-x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
-             status = status \%in\% 1:2, 
-             fot = 0:5, aggre = list(cancer, period, fot))
-st <- survtab_ag(fot ~ cancer + period, data = x, 
-                 surv.method = "lifetable", surv.type = "surv.obs")
-
-plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
-lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
-
-## or
-plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
-}
-\seealso{
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{plot.survtab}
+\alias{plot.survtab}
+\title{\code{plot} method for survtab objects}
+\usage{
+\method{plot}{survtab}(
+  x,
+  y = NULL,
+  subset = NULL,
+  conf.int = TRUE,
+  col = NULL,
+  lty = NULL,
+  ylab = NULL,
+  xlab = NULL,
+  ...
+)
+}
+\arguments{
+\item{x}{a \code{survtab} output object}
+
+\item{y}{survival a character vector of a variable names to plot;
+e.g. \code{y = "r.e2"}}
+
+\item{subset}{a logical condition; \code{obj} is subset accordingly 
+before plotting; use this for limiting to specific strata, 
+e.g. \code{subset = sex == "male"}}
+
+\item{conf.int}{logical; if \code{TRUE}, also plots any confidence intervals
+present in \code{obj} for variables in \code{y}}
+
+\item{col}{line colour; one value for each stratum; will be recycled}
+
+\item{lty}{line type; one value for each stratum; will be recycled}
+
+\item{ylab}{label for Y-axis}
+
+\item{xlab}{label for X-axis}
+
+\item{...}{additional arguments passed on to \code{plot} and 
+\code{lines.survtab}; e.g. \code{ylim} can be defined this way}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Plotting for \code{survtab} objects
+}
+\examples{
+data(sire)
+data(sibr)
+si <- rbind(sire, sibr)
+si$period <- cut(si$dg_date, as.Date(c("1993-01-01", "2004-01-01", "2013-01-01")), right = FALSE)
+si$cancer <- c(rep("rectal", nrow(sire)), rep("breast", nrow(sibr)))
+x <- lexpand(si, birth = bi_date, entry = dg_date, exit = ex_date, 
+             status = status \%in\% 1:2, 
+             fot = 0:5, aggre = list(cancer, period, fot))
+st <- survtab_ag(fot ~ cancer + period, data = x, 
+                 surv.method = "lifetable", surv.type = "surv.obs")
+
+plot(st, "surv.obs", subset = cancer == "breast", ylim = c(0.5, 1), col = "blue")
+lines(st, "surv.obs", subset = cancer == "rectal", col = "red")
+
+## or
+plot(st, "surv.obs", col = c(2,2,4,4), lty = c(1, 2, 1, 2))
+}
+\seealso{
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survtab functions}
diff --git a/man/poisson.ci.Rd b/man/poisson.ci.Rd
index 9541426..89f6fb4 100644
--- a/man/poisson.ci.Rd
+++ b/man/poisson.ci.Rd
@@ -1,36 +1,36 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{poisson.ci}
-\alias{poisson.ci}
-\title{Get rate and exact Poisson confidence intervals}
-\usage{
-poisson.ci(x, pt = 1, conf.level = 0.95)
-}
-\arguments{
-\item{x}{observed}
-
-\item{pt}{expected}
-
-\item{conf.level}{alpha level}
-}
-\value{
-A \code{data.frame} with columns
-\itemize{
-\item \code{x}: arg \code{x}
-\item \code{pt}: arg \code{pt}
-\item \code{rate}: result of \code{x / pt}
-\item \code{lower}: lower bound of CI
-\item \code{upper}: upper bound of CI
-\item \code{conf.level}: arg \code{conf.level}
-}
-}
-\description{
-Computes confidence intervals for Poisson rates
-}
-\examples{
-
-poisson.ci(x = 4, pt = 5, conf.level = 0.95)
-}
-\author{
-epitools
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{poisson.ci}
+\alias{poisson.ci}
+\title{Get rate and exact Poisson confidence intervals}
+\usage{
+poisson.ci(x, pt = 1, conf.level = 0.95)
+}
+\arguments{
+\item{x}{observed}
+
+\item{pt}{expected}
+
+\item{conf.level}{alpha level}
+}
+\value{
+A \code{data.frame} with columns
+\itemize{
+\item \code{x}: arg \code{x}
+\item \code{pt}: arg \code{pt}
+\item \code{rate}: result of \code{x / pt}
+\item \code{lower}: lower bound of CI
+\item \code{upper}: upper bound of CI
+\item \code{conf.level}: arg \code{conf.level}
+}
+}
+\description{
+Computes confidence intervals for Poisson rates
+}
+\examples{
+
+poisson.ci(x = 4, pt = 5, conf.level = 0.95)
+}
+\author{
+epitools
+}
diff --git a/man/popEpi.Rd b/man/popEpi.Rd
index 7770514..ccff5fe 100644
--- a/man/popEpi.Rd
+++ b/man/popEpi.Rd
@@ -1,37 +1,37 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/popEpi_package.r
-\docType{package}
-\name{popEpi}
-\alias{popEpi}
-\title{popEpi: Functions for large-scale epidemiological analysis}
-\description{
-\pkg{popEpi} is built for the needs of registry-based (large-scale)
-epidemiological analysis. This is in most part enabled by the 
-efficient \pkg{data.table} package for handling and aggregating large data sets. 
-
-\pkg{popEpi} currently supplies some utility functions such as \code{\link{splitMulti}}
-and \code{\link{get.yrs}} for preparing large data sets for epidemiological analysis.
-Included are also a a few functions that can be used in 
-epidemiological analysis such as \code{\link{sir}} for estimating
-standardized incidence/mortality ratios (SIRs/SMRs) and \code{\link{survtab}} for 
-estimating observed and relative/net survival as well as cumulative incidence
-functions (CIFs). In particular, \code{survtab} implements the Ederer II 
-(Ederer and Heise (1959)) and
-Pohar Perme estimators (Pohar Perme, Stare, and Esteve (2012) 
-\doi{10.1111/j.1541-0420.2011.01640.x}) and allows for easy 
-age-standardisation.
-
-Since there are many benefits to using \code{data.tables}, \pkg{popEpi} returns
-outputs by default in the \code{data.table} format where appropriate. 
-Since \code{data.table}
-objects are usually modified by reference, this may have surprising side 
-effects for users uninitiated in using \code{data.table}. To ensure
-that appropriate outputs are in the \code{data.frame} format, set
-\code{options("popEpi.datatable" = FALSE)}. However, \code{data.table}
-usage is recommended due to better performance and testing coverage. 
-\code{data.table} is used
-by most functions internally in both cases.
-}
-\details{
-popEpi
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/popEpi_package.r
+\docType{package}
+\name{popEpi}
+\alias{popEpi}
+\title{popEpi: Functions for large-scale epidemiological analysis}
+\description{
+\pkg{popEpi} is built for the needs of registry-based (large-scale)
+epidemiological analysis. This is in most part enabled by the 
+efficient \pkg{data.table} package for handling and aggregating large data sets. 
+
+\pkg{popEpi} currently supplies some utility functions such as \code{\link{splitMulti}}
+and \code{\link{get.yrs}} for preparing large data sets for epidemiological analysis.
+Included are also a a few functions that can be used in 
+epidemiological analysis such as \code{\link{sir}} for estimating
+standardized incidence/mortality ratios (SIRs/SMRs) and \code{\link{survtab}} for 
+estimating observed and relative/net survival as well as cumulative incidence
+functions (CIFs). In particular, \code{survtab} implements the Ederer II 
+(Ederer and Heise (1959)) and
+Pohar Perme estimators (Pohar Perme, Stare, and Esteve (2012) 
+\doi{10.1111/j.1541-0420.2011.01640.x}) and allows for easy 
+age-standardisation.
+
+Since there are many benefits to using \code{data.tables}, \pkg{popEpi} returns
+outputs by default in the \code{data.table} format where appropriate. 
+Since \code{data.table}
+objects are usually modified by reference, this may have surprising side 
+effects for users uninitiated in using \code{data.table}. To ensure
+that appropriate outputs are in the \code{data.frame} format, set
+\code{options("popEpi.datatable" = FALSE)}. However, \code{data.table}
+usage is recommended due to better performance and testing coverage. 
+\code{data.table} is used
+by most functions internally in both cases.
+}
+\details{
+popEpi
+}
diff --git a/man/pophaz.Rd b/man/pophaz.Rd
index aa12e06..490f1f5 100644
--- a/man/pophaz.Rd
+++ b/man/pophaz.Rd
@@ -1,59 +1,59 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/pophaz.R
-\name{pophaz}
-\alias{pophaz}
-\title{Expected / Population Hazard Data Sets Usage in \pkg{popEpi}}
-\description{
-Several functions in \pkg{popEpi} make use of population or expected
-hazards in computing the intended estimates (e.g. \code{\link{survtab}}).
-This document explains using such data sets in this package.
-}
-\details{
-Population hazard data sets (pophaz for short) in \pkg{popEpi} should
-be \code{data.frame}s in the "long" format where one of the columns must be
-named \code{haz} (for hazard), and other columns define the values or 
-levels in variables relating to subjects in your data. For example,
-\code{\link{popmort}} contains Finnish population mortality hazards
-by sex, calendar year, and 1-year age group.
-
-\tabular{rrrr}{
-\code{sex} \tab \code{year} \tab \code{agegroup} \tab \code{haz} \cr
-0 \tab 1951 \tab 0 \tab 0.036363176\cr  
-0 \tab 1951 \tab 1 \tab 0.003616547\cr
-0 \tab 1951 \tab 2 \tab 0.002172384\cr
-0 \tab 1951 \tab 3 \tab 0.001581249\cr
-0 \tab 1951 \tab 4 \tab 0.001180690\cr
-0 \tab 1951 \tab 5 \tab 0.001070595
-}
-
-The names of the columns should match to the names of the variables
-that you have in your subject-level data. Time variables in your pophaz
-may also correspond to \code{Lexis} time scales; see 
-\code{\link{survtab}}.
-
-Any time variables (as they usually have) should be coded consistently:
-When using fractional years in your data, the time variables in your pophaz
-must also be coded in fractional years. When using e.g. \code{Date}s in your
-data, ensure that the pophaz time variables are coded at the level of days
-(or \code{Date}s for calendar time). 
-
-The \code{haz} variable in your pophaz should also be coded consistently
-with the used time variables. E.g. \code{haz} values in life-tables
-reported as deaths per person-year should be multiplied by 365.25 when
-using day-level time variables. Typically you'll have calendar time and age
-expressed in years, which means \code{haz} should be expressed as the number
-of deaths per person-year.
-
-If you have your population hazards in a \code{ratetable} object
-usable by functions in \pkg{survival} and \pkg{relsurv}, you may
-transform them to long-format \code{data.frame}s using
-\code{\link{ratetable_to_long_dt}}. Ensure, however, that the
-created \code{haz} column is coded at the right level (events per
-days or years typically).
-
-National statistical institutions, the WHO, and e.g. the Human
-Life-Table Database supply life-table data.
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/pophaz.R
+\name{pophaz}
+\alias{pophaz}
+\title{Expected / Population Hazard Data Sets Usage in \pkg{popEpi}}
+\description{
+Several functions in \pkg{popEpi} make use of population or expected
+hazards in computing the intended estimates (e.g. \code{\link{survtab}}).
+This document explains using such data sets in this package.
+}
+\details{
+Population hazard data sets (pophaz for short) in \pkg{popEpi} should
+be \code{data.frame}s in the "long" format where one of the columns must be
+named \code{haz} (for hazard), and other columns define the values or 
+levels in variables relating to subjects in your data. For example,
+\code{\link{popmort}} contains Finnish population mortality hazards
+by sex, calendar year, and 1-year age group.
+
+\tabular{rrrr}{
+\code{sex} \tab \code{year} \tab \code{agegroup} \tab \code{haz} \cr
+0 \tab 1951 \tab 0 \tab 0.036363176\cr  
+0 \tab 1951 \tab 1 \tab 0.003616547\cr
+0 \tab 1951 \tab 2 \tab 0.002172384\cr
+0 \tab 1951 \tab 3 \tab 0.001581249\cr
+0 \tab 1951 \tab 4 \tab 0.001180690\cr
+0 \tab 1951 \tab 5 \tab 0.001070595
+}
+
+The names of the columns should match to the names of the variables
+that you have in your subject-level data. Time variables in your pophaz
+may also correspond to \code{Lexis} time scales; see 
+\code{\link{survtab}}.
+
+Any time variables (as they usually have) should be coded consistently:
+When using fractional years in your data, the time variables in your pophaz
+must also be coded in fractional years. When using e.g. \code{Date}s in your
+data, ensure that the pophaz time variables are coded at the level of days
+(or \code{Date}s for calendar time). 
+
+The \code{haz} variable in your pophaz should also be coded consistently
+with the used time variables. E.g. \code{haz} values in life-tables
+reported as deaths per person-year should be multiplied by 365.25 when
+using day-level time variables. Typically you'll have calendar time and age
+expressed in years, which means \code{haz} should be expressed as the number
+of deaths per person-year.
+
+If you have your population hazards in a \code{ratetable} object
+usable by functions in \pkg{survival} and \pkg{relsurv}, you may
+transform them to long-format \code{data.frame}s using
+\code{\link{ratetable_to_long_dt}}. Ensure, however, that the
+created \code{haz} column is coded at the right level (events per
+days or years typically).
+
+National statistical institutions, the WHO, and e.g. the Human
+Life-Table Database supply life-table data.
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/popmort.Rd b/man/popmort.Rd
index 1c9a7cd..f0fe7dd 100644
--- a/man/popmort.Rd
+++ b/man/popmort.Rd
@@ -1,37 +1,37 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{popmort}
-\alias{popmort}
-\title{Population mortality rates in Finland 1951 - 2013 in 101 age groups and
-by gender. This is an example of a population hazard table as used in 
-\pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.}
-\format{
-\code{data.table} with columns
-\itemize{
- \item \code{sex} gender coded as male, female (0, 1)
- \item \code{year} calendar year
- \item \code{agegroup} - coded 0 to 100; one-year age groups
- \item \code{haz} the average population mortality rate per person-year 
- (d/(pyrs), where d is the number of deaths and pyrs is the person-years)
-}
-}
-\source{
-Statistics Finland
-}
-\description{
-Population mortality rates in Finland 1951 - 2013 in 101 age groups and
-by gender. This is an example of a population hazard table as used in 
-\pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.
-}
-\seealso{
-\code{\link{pophaz}}
-
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{meanpop_fi}},
-\code{\link{sibr}},
-\code{\link{sire}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-}
-\concept{popEpi data}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{popmort}
+\alias{popmort}
+\title{Population mortality rates in Finland 1951 - 2013 in 101 age groups and
+by gender. This is an example of a population hazard table as used in 
+\pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.}
+\format{
+\code{data.table} with columns
+\itemize{
+ \item \code{sex} gender coded as male, female (0, 1)
+ \item \code{year} calendar year
+ \item \code{agegroup} - coded 0 to 100; one-year age groups
+ \item \code{haz} the average population mortality rate per person-year 
+ (d/(pyrs), where d is the number of deaths and pyrs is the person-years)
+}
+}
+\source{
+Statistics Finland
+}
+\description{
+Population mortality rates in Finland 1951 - 2013 in 101 age groups and
+by gender. This is an example of a population hazard table as used in 
+\pkg{popEpi}; for the general help page, see \code{\link{pophaz}}.
+}
+\seealso{
+\code{\link{pophaz}}
+
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{meanpop_fi}},
+\code{\link{sibr}},
+\code{\link{sire}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+}
+\concept{popEpi data}
diff --git a/man/prepExpo.Rd b/man/prepExpo.Rd
index 4184410..cc8f554 100644
--- a/man/prepExpo.Rd
+++ b/man/prepExpo.Rd
@@ -1,114 +1,114 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/splitting_utility_functions.R
-\name{prepExpo}
-\alias{prepExpo}
-\title{Prepare Exposure Data for Aggregation}
-\usage{
-prepExpo(
-  lex,
-  freezeScales = "work",
-  cutScale = "per",
-  entry = min(get(cutScale)),
-  exit = max(get(cutScale)),
-  by = "lex.id",
-  breaks = NULL,
-  freezeDummy = NULL,
-  subset = NULL,
-  verbose = FALSE,
-  ...
-)
-}
-\arguments{
-\item{lex}{a \code{\link[Epi]{Lexis}} object with ONLY periods of exposure
-as rows; one or multiple rows per subject allowed}
-
-\item{freezeScales}{a character vector naming \code{Lexis} time scales of exposure
-which should be frozen in periods where no exposure occurs (in the gap
-time periods)}
-
-\item{cutScale}{the \code{Lexis} time scale along which the subject-specific
-ultimate entry and exit times are specified}
-
-\item{entry}{an expression; the time of entry to follow-up which may be earlier, at, or after
-the first time of exposure in \code{freezeScales}; evaluated separately
-for each unique combination of \code{by}, so e.g. with 
-\code{entry = min(Var1)} and \code{by = "lex.id"} it 
-sets the \code{lex.id}-specific minima of \code{Var1} to be the original times
-of entry for each \code{lex.id}}
-
-\item{exit}{the same as \code{entry} but for the ultimate exit time per unique
-combination of \code{by}}
-
-\item{by}{a character vector indicating variable names in \code{lex},
-the unique combinations of which identify separate subjects for which
-to fill gaps in the records from \code{entry} to \code{exit};
-for novices of \code{{\link{data.table}}}, this is passed to a 
-\code{data.table}'s \code{by} argument.}
-
-\item{breaks}{a named list of breaks; 
-e.g. \code{list(work = 0:20,per = 1995:2015)}; passed on to 
-\code{\link{splitMulti}} so see that function's help for more details}
-
-\item{freezeDummy}{a character string; specifies the name for a dummy variable
-that this function will create and add to output which 
-identifies rows where the \code{freezeScales} are frozen and where not
-(\code{0} implies not frozen, \code{1} implies frozen);
-if \code{NULL}, no dummy is created}
-
-\item{subset}{a logical condition to subset data by before computations;
-e.g. \code{subset = sex == "male"}}
-
-\item{verbose}{logical; if \code{TRUE}, the function is chatty and returns
-some messages and timings during its run.}
-
-\item{...}{additional arguments passed on to \code{\link{splitMulti}}}
-}
-\value{
-Returns a \code{Lexis} object that has been split if \code{breaks} is specified.
-The resulting time is also a \code{data.table} if 
-\code{options("popEpi.datatable") == TRUE} (see: \code{?popEpi})
-}
-\description{
-\code{prepExpo} uses a \code{Lexis} object of periods of exposure
-to fill gaps between the periods and overall entry and exit times without
-accumulating exposure time in periods of no exposure, and splits the
-result if requested.
-}
-\details{
-\code{prepExpo} is a convenience function for the purpose of eventually aggregating 
-person-time and events in categories of not only normally progressing 
-\code{Lexis} time scales but also some time scales which should not
-progress sometimes. For example a person may work at a production facility
-only intermittently, meaning exposure time (to work-related substances 
-for example) should not progress outside of periods of work. This allows for
-e.g. a correct aggregation of person-time and events by categories of cumulative
-time of exposure.
-
-Given a \code{Lexis} object containing rows (time lines)
-where a subject is exposed to something (and NO periods without exposure),
-fills any gaps between exposure periods for each unique combination of \code{by}
-and the subject-specific "ultimate" \code{entry} and \code{exit} times,
-"freezes" the cumulative exposure times in periods of no exposure,
-and splits data using \code{breaks} passed to \code{\link{splitMulti}}
-if requested. Results in a (split) \code{Lexis} object where \code{freezeScales}
-do not progress in time periods where no exposure was recorded in \code{lex}.
-
-This function assumes that \code{entry} and \code{exit} arguments are the
-same for each row within a unique combination of variables named in \code{by}.
-E.g. with \code{by = "lex.id"} only each \code{lex.id} has a unique value
-for \code{entry} and \code{exit} at most.
-
-The supplied \code{breaks} split the data using \code{splitMulti}, with
-the exception that breaks supplied concerning any frozen time scales
-ONLY split the rows where the time scales are not frozen. E.g.
-with \code{freezeScales = "work"}, 
-\code{breaks = list(work = 0:10, cal = 1995:2010)} splits all rows over
-\code{"cal"} but only non-frozen rows over \code{"work"}.
-
-Only supports frozen time scales that advance and freeze contemporaneously:
-e.g. it would not currently be possible to take into account the cumulative
-time working at a facility and the cumulative time doing a single task
-at the facility, if the two are not exactly the same. On the other hand
-one might use the same time scale for different exposure types, supply them
-as separate rows, and identify the different exposures using a dummy variable.
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/splitting_utility_functions.R
+\name{prepExpo}
+\alias{prepExpo}
+\title{Prepare Exposure Data for Aggregation}
+\usage{
+prepExpo(
+  lex,
+  freezeScales = "work",
+  cutScale = "per",
+  entry = min(get(cutScale)),
+  exit = max(get(cutScale)),
+  by = "lex.id",
+  breaks = NULL,
+  freezeDummy = NULL,
+  subset = NULL,
+  verbose = FALSE,
+  ...
+)
+}
+\arguments{
+\item{lex}{a \code{\link[Epi]{Lexis}} object with ONLY periods of exposure
+as rows; one or multiple rows per subject allowed}
+
+\item{freezeScales}{a character vector naming \code{Lexis} time scales of exposure
+which should be frozen in periods where no exposure occurs (in the gap
+time periods)}
+
+\item{cutScale}{the \code{Lexis} time scale along which the subject-specific
+ultimate entry and exit times are specified}
+
+\item{entry}{an expression; the time of entry to follow-up which may be earlier, at, or after
+the first time of exposure in \code{freezeScales}; evaluated separately
+for each unique combination of \code{by}, so e.g. with 
+\code{entry = min(Var1)} and \code{by = "lex.id"} it 
+sets the \code{lex.id}-specific minima of \code{Var1} to be the original times
+of entry for each \code{lex.id}}
+
+\item{exit}{the same as \code{entry} but for the ultimate exit time per unique
+combination of \code{by}}
+
+\item{by}{a character vector indicating variable names in \code{lex},
+the unique combinations of which identify separate subjects for which
+to fill gaps in the records from \code{entry} to \code{exit};
+for novices of \code{{\link{data.table}}}, this is passed to a 
+\code{data.table}'s \code{by} argument.}
+
+\item{breaks}{a named list of breaks; 
+e.g. \code{list(work = 0:20,per = 1995:2015)}; passed on to 
+\code{\link{splitMulti}} so see that function's help for more details}
+
+\item{freezeDummy}{a character string; specifies the name for a dummy variable
+that this function will create and add to output which 
+identifies rows where the \code{freezeScales} are frozen and where not
+(\code{0} implies not frozen, \code{1} implies frozen);
+if \code{NULL}, no dummy is created}
+
+\item{subset}{a logical condition to subset data by before computations;
+e.g. \code{subset = sex == "male"}}
+
+\item{verbose}{logical; if \code{TRUE}, the function is chatty and returns
+some messages and timings during its run.}
+
+\item{...}{additional arguments passed on to \code{\link{splitMulti}}}
+}
+\value{
+Returns a \code{Lexis} object that has been split if \code{breaks} is specified.
+The resulting time is also a \code{data.table} if 
+\code{options("popEpi.datatable") == TRUE} (see: \code{?popEpi})
+}
+\description{
+\code{prepExpo} uses a \code{Lexis} object of periods of exposure
+to fill gaps between the periods and overall entry and exit times without
+accumulating exposure time in periods of no exposure, and splits the
+result if requested.
+}
+\details{
+\code{prepExpo} is a convenience function for the purpose of eventually aggregating 
+person-time and events in categories of not only normally progressing 
+\code{Lexis} time scales but also some time scales which should not
+progress sometimes. For example a person may work at a production facility
+only intermittently, meaning exposure time (to work-related substances 
+for example) should not progress outside of periods of work. This allows for
+e.g. a correct aggregation of person-time and events by categories of cumulative
+time of exposure.
+
+Given a \code{Lexis} object containing rows (time lines)
+where a subject is exposed to something (and NO periods without exposure),
+fills any gaps between exposure periods for each unique combination of \code{by}
+and the subject-specific "ultimate" \code{entry} and \code{exit} times,
+"freezes" the cumulative exposure times in periods of no exposure,
+and splits data using \code{breaks} passed to \code{\link{splitMulti}}
+if requested. Results in a (split) \code{Lexis} object where \code{freezeScales}
+do not progress in time periods where no exposure was recorded in \code{lex}.
+
+This function assumes that \code{entry} and \code{exit} arguments are the
+same for each row within a unique combination of variables named in \code{by}.
+E.g. with \code{by = "lex.id"} only each \code{lex.id} has a unique value
+for \code{entry} and \code{exit} at most.
+
+The supplied \code{breaks} split the data using \code{splitMulti}, with
+the exception that breaks supplied concerning any frozen time scales
+ONLY split the rows where the time scales are not frozen. E.g.
+with \code{freezeScales = "work"}, 
+\code{breaks = list(work = 0:10, cal = 1995:2010)} splits all rows over
+\code{"cal"} but only non-frozen rows over \code{"work"}.
+
+Only supports frozen time scales that advance and freeze contemporaneously:
+e.g. it would not currently be possible to take into account the cumulative
+time working at a facility and the cumulative time doing a single task
+at the facility, if the two are not exactly the same. On the other hand
+one might use the same time scale for different exposure types, supply them
+as separate rows, and identify the different exposures using a dummy variable.
+}
diff --git a/man/print.aggre.Rd b/man/print.aggre.Rd
index e78f0da..339ad44 100644
--- a/man/print.aggre.Rd
+++ b/man/print.aggre.Rd
@@ -1,31 +1,31 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{print.aggre}
-\alias{print.aggre}
-\title{Print an \code{aggre} Object}
-\usage{
-\method{print}{aggre}(x, subset = NULL, ...)
-}
-\arguments{
-\item{x}{an \code{aggre} object}
-
-\item{subset}{a logical condition to subset results table by
-before printing; use this to limit to a certain stratum. E.g.
-\code{subset = sex == "male"}}
-
-\item{...}{arguments passed to \code{print.data.table}; try e.g.
-\code{top = 2} for numbers of rows in head and tail printed 
-if the table is large, 
-\code{nrow = 100} for number of rows to print, etc.}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Print method function for \code{aggre} objects; see
-\code{\link{as.aggre}} and \code{\link{aggre}}.
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{print.aggre}
+\alias{print.aggre}
+\title{Print an \code{aggre} Object}
+\usage{
+\method{print}{aggre}(x, subset = NULL, ...)
+}
+\arguments{
+\item{x}{an \code{aggre} object}
+
+\item{subset}{a logical condition to subset results table by
+before printing; use this to limit to a certain stratum. E.g.
+\code{subset = sex == "male"}}
+
+\item{...}{arguments passed to \code{print.data.table}; try e.g.
+\code{top = 2} for numbers of rows in head and tail printed 
+if the table is large, 
+\code{nrow = 100} for number of rows to print, etc.}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Print method function for \code{aggre} objects; see
+\code{\link{as.aggre}} and \code{\link{aggre}}.
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/print.rate.Rd b/man/print.rate.Rd
index 8ffebdc..46820e9 100644
--- a/man/print.rate.Rd
+++ b/man/print.rate.Rd
@@ -1,28 +1,28 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{print.rate}
-\alias{print.rate}
-\title{Print an rate object}
-\usage{
-\method{print}{rate}(x, subset = NULL, ...)
-}
-\arguments{
-\item{x}{an \code{rate} object}
-
-\item{subset}{a logical condition to subset results table by
-before printing; use this to limit to a certain stratum. E.g.
-\code{subset = sex == "female"}}
-
-\item{...}{arguments for data.tables print method, e.g. row.names = FALSE suppresses row numbers.}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Print method function for \code{rate} objects; see
-\code{\link{rate}}.
-}
-\author{
-Matti Rantanen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{print.rate}
+\alias{print.rate}
+\title{Print an rate object}
+\usage{
+\method{print}{rate}(x, subset = NULL, ...)
+}
+\arguments{
+\item{x}{an \code{rate} object}
+
+\item{subset}{a logical condition to subset results table by
+before printing; use this to limit to a certain stratum. E.g.
+\code{subset = sex == "female"}}
+
+\item{...}{arguments for data.tables print method, e.g. row.names = FALSE suppresses row numbers.}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Print method function for \code{rate} objects; see
+\code{\link{rate}}.
+}
+\author{
+Matti Rantanen
+}
diff --git a/man/print.survtab.Rd b/man/print.survtab.Rd
index 9f0354d..81bbe70 100644
--- a/man/print.survtab.Rd
+++ b/man/print.survtab.Rd
@@ -1,41 +1,41 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{print.survtab}
-\alias{print.survtab}
-\title{Print a survtab Object}
-\usage{
-\method{print}{survtab}(x, subset = NULL, ...)
-}
-\arguments{
-\item{x}{a \code{survtab} object}
-
-\item{subset}{a logical condition to subset results table by
-before printing; use this to limit to a certain stratum. E.g.
-\code{subset = sex == "male"}}
-
-\item{...}{arguments passed to \code{print.data.table}; try e.g.
-\code{top = 2} for numbers of rows in head and tail printed 
-if the table is large, 
-\code{nrow = 100} for number of rows to print, etc.}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Print method function for \code{survtab} objects; see
-\code{\link{survtab_ag}}.
-}
-\seealso{
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survtab}()},
-\code{\link{plot.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{print.survtab}
+\alias{print.survtab}
+\title{Print a survtab Object}
+\usage{
+\method{print}{survtab}(x, subset = NULL, ...)
+}
+\arguments{
+\item{x}{a \code{survtab} object}
+
+\item{subset}{a logical condition to subset results table by
+before printing; use this to limit to a certain stratum. E.g.
+\code{subset = sex == "male"}}
+
+\item{...}{arguments passed to \code{print.data.table}; try e.g.
+\code{top = 2} for numbers of rows in head and tail printed 
+if the table is large, 
+\code{nrow = 100} for number of rows to print, etc.}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Print method function for \code{survtab} objects; see
+\code{\link{survtab_ag}}.
+}
+\seealso{
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survtab}()},
+\code{\link{plot.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survtab functions}
diff --git a/man/rate.Rd b/man/rate.Rd
index e1bbd17..abee7c5 100644
--- a/man/rate.Rd
+++ b/man/rate.Rd
@@ -1,123 +1,123 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/incidence_rates.R
-\name{rate}
-\alias{rate}
-\title{Direct-Standardised Incidence/Mortality Rates}
-\usage{
-rate(
-  data,
-  obs = NULL,
-  pyrs = NULL,
-  print = NULL,
-  adjust = NULL,
-  weights = NULL,
-  subset = NULL
-)
-}
-\arguments{
-\item{data}{aggregated data (see e.g. \code{\link{lexpand}}, 
-\code{\link{aggre}} if you have subject-level data)}
-
-\item{obs}{observations variable name in data.
-\link[=flexible_argument]{Flexible input}, typically e.g.
-\code{obs = obs}.}
-
-\item{pyrs}{person-years variable name in data.
-\link[=flexible_argument]{Flexible input}, typically e.g.
-\code{pyrs = pyrs}.}
-
-\item{print}{variable name to stratify the rates.
-\link[=flexible_argument]{Flexible input}, typically e.g.
-\code{print = sex} or \code{print = list(sex, area)}.}
-
-\item{adjust}{variable for adjusting the rates.
-\link[=flexible_argument]{Flexible input}, typically e.g.
-\code{adjust = agegroup}.}
-
-\item{weights}{typically a list of weights or a \code{character} string
-specifying an age group standardization scheme; see
-the \link[=direct_standardization]{dedicated help page} 
-and examples.}
-
-\item{subset}{a logical expression to subset data.}
-}
-\value{
-Returns a \code{data.table} with observations, person-years, rates and
-adjusted rates, if available. Results are stratified by \code{print}.
-Adjusted rates are identified with suffix \code{.adj} and  
-\code{.lo} and \code{.hi} are for confidence intervals lower and upper 
-95\% bounds, respectively.
-The prefix \code{SE.} stands for standard error.
-}
-\description{
-\code{rate} calculates adjusted rates using
-preloaded weights data or user specified weights.
-}
-\details{
-Input data needs to be in aggregated format with observations 
-and person-years. For individual data use \code{\link{lexpand}}, or
-\code{\link{ltable}} and merge person-years manually.
-
-The confidence intervals are based on the normal approximation of the logarithm of the rate.
-The variance of the log rate that is used to derive the confidence intervals 
-is derived using the delta method.
-}
-\examples{
-## Prepare data with lexpand and then reformat agegroup.
-data(sibr)
-x <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,  
-             breaks = list(per = c(1990,2000,2010,2020), age = c(0:17*5,Inf)),
-             aggre = list(agegroup = age, year.cat = per),
-             status =  status != 0)
-
-x$agegroup <- cut(x$agegroup,  c(0:17*5,Inf), right = FALSE)
-
-## calculate rates for selected periods with Nordic 2000 weights:
-r1 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-            adjust = agegroup, weights = 'nordic')
-r1
-
-## use total person-years by stratum as weights (some have zero)
-w <- ltable(x, by.vars = "agegroup", expr = sum(pyrs))
-w[is.na(w$V1),]$V1 <- 0
-
-r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-            adjust = agegroup,
-            weights = w$V1)
-r2
-
-## use data.frame of weights:
-names(w) <- c("agegroup", "weights")
-r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-            adjust = agegroup,
-            weights = w)
-r2
-
-## internal weights (same result as above)
-r3 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
-            adjust = agegroup,
-            weights = "internal")
-r3
-
-}
-\seealso{
-\code{\link{lexpand}}, \code{\link{ltable}}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-
-Other rate functions: 
-\code{\link{rate_ratio}()}
-}
-\author{
-Matti Rantanen, Joonas Miettinen
-}
-\concept{main functions}
-\concept{rate functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/incidence_rates.R
+\name{rate}
+\alias{rate}
+\title{Direct-Standardised Incidence/Mortality Rates}
+\usage{
+rate(
+  data,
+  obs = NULL,
+  pyrs = NULL,
+  print = NULL,
+  adjust = NULL,
+  weights = NULL,
+  subset = NULL
+)
+}
+\arguments{
+\item{data}{aggregated data (see e.g. \code{\link{lexpand}}, 
+\code{\link{aggre}} if you have subject-level data)}
+
+\item{obs}{observations variable name in data.
+\link[=flexible_argument]{Flexible input}, typically e.g.
+\code{obs = obs}.}
+
+\item{pyrs}{person-years variable name in data.
+\link[=flexible_argument]{Flexible input}, typically e.g.
+\code{pyrs = pyrs}.}
+
+\item{print}{variable name to stratify the rates.
+\link[=flexible_argument]{Flexible input}, typically e.g.
+\code{print = sex} or \code{print = list(sex, area)}.}
+
+\item{adjust}{variable for adjusting the rates.
+\link[=flexible_argument]{Flexible input}, typically e.g.
+\code{adjust = agegroup}.}
+
+\item{weights}{typically a list of weights or a \code{character} string
+specifying an age group standardization scheme; see
+the \link[=direct_standardization]{dedicated help page} 
+and examples.}
+
+\item{subset}{a logical expression to subset data.}
+}
+\value{
+Returns a \code{data.table} with observations, person-years, rates and
+adjusted rates, if available. Results are stratified by \code{print}.
+Adjusted rates are identified with suffix \code{.adj} and  
+\code{.lo} and \code{.hi} are for confidence intervals lower and upper 
+95\% bounds, respectively.
+The prefix \code{SE.} stands for standard error.
+}
+\description{
+\code{rate} calculates adjusted rates using
+preloaded weights data or user specified weights.
+}
+\details{
+Input data needs to be in aggregated format with observations 
+and person-years. For individual data use \code{\link{lexpand}}, or
+\code{\link{ltable}} and merge person-years manually.
+
+The confidence intervals are based on the normal approximation of the logarithm of the rate.
+The variance of the log rate that is used to derive the confidence intervals 
+is derived using the delta method.
+}
+\examples{
+## Prepare data with lexpand and then reformat agegroup.
+data(sibr)
+x <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,  
+             breaks = list(per = c(1990,2000,2010,2020), age = c(0:17*5,Inf)),
+             aggre = list(agegroup = age, year.cat = per),
+             status =  status != 0)
+
+x$agegroup <- cut(x$agegroup,  c(0:17*5,Inf), right = FALSE)
+
+## calculate rates for selected periods with Nordic 2000 weights:
+r1 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+            adjust = agegroup, weights = 'nordic')
+r1
+
+## use total person-years by stratum as weights (some have zero)
+w <- ltable(x, by.vars = "agegroup", expr = sum(pyrs))
+w[is.na(w$V1),]$V1 <- 0
+
+r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+            adjust = agegroup,
+            weights = w$V1)
+r2
+
+## use data.frame of weights:
+names(w) <- c("agegroup", "weights")
+r2 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+            adjust = agegroup,
+            weights = w)
+r2
+
+## internal weights (same result as above)
+r3 <- rate( data = x, obs = from0to1, pyrs = pyrs, print = year.cat, 
+            adjust = agegroup,
+            weights = "internal")
+r3
+
+}
+\seealso{
+\code{\link{lexpand}}, \code{\link{ltable}}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+
+Other rate functions: 
+\code{\link{rate_ratio}()}
+}
+\author{
+Matti Rantanen, Joonas Miettinen
+}
+\concept{main functions}
+\concept{rate functions}
diff --git a/man/rate_ratio.Rd b/man/rate_ratio.Rd
index 58a1433..bd122a6 100644
--- a/man/rate_ratio.Rd
+++ b/man/rate_ratio.Rd
@@ -1,72 +1,72 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/incidence_rates_utils.R
-\name{rate_ratio}
-\alias{rate_ratio}
-\title{Confidence intervals for the rate ratios}
-\usage{
-rate_ratio(x, y, crude = FALSE, SE.method = TRUE)
-}
-\arguments{
-\item{x}{a rate-object, vector of two; rate and standard error or observed and person-years.}
-
-\item{y}{a rate-object, vector of two; rate and standard error or observed and person-years.}
-
-\item{crude}{set TRUE to use crude rates; default is FALSE.}
-
-\item{SE.method}{default TRUE; if \code{x} and \code{y} are vectors of observed and 
-person-years, this must be changed to FALSE.}
-}
-\value{
-A vector length of three: rate_ratio, and lower and upper confidence intervals.
-}
-\description{
-Calculate rate ratio with confidence intervals for rate objects or observations and person-years.
-}
-\details{
-Calculate rate ratio of two age standardized rate objects (see \code{\link{rate}}). 
-Multiple rates for each objects is supported if there are an equal number of rates. 
-Another option is to set \code{x} and \code{y} as a vector of two.
-\enumerate{
-  \item rate and its standard error, and  set \code{SE.method = TRUE}.
-  \item observations and person-year, and  set \code{SE.method = FALSE}.
-}
-See examples.
-}
-\examples{
-\donttest{
-# two rate ratios; silly example with female rectal / breast cancer 
-## mortality rates
-data("sire", package = "popEpi")
-data("sibr", package = "popEpi")
-
-BL <- list(per = 2000:2005)
-
-re <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = status == 1, breaks = BL, aggre = list(per))
-br <- lexpand(sibr, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = status == 1, breaks = BL, aggre = list(per))
-
-r_re <- rate(re, obs = "from0to1", pyrs = "pyrs")
-r_br <- rate(br, obs = "from0to1", pyrs = "pyrs")
-
-rate_ratio(r_re, r_br, SE.method = TRUE)
-}
-
-# manually set rates (0.003 and 0.005) and SEs (0.001 and 0.002)
-# so that x = y = c('rate', 'SE')
-rate_ratio(x= c(0.003, 0.001), y= c(0.005, 0.002), SE.method = TRUE) 
-
-# observed numbers (10 and 20) and person-years (30000 and 40000):
-rate_ratio(x = c(10, 30000), y = c(20, 40000), SE.method = FALSE)
-
-}
-\seealso{
-\code{\link{rate}}
-
-Other rate functions: 
-\code{\link{rate}()}
-}
-\author{
-Matti Rantanen
-}
-\concept{rate functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/incidence_rates_utils.R
+\name{rate_ratio}
+\alias{rate_ratio}
+\title{Confidence intervals for the rate ratios}
+\usage{
+rate_ratio(x, y, crude = FALSE, SE.method = TRUE)
+}
+\arguments{
+\item{x}{a rate-object, vector of two; rate and standard error or observed and person-years.}
+
+\item{y}{a rate-object, vector of two; rate and standard error or observed and person-years.}
+
+\item{crude}{set TRUE to use crude rates; default is FALSE.}
+
+\item{SE.method}{default TRUE; if \code{x} and \code{y} are vectors of observed and 
+person-years, this must be changed to FALSE.}
+}
+\value{
+A vector length of three: rate_ratio, and lower and upper confidence intervals.
+}
+\description{
+Calculate rate ratio with confidence intervals for rate objects or observations and person-years.
+}
+\details{
+Calculate rate ratio of two age standardized rate objects (see \code{\link{rate}}). 
+Multiple rates for each objects is supported if there are an equal number of rates. 
+Another option is to set \code{x} and \code{y} as a vector of two.
+\enumerate{
+  \item rate and its standard error, and  set \code{SE.method = TRUE}.
+  \item observations and person-year, and  set \code{SE.method = FALSE}.
+}
+See examples.
+}
+\examples{
+\donttest{
+# two rate ratios; silly example with female rectal / breast cancer 
+## mortality rates
+data("sire", package = "popEpi")
+data("sibr", package = "popEpi")
+
+BL <- list(per = 2000:2005)
+
+re <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = status == 1, breaks = BL, aggre = list(per))
+br <- lexpand(sibr, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = status == 1, breaks = BL, aggre = list(per))
+
+r_re <- rate(re, obs = "from0to1", pyrs = "pyrs")
+r_br <- rate(br, obs = "from0to1", pyrs = "pyrs")
+
+rate_ratio(r_re, r_br, SE.method = TRUE)
+}
+
+# manually set rates (0.003 and 0.005) and SEs (0.001 and 0.002)
+# so that x = y = c('rate', 'SE')
+rate_ratio(x= c(0.003, 0.001), y= c(0.005, 0.002), SE.method = TRUE) 
+
+# observed numbers (10 and 20) and person-years (30000 and 40000):
+rate_ratio(x = c(10, 30000), y = c(20, 40000), SE.method = FALSE)
+
+}
+\seealso{
+\code{\link{rate}}
+
+Other rate functions: 
+\code{\link{rate}()}
+}
+\author{
+Matti Rantanen
+}
+\concept{rate functions}
diff --git a/man/relpois.Rd b/man/relpois.Rd
index dff50ec..ee47af4 100644
--- a/man/relpois.Rd
+++ b/man/relpois.Rd
@@ -1,131 +1,131 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/relative_poisson.R
-\name{relpois}
-\alias{relpois}
-\title{Excess hazard Poisson model}
-\usage{
-relpois(data, formula, fot.breaks = NULL, subset = NULL, check = TRUE, ...)
-}
-\arguments{
-\item{data}{a dataset split with e.g. \code{\link{lexpand}};
-must have expected hazard merged within}
-
-\item{formula}{a formula which is passed on to \code{glm}; see Details}
-
-\item{fot.breaks}{optional; a numeric vector of [a,b) breaks to specify
-survival intervals over the follow-up time; if \code{NULL}, the 
-existing breaks along the mandatory \code{fot} time scale in \code{data}
-are used (e.g. the breaks for \code{fot} supplied to \code{lexpand})}
-
-\item{subset}{a logical vector or condition; e.g. \code{subset = sex == 1};
-limits the data before estimation}
-
-\item{check}{logical; if \code{TRUE}, tabulates excess cases by all
-factor variables in the formula to check for negative / \code{NA} 
-excess cases before fitting the GLM}
-
-\item{...}{any argument passed on to \code{glm}}
-}
-\value{
-A \code{glm} object created using a custom Poisson family construct. Some
-\code{glm} methods are applicable.
-}
-\description{
-Estimate a Poisson piecewise constant excess
-hazards model
-}
-\details{
-\strong{Basics}
-
-\code{relpois} employs a custom link function of the Poisson variety
-to estimate piecewise constant parametric excess hazards. The pieces
-are determined by \code{fot.breaks}. A \code{log(person-years)} offset
-is passed automatically to the \code{glm} call.
-
-\strong{Formula usage}
-
-The formula can be used like any ordinary \code{glm} formula. The user must
-define the outcome in some manner, which is usually \code{lex.Xst} after splitting
-with e.g. \code{lexpand}. The exception is the possibility of including 
-the baseline excess hazard terms by including the 
-reserved term \code{FOT} in the formula.
-
-For example, \code{lex.Xst != 0 ~ FOT + agegr} estimates a model with constant
-excess hazards at the follow-up intervals as specified by 
-the pertinent breaks used in splitting \code{data},
-as well as for the different age groups.
-\code{FOT} is created ad hoc if it is used in the formula.
-If you leave out \code{FOT}, the hazard is effectively
-assumed to be constant across the whole follow-up time. 
-
-You can also simply use your own follow-up time interval variable that
-you have created before calling \code{relpois}. However, when using 
-\code{FOT}, \code{relpois} automatically checks for e.g. 
-negative excess cases in follow-up intervals,
-allowing for quickly finding splitting breaks
-where model estimation is possible. It also drops any data outside the
-follow-up time window.
-
-\strong{Splitting and merging population hazard}
-
-The easiest way to both split and to include population hazard information is 
-by using \code{\link{lexpand}}. You may also fairly easily do it by hand
-by splitting first and then merging in your population hazard information.
-
-
-\strong{Data requirements}
-
-The population hazard information must be available for each record and named
-\code{pop.haz}. The follow-up time variable must be named \code{"fot"} e.g.
-as a result of using \code{lexpand}. The \code{lex.dur} variable must also
-be present, containing person-year information.
-}
-\examples{
-## use the simulated rectal cancer cohort
-data("sire", package = "popEpi")
-sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
-
-## usable straight away after splitting
-fb <- c(0,3/12,6/12,1,2,3,4,5)
-x <- lexpand(sire, birth = bi_date, entry = dg_date,
-             exit = ex_date, status=status,
-             breaks = list(fot=fb), pophaz=popmort)
-rpm <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ FOT + agegr)
- 
-## some methods for glm work. e.g. test for interaction
-\donttest{
-rpm2 <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ FOT*agegr)
-anova(rpm, rpm2, test="LRT")
-AIC(rpm, rpm2)
-## update() won't work currently
-}
-}
-\references{
-Paul W Dickman, Andy Sloggett, Michael Hills, and Timo Hakulinen.
-Regression models for relative survival. 
-Stat Med. 2004 Jan 15;23(1):51-64.
-\doi{10.1002/sim.1597}
-}
-\seealso{
-\code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-
-Other relpois functions: 
-\code{\link{RPL}},
-\code{\link{relpois_ag}()},
-\code{\link{rpcurve}()}
-}
-\author{
-Joonas Miettinen, Karri Seppa
-}
-\concept{main functions}
-\concept{relpois functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/relative_poisson.R
+\name{relpois}
+\alias{relpois}
+\title{Excess hazard Poisson model}
+\usage{
+relpois(data, formula, fot.breaks = NULL, subset = NULL, check = TRUE, ...)
+}
+\arguments{
+\item{data}{a dataset split with e.g. \code{\link{lexpand}};
+must have expected hazard merged within}
+
+\item{formula}{a formula which is passed on to \code{glm}; see Details}
+
+\item{fot.breaks}{optional; a numeric vector of [a,b) breaks to specify
+survival intervals over the follow-up time; if \code{NULL}, the 
+existing breaks along the mandatory \code{fot} time scale in \code{data}
+are used (e.g. the breaks for \code{fot} supplied to \code{lexpand})}
+
+\item{subset}{a logical vector or condition; e.g. \code{subset = sex == 1};
+limits the data before estimation}
+
+\item{check}{logical; if \code{TRUE}, tabulates excess cases by all
+factor variables in the formula to check for negative / \code{NA} 
+excess cases before fitting the GLM}
+
+\item{...}{any argument passed on to \code{glm}}
+}
+\value{
+A \code{glm} object created using a custom Poisson family construct. Some
+\code{glm} methods are applicable.
+}
+\description{
+Estimate a Poisson piecewise constant excess
+hazards model
+}
+\details{
+\strong{Basics}
+
+\code{relpois} employs a custom link function of the Poisson variety
+to estimate piecewise constant parametric excess hazards. The pieces
+are determined by \code{fot.breaks}. A \code{log(person-years)} offset
+is passed automatically to the \code{glm} call.
+
+\strong{Formula usage}
+
+The formula can be used like any ordinary \code{glm} formula. The user must
+define the outcome in some manner, which is usually \code{lex.Xst} after splitting
+with e.g. \code{lexpand}. The exception is the possibility of including 
+the baseline excess hazard terms by including the 
+reserved term \code{FOT} in the formula.
+
+For example, \code{lex.Xst != 0 ~ FOT + agegr} estimates a model with constant
+excess hazards at the follow-up intervals as specified by 
+the pertinent breaks used in splitting \code{data},
+as well as for the different age groups.
+\code{FOT} is created ad hoc if it is used in the formula.
+If you leave out \code{FOT}, the hazard is effectively
+assumed to be constant across the whole follow-up time. 
+
+You can also simply use your own follow-up time interval variable that
+you have created before calling \code{relpois}. However, when using 
+\code{FOT}, \code{relpois} automatically checks for e.g. 
+negative excess cases in follow-up intervals,
+allowing for quickly finding splitting breaks
+where model estimation is possible. It also drops any data outside the
+follow-up time window.
+
+\strong{Splitting and merging population hazard}
+
+The easiest way to both split and to include population hazard information is 
+by using \code{\link{lexpand}}. You may also fairly easily do it by hand
+by splitting first and then merging in your population hazard information.
+
+
+\strong{Data requirements}
+
+The population hazard information must be available for each record and named
+\code{pop.haz}. The follow-up time variable must be named \code{"fot"} e.g.
+as a result of using \code{lexpand}. The \code{lex.dur} variable must also
+be present, containing person-year information.
+}
+\examples{
+## use the simulated rectal cancer cohort
+data("sire", package = "popEpi")
+sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
+
+## usable straight away after splitting
+fb <- c(0,3/12,6/12,1,2,3,4,5)
+x <- lexpand(sire, birth = bi_date, entry = dg_date,
+             exit = ex_date, status=status,
+             breaks = list(fot=fb), pophaz=popmort)
+rpm <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ FOT + agegr)
+ 
+## some methods for glm work. e.g. test for interaction
+\donttest{
+rpm2 <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ FOT*agegr)
+anova(rpm, rpm2, test="LRT")
+AIC(rpm, rpm2)
+## update() won't work currently
+}
+}
+\references{
+Paul W Dickman, Andy Sloggett, Michael Hills, and Timo Hakulinen.
+Regression models for relative survival. 
+Stat Med. 2004 Jan 15;23(1):51-64.
+\doi{10.1002/sim.1597}
+}
+\seealso{
+\code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+
+Other relpois functions: 
+\code{\link{RPL}},
+\code{\link{relpois_ag}()},
+\code{\link{rpcurve}()}
+}
+\author{
+Joonas Miettinen, Karri Seppa
+}
+\concept{main functions}
+\concept{relpois functions}
diff --git a/man/relpois_ag.Rd b/man/relpois_ag.Rd
index 52510d1..1f2681e 100644
--- a/man/relpois_ag.Rd
+++ b/man/relpois_ag.Rd
@@ -1,112 +1,112 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/relative_poisson.R
-\name{relpois_ag}
-\alias{relpois_ag}
-\title{Excess hazard Poisson model}
-\usage{
-relpois_ag(
-  formula,
-  data,
-  d.exp,
-  offset = NULL,
-  breaks = NULL,
-  subset = NULL,
-  piecewise = TRUE,
-  check = TRUE,
-  ...
-)
-}
-\arguments{
-\item{formula}{a formula with the counts of events as the response.
-Passed on to \code{glm}. May contain usage of the \code{offset()} function
-instead of supplying the offset for the Poisson model via the argument
-\code{offset}.}
-
-\item{data}{an \code{aggre} object (an aggregated data set; 
-see \code{\link{as.aggre}} and \code{\link{aggre}})}
-
-\item{d.exp}{the counts of expected cases. Mandatory.
-E.g. \code{d.exp = EXC_CASES}, where \code{EXC_CASES} is a column in data.}
-
-\item{offset}{the offset for the Poisson model, supplied as e.g.
-\code{offset = log(PTIME)}, where \code{PTIME} is a subject-time
-variable in data. Not mandatory, but almost always should be supplied.}
-
-\item{breaks}{optional; a numeric vector of [a,b) breaks to specify
-survival intervals over the follow-up time; if \code{NULL}, the 
-existing breaks along the mandatory time scale mentioned in \code{formula}
-are used}
-
-\item{subset}{a logical vector or condition; e.g. \code{subset = sex == 1};
-limits the data before estimation}
-
-\item{piecewise}{\code{logical}; if \code{TRUE}, and if any time scale
-from data is used (mentioned) in the formula, the time scale is 
-transformed into a factor variable indicating intervals on the time scale.
-Otherwise the time scale left as it is, usually a numeric variable.
-E.g. if \code{formula = counts ~ TS1*VAR1}, \code{TS1} is transformed
-into a factor before fitting model.}
-
-\item{check}{\code{logical}; if \code{TRUE}, performs check on the 
-negativity excess cases by factor-like covariates in formula - 
-negative excess cases will very likely lead to non-converging model}
-
-\item{...}{any other argument passed on to \code{\link[stats]{glm}} such as 
-\code{control} or \code{weights}}
-}
-\value{
-A \code{relpois} object created using a custom Poisson family construct.
-}
-\description{
-Estimate a Poisson Piecewise Constant Excess
-Hazards Model
-}
-\examples{
-## use the simulated rectal cancer cohort
-data(sire, package = "popEpi")
-sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
-
-## create aggregated example data
-fb <- c(0,3/12,6/12,1,2,3,4,5)
-x <- lexpand(sire, birth = bi_date, entry = dg_date,
-             exit = ex_date, status=status \%in\% 1:2,
-             breaks = list(fot=fb), 
-             pophaz=popmort, pp = FALSE,
-             aggre = list(agegr, fot))
-             
-## fit model using aggregated data
-rpm <- relpois_ag(formula = from0to1 ~ fot + agegr,  data = x,
-                  d.exp = d.exp, offset = log(pyrs))
-summary(rpm)
- 
-## the usual functions for handling glm models work
-rpm2 <- update(rpm, . ~ fot*agegr)
-anova(rpm, rpm2, test="LRT")
-AIC(rpm, rpm2)
-
-## other features such as residuals or predicting are not guaranteed
-## to work as intended.
-}
-\seealso{
-\code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-
-Other relpois functions: 
-\code{\link{RPL}},
-\code{\link{relpois}()},
-\code{\link{rpcurve}()}
-}
-\author{
-Joonas Miettinen, Karri Seppa
-}
-\concept{main functions}
-\concept{relpois functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/relative_poisson.R
+\name{relpois_ag}
+\alias{relpois_ag}
+\title{Excess hazard Poisson model}
+\usage{
+relpois_ag(
+  formula,
+  data,
+  d.exp,
+  offset = NULL,
+  breaks = NULL,
+  subset = NULL,
+  piecewise = TRUE,
+  check = TRUE,
+  ...
+)
+}
+\arguments{
+\item{formula}{a formula with the counts of events as the response.
+Passed on to \code{glm}. May contain usage of the \code{offset()} function
+instead of supplying the offset for the Poisson model via the argument
+\code{offset}.}
+
+\item{data}{an \code{aggre} object (an aggregated data set; 
+see \code{\link{as.aggre}} and \code{\link{aggre}})}
+
+\item{d.exp}{the counts of expected cases. Mandatory.
+E.g. \code{d.exp = EXC_CASES}, where \code{EXC_CASES} is a column in data.}
+
+\item{offset}{the offset for the Poisson model, supplied as e.g.
+\code{offset = log(PTIME)}, where \code{PTIME} is a subject-time
+variable in data. Not mandatory, but almost always should be supplied.}
+
+\item{breaks}{optional; a numeric vector of [a,b) breaks to specify
+survival intervals over the follow-up time; if \code{NULL}, the 
+existing breaks along the mandatory time scale mentioned in \code{formula}
+are used}
+
+\item{subset}{a logical vector or condition; e.g. \code{subset = sex == 1};
+limits the data before estimation}
+
+\item{piecewise}{\code{logical}; if \code{TRUE}, and if any time scale
+from data is used (mentioned) in the formula, the time scale is 
+transformed into a factor variable indicating intervals on the time scale.
+Otherwise the time scale left as it is, usually a numeric variable.
+E.g. if \code{formula = counts ~ TS1*VAR1}, \code{TS1} is transformed
+into a factor before fitting model.}
+
+\item{check}{\code{logical}; if \code{TRUE}, performs check on the 
+negativity excess cases by factor-like covariates in formula - 
+negative excess cases will very likely lead to non-converging model}
+
+\item{...}{any other argument passed on to \code{\link[stats]{glm}} such as 
+\code{control} or \code{weights}}
+}
+\value{
+A \code{relpois} object created using a custom Poisson family construct.
+}
+\description{
+Estimate a Poisson Piecewise Constant Excess
+Hazards Model
+}
+\examples{
+## use the simulated rectal cancer cohort
+data(sire, package = "popEpi")
+sire$agegr <- cut(sire$dg_age, c(0,45,60,Inf), right=FALSE)
+
+## create aggregated example data
+fb <- c(0,3/12,6/12,1,2,3,4,5)
+x <- lexpand(sire, birth = bi_date, entry = dg_date,
+             exit = ex_date, status=status \%in\% 1:2,
+             breaks = list(fot=fb), 
+             pophaz=popmort, pp = FALSE,
+             aggre = list(agegr, fot))
+             
+## fit model using aggregated data
+rpm <- relpois_ag(formula = from0to1 ~ fot + agegr,  data = x,
+                  d.exp = d.exp, offset = log(pyrs))
+summary(rpm)
+ 
+## the usual functions for handling glm models work
+rpm2 <- update(rpm, . ~ fot*agegr)
+anova(rpm, rpm2, test="LRT")
+AIC(rpm, rpm2)
+
+## other features such as residuals or predicting are not guaranteed
+## to work as intended.
+}
+\seealso{
+\code{\link{lexpand}}, \code{\link{poisson}}, \code{\link{glm}}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+
+Other relpois functions: 
+\code{\link{RPL}},
+\code{\link{relpois}()},
+\code{\link{rpcurve}()}
+}
+\author{
+Joonas Miettinen, Karri Seppa
+}
+\concept{main functions}
+\concept{relpois functions}
diff --git a/man/robust_values.Rd b/man/robust_values.Rd
index c19dad9..fc8d257 100644
--- a/man/robust_values.Rd
+++ b/man/robust_values.Rd
@@ -1,53 +1,53 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{robust_values}
-\alias{robust_values}
-\title{Convert values to numeric robustly}
-\usage{
-robust_values(num.values, force = FALSE, messages = TRUE)
-}
-\arguments{
-\item{num.values}{values to convert to numeric}
-
-\item{force}{logical; if \code{TRUE}, returns a vector of values where values that cannot be interpreted as numeric are
-set to \code{NA}; if \code{FALSE}, returns the original vector and gives a warning if any value cannot be interpreted as
-numeric.}
-
-\item{messages}{logical; if \code{TRUE}, returns a message of what was done with the \code{num.values}}
-}
-\value{
-A numeric vector.
-}
-\description{
-Brute force solution for ensuring a variable is numeric by 
-coercing a variable of any type first to factor and then to numeric
-}
-\note{
-Returns \code{NULL} if given \code{num.values} is \code{NULL}.
-}
-\examples{
-## this works
-values <- c("1", "3", "5")
-values <- robust_values(values)
-
-## this works
-values <- c("1", "3", "5", NA)
-values <- robust_values(values)
-
-## this returns originals and throws warnings
-values <- c("1", "3", "5", "a")
-suppressWarnings(
-  values <- robust_values(values)
-)
-
-
-## this forces "a" to NA and works otherwise; throws warning about NAs
-values <- c("1", "3", "5", "a")
-suppressWarnings(
-  values <- robust_values(values, force=TRUE)
-)
-
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{robust_values}
+\alias{robust_values}
+\title{Convert values to numeric robustly}
+\usage{
+robust_values(num.values, force = FALSE, messages = TRUE)
+}
+\arguments{
+\item{num.values}{values to convert to numeric}
+
+\item{force}{logical; if \code{TRUE}, returns a vector of values where values that cannot be interpreted as numeric are
+set to \code{NA}; if \code{FALSE}, returns the original vector and gives a warning if any value cannot be interpreted as
+numeric.}
+
+\item{messages}{logical; if \code{TRUE}, returns a message of what was done with the \code{num.values}}
+}
+\value{
+A numeric vector.
+}
+\description{
+Brute force solution for ensuring a variable is numeric by 
+coercing a variable of any type first to factor and then to numeric
+}
+\note{
+Returns \code{NULL} if given \code{num.values} is \code{NULL}.
+}
+\examples{
+## this works
+values <- c("1", "3", "5")
+values <- robust_values(values)
+
+## this works
+values <- c("1", "3", "5", NA)
+values <- robust_values(values)
+
+## this returns originals and throws warnings
+values <- c("1", "3", "5", "a")
+suppressWarnings(
+  values <- robust_values(values)
+)
+
+
+## this forces "a" to NA and works otherwise; throws warning about NAs
+values <- c("1", "3", "5", "a")
+suppressWarnings(
+  values <- robust_values(values, force=TRUE)
+)
+
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/rpcurve.Rd b/man/rpcurve.Rd
index f0f15ac..40cf2b8 100644
--- a/man/rpcurve.Rd
+++ b/man/rpcurve.Rd
@@ -1,74 +1,74 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/relative_poisson_net_survival.R
-\name{rpcurve}
-\alias{rpcurve}
-\title{Marginal piecewise parametric relative survival curve}
-\usage{
-rpcurve(object)
-}
-\arguments{
-\item{object}{a \code{relpois} object}
-}
-\value{
-A `data.table` of relative survival curves.
-}
-\description{
-Fit a marginal relative survival curve based on a \code{relpois} fit
-}
-\details{
-Estimates a marginal curve, i.e. the average of all
-possible individual curves. 
-
-Only supported when the reserved \code{FOT} variable was used in \code{relpois}.
-Computes a curve for each unique combination of covariates (e.g. 4 sets) 
-and returns a weighted average curve based on the counts
-of subjects for each combination (e.g. 1000, 125, 50, 25 respectively). 
-Fairly fast when only categorical variables have been used, otherwise
-go get a cup of coffee.
-
-If delayed entry is present in data due to period analysis limiting,
-the marginal curve is constructed only for those whose follow-up started
-in the respective period.
-}
-\examples{
-\donttest{
-## use the simulated rectal cancer cohort
-data("sire", package = "popEpi")
-ab <- c(0,45,55,65,70,Inf)
-sire$agegr <- cut(sire$dg_age, breaks = ab, right = FALSE)
-
-BL <- list(fot= seq(0,10,1/12))
-pm <- data.frame(popEpi::popmort)
-x <- lexpand(sire, breaks=BL, pophaz=pm, 
-             birth = bi_date, 
-             entry = dg_date, exit = ex_date, 
-             status  = status \%in\% 1:2)
-
-rpm <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ -1+ FOT + agegr, 
-               fot.breaks=c(0,0.25,0.5,1:8,10))
-pmc <- rpcurve(rpm)
-
-## compare with non-parametric estimates
-names(pm) <- c("sex", "per", "age", "haz")
-x$agegr <- cut(x$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
-st <- survtab(fot ~ adjust(agegr), data = x, weights = "internal",
-              pophaz = pm)
-
-
-plot(st, y = "r.e2.as")
-lines(y = pmc$est, x = pmc$Tstop, col="red")
-}
-
-
-
-}
-\seealso{
-Other relpois functions: 
-\code{\link{RPL}},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{relpois functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/relative_poisson_net_survival.R
+\name{rpcurve}
+\alias{rpcurve}
+\title{Marginal piecewise parametric relative survival curve}
+\usage{
+rpcurve(object)
+}
+\arguments{
+\item{object}{a \code{relpois} object}
+}
+\value{
+A `data.table` of relative survival curves.
+}
+\description{
+Fit a marginal relative survival curve based on a \code{relpois} fit
+}
+\details{
+Estimates a marginal curve, i.e. the average of all
+possible individual curves. 
+
+Only supported when the reserved \code{FOT} variable was used in \code{relpois}.
+Computes a curve for each unique combination of covariates (e.g. 4 sets) 
+and returns a weighted average curve based on the counts
+of subjects for each combination (e.g. 1000, 125, 50, 25 respectively). 
+Fairly fast when only categorical variables have been used, otherwise
+go get a cup of coffee.
+
+If delayed entry is present in data due to period analysis limiting,
+the marginal curve is constructed only for those whose follow-up started
+in the respective period.
+}
+\examples{
+\donttest{
+## use the simulated rectal cancer cohort
+data("sire", package = "popEpi")
+ab <- c(0,45,55,65,70,Inf)
+sire$agegr <- cut(sire$dg_age, breaks = ab, right = FALSE)
+
+BL <- list(fot= seq(0,10,1/12))
+pm <- data.frame(popEpi::popmort)
+x <- lexpand(sire, breaks=BL, pophaz=pm, 
+             birth = bi_date, 
+             entry = dg_date, exit = ex_date, 
+             status  = status \%in\% 1:2)
+
+rpm <- relpois(x, formula = lex.Xst \%in\% 1:2 ~ -1+ FOT + agegr, 
+               fot.breaks=c(0,0.25,0.5,1:8,10))
+pmc <- rpcurve(rpm)
+
+## compare with non-parametric estimates
+names(pm) <- c("sex", "per", "age", "haz")
+x$agegr <- cut(x$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
+st <- survtab(fot ~ adjust(agegr), data = x, weights = "internal",
+              pophaz = pm)
+
+
+plot(st, y = "r.e2.as")
+lines(y = pmc$est, x = pmc$Tstop, col="red")
+}
+
+
+
+}
+\seealso{
+Other relpois functions: 
+\code{\link{RPL}},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{relpois functions}
diff --git a/man/setaggre.Rd b/man/setaggre.Rd
index 685e733..fe265c4 100644
--- a/man/setaggre.Rd
+++ b/man/setaggre.Rd
@@ -1,59 +1,59 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/aggregating.R
-\name{setaggre}
-\alias{setaggre}
-\title{Set \code{aggre} attributes to an object by modifying in place}
-\usage{
-setaggre(x, values = NULL, by = NULL, breaks = NULL)
-}
-\arguments{
-\item{x}{a \code{data.frame} or \code{data.table}}
-
-\item{values}{a character string vector; the names of value variables}
-
-\item{by}{a character string vector; the names of variables by which 
-\code{values} have been tabulated}
-
-\item{breaks}{a list of breaks, where each element is a breaks vector
-as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
-fully named, with the names corresponding to time scales at the aggregate
-level in your data. Every unique value in a time scale variable in data must
-also exist in the corresponding vector in the breaks list.}
-}
-\value{
-Returns `x` invisibly after setting attributes to it without taking a copy.
-This function is called for its side effects.
-}
-\description{
-Coerces an R object to an \code{aggre} object, identifying
-the object as one containing aggregated counts, person-years and other
-information. \code{setaggre} modifies in place without taking any copies.
-Retains all other attributes.
-}
-\details{
-\code{setaggre} sets \code{x} to the \code{aggre} class in place 
-without taking a copy as e.g. \code{as.data.frame.XXX} functions do; see e.g. 
-\code{\link[data.table]{setDT}}.
-}
-\examples{
-df <- data.frame(sex = rep(c("male", "female"), each = 5), 
-                 obs = rpois(10, rep(7,5, each=5)), 
-                 pyrs = rpois(10, lambda = 10000))
-## without any breaks
-setaggre(df, values = c("obs", "pyrs"), by = "sex")
-df <- data.frame(df)
-df$FUT <- 0:4
-## with breaks list
-setaggre(df, values = c("obs", "pyrs"), by = "sex", breaks = list(FUT = 0:5))
-}
-\seealso{
-Other aggregation functions: 
-\code{\link{aggre}()},
-\code{\link{as.aggre}()},
-\code{\link{lexpand}()},
-\code{\link{summary.aggre}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{aggregation functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/aggregating.R
+\name{setaggre}
+\alias{setaggre}
+\title{Set \code{aggre} attributes to an object by modifying in place}
+\usage{
+setaggre(x, values = NULL, by = NULL, breaks = NULL)
+}
+\arguments{
+\item{x}{a \code{data.frame} or \code{data.table}}
+
+\item{values}{a character string vector; the names of value variables}
+
+\item{by}{a character string vector; the names of variables by which 
+\code{values} have been tabulated}
+
+\item{breaks}{a list of breaks, where each element is a breaks vector
+as usually passed to e.g. \code{\link{splitLexisDT}}. The list must be
+fully named, with the names corresponding to time scales at the aggregate
+level in your data. Every unique value in a time scale variable in data must
+also exist in the corresponding vector in the breaks list.}
+}
+\value{
+Returns `x` invisibly after setting attributes to it without taking a copy.
+This function is called for its side effects.
+}
+\description{
+Coerces an R object to an \code{aggre} object, identifying
+the object as one containing aggregated counts, person-years and other
+information. \code{setaggre} modifies in place without taking any copies.
+Retains all other attributes.
+}
+\details{
+\code{setaggre} sets \code{x} to the \code{aggre} class in place 
+without taking a copy as e.g. \code{as.data.frame.XXX} functions do; see e.g. 
+\code{\link[data.table]{setDT}}.
+}
+\examples{
+df <- data.frame(sex = rep(c("male", "female"), each = 5), 
+                 obs = rpois(10, rep(7,5, each=5)), 
+                 pyrs = rpois(10, lambda = 10000))
+## without any breaks
+setaggre(df, values = c("obs", "pyrs"), by = "sex")
+df <- data.frame(df)
+df$FUT <- 0:4
+## with breaks list
+setaggre(df, values = c("obs", "pyrs"), by = "sex", breaks = list(FUT = 0:5))
+}
+\seealso{
+Other aggregation functions: 
+\code{\link{aggre}()},
+\code{\link{as.aggre}()},
+\code{\link{lexpand}()},
+\code{\link{summary.aggre}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{aggregation functions}
diff --git a/man/setclass.Rd b/man/setclass.Rd
index f8f50c9..94e7767 100644
--- a/man/setclass.Rd
+++ b/man/setclass.Rd
@@ -1,27 +1,27 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{setclass}
-\alias{setclass}
-\title{Set the class of an object (convenience function for
- \code{setattr(obj, "class", CLASS)}); can add instead of replace}
-\usage{
-setclass(obj, cl, add = FALSE, add.place = "first")
-}
-\arguments{
-\item{obj}{and object for which to set class}
-
-\item{cl}{class to set}
-
-\item{add}{if \code{TRUE}, adds \code{cl} to the 
-classes of the \code{obj}; otherwise replaces the class information}
-
-\item{add.place}{\code{"first"} or \code{"last"}; adds \code{cl}
-to the front or to the back of the \code{obj}'s class vector}
-}
-\description{
-Sets the class of an object in place to \code{cl}
-by replacing or adding
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{setclass}
+\alias{setclass}
+\title{Set the class of an object (convenience function for
+ \code{setattr(obj, "class", CLASS)}); can add instead of replace}
+\usage{
+setclass(obj, cl, add = FALSE, add.place = "first")
+}
+\arguments{
+\item{obj}{and object for which to set class}
+
+\item{cl}{class to set}
+
+\item{add}{if \code{TRUE}, adds \code{cl} to the 
+classes of the \code{obj}; otherwise replaces the class information}
+
+\item{add.place}{\code{"first"} or \code{"last"}; adds \code{cl}
+to the front or to the back of the \code{obj}'s class vector}
+}
+\description{
+Sets the class of an object in place to \code{cl}
+by replacing or adding
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/setcolsnull.Rd b/man/setcolsnull.Rd
index 369622d..460e409 100644
--- a/man/setcolsnull.Rd
+++ b/man/setcolsnull.Rd
@@ -1,41 +1,41 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{setcolsnull}
-\alias{setcolsnull}
-\title{Delete \code{data.table} columns if there}
-\usage{
-setcolsnull(
-  DT = NULL,
-  delete = NULL,
-  keep = NULL,
-  colorder = FALSE,
-  soft = TRUE
-)
-}
-\arguments{
-\item{DT}{a \code{data.table}}
-
-\item{delete}{a character vector of column names to be deleted}
-
-\item{keep}{a character vector of column names to keep; 
-the rest will be removed; \code{keep} overrides \code{delete}}
-
-\item{colorder}{logical; if \code{TRUE}, also does \code{setcolorder} using
-\code{keep}}
-
-\item{soft}{logical; if \code{TRUE}, does not cause an error if any variable
-name in \code{keep} or \code{delete} is missing; \code{soft = FALSE} useful 
-for programming sometimes}
-}
-\value{
-Always returns `NULL` invisibly.
-This function is called for its side effects.
-}
-\description{
-Deletes columns in a \code{data.table} conveniently.
-May only delete columns that are found silently. Sometimes useful in e.g.
-\code{on.exit} expressions.
-}
-\author{
-Joonas Miettinen
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{setcolsnull}
+\alias{setcolsnull}
+\title{Delete \code{data.table} columns if there}
+\usage{
+setcolsnull(
+  DT = NULL,
+  delete = NULL,
+  keep = NULL,
+  colorder = FALSE,
+  soft = TRUE
+)
+}
+\arguments{
+\item{DT}{a \code{data.table}}
+
+\item{delete}{a character vector of column names to be deleted}
+
+\item{keep}{a character vector of column names to keep; 
+the rest will be removed; \code{keep} overrides \code{delete}}
+
+\item{colorder}{logical; if \code{TRUE}, also does \code{setcolorder} using
+\code{keep}}
+
+\item{soft}{logical; if \code{TRUE}, does not cause an error if any variable
+name in \code{keep} or \code{delete} is missing; \code{soft = FALSE} useful 
+for programming sometimes}
+}
+\value{
+Always returns `NULL` invisibly.
+This function is called for its side effects.
+}
+\description{
+Deletes columns in a \code{data.table} conveniently.
+May only delete columns that are found silently. Sometimes useful in e.g.
+\code{on.exit} expressions.
+}
+\author{
+Joonas Miettinen
+}
diff --git a/man/sibr.Rd b/man/sibr.Rd
index 1d1c449..1bd540e 100644
--- a/man/sibr.Rd
+++ b/man/sibr.Rd
@@ -1,46 +1,46 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{sibr}
-\alias{sibr}
-\title{sibr - a simulated cohort of Finnish female breast cancer patients}
-\format{
-data.table with columns
-\itemize{
- \item sex - gender of the patient (1 = female)
- \item bi_date - date of birth
- \item dg_date - date of cancer diagnosis
- \item ex_date - date of exit from follow-up (death or censoring)
- \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
- \item dg_age  - age at diagnosis expressed as fractional years
-}
-}
-\source{
-The Finnish Cancer Registry
-}
-\description{
-\code{sibr} is a simulated cohort pertaining female Finnish breast cancer patients
-diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
-via modest randomization within several time windows. The dataset is additionally
-a random sample of 10 000 cases from the pertaining time window.
-}
-\details{
-The closing date for the pertinent data was 2012-12-31, meaning status information was
-available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
-}
-\seealso{
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{meanpop_fi}},
-\code{\link{popmort}},
-\code{\link{sire}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-
-Other survival data: 
-\code{\link{sire}}
-}
-\author{
-Karri Seppa
-}
-\concept{popEpi data}
-\concept{survival data}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{sibr}
+\alias{sibr}
+\title{sibr - a simulated cohort of Finnish female breast cancer patients}
+\format{
+data.table with columns
+\itemize{
+ \item sex - gender of the patient (1 = female)
+ \item bi_date - date of birth
+ \item dg_date - date of cancer diagnosis
+ \item ex_date - date of exit from follow-up (death or censoring)
+ \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
+ \item dg_age  - age at diagnosis expressed as fractional years
+}
+}
+\source{
+The Finnish Cancer Registry
+}
+\description{
+\code{sibr} is a simulated cohort pertaining female Finnish breast cancer patients
+diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
+via modest randomization within several time windows. The dataset is additionally
+a random sample of 10 000 cases from the pertaining time window.
+}
+\details{
+The closing date for the pertinent data was 2012-12-31, meaning status information was
+available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
+}
+\seealso{
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{meanpop_fi}},
+\code{\link{popmort}},
+\code{\link{sire}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+
+Other survival data: 
+\code{\link{sire}}
+}
+\author{
+Karri Seppa
+}
+\concept{popEpi data}
+\concept{survival data}
diff --git a/man/sir.Rd b/man/sir.Rd
index c9b4714..f1e6446 100644
--- a/man/sir.Rd
+++ b/man/sir.Rd
@@ -1,219 +1,219 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/sir.R
-\name{sir}
-\alias{sir}
-\title{Calculate SIR or SMR}
-\usage{
-sir(
-  coh.data,
-  coh.obs,
-  coh.pyrs,
-  ref.data = NULL,
-  ref.obs = NULL,
-  ref.pyrs = NULL,
-  ref.rate = NULL,
-  subset = NULL,
-  print = NULL,
-  adjust = NULL,
-  mstate = NULL,
-  test.type = "homogeneity",
-  conf.type = "profile",
-  conf.level = 0.95,
-  EAR = FALSE
-)
-}
-\arguments{
-\item{coh.data}{aggregated cohort data, see e.g. \code{\link{lexpand}}}
-
-\item{coh.obs}{variable name for observed cases; quoted or unquoted. A vector when using \code{mstata}.}
-
-\item{coh.pyrs}{variable name for person years in cohort data; 
-quoted (as a string \code{'myvar'}) or unquoted (AKA as a name; \code{myvar})}
-
-\item{ref.data}{population data. Can be left NULL if \code{coh.data} 
-is stratified in \code{print}. See \code{\link{pophaz}} for details.}
-
-\item{ref.obs}{variable name for observed cases; quoted or unquoted}
-
-\item{ref.pyrs}{variable name for person-years in population data; quoted or unquoted}
-
-\item{ref.rate}{population rate variable (cases/person-years). Overwrites 
-arguments \code{ref.pyrs} and \code{ref.obs}. Quoted or unquoted}
-
-\item{subset}{logical condition to select data from \code{coh.data} before any computations}
-
-\item{print}{variable names to stratify results; quoted vector or unquoted named list with functions}
-
-\item{adjust}{variable names for adjusting without stratifying output; quoted vector or unquoted list}
-
-\item{mstate}{set column names for cause specific observations; quoted or unquoted. Relevant only
-when \code{coh.obs} length is two or more. See details.}
-
-\item{test.type}{Test for equal SIRs. Test available are 'homogeneity' and 'trend'.}
-
-\item{conf.type}{Confidence interval type: 'profile'(=default), 'wald' or 'univariate'.}
-
-\item{conf.level}{Level of type-I error in confidence intervals, default 0.05 is 95\% CI.}
-
-\item{EAR}{logical; TRUE calculates Excess Absolute Risks for univariate SIRs.
-(see details)}
-}
-\value{
-A sir-object that is a \code{data.table} with meta information in the attributes.
-}
-\description{
-Poisson modelled standardised incidence or mortality ratios (SIRs / SMRs) i.e. 
-indirect method for calculating standardised rates. SIR is a ratio of observed and expected cases.
-Expected cases are derived by multiplying the strata-specific population rate with the
-corresponding person-years of the cohort.
-}
-\details{
-\code{sir} is a comprehensive tool for modelling SIRs/SMRs with flexible 
-options to adjust and print SIRs, test homogeneity and utilize 
-multi-state data. The cohort data and the variable names for observation 
-counts and person-years are required.
-The reference data is optional, since the cohort data 
-can be stratified (\code{print}) and compared to total.
-
-
-\strong{Adjust and print}
-
-A SIR can be adjusted or standardised using the covariates found in both \code{coh.data} and \code{ref.data}.
-Variable to adjust are given in \code{adjust}.
-Variable names needs to match in both \code{coh.data} and \code{ref.data}. 
-Typical variables to adjust by are gender, age group and calendar period.
-
-\code{print} is used to stratify the SIR output. In other words, the variables 
-assigned to \code{print} are the covariates of the Poisson model.
-Variable levels are treated as categorical.
-Variables can be assigned in both \code{print} and \code{adjust}. 
-This means the output it adjusted and printed by these variables.
-
-\code{print} can also be a list of expressions. This enables changing variable 
-names or transforming variables with functions such as \code{cut} and \code{round}.
-For example, the existing variables \code{agegroup} and \code{year} could be
-transformed to new levels using \code{cut} by
-
-\code{print = list( age.category = cut(agegroup, breaks = c(0,50,75,100)), 
-year.cat = cut(year, seq(1950,2010,20)))}
-
-
-\strong{ref.rate or ref.obs & ref.pyrs}
-
-The population rate variable can be given to the \code{ref.rate} parameter. 
-That is, when using e.g. the \code{popmort} or a comparable data file, one may
-supply \code{ref.rate} instead of \code{ref.obs} and \code{ref.pyrs}, which
-will be ignored if \code{ref.rate} is supplied. 
-
-
-Note that if all the stratifying variables in 
-\code{ref.data} are not listed in \code{adjust}, 
-or when the categories are otherwise combined,
-the (unweighted) mean of rates is used for computing expected cases.
-This might incur a small bias in comparison to when exact numbers of observations
-and person-years are available. 
-
-
-
-\strong{mstate}
-
-E.g. using \code{lexpand} it's possible to compute counts for several outcomes
-so that the population at risk is same for each 
-outcome such as a certain kind of cancer. 
-The transition counts are in wide data format, 
-and the relevant columns can be supplied to \code{sir}
-in a vector via the \code{coh.obs} argument. 
-The name of the corresponding new column in \code{ref.data} is given in
-\code{mstate}. It's recommended to include the \code{mstate} variable in \code{adjust},
-so the corresponding information should also be available in \code{ref.data}.
-More examples in sir-vignette.
-
-This approach is analogous to where SIRs are calculated separately their 
-own function calls.
-
-
-\strong{Other parameters}
-
-\code{univariate} confidence intervals are calculated using exact 
-Poisson intervals (\code{poisson.ci}). The options \code{profile} and \code{wald} are
-is based on a Poisson regression model: profile-likelihood confidence intervals 
-or Wald's normal-approximation. P-value is Poisson model based \code{conf.type}
-or calculated using the method described by Breslow and Day. Function automatically
-switches to another \code{conf.type} if calculation is not possible with a message.
-Usually model fit fails if there is print stratum with zero expected values.
-
-
-The LRT p-value tests the levels of \code{print}. The test can be either 
-\code{"homogeneity"}, a likelihood ratio test where the model variables defined in
-\code{print} (factor) is compared to the constant model.
-Option \code{"trend"} tests if the linear trend of the continuous variable in
-\code{print} is significant (using model comparison).
-
-
-\strong{EAR: Excess Absolute Risk}
-
-Excess Absolute Risk is a simple way to quantify the absolute difference between cohort risk and 
-population risk.
-Make sure that the person-years are calculated accordingly before using EAR. (when using mstate)
-
-Formula for EAR:
-\deqn{EAR = \frac{observed - expected}{person years} \times 1000.}{EAR = (obs - exp)/pyrs * 1000.}
-
-\strong{Data format}
-
-The data should be given in tabulated format. That is the number of observations 
-and person-years are represented for each stratum.
-Note that also individual data is allowed as long as each observations, 
-person-years, and print and adjust variables are presented in columns.
-The extra variables and levels are reduced automatically before estimating SIRs. 
-Example of data format:
-
-\tabular{rrrrr}{
-  sex \tab age \tab period \tab obs \tab pyrs \cr
-  0 \tab 1 \tab 2010 \tab 0 \tab 390 \cr
-  0 \tab 2 \tab 2010 \tab 5 \tab 385 \cr
-  1 \tab 1 \tab 2010 \tab 3 \tab 308 \cr
-  1 \tab 2 \tab 2010 \tab 12 \tab 315
-}
-}
-\examples{
-data(popmort)
-data(sire)
-c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-## SMR due other causes: status = 2
-se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs', 
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup', 'year', 'sex'), print = 'fot')
-se
-## for examples see: vignette('sir')
-
-
-}
-\seealso{
-\code{\link{lexpand}}
-\href{../doc/sir.html}{A SIR calculation vignette}
-
-Other sir functions: 
-\code{\link{lines.sirspline}()},
-\code{\link{plot.sirspline}()},
-\code{\link{sir_exp}()},
-\code{\link{sir_ratio}()},
-\code{\link{sirspline}()}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Matti Rantanen, Joonas Miettinen
-}
-\concept{main functions}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/sir.R
+\name{sir}
+\alias{sir}
+\title{Calculate SIR or SMR}
+\usage{
+sir(
+  coh.data,
+  coh.obs,
+  coh.pyrs,
+  ref.data = NULL,
+  ref.obs = NULL,
+  ref.pyrs = NULL,
+  ref.rate = NULL,
+  subset = NULL,
+  print = NULL,
+  adjust = NULL,
+  mstate = NULL,
+  test.type = "homogeneity",
+  conf.type = "profile",
+  conf.level = 0.95,
+  EAR = FALSE
+)
+}
+\arguments{
+\item{coh.data}{aggregated cohort data, see e.g. \code{\link{lexpand}}}
+
+\item{coh.obs}{variable name for observed cases; quoted or unquoted. A vector when using \code{mstata}.}
+
+\item{coh.pyrs}{variable name for person years in cohort data; 
+quoted (as a string \code{'myvar'}) or unquoted (AKA as a name; \code{myvar})}
+
+\item{ref.data}{population data. Can be left NULL if \code{coh.data} 
+is stratified in \code{print}. See \code{\link{pophaz}} for details.}
+
+\item{ref.obs}{variable name for observed cases; quoted or unquoted}
+
+\item{ref.pyrs}{variable name for person-years in population data; quoted or unquoted}
+
+\item{ref.rate}{population rate variable (cases/person-years). Overwrites 
+arguments \code{ref.pyrs} and \code{ref.obs}. Quoted or unquoted}
+
+\item{subset}{logical condition to select data from \code{coh.data} before any computations}
+
+\item{print}{variable names to stratify results; quoted vector or unquoted named list with functions}
+
+\item{adjust}{variable names for adjusting without stratifying output; quoted vector or unquoted list}
+
+\item{mstate}{set column names for cause specific observations; quoted or unquoted. Relevant only
+when \code{coh.obs} length is two or more. See details.}
+
+\item{test.type}{Test for equal SIRs. Test available are 'homogeneity' and 'trend'.}
+
+\item{conf.type}{Confidence interval type: 'profile'(=default), 'wald' or 'univariate'.}
+
+\item{conf.level}{Level of type-I error in confidence intervals, default 0.05 is 95\% CI.}
+
+\item{EAR}{logical; TRUE calculates Excess Absolute Risks for univariate SIRs.
+(see details)}
+}
+\value{
+A sir-object that is a \code{data.table} with meta information in the attributes.
+}
+\description{
+Poisson modelled standardised incidence or mortality ratios (SIRs / SMRs) i.e. 
+indirect method for calculating standardised rates. SIR is a ratio of observed and expected cases.
+Expected cases are derived by multiplying the strata-specific population rate with the
+corresponding person-years of the cohort.
+}
+\details{
+\code{sir} is a comprehensive tool for modelling SIRs/SMRs with flexible 
+options to adjust and print SIRs, test homogeneity and utilize 
+multi-state data. The cohort data and the variable names for observation 
+counts and person-years are required.
+The reference data is optional, since the cohort data 
+can be stratified (\code{print}) and compared to total.
+
+
+\strong{Adjust and print}
+
+A SIR can be adjusted or standardised using the covariates found in both \code{coh.data} and \code{ref.data}.
+Variable to adjust are given in \code{adjust}.
+Variable names needs to match in both \code{coh.data} and \code{ref.data}. 
+Typical variables to adjust by are gender, age group and calendar period.
+
+\code{print} is used to stratify the SIR output. In other words, the variables 
+assigned to \code{print} are the covariates of the Poisson model.
+Variable levels are treated as categorical.
+Variables can be assigned in both \code{print} and \code{adjust}. 
+This means the output it adjusted and printed by these variables.
+
+\code{print} can also be a list of expressions. This enables changing variable 
+names or transforming variables with functions such as \code{cut} and \code{round}.
+For example, the existing variables \code{agegroup} and \code{year} could be
+transformed to new levels using \code{cut} by
+
+\code{print = list( age.category = cut(agegroup, breaks = c(0,50,75,100)), 
+year.cat = cut(year, seq(1950,2010,20)))}
+
+
+\strong{ref.rate or ref.obs & ref.pyrs}
+
+The population rate variable can be given to the \code{ref.rate} parameter. 
+That is, when using e.g. the \code{popmort} or a comparable data file, one may
+supply \code{ref.rate} instead of \code{ref.obs} and \code{ref.pyrs}, which
+will be ignored if \code{ref.rate} is supplied. 
+
+
+Note that if all the stratifying variables in 
+\code{ref.data} are not listed in \code{adjust}, 
+or when the categories are otherwise combined,
+the (unweighted) mean of rates is used for computing expected cases.
+This might incur a small bias in comparison to when exact numbers of observations
+and person-years are available. 
+
+
+
+\strong{mstate}
+
+E.g. using \code{lexpand} it's possible to compute counts for several outcomes
+so that the population at risk is same for each 
+outcome such as a certain kind of cancer. 
+The transition counts are in wide data format, 
+and the relevant columns can be supplied to \code{sir}
+in a vector via the \code{coh.obs} argument. 
+The name of the corresponding new column in \code{ref.data} is given in
+\code{mstate}. It's recommended to include the \code{mstate} variable in \code{adjust},
+so the corresponding information should also be available in \code{ref.data}.
+More examples in sir-vignette.
+
+This approach is analogous to where SIRs are calculated separately their 
+own function calls.
+
+
+\strong{Other parameters}
+
+\code{univariate} confidence intervals are calculated using exact 
+Poisson intervals (\code{poisson.ci}). The options \code{profile} and \code{wald} are
+is based on a Poisson regression model: profile-likelihood confidence intervals 
+or Wald's normal-approximation. P-value is Poisson model based \code{conf.type}
+or calculated using the method described by Breslow and Day. Function automatically
+switches to another \code{conf.type} if calculation is not possible with a message.
+Usually model fit fails if there is print stratum with zero expected values.
+
+
+The LRT p-value tests the levels of \code{print}. The test can be either 
+\code{"homogeneity"}, a likelihood ratio test where the model variables defined in
+\code{print} (factor) is compared to the constant model.
+Option \code{"trend"} tests if the linear trend of the continuous variable in
+\code{print} is significant (using model comparison).
+
+
+\strong{EAR: Excess Absolute Risk}
+
+Excess Absolute Risk is a simple way to quantify the absolute difference between cohort risk and 
+population risk.
+Make sure that the person-years are calculated accordingly before using EAR. (when using mstate)
+
+Formula for EAR:
+\deqn{EAR = \frac{observed - expected}{person years} \times 1000.}{EAR = (obs - exp)/pyrs * 1000.}
+
+\strong{Data format}
+
+The data should be given in tabulated format. That is the number of observations 
+and person-years are represented for each stratum.
+Note that also individual data is allowed as long as each observations, 
+person-years, and print and adjust variables are presented in columns.
+The extra variables and levels are reduced automatically before estimating SIRs. 
+Example of data format:
+
+\tabular{rrrrr}{
+  sex \tab age \tab period \tab obs \tab pyrs \cr
+  0 \tab 1 \tab 2010 \tab 0 \tab 390 \cr
+  0 \tab 2 \tab 2010 \tab 5 \tab 385 \cr
+  1 \tab 1 \tab 2010 \tab 3 \tab 308 \cr
+  1 \tab 2 \tab 2010 \tab 12 \tab 315
+}
+}
+\examples{
+data(popmort)
+data(sire)
+c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+## SMR due other causes: status = 2
+se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs', 
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup', 'year', 'sex'), print = 'fot')
+se
+## for examples see: vignette('sir')
+
+
+}
+\seealso{
+\code{\link{lexpand}}
+\href{../doc/sir.html}{A SIR calculation vignette}
+
+Other sir functions: 
+\code{\link{lines.sirspline}()},
+\code{\link{plot.sirspline}()},
+\code{\link{sir_exp}()},
+\code{\link{sir_ratio}()},
+\code{\link{sirspline}()}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Matti Rantanen, Joonas Miettinen
+}
+\concept{main functions}
+\concept{sir functions}
diff --git a/man/sir_exp.Rd b/man/sir_exp.Rd
index 814aeda..e6c7690 100644
--- a/man/sir_exp.Rd
+++ b/man/sir_exp.Rd
@@ -1,121 +1,121 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/sir.R
-\name{sir_exp}
-\alias{sir_exp}
-\alias{sir_lex}
-\alias{sir_ag}
-\title{Calculate SMR}
-\usage{
-sir_exp(
-  x,
-  obs,
-  exp,
-  pyrs = NULL,
-  print = NULL,
-  conf.type = "profile",
-  test.type = "homogeneity",
-  conf.level = 0.95,
-  subset = NULL
-)
-
-sir_lex(x, print = NULL, breaks = NULL, ...)
-
-sir_ag(
-  x,
-  obs = "from0to1",
-  print = attr(x, "aggre.meta")$by,
-  exp = "d.exp",
-  pyrs = "pyrs",
-  ...
-)
-}
-\arguments{
-\item{x}{Data set e.g. \code{aggre} or \code{Lexis} object 
-(see: \code{\link{lexpand}})}
-
-\item{obs}{Variable name of the observed cases in the data set}
-
-\item{exp}{Variable name or expression for expected cases}
-
-\item{pyrs}{Variable name for person-years (optional)}
-
-\item{print}{Variables or expression to stratify the results}
-
-\item{conf.type}{select confidence interval type: (default=) `profile`, `wald`, `univariate`}
-
-\item{test.type}{Test for equal SIRs. Test available are 'homogeneity' and 'trend'}
-
-\item{conf.level}{Level of type-I error in confidence intervals, default 0.05 is 95\% CI}
-
-\item{subset}{a logical vector for subsetting data}
-
-\item{breaks}{a named list to split age group (age), period (per) or follow-up (fot).}
-
-\item{...}{pass arguments to \code{sir_exp}}
-}
-\value{
-A sir object
-}
-\description{
-Calculate Standardized Mortality Ratios (SMRs) using 
-a single data set that includes
-observed and expected cases and additionally person-years.
-
-\code{sir_lex} solves SMR from a \code{\link{Lexis}} object 
-calculated with \code{lexpand}.
-
-\code{sir_ag} solves SMR from a \code{\link{aggre}} object 
-calculated using \code{\link{lexpand}}.
-}
-\details{
-These functions are intended to calculate SMRs from a single data set 
-that includes both observed and expected number of cases. For example utilizing the
-argument \code{pop.haz} of the \code{\link{lexpand}}.
-
-\code{sir_lex} automatically exports the transition \code{fromXtoY} using the first
-state in \code{lex.Str} as \code{0} and all other as \code{1}. No missing values
-is allowed in observed, pop.haz or person-years.
-}
-\section{Functions}{
-\itemize{
-\item \code{sir_lex()}: 
-
-\item \code{sir_ag()}: 
-
-}}
-\examples{
-
-\donttest{
-BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-
-## Aggregated data
-x1 <- lexpand(sire, breaks = BL, status = status != 0, 
-              birth = bi_date, entry = dg_date, exit = ex_date,
-              pophaz=popmort,
-              aggre=list(sex, period = per, surv.int = fot))
-sir_ag(x1, print = 'period')
-
-
-# no aggreate or breaks
-x2 <- lexpand(sire, status = status != 0, 
-              birth = bi_date, entry = dg_date, exit = ex_date,
-              pophaz=popmort)
-sir_lex(x2, breaks = BL, print = 'per')
-}
-
-}
-\seealso{
-\code{\link{lexpand}}
-\href{../doc/sir.html}{A SIR calculation vignette}
-
-Other sir functions: 
-\code{\link{lines.sirspline}()},
-\code{\link{plot.sirspline}()},
-\code{\link{sir_ratio}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()}
-}
-\author{
-Matti Rantanen
-}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/sir.R
+\name{sir_exp}
+\alias{sir_exp}
+\alias{sir_lex}
+\alias{sir_ag}
+\title{Calculate SMR}
+\usage{
+sir_exp(
+  x,
+  obs,
+  exp,
+  pyrs = NULL,
+  print = NULL,
+  conf.type = "profile",
+  test.type = "homogeneity",
+  conf.level = 0.95,
+  subset = NULL
+)
+
+sir_lex(x, print = NULL, breaks = NULL, ...)
+
+sir_ag(
+  x,
+  obs = "from0to1",
+  print = attr(x, "aggre.meta")$by,
+  exp = "d.exp",
+  pyrs = "pyrs",
+  ...
+)
+}
+\arguments{
+\item{x}{Data set e.g. \code{aggre} or \code{Lexis} object 
+(see: \code{\link{lexpand}})}
+
+\item{obs}{Variable name of the observed cases in the data set}
+
+\item{exp}{Variable name or expression for expected cases}
+
+\item{pyrs}{Variable name for person-years (optional)}
+
+\item{print}{Variables or expression to stratify the results}
+
+\item{conf.type}{select confidence interval type: (default=) `profile`, `wald`, `univariate`}
+
+\item{test.type}{Test for equal SIRs. Test available are 'homogeneity' and 'trend'}
+
+\item{conf.level}{Level of type-I error in confidence intervals, default 0.05 is 95\% CI}
+
+\item{subset}{a logical vector for subsetting data}
+
+\item{breaks}{a named list to split age group (age), period (per) or follow-up (fot).}
+
+\item{...}{pass arguments to \code{sir_exp}}
+}
+\value{
+A sir object
+}
+\description{
+Calculate Standardized Mortality Ratios (SMRs) using 
+a single data set that includes
+observed and expected cases and additionally person-years.
+
+\code{sir_lex} solves SMR from a \code{\link{Lexis}} object 
+calculated with \code{lexpand}.
+
+\code{sir_ag} solves SMR from a \code{\link{aggre}} object 
+calculated using \code{\link{lexpand}}.
+}
+\details{
+These functions are intended to calculate SMRs from a single data set 
+that includes both observed and expected number of cases. For example utilizing the
+argument \code{pop.haz} of the \code{\link{lexpand}}.
+
+\code{sir_lex} automatically exports the transition \code{fromXtoY} using the first
+state in \code{lex.Str} as \code{0} and all other as \code{1}. No missing values
+is allowed in observed, pop.haz or person-years.
+}
+\section{Functions}{
+\itemize{
+\item \code{sir_lex()}: 
+
+\item \code{sir_ag()}: 
+
+}}
+\examples{
+
+\donttest{
+BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+
+## Aggregated data
+x1 <- lexpand(sire, breaks = BL, status = status != 0, 
+              birth = bi_date, entry = dg_date, exit = ex_date,
+              pophaz=popmort,
+              aggre=list(sex, period = per, surv.int = fot))
+sir_ag(x1, print = 'period')
+
+
+# no aggreate or breaks
+x2 <- lexpand(sire, status = status != 0, 
+              birth = bi_date, entry = dg_date, exit = ex_date,
+              pophaz=popmort)
+sir_lex(x2, breaks = BL, print = 'per')
+}
+
+}
+\seealso{
+\code{\link{lexpand}}
+\href{../doc/sir.html}{A SIR calculation vignette}
+
+Other sir functions: 
+\code{\link{lines.sirspline}()},
+\code{\link{plot.sirspline}()},
+\code{\link{sir_ratio}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()}
+}
+\author{
+Matti Rantanen
+}
+\concept{sir functions}
diff --git a/man/sir_ratio.Rd b/man/sir_ratio.Rd
index 1baeaf4..23ac5d6 100644
--- a/man/sir_ratio.Rd
+++ b/man/sir_ratio.Rd
@@ -1,92 +1,92 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/sir_utils.R
-\name{sir_ratio}
-\alias{sir_ratio}
-\title{Confidence intervals for the ratio of two SIRs/SMRs}
-\usage{
-sir_ratio(
-  x,
-  y,
-  digits = 3,
-  alternative = "two.sided",
-  conf.level = 0.95,
-  type = "exact"
-)
-}
-\arguments{
-\item{x}{a sir-object or a vector of two; observed and expected cases.}
-
-\item{y}{a sir-object or a vector of two; observed and expected cases.}
-
-\item{digits}{number of digits in the output}
-
-\item{alternative}{The null-hypothesis test: (default:) \code{two.sided}, \code{less}, \code{greater}}
-
-\item{conf.level}{the type-I error in confidence intervals, default 0.95 for 95\% CI.}
-
-\item{type}{How the binomial confidence intervals are calculated (default:) \code{exact} or \code{asymptotic}.}
-}
-\value{
-A vector length of three: sir_ratio, and lower and upper confidence intervals.
-}
-\description{
-Calculate ratio of two SIRs/SMRs and the confidence intervals of the ratio.
-}
-\details{
-Function works with pooled sir-objects i.e. the \code{print} argument in \code{sir} is ignored.
-Also \code{x} and \code{y} can be a vector of two where first index is the
-observed cases and second is expected cases (see examples).
-Note that the ratio of two SIR's is only applicable when the age distributions are similar
-in both populations.
-
-\strong{Formula}
-
-The observed number of first sir \code{O1} is considered as a Binomial variable with sample 
-size of \code{O1+O2}. The confidence intervals for Binomial proportion \code{A} 
-is solved using \code{exact} or \code{asymptotic} 
-method. Now the CI for ratio \code{O1/O2} is \code{B = A/(1 - A)}. And further the CI for SIR/SMR 
-is B*E2/E1. (Ederer and Mantel)
-}
-\note{
-Parameter \code{alternative} is always \code{two.sided} when parameter 
-\code{type} is set to \code{asymptotic}.
-}
-\examples{
-## Ratio for sir-object and the same values given manually:
-
-
-## create example dataset
-dt1 <- data.frame(obs = rep(c(5,7), 10),
-                  pyrs = rep(c(250,300,350,400), 5),
-                  var = 1:20)
-Ref <- data.frame(obs = rep(c(50,70,80,100), 5),
-                 pyrs = rep(c(2500,3000,3500,4000), 5),
-                 var = 1:20)
-## sir using the function
-s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
-          ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
-          adjust = var)
-
-## Ratio is simply 1:
-sir_ratio(s1, c(120, 150))
-
-}
-\references{
-Statistics with Confidence: Confidence Intervals and Statistical Guidelines, 
-Douglas Altman, 2000. ISBN: 978-0-727-91375-3
-}
-\seealso{
-\code{\link{sir}}
-\href{../doc/sir.html}{A SIR calculation vignette}
-
-Other sir functions: 
-\code{\link{lines.sirspline}()},
-\code{\link{plot.sirspline}()},
-\code{\link{sir_exp}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()}
-}
-\author{
-Matti Rantanen
-}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/sir_utils.R
+\name{sir_ratio}
+\alias{sir_ratio}
+\title{Confidence intervals for the ratio of two SIRs/SMRs}
+\usage{
+sir_ratio(
+  x,
+  y,
+  digits = 3,
+  alternative = "two.sided",
+  conf.level = 0.95,
+  type = "exact"
+)
+}
+\arguments{
+\item{x}{a sir-object or a vector of two; observed and expected cases.}
+
+\item{y}{a sir-object or a vector of two; observed and expected cases.}
+
+\item{digits}{number of digits in the output}
+
+\item{alternative}{The null-hypothesis test: (default:) \code{two.sided}, \code{less}, \code{greater}}
+
+\item{conf.level}{the type-I error in confidence intervals, default 0.95 for 95\% CI.}
+
+\item{type}{How the binomial confidence intervals are calculated (default:) \code{exact} or \code{asymptotic}.}
+}
+\value{
+A vector length of three: sir_ratio, and lower and upper confidence intervals.
+}
+\description{
+Calculate ratio of two SIRs/SMRs and the confidence intervals of the ratio.
+}
+\details{
+Function works with pooled sir-objects i.e. the \code{print} argument in \code{sir} is ignored.
+Also \code{x} and \code{y} can be a vector of two where first index is the
+observed cases and second is expected cases (see examples).
+Note that the ratio of two SIR's is only applicable when the age distributions are similar
+in both populations.
+
+\strong{Formula}
+
+The observed number of first sir \code{O1} is considered as a Binomial variable with sample 
+size of \code{O1+O2}. The confidence intervals for Binomial proportion \code{A} 
+is solved using \code{exact} or \code{asymptotic} 
+method. Now the CI for ratio \code{O1/O2} is \code{B = A/(1 - A)}. And further the CI for SIR/SMR 
+is B*E2/E1. (Ederer and Mantel)
+}
+\note{
+Parameter \code{alternative} is always \code{two.sided} when parameter 
+\code{type} is set to \code{asymptotic}.
+}
+\examples{
+## Ratio for sir-object and the same values given manually:
+
+
+## create example dataset
+dt1 <- data.frame(obs = rep(c(5,7), 10),
+                  pyrs = rep(c(250,300,350,400), 5),
+                  var = 1:20)
+Ref <- data.frame(obs = rep(c(50,70,80,100), 5),
+                 pyrs = rep(c(2500,3000,3500,4000), 5),
+                 var = 1:20)
+## sir using the function
+s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
+          ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
+          adjust = var)
+
+## Ratio is simply 1:
+sir_ratio(s1, c(120, 150))
+
+}
+\references{
+Statistics with Confidence: Confidence Intervals and Statistical Guidelines, 
+Douglas Altman, 2000. ISBN: 978-0-727-91375-3
+}
+\seealso{
+\code{\link{sir}}
+\href{../doc/sir.html}{A SIR calculation vignette}
+
+Other sir functions: 
+\code{\link{lines.sirspline}()},
+\code{\link{plot.sirspline}()},
+\code{\link{sir_exp}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()}
+}
+\author{
+Matti Rantanen
+}
+\concept{sir functions}
diff --git a/man/sire.Rd b/man/sire.Rd
index 1280f23..f34c12f 100644
--- a/man/sire.Rd
+++ b/man/sire.Rd
@@ -1,45 +1,45 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{sire}
-\alias{sire}
-\title{sire - a simulated cohort of Finnish female rectal cancer patients}
-\format{
-data.table with columns
-\itemize{
- \item sex - gender of the patient (1 = female)
- \item bi_date - date of birth
- \item dg_date - date of cancer diagnosis
- \item ex_date - date of exit from follow-up (death or censoring)
- \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
- \item dg_age  - age at diagnosis expressed as fractional years
-}
-}
-\source{
-The Finnish Cancer Registry
-}
-\description{
-\code{sire} is a simulated cohort pertaining female Finnish rectal cancer patients
-diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
-via modest randomization within several time windows.
-}
-\details{
-The closing date for the pertinent data was 2012-12-31, meaning status information was
-available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
-}
-\seealso{
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{meanpop_fi}},
-\code{\link{popmort}},
-\code{\link{sibr}},
-\code{\link{stdpop101}},
-\code{\link{stdpop18}}
-
-Other survival data: 
-\code{\link{sibr}}
-}
-\author{
-Karri Seppa
-}
-\concept{popEpi data}
-\concept{survival data}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{sire}
+\alias{sire}
+\title{sire - a simulated cohort of Finnish female rectal cancer patients}
+\format{
+data.table with columns
+\itemize{
+ \item sex - gender of the patient (1 = female)
+ \item bi_date - date of birth
+ \item dg_date - date of cancer diagnosis
+ \item ex_date - date of exit from follow-up (death or censoring)
+ \item status  - status of the person at exit; 0 alive; 1 dead due to pertinent cancer; 2 dead due to other causes
+ \item dg_age  - age at diagnosis expressed as fractional years
+}
+}
+\source{
+The Finnish Cancer Registry
+}
+\description{
+\code{sire} is a simulated cohort pertaining female Finnish rectal cancer patients
+diagnosed between 1993-2012. Instead of actual original dates, the dates are masked
+via modest randomization within several time windows.
+}
+\details{
+The closing date for the pertinent data was 2012-12-31, meaning status information was
+available only up to that point --- hence the maximum possible \code{ex_date} is \code{2012-12-31}.
+}
+\seealso{
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{meanpop_fi}},
+\code{\link{popmort}},
+\code{\link{sibr}},
+\code{\link{stdpop101}},
+\code{\link{stdpop18}}
+
+Other survival data: 
+\code{\link{sibr}}
+}
+\author{
+Karri Seppa
+}
+\concept{popEpi data}
+\concept{survival data}
diff --git a/man/sirspline.Rd b/man/sirspline.Rd
index 9918a5c..709ffd4 100644
--- a/man/sirspline.Rd
+++ b/man/sirspline.Rd
@@ -1,164 +1,164 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/sir.R
-\name{sirspline}
-\alias{sirspline}
-\title{Estimate splines for SIR or SMR}
-\usage{
-sirspline(
-  coh.data,
-  coh.obs,
-  coh.pyrs,
-  ref.data = NULL,
-  ref.obs = NULL,
-  ref.pyrs = NULL,
-  ref.rate = NULL,
-  subset = NULL,
-  print = NULL,
-  adjust = NULL,
-  mstate = NULL,
-  spline,
-  knots = NULL,
-  reference.points = NULL,
-  dependent.splines = TRUE
-)
-}
-\arguments{
-\item{coh.data}{cohort data with observations and at risk time variables}
-
-\item{coh.obs}{variable name for observed cases}
-
-\item{coh.pyrs}{variable name for person-years in cohort data}
-
-\item{ref.data}{aggregated population data}
-
-\item{ref.obs}{variable name for observed cases}
-
-\item{ref.pyrs}{variable name for person-years in population data}
-
-\item{ref.rate}{population rate observed/expected. This overwrites the parameters
-\code{ref.pyrs} and \code{ref.obs}.}
-
-\item{subset}{logical condition to subset \code{coh.data} before any computations}
-
-\item{print}{variable names for which to estimate SIRs/SMRs and 
-associated splines separately}
-
-\item{adjust}{variable names for adjusting the expected cases}
-
-\item{mstate}{set column names for cause specific observations. Relevant only
-when coh.obs length is two or more. See help for \code{sir}.}
-
-\item{spline}{variable name(s) for the splines}
-
-\item{knots}{number knots (vector),  pre-defined knots (list of vectors) or for optimal number of knots left NULL}
-
-\item{reference.points}{fixed reference values for rate ratios. If left \code{NULL}
-the smallest value is the reference point (where SIR = 1). 
-Ignored if \code{dependent.splines = FALSE}}
-
-\item{dependent.splines}{logical; if TRUE, all splines are fitted in same model.}
-}
-\value{
-A list of data.frames and vectors.
-Three spline estimates are named as \code{spline.est.A/B/C} and the corresponding values
-in \code{spline.seq.A/B/C} for manual plotting
-}
-\description{
-Splines for standardised incidence or mortality ratio. A useful 
-tool to e.g. check whether a constant SIR can be assumed for all calendar periods,
-age groups or follow-up intervals. Splines can be fitted for these time dimensions
-separately or in the same model.
-}
-\details{
-See \code{\link{sir}} for help on SIR/SMR estimation in general; usage of splines
-is discussed below.
-
-\strong{The spline variables}
-
-The model can include one, two or three splines variables.
-Variables can be included in the same model selecting \code{dependent.splines = TRUE}
-and SIR ratios are calculated (first one is the SIR, others SIR ratios). 
-Reference points vector can be set via \code{reference.points}
-where first element of the vector is the reference point for first ratio.
-
-Variable(s) to fit splines are given as a vector in argument \code{spline}.
-Order will affect the results.
-
-
-\strong{dependent.splines} 
-
-By default dependent.splines is FALSE and all splines are fitted in separate models. 
-If TRUE, the first variable in \code{spline} is a function of a SIR and other(s) are ratios.
-
-\strong{knots}
-
-There are three options to set knots to splines:
-
-Set the number of knots for each spline variable with a \strong{vector}. 
-The knots are automatically placed to the quantiles of observed cases in cohort data. 
-The first and last knots are always the maximum and minimum values, so knot
-value needs to be at least two.
-
-Predefined knot places can be set with a \strong{list} of vectors.
-The vector for each spline in the list specifies the knot places. The lowest 
-and the largest values are the boundary knots and these should be checked beforehand.
-
-If \code{knots} is left \strong{NULL}, the model searches the optimal number 
-of knots by model AIC by fitting models iteratively from 2 to 15 knots and 
-the one with smallest AIC is selected.
-If \code{dependent.splines = TRUE}, the number of knots is searched by fitting each spline
-variable separately.
-
-
-\strong{print}
-
-Splines can be stratified by the levels of variable given in \code{print}. If 
-\code{print} is a vector, only the first variable is accounted for. The knots 
-are placed globally for all levels of \code{print}. This also ensures that the likelihood 
-ratio test is valid.
-Splines are also fitted independently for each level of \code{print}.
-This allows for searching interactions, e.g. by fitting spline for period 
-(\code{splines='period'}) for each age group (\code{print = 'agegroup'}).
-
-
-\strong{p-values}
-
-The output p-value is a test of whether the splines are equal (homogenous)
-at different levels of \code{print}. 
-The test is based on the likelihood ratio test, where the full model 
-includes \code{print} and is 
-compared to a null model without it.
-When \code{(dependent.splines = TRUE)} the p-value returned is a global p-value.
-Otherwise the p-value is spline-specific.
-}
-\examples{
-\donttest{
-## for examples see: vignette('sir')
-}
-}
-\seealso{
-\code{\link{splitMulti}} 
-\href{../doc/sir.html}{A SIR calculation vignette}
-
-Other sir functions: 
-\code{\link{lines.sirspline}()},
-\code{\link{plot.sirspline}()},
-\code{\link{sir_exp}()},
-\code{\link{sir_ratio}()},
-\code{\link{sir}()}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Matti Rantanen, Joonas Miettinen
-}
-\concept{main functions}
-\concept{sir functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/sir.R
+\name{sirspline}
+\alias{sirspline}
+\title{Estimate splines for SIR or SMR}
+\usage{
+sirspline(
+  coh.data,
+  coh.obs,
+  coh.pyrs,
+  ref.data = NULL,
+  ref.obs = NULL,
+  ref.pyrs = NULL,
+  ref.rate = NULL,
+  subset = NULL,
+  print = NULL,
+  adjust = NULL,
+  mstate = NULL,
+  spline,
+  knots = NULL,
+  reference.points = NULL,
+  dependent.splines = TRUE
+)
+}
+\arguments{
+\item{coh.data}{cohort data with observations and at risk time variables}
+
+\item{coh.obs}{variable name for observed cases}
+
+\item{coh.pyrs}{variable name for person-years in cohort data}
+
+\item{ref.data}{aggregated population data}
+
+\item{ref.obs}{variable name for observed cases}
+
+\item{ref.pyrs}{variable name for person-years in population data}
+
+\item{ref.rate}{population rate observed/expected. This overwrites the parameters
+\code{ref.pyrs} and \code{ref.obs}.}
+
+\item{subset}{logical condition to subset \code{coh.data} before any computations}
+
+\item{print}{variable names for which to estimate SIRs/SMRs and 
+associated splines separately}
+
+\item{adjust}{variable names for adjusting the expected cases}
+
+\item{mstate}{set column names for cause specific observations. Relevant only
+when coh.obs length is two or more. See help for \code{sir}.}
+
+\item{spline}{variable name(s) for the splines}
+
+\item{knots}{number knots (vector),  pre-defined knots (list of vectors) or for optimal number of knots left NULL}
+
+\item{reference.points}{fixed reference values for rate ratios. If left \code{NULL}
+the smallest value is the reference point (where SIR = 1). 
+Ignored if \code{dependent.splines = FALSE}}
+
+\item{dependent.splines}{logical; if TRUE, all splines are fitted in same model.}
+}
+\value{
+A list of data.frames and vectors.
+Three spline estimates are named as \code{spline.est.A/B/C} and the corresponding values
+in \code{spline.seq.A/B/C} for manual plotting
+}
+\description{
+Splines for standardised incidence or mortality ratio. A useful 
+tool to e.g. check whether a constant SIR can be assumed for all calendar periods,
+age groups or follow-up intervals. Splines can be fitted for these time dimensions
+separately or in the same model.
+}
+\details{
+See \code{\link{sir}} for help on SIR/SMR estimation in general; usage of splines
+is discussed below.
+
+\strong{The spline variables}
+
+The model can include one, two or three splines variables.
+Variables can be included in the same model selecting \code{dependent.splines = TRUE}
+and SIR ratios are calculated (first one is the SIR, others SIR ratios). 
+Reference points vector can be set via \code{reference.points}
+where first element of the vector is the reference point for first ratio.
+
+Variable(s) to fit splines are given as a vector in argument \code{spline}.
+Order will affect the results.
+
+
+\strong{dependent.splines} 
+
+By default dependent.splines is FALSE and all splines are fitted in separate models. 
+If TRUE, the first variable in \code{spline} is a function of a SIR and other(s) are ratios.
+
+\strong{knots}
+
+There are three options to set knots to splines:
+
+Set the number of knots for each spline variable with a \strong{vector}. 
+The knots are automatically placed to the quantiles of observed cases in cohort data. 
+The first and last knots are always the maximum and minimum values, so knot
+value needs to be at least two.
+
+Predefined knot places can be set with a \strong{list} of vectors.
+The vector for each spline in the list specifies the knot places. The lowest 
+and the largest values are the boundary knots and these should be checked beforehand.
+
+If \code{knots} is left \strong{NULL}, the model searches the optimal number 
+of knots by model AIC by fitting models iteratively from 2 to 15 knots and 
+the one with smallest AIC is selected.
+If \code{dependent.splines = TRUE}, the number of knots is searched by fitting each spline
+variable separately.
+
+
+\strong{print}
+
+Splines can be stratified by the levels of variable given in \code{print}. If 
+\code{print} is a vector, only the first variable is accounted for. The knots 
+are placed globally for all levels of \code{print}. This also ensures that the likelihood 
+ratio test is valid.
+Splines are also fitted independently for each level of \code{print}.
+This allows for searching interactions, e.g. by fitting spline for period 
+(\code{splines='period'}) for each age group (\code{print = 'agegroup'}).
+
+
+\strong{p-values}
+
+The output p-value is a test of whether the splines are equal (homogenous)
+at different levels of \code{print}. 
+The test is based on the likelihood ratio test, where the full model 
+includes \code{print} and is 
+compared to a null model without it.
+When \code{(dependent.splines = TRUE)} the p-value returned is a global p-value.
+Otherwise the p-value is spline-specific.
+}
+\examples{
+\donttest{
+## for examples see: vignette('sir')
+}
+}
+\seealso{
+\code{\link{splitMulti}} 
+\href{../doc/sir.html}{A SIR calculation vignette}
+
+Other sir functions: 
+\code{\link{lines.sirspline}()},
+\code{\link{plot.sirspline}()},
+\code{\link{sir_exp}()},
+\code{\link{sir_ratio}()},
+\code{\link{sir}()}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Matti Rantanen, Joonas Miettinen
+}
+\concept{main functions}
+\concept{sir functions}
diff --git a/man/splitLexisDT.Rd b/man/splitLexisDT.Rd
index 2b824b4..0a479b7 100644
--- a/man/splitLexisDT.Rd
+++ b/man/splitLexisDT.Rd
@@ -1,90 +1,90 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/splitLexisDT.R
-\name{splitLexisDT}
-\alias{splitLexisDT}
-\title{Split case-level observations}
-\usage{
-splitLexisDT(lex, breaks, timeScale, merge = TRUE, drop = TRUE)
-}
-\arguments{
-\item{lex}{a Lexis object, split or not}
-
-\item{breaks}{a vector of \code{[a,b)} breaks to split \code{data} by}
-
-\item{timeScale}{a character string; name of the time scale to split by}
-
-\item{merge}{logical; if \code{TRUE}, retains all variables 
-from the original data - i.e. original variables are
-repeated for all the rows by original subject}
-
-\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
-after expansion that reside outside the time window
-defined by the given breaks}
-}
-\value{
-A \code{data.table} or \code{data.frame} 
-(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-object expanded to accommodate split observations.
-}
-\description{
-Split a \code{Lexis} object along one time scale
-(as \code{\link[Epi]{splitLexis}}) with speed
-}
-\details{
-\code{splitLexisDT} is in essence a \pkg{data.table} version of
-\code{splitLexis} or \code{survSplit} for splitting along a single
-time scale. It requires a Lexis object as input, which may have already
-been split along some time scale.
-
-Unlike \code{splitLexis}, \code{splitLexisDT} drops observed time outside
-the roof and floor of \code{breaks} by default - with \code{drop = FALSE}
-the functions have identical behaviour.
-
-The \code{Lexis} time scale variables can be of any arbitrary 
-format, e.g. \code{Date},
-fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
-or other. However, using \code{date} variables (from package \pkg{date})
-are not recommended, as \code{date} variables are always stored as integers,
-whereas \code{Date} variables (see \code{?as.Date}) are typically stored
-in double ("numeric") format. This allows for breaking days into fractions
-as well, when using e.g. hypothetical years of 365.25 days.
-}
-\examples{
-library(Epi)
-data("sire", package = "popEpi")
-x <- Lexis(data=sire[1000:1100, ], 
-           entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
-           exit=list(per=get.yrs(ex_date)), exit.status=status)
-BL <- list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
-
-x2 <- splitMulti(x, breaks = BL, drop = FALSE)
-
-x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
-x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
-
-x4 <- splitLexis(x,  breaks = BL$fot, time.scale = "fot")
-x4 <- splitLexis(x4, breaks = BL$per, time.scale = "per")
-## all produce identical results
-
-## using Date variables
-x <- Lexis(data=sire[1000:1100, ], 
-           entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
-           exit=list(per=ex_date), exit.status=status)
-BL <- list(fot = 0:5*365.25, per = as.Date(c("2008-01-01", "2013-01-01")))
-
-x2 <- splitMulti(x, breaks = BL, drop = FALSE)
-
-x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
-x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
-
-## splitLexis may not work when using Dates
-}
-\seealso{
-Other splitting functions: 
-\code{\link{lexpand}()},
-\code{\link{splitMulti}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{splitting functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/splitLexisDT.R
+\name{splitLexisDT}
+\alias{splitLexisDT}
+\title{Split case-level observations}
+\usage{
+splitLexisDT(lex, breaks, timeScale, merge = TRUE, drop = TRUE)
+}
+\arguments{
+\item{lex}{a Lexis object, split or not}
+
+\item{breaks}{a vector of \code{[a,b)} breaks to split \code{data} by}
+
+\item{timeScale}{a character string; name of the time scale to split by}
+
+\item{merge}{logical; if \code{TRUE}, retains all variables 
+from the original data - i.e. original variables are
+repeated for all the rows by original subject}
+
+\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
+after expansion that reside outside the time window
+defined by the given breaks}
+}
+\value{
+A \code{data.table} or \code{data.frame} 
+(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+object expanded to accommodate split observations.
+}
+\description{
+Split a \code{Lexis} object along one time scale
+(as \code{\link[Epi]{splitLexis}}) with speed
+}
+\details{
+\code{splitLexisDT} is in essence a \pkg{data.table} version of
+\code{splitLexis} or \code{survSplit} for splitting along a single
+time scale. It requires a Lexis object as input, which may have already
+been split along some time scale.
+
+Unlike \code{splitLexis}, \code{splitLexisDT} drops observed time outside
+the roof and floor of \code{breaks} by default - with \code{drop = FALSE}
+the functions have identical behaviour.
+
+The \code{Lexis} time scale variables can be of any arbitrary 
+format, e.g. \code{Date},
+fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
+or other. However, using \code{date} variables (from package \pkg{date})
+are not recommended, as \code{date} variables are always stored as integers,
+whereas \code{Date} variables (see \code{?as.Date}) are typically stored
+in double ("numeric") format. This allows for breaking days into fractions
+as well, when using e.g. hypothetical years of 365.25 days.
+}
+\examples{
+library(Epi)
+data("sire", package = "popEpi")
+x <- Lexis(data=sire[1000:1100, ], 
+           entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
+           exit=list(per=get.yrs(ex_date)), exit.status=status)
+BL <- list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
+
+x2 <- splitMulti(x, breaks = BL, drop = FALSE)
+
+x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
+x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
+
+x4 <- splitLexis(x,  breaks = BL$fot, time.scale = "fot")
+x4 <- splitLexis(x4, breaks = BL$per, time.scale = "per")
+## all produce identical results
+
+## using Date variables
+x <- Lexis(data=sire[1000:1100, ], 
+           entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
+           exit=list(per=ex_date), exit.status=status)
+BL <- list(fot = 0:5*365.25, per = as.Date(c("2008-01-01", "2013-01-01")))
+
+x2 <- splitMulti(x, breaks = BL, drop = FALSE)
+
+x3 <- splitLexisDT(x, breaks = BL$fot, timeScale = "fot", drop = FALSE)
+x3 <- splitLexisDT(x3, breaks = BL$per, timeScale = "per", drop = FALSE)
+
+## splitLexis may not work when using Dates
+}
+\seealso{
+Other splitting functions: 
+\code{\link{lexpand}()},
+\code{\link{splitMulti}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{splitting functions}
diff --git a/man/splitMulti.Rd b/man/splitMulti.Rd
index 546e256..227a51f 100644
--- a/man/splitMulti.Rd
+++ b/man/splitMulti.Rd
@@ -1,150 +1,150 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/splitMulti.R
-\name{splitMulti}
-\alias{splitMulti}
-\title{Split case-level observations}
-\usage{
-splitMulti(
-  data,
-  breaks = NULL,
-  ...,
-  drop = TRUE,
-  merge = TRUE,
-  verbose = FALSE
-)
-}
-\arguments{
-\item{data}{a Lexis object with event cases as rows}
-
-\item{breaks}{a list of named numeric vectors of breaks; see Details and Examples}
-
-\item{...}{alternate way of supplying breaks as named vectors;
-e.g. \code{fot = 0:5} instead of \code{breaks = list(fot = 0:5)};
-if \code{breaks} is not \code{NULL}, \code{breaks} is used and any breaks
-passed through \code{...} are NOT used; note also that due to partial 
-matching of argument names in R, 
-if you supply e.g. \code{dat = my_breaks} and you 
-do not pass argument \code{data} explicitly (\code{data = my_data}), then R
-interprets this as \code{data = my_breaks} --- so choose the names of your
-time scales wisely}
-
-\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
-after expansion that reside outside the time window
-defined by the given breaks}
-
-\item{merge}{logical; if \code{TRUE}, retains all variables 
-from the original data - i.e. original variables are
-repeated for all the rows by original subject}
-
-\item{verbose}{logical; if \code{TRUE}, the function is chatty 
-and returns some messages along the way}
-}
-\value{
-A \code{data.table} or \code{data.frame} 
-(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
-object expanded to accommodate split observations.
-}
-\description{
-Split a \code{Lexis} object along multiple time scales
-with speed and ease
-}
-\details{
-\code{splitMulti} is in essence a \pkg{data.table} version of
- \code{splitLexis} or \code{survSplit} for splitting along multiple
- time scales.
-It requires a Lexis object as input.
-
-The \code{breaks} must be a list of named vectors of the appropriate type. 
-The breaks are fully explicit and
-left-inclusive and right exclusive, e.g. \code{fot=c(0,5)} 
-forces the data to only include time between
-\code{[0,5)} for each original row (unless \code{drop = FALSE}). 
-Use \code{Inf} or \code{-Inf} for open-ended intervals,
- e.g. \code{per=c(1990,1995,Inf)} creates the intervals 
- \code{[1990,1995), [1995, Inf)}.
- 
-Instead of specifying \code{breaks}, one may make use of the \code{...}
-argument to pass breaks: e.g. 
-
-\code{splitMulti(x, breaks = list(fot = 0:5))} 
-
-is equivalent to
-
-\code{splitMulti(x, fot = 0:5)}.
-
-Multiple breaks can be supplied in the same manner. However, if both
-\code{breaks} and \code{...} are used, only the breaks in \code{breaks}
-are utilized within the function. 
-
-The \code{Lexis} time scale variables can be of any arbitrary 
-format, e.g. \code{Date},
-fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
-or other. However, using \code{date} variables (from package \pkg{date})
-are not recommended, as \code{date} variables are always stored as integers,
-whereas \code{Date} variables (see \code{?as.Date}) are typically stored
-in double ("numeric") format. This allows for breaking days into fractions
-as well, when using e.g. hypothetical years of 365.25 days.
-}
-\examples{
-#### let's prepare data for computing period method survivals
-#### in case there are problems with dates, we first 
-#### convert to fractional years.
-\donttest{
-library("Epi")
-library("data.table")
-data("sire", package = "popEpi")
-x <- Lexis(data=sire[dg_date < ex_date, ], 
-           entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
-           exit=list(per=get.yrs(ex_date)), exit.status=status)
-x2 <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013)))
-# equivalently:
-x2 <- splitMulti(x, fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
-
-## using dates; note: breaks must be expressed as dates or days!
-x <- Lexis(data=sire[dg_date < ex_date, ], 
-           entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
-           exit=list(per=ex_date), exit.status=status)
-BL <- list(fot = seq(0, 5, by = 3/12)*365.242199,
-           per = as.Date(paste0(c(1980:2014),"-01-01")),
-           age = c(0,45,85,Inf)*365.242199)
-x2 <- splitMulti(x, breaks = BL, verbose=TRUE)
-
-
-## multistate example (healty - sick - dead)
-sire2 <- data.frame(sire)
-sire2 <- sire2[sire2$dg_date < sire2$ex_date, ]
-
-set.seed(1L) 
-not_sick <- sample.int(nrow(sire2), 6000L, replace = FALSE)
-sire2$dg_date[not_sick] <- NA
-sire2$status[!is.na(sire2$dg_date) & sire2$status == 0] <- -1
-
-sire2$status[sire2$status==2] <- 1
-sire2$status <- factor(sire2$status, levels = c(0, -1, 1), 
-                       labels = c("healthy", "sick", "dead"))
- 
-xm <- Lexis(data = sire2, 
-            entry = list(fot=0, per=get.yrs(bi_date), age=0), 
-            exit = list(per=get.yrs(ex_date)), exit.status=status)
-xm2 <- cutLexis(xm, cut = get.yrs(xm$dg_date), 
-                timescale = "per", 
-                new.state = "sick")
-xm2[xm2$lex.id == 6L, ]
-
-xm2 <- splitMulti(xm2, breaks = list(fot = seq(0,150,25)))
-xm2[xm2$lex.id == 6L, ]
-}
-
-}
-\seealso{
-\code{\link[Epi]{splitLexis}}, \code{\link[Epi]{Lexis}},  
-\code{\link[survival]{survSplit}}
-
-Other splitting functions: 
-\code{\link{lexpand}()},
-\code{\link{splitLexisDT}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{splitting functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/splitMulti.R
+\name{splitMulti}
+\alias{splitMulti}
+\title{Split case-level observations}
+\usage{
+splitMulti(
+  data,
+  breaks = NULL,
+  ...,
+  drop = TRUE,
+  merge = TRUE,
+  verbose = FALSE
+)
+}
+\arguments{
+\item{data}{a Lexis object with event cases as rows}
+
+\item{breaks}{a list of named numeric vectors of breaks; see Details and Examples}
+
+\item{...}{alternate way of supplying breaks as named vectors;
+e.g. \code{fot = 0:5} instead of \code{breaks = list(fot = 0:5)};
+if \code{breaks} is not \code{NULL}, \code{breaks} is used and any breaks
+passed through \code{...} are NOT used; note also that due to partial 
+matching of argument names in R, 
+if you supply e.g. \code{dat = my_breaks} and you 
+do not pass argument \code{data} explicitly (\code{data = my_data}), then R
+interprets this as \code{data = my_breaks} --- so choose the names of your
+time scales wisely}
+
+\item{drop}{logical; if \code{TRUE}, drops all resulting rows 
+after expansion that reside outside the time window
+defined by the given breaks}
+
+\item{merge}{logical; if \code{TRUE}, retains all variables 
+from the original data - i.e. original variables are
+repeated for all the rows by original subject}
+
+\item{verbose}{logical; if \code{TRUE}, the function is chatty 
+and returns some messages along the way}
+}
+\value{
+A \code{data.table} or \code{data.frame} 
+(depending on \code{options("popEpi.datatable")}; see \code{?popEpi}) 
+object expanded to accommodate split observations.
+}
+\description{
+Split a \code{Lexis} object along multiple time scales
+with speed and ease
+}
+\details{
+\code{splitMulti} is in essence a \pkg{data.table} version of
+ \code{splitLexis} or \code{survSplit} for splitting along multiple
+ time scales.
+It requires a Lexis object as input.
+
+The \code{breaks} must be a list of named vectors of the appropriate type. 
+The breaks are fully explicit and
+left-inclusive and right exclusive, e.g. \code{fot=c(0,5)} 
+forces the data to only include time between
+\code{[0,5)} for each original row (unless \code{drop = FALSE}). 
+Use \code{Inf} or \code{-Inf} for open-ended intervals,
+ e.g. \code{per=c(1990,1995,Inf)} creates the intervals 
+ \code{[1990,1995), [1995, Inf)}.
+ 
+Instead of specifying \code{breaks}, one may make use of the \code{...}
+argument to pass breaks: e.g. 
+
+\code{splitMulti(x, breaks = list(fot = 0:5))} 
+
+is equivalent to
+
+\code{splitMulti(x, fot = 0:5)}.
+
+Multiple breaks can be supplied in the same manner. However, if both
+\code{breaks} and \code{...} are used, only the breaks in \code{breaks}
+are utilized within the function. 
+
+The \code{Lexis} time scale variables can be of any arbitrary 
+format, e.g. \code{Date},
+fractional years (see \code{\link[Epi]{cal.yr}}) and \code{\link{get.yrs}},
+or other. However, using \code{date} variables (from package \pkg{date})
+are not recommended, as \code{date} variables are always stored as integers,
+whereas \code{Date} variables (see \code{?as.Date}) are typically stored
+in double ("numeric") format. This allows for breaking days into fractions
+as well, when using e.g. hypothetical years of 365.25 days.
+}
+\examples{
+#### let's prepare data for computing period method survivals
+#### in case there are problems with dates, we first 
+#### convert to fractional years.
+\donttest{
+library("Epi")
+library("data.table")
+data("sire", package = "popEpi")
+x <- Lexis(data=sire[dg_date < ex_date, ], 
+           entry = list(fot=0, per=get.yrs(dg_date), age=dg_age), 
+           exit=list(per=get.yrs(ex_date)), exit.status=status)
+x2 <- splitMulti(x, breaks = list(fot=seq(0, 5, by = 3/12), per=c(2008, 2013)))
+# equivalently:
+x2 <- splitMulti(x, fot=seq(0, 5, by = 3/12), per=c(2008, 2013))
+
+## using dates; note: breaks must be expressed as dates or days!
+x <- Lexis(data=sire[dg_date < ex_date, ], 
+           entry = list(fot=0, per=dg_date, age=dg_date-bi_date), 
+           exit=list(per=ex_date), exit.status=status)
+BL <- list(fot = seq(0, 5, by = 3/12)*365.242199,
+           per = as.Date(paste0(c(1980:2014),"-01-01")),
+           age = c(0,45,85,Inf)*365.242199)
+x2 <- splitMulti(x, breaks = BL, verbose=TRUE)
+
+
+## multistate example (healty - sick - dead)
+sire2 <- data.frame(sire)
+sire2 <- sire2[sire2$dg_date < sire2$ex_date, ]
+
+set.seed(1L) 
+not_sick <- sample.int(nrow(sire2), 6000L, replace = FALSE)
+sire2$dg_date[not_sick] <- NA
+sire2$status[!is.na(sire2$dg_date) & sire2$status == 0] <- -1
+
+sire2$status[sire2$status==2] <- 1
+sire2$status <- factor(sire2$status, levels = c(0, -1, 1), 
+                       labels = c("healthy", "sick", "dead"))
+ 
+xm <- Lexis(data = sire2, 
+            entry = list(fot=0, per=get.yrs(bi_date), age=0), 
+            exit = list(per=get.yrs(ex_date)), exit.status=status)
+xm2 <- cutLexis(xm, cut = get.yrs(xm$dg_date), 
+                timescale = "per", 
+                new.state = "sick")
+xm2[xm2$lex.id == 6L, ]
+
+xm2 <- splitMulti(xm2, breaks = list(fot = seq(0,150,25)))
+xm2[xm2$lex.id == 6L, ]
+}
+
+}
+\seealso{
+\code{\link[Epi]{splitLexis}}, \code{\link[Epi]{Lexis}},  
+\code{\link[survival]{survSplit}}
+
+Other splitting functions: 
+\code{\link{lexpand}()},
+\code{\link{splitLexisDT}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{splitting functions}
diff --git a/man/stdpop101.Rd b/man/stdpop101.Rd
index ef2f96a..3dcb1a8 100644
--- a/man/stdpop101.Rd
+++ b/man/stdpop101.Rd
@@ -1,35 +1,35 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{stdpop101}
-\alias{stdpop101}
-\title{World standard population by 1 year age groups from 1 to 101. Sums to 100 000.}
-\format{
-data.table with columns
-\itemize{
- \item \code{world_std} weight that sums to 100000 (numeric)
- \item \code{agegroup} age group from 1 to 101 (numeric)
-}
-}
-\source{
-Standard population is from:
-\href{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}{world standard population "101of1"}
-}
-\description{
-World standard population by 1 year age groups from 1 to 101. Sums to 100 000.
-}
-\seealso{
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{meanpop_fi}},
-\code{\link{popmort}},
-\code{\link{sibr}},
-\code{\link{sire}},
-\code{\link{stdpop18}}
-
-Other weights: 
-\code{\link{ICSS}},
-\code{\link{direct_standardization}},
-\code{\link{stdpop18}}
-}
-\concept{popEpi data}
-\concept{weights}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{stdpop101}
+\alias{stdpop101}
+\title{World standard population by 1 year age groups from 1 to 101. Sums to 100 000.}
+\format{
+data.table with columns
+\itemize{
+ \item \code{world_std} weight that sums to 100000 (numeric)
+ \item \code{agegroup} age group from 1 to 101 (numeric)
+}
+}
+\source{
+Standard population is from:
+\href{https://seer.cancer.gov/stdpopulations/stdpop.singleages.html}{world standard population "101of1"}
+}
+\description{
+World standard population by 1 year age groups from 1 to 101. Sums to 100 000.
+}
+\seealso{
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{meanpop_fi}},
+\code{\link{popmort}},
+\code{\link{sibr}},
+\code{\link{sire}},
+\code{\link{stdpop18}}
+
+Other weights: 
+\code{\link{ICSS}},
+\code{\link{direct_standardization}},
+\code{\link{stdpop18}}
+}
+\concept{popEpi data}
+\concept{weights}
diff --git a/man/stdpop18.Rd b/man/stdpop18.Rd
index 3c78f65..7063ac7 100644
--- a/man/stdpop18.Rd
+++ b/man/stdpop18.Rd
@@ -1,37 +1,37 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/data_document.R
-\name{stdpop18}
-\alias{stdpop18}
-\title{Standard populations from 2000: world, Europe and Nordic.}
-\format{
-data.table with columns
-\itemize{
- \item \code{agegroup}, age group in 18 categories (character)
- \item \code{world}, World 2000 standard population (numeric)
- \item \code{europe}, European standard population (numeric)
- \item \code{nordic}, Nordic standard population (numeric)
-}
-}
-\source{
-Nordcan, 2000
-}
-\description{
-World, European, and Nordic standard populations by 18 age categories. 
-Sums to 100000.
-}
-\seealso{
-Other popEpi data: 
-\code{\link{ICSS}},
-\code{\link{meanpop_fi}},
-\code{\link{popmort}},
-\code{\link{sibr}},
-\code{\link{sire}},
-\code{\link{stdpop101}}
-
-Other weights: 
-\code{\link{ICSS}},
-\code{\link{direct_standardization}},
-\code{\link{stdpop101}}
-}
-\concept{popEpi data}
-\concept{weights}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/data_document.R
+\name{stdpop18}
+\alias{stdpop18}
+\title{Standard populations from 2000: world, Europe and Nordic.}
+\format{
+data.table with columns
+\itemize{
+ \item \code{agegroup}, age group in 18 categories (character)
+ \item \code{world}, World 2000 standard population (numeric)
+ \item \code{europe}, European standard population (numeric)
+ \item \code{nordic}, Nordic standard population (numeric)
+}
+}
+\source{
+Nordcan, 2000
+}
+\description{
+World, European, and Nordic standard populations by 18 age categories. 
+Sums to 100000.
+}
+\seealso{
+Other popEpi data: 
+\code{\link{ICSS}},
+\code{\link{meanpop_fi}},
+\code{\link{popmort}},
+\code{\link{sibr}},
+\code{\link{sire}},
+\code{\link{stdpop101}}
+
+Other weights: 
+\code{\link{ICSS}},
+\code{\link{direct_standardization}},
+\code{\link{stdpop101}}
+}
+\concept{popEpi data}
+\concept{weights}
diff --git a/man/summary.aggre.Rd b/man/summary.aggre.Rd
index f86f319..8c98d2d 100644
--- a/man/summary.aggre.Rd
+++ b/man/summary.aggre.Rd
@@ -1,38 +1,38 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{summary.aggre}
-\alias{summary.aggre}
-\title{Summarize an \code{aggre} Object}
-\usage{
-\method{summary}{aggre}(object, by = NULL, subset = NULL, ...)
-}
-\arguments{
-\item{object}{an \code{aggre} object}
-
-\item{by}{list of columns to summarize by - e.g. \code{list(V1, V2)}
-where \code{V1} and \code{V2} are columns in the data.}
-
-\item{subset}{a logical condition to subset results table by
-before summarizing; use this to limit to a certain stratum. E.g.
-\code{subset = sex == "male"}}
-
-\item{...}{unused}
-}
-\value{
-Returns a `data.table` --- a further aggregated version of `object`.
-}
-\description{
-\code{summary} method function for \code{aggre} objects; see
-\code{\link{as.aggre}} and \code{\link{aggre}}.
-}
-\seealso{
-Other aggregation functions: 
-\code{\link{aggre}()},
-\code{\link{as.aggre}()},
-\code{\link{lexpand}()},
-\code{\link{setaggre}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{aggregation functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{summary.aggre}
+\alias{summary.aggre}
+\title{Summarize an \code{aggre} Object}
+\usage{
+\method{summary}{aggre}(object, by = NULL, subset = NULL, ...)
+}
+\arguments{
+\item{object}{an \code{aggre} object}
+
+\item{by}{list of columns to summarize by - e.g. \code{list(V1, V2)}
+where \code{V1} and \code{V2} are columns in the data.}
+
+\item{subset}{a logical condition to subset results table by
+before summarizing; use this to limit to a certain stratum. E.g.
+\code{subset = sex == "male"}}
+
+\item{...}{unused}
+}
+\value{
+Returns a `data.table` --- a further aggregated version of `object`.
+}
+\description{
+\code{summary} method function for \code{aggre} objects; see
+\code{\link{as.aggre}} and \code{\link{aggre}}.
+}
+\seealso{
+Other aggregation functions: 
+\code{\link{aggre}()},
+\code{\link{as.aggre}()},
+\code{\link{lexpand}()},
+\code{\link{setaggre}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{aggregation functions}
diff --git a/man/summary.survtab.Rd b/man/summary.survtab.Rd
index 99b69c9..d3dc1a3 100644
--- a/man/summary.survtab.Rd
+++ b/man/summary.survtab.Rd
@@ -1,100 +1,100 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/S3_definitions.R
-\name{summary.survtab}
-\alias{summary.survtab}
-\title{Summarize a survtab Object}
-\usage{
-\method{summary}{survtab}(object, t = NULL, subset = NULL, q = NULL, ...)
-}
-\arguments{
-\item{object}{a \code{survtab} object}
-
-\item{t}{a vector of times at which time points (actually intervals that
-contain t) to print summary table of survival function estimates by strata;
-values not existing in any interval cause rows containing only \code{NAs} to
-be returned.}
-
-\item{subset}{a logical condition to subset results table by
-before printing; use this to limit to a certain stratum. E.g.
-\code{subset = sex == "male"}}
-
-\item{q}{a named \code{list} of quantiles to include in returned data set,
-where names must match to estimates in \code{object};
-returns intervals where the quantiles are reached first;
-e.g. \code{list(surv.obs = 0.5)} finds the interval where \code{surv.obs}
-is 0.45 and 0.55 at the beginning and end of the interval, respectively;
-returns rows with \code{NA} values for quantiles not reached in estimates
-(e.g. if \code{q = list(surv.obs = 0.5)} but lowest estimate is 0.6);
-see Examples.}
-
-\item{...}{unused; required for congruence with other \code{summary} methods}
-}
-\value{
-A `data.table`: a slice from `object` based on `t`, `subset`, and `q`.
-}
-\description{
-Summary method function for \code{survtab} objects; see
-\code{\link{survtab_ag}}. Returns estimates at given time points
-or all time points if \code{t} and \code{q} are both \code{NULL}.
-}
-\details{
-Note that this function returns the intervals and NOT the time points
-corresponding to quantiles / estimates corresponding to time points.
-If you want precise estimates at time points that are not interval breaks,
-add the time points as breaks and re-estimate the survival time function.
-In interval-based estimation, the estimates denote e.g. probability of 
-dying \emph{during} the interval, so time points within the intervals
-are not usually considered at all. See e.g. Seppa, Dyba, and Hakulinen 
-(2015).
-}
-\examples{
-
-library(Epi)
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-           labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-## pretend some are male
-set.seed(1L)
-x$sex <- rbinom(nrow(x), 1, 0.5)
-## observed survival
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                  surv.type = "cif.obs",
-                  breaks = list(FUT = seq(0, 5, 1/12)))
-
-## estimates at full years of follow-up
-summary(st, t = 1:5)
-
-## interval estimate closest to 75th percentile, i.e.
-## first interval where surv.obs < 0.75 at end
-## (just switch 0.75 to 0.5 for median survival, etc.)
-summary(st, q = list(surv.obs = 0.75))
-## multiple quantiles
-summary(st, q = list(surv.obs = c(0.75, 0.90), CIF_canD = 0.20))
-
-## if you want all estimates in a new data.frame, you can also simply do
-
-x <- as.data.frame(st)
-}
-\references{
-Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
-Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
-\doi{10.1016/B978-0-12-801238-3.02745-8}
-}
-\seealso{
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survtab}()},
-\code{\link{plot.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/S3_definitions.R
+\name{summary.survtab}
+\alias{summary.survtab}
+\title{Summarize a survtab Object}
+\usage{
+\method{summary}{survtab}(object, t = NULL, subset = NULL, q = NULL, ...)
+}
+\arguments{
+\item{object}{a \code{survtab} object}
+
+\item{t}{a vector of times at which time points (actually intervals that
+contain t) to print summary table of survival function estimates by strata;
+values not existing in any interval cause rows containing only \code{NAs} to
+be returned.}
+
+\item{subset}{a logical condition to subset results table by
+before printing; use this to limit to a certain stratum. E.g.
+\code{subset = sex == "male"}}
+
+\item{q}{a named \code{list} of quantiles to include in returned data set,
+where names must match to estimates in \code{object};
+returns intervals where the quantiles are reached first;
+e.g. \code{list(surv.obs = 0.5)} finds the interval where \code{surv.obs}
+is 0.45 and 0.55 at the beginning and end of the interval, respectively;
+returns rows with \code{NA} values for quantiles not reached in estimates
+(e.g. if \code{q = list(surv.obs = 0.5)} but lowest estimate is 0.6);
+see Examples.}
+
+\item{...}{unused; required for congruence with other \code{summary} methods}
+}
+\value{
+A `data.table`: a slice from `object` based on `t`, `subset`, and `q`.
+}
+\description{
+Summary method function for \code{survtab} objects; see
+\code{\link{survtab_ag}}. Returns estimates at given time points
+or all time points if \code{t} and \code{q} are both \code{NULL}.
+}
+\details{
+Note that this function returns the intervals and NOT the time points
+corresponding to quantiles / estimates corresponding to time points.
+If you want precise estimates at time points that are not interval breaks,
+add the time points as breaks and re-estimate the survival time function.
+In interval-based estimation, the estimates denote e.g. probability of 
+dying \emph{during} the interval, so time points within the intervals
+are not usually considered at all. See e.g. Seppa, Dyba, and Hakulinen 
+(2015).
+}
+\examples{
+
+library(Epi)
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+           labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+## pretend some are male
+set.seed(1L)
+x$sex <- rbinom(nrow(x), 1, 0.5)
+## observed survival
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                  surv.type = "cif.obs",
+                  breaks = list(FUT = seq(0, 5, 1/12)))
+
+## estimates at full years of follow-up
+summary(st, t = 1:5)
+
+## interval estimate closest to 75th percentile, i.e.
+## first interval where surv.obs < 0.75 at end
+## (just switch 0.75 to 0.5 for median survival, etc.)
+summary(st, q = list(surv.obs = 0.75))
+## multiple quantiles
+summary(st, q = list(surv.obs = c(0.75, 0.90), CIF_canD = 0.20))
+
+## if you want all estimates in a new data.frame, you can also simply do
+
+x <- as.data.frame(st)
+}
+\references{
+Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
+Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
+\doi{10.1016/B978-0-12-801238-3.02745-8}
+}
+\seealso{
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survtab}()},
+\code{\link{plot.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{survtab functions}
diff --git a/man/survmean.Rd b/man/survmean.Rd
index 4e18b01..22eb9d2 100644
--- a/man/survmean.Rd
+++ b/man/survmean.Rd
@@ -1,299 +1,299 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/mean_survival.R
-\name{survmean}
-\alias{survmean}
-\title{Compute Mean Survival Times Using Extrapolation}
-\usage{
-survmean(
-  formula,
-  data,
-  adjust = NULL,
-  weights = NULL,
-  breaks = NULL,
-  pophaz = NULL,
-  e1.breaks = NULL,
-  e1.pophaz = pophaz,
-  r = "auto",
-  surv.method = "hazard",
-  subset = NULL,
-  verbose = FALSE
-)
-}
-\arguments{
-\item{formula}{a \code{formula}, e.g. \code{FUT ~ V1} or 
-\code{Surv(FUT, lex.Xst) ~ V1}.
-Supplied in the same way as to \code{\link{survtab}}, see that help
-for more info.}
-
-\item{data}{a \code{Lexis} data set; see \code{\link[Epi]{Lexis}}.}
-
-\item{adjust}{variables to adjust estimates by, e.g. \code{adjust = "agegr"}.
-\link[=flexible_argument]{Flexible input}.}
-
-\item{weights}{weights to use to adjust mean survival times. See the
-\link[=direct_standardization]{dedicated help page} for more details on 
-weighting. \code{survmean}
-computes curves separately by all variables to adjust by, computes mean
-survival times, and computes weighted means of the mean survival times.
-See Examples.}
-
-\item{breaks}{a list of breaks defining the time window to compute 
-observed survival in, and the intervals used in estimation. E.g.
-\code{list(FUT = 0:10)} when \code{FUT} is the follow-up time scale in your
-data.}
-
-\item{pophaz}{a data set of population hazards passed to 
-\code{\link{survtab}} (see the 
-\link[=pophaz]{dedicated help page} and the help page of
-\code{survtab} for more information). Defines the 
-population hazard in the time window where observed survival is estimated.}
-
-\item{e1.breaks}{\code{NULL} or a list of breaks defining the time 
-window to compute 
-\strong{expected} survival in, and the intervals used in estimation. E.g.
-\code{list(FUT = 0:100)} when \code{FUT} is the follow-up time scale in your
-data to extrapolate up to 100 years from where the observed survival
-curve ends. \strong{NOTE:} the breaks on the survival time scale
-MUST include the breaks supplied to argument \code{breaks}; see Examples.
-If \code{NULL}, uses decent defaults (maximum follow-up time of 50 years).}
-
-\item{e1.pophaz}{Same as \code{pophaz}, except this defines the 
-population hazard in the time window where \strong{expected} 
-survival is estimated. By default uses the same data as 
-argument \code{pophaz}.}
-
-\item{r}{either a numeric multiplier such as \code{0.995}, \code{"auto"}, or
-\code{"autoX"} where \code{X} is an integer;
-used to determine the relative survival ratio (RSR) persisting after where 
-the estimated observed survival curve ends. See Details.}
-
-\item{surv.method}{passed to \code{survtab}; see that help for more info.}
-
-\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
-subsets the data before computations}
-
-\item{verbose}{\code{logical}; if \code{TRUE}, the function is returns
-some messages and results along the run, which may be useful in debugging}
-}
-\value{
-Returns a \code{data.frame} or \code{data.table} (depending on 
-\code{getOptions("popEpi.datatable")}; see \code{?popEpi}) containing the
-following columns:
-\itemize{
-  \item{est}{: The estimated mean survival time}
-  \item{exp}{: The computed expected survival time}
-  \item{obs}{: Counts of subjects in data}
-  \item{YPLL}{: Years of Potential Life Lost, computed as 
-  (\code{(exp-est)*obs}) - though your time data may be in e.g. days,
-  this column will have the same name regardless.}
-}
-The returned data also has columns named according to the variables
-supplied to the right-hand-side of the formula.
-}
-\description{
-Computes mean survival times based on survival estimation up to
-a point in follow-up time (e.g. 10 years), 
-after which survival is extrapolated
-using an appropriate hazard data file (\code{pophaz}) to yield the "full"
-survival curve. The area under the full survival curve is the mean survival.
-}
-\details{
-\strong{Basics}
-
-\code{survmean} computes mean survival times. For median survival times
-(i.e. where 50 % of subjects have died or met some other event)
-use \code{\link{survtab}}.
-
-The mean survival time is simply the area under the survival curve.
-However, since full follow-up rarely happens, the observed survival curves
-are extrapolated using expected survival: E.g. one might compute observed
-survival till up to 10 years and extrapolate beyond that 
-(till e.g. 50 years) to yield an educated guess on the full observed survival
-curve. 
-
-The area is computed by trapezoidal integration of the area under the curve.
-This function also computes the "full" expected survival curve from
-T = 0 till e.g. T = 50 depending on supplied arguments. The
-expected mean survival time is the area under the 
-mean expected survival curve.
-This function returns the mean expected survival time to be compared with 
-the mean survival time and for computing years of potential life lost (YPLL).
-
-Results can be formed by strata and adjusted for e.g. age by using
-the \code{formula} argument as in \code{survtab}. See also Examples.
-
-\strong{Extrapolation tweaks}
-
-Argument \code{r} controls the relative survival ratio (RSR) assumed to
-persist beyond the time window where observed survival is computed
-(defined by argument \code{breaks}; e.g. up to \code{FUT = 10}).
-The RSR is simply \code{RSR_i = p_oi / p_ei} for a time interval \code{i}, 
-i.e. the observed divided by the expected 
-(conditional, not cumulative) probability of surviving from the beginning of
-a time interval till its end. The cumulative product of \code{RSR_i}
-over time is the (cumulative) relative survival curve. 
-
-
-If \code{r} is numeric, e.g. \code{r = 0.995}, that RSR level is assumed
-to persist beyond the observed survival curve. 
-Numeric \code{r} should be \code{> 0} and expressed at the annual level
-when using fractional years as the scale of the time variables.
-E.g. if RSR is known to be \code{0.95} at the month level, then the
-annualized RSR is \code{0.95^12}. This enables correct usage of the RSR
-with survival intervals of varying lengths. When using day-level time 
-variables (such as \code{Dates}; see \code{as.Date}), numeric \code{r}
-should be expressed at the day level, etc.
-
-If \code{r = "auto"} or \code{r = "auto1"}, this function computes
-RSR estimates internally and automatically uses the \code{RSR_i}
-in the last survival interval in each stratum (and adjusting group)
-and assumes that to persist beyond the observed survival curve.
-Automatic determination of \code{r} is a good starting point,
-but in situations where the RSR estimate is uncertain it may produce poor
-results. Using \code{"autoX"} such as \code{"auto6"} causes \code{survmean}
-to use the mean of the estimated RSRs in the last X survival intervals, 
-which may be more stable.
-Automatic determination will not use values \code{>1} but set them to 1. 
-Visual inspection of the produced curves is always recommended: see
-Examples.
-
-One may also tweak the accuracy and length of extrapolation and 
-expected survival curve computation by using 
-\code{e1.breaks}. By default this is whatever was supplied to \code{breaks}
-for the survival time scale, to which
-
-\code{c(seq(1/12, 1, 1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))}
-
-is added after the maximum value, e.g. with \code{breaks = list(FUT = 0:10)}
-we have 
-
-\code{..., 10+1/12, ..., 11, 11.2, ..., 2, 3, ..., 19, 20, 25, ... 50}
-
-as the \code{e1.breaks}. Supplying \code{e1.breaks} manually requires
-the breaks over time survival time scale supplied to argument \code{breaks}
-to be reiterated in \code{e1.breaks}; see Examples. \strong{NOTE}: the
-default extrapolation breaks assume the time scales in the data to be 
-expressed as fractional years, meaning this will work extremely poorly
-when using e.g. day-level time scales (such as \code{Date} variables). 
-Set the extrapolation breaks manually in such cases.
-}
-\examples{
-
-library(Epi)
-## take 500 subjects randomly for demonstration
-data(sire)
-sire <- sire[sire$dg_date < sire$ex_date, ]
-set.seed(1L)
-sire <- sire[sample(x = nrow(sire), size = 500),]
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-           exit = list(CAL = get.yrs(ex_date)),
-           data = sire,
-           exit.status = factor(status, levels = 0:2,
-                                labels = c("alive", "canD", "othD")),
-           merge = TRUE)
-
-## phony variable
-set.seed(1L)
-x$group <- rbinom(nrow(x), 1, 0.5)
-## age group
-x$agegr <- cut(x$dg_age, c(0,45,60,Inf), right=FALSE)
-
-## population hazards data  set
-pm <- data.frame(popEpi::popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-
-## breaks to define observed survival estimation
-BL <- list(FUT = seq(0, 10, 1/12))
-
-## crude mean survival
-sm1 <- survmean(Surv(FUT, lex.Xst != "alive") ~ 1,
-                pophaz = pm, data = x, weights = NULL,
-                breaks = BL)
-                
-sm1 <- survmean(FUT ~ 1,
-                pophaz = pm, data = x, weights = NULL,
-                breaks = BL)             
-\donttest{
-## mean survival by group                 
-sm2 <- survmean(FUT ~ group,
-                pophaz = pm, data = x, weights = NULL,
-                breaks = BL)
-                
-## ... and adjusted for age using internal weights (counts of subjects)      
-## note: need also longer extrapolation here so that all curves
-## converge to zero in the end.
-eBL <- list(FUT = c(BL$FUT, 11:75))
-sm3 <- survmean(FUT ~ group + adjust(agegr),
-                pophaz = pm, data = x, weights = "internal",
-                breaks = BL, e1.breaks = eBL)
-}
-## visual inspection of how realistic extrapolation is for each stratum;
-## solid lines are observed + extrapolated survivals;
-## dashed lines are expected survivals
-plot(sm1)
-\donttest{
-## plotting object with both stratification and standardization
-## plots curves for each strata-std.group combination
-plot(sm3)
-
-## for finer control of plotting these curves, you may extract
-## from the survmean object using e.g.
-attributes(sm3)$survmean.meta$curves
-
-
-#### using Dates
-
-x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-           exit = list(CAL = ex_date),
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-## phony group variable
-set.seed(1L)
-x$group <- rbinom(nrow(x), 1, 0.5)
-
-                  
-## NOTE: population hazard should be reported at the same scale
-## as time variables in your Lexis data.
-data(popmort, package = "popEpi")
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-## from year to day level
-pm$haz <- pm$haz/365.25 
-pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
-pm$AGE <- pm$AGE*365.25 
-
-BL <- list(FUT = seq(0, 8, 1/12)*365.25)
-eBL <- list(FUT = c(BL$FUT, c(8.25,8.5,9:60)*365.25))
-smd <- survmean(FUT ~ group, data = x, 
-                pophaz = pm, verbose = TRUE, r = "auto5",
-                breaks = BL, e1.breaks = eBL)     
-plot(smd)
-}
-
-
-}
-\seealso{
-Other survmean functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survmean}()},
-\code{\link{plot.survmean}()}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survtab_ag}()},
-\code{\link{survtab}()}
-}
-\author{
-Joonas Miettinen
-}
-\concept{main functions}
-\concept{survmean functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/mean_survival.R
+\name{survmean}
+\alias{survmean}
+\title{Compute Mean Survival Times Using Extrapolation}
+\usage{
+survmean(
+  formula,
+  data,
+  adjust = NULL,
+  weights = NULL,
+  breaks = NULL,
+  pophaz = NULL,
+  e1.breaks = NULL,
+  e1.pophaz = pophaz,
+  r = "auto",
+  surv.method = "hazard",
+  subset = NULL,
+  verbose = FALSE
+)
+}
+\arguments{
+\item{formula}{a \code{formula}, e.g. \code{FUT ~ V1} or 
+\code{Surv(FUT, lex.Xst) ~ V1}.
+Supplied in the same way as to \code{\link{survtab}}, see that help
+for more info.}
+
+\item{data}{a \code{Lexis} data set; see \code{\link[Epi]{Lexis}}.}
+
+\item{adjust}{variables to adjust estimates by, e.g. \code{adjust = "agegr"}.
+\link[=flexible_argument]{Flexible input}.}
+
+\item{weights}{weights to use to adjust mean survival times. See the
+\link[=direct_standardization]{dedicated help page} for more details on 
+weighting. \code{survmean}
+computes curves separately by all variables to adjust by, computes mean
+survival times, and computes weighted means of the mean survival times.
+See Examples.}
+
+\item{breaks}{a list of breaks defining the time window to compute 
+observed survival in, and the intervals used in estimation. E.g.
+\code{list(FUT = 0:10)} when \code{FUT} is the follow-up time scale in your
+data.}
+
+\item{pophaz}{a data set of population hazards passed to 
+\code{\link{survtab}} (see the 
+\link[=pophaz]{dedicated help page} and the help page of
+\code{survtab} for more information). Defines the 
+population hazard in the time window where observed survival is estimated.}
+
+\item{e1.breaks}{\code{NULL} or a list of breaks defining the time 
+window to compute 
+\strong{expected} survival in, and the intervals used in estimation. E.g.
+\code{list(FUT = 0:100)} when \code{FUT} is the follow-up time scale in your
+data to extrapolate up to 100 years from where the observed survival
+curve ends. \strong{NOTE:} the breaks on the survival time scale
+MUST include the breaks supplied to argument \code{breaks}; see Examples.
+If \code{NULL}, uses decent defaults (maximum follow-up time of 50 years).}
+
+\item{e1.pophaz}{Same as \code{pophaz}, except this defines the 
+population hazard in the time window where \strong{expected} 
+survival is estimated. By default uses the same data as 
+argument \code{pophaz}.}
+
+\item{r}{either a numeric multiplier such as \code{0.995}, \code{"auto"}, or
+\code{"autoX"} where \code{X} is an integer;
+used to determine the relative survival ratio (RSR) persisting after where 
+the estimated observed survival curve ends. See Details.}
+
+\item{surv.method}{passed to \code{survtab}; see that help for more info.}
+
+\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
+subsets the data before computations}
+
+\item{verbose}{\code{logical}; if \code{TRUE}, the function is returns
+some messages and results along the run, which may be useful in debugging}
+}
+\value{
+Returns a \code{data.frame} or \code{data.table} (depending on 
+\code{getOptions("popEpi.datatable")}; see \code{?popEpi}) containing the
+following columns:
+\itemize{
+  \item{est}{: The estimated mean survival time}
+  \item{exp}{: The computed expected survival time}
+  \item{obs}{: Counts of subjects in data}
+  \item{YPLL}{: Years of Potential Life Lost, computed as 
+  (\code{(exp-est)*obs}) - though your time data may be in e.g. days,
+  this column will have the same name regardless.}
+}
+The returned data also has columns named according to the variables
+supplied to the right-hand-side of the formula.
+}
+\description{
+Computes mean survival times based on survival estimation up to
+a point in follow-up time (e.g. 10 years), 
+after which survival is extrapolated
+using an appropriate hazard data file (\code{pophaz}) to yield the "full"
+survival curve. The area under the full survival curve is the mean survival.
+}
+\details{
+\strong{Basics}
+
+\code{survmean} computes mean survival times. For median survival times
+(i.e. where 50 % of subjects have died or met some other event)
+use \code{\link{survtab}}.
+
+The mean survival time is simply the area under the survival curve.
+However, since full follow-up rarely happens, the observed survival curves
+are extrapolated using expected survival: E.g. one might compute observed
+survival till up to 10 years and extrapolate beyond that 
+(till e.g. 50 years) to yield an educated guess on the full observed survival
+curve. 
+
+The area is computed by trapezoidal integration of the area under the curve.
+This function also computes the "full" expected survival curve from
+T = 0 till e.g. T = 50 depending on supplied arguments. The
+expected mean survival time is the area under the 
+mean expected survival curve.
+This function returns the mean expected survival time to be compared with 
+the mean survival time and for computing years of potential life lost (YPLL).
+
+Results can be formed by strata and adjusted for e.g. age by using
+the \code{formula} argument as in \code{survtab}. See also Examples.
+
+\strong{Extrapolation tweaks}
+
+Argument \code{r} controls the relative survival ratio (RSR) assumed to
+persist beyond the time window where observed survival is computed
+(defined by argument \code{breaks}; e.g. up to \code{FUT = 10}).
+The RSR is simply \code{RSR_i = p_oi / p_ei} for a time interval \code{i}, 
+i.e. the observed divided by the expected 
+(conditional, not cumulative) probability of surviving from the beginning of
+a time interval till its end. The cumulative product of \code{RSR_i}
+over time is the (cumulative) relative survival curve. 
+
+
+If \code{r} is numeric, e.g. \code{r = 0.995}, that RSR level is assumed
+to persist beyond the observed survival curve. 
+Numeric \code{r} should be \code{> 0} and expressed at the annual level
+when using fractional years as the scale of the time variables.
+E.g. if RSR is known to be \code{0.95} at the month level, then the
+annualized RSR is \code{0.95^12}. This enables correct usage of the RSR
+with survival intervals of varying lengths. When using day-level time 
+variables (such as \code{Dates}; see \code{as.Date}), numeric \code{r}
+should be expressed at the day level, etc.
+
+If \code{r = "auto"} or \code{r = "auto1"}, this function computes
+RSR estimates internally and automatically uses the \code{RSR_i}
+in the last survival interval in each stratum (and adjusting group)
+and assumes that to persist beyond the observed survival curve.
+Automatic determination of \code{r} is a good starting point,
+but in situations where the RSR estimate is uncertain it may produce poor
+results. Using \code{"autoX"} such as \code{"auto6"} causes \code{survmean}
+to use the mean of the estimated RSRs in the last X survival intervals, 
+which may be more stable.
+Automatic determination will not use values \code{>1} but set them to 1. 
+Visual inspection of the produced curves is always recommended: see
+Examples.
+
+One may also tweak the accuracy and length of extrapolation and 
+expected survival curve computation by using 
+\code{e1.breaks}. By default this is whatever was supplied to \code{breaks}
+for the survival time scale, to which
+
+\code{c(seq(1/12, 1, 1/12), seq(1.2, 1.8, 0.2), 2:19, seq(20, 50, 5))}
+
+is added after the maximum value, e.g. with \code{breaks = list(FUT = 0:10)}
+we have 
+
+\code{..., 10+1/12, ..., 11, 11.2, ..., 2, 3, ..., 19, 20, 25, ... 50}
+
+as the \code{e1.breaks}. Supplying \code{e1.breaks} manually requires
+the breaks over time survival time scale supplied to argument \code{breaks}
+to be reiterated in \code{e1.breaks}; see Examples. \strong{NOTE}: the
+default extrapolation breaks assume the time scales in the data to be 
+expressed as fractional years, meaning this will work extremely poorly
+when using e.g. day-level time scales (such as \code{Date} variables). 
+Set the extrapolation breaks manually in such cases.
+}
+\examples{
+
+library(Epi)
+## take 500 subjects randomly for demonstration
+data(sire)
+sire <- sire[sire$dg_date < sire$ex_date, ]
+set.seed(1L)
+sire <- sire[sample(x = nrow(sire), size = 500),]
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+           exit = list(CAL = get.yrs(ex_date)),
+           data = sire,
+           exit.status = factor(status, levels = 0:2,
+                                labels = c("alive", "canD", "othD")),
+           merge = TRUE)
+
+## phony variable
+set.seed(1L)
+x$group <- rbinom(nrow(x), 1, 0.5)
+## age group
+x$agegr <- cut(x$dg_age, c(0,45,60,Inf), right=FALSE)
+
+## population hazards data  set
+pm <- data.frame(popEpi::popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+
+## breaks to define observed survival estimation
+BL <- list(FUT = seq(0, 10, 1/12))
+
+## crude mean survival
+sm1 <- survmean(Surv(FUT, lex.Xst != "alive") ~ 1,
+                pophaz = pm, data = x, weights = NULL,
+                breaks = BL)
+                
+sm1 <- survmean(FUT ~ 1,
+                pophaz = pm, data = x, weights = NULL,
+                breaks = BL)             
+\donttest{
+## mean survival by group                 
+sm2 <- survmean(FUT ~ group,
+                pophaz = pm, data = x, weights = NULL,
+                breaks = BL)
+                
+## ... and adjusted for age using internal weights (counts of subjects)      
+## note: need also longer extrapolation here so that all curves
+## converge to zero in the end.
+eBL <- list(FUT = c(BL$FUT, 11:75))
+sm3 <- survmean(FUT ~ group + adjust(agegr),
+                pophaz = pm, data = x, weights = "internal",
+                breaks = BL, e1.breaks = eBL)
+}
+## visual inspection of how realistic extrapolation is for each stratum;
+## solid lines are observed + extrapolated survivals;
+## dashed lines are expected survivals
+plot(sm1)
+\donttest{
+## plotting object with both stratification and standardization
+## plots curves for each strata-std.group combination
+plot(sm3)
+
+## for finer control of plotting these curves, you may extract
+## from the survmean object using e.g.
+attributes(sm3)$survmean.meta$curves
+
+
+#### using Dates
+
+x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+           exit = list(CAL = ex_date),
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+## phony group variable
+set.seed(1L)
+x$group <- rbinom(nrow(x), 1, 0.5)
+
+                  
+## NOTE: population hazard should be reported at the same scale
+## as time variables in your Lexis data.
+data(popmort, package = "popEpi")
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+## from year to day level
+pm$haz <- pm$haz/365.25 
+pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
+pm$AGE <- pm$AGE*365.25 
+
+BL <- list(FUT = seq(0, 8, 1/12)*365.25)
+eBL <- list(FUT = c(BL$FUT, c(8.25,8.5,9:60)*365.25))
+smd <- survmean(FUT ~ group, data = x, 
+                pophaz = pm, verbose = TRUE, r = "auto5",
+                breaks = BL, e1.breaks = eBL)     
+plot(smd)
+}
+
+
+}
+\seealso{
+Other survmean functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survmean}()},
+\code{\link{plot.survmean}()}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survtab_ag}()},
+\code{\link{survtab}()}
+}
+\author{
+Joonas Miettinen
+}
+\concept{main functions}
+\concept{survmean functions}
diff --git a/man/survtab.Rd b/man/survtab.Rd
index c3d706e..8e3fcae 100644
--- a/man/survtab.Rd
+++ b/man/survtab.Rd
@@ -1,332 +1,332 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/survival_lexis.R
-\name{survtab}
-\alias{survtab}
-\title{Estimate Survival Time Functions}
-\usage{
-survtab(
-  formula,
-  data,
-  adjust = NULL,
-  breaks = NULL,
-  pophaz = NULL,
-  weights = NULL,
-  surv.type = "surv.rel",
-  surv.method = "hazard",
-  relsurv.method = "e2",
-  subset = NULL,
-  conf.level = 0.95,
-  conf.type = "log-log",
-  verbose = FALSE
-)
-}
-\arguments{
-\item{formula}{a \code{formula}; e.g. \code{fot ~ sex},
-where \code{fot} is the time scale over which you wish to estimate a
-survival time function; this
-assumes that \code{lex.Xst} in your data is the status variable in the
-intended format (almost always right). 
-To be explicit, use \code{\link[survival]{Surv}}: e.g. 
-\code{Surv(fot, lex.Xst) ~ sex}. 
-Variables on the right-hand side of the formula
-separated by \code{+} are considered stratifying variables, for which 
-estimates are computed separately. May contain usage of \code{adjust()} 
---- see Details and Examples.}
-
-\item{data}{a \code{Lexis} object with at least the survival time scale}
-
-\item{adjust}{can be used as an alternative to passing variables to 
-argument \code{formula} within a call to \code{adjust()}; e.g.
-\code{adjust = "agegr"}. \link[=flexible_argument]{Flexible input}.}
-
-\item{breaks}{a named list of breaks, e.g.
-\code{list(FUT = 0:5)}. If data is not split in advance, \code{breaks}
-must at the very least contain a vector of breaks to split the survival time 
-scale (mentioned in argument \code{formula}). If data has already been split
-(using e.g. \code{\link{splitMulti}}) along at least the used survival time
-scale, this may be \code{NULL}. It is generally recommended (and sufficient; 
-see Seppa, Dyban and Hakulinen (2015)) to use monthly
-intervals where applicable.}
-
-\item{pophaz}{a \code{data.frame} containing
-expected hazards for the event of interest to occur. See the
-\link[=pophaz]{dedicated help page}. Required when
-\code{surv.type = "surv.rel"} or \code{"cif.rel"}. \code{pophaz} must
-contain one column named \code{"haz"}, and any number of other columns
-identifying levels of variables to do a merge with split data within
-\code{survtab}. Some columns may be time scales, which will
-allow for the expected hazard to vary by e.g. calendar time and age.}
-
-\item{weights}{typically a list of weights or a \code{character} string
-specifying an age group standardization scheme; see
-the \link[=direct_standardization]{dedicated help page} 
-and examples. NOTE: \code{weights = "internal"} is based on the counts
-of persons in follow-up at the start of follow-up (typically T = 0)}
-
-\item{surv.type}{one of \code{'surv.obs'},
-\code{'surv.cause'}, \code{'surv.rel'}, 
-\code{'cif.obs'} or \code{'cif.rel'}; 
-defines what kind of survival time function(s) is/are estimated; see Details}
-
-\item{surv.method}{either \code{'lifetable'} or \code{'hazard'}; determines
-the method of calculating survival time functions, where the former computes
-ratios such as \code{p = d/(n - n.cens)} 
-and the latter utilizes subject-times 
-(typically person-years) for hazard estimates such as \code{d/pyrs} 
-which are used to compute survival time function estimates.
-The former method requires argument \code{n.cens} and the latter 
-argument \code{pyrs} to be supplied.}
-
-\item{relsurv.method}{either \code{'e2'} or \code{'pp'}; 
-defines whether to compute relative survival using the
-EdererII method or using Pohar-Perme weighting;
-ignored if \code{surv.type != "surv.rel"}}
-
-\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
-subsets the data before computations}
-
-\item{conf.level}{confidence level used in confidence intervals; 
-e.g. \code{0.95} for 95 percent confidence intervals}
-
-\item{conf.type}{character string; must be one of \code{"plain"}, 
-\code{"log-log"} and \code{"log"}; 
-defines the transformation used on the survival time
-function to yield confidence 
-intervals via the delta method}
-
-\item{verbose}{logical; if \code{TRUE}, the function is chatty and
-returns some messages and timings along the process}
-}
-\value{
-Returns a table of life time function values and other 
-information with survival intervals as rows.
-Returns some of the following estimates of survival time functions:
-
-\itemize{
- \item \code{surv.obs} - observed (raw, overall) survival
- \item \code{surv.obs.K} - observed cause-specific survival for cause K
- \item \code{CIF_k} - cumulative incidence function for cause \code{k}
- \item \code{CIF.rel} - cumulative incidence function using excess cases
- \item \code{r.e2} -  relative survival, EdererII
- \item \code{r.pp} -  relative survival, Pohar-Perme weighted
-}
-The suffix \code{.as} implies adjusted estimates, and \code{.lo} and
-\code{.hi} imply lower and upper confidence limits, respectively. 
-The prefix \code{SE.} stands for standard error.
-}
-\description{
-This function estimates survival time functions: survival, 
-relative/net survival, and crude/absolute risk functions (CIF).
-}
-\section{Basics}{
-
-
-This function computes interval-based estimates of survival time functions,
-where the intervals are set by the user. For product-limit-based
-estimation see packages \pkg{survival} and \pkg{relsurv}.
-
-if \code{surv.type = 'surv.obs'}, only 'raw' observed survival 
-is estimated over the chosen time intervals. With
-\code{surv.type = 'surv.rel'}, also relative survival estimates 
-are supplied in addition to observed survival figures. 
-
-\code{surv.type = 'cif.obs'} requests cumulative incidence functions (CIF) 
-to be estimated. 
-CIFs are estimated for each competing risk based 
-on a survival-interval-specific proportional hazards
-assumption as described by Chiang (1968).  
-With \code{surv.type = 'cif.rel'}, a CIF is estimated with using 
-excess cases as the ''cause-specific'' cases. Finally, with 
-\code{surv.type = 'surv.cause'}, cause-specific survivals are 
-estimated separately for each separate type of event. 
-
-In hazard-based estimation (\code{surv.method = "hazard"}) survival
-time functions are transformations of the estimated corresponding hazard
-in the intervals. The hazard itself is estimated using counts of events
-(or excess events) and total subject-time in the interval. Life table
-\code{surv.method = "lifetable"} estimates are constructed as transformations 
-of probabilities computed using counts of events and counts of subjects 
-at risk.
-
-
-The vignette \href{../doc/survtab_examples.html}{survtab_examples} 
-has some practical examples.
-}
-
-\section{Relative survival}{
-
- 
-When \code{surv.type = 'surv.rel'}, the user can choose 
-\code{relsurv.method = 'pp'}, whereupon Pohar-Perme weighting is used.
-By default \code{relsurv.method = 'e2'}, i.e. the Ederer II method
-is used to estimate relative survival.
-}
-
-\section{Adjusted estimates}{
-
-
-Adjusted estimates in this context mean computing estimates separately
-by the levels of adjusting variables and returning weighted averages
-of the estimates. For example, computing estimates separately by
-age groups and returning a weighted average estimate (age-adjusted estimate).
-
-Adjusting requires specification of both the adjusting variables and
-the weights for all the levels of the adjusting variables. The former can be
-accomplished by using \code{adjust()} with the argument \code{formula},
-or by supplying variables directly to argument \code{adjust}. E.g. the
-following are all equivalent:
-
-\code{formula = fot ~ sex + adjust(agegr) + adjust(area)}
-
-\code{formula = fot ~ sex + adjust(agegr, area)}
-
-\code{formula  = fot ~ sex, adjust = c("agegr", "area")}
-
-\code{formula  = fot ~ sex, adjust = list(agegr, area)}
-
-The adjusting variables must match with the variable names in the
-argument \code{weights};
-see the \link[=direct_standardization]{dedicated help page}. 
-Typically weights are supplied as a \code{list} or
-a \code{data.frame}. The former can be done by e.g.
-
-\code{weights = list(agegr = VEC1, area = VEC2)},
-
-where \code{VEC1} and \code{VEC2} are vectors of weights (which do not
-have to add up to one). See 
-\href{../doc/survtab_examples.html}{survtab_examples} 
-for an example of using a \code{data.frame} to pass weights.
-}
-
-\section{Period analysis and other data selection schemes}{
-
-
-To calculate e.g. period analysis (delayed entry) estimates, 
-limit the data when/before supplying to this function.See 
-\href{../doc/survtab_examples.html}{survtab_examples}.
-}
-
-\examples{
-\donttest{
-data("sire", package = "popEpi")
-library(Epi)
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-
-## phony group variable
-set.seed(1L)
-x$group <- rbinom(nrow(x), 1, 0.5)
-
-## observed survival. explicit supplying of status:
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-## this assumes the status is lex.Xst (right 99.9 \% of the time)
-st <- survtab(FUT ~ group, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-              
-## relative survival (ederer II)
-data("popmort", package = "popEpi")
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-st <- survtab(FUT ~ group, data = x, 
-              surv.type = "surv.rel",
-              pophaz = pm,
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## ICSS weights usage
-data("ICSS", package = "popEpi")
-cut <- c(0, 30, 50, 70, Inf)
-agegr <- cut(ICSS$age, cut, right = FALSE)
-w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-x$agegr <- cut(x$dg_age, cut, right = FALSE)
-st <- survtab(FUT ~ group + adjust(agegr), data = x, 
-              surv.type = "surv.rel",
-              pophaz = pm, weights = w$ICSS1,
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-#### using dates with survtab
-x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-           exit = list(CAL = ex_date),
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-## phony group variable
-set.seed(1L)
-x$group <- rbinom(nrow(x), 1, 0.5)
-
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)*365.25))    
-                  
-## NOTE: population hazard should be reported at the same scale
-## as time variables in your Lexis data.
-data(popmort, package = "popEpi")
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-## from year to day level
-pm$haz <- pm$haz/365.25 
-pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
-pm$AGE <- pm$AGE*365.25 
-
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
-              surv.type = "surv.rel", relsurv.method = "e2",
-              pophaz = pm,
-              breaks = list(FUT = seq(0, 5, 1/12)*365.25))  
-}
-}
-\references{
-Perme, Maja Pohar, Janez Stare, and Jacques Esteve. 
-"On estimation in relative survival." Biometrics 68.1 (2012): 113-120.
-\doi{10.1111/j.1541-0420.2011.01640.x}
-
-Hakulinen, Timo, Karri Seppa, and Paul C. Lambert. 
-"Choosing the relative survival method for cancer survival estimation." 
-European Journal of Cancer 47.14 (2011): 2202-2210.
-\doi{10.1016/j.ejca.2011.03.011}
- 
-Seppa, Karri, Timo Hakulinen, and Arun Pokhrel. 
-"Choosing the net survival method for cancer survival estimation." 
-European Journal of Cancer (2013).
-\doi{10.1016/j.ejca.2013.09.019}
-
-CHIANG, Chin Long. Introduction to stochastic processes in biostatistics. 
-1968. ISBN-14: 978-0471155003
-
-Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
-Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
-\doi{10.1016/B978-0-12-801238-3.02745-8}
-}
-\seealso{
-\code{\link{splitMulti}}, \code{\link{lexpand}}, 
-\code{\link{ICSS}}, \code{\link{sire}}
-\href{../doc/survtab_examples.html}{The survtab_examples vignette}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab_ag}()}
-
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survtab}()},
-\code{\link{plot.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab_ag}()}
-}
-\concept{main functions}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/survival_lexis.R
+\name{survtab}
+\alias{survtab}
+\title{Estimate Survival Time Functions}
+\usage{
+survtab(
+  formula,
+  data,
+  adjust = NULL,
+  breaks = NULL,
+  pophaz = NULL,
+  weights = NULL,
+  surv.type = "surv.rel",
+  surv.method = "hazard",
+  relsurv.method = "e2",
+  subset = NULL,
+  conf.level = 0.95,
+  conf.type = "log-log",
+  verbose = FALSE
+)
+}
+\arguments{
+\item{formula}{a \code{formula}; e.g. \code{fot ~ sex},
+where \code{fot} is the time scale over which you wish to estimate a
+survival time function; this
+assumes that \code{lex.Xst} in your data is the status variable in the
+intended format (almost always right). 
+To be explicit, use \code{\link[survival]{Surv}}: e.g. 
+\code{Surv(fot, lex.Xst) ~ sex}. 
+Variables on the right-hand side of the formula
+separated by \code{+} are considered stratifying variables, for which 
+estimates are computed separately. May contain usage of \code{adjust()} 
+--- see Details and Examples.}
+
+\item{data}{a \code{Lexis} object with at least the survival time scale}
+
+\item{adjust}{can be used as an alternative to passing variables to 
+argument \code{formula} within a call to \code{adjust()}; e.g.
+\code{adjust = "agegr"}. \link[=flexible_argument]{Flexible input}.}
+
+\item{breaks}{a named list of breaks, e.g.
+\code{list(FUT = 0:5)}. If data is not split in advance, \code{breaks}
+must at the very least contain a vector of breaks to split the survival time 
+scale (mentioned in argument \code{formula}). If data has already been split
+(using e.g. \code{\link{splitMulti}}) along at least the used survival time
+scale, this may be \code{NULL}. It is generally recommended (and sufficient; 
+see Seppa, Dyban and Hakulinen (2015)) to use monthly
+intervals where applicable.}
+
+\item{pophaz}{a \code{data.frame} containing
+expected hazards for the event of interest to occur. See the
+\link[=pophaz]{dedicated help page}. Required when
+\code{surv.type = "surv.rel"} or \code{"cif.rel"}. \code{pophaz} must
+contain one column named \code{"haz"}, and any number of other columns
+identifying levels of variables to do a merge with split data within
+\code{survtab}. Some columns may be time scales, which will
+allow for the expected hazard to vary by e.g. calendar time and age.}
+
+\item{weights}{typically a list of weights or a \code{character} string
+specifying an age group standardization scheme; see
+the \link[=direct_standardization]{dedicated help page} 
+and examples. NOTE: \code{weights = "internal"} is based on the counts
+of persons in follow-up at the start of follow-up (typically T = 0)}
+
+\item{surv.type}{one of \code{'surv.obs'},
+\code{'surv.cause'}, \code{'surv.rel'}, 
+\code{'cif.obs'} or \code{'cif.rel'}; 
+defines what kind of survival time function(s) is/are estimated; see Details}
+
+\item{surv.method}{either \code{'lifetable'} or \code{'hazard'}; determines
+the method of calculating survival time functions, where the former computes
+ratios such as \code{p = d/(n - n.cens)} 
+and the latter utilizes subject-times 
+(typically person-years) for hazard estimates such as \code{d/pyrs} 
+which are used to compute survival time function estimates.
+The former method requires argument \code{n.cens} and the latter 
+argument \code{pyrs} to be supplied.}
+
+\item{relsurv.method}{either \code{'e2'} or \code{'pp'}; 
+defines whether to compute relative survival using the
+EdererII method or using Pohar-Perme weighting;
+ignored if \code{surv.type != "surv.rel"}}
+
+\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
+subsets the data before computations}
+
+\item{conf.level}{confidence level used in confidence intervals; 
+e.g. \code{0.95} for 95 percent confidence intervals}
+
+\item{conf.type}{character string; must be one of \code{"plain"}, 
+\code{"log-log"} and \code{"log"}; 
+defines the transformation used on the survival time
+function to yield confidence 
+intervals via the delta method}
+
+\item{verbose}{logical; if \code{TRUE}, the function is chatty and
+returns some messages and timings along the process}
+}
+\value{
+Returns a table of life time function values and other 
+information with survival intervals as rows.
+Returns some of the following estimates of survival time functions:
+
+\itemize{
+ \item \code{surv.obs} - observed (raw, overall) survival
+ \item \code{surv.obs.K} - observed cause-specific survival for cause K
+ \item \code{CIF_k} - cumulative incidence function for cause \code{k}
+ \item \code{CIF.rel} - cumulative incidence function using excess cases
+ \item \code{r.e2} -  relative survival, EdererII
+ \item \code{r.pp} -  relative survival, Pohar-Perme weighted
+}
+The suffix \code{.as} implies adjusted estimates, and \code{.lo} and
+\code{.hi} imply lower and upper confidence limits, respectively. 
+The prefix \code{SE.} stands for standard error.
+}
+\description{
+This function estimates survival time functions: survival, 
+relative/net survival, and crude/absolute risk functions (CIF).
+}
+\section{Basics}{
+
+
+This function computes interval-based estimates of survival time functions,
+where the intervals are set by the user. For product-limit-based
+estimation see packages \pkg{survival} and \pkg{relsurv}.
+
+if \code{surv.type = 'surv.obs'}, only 'raw' observed survival 
+is estimated over the chosen time intervals. With
+\code{surv.type = 'surv.rel'}, also relative survival estimates 
+are supplied in addition to observed survival figures. 
+
+\code{surv.type = 'cif.obs'} requests cumulative incidence functions (CIF) 
+to be estimated. 
+CIFs are estimated for each competing risk based 
+on a survival-interval-specific proportional hazards
+assumption as described by Chiang (1968).  
+With \code{surv.type = 'cif.rel'}, a CIF is estimated with using 
+excess cases as the ''cause-specific'' cases. Finally, with 
+\code{surv.type = 'surv.cause'}, cause-specific survivals are 
+estimated separately for each separate type of event. 
+
+In hazard-based estimation (\code{surv.method = "hazard"}) survival
+time functions are transformations of the estimated corresponding hazard
+in the intervals. The hazard itself is estimated using counts of events
+(or excess events) and total subject-time in the interval. Life table
+\code{surv.method = "lifetable"} estimates are constructed as transformations 
+of probabilities computed using counts of events and counts of subjects 
+at risk.
+
+
+The vignette \href{../doc/survtab_examples.html}{survtab_examples} 
+has some practical examples.
+}
+
+\section{Relative survival}{
+
+ 
+When \code{surv.type = 'surv.rel'}, the user can choose 
+\code{relsurv.method = 'pp'}, whereupon Pohar-Perme weighting is used.
+By default \code{relsurv.method = 'e2'}, i.e. the Ederer II method
+is used to estimate relative survival.
+}
+
+\section{Adjusted estimates}{
+
+
+Adjusted estimates in this context mean computing estimates separately
+by the levels of adjusting variables and returning weighted averages
+of the estimates. For example, computing estimates separately by
+age groups and returning a weighted average estimate (age-adjusted estimate).
+
+Adjusting requires specification of both the adjusting variables and
+the weights for all the levels of the adjusting variables. The former can be
+accomplished by using \code{adjust()} with the argument \code{formula},
+or by supplying variables directly to argument \code{adjust}. E.g. the
+following are all equivalent:
+
+\code{formula = fot ~ sex + adjust(agegr) + adjust(area)}
+
+\code{formula = fot ~ sex + adjust(agegr, area)}
+
+\code{formula  = fot ~ sex, adjust = c("agegr", "area")}
+
+\code{formula  = fot ~ sex, adjust = list(agegr, area)}
+
+The adjusting variables must match with the variable names in the
+argument \code{weights};
+see the \link[=direct_standardization]{dedicated help page}. 
+Typically weights are supplied as a \code{list} or
+a \code{data.frame}. The former can be done by e.g.
+
+\code{weights = list(agegr = VEC1, area = VEC2)},
+
+where \code{VEC1} and \code{VEC2} are vectors of weights (which do not
+have to add up to one). See 
+\href{../doc/survtab_examples.html}{survtab_examples} 
+for an example of using a \code{data.frame} to pass weights.
+}
+
+\section{Period analysis and other data selection schemes}{
+
+
+To calculate e.g. period analysis (delayed entry) estimates, 
+limit the data when/before supplying to this function.See 
+\href{../doc/survtab_examples.html}{survtab_examples}.
+}
+
+\examples{
+\donttest{
+data("sire", package = "popEpi")
+library(Epi)
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+
+## phony group variable
+set.seed(1L)
+x$group <- rbinom(nrow(x), 1, 0.5)
+
+## observed survival. explicit supplying of status:
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+## this assumes the status is lex.Xst (right 99.9 \% of the time)
+st <- survtab(FUT ~ group, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+              
+## relative survival (ederer II)
+data("popmort", package = "popEpi")
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+st <- survtab(FUT ~ group, data = x, 
+              surv.type = "surv.rel",
+              pophaz = pm,
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## ICSS weights usage
+data("ICSS", package = "popEpi")
+cut <- c(0, 30, 50, 70, Inf)
+agegr <- cut(ICSS$age, cut, right = FALSE)
+w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+x$agegr <- cut(x$dg_age, cut, right = FALSE)
+st <- survtab(FUT ~ group + adjust(agegr), data = x, 
+              surv.type = "surv.rel",
+              pophaz = pm, weights = w$ICSS1,
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+#### using dates with survtab
+x <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+           exit = list(CAL = ex_date),
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+## phony group variable
+set.seed(1L)
+x$group <- rbinom(nrow(x), 1, 0.5)
+
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)*365.25))    
+                  
+## NOTE: population hazard should be reported at the same scale
+## as time variables in your Lexis data.
+data(popmort, package = "popEpi")
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+## from year to day level
+pm$haz <- pm$haz/365.25 
+pm$CAL <- as.Date(paste0(pm$CAL, "-01-01")) 
+pm$AGE <- pm$AGE*365.25 
+
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ group, data = x, 
+              surv.type = "surv.rel", relsurv.method = "e2",
+              pophaz = pm,
+              breaks = list(FUT = seq(0, 5, 1/12)*365.25))  
+}
+}
+\references{
+Perme, Maja Pohar, Janez Stare, and Jacques Esteve. 
+"On estimation in relative survival." Biometrics 68.1 (2012): 113-120.
+\doi{10.1111/j.1541-0420.2011.01640.x}
+
+Hakulinen, Timo, Karri Seppa, and Paul C. Lambert. 
+"Choosing the relative survival method for cancer survival estimation." 
+European Journal of Cancer 47.14 (2011): 2202-2210.
+\doi{10.1016/j.ejca.2011.03.011}
+ 
+Seppa, Karri, Timo Hakulinen, and Arun Pokhrel. 
+"Choosing the net survival method for cancer survival estimation." 
+European Journal of Cancer (2013).
+\doi{10.1016/j.ejca.2013.09.019}
+
+CHIANG, Chin Long. Introduction to stochastic processes in biostatistics. 
+1968. ISBN-14: 978-0471155003
+
+Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
+Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
+\doi{10.1016/B978-0-12-801238-3.02745-8}
+}
+\seealso{
+\code{\link{splitMulti}}, \code{\link{lexpand}}, 
+\code{\link{ICSS}}, \code{\link{sire}}
+\href{../doc/survtab_examples.html}{The survtab_examples vignette}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab_ag}()}
+
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survtab}()},
+\code{\link{plot.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab_ag}()}
+}
+\concept{main functions}
+\concept{survtab functions}
diff --git a/man/survtab_ag.Rd b/man/survtab_ag.Rd
index ac9715b..fbda9c9 100644
--- a/man/survtab_ag.Rd
+++ b/man/survtab_ag.Rd
@@ -1,423 +1,423 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/survival_aggregated.R
-\name{survtab_ag}
-\alias{survtab_ag}
-\title{Estimate Survival Time Functions}
-\usage{
-survtab_ag(
-  formula = NULL,
-  data,
-  adjust = NULL,
-  weights = NULL,
-  surv.breaks = NULL,
-  n = "at.risk",
-  d = "from0to1",
-  n.cens = "from0to0",
-  pyrs = "pyrs",
-  d.exp = "d.exp",
-  n.pp = NULL,
-  d.pp = "d.pp",
-  d.pp.2 = "d.pp.2",
-  n.cens.pp = "n.cens.pp",
-  pyrs.pp = "pyrs.pp",
-  d.exp.pp = "d.exp.pp",
-  surv.type = "surv.rel",
-  surv.method = "hazard",
-  relsurv.method = "e2",
-  subset = NULL,
-  conf.level = 0.95,
-  conf.type = "log-log",
-  verbose = FALSE
-)
-}
-\arguments{
-\item{formula}{a \code{formula}; the response 
-must be the time scale to compute survival time function estimates
-over, e.g. \code{fot ~ sex}. Variables on the right-hand side of the formula
-separated by \code{+} are considered stratifying variables, for which 
-estimates are computed separately. May contain usage of \code{adjust()} 
---- see Details and Examples.}
-
-\item{data}{since popEpi 0.4.0, a \code{data.frame}
-containing variables used in \code{formula} and other arguments.
-\code{aggre} objects are recommended as they contain information on any
-time scales and are therefore safer; for creating \code{aggre} objects see
-\code{\link{as.aggre}} when your data is already aggregated and \code{aggre}
-for aggregating split \code{Lexis} objects.}
-
-\item{adjust}{can be used as an alternative to passing variables to 
-argument \code{formula} within a call to \code{adjust()}; e.g.
-\code{adjust = "agegr"}. \link[=flexible_argument]{Flexible input}.}
-
-\item{weights}{typically a list of weights or a \code{character} string
-specifying an age group standardization scheme; see
-the \link[=direct_standardization]{dedicated help page} 
-and examples. NOTE: \code{weights = "internal"} is based on the counts
-of persons in follow-up at the start of follow-up (typically T = 0)}
-
-\item{surv.breaks}{a vector of breaks on the 
-survival time scale. Optional if \code{data} is an \code{aggre} object
-and mandatory otherwise. Must define each intended interval;
-e.g. \code{surv.breaks = 0:5} when data has intervals defined by 
-breaks \code{seq(0, 5, 1/12)} will aggregate to wider intervals first.
-It is generally recommended (and sufficient; 
-see Seppa, Dyban and Hakulinen (2015)) to use monthly
-intervals where applicable.}
-
-\item{n}{variable containing counts of subjects at-risk at the start of a 
-time interval; e.g. \code{n = "at.risk"}. 
-Required when \code{surv.method = "lifetable"}.
-\link[=flexible_argument]{Flexible input}.}
-
-\item{d}{variable(s) containing counts of subjects experiencing an event. 
-With only one type of event, e.g. \code{d = "deaths"}. With multiple types of 
-events (for CIF or cause-specific survival estimation), supply e.g.
-\code{d = c("canD", "othD")}. If the survival time function to be estimated
-does not use multiple types of events, supplying more than one variable
-to \code{d} simply causes the variables to be added together. 
-Always required. \link[=flexible_argument]{Flexible input}.}
-
-\item{n.cens}{variable containing counts of subjects censored during a 
-survival time interval; E.g. \code{n.cens = "alive"}.
-Required when \code{surv.method = "lifetable"}. 
-\link[=flexible_argument]{Flexible input}.}
-
-\item{pyrs}{variable containing total subject-time accumulated within a 
-survival time interval; E.g. \code{pyrs = "pyrs"}. 
-Required when \code{surv.method = "hazard"}. Flexible input.}
-
-\item{d.exp}{variable denoting total "expected numbers of events" 
-(typically computed \code{pyrs * pop.haz}, where 
-\code{pop.haz} is the expected hazard level) 
-accumulated within a survival time interval; E.g. \code{pyrs = "pyrs"}.
-Required when computing EdererII relative survivals or 
-CIFs based on excess counts of events. Flexible input.}
-
-\item{n.pp}{variable containing total Pohar-Perme weighted counts of
-subjects at risk in an interval,
-supplied as argument \code{n} is supplied. 
-Computed originally on the subject
-level as analogous to \code{pp * as.integer(status == "at-risk")}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{d.pp}{variable(s) containing Pohar-Perme weighted counts of events,
-supplied as argument \code{d} is supplied. Computed originally on the subject
-level as analogous to \code{pp * as.integer(status == some_event)}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{d.pp.2}{variable(s) containing total Pohar-Perme 
-"double-weighted" counts of events,
-supplied as argument \code{d} is supplied. Computed originally on the subject
-level as analogous to \code{pp * pp * as.integer(status == some_event)}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{n.cens.pp}{variable containing total Pohar-Perme weighted counts 
-censorings,
-supplied as argument \code{n.cens} is supplied. 
-Computed originally on the subject
-level as analogous to \code{pp * as.integer(status == "censored")}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{pyrs.pp}{variable containing total Pohar-Perme weighted subject-times,
-supplied as argument \code{pyrs} is supplied. 
-Computed originally on the subject
-level as analogous to \code{pp * pyrs}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{d.exp.pp}{variable containing total Pohar-Perme weighted counts 
-of excess events,
-supplied as argument \code{pyrs} is supplied. 
-Computed originally on the subject
-level as analogous to \code{pp * d.exp}.
-Required when \code{relsurv.method = "pp"}. Flexible input.}
-
-\item{surv.type}{one of \code{'surv.obs'},
-\code{'surv.cause'}, \code{'surv.rel'}, 
-\code{'cif.obs'} or \code{'cif.rel'}; 
-defines what kind of survival time function(s) is/are estimated; see Details}
-
-\item{surv.method}{either \code{'lifetable'} or \code{'hazard'}; determines
-the method of calculating survival time functions, where the former computes
-ratios such as \code{p = d/(n - n.cens)} 
-and the latter utilizes subject-times 
-(typically person-years) for hazard estimates such as \code{d/pyrs} 
-which are used to compute survival time function estimates.
-The former method requires argument \code{n.cens} and the latter 
-argument \code{pyrs} to be supplied.}
-
-\item{relsurv.method}{either \code{'e2'} or \code{'pp'}; 
-defines whether to compute relative survival using the
-EdererII method or using Pohar-Perme weighting;
-ignored if \code{surv.type != "surv.rel"}}
-
-\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
-subsets the data before computations}
-
-\item{conf.level}{confidence level used in confidence intervals; 
-e.g. \code{0.95} for 95 percent confidence intervals}
-
-\item{conf.type}{character string; must be one of \code{"plain"}, 
-\code{"log-log"} and \code{"log"}; 
-defines the transformation used on the survival time
-function to yield confidence 
-intervals via the delta method}
-
-\item{verbose}{logical; if \code{TRUE}, the function is chatty and
-returns some messages and timings along the process}
-}
-\value{
-Returns a table of life time function values and other 
-information with survival intervals as rows.
-Returns some of the following estimates of survival time functions:
-
-\itemize{
- \item \code{surv.obs} - observed (raw, overall) survival
- \item \code{surv.obs.K} - observed cause-specific survival for cause K
- \item \code{CIF_k} - cumulative incidence function for cause \code{k}
- \item \code{CIF.rel} - cumulative incidence function using excess cases
- \item \code{r.e2} -  relative survival, EdererII
- \item \code{r.pp} -  relative survival, Pohar-Perme weighted
-}
-The suffix \code{.as} implies adjusted estimates, and \code{.lo} and
-\code{.hi} imply lower and upper confidence limits, respectively. 
-The prefix \code{SE.} stands for standard error.
-}
-\description{
-This function estimates survival time functions: survival, 
-relative/net survival, and crude/absolute risk functions (CIF).
-}
-\section{Basics}{
-
-
-This function computes interval-based estimates of survival time functions,
-where the intervals are set by the user. For product-limit-based
-estimation see packages \pkg{survival} and \pkg{relsurv}.
-
-if \code{surv.type = 'surv.obs'}, only 'raw' observed survival 
-is estimated over the chosen time intervals. With
-\code{surv.type = 'surv.rel'}, also relative survival estimates 
-are supplied in addition to observed survival figures. 
-
-\code{surv.type = 'cif.obs'} requests cumulative incidence functions (CIF) 
-to be estimated. 
-CIFs are estimated for each competing risk based 
-on a survival-interval-specific proportional hazards
-assumption as described by Chiang (1968).  
-With \code{surv.type = 'cif.rel'}, a CIF is estimated with using 
-excess cases as the ''cause-specific'' cases. Finally, with 
-\code{surv.type = 'surv.cause'}, cause-specific survivals are 
-estimated separately for each separate type of event. 
-
-In hazard-based estimation (\code{surv.method = "hazard"}) survival
-time functions are transformations of the estimated corresponding hazard
-in the intervals. The hazard itself is estimated using counts of events
-(or excess events) and total subject-time in the interval. Life table
-\code{surv.method = "lifetable"} estimates are constructed as transformations 
-of probabilities computed using counts of events and counts of subjects 
-at risk.
-
-
-The vignette \href{../doc/survtab_examples.html}{survtab_examples} 
-has some practical examples.
-}
-
-\section{Relative survival}{
-
- 
-When \code{surv.type = 'surv.rel'}, the user can choose 
-\code{relsurv.method = 'pp'}, whereupon Pohar-Perme weighting is used.
-By default \code{relsurv.method = 'e2'}, i.e. the Ederer II method
-is used to estimate relative survival.
-}
-
-\section{Adjusted estimates}{
-
-
-Adjusted estimates in this context mean computing estimates separately
-by the levels of adjusting variables and returning weighted averages
-of the estimates. For example, computing estimates separately by
-age groups and returning a weighted average estimate (age-adjusted estimate).
-
-Adjusting requires specification of both the adjusting variables and
-the weights for all the levels of the adjusting variables. The former can be
-accomplished by using \code{adjust()} with the argument \code{formula},
-or by supplying variables directly to argument \code{adjust}. E.g. the
-following are all equivalent:
-
-\code{formula = fot ~ sex + adjust(agegr) + adjust(area)}
-
-\code{formula = fot ~ sex + adjust(agegr, area)}
-
-\code{formula  = fot ~ sex, adjust = c("agegr", "area")}
-
-\code{formula  = fot ~ sex, adjust = list(agegr, area)}
-
-The adjusting variables must match with the variable names in the
-argument \code{weights};
-see the \link[=direct_standardization]{dedicated help page}. 
-Typically weights are supplied as a \code{list} or
-a \code{data.frame}. The former can be done by e.g.
-
-\code{weights = list(agegr = VEC1, area = VEC2)},
-
-where \code{VEC1} and \code{VEC2} are vectors of weights (which do not
-have to add up to one). See 
-\href{../doc/survtab_examples.html}{survtab_examples} 
-for an example of using a \code{data.frame} to pass weights.
-}
-
-\section{Period analysis and other data selection schemes}{
-
-
-To calculate e.g. period analysis (delayed entry) estimates, 
-limit the data when/before supplying to this function.See 
-\href{../doc/survtab_examples.html}{survtab_examples}.
-}
-
-\section{Data requirements}{
-
-
-\code{survtab_ag} computes estimates of survival time functions using 
-pre-aggregated data. For using subject-level data directly, use 
-\code{\link{survtab}}. For aggregating data, see \code{\link{lexpand}}
-and \code{\link{aggre}}. 
-
-By default, and if data is an \code{aggre} object (not mandatory), 
-\code{survtab_ag} makes use of the exact same breaks that were used in 
-splitting the original data (with e.g. \code{lexpand}), so it is not 
-necessary to specify any \code{surv.breaks}. If specified, the 
-\code{surv.breaks} must be a subset of the pertinent 
-pre-existing breaks. When data is not an \code{aggre} object, breaks
-must always be specified. Interval lengths (\code{delta} in output) are 
-also calculated based on whichever breaks are used, 
-so the upper limit of the breaks should
-therefore be meaningful and never e.g. \code{Inf}.
-}
-
-\examples{
-## see more examples with explanations in vignette("survtab_examples")
-
-#### survtab_ag usage
-
-data("sire", package = "popEpi")
-## prepare data for e.g. 5-year "period analysis" for 2008-2012
-## note: sire is a simulated cohort integrated into popEpi.
-BL <- list(fot=seq(0, 5, by = 1/12),
-           per = c("2008-01-01", "2013-01-01"))
-x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status \%in\% 1:2,
-             breaks = BL,
-             pophaz = popmort,
-             aggre = list(fot))
-             
-## calculate relative EdererII period method
-## NOTE: x is an aggre object here, so surv.breaks are deduced
-## automatically
-st <- survtab_ag(fot ~ 1, data = x)
-
-summary(st, t = 1:5) ## annual estimates
-summary(st, q = list(r.e2 = 0.75)) ## 1st interval where r.e2 < 0.75 at end
-\donttest{
-plot(st)
-
-
-## non-aggre data: first call to survtab_ag would fail
-df <- data.frame(x)
-# st <- survtab_ag(fot ~ 1, data = x)
-st <- survtab_ag(fot ~ 1, data = x, surv.breaks = BL$fot)
-
-## calculate age-standardised 5-year relative survival ratio using 
-## Ederer II method and period approach 
-
-sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
-BL <- list(fot=seq(0, 5, by = 1/12),
-           per = c("2008-01-01", "2013-01-01"))
-x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = status \%in\% 1:2,
-             breaks = BL,
-             pophaz = popmort,
-             aggre = list(agegr, fot))
-
-## age standardisation using internal weights (age distribution of 
-## patients diagnosed within the period window)
-## (NOTE: what is done here is equivalent to using weights = "internal")
-w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0], FUN = sum)
-names(w) <- c("agegr", "weights")
-
-st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
-plot(st, y = "r.e2.as", col = c("blue"))
-
-## age standardisation using ICSS1 weights
-data(ICSS)
-cut <- c(0, 45, 55, 65, 75, Inf)
-agegr <- cut(ICSS$age, cut, right = FALSE)
-w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
-names(w) <- c("agegr", "weights")
-
-st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
-lines(st, y = "r.e2.as", col = c("red"))
-
-
-## cause-specific survival
-sire$stat <- factor(sire$status, 0:2, c("alive", "canD", "othD"))
-x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-             status = stat,
-             breaks = BL,
-             pophaz = popmort,
-             aggre = list(agegr, fot))
-st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w,
-                 d = c("fromalivetocanD", "fromalivetoothD"),
-                 surv.type = "surv.cause")
-plot(st, y = "surv.obs.fromalivetocanD.as")
-lines(st, y = "surv.obs.fromalivetoothD.as", col = "red")
-
-
-}
-}
-\references{
-Perme, Maja Pohar, Janez Stare, and Jacques Esteve. 
-"On estimation in relative survival." Biometrics 68.1 (2012): 113-120.
-\doi{10.1111/j.1541-0420.2011.01640.x}
-
-Hakulinen, Timo, Karri Seppa, and Paul C. Lambert. 
-"Choosing the relative survival method for cancer survival estimation." 
-European Journal of Cancer 47.14 (2011): 2202-2210.
-\doi{10.1016/j.ejca.2011.03.011}
- 
-Seppa, Karri, Timo Hakulinen, and Arun Pokhrel. 
-"Choosing the net survival method for cancer survival estimation." 
-European Journal of Cancer (2013).
-\doi{10.1016/j.ejca.2013.09.019}
-
-CHIANG, Chin Long. Introduction to stochastic processes in biostatistics. 
-1968. ISBN-14: 978-0471155003
-
-Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
-Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
-\doi{10.1016/B978-0-12-801238-3.02745-8}
-}
-\seealso{
-\code{\link{splitMulti}}, \code{\link{lexpand}}, 
-\code{\link{ICSS}}, \code{\link{sire}}
-\href{../doc/survtab_examples.html}{The survtab_examples vignette}
-
-Other main functions: 
-\code{\link{Surv}()},
-\code{\link{rate}()},
-\code{\link{relpois_ag}()},
-\code{\link{relpois}()},
-\code{\link{sirspline}()},
-\code{\link{sir}()},
-\code{\link{survmean}()},
-\code{\link{survtab}()}
-
-Other survtab functions: 
-\code{\link{Surv}()},
-\code{\link{lines.survtab}()},
-\code{\link{plot.survtab}()},
-\code{\link{print.survtab}()},
-\code{\link{summary.survtab}()},
-\code{\link{survtab}()}
-}
-\concept{main functions}
-\concept{survtab functions}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/survival_aggregated.R
+\name{survtab_ag}
+\alias{survtab_ag}
+\title{Estimate Survival Time Functions}
+\usage{
+survtab_ag(
+  formula = NULL,
+  data,
+  adjust = NULL,
+  weights = NULL,
+  surv.breaks = NULL,
+  n = "at.risk",
+  d = "from0to1",
+  n.cens = "from0to0",
+  pyrs = "pyrs",
+  d.exp = "d.exp",
+  n.pp = NULL,
+  d.pp = "d.pp",
+  d.pp.2 = "d.pp.2",
+  n.cens.pp = "n.cens.pp",
+  pyrs.pp = "pyrs.pp",
+  d.exp.pp = "d.exp.pp",
+  surv.type = "surv.rel",
+  surv.method = "hazard",
+  relsurv.method = "e2",
+  subset = NULL,
+  conf.level = 0.95,
+  conf.type = "log-log",
+  verbose = FALSE
+)
+}
+\arguments{
+\item{formula}{a \code{formula}; the response 
+must be the time scale to compute survival time function estimates
+over, e.g. \code{fot ~ sex}. Variables on the right-hand side of the formula
+separated by \code{+} are considered stratifying variables, for which 
+estimates are computed separately. May contain usage of \code{adjust()} 
+--- see Details and Examples.}
+
+\item{data}{since popEpi 0.4.0, a \code{data.frame}
+containing variables used in \code{formula} and other arguments.
+\code{aggre} objects are recommended as they contain information on any
+time scales and are therefore safer; for creating \code{aggre} objects see
+\code{\link{as.aggre}} when your data is already aggregated and \code{aggre}
+for aggregating split \code{Lexis} objects.}
+
+\item{adjust}{can be used as an alternative to passing variables to 
+argument \code{formula} within a call to \code{adjust()}; e.g.
+\code{adjust = "agegr"}. \link[=flexible_argument]{Flexible input}.}
+
+\item{weights}{typically a list of weights or a \code{character} string
+specifying an age group standardization scheme; see
+the \link[=direct_standardization]{dedicated help page} 
+and examples. NOTE: \code{weights = "internal"} is based on the counts
+of persons in follow-up at the start of follow-up (typically T = 0)}
+
+\item{surv.breaks}{a vector of breaks on the 
+survival time scale. Optional if \code{data} is an \code{aggre} object
+and mandatory otherwise. Must define each intended interval;
+e.g. \code{surv.breaks = 0:5} when data has intervals defined by 
+breaks \code{seq(0, 5, 1/12)} will aggregate to wider intervals first.
+It is generally recommended (and sufficient; 
+see Seppa, Dyban and Hakulinen (2015)) to use monthly
+intervals where applicable.}
+
+\item{n}{variable containing counts of subjects at-risk at the start of a 
+time interval; e.g. \code{n = "at.risk"}. 
+Required when \code{surv.method = "lifetable"}.
+\link[=flexible_argument]{Flexible input}.}
+
+\item{d}{variable(s) containing counts of subjects experiencing an event. 
+With only one type of event, e.g. \code{d = "deaths"}. With multiple types of 
+events (for CIF or cause-specific survival estimation), supply e.g.
+\code{d = c("canD", "othD")}. If the survival time function to be estimated
+does not use multiple types of events, supplying more than one variable
+to \code{d} simply causes the variables to be added together. 
+Always required. \link[=flexible_argument]{Flexible input}.}
+
+\item{n.cens}{variable containing counts of subjects censored during a 
+survival time interval; E.g. \code{n.cens = "alive"}.
+Required when \code{surv.method = "lifetable"}. 
+\link[=flexible_argument]{Flexible input}.}
+
+\item{pyrs}{variable containing total subject-time accumulated within a 
+survival time interval; E.g. \code{pyrs = "pyrs"}. 
+Required when \code{surv.method = "hazard"}. Flexible input.}
+
+\item{d.exp}{variable denoting total "expected numbers of events" 
+(typically computed \code{pyrs * pop.haz}, where 
+\code{pop.haz} is the expected hazard level) 
+accumulated within a survival time interval; E.g. \code{pyrs = "pyrs"}.
+Required when computing EdererII relative survivals or 
+CIFs based on excess counts of events. Flexible input.}
+
+\item{n.pp}{variable containing total Pohar-Perme weighted counts of
+subjects at risk in an interval,
+supplied as argument \code{n} is supplied. 
+Computed originally on the subject
+level as analogous to \code{pp * as.integer(status == "at-risk")}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{d.pp}{variable(s) containing Pohar-Perme weighted counts of events,
+supplied as argument \code{d} is supplied. Computed originally on the subject
+level as analogous to \code{pp * as.integer(status == some_event)}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{d.pp.2}{variable(s) containing total Pohar-Perme 
+"double-weighted" counts of events,
+supplied as argument \code{d} is supplied. Computed originally on the subject
+level as analogous to \code{pp * pp * as.integer(status == some_event)}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{n.cens.pp}{variable containing total Pohar-Perme weighted counts 
+censorings,
+supplied as argument \code{n.cens} is supplied. 
+Computed originally on the subject
+level as analogous to \code{pp * as.integer(status == "censored")}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{pyrs.pp}{variable containing total Pohar-Perme weighted subject-times,
+supplied as argument \code{pyrs} is supplied. 
+Computed originally on the subject
+level as analogous to \code{pp * pyrs}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{d.exp.pp}{variable containing total Pohar-Perme weighted counts 
+of excess events,
+supplied as argument \code{pyrs} is supplied. 
+Computed originally on the subject
+level as analogous to \code{pp * d.exp}.
+Required when \code{relsurv.method = "pp"}. Flexible input.}
+
+\item{surv.type}{one of \code{'surv.obs'},
+\code{'surv.cause'}, \code{'surv.rel'}, 
+\code{'cif.obs'} or \code{'cif.rel'}; 
+defines what kind of survival time function(s) is/are estimated; see Details}
+
+\item{surv.method}{either \code{'lifetable'} or \code{'hazard'}; determines
+the method of calculating survival time functions, where the former computes
+ratios such as \code{p = d/(n - n.cens)} 
+and the latter utilizes subject-times 
+(typically person-years) for hazard estimates such as \code{d/pyrs} 
+which are used to compute survival time function estimates.
+The former method requires argument \code{n.cens} and the latter 
+argument \code{pyrs} to be supplied.}
+
+\item{relsurv.method}{either \code{'e2'} or \code{'pp'}; 
+defines whether to compute relative survival using the
+EdererII method or using Pohar-Perme weighting;
+ignored if \code{surv.type != "surv.rel"}}
+
+\item{subset}{a logical condition; e.g. \code{subset = sex == 1}; 
+subsets the data before computations}
+
+\item{conf.level}{confidence level used in confidence intervals; 
+e.g. \code{0.95} for 95 percent confidence intervals}
+
+\item{conf.type}{character string; must be one of \code{"plain"}, 
+\code{"log-log"} and \code{"log"}; 
+defines the transformation used on the survival time
+function to yield confidence 
+intervals via the delta method}
+
+\item{verbose}{logical; if \code{TRUE}, the function is chatty and
+returns some messages and timings along the process}
+}
+\value{
+Returns a table of life time function values and other 
+information with survival intervals as rows.
+Returns some of the following estimates of survival time functions:
+
+\itemize{
+ \item \code{surv.obs} - observed (raw, overall) survival
+ \item \code{surv.obs.K} - observed cause-specific survival for cause K
+ \item \code{CIF_k} - cumulative incidence function for cause \code{k}
+ \item \code{CIF.rel} - cumulative incidence function using excess cases
+ \item \code{r.e2} -  relative survival, EdererII
+ \item \code{r.pp} -  relative survival, Pohar-Perme weighted
+}
+The suffix \code{.as} implies adjusted estimates, and \code{.lo} and
+\code{.hi} imply lower and upper confidence limits, respectively. 
+The prefix \code{SE.} stands for standard error.
+}
+\description{
+This function estimates survival time functions: survival, 
+relative/net survival, and crude/absolute risk functions (CIF).
+}
+\section{Basics}{
+
+
+This function computes interval-based estimates of survival time functions,
+where the intervals are set by the user. For product-limit-based
+estimation see packages \pkg{survival} and \pkg{relsurv}.
+
+if \code{surv.type = 'surv.obs'}, only 'raw' observed survival 
+is estimated over the chosen time intervals. With
+\code{surv.type = 'surv.rel'}, also relative survival estimates 
+are supplied in addition to observed survival figures. 
+
+\code{surv.type = 'cif.obs'} requests cumulative incidence functions (CIF) 
+to be estimated. 
+CIFs are estimated for each competing risk based 
+on a survival-interval-specific proportional hazards
+assumption as described by Chiang (1968).  
+With \code{surv.type = 'cif.rel'}, a CIF is estimated with using 
+excess cases as the ''cause-specific'' cases. Finally, with 
+\code{surv.type = 'surv.cause'}, cause-specific survivals are 
+estimated separately for each separate type of event. 
+
+In hazard-based estimation (\code{surv.method = "hazard"}) survival
+time functions are transformations of the estimated corresponding hazard
+in the intervals. The hazard itself is estimated using counts of events
+(or excess events) and total subject-time in the interval. Life table
+\code{surv.method = "lifetable"} estimates are constructed as transformations 
+of probabilities computed using counts of events and counts of subjects 
+at risk.
+
+
+The vignette \href{../doc/survtab_examples.html}{survtab_examples} 
+has some practical examples.
+}
+
+\section{Relative survival}{
+
+ 
+When \code{surv.type = 'surv.rel'}, the user can choose 
+\code{relsurv.method = 'pp'}, whereupon Pohar-Perme weighting is used.
+By default \code{relsurv.method = 'e2'}, i.e. the Ederer II method
+is used to estimate relative survival.
+}
+
+\section{Adjusted estimates}{
+
+
+Adjusted estimates in this context mean computing estimates separately
+by the levels of adjusting variables and returning weighted averages
+of the estimates. For example, computing estimates separately by
+age groups and returning a weighted average estimate (age-adjusted estimate).
+
+Adjusting requires specification of both the adjusting variables and
+the weights for all the levels of the adjusting variables. The former can be
+accomplished by using \code{adjust()} with the argument \code{formula},
+or by supplying variables directly to argument \code{adjust}. E.g. the
+following are all equivalent:
+
+\code{formula = fot ~ sex + adjust(agegr) + adjust(area)}
+
+\code{formula = fot ~ sex + adjust(agegr, area)}
+
+\code{formula  = fot ~ sex, adjust = c("agegr", "area")}
+
+\code{formula  = fot ~ sex, adjust = list(agegr, area)}
+
+The adjusting variables must match with the variable names in the
+argument \code{weights};
+see the \link[=direct_standardization]{dedicated help page}. 
+Typically weights are supplied as a \code{list} or
+a \code{data.frame}. The former can be done by e.g.
+
+\code{weights = list(agegr = VEC1, area = VEC2)},
+
+where \code{VEC1} and \code{VEC2} are vectors of weights (which do not
+have to add up to one). See 
+\href{../doc/survtab_examples.html}{survtab_examples} 
+for an example of using a \code{data.frame} to pass weights.
+}
+
+\section{Period analysis and other data selection schemes}{
+
+
+To calculate e.g. period analysis (delayed entry) estimates, 
+limit the data when/before supplying to this function.See 
+\href{../doc/survtab_examples.html}{survtab_examples}.
+}
+
+\section{Data requirements}{
+
+
+\code{survtab_ag} computes estimates of survival time functions using 
+pre-aggregated data. For using subject-level data directly, use 
+\code{\link{survtab}}. For aggregating data, see \code{\link{lexpand}}
+and \code{\link{aggre}}. 
+
+By default, and if data is an \code{aggre} object (not mandatory), 
+\code{survtab_ag} makes use of the exact same breaks that were used in 
+splitting the original data (with e.g. \code{lexpand}), so it is not 
+necessary to specify any \code{surv.breaks}. If specified, the 
+\code{surv.breaks} must be a subset of the pertinent 
+pre-existing breaks. When data is not an \code{aggre} object, breaks
+must always be specified. Interval lengths (\code{delta} in output) are 
+also calculated based on whichever breaks are used, 
+so the upper limit of the breaks should
+therefore be meaningful and never e.g. \code{Inf}.
+}
+
+\examples{
+## see more examples with explanations in vignette("survtab_examples")
+
+#### survtab_ag usage
+
+data("sire", package = "popEpi")
+## prepare data for e.g. 5-year "period analysis" for 2008-2012
+## note: sire is a simulated cohort integrated into popEpi.
+BL <- list(fot=seq(0, 5, by = 1/12),
+           per = c("2008-01-01", "2013-01-01"))
+x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status \%in\% 1:2,
+             breaks = BL,
+             pophaz = popmort,
+             aggre = list(fot))
+             
+## calculate relative EdererII period method
+## NOTE: x is an aggre object here, so surv.breaks are deduced
+## automatically
+st <- survtab_ag(fot ~ 1, data = x)
+
+summary(st, t = 1:5) ## annual estimates
+summary(st, q = list(r.e2 = 0.75)) ## 1st interval where r.e2 < 0.75 at end
+\donttest{
+plot(st)
+
+
+## non-aggre data: first call to survtab_ag would fail
+df <- data.frame(x)
+# st <- survtab_ag(fot ~ 1, data = x)
+st <- survtab_ag(fot ~ 1, data = x, surv.breaks = BL$fot)
+
+## calculate age-standardised 5-year relative survival ratio using 
+## Ederer II method and period approach 
+
+sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
+BL <- list(fot=seq(0, 5, by = 1/12),
+           per = c("2008-01-01", "2013-01-01"))
+x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = status \%in\% 1:2,
+             breaks = BL,
+             pophaz = popmort,
+             aggre = list(agegr, fot))
+
+## age standardisation using internal weights (age distribution of 
+## patients diagnosed within the period window)
+## (NOTE: what is done here is equivalent to using weights = "internal")
+w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0], FUN = sum)
+names(w) <- c("agegr", "weights")
+
+st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
+plot(st, y = "r.e2.as", col = c("blue"))
+
+## age standardisation using ICSS1 weights
+data(ICSS)
+cut <- c(0, 45, 55, 65, 75, Inf)
+agegr <- cut(ICSS$age, cut, right = FALSE)
+w <- aggregate(ICSS1~agegr, data = ICSS, FUN = sum)
+names(w) <- c("agegr", "weights")
+
+st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
+lines(st, y = "r.e2.as", col = c("red"))
+
+
+## cause-specific survival
+sire$stat <- factor(sire$status, 0:2, c("alive", "canD", "othD"))
+x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+             status = stat,
+             breaks = BL,
+             pophaz = popmort,
+             aggre = list(agegr, fot))
+st <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w,
+                 d = c("fromalivetocanD", "fromalivetoothD"),
+                 surv.type = "surv.cause")
+plot(st, y = "surv.obs.fromalivetocanD.as")
+lines(st, y = "surv.obs.fromalivetoothD.as", col = "red")
+
+
+}
+}
+\references{
+Perme, Maja Pohar, Janez Stare, and Jacques Esteve. 
+"On estimation in relative survival." Biometrics 68.1 (2012): 113-120.
+\doi{10.1111/j.1541-0420.2011.01640.x}
+
+Hakulinen, Timo, Karri Seppa, and Paul C. Lambert. 
+"Choosing the relative survival method for cancer survival estimation." 
+European Journal of Cancer 47.14 (2011): 2202-2210.
+\doi{10.1016/j.ejca.2011.03.011}
+ 
+Seppa, Karri, Timo Hakulinen, and Arun Pokhrel. 
+"Choosing the net survival method for cancer survival estimation." 
+European Journal of Cancer (2013).
+\doi{10.1016/j.ejca.2013.09.019}
+
+CHIANG, Chin Long. Introduction to stochastic processes in biostatistics. 
+1968. ISBN-14: 978-0471155003
+
+Seppa K., Dyba T. and Hakulinen T.: Cancer Survival, 
+Reference Module in Biomedical Sciences. Elsevier. 08-Jan-2015.
+\doi{10.1016/B978-0-12-801238-3.02745-8}
+}
+\seealso{
+\code{\link{splitMulti}}, \code{\link{lexpand}}, 
+\code{\link{ICSS}}, \code{\link{sire}}
+\href{../doc/survtab_examples.html}{The survtab_examples vignette}
+
+Other main functions: 
+\code{\link{Surv}()},
+\code{\link{rate}()},
+\code{\link{relpois_ag}()},
+\code{\link{relpois}()},
+\code{\link{sirspline}()},
+\code{\link{sir}()},
+\code{\link{survmean}()},
+\code{\link{survtab}()}
+
+Other survtab functions: 
+\code{\link{Surv}()},
+\code{\link{lines.survtab}()},
+\code{\link{plot.survtab}()},
+\code{\link{print.survtab}()},
+\code{\link{summary.survtab}()},
+\code{\link{survtab}()}
+}
+\concept{main functions}
+\concept{survtab functions}
diff --git a/man/try2int.Rd b/man/try2int.Rd
index b56f4fb..3787a1f 100644
--- a/man/try2int.Rd
+++ b/man/try2int.Rd
@@ -1,31 +1,31 @@
-% Generated by roxygen2: do not edit by hand
-% Please edit documentation in R/utility_functions.R
-\name{try2int}
-\alias{try2int}
-\title{Attempt coercion to integer}
-\source{
-\href{https://stackoverflow.com/questions/3476782/how-to-check-if-the-number-is-integer}{Stackoverflow thread}
-}
-\usage{
-try2int(obj, tol = .Machine$double.eps^0.5)
-}
-\arguments{
-\item{obj}{a numeric vector}
-
-\item{tol}{tolerance; if each numeric value in \code{obj} deviate from
-the corresponding integers at most the value of \code{tol}, they are considered
-to be integers; e.g. by default \code{1 + .Machine$double.eps} is considered
-to be an integer but \code{1 + .Machine$double.eps^0.49} is not.}
-}
-\value{
-An `integer` vector if no information is lost in coercion; else `numeric` 
-vector.
-}
-\description{
-Attempts to convert a numeric object to integer, 
-but won't if loss of information is imminent (if values after decimal
-are not zero for even one value in \code{obj})
-}
-\author{
-James Arnold
-}
+% Generated by roxygen2: do not edit by hand
+% Please edit documentation in R/utility_functions.R
+\name{try2int}
+\alias{try2int}
+\title{Attempt coercion to integer}
+\source{
+\href{https://stackoverflow.com/questions/3476782/how-to-check-if-the-number-is-integer}{Stackoverflow thread}
+}
+\usage{
+try2int(obj, tol = .Machine$double.eps^0.5)
+}
+\arguments{
+\item{obj}{a numeric vector}
+
+\item{tol}{tolerance; if each numeric value in \code{obj} deviate from
+the corresponding integers at most the value of \code{tol}, they are considered
+to be integers; e.g. by default \code{1 + .Machine$double.eps} is considered
+to be an integer but \code{1 + .Machine$double.eps^0.49} is not.}
+}
+\value{
+An `integer` vector if no information is lost in coercion; else `numeric` 
+vector.
+}
+\description{
+Attempts to convert a numeric object to integer, 
+but won't if loss of information is imminent (if values after decimal
+are not zero for even one value in \code{obj})
+}
+\author{
+James Arnold
+}
diff --git a/tests/testthat.R b/tests/testthat.R
index 16a635f..71d474e 100644
--- a/tests/testthat.R
+++ b/tests/testthat.R
@@ -1,15 +1,15 @@
-
-if (requireNamespace("testthat")) {
-  library("testthat")
-  library("popEpi")
-  library("data.table")
-
-  using_r_devel <- grepl(pattern = "devel", x = R.version$status)
-  if (using_r_devel) {
-    ## memory leak problem in data.table 1.11.2 in R-devel (3.6.0 atm)
-    requireNamespace("data.table")
-    data.table::setDTthreads(threads = 1L)
-  }
-
-  testthat::test_check("popEpi")
-}
+
+if (requireNamespace("testthat")) {
+  library("testthat")
+  library("popEpi")
+  library("data.table")
+
+  using_r_devel <- grepl(pattern = "devel", x = R.version$status)
+  if (using_r_devel) {
+    ## memory leak problem in data.table 1.11.2 in R-devel (3.6.0 atm)
+    requireNamespace("data.table")
+    data.table::setDTthreads(threads = 1L)
+  }
+
+  testthat::test_check("popEpi")
+}
diff --git a/tests/testthat/Rplots.pdf b/tests/testthat/Rplots.pdf
deleted file mode 100644
index 392c5bd..0000000
Binary files a/tests/testthat/Rplots.pdf and /dev/null differ
diff --git a/tests/testthat/test_Surv.R b/tests/testthat/test_Surv.R
index 6508ed9..79612ad 100644
--- a/tests/testthat/test_Surv.R
+++ b/tests/testthat/test_Surv.R
@@ -1,25 +1,25 @@
-
-
-
-
-
-testthat::context("survival::Surv and popEpi::Surv harmony")
-
-
-
-
-testthat::test_that("survival::Surv and popEpi::Surv's formals are in harmony",{
-  
-  popepi_formals <- as.list(formals(popEpi::Surv))
-  survival_formals <- as.list(formals(survival::Surv))
-  testthat::expect_identical(
-    object = popepi_formals,
-    expected = survival_formals
-  )
-  
-})
-
-
-
-
-
+
+
+
+
+
+testthat::context("survival::Surv and popEpi::Surv harmony")
+
+
+
+
+testthat::test_that("survival::Surv and popEpi::Surv's formals are in harmony",{
+  
+  popepi_formals <- as.list(formals(popEpi::Surv))
+  survival_formals <- as.list(formals(survival::Surv))
+  testthat::expect_identical(
+    object = popepi_formals,
+    expected = survival_formals
+  )
+  
+})
+
+
+
+
+
diff --git a/tests/testthat/test_aggre.R b/tests/testthat/test_aggre.R
index 10f67e3..f921be5 100644
--- a/tests/testthat/test_aggre.R
+++ b/tests/testthat/test_aggre.R
@@ -1,228 +1,228 @@
-testthat::context("aggre")
-
-testthat::test_that("aggre leaves original data untouched", {
-  
-  x <- sire[1:100,]
-  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
-  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, breaks=BL)
-  
-  ## scramble order
-  set.seed(1L)
-  x <- x[sample(x = .N, size = .N, replace = FALSE)]
-  setkeyv(x, NULL)
-  
-  setDT(x)
-  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
-  
-  xor <- copy(x)
-  
-  ag1 <- aggre(x, by = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
-  
-  testthat::expect_identical(x, xor)
-})
-
-testthat::test_that("aggre works with by = NULL", {
-  
-  sr <- popEpi::sire[dg_date < ex_date,][1:1000,]
-  
-  BL <- list(fot= seq(0,20,1), age= c(0:100, Inf), per= c(1960:2014))
-  x <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, breaks=BL)
-  
-  ag1 <- aggre(x, by = NULL)
-  testthat::expect_equal(as.numeric(ag1), c(9539.1903286174274, 1000, 373, 627))
-  
-})
-
-testthat::test_that("aggre and lexpand produce the same results", {
-  sr <- popEpi::sire[dg_date < ex_date,][1:1000,]
-  
-  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
-  x <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, breaks=BL)
-  if (!is.data.table(x)) setDF2DT(x)
-  
-  e <- quote(list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
-  v <- c("gender", "sex", "sex", "surv.int", "per", "agegr")
-  
-  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"))
-  x2 <- aggre(x, by = e, verbose = FALSE)
-  x3 <- aggre(x, by = e, type = "full", verbose = FALSE)
-  x4 <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, aggre.type = "non-empty",
-                breaks=BL, aggre = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
-  x5 <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, aggre.type = "cartesian",
-                breaks=BL, aggre = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
-  
-  x[, fot := popEpi:::cutLow(fot, BL$fot)]
-  x[, age := popEpi:::cutLow(age, BL$age)]
-  x[, per := popEpi:::cutLow(per, BL$per)]
-  
-  x <- x[, list(pyrs = sum(lex.dur), obs = sum(lex.Xst)), keyby = e]
-  x <- x[pyrs > 0 & !is.na(pyrs)]
-  
-  if (!is.data.table(x2)) setDF2DT(x2)
-  if (!is.data.table(x3)) setDF2DT(x3)
-  if (!is.data.table(x4)) setDF2DT(x4)
-  if (!is.data.table(x5)) setDF2DT(x5)
-  
-  setkeyv(x, v)
-  setkeyv(x2, v)
-  setkeyv(x3, v)
-  setkeyv(x4, v)
-  setkeyv(x5, v)
-  
-  testthat::expect_equal(x2$pyrs, x$pyrs, tolerance = 1e-05)
-  testthat::expect_equal(x2$from0to1, x$obs, tolerance = 1e-05)
-  
-  testthat::expect_equal(sum(x2$pyrs), sum(x3$pyrs), tolerance = 1e-05)
-  testthat::expect_equal(sum(x2$from0to1), sum(x3$from0to1), tolerance = 1e-05)
-  
-  testthat::expect_equal(sum(x2$pyrs), sum(x4$pyrs), tolerance = 1e-05)
-  testthat::expect_equal(sum(x2$from0to1), sum(x4$from0to1), tolerance = 1e-05)
-  
-  testthat::expect_equal(x3$pyrs, x5$pyrs, tolerance = 1e-05)
-  testthat::expect_equal(x3$from0to0, x5$from0to0, tolerance = 1e-05)
-  testthat::expect_equal(sum(x3$from0to1), sum(x5$from0to1), tolerance = 1e-05)
-  
-  testthat::expect_equal(x2$pyrs, x4$pyrs, tolerance = 1e-05)
-  testthat::expect_equal(x2$from0to0, x4$from0to0, tolerance = 1e-05)
-  testthat::expect_equal(sum(x2$from0to1), sum(x4$from0to1), tolerance = 1e-05)
-})
-
-
-testthat::test_that("aggre()'s by argument works flexibly", {
-  library(Epi)
-  BL <- list(fot = 0:5, per = c(1995,2015))
-  for (cond in c(FALSE, TRUE)) {
-    x <- Lexis(data = sire[dg_date < ex_date,][1:500, ], entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
-               exit = list(per = get.yrs(ex_date)), exit.status = status, 
-               entry.status = 0)
-    x <- splitMulti(x, breaks = BL)
-    setDF(x)
-    setattr(x, "class", c("Lexis", "data.frame"))
-    x$agegr <- cut(x$dg_age, 2)
-    if (cond) {
-      forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"))
-      alloc.col(x)
-    }
-    
-    a <- aggre(x, by = list(agegr = cut(dg_age, 2), sex, fot, per = per), type = "unique")
-    b <- aggre(x, by = c("agegr", "sex", "fot", "per"), type = "unique")
-    
-    testthat::expect_equal(a, b)
-    
-    a <- aggre(x, by = cut(dg_age, 2), type = "unique")
-    setnames(a, "cut", "agegr")
-    attr(a, "aggre.meta")$by <- "agegr"
-    b <- aggre(x, by = c("agegr"), type = "unique")
-    c <- aggre(x, by = list(agegr = cut(dg_age, 2)), type = "unique")
-    d<- aggre(x, by = agegr, type = "unique")
-    
-    testthat::expect_equal(a, b)
-    testthat::expect_equal(b, c)
-    testthat::expect_equal(c, d)
-  }
-  
-  
-})
-
-testthat::test_that("subset argument works properly", {
-  
-  
-  x <- sire[dg_date < ex_date, ][1:1000,]
-  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
-  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, breaks=BL)
-  # setDT2DF(x)
-  x2 <- x[x$dg_age <= 55L, ]
-  
-  setDT(x)
-  setDT(x2)
-  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
-  forceLexisDT(x2, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
-  
-  ag <- quote(list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
-  ag1 <- aggre(x, by = ag, subset = dg_age <= 55L)
-  ag2 <- aggre(x2, by = ag)
-  
-  ag3 <- aggre(x, by = ag, type = "full", subset = dg_age <= 55L)
-  ag4 <- aggre(x2, by = ag, type = "full") 
-  
-  testthat::expect_identical(ag1, ag2)
-  testthat::expect_identical(ag3, ag4)
-  
-})
-
-
-testthat::test_that("at.risk column works as intended", {
-  ## normal case - no late entry. Just lots of breaks.
-  popEpi:::skip_normally()
-  x <- sire[dg_date < ex_date, ][1:1000,]
-  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
-  
-  x <- Lexis(data = x, 
-             entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
-             exit = list(per = get.yrs(ex_date)), exit.status = status, 
-             entry.status = 0)
-  
-  x <- splitMulti(x, breaks = BL, drop = TRUE)
-  
-  ag <- aggre(x, by = list(sex, fot))
-  setkey(ag, sex, fot)
-  
-  ## total events and changes in at.risk should be congruent here
-  ag[, ndiff := at.risk - c(at.risk[-1], NA), by = list(sex)]
-  ag[!is.na(ndiff), events := from0to0 + from0to1 + from0to2]
-  
-  testthat::expect_equal(ag$ndiff, ag$events)
-  
-  ## compare at.risk with manually computed at.risk and events
-  x[, evented := detectEvents(x, breaks = attr(x, "breaks"), by = "lex.id") != 0L]
-  x[, normalEntry := fot %in% BL$fot]
-  x[, cutFot := cutLow(fot, BL$fot)]
-  byDT <- CJ(sex = 1, 
-             cutFot = BL$fot[-length(BL$fot)])
-  n.start <- x[byDT, .(sum(normalEntry & !duplicated(lex.id)),
-                       sum(evented)), by = .EACHI,
-               on = names(byDT)]
-  n.start[is.na(ag$ndiff), V2 := NA]
-  testthat::expect_equal(ag$at.risk, n.start$V1)
-  testthat::expect_equal(ag$ndiff, n.start$V2)
-})
-
-
-
-testthat::test_that("at.risk column works as intended, Vol. 2", {
-  popEpi:::skip_normally()
-  ## period analysis case - some observations are late entry.
-  data(sire)
-  
-  BL <- list(fot=seq(0, 5, by = 1/12),
-             per = c(2008,2013))
-  
-  x <- Lexis(data = sire[dg_date < ex_date,], 
-             entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
-             exit = list(per = get.yrs(ex_date)), exit.status = status, 
-             entry.status = 0)
-  
-  x <- splitMulti(x, breaks = BL, drop = TRUE)
-  
-  a <- aggre(x, by = list(sex, per, fot))
-  setkey(a, sex, per, fot)
-  a[, ndiff := at.risk - c(at.risk[-1], NA), by = list(sex, per)]
-  a[!is.na(ndiff), events := from0to0 + from0to1 + from0to2]
-  
-  x[, normalEntry := fot %in% BL$fot]
-  x[, cutPer := cutLow(per, BL$per)]
-  x[, cutFot := cutLow(fot, BL$fot)]
-  byDT <- CJ(sex = 1, cutPer = BL$per[-length(BL$per)], 
-             cutFot = BL$fot[-length(BL$fot)])
-  n.start <- x[byDT, sum(normalEntry & !duplicated(lex.id)), by = .EACHI,
-               on = names(byDT)]
-  
-  testthat::expect_equal(a$at.risk, n.start$V1)
-})
-
+testthat::context("aggre")
+
+testthat::test_that("aggre leaves original data untouched", {
+  
+  x <- sire[1:100,]
+  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
+  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, breaks=BL)
+  
+  ## scramble order
+  set.seed(1L)
+  x <- x[sample(x = .N, size = .N, replace = FALSE)]
+  setkeyv(x, NULL)
+  
+  setDT(x)
+  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
+  
+  xor <- copy(x)
+  
+  ag1 <- aggre(x, by = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
+  
+  testthat::expect_identical(x, xor)
+})
+
+testthat::test_that("aggre works with by = NULL", {
+  
+  sr <- popEpi::sire[dg_date < ex_date,][1:1000,]
+  
+  BL <- list(fot= seq(0,20,1), age= c(0:100, Inf), per= c(1960:2014))
+  x <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, breaks=BL)
+  
+  ag1 <- aggre(x, by = NULL)
+  testthat::expect_equal(as.numeric(ag1), c(9539.1903286174274, 1000, 373, 627))
+  
+})
+
+testthat::test_that("aggre and lexpand produce the same results", {
+  sr <- popEpi::sire[dg_date < ex_date,][1:1000,]
+  
+  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
+  x <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, breaks=BL)
+  if (!is.data.table(x)) setDF2DT(x)
+  
+  e <- quote(list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
+  v <- c("gender", "sex", "sex", "surv.int", "per", "agegr")
+  
+  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"))
+  x2 <- aggre(x, by = e, verbose = FALSE)
+  x3 <- aggre(x, by = e, type = "full", verbose = FALSE)
+  x4 <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, aggre.type = "non-empty",
+                breaks=BL, aggre = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
+  x5 <- lexpand(sr, birth  = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, aggre.type = "cartesian",
+                breaks=BL, aggre = list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
+  
+  x[, fot := popEpi:::cutLow(fot, BL$fot)]
+  x[, age := popEpi:::cutLow(age, BL$age)]
+  x[, per := popEpi:::cutLow(per, BL$per)]
+  
+  x <- x[, list(pyrs = sum(lex.dur), obs = sum(lex.Xst)), keyby = e]
+  x <- x[pyrs > 0 & !is.na(pyrs)]
+  
+  if (!is.data.table(x2)) setDF2DT(x2)
+  if (!is.data.table(x3)) setDF2DT(x3)
+  if (!is.data.table(x4)) setDF2DT(x4)
+  if (!is.data.table(x5)) setDF2DT(x5)
+  
+  setkeyv(x, v)
+  setkeyv(x2, v)
+  setkeyv(x3, v)
+  setkeyv(x4, v)
+  setkeyv(x5, v)
+  
+  testthat::expect_equal(x2$pyrs, x$pyrs, tolerance = 1e-05)
+  testthat::expect_equal(x2$from0to1, x$obs, tolerance = 1e-05)
+  
+  testthat::expect_equal(sum(x2$pyrs), sum(x3$pyrs), tolerance = 1e-05)
+  testthat::expect_equal(sum(x2$from0to1), sum(x3$from0to1), tolerance = 1e-05)
+  
+  testthat::expect_equal(sum(x2$pyrs), sum(x4$pyrs), tolerance = 1e-05)
+  testthat::expect_equal(sum(x2$from0to1), sum(x4$from0to1), tolerance = 1e-05)
+  
+  testthat::expect_equal(x3$pyrs, x5$pyrs, tolerance = 1e-05)
+  testthat::expect_equal(x3$from0to0, x5$from0to0, tolerance = 1e-05)
+  testthat::expect_equal(sum(x3$from0to1), sum(x5$from0to1), tolerance = 1e-05)
+  
+  testthat::expect_equal(x2$pyrs, x4$pyrs, tolerance = 1e-05)
+  testthat::expect_equal(x2$from0to0, x4$from0to0, tolerance = 1e-05)
+  testthat::expect_equal(sum(x2$from0to1), sum(x4$from0to1), tolerance = 1e-05)
+})
+
+
+testthat::test_that("aggre()'s by argument works flexibly", {
+  library(Epi)
+  BL <- list(fot = 0:5, per = c(1995,2015))
+  for (cond in c(FALSE, TRUE)) {
+    x <- Lexis(data = sire[dg_date < ex_date,][1:500, ], entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
+               exit = list(per = get.yrs(ex_date)), exit.status = status, 
+               entry.status = 0)
+    x <- splitMulti(x, breaks = BL)
+    setDF(x)
+    setattr(x, "class", c("Lexis", "data.frame"))
+    x$agegr <- cut(x$dg_age, 2)
+    if (cond) {
+      forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"))
+      alloc.col(x)
+    }
+    
+    a <- aggre(x, by = list(agegr = cut(dg_age, 2), sex, fot, per = per), type = "unique")
+    b <- aggre(x, by = c("agegr", "sex", "fot", "per"), type = "unique")
+    
+    testthat::expect_equal(a, b)
+    
+    a <- aggre(x, by = cut(dg_age, 2), type = "unique")
+    setnames(a, "cut", "agegr")
+    attr(a, "aggre.meta")$by <- "agegr"
+    b <- aggre(x, by = c("agegr"), type = "unique")
+    c <- aggre(x, by = list(agegr = cut(dg_age, 2)), type = "unique")
+    d<- aggre(x, by = agegr, type = "unique")
+    
+    testthat::expect_equal(a, b)
+    testthat::expect_equal(b, c)
+    testthat::expect_equal(c, d)
+  }
+  
+  
+})
+
+testthat::test_that("subset argument works properly", {
+  
+  
+  x <- sire[dg_date < ex_date, ][1:1000,]
+  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
+  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, breaks=BL)
+  # setDT2DF(x)
+  x2 <- x[x$dg_age <= 55L, ]
+  
+  setDT(x)
+  setDT(x2)
+  forceLexisDT(x, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
+  forceLexisDT(x2, breaks = BL, allScales = c("fot", "per", "age"), key = FALSE)
+  
+  ag <- quote(list(gender = factor(sex, 1, "f"), sex, surv.int = fot, per, agegr = age))
+  ag1 <- aggre(x, by = ag, subset = dg_age <= 55L)
+  ag2 <- aggre(x2, by = ag)
+  
+  ag3 <- aggre(x, by = ag, type = "full", subset = dg_age <= 55L)
+  ag4 <- aggre(x2, by = ag, type = "full") 
+  
+  testthat::expect_identical(ag1, ag2)
+  testthat::expect_identical(ag3, ag4)
+  
+})
+
+
+testthat::test_that("at.risk column works as intended", {
+  ## normal case - no late entry. Just lots of breaks.
+  popEpi:::skip_normally()
+  x <- sire[dg_date < ex_date, ][1:1000,]
+  BL <- list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014))
+  
+  x <- Lexis(data = x, 
+             entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
+             exit = list(per = get.yrs(ex_date)), exit.status = status, 
+             entry.status = 0)
+  
+  x <- splitMulti(x, breaks = BL, drop = TRUE)
+  
+  ag <- aggre(x, by = list(sex, fot))
+  setkey(ag, sex, fot)
+  
+  ## total events and changes in at.risk should be congruent here
+  ag[, ndiff := at.risk - c(at.risk[-1], NA), by = list(sex)]
+  ag[!is.na(ndiff), events := from0to0 + from0to1 + from0to2]
+  
+  testthat::expect_equal(ag$ndiff, ag$events)
+  
+  ## compare at.risk with manually computed at.risk and events
+  x[, evented := detectEvents(x, breaks = attr(x, "breaks"), by = "lex.id") != 0L]
+  x[, normalEntry := fot %in% BL$fot]
+  x[, cutFot := cutLow(fot, BL$fot)]
+  byDT <- CJ(sex = 1, 
+             cutFot = BL$fot[-length(BL$fot)])
+  n.start <- x[byDT, .(sum(normalEntry & !duplicated(lex.id)),
+                       sum(evented)), by = .EACHI,
+               on = names(byDT)]
+  n.start[is.na(ag$ndiff), V2 := NA]
+  testthat::expect_equal(ag$at.risk, n.start$V1)
+  testthat::expect_equal(ag$ndiff, n.start$V2)
+})
+
+
+
+testthat::test_that("at.risk column works as intended, Vol. 2", {
+  popEpi:::skip_normally()
+  ## period analysis case - some observations are late entry.
+  data(sire)
+  
+  BL <- list(fot=seq(0, 5, by = 1/12),
+             per = c(2008,2013))
+  
+  x <- Lexis(data = sire[dg_date < ex_date,], 
+             entry = list(fot = 0, age = dg_age, per = get.yrs(dg_date)),
+             exit = list(per = get.yrs(ex_date)), exit.status = status, 
+             entry.status = 0)
+  
+  x <- splitMulti(x, breaks = BL, drop = TRUE)
+  
+  a <- aggre(x, by = list(sex, per, fot))
+  setkey(a, sex, per, fot)
+  a[, ndiff := at.risk - c(at.risk[-1], NA), by = list(sex, per)]
+  a[!is.na(ndiff), events := from0to0 + from0to1 + from0to2]
+  
+  x[, normalEntry := fot %in% BL$fot]
+  x[, cutPer := cutLow(per, BL$per)]
+  x[, cutFot := cutLow(fot, BL$fot)]
+  byDT <- CJ(sex = 1, cutPer = BL$per[-length(BL$per)], 
+             cutFot = BL$fot[-length(BL$fot)])
+  n.start <- x[byDT, sum(normalEntry & !duplicated(lex.id)), by = .EACHI,
+               on = names(byDT)]
+  
+  testthat::expect_equal(a$at.risk, n.start$V1)
+})
+
diff --git a/tests/testthat/test_epi.R b/tests/testthat/test_epi.R
index 9fa2fb7..f45da12 100644
--- a/tests/testthat/test_epi.R
+++ b/tests/testthat/test_epi.R
@@ -1,35 +1,35 @@
-testthat::context("Epi subsetting methods OK")
-
-
-
-
-
-testthat::test_that("[.Epi exists and works", {
-  
-  xcoh <- structure( list( id = c("A", "B", "C"),
-                           birth = c("14/07/1952", "01/04/1954", "10/06/1987"),
-                           entry = c("04/08/1965", "08/09/1972", "23/12/1991"),
-                           exit = c("27/06/1997", "23/05/1995", "24/07/1998"),
-                           fail = c(1, 0, 1) ),
-                     .Names = c("id", "birth", "entry", "exit", "fail"),
-                     row.names = c("1", "2", "3"),
-                     class = "data.frame" )
-  
-  
-  xcoh <- cal.yr( xcoh, format="%d/%m/%Y", wh=2:4 )
-  
-  Lcoh <- Lexis( entry = list( per=entry ),
-                 exit = list( per=exit, age=exit-birth ),
-                 exit.status = fail,
-                 entry.status = 0,
-                 data = xcoh )
-  
-  e1 <- subset(Lcoh, fail == 1)
-  e2 <- Lcoh[Lcoh$fail == 1, ]
-
-  testthat::expect_identical(e1, e2)
-})
-
-
-
-
+testthat::context("Epi subsetting methods OK")
+
+
+
+
+
+testthat::test_that("[.Epi exists and works", {
+  
+  xcoh <- structure( list( id = c("A", "B", "C"),
+                           birth = c("14/07/1952", "01/04/1954", "10/06/1987"),
+                           entry = c("04/08/1965", "08/09/1972", "23/12/1991"),
+                           exit = c("27/06/1997", "23/05/1995", "24/07/1998"),
+                           fail = c(1, 0, 1) ),
+                     .Names = c("id", "birth", "entry", "exit", "fail"),
+                     row.names = c("1", "2", "3"),
+                     class = "data.frame" )
+  
+  
+  xcoh <- cal.yr( xcoh, format="%d/%m/%Y", wh=2:4 )
+  
+  Lcoh <- Lexis( entry = list( per=entry ),
+                 exit = list( per=exit, age=exit-birth ),
+                 exit.status = fail,
+                 entry.status = 0,
+                 data = xcoh )
+  
+  e1 <- subset(Lcoh, fail == 1)
+  e2 <- Lcoh[Lcoh$fail == 1, ]
+
+  testthat::expect_identical(e1, e2)
+})
+
+
+
+
diff --git a/tests/testthat/test_expo.R b/tests/testthat/test_expo.R
index 67d7189..34a360b 100644
--- a/tests/testthat/test_expo.R
+++ b/tests/testthat/test_expo.R
@@ -1,50 +1,50 @@
-testthat::context("Testing aggregation by categories of exposure")
-
-testthat::test_that("prepExpo works in the simple case", {
-  popEpi:::skip_normally()
-  library(Epi)
-  
-  df <- data.frame(id = "A", birth  = c(1952.4534), 
-                   entry = c(1965.4746, 1972.42845, 1991.78643),
-                   exit = c(1968.56346, 1979.32478, 1997.32432), fail = 0)
-  
-  # Define as Lexis object with timescales calendar time and age
-  x <- Lexis( entry = list(work = 0, per=entry ),
-                 exit = list( per=exit, age=exit-birth ),
-                 exit.status = fail,
-                 entry.status = 0,
-                 id = id,
-                 data = df )
-  
-  x2 <- prepExpo(x, freezeScales = "work", 
-                 cutScale = "per", 
-                 entry = 1964, 
-                 exit = 2012, by = "lex.id")
-  cd <- cumsum(x$lex.dur)
-  exp_work <-  c(0, 0, cd[1], cd[1], cd[2], cd[2], cd[3])
-  testthat::expect_equal(x2$work, exp_work)
-  testthat::expect_equal(x2$per, 1964+c(0,cumsum(x2$lex.dur)[-nrow(x2)]))
-  
-  BL <- list(work = 0:50, age = c(0,18,Inf), per = 1963:2014)
-  x2 <- prepExpo(x, freezeScales = "work", 
-                 cutScale = "per", 
-                 entry = 1964, 
-                 # verbose = TRUE,
-                 exit = 2012, by = "lex.id", 
-                 breaks = BL)
-  ag <- aggre(x2, by = list(lex.id, per, age))
-  
-  xx <- Lexis(entry = list(per = 1964, age = 1964-birth), exit = list(per=2012), data = df[1,])
-  ag2 <- splitMulti(xx, breaks = BL[c("per","age")])
-  ag2 <- aggre(ag2, by = list(lex.id, per, age))
-  
-  setkeyv(ag, c("lex.id","per"))
-  setkeyv(ag2, c("lex.id","per"))
-  
-  testthat::expect_equal(ag$pyrs, ag2$pyrs)
-  
-  
-  
-  
-})
-
+testthat::context("Testing aggregation by categories of exposure")
+
+testthat::test_that("prepExpo works in the simple case", {
+  popEpi:::skip_normally()
+  library(Epi)
+  
+  df <- data.frame(id = "A", birth  = c(1952.4534), 
+                   entry = c(1965.4746, 1972.42845, 1991.78643),
+                   exit = c(1968.56346, 1979.32478, 1997.32432), fail = 0)
+  
+  # Define as Lexis object with timescales calendar time and age
+  x <- Lexis( entry = list(work = 0, per=entry ),
+                 exit = list( per=exit, age=exit-birth ),
+                 exit.status = fail,
+                 entry.status = 0,
+                 id = id,
+                 data = df )
+  
+  x2 <- prepExpo(x, freezeScales = "work", 
+                 cutScale = "per", 
+                 entry = 1964, 
+                 exit = 2012, by = "lex.id")
+  cd <- cumsum(x$lex.dur)
+  exp_work <-  c(0, 0, cd[1], cd[1], cd[2], cd[2], cd[3])
+  testthat::expect_equal(x2$work, exp_work)
+  testthat::expect_equal(x2$per, 1964+c(0,cumsum(x2$lex.dur)[-nrow(x2)]))
+  
+  BL <- list(work = 0:50, age = c(0,18,Inf), per = 1963:2014)
+  x2 <- prepExpo(x, freezeScales = "work", 
+                 cutScale = "per", 
+                 entry = 1964, 
+                 # verbose = TRUE,
+                 exit = 2012, by = "lex.id", 
+                 breaks = BL)
+  ag <- aggre(x2, by = list(lex.id, per, age))
+  
+  xx <- Lexis(entry = list(per = 1964, age = 1964-birth), exit = list(per=2012), data = df[1,])
+  ag2 <- splitMulti(xx, breaks = BL[c("per","age")])
+  ag2 <- aggre(ag2, by = list(lex.id, per, age))
+  
+  setkeyv(ag, c("lex.id","per"))
+  setkeyv(ag2, c("lex.id","per"))
+  
+  testthat::expect_equal(ag$pyrs, ag2$pyrs)
+  
+  
+  
+  
+})
+
diff --git a/tests/testthat/test_lexpand.R b/tests/testthat/test_lexpand.R
index 442251b..f00c49b 100644
--- a/tests/testthat/test_lexpand.R
+++ b/tests/testthat/test_lexpand.R
@@ -1,372 +1,372 @@
-testthat::context("lexpand sanity checks")
-
-
-testthat::test_that("lexpand arguments can be passed as symbol, expression, character name of variable, and symbol of a character variable", {
-  popEpi:::skip_normally()
-  sr <- copy(sire)[dg_date < ex_date, ][1:100,]
-  sr[, id := as.character(1:.N)]
-  
-  x <- lexpand(sr, fot = c(0, Inf), 
-               birth = "bi_date", entry = dg_date, exit = "ex_date", 
-               status = status %in% 1:2, id = "id")
-  
-  x2 <- lexpand(sr, fot = c(0, Inf), 
-               birth = bi_date, entry = "dg_date", exit = ex_date, 
-               status = status %in% 1:2, id = id)
-  
-  
-  x3 <- lexpand(sr, fot = c(0, Inf), 
-                birth = bi_date, entry = dg_date, exit = ex_date, 
-                status = status %in% 1:2, id = id)
-  
-  testthat::expect_identical(x, x2)
-  testthat::expect_identical(x, x3)
-})
-
-
-
-testthat::test_that("original total pyrs equals pyrs after splitting w/ large number of breaks", {
-  popEpi:::skip_normally()
-  x <- copy(sire)[dg_date < ex_date, ]
-  x[, fot := get.yrs(ex_date, year.length = "actual") - get.yrs(dg_date, year.length = "actual")]
-  totpyrs <- x[, sum(fot)]
-  
-  x <- lexpand(sire, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks=list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014)))
-  setDT(x)
-  totpyrs_split <- x[, sum(lex.dur)]
-  
-  testthat::expect_equal(totpyrs, totpyrs_split, tolerance = 1e-05)
-})
-
-
-
-testthat::test_that("pp not added to data if pp = FALSE but pop.haz is", {
-  x <- lexpand(sire[dg_date < ex_date, ][0:100], 
-               birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks=list(fot=0:5), 
-               pophaz=data.table(popEpi::popmort), 
-               pp = FALSE)
-  testthat::expect_equal(intersect(names(x), c("pp", "pop.haz")),  "pop.haz")
-  testthat::expect_true(!any(is.na(x$pop.haz)))
-})
-
-
-
-testthat::test_that("lexpand produces the same results with internal/external dropping", {
-  popEpi:::skip_normally()
-  x <- lexpand(sire[dg_date < ex_date, ], 
-               birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks=list(fot=0:5), pophaz=data.table(popEpi::popmort), 
-               pp = TRUE, drop = TRUE)
-  x2 <-lexpand(sire[dg_date < ex_date, ], 
-               birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks=list(fot=0:5), pophaz=data.table(popEpi::popmort), 
-               pp = TRUE, drop = FALSE)
-  x2 <-popEpi:::intelliDrop(x2, breaks = list(fot=0:5), dropNegDur = TRUE)
-  setDT(x)
-  setDT(x2)
-  popEpi:::doTestBarrage(dt1 = x, dt2 = x2, allScales = c("fot", "per", "age"))
-})
-
-
-testthat::test_that("lexpanding with aggre.type = 'unique' works", {
-  popEpi:::skip_normally()
-  
-  BL <- list(fot = 0:5, age = seq(0,100, 5))
-  ag1 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = BL, status = status,
-                 birth = bi_date, entry = dg_date, exit = ex_date)
-  setDT(ag1)
-  ag1 <- ag1[, list(pyrs = sum(lex.dur), from0to1 = sum(lex.Xst == 1L)), 
-           keyby = list(fot = popEpi:::cutLow(fot, BL$fot), 
-                        age = popEpi:::cutLow(age, BL$age))]
-  ag2 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = BL, status = status,
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 aggre = list(fot, age), aggre.type = "unique")
-  setDT(ag2)
-  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
-  
-})
-
-testthat::test_that("lexpanding with aggre.type = 'cartesian' works; no time scales used", {
-  popEpi:::skip_normally()
-  
-  BL <- list(fot = c(0,Inf))
-  ag1 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = BL, status = status, entry.status = 0L,
-                 birth = bi_date, entry = dg_date, exit = ex_date)
-  setDT(ag1)
-  forceLexisDT(ag1, breaks = BL, allScales = c("fot", "per", "age"))
-  
-  e <- quote(list(sex = factor(sex, 0:1, c("m", "f")),
-            period = cut(get.yrs(dg_date), get.yrs(as.Date(paste0(seq(1970, 2015, 5), "-01-01"))))))
-  ag1[, c("sex", "period") := eval(e)]
-  ceejay <- do.call(CJ, lapply(ag1[, list(sex, period)], function(x) {if (is.factor(x)) levels(x) else unique(x)}))
-  setkey(ceejay, sex, period); setkey(ag1, sex, period)
-  ag1 <- ag1[ceejay, list(pyrs = sum(lex.dur), 
-                          from0to1 = sum(lex.Xst == 1L)), by = .EACHI]
-  ag1[is.na(pyrs), pyrs := 0]
-  ag1[is.na(from0to1), from0to1 := 0]
-  
-  ag2 <- lexpand(sire[dg_date < ex_date, ],
-                 breaks = BL, 
-                 status = status, entry.status = 0L,
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 aggre = list(sex = factor(sex, 0:1, c("m", "f")),
-                              period = cut(get.yrs(dg_date), get.yrs(as.Date(paste0(seq(1970, 2015, 5), "-01-01"))))), 
-                 aggre.type = "cartesian")
-  
-  setDT(ag2)
-  setkeyv(ag1, c("sex", "period"))
-  setkeyv(ag2, c("sex", "period"))
-  testthat::expect_equal(sum(ag1$pyrs), sum(ag2$pyrs))
-  testthat::expect_equal(sum(ag1$from0to1), sum(ag2$from0to1))
-  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
-  
-})
-
-testthat::test_that("lexpanding with aggre.type = 'cartesian' works; only time scales used", {
-  popEpi:::skip_normally()
-  
-  BL <- list(fot = 0:5, age = seq(0,100, 5))
-  ag1 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = BL, status = status, entry.status = 0L,
-                 birth = bi_date, entry = dg_date, exit = ex_date)
-  setDT(ag1)
-  forceLexisDT(ag1, breaks = BL, allScales = c("fot", "per", "age"))
-  
-  ag3 <- aggre(ag1, by = list(fot, age), type = "cartesian")
-  setDT(ag3)
-  
-  ag4 <- aggre(ag1, by = list(fot, age), type = "unique")
-  setDT(ag4)
-  
-  ag1[, `:=`(fot = try2int(popEpi:::cutLow(fot, c(BL$fot, Inf))), 
-             age = try2int(popEpi:::cutLow(age, c(BL$age, Inf))))]
-  ceejay <- do.call(CJ, lapply(BL, function(x) x[-length(x)]))
-  setkey(ceejay, fot, age); setkey(ag1, fot, age)
-  ag1 <- ag1[ceejay, list(pyrs = sum(lex.dur), 
-                          from0to1 = sum(lex.Xst == 1L)), by = .EACHI]
-  ag1[is.na(pyrs), pyrs := 0]
-  ag1[is.na(from0to1), from0to1 := 0]
-  
-  ag2 <- lexpand(sire[dg_date < ex_date, ],
-                 breaks = list(fot = 0:5, age = seq(0,100, 5)), 
-                 status = status, entry.status = 0L,
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 aggre = list(fot, age), aggre.type = "cartesian")
-  
-  setDT(ag2)
-  setkeyv(ag1, c("fot", "age"))
-  setkeyv(ag2, c("fot", "age"))
-  setkeyv(ag3, c("fot", "age"))
-  testthat::expect_equal(sum(ag1$pyrs), sum(ag3$pyrs))
-  testthat::expect_equal(sum(ag1$from0to1), sum(ag3$from0to1))
-  testthat::expect_equal(ag1$pyrs, ag3$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag3$from0to1)
-  
-  testthat::expect_equal(sum(ag1$pyrs), sum(ag2$pyrs))
-  testthat::expect_equal(sum(ag1$from0to1), sum(ag2$from0to1))
-  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
-  
-})
-
-
-testthat::test_that("lexpanding and aggregating to years works", {
-  ag1 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = list(per=2000:2014), status = status,
-                 birth = bi_date, entry = dg_date, exit = ex_date)
-  setDT(ag1)
-  ag1[, `:=`(per = as.integer(popEpi:::cutLow(per, 2000:2014)))]
-  ag1 <- ag1[, list(pyrs = sum(lex.dur), from0to1 = sum(lex.Xst == 1L)), keyby = per]
-  
-  ag2 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = list(per = 2000:2014), status = status,
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 aggre = list(per), aggre.type = "unique")
-  setDT(ag2)
-  ag3 <- lexpand(sire[dg_date < ex_date, ], 
-                 breaks = list(per = 2000:2014, age = c(seq(0,100,5),Inf), fot = c(0:10, Inf)), 
-                 status = status,
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 aggre = list(y = per), aggre.type = "unique")
-  setDT(ag3)
-  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
-  testthat::expect_equal(ag1$pyrs, ag3$pyrs)
-  testthat::expect_equal(ag1$from0to1, ag3$from0to1)
-  
-})
-
-# Aggre check (to totpyrs) -----------------------------------------------------
-
-testthat::test_that("lexpand aggre produces correct results", {
-  popEpi:::skip_normally()
-  x <- copy(sire)[dg_date < ex_date, ]
-  x[, fot := get.yrs(ex_date, year.length = "actual") - get.yrs(dg_date, year.length = "actual")]
-  totpyrs <- x[, sum(fot)]
-  counts <- x[, .N, by = .(status)]
-  
-  x <- lexpand(sire[dg_date < ex_date, ], 
-               birth = bi_date, entry = dg_date, exit = ex_date,
-               breaks=list(fot=c(0,5,10,50,Inf), age=c(seq(0,85,5),Inf), per = 1993:2013), 
-               status=status, aggre = list(fot, age, per))
-  setDT(x)
-  row_length <- x[,list( length(unique(age)), length(unique(per)), length(unique(fot)))]
-  
-  testthat::expect_equal( x[,sum(pyrs)], totpyrs, tolerance = 0.001)
-  testthat::expect_equal( x[,sum(from0to0)], counts[1,N])
-  testthat::expect_equal( x[,sum(from0to1)], counts[2,N])
-  testthat::expect_equal( x[,sum(from0to2)], counts[3,N]) 
-  #expect_equal( prod(row_length), x[,.N]) 
-})
-
-testthat::test_that('lexpand aggre: multistate column names correct', {
-  
-  x <- lexpand(sire[dg_date < ex_date, ][0:100], 
-               birth = bi_date, entry = dg_date, exit = ex_date,
-               breaks=list(fot=c(0,5,10,50,Inf), age=c(seq(0,85,5),Inf), 
-                           per = 1993:2013), 
-               status=status, aggre = list(fot, age, per))
-  
-  testthat::expect_equal(intersect(names(x), c('from0to0','from0to1','from0to2')), 
-               c('from0to0','from0to1','from0to2'))  
-})
-
-
-# overlapping time lines --------------------------------------------------
-
-testthat::test_that('lexpansion w/ overlapping = TRUE/FALSE produces double/undoubled pyrs', {
-  popEpi:::skip_normally()
-  
-  sire2 <- copy(sire)[dg_date < ex_date, ][1:100]
-  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
-  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
-  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
-  sire2[, id := 1:.N]
-  sire2 <- sire2[rep(1:.N, each=2)]
-  
-  sire2[seq(2,.N, by=2), dg_yrs := (ex_yrs + dg_yrs)/2L]
-  sire2[, dg_age := dg_yrs-bi_yrs]
-  
-  x <- lexpand(sire2, birth = "bi_yrs", entry = "bi_yrs", event="dg_yrs", 
-               exit = "ex_yrs", status="status", entry.status = 0L, id = "id", 
-               overlapping = TRUE)
-  setDT(x)
-  testthat::expect_equal(x[, sum(lex.dur), keyby=lex.id]$V1, sire2[, sum(ex_yrs-bi_yrs), keyby=id]$V1)  
-  
-  x <- lexpand(sire2, birth = "bi_yrs", entry = "bi_yrs", event="dg_yrs", 
-               exit = "ex_yrs", status="status", entry.status = 0L, id = "id", 
-               overlapping = FALSE)
-  setDT(x)
-  testthat::expect_equal(x[, sum(lex.dur), keyby=lex.id]$V1, 
-               sire2[!duplicated(id), sum(ex_yrs-bi_yrs), keyby=id]$V1)  
-})
-
-
-
-testthat::test_that("different specifications of time vars work with event defined and overlapping=FALSE", {
-  
-  dt <- data.table(bi_date = as.Date('1949-01-01'), 
-                   dg_date = as.Date(paste0(1999:2000, "-01-01")), 
-                   start = as.Date("1997-01-01"),
-                   end = as.Date('2002-01-01'), 
-                   status = c(1,2), id=1)
-  
-  ## birth -> entry -> event -> exit
-  x1 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = start, exit = end, event = dg_date, 
-                id = id, overlapping = FALSE,  entry.status = 0, status = status,
-                merge = FALSE)
-  testthat::expect_equal(x1$lex.dur, c(2,1,2))
-  testthat::expect_equal(x1$age, c(48,50,51))
-  testthat::expect_equal(x1$lex.Cst, 0:2)
-  testthat::expect_equal(x1$lex.Xst, c(1,2,2))
-  
-  ## birth -> entry = event -> exit
-  testthat::expect_error(
-    lexpand(data = dt, subset = NULL, 
-            birth = bi_date, entry = dg_date, exit = end, event = dg_date,
-            id = id, overlapping = FALSE,  entry.status = 0, status = status,
-            merge = FALSE), 
-    regexp = paste0("some rows have simultaneous 'entry' and 'event', ",
-                    "which is not supported with overlapping = FALSE; ",
-                    "perhaps separate them by one day?")
-    )
-  
-  ## birth = entry -> event -> exit
-  x3 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = bi_date, exit = end, event = dg_date,
-                id = id, overlapping = FALSE, entry.status = 0, status = status,
-                merge = FALSE)
-  testthat::expect_equal(x3$lex.dur, c(50,1,2))
-  testthat::expect_equal(x3$age, c(0,50,51))
-  testthat::expect_equal(x3$lex.Cst, 0:2)
-  testthat::expect_equal(x3$lex.Xst, c(1,2,2))
-  
-  ## birth -> entry -> event = exit
-  testthat::expect_error(
-    lexpand(data = dt, subset = NULL, 
-            birth = bi_date, entry = dg_date, exit = end, event = end,
-            id = id, overlapping = FALSE,  entry.status = 0, status = status,
-            merge = FALSE), 
-    regexp = paste0("subject\\(s\\) defined by lex.id had several rows ",
-                    "where 'event' time had the same value, which is not ",
-                    "supported with overlapping = FALSE; perhaps separate ",
-                    "them by one day?")
-  )
-  
-  ## birth = entry -> event -> exit
-  x6 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = bi_date, exit = end, event = dg_date,
-                id = id, overlapping = FALSE,  entry.status = 0, status = status,
-                merge = FALSE)
-  testthat::expect_equal(x6$lex.dur, c(50,1,2))
-  testthat::expect_equal(x6$age, c(0,50,51))
-  testthat::expect_equal(x6$lex.Cst, 0:2)
-  testthat::expect_equal(x6$lex.Xst, c(1,2,2))
-  
-})
-
-
-testthat::test_that("lexpand drops persons outside breaks window correctly", {
-  popEpi:::skip_normally()
-  
-  dt <- data.table(bi_date = as.Date('1949-01-01'), 
-                   dg_date = as.Date(paste0(2000, "-01-01")), 
-                   start = as.Date("1997-01-01"),
-                   end = as.Date('2002-01-01'), 
-                   status = c(2), id=1)
-  
-  ## by age
-  x1 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = start, exit = end, event = dg_date, 
-                id = id, overlapping = FALSE,  entry.status = 0, status = status,
-                merge = FALSE, breaks = list(age = 50:55))
-  testthat::expect_equal(x1$age, c(50, 51, 52))
-  
-  ## by period
-  x1 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = start, exit = end, event = dg_date, 
-                id = id, overlapping = FALSE,  entry.status = 0, status = status,
-                merge = FALSE, breaks = list(per = 2000:2005))
-  testthat::expect_equal(x1$per, c(2000, 2001))
-  
-  
-  ## by fot
-  x1 <- lexpand(data = dt, subset = NULL, 
-                birth = bi_date, entry = start, exit = end, event = dg_date, 
-                id = id, overlapping = FALSE,  entry.status = 0, status = status,
-                merge = FALSE, breaks = list(fot = 2:5))
-  testthat::expect_equal(x1$fot, c(2, 3, 4))
-})
-
+testthat::context("lexpand sanity checks")
+
+
+testthat::test_that("lexpand arguments can be passed as symbol, expression, character name of variable, and symbol of a character variable", {
+  popEpi:::skip_normally()
+  sr <- copy(sire)[dg_date < ex_date, ][1:100,]
+  sr[, id := as.character(1:.N)]
+  
+  x <- lexpand(sr, fot = c(0, Inf), 
+               birth = "bi_date", entry = dg_date, exit = "ex_date", 
+               status = status %in% 1:2, id = "id")
+  
+  x2 <- lexpand(sr, fot = c(0, Inf), 
+               birth = bi_date, entry = "dg_date", exit = ex_date, 
+               status = status %in% 1:2, id = id)
+  
+  
+  x3 <- lexpand(sr, fot = c(0, Inf), 
+                birth = bi_date, entry = dg_date, exit = ex_date, 
+                status = status %in% 1:2, id = id)
+  
+  testthat::expect_identical(x, x2)
+  testthat::expect_identical(x, x3)
+})
+
+
+
+testthat::test_that("original total pyrs equals pyrs after splitting w/ large number of breaks", {
+  popEpi:::skip_normally()
+  x <- copy(sire)[dg_date < ex_date, ]
+  x[, fot := get.yrs(ex_date, year.length = "actual") - get.yrs(dg_date, year.length = "actual")]
+  totpyrs <- x[, sum(fot)]
+  
+  x <- lexpand(sire, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks=list(fot= seq(0,20,1/12), age= c(0:100, Inf), per= c(1960:2014)))
+  setDT(x)
+  totpyrs_split <- x[, sum(lex.dur)]
+  
+  testthat::expect_equal(totpyrs, totpyrs_split, tolerance = 1e-05)
+})
+
+
+
+testthat::test_that("pp not added to data if pp = FALSE but pop.haz is", {
+  x <- lexpand(sire[dg_date < ex_date, ][0:100], 
+               birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks=list(fot=0:5), 
+               pophaz=data.table(popEpi::popmort), 
+               pp = FALSE)
+  testthat::expect_equal(intersect(names(x), c("pp", "pop.haz")),  "pop.haz")
+  testthat::expect_true(!any(is.na(x$pop.haz)))
+})
+
+
+
+testthat::test_that("lexpand produces the same results with internal/external dropping", {
+  popEpi:::skip_normally()
+  x <- lexpand(sire[dg_date < ex_date, ], 
+               birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks=list(fot=0:5), pophaz=data.table(popEpi::popmort), 
+               pp = TRUE, drop = TRUE)
+  x2 <-lexpand(sire[dg_date < ex_date, ], 
+               birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks=list(fot=0:5), pophaz=data.table(popEpi::popmort), 
+               pp = TRUE, drop = FALSE)
+  x2 <-popEpi:::intelliDrop(x2, breaks = list(fot=0:5), dropNegDur = TRUE)
+  setDT(x)
+  setDT(x2)
+  popEpi:::doTestBarrage(dt1 = x, dt2 = x2, allScales = c("fot", "per", "age"))
+})
+
+
+testthat::test_that("lexpanding with aggre.type = 'unique' works", {
+  popEpi:::skip_normally()
+  
+  BL <- list(fot = 0:5, age = seq(0,100, 5))
+  ag1 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = BL, status = status,
+                 birth = bi_date, entry = dg_date, exit = ex_date)
+  setDT(ag1)
+  ag1 <- ag1[, list(pyrs = sum(lex.dur), from0to1 = sum(lex.Xst == 1L)), 
+           keyby = list(fot = popEpi:::cutLow(fot, BL$fot), 
+                        age = popEpi:::cutLow(age, BL$age))]
+  ag2 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = BL, status = status,
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 aggre = list(fot, age), aggre.type = "unique")
+  setDT(ag2)
+  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
+  
+})
+
+testthat::test_that("lexpanding with aggre.type = 'cartesian' works; no time scales used", {
+  popEpi:::skip_normally()
+  
+  BL <- list(fot = c(0,Inf))
+  ag1 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = BL, status = status, entry.status = 0L,
+                 birth = bi_date, entry = dg_date, exit = ex_date)
+  setDT(ag1)
+  forceLexisDT(ag1, breaks = BL, allScales = c("fot", "per", "age"))
+  
+  e <- quote(list(sex = factor(sex, 0:1, c("m", "f")),
+            period = cut(get.yrs(dg_date), get.yrs(as.Date(paste0(seq(1970, 2015, 5), "-01-01"))))))
+  ag1[, c("sex", "period") := eval(e)]
+  ceejay <- do.call(CJ, lapply(ag1[, list(sex, period)], function(x) {if (is.factor(x)) levels(x) else unique(x)}))
+  setkey(ceejay, sex, period); setkey(ag1, sex, period)
+  ag1 <- ag1[ceejay, list(pyrs = sum(lex.dur), 
+                          from0to1 = sum(lex.Xst == 1L)), by = .EACHI]
+  ag1[is.na(pyrs), pyrs := 0]
+  ag1[is.na(from0to1), from0to1 := 0]
+  
+  ag2 <- lexpand(sire[dg_date < ex_date, ],
+                 breaks = BL, 
+                 status = status, entry.status = 0L,
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 aggre = list(sex = factor(sex, 0:1, c("m", "f")),
+                              period = cut(get.yrs(dg_date), get.yrs(as.Date(paste0(seq(1970, 2015, 5), "-01-01"))))), 
+                 aggre.type = "cartesian")
+  
+  setDT(ag2)
+  setkeyv(ag1, c("sex", "period"))
+  setkeyv(ag2, c("sex", "period"))
+  testthat::expect_equal(sum(ag1$pyrs), sum(ag2$pyrs))
+  testthat::expect_equal(sum(ag1$from0to1), sum(ag2$from0to1))
+  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
+  
+})
+
+testthat::test_that("lexpanding with aggre.type = 'cartesian' works; only time scales used", {
+  popEpi:::skip_normally()
+  
+  BL <- list(fot = 0:5, age = seq(0,100, 5))
+  ag1 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = BL, status = status, entry.status = 0L,
+                 birth = bi_date, entry = dg_date, exit = ex_date)
+  setDT(ag1)
+  forceLexisDT(ag1, breaks = BL, allScales = c("fot", "per", "age"))
+  
+  ag3 <- aggre(ag1, by = list(fot, age), type = "cartesian")
+  setDT(ag3)
+  
+  ag4 <- aggre(ag1, by = list(fot, age), type = "unique")
+  setDT(ag4)
+  
+  ag1[, `:=`(fot = try2int(popEpi:::cutLow(fot, c(BL$fot, Inf))), 
+             age = try2int(popEpi:::cutLow(age, c(BL$age, Inf))))]
+  ceejay <- do.call(CJ, lapply(BL, function(x) x[-length(x)]))
+  setkey(ceejay, fot, age); setkey(ag1, fot, age)
+  ag1 <- ag1[ceejay, list(pyrs = sum(lex.dur), 
+                          from0to1 = sum(lex.Xst == 1L)), by = .EACHI]
+  ag1[is.na(pyrs), pyrs := 0]
+  ag1[is.na(from0to1), from0to1 := 0]
+  
+  ag2 <- lexpand(sire[dg_date < ex_date, ],
+                 breaks = list(fot = 0:5, age = seq(0,100, 5)), 
+                 status = status, entry.status = 0L,
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 aggre = list(fot, age), aggre.type = "cartesian")
+  
+  setDT(ag2)
+  setkeyv(ag1, c("fot", "age"))
+  setkeyv(ag2, c("fot", "age"))
+  setkeyv(ag3, c("fot", "age"))
+  testthat::expect_equal(sum(ag1$pyrs), sum(ag3$pyrs))
+  testthat::expect_equal(sum(ag1$from0to1), sum(ag3$from0to1))
+  testthat::expect_equal(ag1$pyrs, ag3$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag3$from0to1)
+  
+  testthat::expect_equal(sum(ag1$pyrs), sum(ag2$pyrs))
+  testthat::expect_equal(sum(ag1$from0to1), sum(ag2$from0to1))
+  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
+  
+})
+
+
+testthat::test_that("lexpanding and aggregating to years works", {
+  ag1 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = list(per=2000:2014), status = status,
+                 birth = bi_date, entry = dg_date, exit = ex_date)
+  setDT(ag1)
+  ag1[, `:=`(per = as.integer(popEpi:::cutLow(per, 2000:2014)))]
+  ag1 <- ag1[, list(pyrs = sum(lex.dur), from0to1 = sum(lex.Xst == 1L)), keyby = per]
+  
+  ag2 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = list(per = 2000:2014), status = status,
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 aggre = list(per), aggre.type = "unique")
+  setDT(ag2)
+  ag3 <- lexpand(sire[dg_date < ex_date, ], 
+                 breaks = list(per = 2000:2014, age = c(seq(0,100,5),Inf), fot = c(0:10, Inf)), 
+                 status = status,
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 aggre = list(y = per), aggre.type = "unique")
+  setDT(ag3)
+  testthat::expect_equal(ag1$pyrs, ag2$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag2$from0to1)
+  testthat::expect_equal(ag1$pyrs, ag3$pyrs)
+  testthat::expect_equal(ag1$from0to1, ag3$from0to1)
+  
+})
+
+# Aggre check (to totpyrs) -----------------------------------------------------
+
+testthat::test_that("lexpand aggre produces correct results", {
+  popEpi:::skip_normally()
+  x <- copy(sire)[dg_date < ex_date, ]
+  x[, fot := get.yrs(ex_date, year.length = "actual") - get.yrs(dg_date, year.length = "actual")]
+  totpyrs <- x[, sum(fot)]
+  counts <- x[, .N, by = .(status)]
+  
+  x <- lexpand(sire[dg_date < ex_date, ], 
+               birth = bi_date, entry = dg_date, exit = ex_date,
+               breaks=list(fot=c(0,5,10,50,Inf), age=c(seq(0,85,5),Inf), per = 1993:2013), 
+               status=status, aggre = list(fot, age, per))
+  setDT(x)
+  row_length <- x[,list( length(unique(age)), length(unique(per)), length(unique(fot)))]
+  
+  testthat::expect_equal( x[,sum(pyrs)], totpyrs, tolerance = 0.001)
+  testthat::expect_equal( x[,sum(from0to0)], counts[1,N])
+  testthat::expect_equal( x[,sum(from0to1)], counts[2,N])
+  testthat::expect_equal( x[,sum(from0to2)], counts[3,N]) 
+  #expect_equal( prod(row_length), x[,.N]) 
+})
+
+testthat::test_that('lexpand aggre: multistate column names correct', {
+  
+  x <- lexpand(sire[dg_date < ex_date, ][0:100], 
+               birth = bi_date, entry = dg_date, exit = ex_date,
+               breaks=list(fot=c(0,5,10,50,Inf), age=c(seq(0,85,5),Inf), 
+                           per = 1993:2013), 
+               status=status, aggre = list(fot, age, per))
+  
+  testthat::expect_equal(intersect(names(x), c('from0to0','from0to1','from0to2')), 
+               c('from0to0','from0to1','from0to2'))  
+})
+
+
+# overlapping time lines --------------------------------------------------
+
+testthat::test_that('lexpansion w/ overlapping = TRUE/FALSE produces double/undoubled pyrs', {
+  popEpi:::skip_normally()
+  
+  sire2 <- copy(sire)[dg_date < ex_date, ][1:100]
+  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
+  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
+  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
+  sire2[, id := 1:.N]
+  sire2 <- sire2[rep(1:.N, each=2)]
+  
+  sire2[seq(2,.N, by=2), dg_yrs := (ex_yrs + dg_yrs)/2L]
+  sire2[, dg_age := dg_yrs-bi_yrs]
+  
+  x <- lexpand(sire2, birth = "bi_yrs", entry = "bi_yrs", event="dg_yrs", 
+               exit = "ex_yrs", status="status", entry.status = 0L, id = "id", 
+               overlapping = TRUE)
+  setDT(x)
+  testthat::expect_equal(x[, sum(lex.dur), keyby=lex.id]$V1, sire2[, sum(ex_yrs-bi_yrs), keyby=id]$V1)  
+  
+  x <- lexpand(sire2, birth = "bi_yrs", entry = "bi_yrs", event="dg_yrs", 
+               exit = "ex_yrs", status="status", entry.status = 0L, id = "id", 
+               overlapping = FALSE)
+  setDT(x)
+  testthat::expect_equal(x[, sum(lex.dur), keyby=lex.id]$V1, 
+               sire2[!duplicated(id), sum(ex_yrs-bi_yrs), keyby=id]$V1)  
+})
+
+
+
+testthat::test_that("different specifications of time vars work with event defined and overlapping=FALSE", {
+  
+  dt <- data.table(bi_date = as.Date('1949-01-01'), 
+                   dg_date = as.Date(paste0(1999:2000, "-01-01")), 
+                   start = as.Date("1997-01-01"),
+                   end = as.Date('2002-01-01'), 
+                   status = c(1,2), id=1)
+  
+  ## birth -> entry -> event -> exit
+  x1 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = start, exit = end, event = dg_date, 
+                id = id, overlapping = FALSE,  entry.status = 0, status = status,
+                merge = FALSE)
+  testthat::expect_equal(x1$lex.dur, c(2,1,2))
+  testthat::expect_equal(x1$age, c(48,50,51))
+  testthat::expect_equal(x1$lex.Cst, 0:2)
+  testthat::expect_equal(x1$lex.Xst, c(1,2,2))
+  
+  ## birth -> entry = event -> exit
+  testthat::expect_error(
+    lexpand(data = dt, subset = NULL, 
+            birth = bi_date, entry = dg_date, exit = end, event = dg_date,
+            id = id, overlapping = FALSE,  entry.status = 0, status = status,
+            merge = FALSE), 
+    regexp = paste0("some rows have simultaneous 'entry' and 'event', ",
+                    "which is not supported with overlapping = FALSE; ",
+                    "perhaps separate them by one day?")
+    )
+  
+  ## birth = entry -> event -> exit
+  x3 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = bi_date, exit = end, event = dg_date,
+                id = id, overlapping = FALSE, entry.status = 0, status = status,
+                merge = FALSE)
+  testthat::expect_equal(x3$lex.dur, c(50,1,2))
+  testthat::expect_equal(x3$age, c(0,50,51))
+  testthat::expect_equal(x3$lex.Cst, 0:2)
+  testthat::expect_equal(x3$lex.Xst, c(1,2,2))
+  
+  ## birth -> entry -> event = exit
+  testthat::expect_error(
+    lexpand(data = dt, subset = NULL, 
+            birth = bi_date, entry = dg_date, exit = end, event = end,
+            id = id, overlapping = FALSE,  entry.status = 0, status = status,
+            merge = FALSE), 
+    regexp = paste0("subject\\(s\\) defined by lex.id had several rows ",
+                    "where 'event' time had the same value, which is not ",
+                    "supported with overlapping = FALSE; perhaps separate ",
+                    "them by one day?")
+  )
+  
+  ## birth = entry -> event -> exit
+  x6 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = bi_date, exit = end, event = dg_date,
+                id = id, overlapping = FALSE,  entry.status = 0, status = status,
+                merge = FALSE)
+  testthat::expect_equal(x6$lex.dur, c(50,1,2))
+  testthat::expect_equal(x6$age, c(0,50,51))
+  testthat::expect_equal(x6$lex.Cst, 0:2)
+  testthat::expect_equal(x6$lex.Xst, c(1,2,2))
+  
+})
+
+
+testthat::test_that("lexpand drops persons outside breaks window correctly", {
+  popEpi:::skip_normally()
+  
+  dt <- data.table(bi_date = as.Date('1949-01-01'), 
+                   dg_date = as.Date(paste0(2000, "-01-01")), 
+                   start = as.Date("1997-01-01"),
+                   end = as.Date('2002-01-01'), 
+                   status = c(2), id=1)
+  
+  ## by age
+  x1 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = start, exit = end, event = dg_date, 
+                id = id, overlapping = FALSE,  entry.status = 0, status = status,
+                merge = FALSE, breaks = list(age = 50:55))
+  testthat::expect_equal(x1$age, c(50, 51, 52))
+  
+  ## by period
+  x1 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = start, exit = end, event = dg_date, 
+                id = id, overlapping = FALSE,  entry.status = 0, status = status,
+                merge = FALSE, breaks = list(per = 2000:2005))
+  testthat::expect_equal(x1$per, c(2000, 2001))
+  
+  
+  ## by fot
+  x1 <- lexpand(data = dt, subset = NULL, 
+                birth = bi_date, entry = start, exit = end, event = dg_date, 
+                id = id, overlapping = FALSE,  entry.status = 0, status = status,
+                merge = FALSE, breaks = list(fot = 2:5))
+  testthat::expect_equal(x1$fot, c(2, 3, 4))
+})
+
diff --git a/tests/testthat/test_prevtab.R b/tests/testthat/test_prevtab.R
index 9f690d2..4f73b0e 100644
--- a/tests/testthat/test_prevtab.R
+++ b/tests/testthat/test_prevtab.R
@@ -1,22 +1,22 @@
-# context("prevtab")
-# 
-# 
-# 
-# test_that("prevtab produces intended results", {
-#   
-#   
-#   mp <- data.table(popEpi::meanpop_fi)
-#   setnames(mp, c("year", "agegroup"), c("per", "age"))
-#   x <- lexpand(popEpi::sire, 
-#                birth = bi_date, entry = dg_date, exit = ex_date,
-#                status = status %in% 1:2)
-#   pt1 <- prevtab(per ~ sex + fot, data = x, meanpop = mp,
-#                  breaks = list(per = 2009:2014, fot = c(0,1,5,10,Inf)),
-#                  adjust = "agegroup", weights = "internal")
-#   pt2 <- prevtab(per ~ sex + work, data = x, meanpop = mp,
-#                  breaks = list(per = 2009:2014, work = c(0,1,5,10,Inf)),
-#                  adjust = "agegroup", weights = "internal")
-#   
-#   
-# })
-
+# context("prevtab")
+# 
+# 
+# 
+# test_that("prevtab produces intended results", {
+#   
+#   
+#   mp <- data.table(popEpi::meanpop_fi)
+#   setnames(mp, c("year", "agegroup"), c("per", "age"))
+#   x <- lexpand(popEpi::sire, 
+#                birth = bi_date, entry = dg_date, exit = ex_date,
+#                status = status %in% 1:2)
+#   pt1 <- prevtab(per ~ sex + fot, data = x, meanpop = mp,
+#                  breaks = list(per = 2009:2014, fot = c(0,1,5,10,Inf)),
+#                  adjust = "agegroup", weights = "internal")
+#   pt2 <- prevtab(per ~ sex + work, data = x, meanpop = mp,
+#                  breaks = list(per = 2009:2014, work = c(0,1,5,10,Inf)),
+#                  adjust = "agegroup", weights = "internal")
+#   
+#   
+# })
+
diff --git a/tests/testthat/test_rate.R b/tests/testthat/test_rate.R
index 8bac005..74b3c51 100644
--- a/tests/testthat/test_rate.R
+++ b/tests/testthat/test_rate.R
@@ -1,302 +1,302 @@
-testthat::context('rate')
-
-# simultate test data
-set.seed(5)
-p18 <- data.table( OBS=round(runif(36)*10), PYRS=round(runif(36)*10000), AGEGROUP=1:18, COV = rep(c(1,2), each = 18))
-set.seed(5)
-p20 <- data.table( OBS=round(runif(20)*10), PYRS=round(runif(20)*10000), AGEGROUP=1:20, COV = rep(c(1,2), each = 20))
-set.seed(5)
-p101 <- data.table( OBS=round(runif(101)*10), PYRS=round(runif(101)*10000), AGEGROUP=1:101, COV = rep(c(1,2), each = 101))
-p18b <- data.table(p18)
-setnames(p18b, c('OBS','PYRS','AGEGROUP'), c('obs','pyrs','agegroup'))
-wv <- c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)
-
-
-testthat::test_that("rate works with different weights", {
-  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18))
-  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
-  w3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'cohort')
-  w4 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL)
-  w5 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL)
-  
-  testthat::expect_equal(sum(w1$PYRS), p18[,sum(PYRS)])
-  testthat::expect_equal(sum(w2$OBS), p20[,sum(OBS)])
-  testthat::expect_equal(sum(w3$OBS), p20[,sum(OBS)])
-  testthat::expect_equal(w4$rate, p20[,list(sum(OBS)/sum(PYRS)), by ='COV'][, V1])
-  testthat::expect_equal(w5$rate, p20[,list(sum(OBS)/sum(PYRS))][, V1])
-  testthat::expect_is(w1, 'rate')
-  testthat::expect_is(w2, 'rate')
-  testthat::expect_is(w2, 'data.frame')
-  if(getOption("popEpi.datatable")) {
-    testthat::expect_is(w2, 'data.table')
-  }
-})
-
-testthat::test_that("rate CIs and SEs are correct", {
-  
-  ci <-data.table(agegroup = c(1,2,3,4),
-                  obs=c(4,13,8,7),
-                  pyrs=c(96,237,105,32),
-                  rate=c(4.2,5.5,7.6,21.9),
-                  std.pop=c(2773,2556,1113,184))
-  
-  a1 <- ci[,sum(obs/pyrs*std.pop)/sum(std.pop)] # oikea estimaatti
-  a2 <- ci[,sqrt(sum(std.pop^2*((obs/pyrs)*(1-obs/pyrs))/pyrs))/sum(std.pop)] # myös oikea tulos  
-  
-  ci0 <- rate(data = ci, obs = 'obs', pyrs = 'pyrs', print = NULL, adjust = 'agegroup', weights = list(agegroup = c(2773,2556,1113,184)))
-  testthat::expect_equal(ci0[,SE.rate.adj], a2, tolerance=0.0005) # test
-  testthat::expect_equal(ci0[,rate.adj],a1)
-
-  # another...
-  ci <-data.table(agegroup = c(1,2,3,4),
-                  obs=c(4,13,8,7),
-                  pyrs=c(960,2370,1050,320),
-                  rate=c(4.2,5.5,7.6,21.9),
-                  std.pop=c(2773,2556,1113,184))
-  
-  a1 <- ci[,sum(obs/pyrs*std.pop)/sum(std.pop)] # oikea estimaatti
-  a2 <- ci[,sqrt(sum(std.pop^2*((obs/pyrs)*(1-obs/pyrs))/pyrs))/sum(std.pop)] # myös oikea tulos  
-  
-  ci0 <- rate(data = ci, obs = 'obs', pyrs = 'pyrs', print = NULL, adjust = 'agegroup', weights = list(agegroup = c(2773,2556,1113,184)))
-  testthat::expect_equal(c(ci0$rate.adj.lo, ci0$rate.adj.hi),c(a1 - a2*1.96, a1 + a2*1.96), tolerance = 0.0006)
-  testthat::expect_gt(ci0[,SE.rate.adj], a2) # WHYY?!
-  testthat::expect_equal(ci0[,rate.adj],a1) # ok
-})
-
-testthat::test_that("makeWeightsDT works in rate", {
-  set.seed(5)
-  p18 <- data.table( OBS=round(runif(36)*10), PYRS=round(runif(36)*10000), AGEGROUP=1:18, COV = rep(c(1,2), each = 18))
-  op <- c('OBS',  'PYRS')
-  mw1 <- makeWeightsDT(p18, adjust = substitute(factor(AGEGROUP, 1:18, 1:18)), weights = wv, print = NULL, values = list(op))
-  mw2 <- makeWeightsDT(p18, adjust = NULL, weights = NULL, print = substitute(COV), values = list(op))
-  
-  attlist <- attr( mw1, 'makeWeightsDT')
-  
-  testthat::expect_equal(c(attlist$adVars, attlist$vaVars, 'weights'), c('factor','OBS','PYRS','weights') )
-  
-  testthat::expect_equal(mw1[,as.character(factor)], as.character(1:18))
-  testthat::expect_equal(mw2[,OBS], p18[,sum(OBS), by=COV][,V1])
-})
-
-
-testthat::test_that("names dont cause problems", {
-  w1 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'nordic')
-  w2 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'cohort')
-  w3 <- rate(data = p18b, obs =  obs,  pyrs =  pyrs,  print =  COV,  adjust =  agegroup,  weights = 'cohort')
-  w5 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'world_1966_18of5')
-  w6 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
-  w7a <- rate(data = p20, obs = OBS, pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
-  w7b <- rate(data = p20, obs = OBS, pyrs = PYRS, print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
-  
-  wr <- p18b[,list(obs=sum(obs),pyrs =sum(pyrs)), by ='COV']
-  
-  testthat::expect_equal(w2$obs, w1$obs)
-  testthat::expect_equal(w2$pyrs, w1$pyrs)
-  testthat::expect_equal(wr$obs, w1$obs)
-  testthat::expect_equal(wr$pyrs, w1$pyrs)
-  testthat::expect_equal(w2, w3, check.attributes = FALSE)
-})
-
-
-testthat::test_that("rate works with different weights an subset", {
-  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = c(1:18))
-  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = c(1:18), subset = COV==1)
-  s1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18), subset = COV==1)
-  s2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP',  weights = 'world_2000_20of5', subset = COV == 2)
-  s3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'cohort', subset = COV==1)
-  s4 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL, subset = AGEGROUP != 1)
-  s5 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL, subset = COV == 1)
-  
-  
-  testthat::expect_equal(sum(s1$PYRS), p18[COV==1,sum(PYRS)])
-  testthat::expect_equal(sum(s2$OBS), p20[COV==2,sum(OBS)])
-  testthat::expect_equal(sum(s3$OBS), p20[COV==1,sum(OBS)])
-  testthat::expect_equal(s4$rate, p20[AGEGROUP!= 1,list(sum(OBS)/sum(PYRS)), by ='COV'][, V1])
-  testthat::expect_equal(s5$rate, p20[COV==1,list(sum(OBS)/sum(PYRS))][, V1])
-  testthat::expect_is(s3, 'rate')
-})
-
-testthat::test_that("rate works with different weights and syntaxies", {
-  
-  
-  wv <- c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)
-  
-  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL)
-  s1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1))
-  s2 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = wv)
-  #s3 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = NULL) ???
-  testthat::expect_is(s1, 'rate')
-  
-  testthat::expect_equal(sum(s1$PYRS), p18[,sum(PYRS)])
-  testthat::expect_equal(sum(s2$OBS), p18[,sum(OBS)])
-  #expect_equal(sum(s3$OBS), p18[,sum(OBS)])
-  
-  
-  # non working syntaxes
-  testthat::expect_error( 
-    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = list(1:18))
-  ) # non named list
-  testthat::expect_error(
-    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = list(c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)))
-  )
-  testthat::expect_error(
-    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = list(factor(AGEGROUP, 1:18, 1:18), COV), weights = list( wv,  c(0.5,0.5)))
-  ) # a list length of 2
-  testthat::expect_error(
-    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = c('AGEGROUP','COV'), weights = list(COV = c(.5,.5), AGEGROUP = 1:18))
-  ) # duplicated names
-  
-  # working
-  s10 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
-              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18)), 
-              weights = list(agegr = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)))
-  
-  s11 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
-              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18), gender = factor(COV, 1:2, 1:2)), 
-              weights = list(agegr = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1), gender = c(1,1)))
-  
-  s12 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
-              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18)), 
-              weights = list(agegr = wv))
-  
-  
-  s13 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = list(AGEGROUP = 1:18))
-  s14a <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = c('AGEGROUP','COV'), weights = list(AGEGROUP = 1:18, COV = c(.5,.5))) # SAMA1
-  s14b <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = c('AGEGROUP','COV'), weights = list(COV = c(.5,.5), AGEGROUP = 1:18)) # SAMA1
-  
-  s16a <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5') #
-  s16b <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = list(AGEGROUP), weights = 'world_2000_20of5') #
-  s16c <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = AGEGROUP, weights = 'world_2000_20of5') #
-  
-  
-  
-  # Works
-  s21 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = (1:18), subset = COV==1)
-  s22 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5', subset = COV == 2)
-  s23 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL, subset = AGEGROUP != 1)
-  s24 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL, subset = COV == 1)
-  
-  ## internal weights
-  s23a <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'internal')
-  s23b <- rate(data = p20, obs = 'OBS', pyrs = PYRS,   print = 'COV', adjust = 'AGEGROUP', weights = 'internal')
-  s23c <- rate(data = p20, obs = OBS, pyrs = PYRS,   print = COV, adjust = AGEGROUP, weights = 'internal')
-  s24a <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = list(AGEGROUP, COV), weights = "internal")
-  s24b <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = c('AGEGROUP', 'COV'), weights = "internal")
-  
-  ## update and getCall
-  s2B <- update(s0, adjust = factor(AGEGROUP, 1:18, 1:18), weights = wv)
-  testthat::expect_equal(data.table(s2B), data.table(s2))
-  
-})
-
-
-testthat::test_that("in rate levels of adjust are printable", {
-  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = list(COV, AG=findInterval(AGEGROUP, c(0,8,20))), adjust ='AGEGROUP', weights = 'cohort')
-  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = list(COV, ag=findInterval(AGEGROUP, c(1,8,20))), adjust ='AGEGROUP', weights = 'world_2000_20of5')
-  testthat::expect_equal(w1[,.N]  ,4)
-})
-
-testthat::test_that("rate.plot doesnt throw an error", {
-  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = list(COV, AG=findInterval(AGEGROUP, c(0,8,20))), adjust ='AGEGROUP', weights = 'cohort')
-  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = list(COV), adjust ='AGEGROUP', weights = 'world_2000_20of5')
-  w3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS')
-  testthat::expect_silent(plot(w1))
-  testthat::expect_silent(plot(w2, col = 4:5, conf.int = FALSE))
-  testthat::expect_silent(plot(w3, eps = 0.01))
-})
-
-
-testthat::test_that("rate works with missing values", {
-  p18c <- copy(p18)
-  p18c[c(1,6), PYRS := NA]
-  testthat::expect_warning( rate(data = p18c, obs = 'OBS', pyrs = 'PYRS', adjust = 'AGEGROUP', weights = 1:18), "Data contains 2 NA values." )
-})
-
-testthat::test_that("rate standard error and CIS works", {
-  se <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = 'AGEGROUP', weights = 1:18)
-  
-  testthat::expect_equal(se[, SE.rate.adj] , se[, SE.rate], tolerance = 0.00002)
-  testthat::expect_lt(se[, rate.adj.lo], se[, rate.adj])
-  testthat::expect_lt(se[, rate.adj], se[, rate.adj.hi])
-  testthat::expect_lt(se[, rate.lo], se[, rate])
-  testthat::expect_lt(se[, rate], se[, rate.hi])
-  testthat::expect_equal(se[, sqrt(OBS/PYRS^2)], se[, SE.rate])
-  testthat::expect_equal( se[, rate.adj - SE.rate.adj*2], se[,rate.adj.lo], tolerance = 0.00002)
-  testthat::expect_equal( se[, rate.adj + SE.rate.adj*2], se[,rate.adj.hi], tolerance = 0.00002)
-})
-
-
-
-testthat::test_that("rate_ratio works", {
-
-  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18))
-  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
-  w3 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV')
-  cru <- round((88/109818)/(78/110421),3)
-  
-  
-  # rate obj, one line
-  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = FALSE, SE.method = FALSE),
-                rate_ratio( w1[1], w1[2], crude = TRUE, SE.method = FALSE))  # oikein, crude ignoorattu
-  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = FALSE, SE.method = TRUE)$rate_ratio, 
-                round(w1[1, rate.adj]/ w1[2, rate.adj],3))
-  
-  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = TRUE, SE.method = TRUE),
-                data.frame(rate_ratio = 1.134, lower=1.087, upper=1.182))
-  
-  # rate obj crude
-  suppressMessages(
-    testthat::expect_equal( rate_ratio( w3[1], w3[2], crude = TRUE, SE.method = TRUE),  # oikein, hiljaa
-                  rate_ratio( w3[1], w3[2], crude = FALSE, SE.method = TRUE))  # oikein, message
-  )
-  # rate + SE
-  x <- c(w1[1, rate], w1[1, SE.rate])
-  y <- c(w1[2, rate], w1[2, SE.rate])
-  
-  testthat::expect_error( rate_ratio( x, y, crude = FALSE, SE.method = FALSE) ) # error, käyttäjän vastuu?
-  testthat::expect_error( rate_ratio( x, y, crude = TRUE,  SE.method = FALSE) )  # error, käyttäjän vastuu?
-  testthat::expect_equal( rate_ratio( x, y, crude = FALSE, SE.method = TRUE), 
-                rate_ratio( x, y, crude = TRUE,  SE.method = TRUE))
-  
-  # Obs + Pyrs
-  testthat::expect_warning( rate_ratio( c(88,78), c(109818,110421), crude = FALSE, SE.method = TRUE) ) # oikein, message
-  a <- rate_ratio( c(88,78), c(109818,110421), crude = TRUE, SE.method = FALSE)$rate_ratio
-  testthat::expect_equal(a, (c(88,78)/c(109818,110421))[1]/(c(88,78)/c(109818,110421))[2], tolerance = 0.001 )
-  
-  # multi row rate objects
-  a0 <- rate_ratio( w1, w2, crude = FALSE, SE.method = TRUE) 
-  b0 <- rate_ratio( w1[1], w2[1], crude = FALSE, SE.method = TRUE) 
-  c0 <- rate_ratio( w1[2], w2[2], crude = FALSE, SE.method = TRUE) 
-  
-  testthat::expect_equal(a0[1,], b0)
-  testthat::expect_equal(c(a0[2,]), c(c0))
-
-  #rate_ratio( w1, w2[1], crude = FALSE, SE.method = TRUE) # väärä, pitäisi tulla data.frame
-  #rate_ratio( w1, c(0.0005,0.00002), crude = FALSE, SE.method = TRUE) # väärä, pitäisi tulla data.frame
- 
-})
-
-
-testthat::test_that("warnings and stops works properly", {
-  testthat::expect_error(
-    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = list(1:18, 2:19))
-    )
-  testthat::expect_error( stdr.weights(c('wold00_1','world66_5')) )
-  testthat::expect_error( stdr.weights(c('wold00_20of5')) )
-})
-
-
-
-testthat::test_that("stdr.weights returns correct datasets", {
-  al <- c('world_1966_18of5','europe','nordic',
-          "world_2000_18of5","world_2000_101of1", 
-          "world_2000_20of5")
-  le <- c(18,18,18,18,101,20)
-  testthat::expect_equal( stdr.weights(al[1])[,.N], le[1])
-  testthat::expect_equal( stdr.weights(al[2])[,.N], le[2])
-  testthat::expect_equal( stdr.weights(al[3])[,.N], le[3])
-  testthat::expect_equal( stdr.weights(al[4])[,.N], le[4])
-  testthat::expect_equal( stdr.weights(al[5])[,.N], le[5])
-  testthat::expect_equal( stdr.weights(al[6])[,.N], le[6])
-})
-
+testthat::context('rate')
+
+# simultate test data
+set.seed(5)
+p18 <- data.table( OBS=round(runif(36)*10), PYRS=round(runif(36)*10000), AGEGROUP=1:18, COV = rep(c(1,2), each = 18))
+set.seed(5)
+p20 <- data.table( OBS=round(runif(20)*10), PYRS=round(runif(20)*10000), AGEGROUP=1:20, COV = rep(c(1,2), each = 20))
+set.seed(5)
+p101 <- data.table( OBS=round(runif(101)*10), PYRS=round(runif(101)*10000), AGEGROUP=1:101, COV = rep(c(1,2), each = 101))
+p18b <- data.table(p18)
+setnames(p18b, c('OBS','PYRS','AGEGROUP'), c('obs','pyrs','agegroup'))
+wv <- c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)
+
+
+testthat::test_that("rate works with different weights", {
+  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18))
+  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
+  w3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'cohort')
+  w4 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL)
+  w5 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL)
+  
+  testthat::expect_equal(sum(w1$PYRS), p18[,sum(PYRS)])
+  testthat::expect_equal(sum(w2$OBS), p20[,sum(OBS)])
+  testthat::expect_equal(sum(w3$OBS), p20[,sum(OBS)])
+  testthat::expect_equal(w4$rate, p20[,list(sum(OBS)/sum(PYRS)), by ='COV'][, V1])
+  testthat::expect_equal(w5$rate, p20[,list(sum(OBS)/sum(PYRS))][, V1])
+  testthat::expect_is(w1, 'rate')
+  testthat::expect_is(w2, 'rate')
+  testthat::expect_is(w2, 'data.frame')
+  if(getOption("popEpi.datatable")) {
+    testthat::expect_is(w2, 'data.table')
+  }
+})
+
+testthat::test_that("rate CIs and SEs are correct", {
+  
+  ci <-data.table(agegroup = c(1,2,3,4),
+                  obs=c(4,13,8,7),
+                  pyrs=c(96,237,105,32),
+                  rate=c(4.2,5.5,7.6,21.9),
+                  std.pop=c(2773,2556,1113,184))
+  
+  a1 <- ci[,sum(obs/pyrs*std.pop)/sum(std.pop)] # oikea estimaatti
+  a2 <- ci[,sqrt(sum(std.pop^2*((obs/pyrs)*(1-obs/pyrs))/pyrs))/sum(std.pop)] # myös oikea tulos  
+  
+  ci0 <- rate(data = ci, obs = 'obs', pyrs = 'pyrs', print = NULL, adjust = 'agegroup', weights = list(agegroup = c(2773,2556,1113,184)))
+  testthat::expect_equal(ci0[,SE.rate.adj], a2, tolerance=0.0005) # test
+  testthat::expect_equal(ci0[,rate.adj],a1)
+
+  # another...
+  ci <-data.table(agegroup = c(1,2,3,4),
+                  obs=c(4,13,8,7),
+                  pyrs=c(960,2370,1050,320),
+                  rate=c(4.2,5.5,7.6,21.9),
+                  std.pop=c(2773,2556,1113,184))
+  
+  a1 <- ci[,sum(obs/pyrs*std.pop)/sum(std.pop)] # oikea estimaatti
+  a2 <- ci[,sqrt(sum(std.pop^2*((obs/pyrs)*(1-obs/pyrs))/pyrs))/sum(std.pop)] # myös oikea tulos  
+  
+  ci0 <- rate(data = ci, obs = 'obs', pyrs = 'pyrs', print = NULL, adjust = 'agegroup', weights = list(agegroup = c(2773,2556,1113,184)))
+  testthat::expect_equal(c(ci0$rate.adj.lo, ci0$rate.adj.hi),c(a1 - a2*1.96, a1 + a2*1.96), tolerance = 0.0006)
+  testthat::expect_gt(ci0[,SE.rate.adj], a2) # WHYY?!
+  testthat::expect_equal(ci0[,rate.adj],a1) # ok
+})
+
+testthat::test_that("makeWeightsDT works in rate", {
+  set.seed(5)
+  p18 <- data.table( OBS=round(runif(36)*10), PYRS=round(runif(36)*10000), AGEGROUP=1:18, COV = rep(c(1,2), each = 18))
+  op <- c('OBS',  'PYRS')
+  mw1 <- makeWeightsDT(p18, adjust = substitute(factor(AGEGROUP, 1:18, 1:18)), weights = wv, print = NULL, values = list(op))
+  mw2 <- makeWeightsDT(p18, adjust = NULL, weights = NULL, print = substitute(COV), values = list(op))
+  
+  attlist <- attr( mw1, 'makeWeightsDT')
+  
+  testthat::expect_equal(c(attlist$adVars, attlist$vaVars, 'weights'), c('factor','OBS','PYRS','weights') )
+  
+  testthat::expect_equal(mw1[,as.character(factor)], as.character(1:18))
+  testthat::expect_equal(mw2[,OBS], p18[,sum(OBS), by=COV][,V1])
+})
+
+
+testthat::test_that("names dont cause problems", {
+  w1 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'nordic')
+  w2 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'cohort')
+  w3 <- rate(data = p18b, obs =  obs,  pyrs =  pyrs,  print =  COV,  adjust =  agegroup,  weights = 'cohort')
+  w5 <- rate(data = p18b, obs = 'obs', pyrs = 'pyrs', print = 'COV', adjust = 'agegroup', weights = 'world_1966_18of5')
+  w6 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
+  w7a <- rate(data = p20, obs = OBS, pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
+  w7b <- rate(data = p20, obs = OBS, pyrs = PYRS, print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
+  
+  wr <- p18b[,list(obs=sum(obs),pyrs =sum(pyrs)), by ='COV']
+  
+  testthat::expect_equal(w2$obs, w1$obs)
+  testthat::expect_equal(w2$pyrs, w1$pyrs)
+  testthat::expect_equal(wr$obs, w1$obs)
+  testthat::expect_equal(wr$pyrs, w1$pyrs)
+  testthat::expect_equal(w2, w3, check.attributes = FALSE)
+})
+
+
+testthat::test_that("rate works with different weights an subset", {
+  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = c(1:18))
+  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = c(1:18), subset = COV==1)
+  s1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18), subset = COV==1)
+  s2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP',  weights = 'world_2000_20of5', subset = COV == 2)
+  s3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'cohort', subset = COV==1)
+  s4 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL, subset = AGEGROUP != 1)
+  s5 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL, subset = COV == 1)
+  
+  
+  testthat::expect_equal(sum(s1$PYRS), p18[COV==1,sum(PYRS)])
+  testthat::expect_equal(sum(s2$OBS), p20[COV==2,sum(OBS)])
+  testthat::expect_equal(sum(s3$OBS), p20[COV==1,sum(OBS)])
+  testthat::expect_equal(s4$rate, p20[AGEGROUP!= 1,list(sum(OBS)/sum(PYRS)), by ='COV'][, V1])
+  testthat::expect_equal(s5$rate, p20[COV==1,list(sum(OBS)/sum(PYRS))][, V1])
+  testthat::expect_is(s3, 'rate')
+})
+
+testthat::test_that("rate works with different weights and syntaxies", {
+  
+  
+  wv <- c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)
+  
+  s0 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL)
+  s1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1))
+  s2 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = wv)
+  #s3 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = NULL) ???
+  testthat::expect_is(s1, 'rate')
+  
+  testthat::expect_equal(sum(s1$PYRS), p18[,sum(PYRS)])
+  testthat::expect_equal(sum(s2$OBS), p18[,sum(OBS)])
+  #expect_equal(sum(s3$OBS), p18[,sum(OBS)])
+  
+  
+  # non working syntaxes
+  testthat::expect_error( 
+    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = list(1:18))
+  ) # non named list
+  testthat::expect_error(
+    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = factor(AGEGROUP, 1:18, 1:18), weights = list(c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)))
+  )
+  testthat::expect_error(
+    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = list(factor(AGEGROUP, 1:18, 1:18), COV), weights = list( wv,  c(0.5,0.5)))
+  ) # a list length of 2
+  testthat::expect_error(
+    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = c('AGEGROUP','COV'), weights = list(COV = c(.5,.5), AGEGROUP = 1:18))
+  ) # duplicated names
+  
+  # working
+  s10 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
+              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18)), 
+              weights = list(agegr = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1)))
+  
+  s11 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
+              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18), gender = factor(COV, 1:2, 1:2)), 
+              weights = list(agegr = c(.1,.1,.1,.2,.2,.2,.2,.3,.3,.4,.5,.5,.5,.4,.4,.3,.2,.1), gender = c(1,1)))
+  
+  s12 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', 
+              adjust = list(agegr = factor(AGEGROUP, 1:18, 1:18)), 
+              weights = list(agegr = wv))
+  
+  
+  s13 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = 'AGEGROUP', weights = list(AGEGROUP = 1:18))
+  s14a <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = c('AGEGROUP','COV'), weights = list(AGEGROUP = 1:18, COV = c(.5,.5))) # SAMA1
+  s14b <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = c('AGEGROUP','COV'), weights = list(COV = c(.5,.5), AGEGROUP = 1:18)) # SAMA1
+  
+  s16a <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5') #
+  s16b <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = list(AGEGROUP), weights = 'world_2000_20of5') #
+  s16c <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = AGEGROUP, weights = 'world_2000_20of5') #
+  
+  
+  
+  # Works
+  s21 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = (1:18), subset = COV==1)
+  s22 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5', subset = COV == 2)
+  s23 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = NULL, weights = NULL, subset = AGEGROUP != 1)
+  s24 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = NULL, adjust = NULL, weights = NULL, subset = COV == 1)
+  
+  ## internal weights
+  s23a <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'internal')
+  s23b <- rate(data = p20, obs = 'OBS', pyrs = PYRS,   print = 'COV', adjust = 'AGEGROUP', weights = 'internal')
+  s23c <- rate(data = p20, obs = OBS, pyrs = PYRS,   print = COV, adjust = AGEGROUP, weights = 'internal')
+  s24a <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = list(AGEGROUP, COV), weights = "internal")
+  s24b <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = c('AGEGROUP', 'COV'), weights = "internal")
+  
+  ## update and getCall
+  s2B <- update(s0, adjust = factor(AGEGROUP, 1:18, 1:18), weights = wv)
+  testthat::expect_equal(data.table(s2B), data.table(s2))
+  
+})
+
+
+testthat::test_that("in rate levels of adjust are printable", {
+  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = list(COV, AG=findInterval(AGEGROUP, c(0,8,20))), adjust ='AGEGROUP', weights = 'cohort')
+  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = list(COV, ag=findInterval(AGEGROUP, c(1,8,20))), adjust ='AGEGROUP', weights = 'world_2000_20of5')
+  testthat::expect_equal(w1[,.N]  ,4)
+})
+
+testthat::test_that("rate.plot doesnt throw an error", {
+  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = list(COV, AG=findInterval(AGEGROUP, c(0,8,20))), adjust ='AGEGROUP', weights = 'cohort')
+  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = list(COV), adjust ='AGEGROUP', weights = 'world_2000_20of5')
+  w3 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS')
+  testthat::expect_silent(plot(w1))
+  testthat::expect_silent(plot(w2, col = 4:5, conf.int = FALSE))
+  testthat::expect_silent(plot(w3, eps = 0.01))
+})
+
+
+testthat::test_that("rate works with missing values", {
+  p18c <- copy(p18)
+  p18c[c(1,6), PYRS := NA]
+  testthat::expect_warning( rate(data = p18c, obs = 'OBS', pyrs = 'PYRS', adjust = 'AGEGROUP', weights = 1:18), "Data contains 2 NA values." )
+})
+
+testthat::test_that("rate standard error and CIS works", {
+  se <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', adjust = 'AGEGROUP', weights = 1:18)
+  
+  testthat::expect_equal(se[, SE.rate.adj] , se[, SE.rate], tolerance = 0.00002)
+  testthat::expect_lt(se[, rate.adj.lo], se[, rate.adj])
+  testthat::expect_lt(se[, rate.adj], se[, rate.adj.hi])
+  testthat::expect_lt(se[, rate.lo], se[, rate])
+  testthat::expect_lt(se[, rate], se[, rate.hi])
+  testthat::expect_equal(se[, sqrt(OBS/PYRS^2)], se[, SE.rate])
+  testthat::expect_equal( se[, rate.adj - SE.rate.adj*2], se[,rate.adj.lo], tolerance = 0.00002)
+  testthat::expect_equal( se[, rate.adj + SE.rate.adj*2], se[,rate.adj.hi], tolerance = 0.00002)
+})
+
+
+
+testthat::test_that("rate_ratio works", {
+
+  w1 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = c(1:18))
+  w2 <- rate(data = p20, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = 'world_2000_20of5')
+  w3 <- rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV')
+  cru <- round((88/109818)/(78/110421),3)
+  
+  
+  # rate obj, one line
+  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = FALSE, SE.method = FALSE),
+                rate_ratio( w1[1], w1[2], crude = TRUE, SE.method = FALSE))  # oikein, crude ignoorattu
+  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = FALSE, SE.method = TRUE)$rate_ratio, 
+                round(w1[1, rate.adj]/ w1[2, rate.adj],3))
+  
+  testthat::expect_equal( rate_ratio( w1[1], w1[2], crude = TRUE, SE.method = TRUE),
+                data.frame(rate_ratio = 1.134, lower=1.087, upper=1.182))
+  
+  # rate obj crude
+  suppressMessages(
+    testthat::expect_equal( rate_ratio( w3[1], w3[2], crude = TRUE, SE.method = TRUE),  # oikein, hiljaa
+                  rate_ratio( w3[1], w3[2], crude = FALSE, SE.method = TRUE))  # oikein, message
+  )
+  # rate + SE
+  x <- c(w1[1, rate], w1[1, SE.rate])
+  y <- c(w1[2, rate], w1[2, SE.rate])
+  
+  testthat::expect_error( rate_ratio( x, y, crude = FALSE, SE.method = FALSE) ) # error, käyttäjän vastuu?
+  testthat::expect_error( rate_ratio( x, y, crude = TRUE,  SE.method = FALSE) )  # error, käyttäjän vastuu?
+  testthat::expect_equal( rate_ratio( x, y, crude = FALSE, SE.method = TRUE), 
+                rate_ratio( x, y, crude = TRUE,  SE.method = TRUE))
+  
+  # Obs + Pyrs
+  testthat::expect_warning( rate_ratio( c(88,78), c(109818,110421), crude = FALSE, SE.method = TRUE) ) # oikein, message
+  a <- rate_ratio( c(88,78), c(109818,110421), crude = TRUE, SE.method = FALSE)$rate_ratio
+  testthat::expect_equal(a, (c(88,78)/c(109818,110421))[1]/(c(88,78)/c(109818,110421))[2], tolerance = 0.001 )
+  
+  # multi row rate objects
+  a0 <- rate_ratio( w1, w2, crude = FALSE, SE.method = TRUE) 
+  b0 <- rate_ratio( w1[1], w2[1], crude = FALSE, SE.method = TRUE) 
+  c0 <- rate_ratio( w1[2], w2[2], crude = FALSE, SE.method = TRUE) 
+  
+  testthat::expect_equal(a0[1,], b0)
+  testthat::expect_equal(c(a0[2,]), c(c0))
+
+  #rate_ratio( w1, w2[1], crude = FALSE, SE.method = TRUE) # väärä, pitäisi tulla data.frame
+  #rate_ratio( w1, c(0.0005,0.00002), crude = FALSE, SE.method = TRUE) # väärä, pitäisi tulla data.frame
+ 
+})
+
+
+testthat::test_that("warnings and stops works properly", {
+  testthat::expect_error(
+    rate(data = p18, obs = 'OBS', pyrs = 'PYRS', print = 'COV', adjust = 'AGEGROUP', weights = list(1:18, 2:19))
+    )
+  testthat::expect_error( stdr.weights(c('wold00_1','world66_5')) )
+  testthat::expect_error( stdr.weights(c('wold00_20of5')) )
+})
+
+
+
+testthat::test_that("stdr.weights returns correct datasets", {
+  al <- c('world_1966_18of5','europe','nordic',
+          "world_2000_18of5","world_2000_101of1", 
+          "world_2000_20of5")
+  le <- c(18,18,18,18,101,20)
+  testthat::expect_equal( stdr.weights(al[1])[,.N], le[1])
+  testthat::expect_equal( stdr.weights(al[2])[,.N], le[2])
+  testthat::expect_equal( stdr.weights(al[3])[,.N], le[3])
+  testthat::expect_equal( stdr.weights(al[4])[,.N], le[4])
+  testthat::expect_equal( stdr.weights(al[5])[,.N], le[5])
+  testthat::expect_equal( stdr.weights(al[6])[,.N], le[6])
+})
+
diff --git a/tests/testthat/test_relpois_mean_curve.R b/tests/testthat/test_relpois_mean_curve.R
index c7fe07d..405a236 100644
--- a/tests/testthat/test_relpois_mean_curve.R
+++ b/tests/testthat/test_relpois_mean_curve.R
@@ -1,68 +1,68 @@
-testthat::context("test rpcurve vs. survtab congruence")
-
-testthat::test_that("rpcurve and survtab e2 are approximately congruent", {
-  popEpi:::skip_normally()
-  
-  sire2 <- popEpi::sire[dg_date < ex_date, ]
-  sire2[, "agegr" := cut(dg_age, breaks = c(0,45,70,Inf))]
-  
-  fb <- c(0,3/12,6/12,1:8,10)
-  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               fot=fb,  pophaz=data.table(popEpi::popmort))
-  data.table::setDT(x)
-  data.table::setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  rp <- relpois(x, formula = lex.Xst %in% 1:2 ~ -1 + FOT+agegr)
-  mc <- rpcurve(rp)
-  
-  x[, pop.haz := NULL]
-  pm <- data.table(popEpi::popmort)
-  setnames(pm, c("year", "agegroup"), c("per", "age"))
-  w <- as.numeric(table(x$agegr))
-  st <- survtab(Surv(fot, lex.Xst) ~ adjust(agegr), 
-                pophaz = pm, weights = w,
-                relsurv.method = "e2",
-                data= x, breaks = list(fot = seq(0, 10, 1/12)))
-  setDT(mc)
-  setDT(st)
-  
-  testthat::expect_equal(st[Tstop %in% fb]$r.e2.as, mc$est, tolerance=0.0136, scale=1L)
-  
-  ## added old results on 2016-03-19, 
-  ## ref = 4feb1ca37489737332cebf33d24550a9951a7630
-  old_res <- c(0.9253749, 0.8801775, 0.8123319, 0.7237591, 0.6679470, 
-               0.6315218, 0.6035761, 0.5826368, 0.5645760, 0.5519045, 0.5368186)
-  
-  testthat::expect_equal(old_res, mc$est, tolerance=1e-5, scale=1L)
-})
-
-
-
-# comparison with flexsurv; maybe not needed ----------------------------------
-# library(flexsurv)
-# sire2 <- lexpand(sire2, fot=c(0, 10), status)
-# sire2[, year := year(ex_date)]
-# sire2[, agegroup := as.integer(as.integer(ex_date-bi_date)/365.25)]
-# sire2[agegroup > 100, agegroup := 100L]
-# 
-# sire2 <- data.table:::merge.data.table(sire2, popmort, all.x=FALSE, all.y=FALSE, by=c("sex","year","agegroup"))
-# sire2[lex.Xst == 0, haz := 0] ## not really needed, nothing changes even when it works
-# 
-# ## spline model does not work with bhazard (nothing changes)
-# fl <- flexsurvspline(Surv(lex.dur, lex.Xst %in% 1:2) ~ agegr, data=sire2, k = 2, bhazard=sire2$haz)
-# ## this works
-# fl <- flexsurvreg(Surv(lex.dur, lex.Xst %in% 1:2) ~ agegr, data=sire2, dist="gengamma", bhazard=sire2$haz)
-# su <- summary.flexsurvreg(fl, newdata = sire2, ci = FALSE, t = fb, B = 0)
-# su <- rbindlist(su)
-# su <- su[, list(netsurv = mean(est)), by=time]
-# 
-# plot(netsurv~time, data=su, type="l")
-# lines(est~Tstop, data=mc, col="red")
-# # lines(lo~Tstop, data=mc, col="red")
-# # lines(hi~Tstop, data=mc, col="red")
-# lines(st, "r.e2.as", col="blue", conf.int=FALSE)
-
-
-
-
-
+testthat::context("test rpcurve vs. survtab congruence")
+
+testthat::test_that("rpcurve and survtab e2 are approximately congruent", {
+  popEpi:::skip_normally()
+  
+  sire2 <- popEpi::sire[dg_date < ex_date, ]
+  sire2[, "agegr" := cut(dg_age, breaks = c(0,45,70,Inf))]
+  
+  fb <- c(0,3/12,6/12,1:8,10)
+  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               fot=fb,  pophaz=data.table(popEpi::popmort))
+  data.table::setDT(x)
+  data.table::setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  rp <- relpois(x, formula = lex.Xst %in% 1:2 ~ -1 + FOT+agegr)
+  mc <- rpcurve(rp)
+  
+  x[, pop.haz := NULL]
+  pm <- data.table(popEpi::popmort)
+  setnames(pm, c("year", "agegroup"), c("per", "age"))
+  w <- as.numeric(table(x$agegr))
+  st <- survtab(Surv(fot, lex.Xst) ~ adjust(agegr), 
+                pophaz = pm, weights = w,
+                relsurv.method = "e2",
+                data= x, breaks = list(fot = seq(0, 10, 1/12)))
+  setDT(mc)
+  setDT(st)
+  
+  testthat::expect_equal(st[Tstop %in% fb]$r.e2.as, mc$est, tolerance=0.0136, scale=1L)
+  
+  ## added old results on 2016-03-19, 
+  ## ref = 4feb1ca37489737332cebf33d24550a9951a7630
+  old_res <- c(0.9253749, 0.8801775, 0.8123319, 0.7237591, 0.6679470, 
+               0.6315218, 0.6035761, 0.5826368, 0.5645760, 0.5519045, 0.5368186)
+  
+  testthat::expect_equal(old_res, mc$est, tolerance=1e-5, scale=1L)
+})
+
+
+
+# comparison with flexsurv; maybe not needed ----------------------------------
+# library(flexsurv)
+# sire2 <- lexpand(sire2, fot=c(0, 10), status)
+# sire2[, year := year(ex_date)]
+# sire2[, agegroup := as.integer(as.integer(ex_date-bi_date)/365.25)]
+# sire2[agegroup > 100, agegroup := 100L]
+# 
+# sire2 <- data.table:::merge.data.table(sire2, popmort, all.x=FALSE, all.y=FALSE, by=c("sex","year","agegroup"))
+# sire2[lex.Xst == 0, haz := 0] ## not really needed, nothing changes even when it works
+# 
+# ## spline model does not work with bhazard (nothing changes)
+# fl <- flexsurvspline(Surv(lex.dur, lex.Xst %in% 1:2) ~ agegr, data=sire2, k = 2, bhazard=sire2$haz)
+# ## this works
+# fl <- flexsurvreg(Surv(lex.dur, lex.Xst %in% 1:2) ~ agegr, data=sire2, dist="gengamma", bhazard=sire2$haz)
+# su <- summary.flexsurvreg(fl, newdata = sire2, ci = FALSE, t = fb, B = 0)
+# su <- rbindlist(su)
+# su <- su[, list(netsurv = mean(est)), by=time]
+# 
+# plot(netsurv~time, data=su, type="l")
+# lines(est~Tstop, data=mc, col="red")
+# # lines(lo~Tstop, data=mc, col="red")
+# # lines(hi~Tstop, data=mc, col="red")
+# lines(st, "r.e2.as", col="blue", conf.int=FALSE)
+
+
+
+
+
diff --git a/tests/testthat/test_sir.R b/tests/testthat/test_sir.R
index 57f2f17..9337a30 100644
--- a/tests/testthat/test_sir.R
+++ b/tests/testthat/test_sir.R
@@ -1,438 +1,438 @@
-testthat::context("SIR")
-
-
-testthat::test_that("SIR w/ coh=ref=popEpi::sire", {
-  ## don't skip on CRAN
-  sire2 <- copy(popEpi::sire)
-  sire2[, agegroup := cut(dg_age, breaks = c(0:17*5, Inf))]
-  levels(sire2$agegroup) <- 1:18
-  sire2[, ex_y := year(ex_date)]
-  sire2[, dur := as.integer(ex_date-dg_date)/365.242199]
-  ltre <- ltable(sire2[status != 0], c("ex_y", "agegroup"), 
-                 expr = list(obs=.N, pyrs=sum(dur)))
-  setDT(ltre)
-  ltre[is.na(pyrs), pyrs := 0]
-  
-  sibr2 <- copy(popEpi::sibr)
-  sibr2[, agegroup := cut(dg_age, breaks = c(0:17*5, Inf))]
-  levels(sibr2$agegroup) <- 1:18
-  sibr2[, ex_y := year(ex_date)]
-  sibr2[, dur := as.integer(ex_date-dg_date)/365.242199]
-  ltbr <- ltable(sibr2[status != 0], c("ex_y", "agegroup"), 
-                 expr = list(obs=.N, pyrs=sum(dur)))
-  setDT(ltbr)
-  ltbr[is.na(pyrs), pyrs := 0]
-
-  
-  
-  sl <- sir(coh.data=ltre, coh.obs="obs", coh.pyrs="pyrs",
-            ref.data=ltre, ref.obs="obs", ref.pyrs="pyrs", 
-            adjust= c("agegroup","ex_y"))
-  plot(sl)
-  ## SIR w/ coh=ref=popEpi::sire
-  ## don't skip on CRAN
-  testthat::expect_equal(sl$sir, 1)
-  testthat::expect_equal(sl$pyrs, 13783.81, tolerance=0.01)
-  testthat::expect_equal(sl$expected, 4595)
-  testthat::expect_equal(sl$observed, 4595)
-  
-  
-  sl <- sir(coh.data=ltre, coh.obs="obs", coh.pyrs="pyrs",
-            ref.data=ltbr, ref.obs="obs", ref.pyrs="pyrs",
-            adjust= c("agegroup","ex_y"))
-  
-  ## SIR w/ coh=ref=popEpi::sire"
-  testthat::expect_equal(sl$sir, 1.39, tolerance=0.01)
-  testthat::expect_equal(sl$pyrs, 13783.81, tolerance=0.01)
-  testthat::expect_equal(sl$expected, 3305.04, tolerance=0.01)
-  testthat::expect_equal(sl$observed, 4595)
-})
-
-
-
-# SIR mstate, subset + lexpand aggre ---------------------------------------
-
-# same model
-c <- lexpand( popEpi::sire[dg_date<ex_date,], status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1990:2013, age = 0:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-# different models
-c2 <- lexpand( popEpi::sire[dg_date<ex_date,], status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-               breaks = list(per = 1990:2010, age = 0:100, fot = c(0,10,20,Inf)), 
-               aggre = list(fot, agegroup = age, year = per, sex) )
-
-
-testthat::test_that("misc things work", {
-  ms1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', 
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
-  plot(ms1)
-  
-  # only one print level with no variation
-  testthat::expect_message( ms2 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type= 'profile',
-                              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                              adjust = c('agegroup','year','sex'), 
-                              print = list(sex)) )
-  testthat::expect_equal(attributes(ms2)$sir.meta$conf.type, 'profile')
-  
-  # coef and confint ---
-  testthat::expect_equal(as.numeric(coef(ms1)), ms1$sir)
-  testthat::expect_equal(data.table(confint(ms1)), data.table('2.5 %' = ms1$sir.lo, '97.5 %' = ms1$sir.hi), tolerance = 0.001)
-})
-
-
-testthat::test_that("confidence intervals an other options works", {
-  
-  # CI's
-  ci1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(FOT = fot, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
-  testthat::expect_message( ci2 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'wald',
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(FOT = fot, sex, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) )))
-  ci3 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'univariate',
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(FOT = fot, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
-  
-  testthat::expect_equal(attributes(ci1)$sir.meta$conf.type,'profile')
-  testthat::expect_equal(attributes(ci2)$sir.meta$conf.type,'wald')
-  testthat::expect_equal(attributes(ci3)$sir.meta$conf.type,'univariate')
-  
-  testthat::expect_message( sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'wald',
-                       ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                       adjust = c('agegroup','year','sex'), 
-                       print = list(sex )))
-                  
-  testthat::expect_message( ci4 <-  sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'profile',
-                               ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                               adjust = c('agegroup','year','sex'), 
-                               print = list(dg = cut(agegroup, c(0,5,10,20,40,75,120)) )) )
-  testthat::expect_equal(attributes(ci4)$sir.meta$conf.type,'wald')
-    
-  
-  testthat::expect_message( ci5 <- sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'profile',
-                              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                              adjust = c('agegroup','year','sex'), 
-                              print = list(sex=sex,FOT = fot, 
-                                           age.cat = cut(agegroup, c(0,50,75,120)), 
-                                           per = cut(year, c(1990,2010,2014), dig.lab = 8) )) )
-  # ear
-  ea <- sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'wald', EAR = TRUE,
-             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-             adjust = c('agegroup','year','sex'), conf.level = 0.95) 
-  testthat::expect_equal(ea[,EAR], round((1490-1482.13)/39905.92*1000,3))  
-  
-  # Update
-  ud1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(fot))
-  ud2 <- update(ud1, print = list(fot,  age.cat = cut(agegroup, c(0,75,120))))
-  
-  ud3 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
-              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-              adjust = c('agegroup','year','sex'), 
-              print = list(fot,  age.cat = cut(agegroup, c(0,75,120))))
-  testthat::expect_equal(ud2, ud3)
-})
-
-
-testthat::test_that("SIR works with multistate aggregated lexpand data", {
-  ## don't skip on CRAN
-
-  testthat::expect_warning(
-    se <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs', 
-               subset = year %in% 1990:2009,
-               ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-               adjust = c('agegroup','year','sex'), print =c('cause','fot'), mstate = 'cause')
-  )
-  
-  dummy.coh <- data.table(agegroup = 1:18, sex = 1, from0to1 = round(runif(18)), 
-                          from0to2 = round(runif(18)+0.4),
-                          pyrs = (rnorm(18)+100)*200)
-  dummy.ref <- data.table(agegroup = rep(1:18,times=2), sex = 1, obs = floor(runif(36)*100),
-                          categ2 = rep(c(1,2),each=18),pyrs = (rnorm(36)+100)*2000)
-  
-  sm <- sir( coh.data = dummy.coh, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs', 
-             ref.data = dummy.ref, ref.obs = 'obs', ref.pyrs = 'pyrs', mstate = 'categ2',
-             adjust = c('agegroup','sex','categ2'), print =c('categ2'))
-  
-  setkey(12345)
-  dummy.coh2 <- data.table(agegroup = rep(1:18,times=6), sex = rep(1:2, each= 18), obs = floor(runif(18*6)*100), level = rep(1:6,each=18),pyrs = (rnorm(36)+100)*2000)
-                           
-  sg <- sir( coh.data = dummy.coh2, coh.obs = c('obs'), coh.pyrs = 'pyrs', 
-             adjust = c('agegroup'), print =c('level'))
-  testthat::expect_equal(attributes(sg)$sir.meta$pooled[,sir], 1)
-  testthat::expect_equal(sg[,.N], 6)
-    
-
-  s1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', subset = year %in% 1990:2009,
-             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-             adjust = c('agegroup','year','sex'), print =c('fot'))
-  
-  s2 <- sir( coh.data = c2, coh.obs = c('from0to2'), coh.pyrs = 'pyrs',
-             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-             adjust = c('agegroup','year','sex'), print =c('fot'))
-  s12 <- rbind( cbind(cause=1L, data.table(s1)), 
-                cbind(cause=2L, data.table(s2)))
-  
-  # compare by hand calculated sir
-  
-  r <- merge(c2, data.table(popEpi::popmort), c('sex','agegroup','year'), all.x=TRUE)
-  setDT(r)
-  r[, exp := haz*pyrs]
-  est <- r[, list(observed=sum(from0to1, na.rm=TRUE),expected=sum(exp, na.rm=TRUE)), by=.(fot)]
-  est <- round(est,4)
-  
-  testthat::expect_is(object = se, class = 'sir')
-  testthat::expect_equivalent(se, s12)
-  setDT(s1)
-  testthat::expect_equivalent(s1[,1:3, with=FALSE], est)
-})
-
-
-
-# SIR utils ---------------------------------------------------------------
-
-testthat::test_that('Util functions work', {
-  testthat::expect_equal( poisson.ci(5,5)$rate, 1)
-  testthat::expect_equal(p.round(0.0000001), "< 0.001")
-})
-
-testthat::test_that("data.table subsettin works", {
-  
-  ss <- sir( coh.data = c2, coh.obs = c('from0to2'), coh.pyrs = 'pyrs',
-             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-             adjust = c('agegroup','year','sex'), print =c('fot'))
-  testthat::expect_equal(ss[1,.N], 1)
-  testthat::expect_equal(c(ss[,.(fot)]), list(fot = c(0,10)))
-})
-
-
-
-# sir splines -------------------------------------------------------------
-
-if (requireNamespace("splines")) {
-  testthat::test_that("SIR spline throws errors correctly", {
-    popEpi:::skip_normally()
-    
-    sp0 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
-                                          subset = year %in% 1990:2008,
-                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                                          adjust = c('agegroup','year','sex'), print = NULL,
-                                          spline=c('agegroup','year','fot') )))
-    sp1 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                                          subset = year %in% 1990:2008,
-                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                                          adjust = c('agegroup','year','sex'), print =c('cause'),
-                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
-    sp2 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                                          subset = year %in% 1990:2008,
-                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', dependent.spline=FALSE,
-                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
-                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
-    sp3 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                                          subset = year %in% 1990:2008,
-                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
-                                          mstate = 'cause', spline='agegroup') ))
-    
-    sp4 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                                          subset = year %in% 1990:2008,
-                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', reference.points = c(2000,4),
-                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
-                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
-    testthat::expect_is( object = sp1, class = 'sirspline')
-    testthat::expect_is( object = sp2, class = 'sirspline')
-    testthat::expect_is( object = sp3, class = 'sirspline')
-    testthat::expect_is( object = sp4, class = 'sirspline')
-  })  
-}
-
-testthat::test_that("print accepts a function and subset works", {
-  popEpi:::skip_normally()
-  testthat::expect_warning(
-    pl1 <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                subset = year %in% 1990:2008,
-                ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                adjust = c('agegroup','year','sex'), print = list(year.int = findInterval(year,c(1989,2000,2010))),
-                mstate = 'cause')
-  )
-  testthat::expect_warning(
-    pl2 <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
-                subset = year %in% 1990:2008,
-                ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
-                adjust = c('agegroup','year','sex'), print = list(year.int = findInterval(year,c(1989,2000,2010))),
-                mstate = 'cause')
-  )
-  setDT(pl1)
-  testthat::expect_equal( pl1[,year.int], 1:2)
-  testthat::expect_is(object=pl2, 'sir')
-})
-
-
-
-testthat::test_that("sir_ratio", {
-  ## don't skip on CRAN
-  dt1 <- data.table(obs = rep(c(5,7), 10),
-                    pyrs = rep(c(250,300,350,400), 5),
-                    var = 1:20)
-  
-  dt2 <- data.table(obs = rep(c(10,13), 10),
-                    pyrs = rep(c(250,300,350,400), 5),
-                    var = 1:20)
-  
-  Ref <- data.table(obs = rep(c(50,70,80,100), 5),
-                    pyrs = rep(c(2500,3000,3500,4000), 5),
-                    var = 1:20)
-  
-  
-  s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
-            ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
-            adjust = var)
-  
-  
-  s2 <- sir(coh.data = dt2, coh.obs = obs, coh.pyrs = pyrs, 
-            ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
-            adjust = var)
-  
-  
-  ## SIR w/ coh=ref=popEpi::sire
-  ## don't skip on CRAN
-  # Test the example from the book (statistics with confidence):
-  testthat::expect_equal( sir_ratio(x = c(64, 45.6), y = c(25,23.7), digits=2)[2:3], c(lower=0.83, upper=2.21))
-  # other tests
-  testthat::expect_message(sir_ratio(s1, s2, digits = 2, alternative = 'less', conf.level = 0.95, type = 'asymptotic'))
-  testthat::expect_equal(sir_ratio(s1, s2), sir_ratio(c(120,150), c(230,150)), tolerance=0.001)
-  testthat::expect_equal(sir_ratio(s1, c(230,150)), sir_ratio(c(120,150), s2), tolerance=0.001)
-  testthat::expect_equal(sir_ratio(s1, s2)[1], c(sir_ratio =round((120/150)/(230/150),3)), tolerance=0.001)
-})
-
-
-testthat::test_that("sir_exp and sir_ag and sir_lexis are working", {
-  ## don't skip on CRAN
-  # from0to1 ei oo vakio
-  
-  BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-  suppressMessages(
-    # aggregated with breaks
-    x1 <- lexpand(sire, breaks = BL, status = status != 0, 
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 pophaz=popmort,
-                 aggre=list(sex, period = per, surv.int = fot))
-  )
-  suppressMessages(
-    # no aggreate but breaks
-    x2 <- lexpand(sire, breaks = BL, status = status != 0, 
-                 birth = bi_date, entry = dg_date, exit = ex_date,
-                 pophaz=popmort)
-  )
-
-  suppressMessages(
-    x3 <- lexpand(sire, status = status != 0,
-                  birth = bi_date, entry = dg_date, exit = ex_date,
-                  pophaz=popmort)
-  )
-  suppressMessages(
-    x4 <- lexpand(sire, status = status != 0,
-                  birth = bi_date, entry = bi_date, exit = ex_date,
-                  pophaz=popmort)
-  )
-  # library(Epi)
-  # # no aggreate but breaks
-  # suppressMessages(
-  #   x5 <- Lexis(entry = list(FUT = 0, AGE = 0, CAL = get.yrs(bi_date)), 
-  #              exit = list(CAL = get.yrs(ex_date)), 
-  #              data = sire[sire$dg_date < sire$ex_date, ],
-  #              exit.status = factor(status, levels = 0:2, 
-  #                                   labels = c("alive", "canD", "othD")), 
-  #              merge = TRUE)
-  #   )
-  
-   
-   
-  # AGGRE
-  testthat::expect_message( a1 <- sir_ag(x1, obs = 'from0to1', conf.type = 'profile') )
-  testthat::expect_message( a0 <- sir_ag(x1, obs = from0to1, conf.type = 'profile') )
-  a0 <- sir_ag(x1, obs = 'from0to1', print = list(period))
-  a0 <- sir_ag(x1, obs = 'from0to1', print = list(per=period))
-  testthat::expect_message( a0 <- sir_ag(x1, obs = from0to1, exp = d.exp, pyrs = pyrs) )
-  a0 <- sir_ag(x1, obs = from0to1, exp = d.exp, pyrs = pyrs, print = period)
-  
-  testthat::expect_message(a2 <- sir_ag(x1, print = attr(x, "aggre.meta")$by))
-  testthat::expect_message(a3 <- sir_ag(x1))
-  testthat::expect_equal(data.table(a2),data.table(a3))
-
-  
-  a4 <- sir_ag(x1, obs = 'from0to1', print = surv.int)
-  a4b <- sir_ag(x1, obs = 'from0to1', print = c('surv.int','period'))  
-  a5 <- sir_ag(x1, obs = 'from0to1', print = list(surv.int,period))  
-  a6 <- sir_ag(x1, obs = 'from0to1', print = c('period','surv.int'))
-  setcolorder(a5, c(2,1,3:9))
-  setkey(a5, period)
-  setkey(a6, period)
-  testthat::expect_equal(data.table(a5)[order(period, surv.int)], data.table(a6)[order(period, surv.int)])
-  testthat::expect_error(sir_ag(x2, print = NULL))
-  
-  # UPDATE
-  a4c <- update(a4, print = c('surv.int','period')) # update to l2
-  testthat::expect_equal(a4c,a4b)
-  
-  
-  # LEXIS
-  testthat::expect_error(sir_lex(x1))
-  l1 <- sir_lex(x2, print = attr(x, "aggre.meta")$by)
-  l2 <- sir_lex(x2, print = 'per')
-  l2 <- sir_lex(x2, print = per)
-  l3 <- sir_lex(x2, print = NULL)
-  testthat::expect_equal(l3[,.N], 1)
-  
-  l1 <- sir_lex(x2, print = NULL)
-  l2 <- sir_lex(x2, print = 'per')
-  l2 <- sir_lex(x2, print = per)
-  l3 <- sir_lex(x2, print = NULL)
-  
-  
-  # breaks...
-  BL1 <- list(fot = 0:5, per = c(2003,2008,2013))
-  b1 <- sir_lex(x3, breaks = BL1, print = c('fot','per')) 
-  BL <- list(fot = 0:5, per = c('2003-01-01','2008-01-01','2013-01-01'), age = c(0,75,100))
-  b3 <- sir_lex(x3, breaks = BL, print = c('fot','per','age'))
-  testthat::expect_equal(b3[,.N], 24)
-  
-  BL2 <- list(fot = 0:5, age = c(0,50,70,100))
-  b3 <- sir_lex(x2, breaks = BL2, print = c('fot','age')) # already split data
-  testthat::expect_equal(b3[,.N], 15)
-  
-  # Character date
-  BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
-  b5 <- sir_lex(x3, breaks = BL, print = c('fot','per'))
-  BL <- list(fot = 0:5, per = c(2002.999, 2007.999, 2013.001))
-  b6 <- sir_lex(x3, breaks = BL, print = c('fot','per'))
-  testthat::expect_equal(data.table(b5), data.table(b6), tolerance = 0.001)
-  
-  # EXPECTED
-  # without pyrs;
-  e0 <- sir_exp(x1, obs = "from0to1", exp = "d.exp", pyrs = , print = period)
-  e1 <- sir_exp(x1, obs = "from0to1", exp = "d.exp", pyrs = "pyrs", print = period)
-  e1 <- sir_exp(x1, obs = from0to1, exp = d.exp, pyrs = pyrs, print = period)
-  e2 <- sir_exp(x2, obs = "lex.Xst", exp = list(exp = pop.haz*lex.dur), pyrs = "lex.dur", 
-                print = list(period = cut(per, c(2003,2008, 2013), right = FALSE)))
-  e1 <- data.table(e1)
-  e2 <- data.table(e2) 
-  testthat::expect_equal(e1[,.(pyrs, sir, sir.lo, sir.hi, p_value)], e2[,.(pyrs, sir, sir.lo, sir.hi, p_value)])
-  
-  # WARNINGS (when missing pop.haz)
-  testthat::expect_error( f1 <- sir_lex(x4) )
-  testthat::expect_error( f1 <- sir_exp(x4, obs ='lex.Xst', exp = list(exp = lex.dur*pop.haz), pyrs ='lex.dur') )
-  
-  # SIR_RATIO
-  testthat::expect_equal(sir_ratio(a2, a3)[1], c(sir_ratio = 1)) 
-  plot(a2)
-  plot(a4)
-})
-
+testthat::context("SIR")
+
+
+testthat::test_that("SIR w/ coh=ref=popEpi::sire", {
+  ## don't skip on CRAN
+  sire2 <- copy(popEpi::sire)
+  sire2[, agegroup := cut(dg_age, breaks = c(0:17*5, Inf))]
+  levels(sire2$agegroup) <- 1:18
+  sire2[, ex_y := year(ex_date)]
+  sire2[, dur := as.integer(ex_date-dg_date)/365.242199]
+  ltre <- ltable(sire2[status != 0], c("ex_y", "agegroup"), 
+                 expr = list(obs=.N, pyrs=sum(dur)))
+  setDT(ltre)
+  ltre[is.na(pyrs), pyrs := 0]
+  
+  sibr2 <- copy(popEpi::sibr)
+  sibr2[, agegroup := cut(dg_age, breaks = c(0:17*5, Inf))]
+  levels(sibr2$agegroup) <- 1:18
+  sibr2[, ex_y := year(ex_date)]
+  sibr2[, dur := as.integer(ex_date-dg_date)/365.242199]
+  ltbr <- ltable(sibr2[status != 0], c("ex_y", "agegroup"), 
+                 expr = list(obs=.N, pyrs=sum(dur)))
+  setDT(ltbr)
+  ltbr[is.na(pyrs), pyrs := 0]
+
+  
+  
+  sl <- sir(coh.data=ltre, coh.obs="obs", coh.pyrs="pyrs",
+            ref.data=ltre, ref.obs="obs", ref.pyrs="pyrs", 
+            adjust= c("agegroup","ex_y"))
+  plot(sl)
+  ## SIR w/ coh=ref=popEpi::sire
+  ## don't skip on CRAN
+  testthat::expect_equal(sl$sir, 1)
+  testthat::expect_equal(sl$pyrs, 13783.81, tolerance=0.01)
+  testthat::expect_equal(sl$expected, 4595)
+  testthat::expect_equal(sl$observed, 4595)
+  
+  
+  sl <- sir(coh.data=ltre, coh.obs="obs", coh.pyrs="pyrs",
+            ref.data=ltbr, ref.obs="obs", ref.pyrs="pyrs",
+            adjust= c("agegroup","ex_y"))
+  
+  ## SIR w/ coh=ref=popEpi::sire"
+  testthat::expect_equal(sl$sir, 1.39, tolerance=0.01)
+  testthat::expect_equal(sl$pyrs, 13783.81, tolerance=0.01)
+  testthat::expect_equal(sl$expected, 3305.04, tolerance=0.01)
+  testthat::expect_equal(sl$observed, 4595)
+})
+
+
+
+# SIR mstate, subset + lexpand aggre ---------------------------------------
+
+# same model
+c <- lexpand( popEpi::sire[dg_date<ex_date,], status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1990:2013, age = 0:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+# different models
+c2 <- lexpand( popEpi::sire[dg_date<ex_date,], status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+               breaks = list(per = 1990:2010, age = 0:100, fot = c(0,10,20,Inf)), 
+               aggre = list(fot, agegroup = age, year = per, sex) )
+
+
+testthat::test_that("misc things work", {
+  ms1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', 
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
+  plot(ms1)
+  
+  # only one print level with no variation
+  testthat::expect_message( ms2 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type= 'profile',
+                              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                              adjust = c('agegroup','year','sex'), 
+                              print = list(sex)) )
+  testthat::expect_equal(attributes(ms2)$sir.meta$conf.type, 'profile')
+  
+  # coef and confint ---
+  testthat::expect_equal(as.numeric(coef(ms1)), ms1$sir)
+  testthat::expect_equal(data.table(confint(ms1)), data.table('2.5 %' = ms1$sir.lo, '97.5 %' = ms1$sir.hi), tolerance = 0.001)
+})
+
+
+testthat::test_that("confidence intervals an other options works", {
+  
+  # CI's
+  ci1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(FOT = fot, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
+  testthat::expect_message( ci2 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'wald',
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(FOT = fot, sex, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) )))
+  ci3 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'univariate',
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(FOT = fot, age.cat = cut(agegroup, c(0,75,120)), per = cut(year, c(1990,2010,2014), dig.lab = 8) ))
+  
+  testthat::expect_equal(attributes(ci1)$sir.meta$conf.type,'profile')
+  testthat::expect_equal(attributes(ci2)$sir.meta$conf.type,'wald')
+  testthat::expect_equal(attributes(ci3)$sir.meta$conf.type,'univariate')
+  
+  testthat::expect_message( sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'wald',
+                       ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                       adjust = c('agegroup','year','sex'), 
+                       print = list(sex )))
+                  
+  testthat::expect_message( ci4 <-  sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'profile',
+                               ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                               adjust = c('agegroup','year','sex'), 
+                               print = list(dg = cut(agegroup, c(0,5,10,20,40,75,120)) )) )
+  testthat::expect_equal(attributes(ci4)$sir.meta$conf.type,'wald')
+    
+  
+  testthat::expect_message( ci5 <- sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'profile',
+                              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                              adjust = c('agegroup','year','sex'), 
+                              print = list(sex=sex,FOT = fot, 
+                                           age.cat = cut(agegroup, c(0,50,75,120)), 
+                                           per = cut(year, c(1990,2010,2014), dig.lab = 8) )) )
+  # ear
+  ea <- sir( coh.data = c, coh.obs = c('from0to2'), coh.pyrs = 'pyrs', conf.type = 'wald', EAR = TRUE,
+             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+             adjust = c('agegroup','year','sex'), conf.level = 0.95) 
+  testthat::expect_equal(ea[,EAR], round((1490-1482.13)/39905.92*1000,3))  
+  
+  # Update
+  ud1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(fot))
+  ud2 <- update(ud1, print = list(fot,  age.cat = cut(agegroup, c(0,75,120))))
+  
+  ud3 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', conf.type = 'profile',
+              ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+              adjust = c('agegroup','year','sex'), 
+              print = list(fot,  age.cat = cut(agegroup, c(0,75,120))))
+  testthat::expect_equal(ud2, ud3)
+})
+
+
+testthat::test_that("SIR works with multistate aggregated lexpand data", {
+  ## don't skip on CRAN
+
+  testthat::expect_warning(
+    se <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs', 
+               subset = year %in% 1990:2009,
+               ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+               adjust = c('agegroup','year','sex'), print =c('cause','fot'), mstate = 'cause')
+  )
+  
+  dummy.coh <- data.table(agegroup = 1:18, sex = 1, from0to1 = round(runif(18)), 
+                          from0to2 = round(runif(18)+0.4),
+                          pyrs = (rnorm(18)+100)*200)
+  dummy.ref <- data.table(agegroup = rep(1:18,times=2), sex = 1, obs = floor(runif(36)*100),
+                          categ2 = rep(c(1,2),each=18),pyrs = (rnorm(36)+100)*2000)
+  
+  sm <- sir( coh.data = dummy.coh, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs', 
+             ref.data = dummy.ref, ref.obs = 'obs', ref.pyrs = 'pyrs', mstate = 'categ2',
+             adjust = c('agegroup','sex','categ2'), print =c('categ2'))
+  
+  setkey(12345)
+  dummy.coh2 <- data.table(agegroup = rep(1:18,times=6), sex = rep(1:2, each= 18), obs = floor(runif(18*6)*100), level = rep(1:6,each=18),pyrs = (rnorm(36)+100)*2000)
+                           
+  sg <- sir( coh.data = dummy.coh2, coh.obs = c('obs'), coh.pyrs = 'pyrs', 
+             adjust = c('agegroup'), print =c('level'))
+  testthat::expect_equal(attributes(sg)$sir.meta$pooled[,sir], 1)
+  testthat::expect_equal(sg[,.N], 6)
+    
+
+  s1 <- sir( coh.data = c, coh.obs = c('from0to1'), coh.pyrs = 'pyrs', subset = year %in% 1990:2009,
+             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+             adjust = c('agegroup','year','sex'), print =c('fot'))
+  
+  s2 <- sir( coh.data = c2, coh.obs = c('from0to2'), coh.pyrs = 'pyrs',
+             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+             adjust = c('agegroup','year','sex'), print =c('fot'))
+  s12 <- rbind( cbind(cause=1L, data.table(s1)), 
+                cbind(cause=2L, data.table(s2)))
+  
+  # compare by hand calculated sir
+  
+  r <- merge(c2, data.table(popEpi::popmort), c('sex','agegroup','year'), all.x=TRUE)
+  setDT(r)
+  r[, exp := haz*pyrs]
+  est <- r[, list(observed=sum(from0to1, na.rm=TRUE),expected=sum(exp, na.rm=TRUE)), by=.(fot)]
+  est <- round(est,4)
+  
+  testthat::expect_is(object = se, class = 'sir')
+  testthat::expect_equivalent(se, s12)
+  setDT(s1)
+  testthat::expect_equivalent(s1[,1:3, with=FALSE], est)
+})
+
+
+
+# SIR utils ---------------------------------------------------------------
+
+testthat::test_that('Util functions work', {
+  testthat::expect_equal( poisson.ci(5,5)$rate, 1)
+  testthat::expect_equal(p.round(0.0000001), "< 0.001")
+})
+
+testthat::test_that("data.table subsettin works", {
+  
+  ss <- sir( coh.data = c2, coh.obs = c('from0to2'), coh.pyrs = 'pyrs',
+             ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+             adjust = c('agegroup','year','sex'), print =c('fot'))
+  testthat::expect_equal(ss[1,.N], 1)
+  testthat::expect_equal(c(ss[,.(fot)]), list(fot = c(0,10)))
+})
+
+
+
+# sir splines -------------------------------------------------------------
+
+if (requireNamespace("splines")) {
+  testthat::test_that("SIR spline throws errors correctly", {
+    popEpi:::skip_normally()
+    
+    sp0 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
+                                          subset = year %in% 1990:2008,
+                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                                          adjust = c('agegroup','year','sex'), print = NULL,
+                                          spline=c('agegroup','year','fot') )))
+    sp1 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                                          subset = year %in% 1990:2008,
+                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                                          adjust = c('agegroup','year','sex'), print =c('cause'),
+                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
+    sp2 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                                          subset = year %in% 1990:2008,
+                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', dependent.spline=FALSE,
+                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
+                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
+    sp3 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                                          subset = year %in% 1990:2008,
+                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
+                                          mstate = 'cause', spline='agegroup') ))
+    
+    sp4 <- suppressWarnings(try(sirspline( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                                          subset = year %in% 1990:2008,
+                                          ref.data = data.table(popEpi::popmort), ref.rate = 'haz', reference.points = c(2000,4),
+                                          adjust = c('agegroup','year','sex'), print =c('cause'), 
+                                          mstate = 'cause', spline=c('agegroup','year','fot') )))
+    testthat::expect_is( object = sp1, class = 'sirspline')
+    testthat::expect_is( object = sp2, class = 'sirspline')
+    testthat::expect_is( object = sp3, class = 'sirspline')
+    testthat::expect_is( object = sp4, class = 'sirspline')
+  })  
+}
+
+testthat::test_that("print accepts a function and subset works", {
+  popEpi:::skip_normally()
+  testthat::expect_warning(
+    pl1 <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                subset = year %in% 1990:2008,
+                ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                adjust = c('agegroup','year','sex'), print = list(year.int = findInterval(year,c(1989,2000,2010))),
+                mstate = 'cause')
+  )
+  testthat::expect_warning(
+    pl2 <- sir( coh.data = c, coh.obs = c('from0to1','from0to2'), coh.pyrs = 'pyrs',
+                subset = year %in% 1990:2008,
+                ref.data = data.table(popEpi::popmort), ref.rate = 'haz', 
+                adjust = c('agegroup','year','sex'), print = list(year.int = findInterval(year,c(1989,2000,2010))),
+                mstate = 'cause')
+  )
+  setDT(pl1)
+  testthat::expect_equal( pl1[,year.int], 1:2)
+  testthat::expect_is(object=pl2, 'sir')
+})
+
+
+
+testthat::test_that("sir_ratio", {
+  ## don't skip on CRAN
+  dt1 <- data.table(obs = rep(c(5,7), 10),
+                    pyrs = rep(c(250,300,350,400), 5),
+                    var = 1:20)
+  
+  dt2 <- data.table(obs = rep(c(10,13), 10),
+                    pyrs = rep(c(250,300,350,400), 5),
+                    var = 1:20)
+  
+  Ref <- data.table(obs = rep(c(50,70,80,100), 5),
+                    pyrs = rep(c(2500,3000,3500,4000), 5),
+                    var = 1:20)
+  
+  
+  s1 <- sir(coh.data = dt1, coh.obs = obs, coh.pyrs = pyrs, 
+            ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
+            adjust = var)
+  
+  
+  s2 <- sir(coh.data = dt2, coh.obs = obs, coh.pyrs = pyrs, 
+            ref.data = Ref, ref.obs = obs, ref.pyrs = pyrs,
+            adjust = var)
+  
+  
+  ## SIR w/ coh=ref=popEpi::sire
+  ## don't skip on CRAN
+  # Test the example from the book (statistics with confidence):
+  testthat::expect_equal( sir_ratio(x = c(64, 45.6), y = c(25,23.7), digits=2)[2:3], c(lower=0.83, upper=2.21))
+  # other tests
+  testthat::expect_message(sir_ratio(s1, s2, digits = 2, alternative = 'less', conf.level = 0.95, type = 'asymptotic'))
+  testthat::expect_equal(sir_ratio(s1, s2), sir_ratio(c(120,150), c(230,150)), tolerance=0.001)
+  testthat::expect_equal(sir_ratio(s1, c(230,150)), sir_ratio(c(120,150), s2), tolerance=0.001)
+  testthat::expect_equal(sir_ratio(s1, s2)[1], c(sir_ratio =round((120/150)/(230/150),3)), tolerance=0.001)
+})
+
+
+testthat::test_that("sir_exp and sir_ag and sir_lexis are working", {
+  ## don't skip on CRAN
+  # from0to1 ei oo vakio
+  
+  BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+  suppressMessages(
+    # aggregated with breaks
+    x1 <- lexpand(sire, breaks = BL, status = status != 0, 
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 pophaz=popmort,
+                 aggre=list(sex, period = per, surv.int = fot))
+  )
+  suppressMessages(
+    # no aggreate but breaks
+    x2 <- lexpand(sire, breaks = BL, status = status != 0, 
+                 birth = bi_date, entry = dg_date, exit = ex_date,
+                 pophaz=popmort)
+  )
+
+  suppressMessages(
+    x3 <- lexpand(sire, status = status != 0,
+                  birth = bi_date, entry = dg_date, exit = ex_date,
+                  pophaz=popmort)
+  )
+  suppressMessages(
+    x4 <- lexpand(sire, status = status != 0,
+                  birth = bi_date, entry = bi_date, exit = ex_date,
+                  pophaz=popmort)
+  )
+  # library(Epi)
+  # # no aggreate but breaks
+  # suppressMessages(
+  #   x5 <- Lexis(entry = list(FUT = 0, AGE = 0, CAL = get.yrs(bi_date)), 
+  #              exit = list(CAL = get.yrs(ex_date)), 
+  #              data = sire[sire$dg_date < sire$ex_date, ],
+  #              exit.status = factor(status, levels = 0:2, 
+  #                                   labels = c("alive", "canD", "othD")), 
+  #              merge = TRUE)
+  #   )
+  
+   
+   
+  # AGGRE
+  testthat::expect_message( a1 <- sir_ag(x1, obs = 'from0to1', conf.type = 'profile') )
+  testthat::expect_message( a0 <- sir_ag(x1, obs = from0to1, conf.type = 'profile') )
+  a0 <- sir_ag(x1, obs = 'from0to1', print = list(period))
+  a0 <- sir_ag(x1, obs = 'from0to1', print = list(per=period))
+  testthat::expect_message( a0 <- sir_ag(x1, obs = from0to1, exp = d.exp, pyrs = pyrs) )
+  a0 <- sir_ag(x1, obs = from0to1, exp = d.exp, pyrs = pyrs, print = period)
+  
+  testthat::expect_message(a2 <- sir_ag(x1, print = attr(x, "aggre.meta")$by))
+  testthat::expect_message(a3 <- sir_ag(x1))
+  testthat::expect_equal(data.table(a2),data.table(a3))
+
+  
+  a4 <- sir_ag(x1, obs = 'from0to1', print = surv.int)
+  a4b <- sir_ag(x1, obs = 'from0to1', print = c('surv.int','period'))  
+  a5 <- sir_ag(x1, obs = 'from0to1', print = list(surv.int,period))  
+  a6 <- sir_ag(x1, obs = 'from0to1', print = c('period','surv.int'))
+  setcolorder(a5, c(2,1,3:9))
+  setkey(a5, period)
+  setkey(a6, period)
+  testthat::expect_equal(data.table(a5)[order(period, surv.int)], data.table(a6)[order(period, surv.int)])
+  testthat::expect_error(sir_ag(x2, print = NULL))
+  
+  # UPDATE
+  a4c <- update(a4, print = c('surv.int','period')) # update to l2
+  testthat::expect_equal(a4c,a4b)
+  
+  
+  # LEXIS
+  testthat::expect_error(sir_lex(x1))
+  l1 <- sir_lex(x2, print = attr(x, "aggre.meta")$by)
+  l2 <- sir_lex(x2, print = 'per')
+  l2 <- sir_lex(x2, print = per)
+  l3 <- sir_lex(x2, print = NULL)
+  testthat::expect_equal(l3[,.N], 1)
+  
+  l1 <- sir_lex(x2, print = NULL)
+  l2 <- sir_lex(x2, print = 'per')
+  l2 <- sir_lex(x2, print = per)
+  l3 <- sir_lex(x2, print = NULL)
+  
+  
+  # breaks...
+  BL1 <- list(fot = 0:5, per = c(2003,2008,2013))
+  b1 <- sir_lex(x3, breaks = BL1, print = c('fot','per')) 
+  BL <- list(fot = 0:5, per = c('2003-01-01','2008-01-01','2013-01-01'), age = c(0,75,100))
+  b3 <- sir_lex(x3, breaks = BL, print = c('fot','per','age'))
+  testthat::expect_equal(b3[,.N], 24)
+  
+  BL2 <- list(fot = 0:5, age = c(0,50,70,100))
+  b3 <- sir_lex(x2, breaks = BL2, print = c('fot','age')) # already split data
+  testthat::expect_equal(b3[,.N], 15)
+  
+  # Character date
+  BL <- list(fot = 0:5, per = c("2003-01-01","2008-01-01", "2013-01-01"))
+  b5 <- sir_lex(x3, breaks = BL, print = c('fot','per'))
+  BL <- list(fot = 0:5, per = c(2002.999, 2007.999, 2013.001))
+  b6 <- sir_lex(x3, breaks = BL, print = c('fot','per'))
+  testthat::expect_equal(data.table(b5), data.table(b6), tolerance = 0.001)
+  
+  # EXPECTED
+  # without pyrs;
+  e0 <- sir_exp(x1, obs = "from0to1", exp = "d.exp", pyrs = , print = period)
+  e1 <- sir_exp(x1, obs = "from0to1", exp = "d.exp", pyrs = "pyrs", print = period)
+  e1 <- sir_exp(x1, obs = from0to1, exp = d.exp, pyrs = pyrs, print = period)
+  e2 <- sir_exp(x2, obs = "lex.Xst", exp = list(exp = pop.haz*lex.dur), pyrs = "lex.dur", 
+                print = list(period = cut(per, c(2003,2008, 2013), right = FALSE)))
+  e1 <- data.table(e1)
+  e2 <- data.table(e2) 
+  testthat::expect_equal(e1[,.(pyrs, sir, sir.lo, sir.hi, p_value)], e2[,.(pyrs, sir, sir.lo, sir.hi, p_value)])
+  
+  # WARNINGS (when missing pop.haz)
+  testthat::expect_error( f1 <- sir_lex(x4) )
+  testthat::expect_error( f1 <- sir_exp(x4, obs ='lex.Xst', exp = list(exp = lex.dur*pop.haz), pyrs ='lex.dur') )
+  
+  # SIR_RATIO
+  testthat::expect_equal(sir_ratio(a2, a3)[1], c(sir_ratio = 1)) 
+  plot(a2)
+  plot(a4)
+})
+
diff --git a/tests/testthat/test_splitLexisDT.R b/tests/testthat/test_splitLexisDT.R
index 3b8e5bb..09a7ce3 100644
--- a/tests/testthat/test_splitLexisDT.R
+++ b/tests/testthat/test_splitLexisDT.R
@@ -1,134 +1,134 @@
-testthat::context("Compare splitLexisDT results with splitLexis results")
-
-testthat::test_that("splitLexisDT and splitLexis are congruent", {
-  popEpi:::skip_normally()
-  
-  library(Epi)
-  
-  sire2 <- data.table::copy(popEpi::sire)
-  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
-  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
-  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
-  sire2[, id := 1:.N]
-  
-  BL <- list(fot = 0:5, fot = 2:6, fot = c(10,Inf), fot = c(0, Inf),
-             per = 1990:1995, per = seq(1950,2010,10), per = c(1900, 2100), 
-             age = seq(0,150,5), age = c(25,100), age = c(0, 60))
-  
-  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
-             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L)
-  setDT(x)
-  forceLexisDT(x, breaks = list(fot=NULL,per=NULL,age=NULL),
-               allScales = c("fot", "per", "age"))
-  
-  
-  # x2 <- splitLexis(x, breaks = BL[[3]], time.scale = "fot")
-  # x3 <- splitLexisDT(x, breaks = BL[[3]], timeScale = "fot", drop = FALSE)
-  # x2 <- intelliDrop(setDT(x2),  breaks = list(fot = BL[[3]]))
-  # x3 <- intelliDrop(x3,  breaks = list(fot = BL[[3]]))
-  # x4 <- splitLexisDT(x, breaks = BL[[3]], timeScale = "fot", drop = TRUE)
-  
-  
-  # one row per id ---------------------------------------------------------------
-  
-  testthat::test_that("splitLexisDT and splitLexis congruent with one row per id", {
-    testthat::expect_identical(1L, 1L) ## to trigger testing...
-    for (sc in seq_along(BL)) {
-      testthat::test_that(paste0("results congruent using breaks ", sc), {
-        testthat::expect_identical(1L, 1L) ## to trigger testing...
-        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
-      })
-    }
-  })
-  
-  
-  
-  # multiple rows per id ---------------------------------------------------------
-  
-  sire2 <- sire2[rep(1:.N, each = 2)]
-  
-  
-  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
-             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L, id = id)
-  
-  
-  testthat::test_that("splitLexisDT and splitLexis congruent with multiple rows per id", {
-    testthat::expect_identical(1L, 1L) ## to trigger testing...
-    for (sc in seq_along(BL)) {
-      testthat::test_that(paste0("results congruent using breaks ", sc), {
-        testthat::expect_identical(1L, 1L) ## to trigger testing...
-        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
-      })
-    }
-  })
-  
-  
-  # multistate using Lexis -----------------------------------------------------
-  
-  sire2[, "EX" := factor(status, levels = 0:2, ordered = TRUE)]    
-  sire2[, "EN" := factor(0L, levels = 0:2, ordered = TRUE)]
-  levels(sire2$EX) <- levels(sire2$EN) <- c("ok", "dead", "dead")
-  
-  x <- Lexis(data = sire2[dg_date < ex_date & !duplicated(id)], 
-             entry = list(fot=0, per=bi_yrs, age=0),
-             exit = list(per=ex_yrs), 
-             merge=TRUE, 
-             exit.status=EX, 
-             entry.status = EN, 
-             id = id)
-  
-  x <- cutLexis(x, cut = x$dg_yrs, timescale = "per", new.state = "sick", precursor.state = "ok")
-  setDT(x)
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  
-  testthat::test_that("splitLexisDT and splitLexis congruent with multiple Lexis states per id", {
-    testthat::expect_identical(1L, 1L) ## to trigger testing...
-    for (sc in seq_along(BL)) {
-      testthat::test_that(paste0("results congruent using breaks ", sc), {
-        testthat::expect_identical(1L, 1L) ## to trigger testing...
-        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
-      })
-    }
-  })
-  
-  # using mstate package -------------------------------------------------------
-  if (requireNamespace("mstate")) {
-    testthat::test_that("splitLexisDT and splitLexis congruent using mstate data", {
-      testthat::expect_identical(1L, 1L) ## to trigger testing...
-      ## taken directly from ?msprep examples
-      library(mstate)
-      tmat <- trans.illdeath()
-      # some data in wide format
-      tg <- data.frame(stt=rep(0,6),sts=rep(0,6),
-                      illt=c(1,1,6,6,8,9),ills=c(1,0,1,1,0,1),
-                      dt=c(5,1,9,7,8,12),ds=c(1,1,1,1,1,1),
-                      x1=c(1,1,1,2,2,2),x2=c(6:1))
-      tg$x1 <- factor(tg$x1,labels=c("male","female"))
-      tg$patid <- factor(2:7,levels=1:8,labels=as.character(1:8))
-      # define time, status and covariates also as matrices
-      tt <- matrix(c(rep(NA,6),tg$illt,tg$dt),6,3)
-      st <- matrix(c(rep(NA,6),tg$ills,tg$ds),6,3)
-      keepmat <- data.frame(gender=tg$x1,age=tg$x2)
-      # data in long format using msprep
-      df <- msprep(time=tt,status=st,trans=tmat,keep=as.matrix(keepmat))
-      
-      x <- Lexis(data=df, entry=list(TT = Tstart), exit=list(TT = Tstop), 
-                merge=TRUE, exit.status=to, entry.status = from, id = id)
-      setDT(x)
-      setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-      BL2 <- list(TT = 0:4, TT = 1:2, TT = 0:1, TT = c(1,Inf))
-      
-      for (sc in seq_along(BL2)) {
-        testthat::test_that(paste0("results congruent using breaks ", sc), {
-          testthat::expect_identical(1L, 1L) ## to trigger testing...
-          popEpi:::compareSLDTWithEpi(data = x, breaks = BL2[[sc]], timeScale = "TT")
-        })
-      }
-    })
-  }
-  
-})
-
-
-
-
+testthat::context("Compare splitLexisDT results with splitLexis results")
+
+testthat::test_that("splitLexisDT and splitLexis are congruent", {
+  popEpi:::skip_normally()
+  
+  library(Epi)
+  
+  sire2 <- data.table::copy(popEpi::sire)
+  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
+  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
+  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
+  sire2[, id := 1:.N]
+  
+  BL <- list(fot = 0:5, fot = 2:6, fot = c(10,Inf), fot = c(0, Inf),
+             per = 1990:1995, per = seq(1950,2010,10), per = c(1900, 2100), 
+             age = seq(0,150,5), age = c(25,100), age = c(0, 60))
+  
+  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
+             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L)
+  setDT(x)
+  forceLexisDT(x, breaks = list(fot=NULL,per=NULL,age=NULL),
+               allScales = c("fot", "per", "age"))
+  
+  
+  # x2 <- splitLexis(x, breaks = BL[[3]], time.scale = "fot")
+  # x3 <- splitLexisDT(x, breaks = BL[[3]], timeScale = "fot", drop = FALSE)
+  # x2 <- intelliDrop(setDT(x2),  breaks = list(fot = BL[[3]]))
+  # x3 <- intelliDrop(x3,  breaks = list(fot = BL[[3]]))
+  # x4 <- splitLexisDT(x, breaks = BL[[3]], timeScale = "fot", drop = TRUE)
+  
+  
+  # one row per id ---------------------------------------------------------------
+  
+  testthat::test_that("splitLexisDT and splitLexis congruent with one row per id", {
+    testthat::expect_identical(1L, 1L) ## to trigger testing...
+    for (sc in seq_along(BL)) {
+      testthat::test_that(paste0("results congruent using breaks ", sc), {
+        testthat::expect_identical(1L, 1L) ## to trigger testing...
+        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
+      })
+    }
+  })
+  
+  
+  
+  # multiple rows per id ---------------------------------------------------------
+  
+  sire2 <- sire2[rep(1:.N, each = 2)]
+  
+  
+  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
+             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L, id = id)
+  
+  
+  testthat::test_that("splitLexisDT and splitLexis congruent with multiple rows per id", {
+    testthat::expect_identical(1L, 1L) ## to trigger testing...
+    for (sc in seq_along(BL)) {
+      testthat::test_that(paste0("results congruent using breaks ", sc), {
+        testthat::expect_identical(1L, 1L) ## to trigger testing...
+        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
+      })
+    }
+  })
+  
+  
+  # multistate using Lexis -----------------------------------------------------
+  
+  sire2[, "EX" := factor(status, levels = 0:2, ordered = TRUE)]    
+  sire2[, "EN" := factor(0L, levels = 0:2, ordered = TRUE)]
+  levels(sire2$EX) <- levels(sire2$EN) <- c("ok", "dead", "dead")
+  
+  x <- Lexis(data = sire2[dg_date < ex_date & !duplicated(id)], 
+             entry = list(fot=0, per=bi_yrs, age=0),
+             exit = list(per=ex_yrs), 
+             merge=TRUE, 
+             exit.status=EX, 
+             entry.status = EN, 
+             id = id)
+  
+  x <- cutLexis(x, cut = x$dg_yrs, timescale = "per", new.state = "sick", precursor.state = "ok")
+  setDT(x)
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  
+  testthat::test_that("splitLexisDT and splitLexis congruent with multiple Lexis states per id", {
+    testthat::expect_identical(1L, 1L) ## to trigger testing...
+    for (sc in seq_along(BL)) {
+      testthat::test_that(paste0("results congruent using breaks ", sc), {
+        testthat::expect_identical(1L, 1L) ## to trigger testing...
+        popEpi:::compareSLDTWithEpi(data = x, breaks = BL[[sc]], timeScale = names(BL)[sc])
+      })
+    }
+  })
+  
+  # using mstate package -------------------------------------------------------
+  if (requireNamespace("mstate")) {
+    testthat::test_that("splitLexisDT and splitLexis congruent using mstate data", {
+      testthat::expect_identical(1L, 1L) ## to trigger testing...
+      ## taken directly from ?msprep examples
+      library(mstate)
+      tmat <- trans.illdeath()
+      # some data in wide format
+      tg <- data.frame(stt=rep(0,6),sts=rep(0,6),
+                      illt=c(1,1,6,6,8,9),ills=c(1,0,1,1,0,1),
+                      dt=c(5,1,9,7,8,12),ds=c(1,1,1,1,1,1),
+                      x1=c(1,1,1,2,2,2),x2=c(6:1))
+      tg$x1 <- factor(tg$x1,labels=c("male","female"))
+      tg$patid <- factor(2:7,levels=1:8,labels=as.character(1:8))
+      # define time, status and covariates also as matrices
+      tt <- matrix(c(rep(NA,6),tg$illt,tg$dt),6,3)
+      st <- matrix(c(rep(NA,6),tg$ills,tg$ds),6,3)
+      keepmat <- data.frame(gender=tg$x1,age=tg$x2)
+      # data in long format using msprep
+      df <- msprep(time=tt,status=st,trans=tmat,keep=as.matrix(keepmat))
+      
+      x <- Lexis(data=df, entry=list(TT = Tstart), exit=list(TT = Tstop), 
+                merge=TRUE, exit.status=to, entry.status = from, id = id)
+      setDT(x)
+      setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+      BL2 <- list(TT = 0:4, TT = 1:2, TT = 0:1, TT = c(1,Inf))
+      
+      for (sc in seq_along(BL2)) {
+        testthat::test_that(paste0("results congruent using breaks ", sc), {
+          testthat::expect_identical(1L, 1L) ## to trigger testing...
+          popEpi:::compareSLDTWithEpi(data = x, breaks = BL2[[sc]], timeScale = "TT")
+        })
+      }
+    })
+  }
+  
+})
+
+
+
+
diff --git a/tests/testthat/test_splitMulti.R b/tests/testthat/test_splitMulti.R
index 38356f4..5325d61 100644
--- a/tests/testthat/test_splitMulti.R
+++ b/tests/testthat/test_splitMulti.R
@@ -1,145 +1,145 @@
-testthat::context("Compare splitMulti results with splitLexis results")
-
-
-
-
-
-testthat::test_that("splitMulti and splitLexis are congruent", {
-  testthat::expect_identical(1L, 1L) ## to trigger testing...
-  popEpi:::skip_normally()
-  library(Epi)
-  
-  sire2 <- copy(sire)
-  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
-  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
-  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
-  sire2[, id := 1:.N]
-  
-  BL1 <- list(fot = 0:5, per = 1990:1995, age = c(0, 60, Inf))
-  
-  BL2 <- list(fot = c(10,Inf), age = seq(0,150,5))
-  
-  BL3 <- list(fot = c(5, Inf), per = c(1900, 2100), age = c(25,100))
-  
-  BL4 <- list(fot = 0:10)
-  
-  BL5 <- list(fot = 5:10)
-  
-  BL6 <- list(per = 1990:2000, age = c(50,70))
-  
-  
-  BL <- list(BL1, BL2, BL3, BL4, BL5, BL6)
-  
-  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
-             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L)
-  setDT(x)
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  
-  
-  # one row per id ---------------------------------------------------------------
-  
-  testthat::test_that("splitMulti and splitLexis congruent with one row per id", {
-    testthat::expect_identical(1L, 1L) ## to trigger testing...
-    for (sc in seq_along(BL)) {
-      compareSMWithEpi(x, BL[[sc]])
-    }
-  })
-  
-  
-  
-  # multiple rows per id ---------------------------------------------------------
-  
-  sire2 <- sire2[rep(1:.N, each = 2)]
-  
-  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
-             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L, id = id)
-  setDT(x)
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  
-  for (sc in seq_along(BL)) {
-    testthat::test_that(paste0("splitLexisDT and splitLexis congruent with multiple rows per id with breaks no. ", sc), {
-      testthat::expect_identical(1L, 1L) ## to trigger testing...
-      compareSMWithEpi(x, BL[[sc]])
-    })
-  }
-  
-  # multistate using Lexis -----------------------------------------------------
-  
-  
-  sire2[, "EX" := factor(status, levels = 0:2, ordered = TRUE)]    
-  sire2[, "EN" := factor(0L, levels = 0:2, ordered = TRUE)]
-  levels(sire2$EX) <- levels(sire2$EN) <- c("ok", "dead", "dead")
-  
-  x <- Lexis(data=sire2[dg_date < ex_date & !duplicated(id)], entry=list(fot=0, per=bi_yrs, age=0),
-             exit=list(per=ex_yrs), merge=TRUE, exit.status=EX, entry.status = EN, id = id)
-  
-  x <- cutLexis(x, cut = x$dg_yrs, timescale = "per", new.state = "sick", precursor.state = "ok")
-  setDT(x)   
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  
-  BL[[1L]] <- NULL ## this would drop all rows in split data
-  
-  for (sc in seq_along(BL)) {
-    testthat::test_that(paste0("splitLexisDT and splitLexis congruent with multiple Lexis states per id using breaks list no. ", sc), {
-      testthat::expect_identical(1L, 1L) ## to trigger testing...
-      compareSMWithEpi(x, BL[[sc]])
-    })
-  }
-  
-  
-})
-
-
-
-
-testthat::test_that("splitMulti agrees with splitLexis, vol. II", {
-  
-  library("Epi")
-  
-  data(nickel, package = "Epi")
-  
-  lex <- Lexis( entry = list(age=agein,
-                             per=agein+dob),
-                exit = list(age=ageout),
-                exit.status = factor(icd>0, labels=c("Alive","Dead")),
-                entry.status = factor(0, 0:1, labels = c("Alive", "Dead")),
-                data = nickel )
-  
-  set.seed(1337)
-  lex$lex.id <- sample(paste0("abcd_", 1:nrow(lex)), size = nrow(lex))
-  
-  lex_copy <- copy(lex)
-  
-  BL <- list(
-    per = 1920:1990,
-    age = 0:100
-  )
-  
-  epi_s1 <- splitLexis(lex, breaks = BL$per, time.scale = "per")
-  epi_s2 <- splitLexis(epi_s1, breaks = BL$age, time.scale = "age")
-  
-  pop_s1 <- splitLexisDT(lex, breaks=BL$per, timeScale="per", drop = FALSE)
-  pop_s2 <- splitLexisDT(pop_s1, breaks=BL$age, timeScale="age" , drop = FALSE)
-  
-  pop_sm <- splitMulti(lex, breaks = BL, drop = FALSE)
-  
-  testthat::expect_equal(
-    setDT(epi_s2), setDT(pop_sm), check.attributes = FALSE
-  )
-  testthat::expect_equal(
-    setDT(epi_s2), setDT(pop_s2), check.attributes = FALSE
-  )
-  testthat::expect_identical(
-    lex, lex_copy
-  )
-  
-})
-
-
-
-
-
-
-
-
-
+testthat::context("Compare splitMulti results with splitLexis results")
+
+
+
+
+
+testthat::test_that("splitMulti and splitLexis are congruent", {
+  testthat::expect_identical(1L, 1L) ## to trigger testing...
+  popEpi:::skip_normally()
+  library(Epi)
+  
+  sire2 <- copy(sire)
+  sire2[, dg_yrs := get.yrs(dg_date, "actual")]
+  sire2[, ex_yrs := get.yrs(ex_date, "actual")]
+  sire2[, bi_yrs := get.yrs(bi_date, "actual")]
+  sire2[, id := 1:.N]
+  
+  BL1 <- list(fot = 0:5, per = 1990:1995, age = c(0, 60, Inf))
+  
+  BL2 <- list(fot = c(10,Inf), age = seq(0,150,5))
+  
+  BL3 <- list(fot = c(5, Inf), per = c(1900, 2100), age = c(25,100))
+  
+  BL4 <- list(fot = 0:10)
+  
+  BL5 <- list(fot = 5:10)
+  
+  BL6 <- list(per = 1990:2000, age = c(50,70))
+  
+  
+  BL <- list(BL1, BL2, BL3, BL4, BL5, BL6)
+  
+  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
+             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L)
+  setDT(x)
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  
+  
+  # one row per id ---------------------------------------------------------------
+  
+  testthat::test_that("splitMulti and splitLexis congruent with one row per id", {
+    testthat::expect_identical(1L, 1L) ## to trigger testing...
+    for (sc in seq_along(BL)) {
+      compareSMWithEpi(x, BL[[sc]])
+    }
+  })
+  
+  
+  
+  # multiple rows per id ---------------------------------------------------------
+  
+  sire2 <- sire2[rep(1:.N, each = 2)]
+  
+  x <- Lexis(data=sire2[dg_date < ex_date], entry=list(fot=0, per=dg_yrs, age=dg_age),
+             exit=list(per=ex_yrs), merge=TRUE, exit.status=1L, entry.status = 0L, id = id)
+  setDT(x)
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  
+  for (sc in seq_along(BL)) {
+    testthat::test_that(paste0("splitLexisDT and splitLexis congruent with multiple rows per id with breaks no. ", sc), {
+      testthat::expect_identical(1L, 1L) ## to trigger testing...
+      compareSMWithEpi(x, BL[[sc]])
+    })
+  }
+  
+  # multistate using Lexis -----------------------------------------------------
+  
+  
+  sire2[, "EX" := factor(status, levels = 0:2, ordered = TRUE)]    
+  sire2[, "EN" := factor(0L, levels = 0:2, ordered = TRUE)]
+  levels(sire2$EX) <- levels(sire2$EN) <- c("ok", "dead", "dead")
+  
+  x <- Lexis(data=sire2[dg_date < ex_date & !duplicated(id)], entry=list(fot=0, per=bi_yrs, age=0),
+             exit=list(per=ex_yrs), merge=TRUE, exit.status=EX, entry.status = EN, id = id)
+  
+  x <- cutLexis(x, cut = x$dg_yrs, timescale = "per", new.state = "sick", precursor.state = "ok")
+  setDT(x)   
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  
+  BL[[1L]] <- NULL ## this would drop all rows in split data
+  
+  for (sc in seq_along(BL)) {
+    testthat::test_that(paste0("splitLexisDT and splitLexis congruent with multiple Lexis states per id using breaks list no. ", sc), {
+      testthat::expect_identical(1L, 1L) ## to trigger testing...
+      compareSMWithEpi(x, BL[[sc]])
+    })
+  }
+  
+  
+})
+
+
+
+
+testthat::test_that("splitMulti agrees with splitLexis, vol. II", {
+  
+  library("Epi")
+  
+  data(nickel, package = "Epi")
+  
+  lex <- Lexis( entry = list(age=agein,
+                             per=agein+dob),
+                exit = list(age=ageout),
+                exit.status = factor(icd>0, labels=c("Alive","Dead")),
+                entry.status = factor(0, 0:1, labels = c("Alive", "Dead")),
+                data = nickel )
+  
+  set.seed(1337)
+  lex$lex.id <- sample(paste0("abcd_", 1:nrow(lex)), size = nrow(lex))
+  
+  lex_copy <- copy(lex)
+  
+  BL <- list(
+    per = 1920:1990,
+    age = 0:100
+  )
+  
+  epi_s1 <- splitLexis(lex, breaks = BL$per, time.scale = "per")
+  epi_s2 <- splitLexis(epi_s1, breaks = BL$age, time.scale = "age")
+  
+  pop_s1 <- splitLexisDT(lex, breaks=BL$per, timeScale="per", drop = FALSE)
+  pop_s2 <- splitLexisDT(pop_s1, breaks=BL$age, timeScale="age" , drop = FALSE)
+  
+  pop_sm <- splitMulti(lex, breaks = BL, drop = FALSE)
+  
+  testthat::expect_equal(
+    setDT(epi_s2), setDT(pop_sm), check.attributes = FALSE
+  )
+  testthat::expect_equal(
+    setDT(epi_s2), setDT(pop_s2), check.attributes = FALSE
+  )
+  testthat::expect_identical(
+    lex, lex_copy
+  )
+  
+})
+
+
+
+
+
+
+
+
+
diff --git a/tests/testthat/test_splitting_attributes.R b/tests/testthat/test_splitting_attributes.R
index 7bd4bee..9f7e541 100644
--- a/tests/testthat/test_splitting_attributes.R
+++ b/tests/testthat/test_splitting_attributes.R
@@ -1,93 +1,93 @@
-testthat::context("Attributes of data split by popEpi funs")
-
-
-
-
-
-testthat::test_that("popEpi splitters produce correct attributes", {
-  
-  library("Epi")
-  library("data.table")
-  
-  sire <- setDT(copy(popEpi::sire))
-  sire[, "dg_yrs" := get.yrs(dg_date, "actual")]
-  sire[, "ex_yrs" := get.yrs(ex_date, "actual")]
-  sire[, "bi_yrs" := get.yrs(bi_date, "actual")]
-  sire[, "id" := 1:.N]
-  
-  sire <- Lexis(
-    data = sire[dg_date < ex_date], 
-    entry = list(fot=0, per=dg_yrs, age=dg_age),
-    exit = list(per=ex_yrs),
-    merge = TRUE, 
-    exit.status = 1L, entry.status = 0L
-  )
-  BL <- list(age = c(0, 50), fot = c(0, 5))
-  sm_1 <- splitMulti(sire, breaks = BL["fot"], drop = TRUE)
-  sm_1 <- splitMulti(sm_1, breaks = BL["age"], drop = TRUE)
-  
-  sm_2 <- splitMulti(sire, breaks = BL, drop = TRUE)
-  
-  sl <- splitLexisDT(sire, breaks = BL$fot, timeScale = "fot", drop = TRUE)
-  sl <- splitLexisDT(sl, breaks = BL$age, timeScale = "age", drop = TRUE)
-  
-  lp <- lexpand(data.table(sire)[, .(bi_yrs, dg_yrs, ex_yrs, status)], 
-                birth = bi_yrs, entry = dg_yrs, exit = ex_yrs, status = status,
-                breaks = BL)
-  
-  lapply(list(sm_1, sm_2, sl, lp), function(lex) {
-    testthat::expect_identical(
-      attr(lex, "breaks"),
-      list(fot = c(0, 5), per = NULL, age = c(0, 50))
-    )
-    testthat::expect_identical(
-      attr(lex, "time.scales"),
-      c("fot", "per", "age")
-    ) 
-  })
-  
-})
-
-
-
-
-
-testthat::test_that("popEpi splitters retain time.since attribute", {
-  ## based on simLexis example from Epi 2.19
-  library("Epi")
-  library("data.table")
-  
-  data("DMlate", package = "Epi")
-  dml <- Lexis( entry = list(Per=dodm, Age=dodm-dobth, DMdur=0 ),
-                exit = list(Per=dox),
-                exit.status = factor(!is.na(dodth),labels=c("DM","Dead")),
-                data = DMlate[runif(nrow(DMlate))<0.1,] )
-  # Split follow-up at insulin, introduce a new timescale,
-  # and split non-precursor states
-  dmi <- cutLexis( dml, cut = dml$doins,
-                   pre = "DM",
-                   new.state = "Ins",
-                   new.scale = "t.Ins",
-                   split.states = TRUE )
-  
-  # Split the follow in 1-year intervals for modelling
-  Si <- splitLexis( dmi, 0:30/2, "DMdur" )
-  
-  
-  sldt <- splitLexisDT(dmi, breaks = 0:30/2, timeScale = "DMdur")
-  sm <- splitMulti(dmi, breaks = list(DMdur = 0:30/2))
-  
-  lex_attr_nms <- c("time.since", "breaks", "time.scales")
-  testthat::expect_identical(
-    attributes(Si)[lex_attr_nms], attributes(sm)[lex_attr_nms]
-  )
-  testthat::expect_identical(
-    attributes(Si)[lex_attr_nms], attributes(sldt)[lex_attr_nms]
-  )
-  
-})
-
-
-
-
-
+testthat::context("Attributes of data split by popEpi funs")
+
+
+
+
+
+testthat::test_that("popEpi splitters produce correct attributes", {
+  
+  library("Epi")
+  library("data.table")
+  
+  sire <- setDT(copy(popEpi::sire))
+  sire[, "dg_yrs" := get.yrs(dg_date, "actual")]
+  sire[, "ex_yrs" := get.yrs(ex_date, "actual")]
+  sire[, "bi_yrs" := get.yrs(bi_date, "actual")]
+  sire[, "id" := 1:.N]
+  
+  sire <- Lexis(
+    data = sire[dg_date < ex_date], 
+    entry = list(fot=0, per=dg_yrs, age=dg_age),
+    exit = list(per=ex_yrs),
+    merge = TRUE, 
+    exit.status = 1L, entry.status = 0L
+  )
+  BL <- list(age = c(0, 50), fot = c(0, 5))
+  sm_1 <- splitMulti(sire, breaks = BL["fot"], drop = TRUE)
+  sm_1 <- splitMulti(sm_1, breaks = BL["age"], drop = TRUE)
+  
+  sm_2 <- splitMulti(sire, breaks = BL, drop = TRUE)
+  
+  sl <- splitLexisDT(sire, breaks = BL$fot, timeScale = "fot", drop = TRUE)
+  sl <- splitLexisDT(sl, breaks = BL$age, timeScale = "age", drop = TRUE)
+  
+  lp <- lexpand(data.table(sire)[, .(bi_yrs, dg_yrs, ex_yrs, status)], 
+                birth = bi_yrs, entry = dg_yrs, exit = ex_yrs, status = status,
+                breaks = BL)
+  
+  lapply(list(sm_1, sm_2, sl, lp), function(lex) {
+    testthat::expect_identical(
+      attr(lex, "breaks"),
+      list(fot = c(0, 5), per = NULL, age = c(0, 50))
+    )
+    testthat::expect_identical(
+      attr(lex, "time.scales"),
+      c("fot", "per", "age")
+    ) 
+  })
+  
+})
+
+
+
+
+
+testthat::test_that("popEpi splitters retain time.since attribute", {
+  ## based on simLexis example from Epi 2.19
+  library("Epi")
+  library("data.table")
+  
+  data("DMlate", package = "Epi")
+  dml <- Lexis( entry = list(Per=dodm, Age=dodm-dobth, DMdur=0 ),
+                exit = list(Per=dox),
+                exit.status = factor(!is.na(dodth),labels=c("DM","Dead")),
+                data = DMlate[runif(nrow(DMlate))<0.1,] )
+  # Split follow-up at insulin, introduce a new timescale,
+  # and split non-precursor states
+  dmi <- cutLexis( dml, cut = dml$doins,
+                   pre = "DM",
+                   new.state = "Ins",
+                   new.scale = "t.Ins",
+                   split.states = TRUE )
+  
+  # Split the follow in 1-year intervals for modelling
+  Si <- splitLexis( dmi, 0:30/2, "DMdur" )
+  
+  
+  sldt <- splitLexisDT(dmi, breaks = 0:30/2, timeScale = "DMdur")
+  sm <- splitMulti(dmi, breaks = list(DMdur = 0:30/2))
+  
+  lex_attr_nms <- c("time.since", "breaks", "time.scales")
+  testthat::expect_identical(
+    attributes(Si)[lex_attr_nms], attributes(sm)[lex_attr_nms]
+  )
+  testthat::expect_identical(
+    attributes(Si)[lex_attr_nms], attributes(sldt)[lex_attr_nms]
+  )
+  
+})
+
+
+
+
+
diff --git a/tests/testthat/test_splitting_breaks.R b/tests/testthat/test_splitting_breaks.R
index 7842ff4..f33b544 100644
--- a/tests/testthat/test_splitting_breaks.R
+++ b/tests/testthat/test_splitting_breaks.R
@@ -1,80 +1,80 @@
-testthat::context("breaks attributes resulting from splitting")
-
-
-testthat::test_that("splitMulti produces intended breaks list", {
-  popEpi:::skip_normally()
-  x <- data.table(popEpi::sibr)
-  x <- x[dg_date < ex_date & bi_date < dg_date]
-  
-  x <- Lexis(
-    data = x, 
-    entry = list(
-      per = get.yrs(dg_date), fot = 0.0, 
-      age = get.yrs(dg_date)-get.yrs(bi_date)
-    ),
-    exit = list(
-      per = get.yrs(ex_date)
-    ), entry.status = 0L, exit.status = 1L
-  )
-  forceLexisDT(x, breaks = get_breaks(x), allScales = timeScales(x))
-  
-  BL <- list(fot = 2:8, per = 1990:2000, age = seq(0,100, 10))
-  xx <- splitMulti(x, breaks = BL, drop = TRUE)
-  
-  BL2 <- list(fot = 4:7, per = 1991:1999, age = seq(50,70, 10))
-  xxx <- splitMulti(xx, breaks = BL, drop = TRUE)
-  
-  testthat::expect_equal(breaks(xx, "fot"), BL$fot)
-  testthat::expect_equal(breaks(xx, "per"), BL$per)
-  testthat::expect_equal(breaks(xx, "age"), BL$age)
-  
-  testthat::expect_equal(breaks(xxx, "fot"), BL$fot)
-  testthat::expect_equal(breaks(xxx, "per"), BL$per)
-  testthat::expect_equal(breaks(xxx, "age"), BL$age)
-})
-
-
-
-testthat::test_that("splitLexisDT produces intended breaks list", {
-  popEpi:::skip_normally()
-  x <- data.table(popEpi::sibr)[dg_date < ex_date, ]
-  
-  x <- Lexis(
-    data = x, 
-    entry = list(
-      per = get.yrs(dg_date), fot = 0.0, 
-      age = get.yrs(dg_date)-get.yrs(bi_date)
-    ),
-    exit = list(
-      per = get.yrs(ex_date)
-    ), entry.status = 0L, exit.status = 1L
-  )
-  forceLexisDT(x, breaks = get_breaks(x), allScales = timeScales(x))
-  
-  br <- 2:10
-  xx <- splitLexisDT(x, breaks = br, timeScale = "fot", drop = TRUE)
-  
-  br2 <- 2:12
-  xxx <- splitLexisDT(xx, breaks = br, timeScale = "fot", drop = TRUE)
-  
-  testthat::expect_equal(breaks(xx, "fot"), br)
-  testthat::expect_equal(breaks(xxx, "fot"), unique(br, br2))
-  
-  
-  br <- 0:8
-  xx <- splitLexisDT(x, breaks = br, timeScale = "fot", drop = FALSE)
-  
-  br2 <- 2:10
-  xxx <- splitLexisDT(xx, breaks = br2, timeScale = "fot", drop = FALSE)
-  
-  br3 <- seq(9, 12, 0.5)
-  xxxx <- splitLexisDT(xxx, breaks = br3, timeScale = "fot", drop = FALSE)
-  
-  testthat::expect_equal(breaks(xx, "fot"), br)
-  testthat::expect_equal(breaks(xxx, "fot"), unique(c(br, br2)))
-  testthat::expect_equal(breaks(xxxx, "fot"), sort(unique(c(br, br3))))
-})
-
-
-
-
+testthat::context("breaks attributes resulting from splitting")
+
+
+testthat::test_that("splitMulti produces intended breaks list", {
+  popEpi:::skip_normally()
+  x <- data.table(popEpi::sibr)
+  x <- x[dg_date < ex_date & bi_date < dg_date]
+  
+  x <- Lexis(
+    data = x, 
+    entry = list(
+      per = get.yrs(dg_date), fot = 0.0, 
+      age = get.yrs(dg_date)-get.yrs(bi_date)
+    ),
+    exit = list(
+      per = get.yrs(ex_date)
+    ), entry.status = 0L, exit.status = 1L
+  )
+  forceLexisDT(x, breaks = get_breaks(x), allScales = timeScales(x))
+  
+  BL <- list(fot = 2:8, per = 1990:2000, age = seq(0,100, 10))
+  xx <- splitMulti(x, breaks = BL, drop = TRUE)
+  
+  BL2 <- list(fot = 4:7, per = 1991:1999, age = seq(50,70, 10))
+  xxx <- splitMulti(xx, breaks = BL, drop = TRUE)
+  
+  testthat::expect_equal(breaks(xx, "fot"), BL$fot)
+  testthat::expect_equal(breaks(xx, "per"), BL$per)
+  testthat::expect_equal(breaks(xx, "age"), BL$age)
+  
+  testthat::expect_equal(breaks(xxx, "fot"), BL$fot)
+  testthat::expect_equal(breaks(xxx, "per"), BL$per)
+  testthat::expect_equal(breaks(xxx, "age"), BL$age)
+})
+
+
+
+testthat::test_that("splitLexisDT produces intended breaks list", {
+  popEpi:::skip_normally()
+  x <- data.table(popEpi::sibr)[dg_date < ex_date, ]
+  
+  x <- Lexis(
+    data = x, 
+    entry = list(
+      per = get.yrs(dg_date), fot = 0.0, 
+      age = get.yrs(dg_date)-get.yrs(bi_date)
+    ),
+    exit = list(
+      per = get.yrs(ex_date)
+    ), entry.status = 0L, exit.status = 1L
+  )
+  forceLexisDT(x, breaks = get_breaks(x), allScales = timeScales(x))
+  
+  br <- 2:10
+  xx <- splitLexisDT(x, breaks = br, timeScale = "fot", drop = TRUE)
+  
+  br2 <- 2:12
+  xxx <- splitLexisDT(xx, breaks = br, timeScale = "fot", drop = TRUE)
+  
+  testthat::expect_equal(breaks(xx, "fot"), br)
+  testthat::expect_equal(breaks(xxx, "fot"), unique(br, br2))
+  
+  
+  br <- 0:8
+  xx <- splitLexisDT(x, breaks = br, timeScale = "fot", drop = FALSE)
+  
+  br2 <- 2:10
+  xxx <- splitLexisDT(xx, breaks = br2, timeScale = "fot", drop = FALSE)
+  
+  br3 <- seq(9, 12, 0.5)
+  xxxx <- splitLexisDT(xxx, breaks = br3, timeScale = "fot", drop = FALSE)
+  
+  testthat::expect_equal(breaks(xx, "fot"), br)
+  testthat::expect_equal(breaks(xxx, "fot"), unique(c(br, br2)))
+  testthat::expect_equal(breaks(xxxx, "fot"), sort(unique(c(br, br3))))
+})
+
+
+
+
diff --git a/tests/testthat/test_splitting_randomly_on_fixed_data.R b/tests/testthat/test_splitting_randomly_on_fixed_data.R
index 4d3a09c..8a41844 100644
--- a/tests/testthat/test_splitting_randomly_on_fixed_data.R
+++ b/tests/testthat/test_splitting_randomly_on_fixed_data.R
@@ -1,140 +1,140 @@
-testthat::context("Splitting tests on fixed data and random breaks")
-
-
-
-
-
-testthat::test_that("splitting funs congruent with random splitting and fixed data", {
-
-  popEpi:::skip_normally()
-
-  library("Epi")
-  library("data.table")
-
-  data("occup", package = "Epi")
-
-  occup <- Epi::Lexis(
-    entry = list(age = AoE, per = DoE),
-    exit = list(per = DoX),
-    entry.status = 0L,
-    exit.status = as.integer(Xst == "D"),
-    data = occup
-  )
-  
-  
-  data("DMlate", package = "Epi")
-  
-  DMlate <- Epi::Lexis(
-    entry = list(fot = 0, per = dodm, age = dodm-dobth),
-    exit = list(per = dox),
-    entry.status = 0L,
-    exit.status = as.integer(!is.na(dodth)),
-    data = DMlate[DMlate$dox>DMlate$dodm, ]
-  )
-  
-  data("DMrand", package = "Epi")
-  
-  DMrand <- Epi::Lexis(
-    entry = list(fot = 0, per = dodm, age = dodm-dobth),
-    exit = list(per = dox),
-    entry.status = 0L,
-    exit.status = as.integer(!is.na(dodth)),
-    data = DMrand[DMrand$dox>DMrand$dodm, ]
-  )
-  
-  data("thoro", package = "Epi")
-  
-  thoro <- Epi::Lexis(
-    entry = list(fot = 0L, per = injecdat, age = injecdat-birthdat),
-    exit = list(per = exitdat),
-    entry.status = 0L,
-    exit.status = as.integer(exitstat == 1),
-    data = thoro[thoro$injecdat < thoro$exitdat, ]
-  )
-  
-  sire <- setDT(copy(popEpi::sire))
-  sire[, "dg_yrs" := get.yrs(dg_date, "actual")]
-  sire[, "ex_yrs" := get.yrs(ex_date, "actual")]
-  sire[, "bi_yrs" := get.yrs(bi_date, "actual")]
-  sire[, "id":= 1:.N]
-  sire <- Lexis(
-    data=sire[dg_date < ex_date],
-    entry=list(fot=0, per=dg_yrs, age=dg_age),
-    exit=list(per=ex_yrs),
-    merge=TRUE,
-    exit.status = status,
-    entry.status = 0L
-  )
-
-
-  sibr <- setDT(copy(popEpi::sibr))
-  sibr[, "dg_yrs" := get.yrs(dg_date, "actual")]
-  sibr[, "ex_yrs" := get.yrs(ex_date, "actual")]
-  sibr[, "bi_yrs" := get.yrs(bi_date, "actual")]
-  sibr[, "id":= 1:.N]
-  sibr <- Lexis(
-    data=sibr[dg_date < ex_date],
-    entry=list(fot=0, per=dg_yrs, age=dg_age),
-    exit=list(per=ex_yrs),
-    merge=TRUE,
-    exit.status = status,
-    entry.status = 0L
-  )
-
-  lex_list <- list(
-    occup = occup,
-    DMlate = DMlate,
-    DMrand = DMrand,
-    thoro = thoro,
-    sire = sire,
-    sibr = sibr
-  )
-  lex_list[] <- lapply(lex_list, function(df) {
-    df$lex.id <- sample(df$lex.id, nrow(df), replace = FALSE)
-    df
-  })
-
-  n_random_splits <- 500
-  invisible(lapply(names(lex_list), function(lex_nm) {
-    lapply(1:n_random_splits, function(i) {
-      used_seed <- get_random_seed()
-      set.seed(used_seed)
-      l <- random_splitting_on(lex = lex_list[[lex_nm]], n.max.breaks = 50)
-      # list contents in order: Epi::splitLexis, splitLexisDT, splitMulti
-
-      lapply(2:length(l), function(list_pos) {
-
-        tt_msg <- paste0(
-          "Epi::splitLexis and ", c("splitLexisDT", "splitMulti")[list_pos-1],
-          " are in agreement in data '", lex_nm, "' using seed ", used_seed
-        )
-
-        testthat::test_that(tt_msg, {
-          testthat::expect_equal(l[[1]], l[[list_pos]], check.attributes = FALSE)
-        })
-
-      })
-
-      testthat::test_that(
-        paste0(
-          "splitLexisDT and splitMulti are in agreement in data '", lex_nm,
-          "' using seed ", used_seed
-        ),
-        testthat::expect_equal(l[[2]], l[[3]], check.attributes = TRUE)
-      )
-
-    })
-  }))
-
-
-
-})
-
-
-
-
-
-
-
-
-
+testthat::context("Splitting tests on fixed data and random breaks")
+
+
+
+
+
+testthat::test_that("splitting funs congruent with random splitting and fixed data", {
+
+  popEpi:::skip_normally()
+
+  library("Epi")
+  library("data.table")
+
+  data("occup", package = "Epi")
+
+  occup <- Epi::Lexis(
+    entry = list(age = AoE, per = DoE),
+    exit = list(per = DoX),
+    entry.status = 0L,
+    exit.status = as.integer(Xst == "D"),
+    data = occup
+  )
+  
+  
+  data("DMlate", package = "Epi")
+  
+  DMlate <- Epi::Lexis(
+    entry = list(fot = 0, per = dodm, age = dodm-dobth),
+    exit = list(per = dox),
+    entry.status = 0L,
+    exit.status = as.integer(!is.na(dodth)),
+    data = DMlate[DMlate$dox>DMlate$dodm, ]
+  )
+  
+  data("DMrand", package = "Epi")
+  
+  DMrand <- Epi::Lexis(
+    entry = list(fot = 0, per = dodm, age = dodm-dobth),
+    exit = list(per = dox),
+    entry.status = 0L,
+    exit.status = as.integer(!is.na(dodth)),
+    data = DMrand[DMrand$dox>DMrand$dodm, ]
+  )
+  
+  data("thoro", package = "Epi")
+  
+  thoro <- Epi::Lexis(
+    entry = list(fot = 0L, per = injecdat, age = injecdat-birthdat),
+    exit = list(per = exitdat),
+    entry.status = 0L,
+    exit.status = as.integer(exitstat == 1),
+    data = thoro[thoro$injecdat < thoro$exitdat, ]
+  )
+  
+  sire <- setDT(copy(popEpi::sire))
+  sire[, "dg_yrs" := get.yrs(dg_date, "actual")]
+  sire[, "ex_yrs" := get.yrs(ex_date, "actual")]
+  sire[, "bi_yrs" := get.yrs(bi_date, "actual")]
+  sire[, "id":= 1:.N]
+  sire <- Lexis(
+    data=sire[dg_date < ex_date],
+    entry=list(fot=0, per=dg_yrs, age=dg_age),
+    exit=list(per=ex_yrs),
+    merge=TRUE,
+    exit.status = status,
+    entry.status = 0L
+  )
+
+
+  sibr <- setDT(copy(popEpi::sibr))
+  sibr[, "dg_yrs" := get.yrs(dg_date, "actual")]
+  sibr[, "ex_yrs" := get.yrs(ex_date, "actual")]
+  sibr[, "bi_yrs" := get.yrs(bi_date, "actual")]
+  sibr[, "id":= 1:.N]
+  sibr <- Lexis(
+    data=sibr[dg_date < ex_date],
+    entry=list(fot=0, per=dg_yrs, age=dg_age),
+    exit=list(per=ex_yrs),
+    merge=TRUE,
+    exit.status = status,
+    entry.status = 0L
+  )
+
+  lex_list <- list(
+    occup = occup,
+    DMlate = DMlate,
+    DMrand = DMrand,
+    thoro = thoro,
+    sire = sire,
+    sibr = sibr
+  )
+  lex_list[] <- lapply(lex_list, function(df) {
+    df$lex.id <- sample(df$lex.id, nrow(df), replace = FALSE)
+    df
+  })
+
+  n_random_splits <- 500
+  invisible(lapply(names(lex_list), function(lex_nm) {
+    lapply(1:n_random_splits, function(i) {
+      used_seed <- get_random_seed()
+      set.seed(used_seed)
+      l <- random_splitting_on(lex = lex_list[[lex_nm]], n.max.breaks = 50)
+      # list contents in order: Epi::splitLexis, splitLexisDT, splitMulti
+
+      lapply(2:length(l), function(list_pos) {
+
+        tt_msg <- paste0(
+          "Epi::splitLexis and ", c("splitLexisDT", "splitMulti")[list_pos-1],
+          " are in agreement in data '", lex_nm, "' using seed ", used_seed
+        )
+
+        testthat::test_that(tt_msg, {
+          testthat::expect_equal(l[[1]], l[[list_pos]], check.attributes = FALSE)
+        })
+
+      })
+
+      testthat::test_that(
+        paste0(
+          "splitLexisDT and splitMulti are in agreement in data '", lex_nm,
+          "' using seed ", used_seed
+        ),
+        testthat::expect_equal(l[[2]], l[[3]], check.attributes = TRUE)
+      )
+
+    })
+  }))
+
+
+
+})
+
+
+
+
+
+
+
+
+
diff --git a/tests/testthat/test_splitting_randomly_on_random_data.R b/tests/testthat/test_splitting_randomly_on_random_data.R
index a408f7e..494aecf 100644
--- a/tests/testthat/test_splitting_randomly_on_random_data.R
+++ b/tests/testthat/test_splitting_randomly_on_random_data.R
@@ -1,60 +1,60 @@
-testthat::context("Splitting tests on random data and random breaks")
-
-
-
-
-
-testthat::test_that("splitting funs congruent with random splitting and random data", {
-
-  popEpi:::skip_normally()
-
-  library("Epi")
-  library("data.table")
-
-  n_datasets <- 100
-  n_random_splits <- 50
-  lapply(seq_len(n_datasets), function(lex_nm) {
-
-    data_seed <- get_random_seed()
-    lex <- random_Lexis(
-      n.rows = 100:2000,
-      n.time.scales = 1:10,
-      n.statuses = 2:10,
-      n.other.vars = 1
-    )
-
-    lapply(seq_len(n_random_splits), function(i) {
-      split_seed <- get_random_seed()
-      set.seed(split_seed)
-      l <- random_splitting_on(lex = lex, n.max.breaks = 50)
-      # list contents in order: Epi::splitLexis, splitLexisDT, splitMulti
-
-      lapply(2:length(l), function(list_pos) {
-
-        tt_msg <- paste0(
-          "Epi::splitLexis and ", c("splitLexisDT", "splitMulti")[list_pos-1],
-          " are in agreement in data '", lex_nm,
-          "' using data seed ", data_seed, " and splitting seed ", split_seed
-        )
-
-        testthat::test_that(tt_msg, {
-          testthat::expect_equal(l[[1]], l[[list_pos]], check.attributes = FALSE)
-        })
-
-      })
-
-      testthat::test_that(
-        paste0(
-          "splitLexisDT and splitMulti are in agreement in data '", lex_nm,
-          "' using data seed ", data_seed, " and splitting seed ", split_seed
-        ),
-        testthat::expect_equal(l[[2]], l[[3]], check.attributes = TRUE)
-      )
-
-    })
-  })
-
-
-
-})
-
+testthat::context("Splitting tests on random data and random breaks")
+
+
+
+
+
+testthat::test_that("splitting funs congruent with random splitting and random data", {
+
+  popEpi:::skip_normally()
+
+  library("Epi")
+  library("data.table")
+
+  n_datasets <- 100
+  n_random_splits <- 50
+  lapply(seq_len(n_datasets), function(lex_nm) {
+
+    data_seed <- get_random_seed()
+    lex <- random_Lexis(
+      n.rows = 100:2000,
+      n.time.scales = 1:10,
+      n.statuses = 2:10,
+      n.other.vars = 1
+    )
+
+    lapply(seq_len(n_random_splits), function(i) {
+      split_seed <- get_random_seed()
+      set.seed(split_seed)
+      l <- random_splitting_on(lex = lex, n.max.breaks = 50)
+      # list contents in order: Epi::splitLexis, splitLexisDT, splitMulti
+
+      lapply(2:length(l), function(list_pos) {
+
+        tt_msg <- paste0(
+          "Epi::splitLexis and ", c("splitLexisDT", "splitMulti")[list_pos-1],
+          " are in agreement in data '", lex_nm,
+          "' using data seed ", data_seed, " and splitting seed ", split_seed
+        )
+
+        testthat::test_that(tt_msg, {
+          testthat::expect_equal(l[[1]], l[[list_pos]], check.attributes = FALSE)
+        })
+
+      })
+
+      testthat::test_that(
+        paste0(
+          "splitLexisDT and splitMulti are in agreement in data '", lex_nm,
+          "' using data seed ", data_seed, " and splitting seed ", split_seed
+        ),
+        testthat::expect_equal(l[[2]], l[[3]], check.attributes = TRUE)
+      )
+
+    })
+  })
+
+
+
+})
+
diff --git a/tests/testthat/test_survmean.R b/tests/testthat/test_survmean.R
index 4a26725..c6b3490 100644
--- a/tests/testthat/test_survmean.R
+++ b/tests/testthat/test_survmean.R
@@ -1,301 +1,301 @@
-testthat::context("mean survival testing")
-library("data.table")
-library("Epi")
-
-testthat::test_that("survmean() agrees with old results", {
-  popEpi:::skip_normally()
-  
-  sr <- data.table(popEpi::sire)[dg_date < ex_date, ]
-  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = sr,
-             exit.status = factor(status, levels = 0:2,
-                                  labels = c("alive", "canD", "othD")),
-             entry.status = factor(0, levels = 0:2,
-                                   labels = c("alive", "canD", "othD")),
-             merge = TRUE)
-  
-  ## observed survival
-  pm <- data.table(popEpi::popmort)
-  names(pm) <- c("sex", "CAL", "AGE", "haz")
-  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ agegr,
-                 pophaz = pm, data = x,
-                 breaks = list(FUT = seq(0, 10, 1/12)),
-                 e1.breaks = list(FUT = c(seq(0, 10, 1/12), 11:100)))
-  
-  ## values to test against computed on 2016-03-04;
-  ## git ref: 5077677
-  testthat::expect_equal(sm$est, c(33.951439, 21.611419,  7.604318), tol = 0.005, scale = 1)
-  testthat::expect_equal(sm$exp, c(45.25686, 31.22712, 13.06725), tol = 0.005, scale = 1)
-  
-  
-})
-
-
-
-# test_that("survmean() agrees with results computed using pkg survival", {
-#   popEpi:::skip_normally()
-#   
-#   
-#   BL <- list(fot= seq(0,15,1/24))
-#   eBL <- list(fot = unique(c(BL$fot, seq(15, 115,0.5))))
-#   
-#   sire2 <- data.table(popEpi::sire)[dg_date<ex_date, ]
-#   sire2$statusf <- factor(sire2$status, levels = 0:2, 
-#                           labels = c("alive", "canD", "othD"))
-#   
-#   x <- lexpand(sire2, 
-#                birth  = bi_date, entry = dg_date, exit = ex_date,
-#                status = statusf,
-#                breaks = NULL)
-#   
-#   popmort_sm <- setDT(copy(popEpi::popmort))
-#   setnames(popmort_sm, c("agegroup", "year"), c("age", "per"))
-#   
-#   sm <- survmean(Surv(fot, event = lex.Xst) ~ 1, 
-#                  breaks = BL, e1.breaks = eBL, r = 1,
-#                  pophaz = popmort_sm, data = x)
-#   
-#   st <- survtab(Surv(fot, event = lex.Xst) ~ 1, 
-#                 breaks = BL, 
-#                 pophaz = popmort_sm, data = x)
-#   
-#   sm_curve <- copy(attr(sm, "survmean.meta")$curve)
-#   td01 <- readRDS("tests/testthat/survmean_test_data_01.rds")
-#   
-#   expect_equal(sm_curve[survmean_type == "est", surv], 
-#                td01$est, 
-#                scale = 1L, 
-#                tolerance = 0.001)
-# })
-
-
-testthat::test_that("survmean expected survival curve corresponds to full Ederer I", {
-  popEpi:::skip_normally()
-  
-  sr <- data.table(sire)[dg_date < ex_date, ]
-  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = sr,
-             exit.status = factor(status, levels = 0:2,
-                                  labels = c("alive", "canD", "othD")),
-             entry.status = factor(0, levels = 0:2,
-                                   labels = c("alive", "canD", "othD")),
-             merge = TRUE)
-  
-  pm <- copy(popEpi::popmort)
-  names(pm) <- c("sex", "CAL", "AGE", "haz")
-  
-  BL <- list(FUT = seq(0, 10, 1/12))
-  eBL <- list(FUT = c(BL$FUT, seq(11,110,1/2)))
-  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
-                 pophaz = pm, data = x,
-                 breaks = BL, 
-                 e1.breaks = eBL)
-  
-  ## pure Ederer I curve
-  setDT(x)
-  x[, lex.dur := 110]
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  e1 <- comp_e1(x = x, breaks = eBL, 
-                pophaz = pm, survScale = "FUT")
-  setkeyv(e1, "FUT")
-  e1[, "delta" := diff(eBL$FUT)]
-  e1[, "l1" := c(1,surv.exp[-.N])]
-  sm.e1 <- e1[, sum((l1+surv.exp)/2L*delta)]
-  
-  
-  testthat::expect_equal(sm$exp, sm.e1, tol = 0.0005, scale = 1)
-  
-})
-
-testthat::test_that("survmean period method is useful", {
-  popEpi:::skip_normally()
-  
-  sr <- data.table(popEpi::sire)[dg_date < ex_date, ]
-  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = sr,
-             exit.status = factor(status, levels = 0:2,
-                                  labels = c("alive", "canD", "othD")),
-             entry.status = factor(0, levels = 0:2,
-                                   labels = c("alive", "canD", "othD")),
-             merge = TRUE)
-  
-  pm <- data.table(popEpi::popmort)
-  names(pm) <- c("sex", "CAL", "AGE", "haz")
-  
-  BL <- list(FUT = seq(0, 10, 1/12))
-  eBL <- list(FUT = c(BL$FUT, seq(max(BL$FUT),110,1/2)))
-  eBL$FUT <- sort(unique(eBL$FUT))
-  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
-                 subset = dg_date >= "1998-01-01" & dg_date < "2003-01-01",
-                 pophaz = pm, data = x,
-                 r = 1,
-                 breaks = BL, 
-                 e1.breaks = eBL)
-  
-  
-  BL <- list(FUT = seq(0, 5, 1/12), CAL = c(1998,2003))
-  eBL <- list(FUT = c(BL$FUT, seq(max(BL$FUT), 10, 1/12), seq(10,110,1/2)))
-  eBL$FUT <- sort(unique(eBL$FUT))
-  smp <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
-                  pophaz = pm, data = x,
-                  breaks = BL, 
-                  r = 1,
-                  e1.breaks = eBL)
-  
-  
-  
-  testthat::expect_equal(sm$obs, smp$obs)
-  testthat::expect_equal(sm$exp, smp$exp)
-  testthat::expect_equal(smp$est, 10.01542, tol = 0.0005, scale = 1)
-  testthat::expect_equal(sm$est, 10.0216, tol = 0.0005, scale = 1)
-  
-})
-
-
-
-testthat::test_that("Dates and frac. yrs produce congruent results", {
-  popEpi:::skip_normally()
-  
-  x <- data.table(popEpi::sire)
-  x <- x[dg_date<ex_date]
-  
-  ## phony group variable
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-  
-  
-  ## yrs
-  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-              exit = list(CAL = get.yrs(ex_date)), 
-              data = x,
-              exit.status = factor(status, levels = 0:2, 
-                                   labels = c("alive", "canD", "othD")), 
-              entry.status = factor(0, levels = 0:2,
-                                    labels = c("alive", "canD", "othD")),
-              merge = TRUE)
-  
-  ## dates
-  xd <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-              exit = list(CAL = ex_date),
-              data = x,
-              exit.status = factor(status, levels = 0:2, 
-                                   labels = c("alive", "canD", "othD")), 
-              entry.status = factor(0, levels = 0:2,
-                                    labels = c("alive", "canD", "othD")),
-              merge = TRUE)
-  yd <- 365.242199
-  BLy <- list(FUT = seq(0, 9, 1/4))
-  BLd <- lapply(BLy, function(el) el * yd)
-  eBLy <- list(FUT = c(seq(0, 9, 1/4), 9.5, 10:75))
-  eBLd <- lapply(eBLy, function(el) el * yd)
-  
-  pmy <- data.table(popEpi::popmort)
-  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
-  
-  pmd <- data.table(pmy)
-  pmd[, CAL := as.Date(paste0(CAL, "-01-01"))]
-  pmd[, AGE := AGE * yd]
-  pmd[, haz := haz/yd]
-  
-  #### hazard method
-  ## observed survival & Ederer II
-  
-  sty <- survmean(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                  surv.method = "hazard",
-                  e1.breaks = eBLy,
-                  breaks = BLy, pophaz = pmy)
-  
-  std <- survmean(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                  surv.method = "hazard",
-                  e1.breaks = eBLd,
-                  breaks = BLd, pophaz = pmd)    
-  cuy <- data.table(attributes(sty)$survmean.meta$curves)
-  cud <- data.table(attributes(std)$survmean.meta$curves)
-  
-  std[, c("est", "exp", "YPLL") := lapply(.SD, function(col) col/yd),
-      .SDcols = c("est", "exp", "YPLL")]
-  
-  testthat::expect_equal(sty$est, std$est, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$exp, std$exp, scale = 1L, tolerance = 0.001)
-  testthat::expect_equal(sty$obs, std$obs)
-  
-  testthat::expect_equal(cuy$surv, cud$surv, scale = 1L, tolerance = 0.00005)
-  
-  #### lifetable method
-  ## observed survival & Ederer II
-  
-  sty <- survmean(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                  surv.method = "lifetable",
-                  e1.breaks = eBLy,
-                  breaks = BLy, pophaz = pmy)
-  
-  std <- survmean(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                  surv.method = "lifetable",
-                  e1.breaks = eBLd,
-                  breaks = BLd, pophaz = pmd)    
-  cuy <- data.table(attributes(sty)$survmean.meta$curves)
-  cud <- data.table(attributes(std)$survmean.meta$curves)
-  
-  std[, c("est", "exp", "YPLL") := lapply(.SD, function(col) col/yd),
-      .SDcols = c("est", "exp", "YPLL")]
-  
-  testthat::expect_equal(sty$est, std$est, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$exp, std$exp, scale = 1L, tolerance = 0.001)
-  testthat::expect_equal(sty$obs, std$obs)
-  
-  testthat::expect_equal(cuy$surv, cud$surv, scale = 1L, tolerance = 0.00005)
-  
-  
-})
-
-
-
-
-testthat::test_that("updating works for survmean objects", {
-  
-  popEpi:::skip_normally()
-  
-  set.seed(1)
-  sr <- setDT(popEpi::sire[sample(1:.N, 100), ])
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = sr,
-             exit.status = factor(status, levels = 0:2,
-                                  labels = c("alive", "canD", "othD")),
-             entry.status = factor(0, levels = 0:2,
-                                   labels = c("alive", "canD", "othD")),
-             merge = TRUE)
-  
-  pm <- data.table(popEpi::popmort)
-  names(pm) <- c("sex", "CAL", "AGE", "haz")
-  
-  BL <- list(FUT = 1:10)
-  fo <- Surv(time = FUT, event = lex.Xst != "alive") ~ 1
-  e <- quote(
-    survmean(formula = fo, data = x, breaks = BL,pophaz = pm)
-  )
-  sm <- eval(e)
-  
-  fo <- update(fo, ~.+sex)
-  sm2 <- eval(e)
-  
-  sm3 <- update(sm, .~ + sex)
-  
-  testthat::expect_equal(data.frame(sm2), data.frame(sm3))
-  testthat::expect_equal(formula(sm2), fo)
-  testthat::expect_equal(formula(sm3), fo)
-  testthat::expect_equal(getCall(sm2), e)
-  
-})
-
-
+testthat::context("mean survival testing")
+library("data.table")
+library("Epi")
+
+testthat::test_that("survmean() agrees with old results", {
+  popEpi:::skip_normally()
+  
+  sr <- data.table(popEpi::sire)[dg_date < ex_date, ]
+  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = sr,
+             exit.status = factor(status, levels = 0:2,
+                                  labels = c("alive", "canD", "othD")),
+             entry.status = factor(0, levels = 0:2,
+                                   labels = c("alive", "canD", "othD")),
+             merge = TRUE)
+  
+  ## observed survival
+  pm <- data.table(popEpi::popmort)
+  names(pm) <- c("sex", "CAL", "AGE", "haz")
+  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ agegr,
+                 pophaz = pm, data = x,
+                 breaks = list(FUT = seq(0, 10, 1/12)),
+                 e1.breaks = list(FUT = c(seq(0, 10, 1/12), 11:100)))
+  
+  ## values to test against computed on 2016-03-04;
+  ## git ref: 5077677
+  testthat::expect_equal(sm$est, c(33.951439, 21.611419,  7.604318), tol = 0.005, scale = 1)
+  testthat::expect_equal(sm$exp, c(45.25686, 31.22712, 13.06725), tol = 0.005, scale = 1)
+  
+  
+})
+
+
+
+# test_that("survmean() agrees with results computed using pkg survival", {
+#   popEpi:::skip_normally()
+#   
+#   
+#   BL <- list(fot= seq(0,15,1/24))
+#   eBL <- list(fot = unique(c(BL$fot, seq(15, 115,0.5))))
+#   
+#   sire2 <- data.table(popEpi::sire)[dg_date<ex_date, ]
+#   sire2$statusf <- factor(sire2$status, levels = 0:2, 
+#                           labels = c("alive", "canD", "othD"))
+#   
+#   x <- lexpand(sire2, 
+#                birth  = bi_date, entry = dg_date, exit = ex_date,
+#                status = statusf,
+#                breaks = NULL)
+#   
+#   popmort_sm <- setDT(copy(popEpi::popmort))
+#   setnames(popmort_sm, c("agegroup", "year"), c("age", "per"))
+#   
+#   sm <- survmean(Surv(fot, event = lex.Xst) ~ 1, 
+#                  breaks = BL, e1.breaks = eBL, r = 1,
+#                  pophaz = popmort_sm, data = x)
+#   
+#   st <- survtab(Surv(fot, event = lex.Xst) ~ 1, 
+#                 breaks = BL, 
+#                 pophaz = popmort_sm, data = x)
+#   
+#   sm_curve <- copy(attr(sm, "survmean.meta")$curve)
+#   td01 <- readRDS("tests/testthat/survmean_test_data_01.rds")
+#   
+#   expect_equal(sm_curve[survmean_type == "est", surv], 
+#                td01$est, 
+#                scale = 1L, 
+#                tolerance = 0.001)
+# })
+
+
+testthat::test_that("survmean expected survival curve corresponds to full Ederer I", {
+  popEpi:::skip_normally()
+  
+  sr <- data.table(sire)[dg_date < ex_date, ]
+  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = sr,
+             exit.status = factor(status, levels = 0:2,
+                                  labels = c("alive", "canD", "othD")),
+             entry.status = factor(0, levels = 0:2,
+                                   labels = c("alive", "canD", "othD")),
+             merge = TRUE)
+  
+  pm <- copy(popEpi::popmort)
+  names(pm) <- c("sex", "CAL", "AGE", "haz")
+  
+  BL <- list(FUT = seq(0, 10, 1/12))
+  eBL <- list(FUT = c(BL$FUT, seq(11,110,1/2)))
+  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
+                 pophaz = pm, data = x,
+                 breaks = BL, 
+                 e1.breaks = eBL)
+  
+  ## pure Ederer I curve
+  setDT(x)
+  x[, lex.dur := 110]
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  e1 <- comp_e1(x = x, breaks = eBL, 
+                pophaz = pm, survScale = "FUT")
+  setkeyv(e1, "FUT")
+  e1[, "delta" := diff(eBL$FUT)]
+  e1[, "l1" := c(1,surv.exp[-.N])]
+  sm.e1 <- e1[, sum((l1+surv.exp)/2L*delta)]
+  
+  
+  testthat::expect_equal(sm$exp, sm.e1, tol = 0.0005, scale = 1)
+  
+})
+
+testthat::test_that("survmean period method is useful", {
+  popEpi:::skip_normally()
+  
+  sr <- data.table(popEpi::sire)[dg_date < ex_date, ]
+  sr$agegr <- cut(sr$dg_age, c(0,45,60,Inf), right=FALSE)
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = sr,
+             exit.status = factor(status, levels = 0:2,
+                                  labels = c("alive", "canD", "othD")),
+             entry.status = factor(0, levels = 0:2,
+                                   labels = c("alive", "canD", "othD")),
+             merge = TRUE)
+  
+  pm <- data.table(popEpi::popmort)
+  names(pm) <- c("sex", "CAL", "AGE", "haz")
+  
+  BL <- list(FUT = seq(0, 10, 1/12))
+  eBL <- list(FUT = c(BL$FUT, seq(max(BL$FUT),110,1/2)))
+  eBL$FUT <- sort(unique(eBL$FUT))
+  sm <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
+                 subset = dg_date >= "1998-01-01" & dg_date < "2003-01-01",
+                 pophaz = pm, data = x,
+                 r = 1,
+                 breaks = BL, 
+                 e1.breaks = eBL)
+  
+  
+  BL <- list(FUT = seq(0, 5, 1/12), CAL = c(1998,2003))
+  eBL <- list(FUT = c(BL$FUT, seq(max(BL$FUT), 10, 1/12), seq(10,110,1/2)))
+  eBL$FUT <- sort(unique(eBL$FUT))
+  smp <- survmean(Surv(time = FUT, event = lex.Xst != "alive") ~ 1,
+                  pophaz = pm, data = x,
+                  breaks = BL, 
+                  r = 1,
+                  e1.breaks = eBL)
+  
+  
+  
+  testthat::expect_equal(sm$obs, smp$obs)
+  testthat::expect_equal(sm$exp, smp$exp)
+  testthat::expect_equal(smp$est, 10.01542, tol = 0.0005, scale = 1)
+  testthat::expect_equal(sm$est, 10.0216, tol = 0.0005, scale = 1)
+  
+})
+
+
+
+testthat::test_that("Dates and frac. yrs produce congruent results", {
+  popEpi:::skip_normally()
+  
+  x <- data.table(popEpi::sire)
+  x <- x[dg_date<ex_date]
+  
+  ## phony group variable
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+  
+  
+  ## yrs
+  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+              exit = list(CAL = get.yrs(ex_date)), 
+              data = x,
+              exit.status = factor(status, levels = 0:2, 
+                                   labels = c("alive", "canD", "othD")), 
+              entry.status = factor(0, levels = 0:2,
+                                    labels = c("alive", "canD", "othD")),
+              merge = TRUE)
+  
+  ## dates
+  xd <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+              exit = list(CAL = ex_date),
+              data = x,
+              exit.status = factor(status, levels = 0:2, 
+                                   labels = c("alive", "canD", "othD")), 
+              entry.status = factor(0, levels = 0:2,
+                                    labels = c("alive", "canD", "othD")),
+              merge = TRUE)
+  yd <- 365.242199
+  BLy <- list(FUT = seq(0, 9, 1/4))
+  BLd <- lapply(BLy, function(el) el * yd)
+  eBLy <- list(FUT = c(seq(0, 9, 1/4), 9.5, 10:75))
+  eBLd <- lapply(eBLy, function(el) el * yd)
+  
+  pmy <- data.table(popEpi::popmort)
+  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
+  
+  pmd <- data.table(pmy)
+  pmd[, CAL := as.Date(paste0(CAL, "-01-01"))]
+  pmd[, AGE := AGE * yd]
+  pmd[, haz := haz/yd]
+  
+  #### hazard method
+  ## observed survival & Ederer II
+  
+  sty <- survmean(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                  surv.method = "hazard",
+                  e1.breaks = eBLy,
+                  breaks = BLy, pophaz = pmy)
+  
+  std <- survmean(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                  surv.method = "hazard",
+                  e1.breaks = eBLd,
+                  breaks = BLd, pophaz = pmd)    
+  cuy <- data.table(attributes(sty)$survmean.meta$curves)
+  cud <- data.table(attributes(std)$survmean.meta$curves)
+  
+  std[, c("est", "exp", "YPLL") := lapply(.SD, function(col) col/yd),
+      .SDcols = c("est", "exp", "YPLL")]
+  
+  testthat::expect_equal(sty$est, std$est, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$exp, std$exp, scale = 1L, tolerance = 0.001)
+  testthat::expect_equal(sty$obs, std$obs)
+  
+  testthat::expect_equal(cuy$surv, cud$surv, scale = 1L, tolerance = 0.00005)
+  
+  #### lifetable method
+  ## observed survival & Ederer II
+  
+  sty <- survmean(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                  surv.method = "lifetable",
+                  e1.breaks = eBLy,
+                  breaks = BLy, pophaz = pmy)
+  
+  std <- survmean(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                  surv.method = "lifetable",
+                  e1.breaks = eBLd,
+                  breaks = BLd, pophaz = pmd)    
+  cuy <- data.table(attributes(sty)$survmean.meta$curves)
+  cud <- data.table(attributes(std)$survmean.meta$curves)
+  
+  std[, c("est", "exp", "YPLL") := lapply(.SD, function(col) col/yd),
+      .SDcols = c("est", "exp", "YPLL")]
+  
+  testthat::expect_equal(sty$est, std$est, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$exp, std$exp, scale = 1L, tolerance = 0.001)
+  testthat::expect_equal(sty$obs, std$obs)
+  
+  testthat::expect_equal(cuy$surv, cud$surv, scale = 1L, tolerance = 0.00005)
+  
+  
+})
+
+
+
+
+testthat::test_that("updating works for survmean objects", {
+  
+  popEpi:::skip_normally()
+  
+  set.seed(1)
+  sr <- setDT(popEpi::sire[sample(1:.N, 100), ])
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = sr,
+             exit.status = factor(status, levels = 0:2,
+                                  labels = c("alive", "canD", "othD")),
+             entry.status = factor(0, levels = 0:2,
+                                   labels = c("alive", "canD", "othD")),
+             merge = TRUE)
+  
+  pm <- data.table(popEpi::popmort)
+  names(pm) <- c("sex", "CAL", "AGE", "haz")
+  
+  BL <- list(FUT = 1:10)
+  fo <- Surv(time = FUT, event = lex.Xst != "alive") ~ 1
+  e <- quote(
+    survmean(formula = fo, data = x, breaks = BL,pophaz = pm)
+  )
+  sm <- eval(e)
+  
+  fo <- update(fo, ~.+sex)
+  sm2 <- eval(e)
+  
+  sm3 <- update(sm, .~ + sex)
+  
+  testthat::expect_equal(data.frame(sm2), data.frame(sm3))
+  testthat::expect_equal(formula(sm2), fo)
+  testthat::expect_equal(formula(sm3), fo)
+  testthat::expect_equal(getCall(sm2), e)
+  
+})
+
+
diff --git a/tests/testthat/test_survtab_adjusted.R b/tests/testthat/test_survtab_adjusted.R
index 7a72907..1d21495 100644
--- a/tests/testthat/test_survtab_adjusted.R
+++ b/tests/testthat/test_survtab_adjusted.R
@@ -1,77 +1,77 @@
-testthat::context("adjusted curves from survtab")
-
-testthat::test_that("adjusted SEs and curves are what expected", {
-  popEpi:::skip_normally()
-  
-  sire <- data.table::data.table(popEpi::sire)
-  
-  ## calculate age-standardised 5-year relative survival ratio using
-  ## Ederer II method and period approach
-  
-  sire$agegr <- cut(sire$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
-  BL <- list(fot=seq(0, 5, by = 1/12),
-             per = c("2008-01-01", "2013-01-01"))
-  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks = BL,
-               pophaz = popmort,
-               aggre = list(agegr, fot))
-  
-  ## age standardisation using internal weights (age distribution of
-  ## patients diagnosed within the period window)
-  ## (NOTE: what is done here is equivalent to using weights = "internal")
-  w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0, ], FUN = sum)
-  names(w) <- c("agegr", "weights")
-  
-  st_raw <- survtab_ag(fot ~ agegr, data = x)
-  st_raw_pp <- survtab_ag(fot ~ agegr, data = x,  
-                          relsurv.method = "pp", 
-                          d.pp = "from0to1.pp", 
-                          d.pp.2 = "from0to1.pp.2", 
-                          n.cens.pp = "from0to0.pp", 
-                          pyrs.pp = "ptime.pp",
-                          n.cens = "from0to0")
-  
-  st_adj <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
-  st_adj_pp <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w, 
-                          relsurv.method = "pp", 
-                          d.pp = "from0to1.pp", 
-                          d.pp.2 = "from0to1.pp.2", 
-                          n.cens.pp = "from0to0.pp", 
-                          pyrs.pp = "ptime.pp",
-                          n.cens = "from0to0")
-  
-  dt_raw <- data.table::data.table(st_raw)
-  dt_raw_pp <- data.table::data.table(st_raw_pp)
-  data.table::set(
-    dt_raw, j = c("r.pp", "SE.r.pp"), 
-    value = mget(c("r.pp", "SE.r.pp"), as.environment(dt_raw_pp))
-  )
-  
-  dt_adj <- data.table::data.table(st_adj)
-  dt_adj_pp <- data.table::data.table(st_adj_pp)
-  data.table::set(
-    dt_adj, j = c("r.pp.as", "SE.r.pp.as"), 
-    value = mget(c("r.pp.as", "SE.r.pp.as"), as.environment(dt_adj_pp))
-  )
-  
-  
-  w$weights <- w$weights/sum(w$weights)
-  dt_raw <- merge(dt_raw, w, by = "agegr", all = TRUE)
-  dt_raw_adj <- dt_raw[, .("surv.obs.as" = sum(weights * surv.obs), 
-                           "SE.surv.obs.as" = sqrt(sum((weights^2)*(SE.surv.obs^2))),
-                           "r.e2.as" = sum(weights * r.e2), 
-                           "SE.r.e2.as" = sqrt(sum((weights^2)*(SE.r.e2^2))),
-                           "r.pp.as" = sum(weights * r.pp), 
-                           "SE.r.pp.as" = sqrt(sum((weights^2)*(SE.r.pp^2))),
-                           "pyrs" = sum(pyrs)), 
-                       keyby = "surv.int"]
-  
-  testthat::expect_equal(
-    dt_raw_adj, dt_adj[, .SD, .SDcols = names(dt_raw_adj)]
-  )
-  
-  
-  
-})
-
+testthat::context("adjusted curves from survtab")
+
+testthat::test_that("adjusted SEs and curves are what expected", {
+  popEpi:::skip_normally()
+  
+  sire <- data.table::data.table(popEpi::sire)
+  
+  ## calculate age-standardised 5-year relative survival ratio using
+  ## Ederer II method and period approach
+  
+  sire$agegr <- cut(sire$dg_age, c(0,45,55,65,75,Inf), right = FALSE)
+  BL <- list(fot=seq(0, 5, by = 1/12),
+             per = c("2008-01-01", "2013-01-01"))
+  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks = BL,
+               pophaz = popmort,
+               aggre = list(agegr, fot))
+  
+  ## age standardisation using internal weights (age distribution of
+  ## patients diagnosed within the period window)
+  ## (NOTE: what is done here is equivalent to using weights = "internal")
+  w <- aggregate(at.risk ~ agegr, data = x[x$fot == 0, ], FUN = sum)
+  names(w) <- c("agegr", "weights")
+  
+  st_raw <- survtab_ag(fot ~ agegr, data = x)
+  st_raw_pp <- survtab_ag(fot ~ agegr, data = x,  
+                          relsurv.method = "pp", 
+                          d.pp = "from0to1.pp", 
+                          d.pp.2 = "from0to1.pp.2", 
+                          n.cens.pp = "from0to0.pp", 
+                          pyrs.pp = "ptime.pp",
+                          n.cens = "from0to0")
+  
+  st_adj <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w)
+  st_adj_pp <- survtab_ag(fot ~ adjust(agegr), data = x, weights = w, 
+                          relsurv.method = "pp", 
+                          d.pp = "from0to1.pp", 
+                          d.pp.2 = "from0to1.pp.2", 
+                          n.cens.pp = "from0to0.pp", 
+                          pyrs.pp = "ptime.pp",
+                          n.cens = "from0to0")
+  
+  dt_raw <- data.table::data.table(st_raw)
+  dt_raw_pp <- data.table::data.table(st_raw_pp)
+  data.table::set(
+    dt_raw, j = c("r.pp", "SE.r.pp"), 
+    value = mget(c("r.pp", "SE.r.pp"), as.environment(dt_raw_pp))
+  )
+  
+  dt_adj <- data.table::data.table(st_adj)
+  dt_adj_pp <- data.table::data.table(st_adj_pp)
+  data.table::set(
+    dt_adj, j = c("r.pp.as", "SE.r.pp.as"), 
+    value = mget(c("r.pp.as", "SE.r.pp.as"), as.environment(dt_adj_pp))
+  )
+  
+  
+  w$weights <- w$weights/sum(w$weights)
+  dt_raw <- merge(dt_raw, w, by = "agegr", all = TRUE)
+  dt_raw_adj <- dt_raw[, .("surv.obs.as" = sum(weights * surv.obs), 
+                           "SE.surv.obs.as" = sqrt(sum((weights^2)*(SE.surv.obs^2))),
+                           "r.e2.as" = sum(weights * r.e2), 
+                           "SE.r.e2.as" = sqrt(sum((weights^2)*(SE.r.e2^2))),
+                           "r.pp.as" = sum(weights * r.pp), 
+                           "SE.r.pp.as" = sqrt(sum((weights^2)*(SE.r.pp^2))),
+                           "pyrs" = sum(pyrs)), 
+                       keyby = "surv.int"]
+  
+  testthat::expect_equal(
+    dt_raw_adj, dt_adj[, .SD, .SDcols = names(dt_raw_adj)]
+  )
+  
+  
+  
+})
+
diff --git a/tests/testthat/test_survtab_bad_surv_ints.R b/tests/testthat/test_survtab_bad_surv_ints.R
index 3cdab32..c40faf2 100644
--- a/tests/testthat/test_survtab_bad_surv_ints.R
+++ b/tests/testthat/test_survtab_bad_surv_ints.R
@@ -1,95 +1,95 @@
-testthat::context("Testing empty survival intervals in survtab")
-
-testthat::test_that("removing consecutively bad surv.ints is logical w/ & w/out adjusting", {
-  
-  sire2 <- sire[dg_date < ex_date, ]
-  sire2[, agegr := cut(dg_age, c(0,45,60,Inf), right=FALSE, labels=FALSE)]
-  
-  BL <- list(fot= seq(0,10,1/12), per=c(2008,2013))
-  
-  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, breaks=BL)
-  setDT(x)
-  
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  ## NOTE: neither should give any messages!
-  testthat::expect_message({
-    st1 <-  survtab(Surv(fot, lex.Xst) ~ agegr,
-                        subset = !(agegr == 1L & fot > 8.49),
-                        data = x, surv.type="surv.obs")
-    }, regexp = NA)
-  
-  ## INTENTION: 7.5+ intervals empty for one age group.
-  ## this should make adjusted estimates missing altogether for 7.5+.
-  testthat::expect_message({
-    st2 <- survtab(Surv(fot, lex.Xst) ~ adjust(agegr), 
-                      data = x, surv.type="surv.obs",
-                      subset = !(agegr == 1L & fot > 8.49),
-                      weights = list(agegr = c(0.33, 0.33, 0.33)))
-  }, regexp = NA)
-  setDT(st1)
-  setDT(st2)
-  
-  
-  testthat::expect_equal(st1[agegr==3 & Tstop>8.5, .N] ,  18L)
-  testthat::expect_equal(st1[agegr==1 & Tstop>8.5, .N] ,  0L)
-  testthat::expect_equal(st2[Tstop > 8.5, .N] , 0L)
-})
-
-## non-consecutively bad surv.ints ---------------------------------------------
-
-testthat::test_that("survtab_ag messages & results due to non-consecutively bad surv.ints are OK", {
-  ## non-consecutively bad surv.ints (missing years 5-6)
-  sire2 <- sire[dg_date < ex_date, ]
-  sire2[, agegr := cut(dg_age, c(0,45,60,Inf), right=FALSE, labels=FALSE)]
-  sire2 <- sire2[!(dg_age > 60 & as.integer(as.integer(ex_date-dg_date)/365.25) %in% 5:6)]
-  BL <- list(fot= seq(0,10,1/12), per=c(2008,2013))
-  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks=BL)
-  tf1 <- quote(
-    st1 <-  survtab(Surv(fot, lex.Xst)~1, data = x, surv.type="surv.obs", 
-                        subset=!(fot >= 5 & fot < 7))
-  )
-  
-  tf2 <- quote(
-    st2 <- survtab(Surv(fot, lex.Xst)~adjust(agegr), data = x, surv.type="surv.obs",
-                       subset=!(agegr==3 & fot >= 5 & fot < 7), 
-                       weights = list(agegr = c(0.33, 0.33, 0.33)))
-  )
-  
-  ## NOTE: \\ needed before "(" or ")"
-  msgs <- c(paste0("The total person-time was zero in some survival ",
-                   "intervals summed to the margins \\(over any stratifying ",
-                   "/ adjusting variables\\) _non-consecutively_, i.e. some ",
-                   "intervals after an empty interval had person-time in ",
-                   "them. Keeping all survival intervals with some estimates ",
-                   "as NA for inspection."),
-            "Some cumulative surv.obs were zero or NA:")
-  testthat::expect_message(eval(tf1), msgs[1],ignore.case=TRUE)
-  testthat::expect_message(eval(tf1), msgs[2],ignore.case=TRUE)
-  
-  setDT(st1)
-  
-  testthat::expect_equal(st1[is.na(surv.obs), .N], 60L)
-  
-  msgs <- c(paste0("The total person-time was zero in some survival ",
-                   "intervals, when summed to the variable\\(s\\) ",
-                   "'agegr' \\(i.e. over all other variables, if any",
-                   "\\) _non-consecutively_, i.e. some intervals after ",
-                   "an empty interval had person-time in them. ",
-                   "Keeping all survival intervals with some ",
-                   "estimates as NA for inspection."),
-            "Some cumulative surv.obs were zero or NA:")
-  
-  testthat::expect_message(eval(tf2), msgs[1])
-  testthat::expect_message(eval(tf2), msgs[2])
-  
-  setDT(st2)
-  testthat::expect_equal(st2[is.na(surv.obs.as), .N], 60L)
-})
-
-
-
-
-
+testthat::context("Testing empty survival intervals in survtab")
+
+testthat::test_that("removing consecutively bad surv.ints is logical w/ & w/out adjusting", {
+  
+  sire2 <- sire[dg_date < ex_date, ]
+  sire2[, agegr := cut(dg_age, c(0,45,60,Inf), right=FALSE, labels=FALSE)]
+  
+  BL <- list(fot= seq(0,10,1/12), per=c(2008,2013))
+  
+  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, breaks=BL)
+  setDT(x)
+  
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  ## NOTE: neither should give any messages!
+  testthat::expect_message({
+    st1 <-  survtab(Surv(fot, lex.Xst) ~ agegr,
+                        subset = !(agegr == 1L & fot > 8.49),
+                        data = x, surv.type="surv.obs")
+    }, regexp = NA)
+  
+  ## INTENTION: 7.5+ intervals empty for one age group.
+  ## this should make adjusted estimates missing altogether for 7.5+.
+  testthat::expect_message({
+    st2 <- survtab(Surv(fot, lex.Xst) ~ adjust(agegr), 
+                      data = x, surv.type="surv.obs",
+                      subset = !(agegr == 1L & fot > 8.49),
+                      weights = list(agegr = c(0.33, 0.33, 0.33)))
+  }, regexp = NA)
+  setDT(st1)
+  setDT(st2)
+  
+  
+  testthat::expect_equal(st1[agegr==3 & Tstop>8.5, .N] ,  18L)
+  testthat::expect_equal(st1[agegr==1 & Tstop>8.5, .N] ,  0L)
+  testthat::expect_equal(st2[Tstop > 8.5, .N] , 0L)
+})
+
+## non-consecutively bad surv.ints ---------------------------------------------
+
+testthat::test_that("survtab_ag messages & results due to non-consecutively bad surv.ints are OK", {
+  ## non-consecutively bad surv.ints (missing years 5-6)
+  sire2 <- sire[dg_date < ex_date, ]
+  sire2[, agegr := cut(dg_age, c(0,45,60,Inf), right=FALSE, labels=FALSE)]
+  sire2 <- sire2[!(dg_age > 60 & as.integer(as.integer(ex_date-dg_date)/365.25) %in% 5:6)]
+  BL <- list(fot= seq(0,10,1/12), per=c(2008,2013))
+  x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks=BL)
+  tf1 <- quote(
+    st1 <-  survtab(Surv(fot, lex.Xst)~1, data = x, surv.type="surv.obs", 
+                        subset=!(fot >= 5 & fot < 7))
+  )
+  
+  tf2 <- quote(
+    st2 <- survtab(Surv(fot, lex.Xst)~adjust(agegr), data = x, surv.type="surv.obs",
+                       subset=!(agegr==3 & fot >= 5 & fot < 7), 
+                       weights = list(agegr = c(0.33, 0.33, 0.33)))
+  )
+  
+  ## NOTE: \\ needed before "(" or ")"
+  msgs <- c(paste0("The total person-time was zero in some survival ",
+                   "intervals summed to the margins \\(over any stratifying ",
+                   "/ adjusting variables\\) _non-consecutively_, i.e. some ",
+                   "intervals after an empty interval had person-time in ",
+                   "them. Keeping all survival intervals with some estimates ",
+                   "as NA for inspection."),
+            "Some cumulative surv.obs were zero or NA:")
+  testthat::expect_message(eval(tf1), msgs[1],ignore.case=TRUE)
+  testthat::expect_message(eval(tf1), msgs[2],ignore.case=TRUE)
+  
+  setDT(st1)
+  
+  testthat::expect_equal(st1[is.na(surv.obs), .N], 60L)
+  
+  msgs <- c(paste0("The total person-time was zero in some survival ",
+                   "intervals, when summed to the variable\\(s\\) ",
+                   "'agegr' \\(i.e. over all other variables, if any",
+                   "\\) _non-consecutively_, i.e. some intervals after ",
+                   "an empty interval had person-time in them. ",
+                   "Keeping all survival intervals with some ",
+                   "estimates as NA for inspection."),
+            "Some cumulative surv.obs were zero or NA:")
+  
+  testthat::expect_message(eval(tf2), msgs[1])
+  testthat::expect_message(eval(tf2), msgs[2])
+  
+  setDT(st2)
+  testthat::expect_equal(st2[is.na(surv.obs.as), .N], 60L)
+})
+
+
+
+
+
diff --git a/tests/testthat/test_survtab_observed.R b/tests/testthat/test_survtab_observed.R
index fd12833..babd545 100644
--- a/tests/testthat/test_survtab_observed.R
+++ b/tests/testthat/test_survtab_observed.R
@@ -1,144 +1,144 @@
-testthat::context("CIF's & surv.obs's congruence & comparison w/ survival::survfit")
-
-testthat::test_that("surv.obs about the same as Kaplan-Meier & CIFs close to Aalen-Johansen", {
-  
-  BL <- list(fot= seq(0,19,1/12), per=c(2008,2013))
-  sire2 <- sire[dg_date<ex_date, ]
-  sire2$statusf <- factor(sire2$status, levels = 0:2, 
-                         labels = c("alive", "canD", "othD"))
-  
-  x <- lexpand(sire2, 
-               birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = statusf,
-               breaks=BL)
-  st <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="cif.obs")
-  setDT(x)
-  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-  setDT(st)
-  
-  testthat::test_that("CIFs and surv.obs sum to 1", {
-    testthat::expect_equal(st[, CIF_canD + CIF_othD + surv.obs] ,  rep(1, times = st[,.N]), tolerance = 0.0001, scale=1)
-  })
-  
-  x <- lexpand(sire2, 
-               birth  = bi_date, entry = dg_date, exit = ex_date,
-               status = statusf,
-               breaks = BL["per"])
-  
-  fb <- setdiff(BL$fot, 0)
-  su.km  <- survival::survfit(Surv(time=fot, time2=fot+lex.dur, event = lex.Xst!="alive") ~ 1, data = x, id = lex.id)
-  su.km  <- summary(su.km, times = fb)
-  su.km  <- cbind(data.table(time = su.km$time), data.table(su.km$surv))
-  
-  su.cif <- survival::survfit(Surv(time=fot, time2=fot+lex.dur, event = lex.Xst)~1, data=x, id = lex.id)
-  su.cif <- summary(su.cif, times = fb)
-  ## see issue #125
-  prev_var <- intersect(names(su.cif), c("prev", "pstate"))
-  stopifnot(length(prev_var) == 1L)
-  curve_nms <- dimnames(su.cif[["table"]])[[1]]
-  keep_curve <- curve_nms %in% c("canD", "othD")
-  cif <- cbind(data.table(time = su.cif$time), data.table(su.cif[[prev_var]][, keep_curve]))
-  
-  testthat::expect_equal(st[, surv.obs] ,  su.km[, V1], tolerance = 0.0032, scale=1)
-  
-  testthat::expect_equal(cif$V1, st$CIF_canD, tolerance = 0.0022, scale=1)
-  testthat::expect_equal(cif$V2, st$CIF_othD, tolerance = 0.0011, scale=1)
-})
-
-
-# custom status var -------------------------------------------------------
-
-testthat::test_that("survtab status argument works as expected", {
-  popEpi:::skip_normally()
-  
-  BL <- list(fot= seq(0,19,1/12), per=c(2008,2013))
-  sr <- sire[dg_date < ex_date, ]
-  
-  sr$statusf <- factor(sr$status, 0:2, labels = c("alive", "canD", "othD"))
-  sr$statusb <- as.integer(sr$status %in% 1:2)
-  
-  st <- NULL
-  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
-               exit = ex_date, status = status)
-  testthat::expect_error(
-    suppressWarnings(
-      st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
-                    breaks = list(fot = 0:5))
-    ), 
-    regexp = paste0("Some status indicators (3648 values in total) were NA. ",
-                    "Usual suspects: original status variable has NA values, ",
-                    "or you have numeric status variable with more than two ",
-                    "levels and you did not assign e.g. type = 'mstate' ",
-                    "(e.g. Surv(time = c(1,1,1), event = c(0,1,2), ",
-                    "type = 'mstate') works)."),
-    fixed = TRUE
-  )
-
-  st <- NULL
-  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
-               exit = ex_date, status = statusf)
-  st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
-                    breaks = list(fot = 0:5))
-  testthat::expect_equal(class(st)[1L], "survtab")
-  
-  st <- NULL
-  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
-               exit = ex_date, status = statusb)
-  st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
-                    breaks = list(fot = 0:5))
-  testthat::expect_equal(class(st)[1L], "survtab")
-  
-})
-
-
-testthat::test_that("survtab works with more complicated estimation", {
-  library(Epi)
-  
-  library(data.table)
-  
-  x <- data.table(popEpi::sire[sire$dg_date < sire$ex_date, ])
-  
-  ## pretend some are male
-  set.seed(1L)
-  x$sex <- rbinom(nrow(x), 1, 0.5)
-  
-  ## period
-  x$period <- cut(year(x$dg_date), c(1993,1998,2003,2008,2013), right = FALSE)
-  
-  # age group
-  x$agegr <- cut(x$dg_age, 4)
-  
-  x$stat <- factor(x$status, levels = 0:2, 
-                   labels = c("alive", "canD", "othD"))
-  x$enStat <- factor(rep(0L, nrow(x)), levels = 0:2, 
-                     labels = c("alive", "canD", "othD"))
-  xl <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-              exit = list(CAL = get.yrs(ex_date)), 
-              data = x,
-              exit.status = stat, entry.status = enStat, 
-              merge = TRUE)
-  
-  
-  ## observed survival
-  st1 <- survtab(Surv(time = FUT, event = lex.Xst) ~ factor(sex, 0:1, c("male", "female")) + period 
-                     + adjust(agegr), data = xl, 
-                     weights = list(agegr = as.numeric(table(x$agegr))),
-                     surv.type = "surv.obs",
-                     breaks = list(FUT = seq(0, 5, 1/12)))
-  
-  ag <- splitLexisDT(xl, breaks = seq(0, 5, 1/12), timeScale = "FUT")
-  ag[, lex.Cst := as.integer(lex.Cst)]
-  ag[, lex.Cst := 0L]
-  ag[, lex.Xst := as.integer(lex.Xst != "alive")]
-  ag <- aggre(ag, by = list(sex, period, agegr, FUT))
-  
-  st2 <- survtab_ag(FUT ~ factor(sex, 0:1, c("male", "female")) + period + adjust(agegr), 
-                    weights = list(agegr = as.numeric(table(x$agegr))),
-                    data = ag, surv.type = "surv.obs", d = "from0to1")
-
-  testthat::expect_equal(st1$surv.obs.as.lo, st2$surv.obs.as.lo)
-  testthat::expect_equivalent(st1, st2)
-})
-
-
-
+testthat::context("CIF's & surv.obs's congruence & comparison w/ survival::survfit")
+
+testthat::test_that("surv.obs about the same as Kaplan-Meier & CIFs close to Aalen-Johansen", {
+  
+  BL <- list(fot= seq(0,19,1/12), per=c(2008,2013))
+  sire2 <- sire[dg_date<ex_date, ]
+  sire2$statusf <- factor(sire2$status, levels = 0:2, 
+                         labels = c("alive", "canD", "othD"))
+  
+  x <- lexpand(sire2, 
+               birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = statusf,
+               breaks=BL)
+  st <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="cif.obs")
+  setDT(x)
+  setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+  setDT(st)
+  
+  testthat::test_that("CIFs and surv.obs sum to 1", {
+    testthat::expect_equal(st[, CIF_canD + CIF_othD + surv.obs] ,  rep(1, times = st[,.N]), tolerance = 0.0001, scale=1)
+  })
+  
+  x <- lexpand(sire2, 
+               birth  = bi_date, entry = dg_date, exit = ex_date,
+               status = statusf,
+               breaks = BL["per"])
+  
+  fb <- setdiff(BL$fot, 0)
+  su.km  <- survival::survfit(Surv(time=fot, time2=fot+lex.dur, event = lex.Xst!="alive") ~ 1, data = x, id = lex.id)
+  su.km  <- summary(su.km, times = fb)
+  su.km  <- cbind(data.table(time = su.km$time), data.table(su.km$surv))
+  
+  su.cif <- survival::survfit(Surv(time=fot, time2=fot+lex.dur, event = lex.Xst)~1, data=x, id = lex.id)
+  su.cif <- summary(su.cif, times = fb)
+  ## see issue #125
+  prev_var <- intersect(names(su.cif), c("prev", "pstate"))
+  stopifnot(length(prev_var) == 1L)
+  curve_nms <- dimnames(su.cif[["table"]])[[1]]
+  keep_curve <- curve_nms %in% c("canD", "othD")
+  cif <- cbind(data.table(time = su.cif$time), data.table(su.cif[[prev_var]][, keep_curve]))
+  
+  testthat::expect_equal(st[, surv.obs] ,  su.km[, V1], tolerance = 0.0032, scale=1)
+  
+  testthat::expect_equal(cif$V1, st$CIF_canD, tolerance = 0.0022, scale=1)
+  testthat::expect_equal(cif$V2, st$CIF_othD, tolerance = 0.0011, scale=1)
+})
+
+
+# custom status var -------------------------------------------------------
+
+testthat::test_that("survtab status argument works as expected", {
+  popEpi:::skip_normally()
+  
+  BL <- list(fot= seq(0,19,1/12), per=c(2008,2013))
+  sr <- sire[dg_date < ex_date, ]
+  
+  sr$statusf <- factor(sr$status, 0:2, labels = c("alive", "canD", "othD"))
+  sr$statusb <- as.integer(sr$status %in% 1:2)
+  
+  st <- NULL
+  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
+               exit = ex_date, status = status)
+  testthat::expect_error(
+    suppressWarnings(
+      st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
+                    breaks = list(fot = 0:5))
+    ), 
+    regexp = paste0("Some status indicators (3648 values in total) were NA. ",
+                    "Usual suspects: original status variable has NA values, ",
+                    "or you have numeric status variable with more than two ",
+                    "levels and you did not assign e.g. type = 'mstate' ",
+                    "(e.g. Surv(time = c(1,1,1), event = c(0,1,2), ",
+                    "type = 'mstate') works)."),
+    fixed = TRUE
+  )
+
+  st <- NULL
+  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
+               exit = ex_date, status = statusf)
+  st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
+                    breaks = list(fot = 0:5))
+  testthat::expect_equal(class(st)[1L], "survtab")
+  
+  st <- NULL
+  x <- lexpand(sr, birth  = bi_date, entry = dg_date, 
+               exit = ex_date, status = statusb)
+  st <- survtab(Surv(fot, lex.Xst) ~ 1, data = x, surv.type = "surv.obs", 
+                    breaks = list(fot = 0:5))
+  testthat::expect_equal(class(st)[1L], "survtab")
+  
+})
+
+
+testthat::test_that("survtab works with more complicated estimation", {
+  library(Epi)
+  
+  library(data.table)
+  
+  x <- data.table(popEpi::sire[sire$dg_date < sire$ex_date, ])
+  
+  ## pretend some are male
+  set.seed(1L)
+  x$sex <- rbinom(nrow(x), 1, 0.5)
+  
+  ## period
+  x$period <- cut(year(x$dg_date), c(1993,1998,2003,2008,2013), right = FALSE)
+  
+  # age group
+  x$agegr <- cut(x$dg_age, 4)
+  
+  x$stat <- factor(x$status, levels = 0:2, 
+                   labels = c("alive", "canD", "othD"))
+  x$enStat <- factor(rep(0L, nrow(x)), levels = 0:2, 
+                     labels = c("alive", "canD", "othD"))
+  xl <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+              exit = list(CAL = get.yrs(ex_date)), 
+              data = x,
+              exit.status = stat, entry.status = enStat, 
+              merge = TRUE)
+  
+  
+  ## observed survival
+  st1 <- survtab(Surv(time = FUT, event = lex.Xst) ~ factor(sex, 0:1, c("male", "female")) + period 
+                     + adjust(agegr), data = xl, 
+                     weights = list(agegr = as.numeric(table(x$agegr))),
+                     surv.type = "surv.obs",
+                     breaks = list(FUT = seq(0, 5, 1/12)))
+  
+  ag <- splitLexisDT(xl, breaks = seq(0, 5, 1/12), timeScale = "FUT")
+  ag[, lex.Cst := as.integer(lex.Cst)]
+  ag[, lex.Cst := 0L]
+  ag[, lex.Xst := as.integer(lex.Xst != "alive")]
+  ag <- aggre(ag, by = list(sex, period, agegr, FUT))
+  
+  st2 <- survtab_ag(FUT ~ factor(sex, 0:1, c("male", "female")) + period + adjust(agegr), 
+                    weights = list(agegr = as.numeric(table(x$agegr))),
+                    data = ag, surv.type = "surv.obs", d = "from0to1")
+
+  testthat::expect_equal(st1$surv.obs.as.lo, st2$surv.obs.as.lo)
+  testthat::expect_equivalent(st1, st2)
+})
+
+
+
diff --git a/tests/testthat/test_survtab_relative.R b/tests/testthat/test_survtab_relative.R
index a51e2ad..e1d7a2c 100644
--- a/tests/testthat/test_survtab_relative.R
+++ b/tests/testthat/test_survtab_relative.R
@@ -1,196 +1,196 @@
-if (requireNamespace("relsurv")) {
-testthat::context("popEpi::survtab vs. relsurv::rs.surv")
-  # survtab vs. relsurv::rs.surv --------------------------------------------
-  testthat::test_that("relative survivals about the same as relsurv's", {
-    
-    
-    library(relsurv)
-    library(Epi)
-    
-    # male
-    pm <- data.table(popEpi::popmort)
-    # pm[, surv := 1L]
-    pm[, surv := exp(-haz)]
-    pm.m <- cast_simple(pm[sex==0], columns = 'year', rows = 'agegroup',  values='surv')
-    pm.m[,agegroup := NULL]
-    pm.m <- as.matrix(pm.m)
-    # female
-    pm.f <- cast_simple(pm[sex==1], columns = 'year', rows = 'agegroup',  values='surv')
-    pm.f[,agegroup := NULL]
-    pm.f <- as.matrix(pm.f)
-    
-    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
-    
-    pm[, surv := NULL]
-    
-    sire2 <- sire[dg_date<ex_date, ]
-    sire2[, Tstop  := as.integer(ex_date - dg_date)]
-    sire2[, dg_age := as.integer(dg_date - bi_date)]
-    
-    x <- Lexis(entry = list(age = dg_age, per = dg_date, fot = 0L),
-              exit = list(fot = Tstop), 
-              exit.status = as.integer(status %in% 1:2),
-              entry.status = 0L, data = sire2)
-    setDT(x)
-    setattr(x, "class", c("Lexis","data.table", "data.frame"))
-    
-    ## rs.surv
-    ## sex must be coded c(1,2) (male, female)
-    x[, paste0(c("bi", "dg", "ex"), "_date") := lapply(.SD, as.Date),
-      .SDcols = paste0(c("bi", "dg", "ex"), "_date")]
-    x[, per := as.Date(per)]
-    x[, sex := 2L]
-    rs.e2 <- rs.surv(Surv(lex.dur, lex.Xst!=0) ~ 1 + ratetable(age=age, sex=sex, year=per),
-                    ratetable = popm, data = x, method = 'ederer2', type = "fleming-harrington", fin.date=ex_date)
-    rs.pp <- rs.surv(Surv(lex.dur, lex.Xst!=0) ~ 1 + ratetable(age=age, sex=sex, year=per),
-                    ratetable = popm, data = x, method = 'pohar-perme', type = "fleming-harrington", fin.date=ex_date)
-    x[, sex := 1L]
-    
-    ## survtab
-    fb <- seq(0, 19, 1/24)
-    
-    x[, lex.dur := lex.dur/365.242199]
-    x[, age := age/365.242199]
-    x[, per := get.yrs(per, year.length = "approx")]
-    
-    setnames(pm, c("year", "agegroup"), c("per", "age"))
-    st.e2 <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="surv.rel", 
-                        relsurv.method="e2", pophaz = pm, breaks = list(fot = fb))
-    st.pp <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="surv.rel", 
-                        relsurv.method="pp", pophaz = pm, breaks = list(fot = fb))
-    setDT(st.e2)
-    setDT(st.pp)
-    
-    ## rs.surv
-    fb <- fb[-1]
-    fbd <- fb*365.242199
-    
-    su.e2 <- summary(rs.e2, times = fbd)
-    su.e2 <- cbind(data.table(time = fb), data.table(su.e2$surv))
-    su.pp <- summary(rs.pp, times = fbd)
-    su.pp <- cbind(data.table(time = fb), data.table(su.pp$surv))
-    
-    testthat::expect_equal(st.e2[, r.e2] ,  su.e2[, V1], tolerance = 0.000226, scale = 1L)
-    testthat::expect_equal(st.pp[, r.pp] ,  su.pp[, V1], tolerance = 0.00292, scale = 1L)
-  })
-
-  # relpois vs. relsurv::rsadd ---------------------------------------------
-
-  testthat::test_that("relpois congruent with relsurv::rsadd", {
-    popEpi:::skip_normally()
-    
-    
-    library(relsurv)
-    
-    # male
-    pm <- data.table(popEpi::popmort)
-    # pm[, surv := 1L]
-    pm[, surv := exp(-haz)]
-    pm.m <- cast_simple(pm[sex==0], columns = 'year', rows = 'agegroup',  values='surv')
-    pm.m[,agegroup := NULL]
-    pm.m <- as.matrix(pm.m)
-    # female
-    pm.f <- cast_simple(pm[sex==1], columns = 'year', rows = 'agegroup',  values='surv')
-    pm.f[,agegroup := NULL]
-    pm.f <- as.matrix(pm.f)
-    
-    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
-    
-    pm[, surv := NULL]
-    
-    sire2 <- copy(sire)
-    sire2[, Tstop  := as.integer(ex_date - dg_date)]
-    sire2[, dg_age := as.integer(dg_date - bi_date)]
-    
-    
-    sire2[, agegr := cut(dg_age/365.25, breaks = c(0,45,70,Inf))]
-    x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2,
-                breaks=list(fot=0:5), pophaz=pm)
-    rp <- relpois(x, formula = lex.Xst%in%1:2 ~ -1+FOT+agegr)
-    setDT(x)
-    setattr(x, "class", c("Lexis", "data.table", "data.frame"))
-    sire2[, sex := 2L]
-    sire2[, paste0(c("bi", "dg", "ex"), "_date") := lapply(.SD, as.Date),
-          .SDcols = paste0(c("bi", "dg", "ex"), "_date")]
-    
-    rs <- relsurv::rsadd(Surv(Tstop, event = status %in% 1:2) ~ ratetable(age=dg_age, sex=sex, year=dg_date) + agegr,
-                        int = 0:5,
-                        ratetable = popm, data = sire2, method = "glm.poi")
-    
-    
-    testthat::expect_equal(coef(rp)[1:5] ,  coef(rs)[3:7], tolerance = 0.055, scale=1, check.attributes=FALSE)
-  })
-
-
-
-  testthat::test_that("Ederer I expected survival curve agrees with survival::survexp", {
-    
-    
-    library(relsurv)
-    library(Epi)
-    
-    # male
-    pm <- data.table(popEpi::popmort)
-    # pm[, surv := 1L]
-    pm[, surv := exp(-haz)]
-    pm.m <- cast_simple(pm[sex == 0], columns = 'year', rows = 'agegroup', values = 'surv')
-    pm.m[,agegroup := NULL]
-    pm.m <- as.matrix(pm.m)
-    # female
-    pm.f <- cast_simple(pm[sex == 1], columns = 'year', rows = 'agegroup', values = 'surv')
-    pm.f[,agegroup := NULL]
-    pm.f <- as.matrix(pm.f)
-    
-    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
-    
-    pm[, surv := NULL]
-    
-    sire2 <- popEpi::sire[dg_date < ex_date, ]
-    sire2 <- sire2[1:500, ]
-    sire2[, "Tstop"  := as.integer(ex_date - dg_date)]
-    sire2[, "dg_age" := as.integer(dg_date - bi_date)]
-    
-    x <- Lexis(entry = list(age = dg_age, per = dg_date, fot = 0L),
-              exit = list(fot = Tstop), 
-              exit.status = as.integer(status %in% 1:2),
-              entry.status = 0L, data = sire2)
-    setDT(x)
-    setattr(x, "class", c("Lexis","data.table", "data.frame"))
-    x[, "per" := as.Date(per)]
-    
-    ## rs.surv
-    ## sex must be coded c(1,2) (male, female)
-    x[, "sex" := 2L]
-    
-    fb <- seq(0, 19, 1/24)
-    su <- survexp(~1, data = x, ratetab = popm, method = "ederer", 
-                  rmap = list(sex = "female", year = per, age = age),
-                  times = fb*365.242199)
-    
-    x[, "sex" := 1L]
-    x[, "lex.dur" := max(fb)] ## not really needed but illustrative
-    x[, "age" := age/365.242199]
-    x[, "per" := get.yrs(per, year.length = "approx")]
-    
-    setnames(pm, c("year", "agegroup"), c("per", "age"))
-    e1 <- comp_e1(x, breaks = list(fot = fb), pophaz = pm, survScale = "fot")
-    
-    ## rs.surv
-    fb <- fb[-1]
-    fbd <- fb*365.242199
-    
-    su <- data.table(time = su$time, surv.exp = su$surv)
-    su <- su[time != 0L]
-    su[, time := time / 365.242199]
-    setnames(su, "time", "fot")
-    
-    testthat::expect_equal(e1, su, tolerance = 0.000004575, scale = 1L)
-    testthat::expect_equal(max(abs(e1$surv.exp - su$surv.exp)), 0L, 
-                tolerance = 0.0000103, scale = 1L)
-    
-    
-  })
-
-}
-
+if (requireNamespace("relsurv")) {
+testthat::context("popEpi::survtab vs. relsurv::rs.surv")
+  # survtab vs. relsurv::rs.surv --------------------------------------------
+  testthat::test_that("relative survivals about the same as relsurv's", {
+    
+    
+    library(relsurv)
+    library(Epi)
+    
+    # male
+    pm <- data.table(popEpi::popmort)
+    # pm[, surv := 1L]
+    pm[, surv := exp(-haz)]
+    pm.m <- cast_simple(pm[sex==0], columns = 'year', rows = 'agegroup',  values='surv')
+    pm.m[,agegroup := NULL]
+    pm.m <- as.matrix(pm.m)
+    # female
+    pm.f <- cast_simple(pm[sex==1], columns = 'year', rows = 'agegroup',  values='surv')
+    pm.f[,agegroup := NULL]
+    pm.f <- as.matrix(pm.f)
+    
+    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
+    
+    pm[, surv := NULL]
+    
+    sire2 <- sire[dg_date<ex_date, ]
+    sire2[, Tstop  := as.integer(ex_date - dg_date)]
+    sire2[, dg_age := as.integer(dg_date - bi_date)]
+    
+    x <- Lexis(entry = list(age = dg_age, per = dg_date, fot = 0L),
+              exit = list(fot = Tstop), 
+              exit.status = as.integer(status %in% 1:2),
+              entry.status = 0L, data = sire2)
+    setDT(x)
+    setattr(x, "class", c("Lexis","data.table", "data.frame"))
+    
+    ## rs.surv
+    ## sex must be coded c(1,2) (male, female)
+    x[, paste0(c("bi", "dg", "ex"), "_date") := lapply(.SD, as.Date),
+      .SDcols = paste0(c("bi", "dg", "ex"), "_date")]
+    x[, per := as.Date(per)]
+    x[, sex := 2L]
+    rs.e2 <- rs.surv(Surv(lex.dur, lex.Xst!=0) ~ 1 + ratetable(age=age, sex=sex, year=per),
+                    ratetable = popm, data = x, method = 'ederer2', type = "fleming-harrington", fin.date=ex_date)
+    rs.pp <- rs.surv(Surv(lex.dur, lex.Xst!=0) ~ 1 + ratetable(age=age, sex=sex, year=per),
+                    ratetable = popm, data = x, method = 'pohar-perme', type = "fleming-harrington", fin.date=ex_date)
+    x[, sex := 1L]
+    
+    ## survtab
+    fb <- seq(0, 19, 1/24)
+    
+    x[, lex.dur := lex.dur/365.242199]
+    x[, age := age/365.242199]
+    x[, per := get.yrs(per, year.length = "approx")]
+    
+    setnames(pm, c("year", "agegroup"), c("per", "age"))
+    st.e2 <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="surv.rel", 
+                        relsurv.method="e2", pophaz = pm, breaks = list(fot = fb))
+    st.pp <- survtab(Surv(fot, event = lex.Xst) ~ 1, data = x, surv.type="surv.rel", 
+                        relsurv.method="pp", pophaz = pm, breaks = list(fot = fb))
+    setDT(st.e2)
+    setDT(st.pp)
+    
+    ## rs.surv
+    fb <- fb[-1]
+    fbd <- fb*365.242199
+    
+    su.e2 <- summary(rs.e2, times = fbd)
+    su.e2 <- cbind(data.table(time = fb), data.table(su.e2$surv))
+    su.pp <- summary(rs.pp, times = fbd)
+    su.pp <- cbind(data.table(time = fb), data.table(su.pp$surv))
+    
+    testthat::expect_equal(st.e2[, r.e2] ,  su.e2[, V1], tolerance = 0.000226, scale = 1L)
+    testthat::expect_equal(st.pp[, r.pp] ,  su.pp[, V1], tolerance = 0.00292, scale = 1L)
+  })
+
+  # relpois vs. relsurv::rsadd ---------------------------------------------
+
+  testthat::test_that("relpois congruent with relsurv::rsadd", {
+    popEpi:::skip_normally()
+    
+    
+    library(relsurv)
+    
+    # male
+    pm <- data.table(popEpi::popmort)
+    # pm[, surv := 1L]
+    pm[, surv := exp(-haz)]
+    pm.m <- cast_simple(pm[sex==0], columns = 'year', rows = 'agegroup',  values='surv')
+    pm.m[,agegroup := NULL]
+    pm.m <- as.matrix(pm.m)
+    # female
+    pm.f <- cast_simple(pm[sex==1], columns = 'year', rows = 'agegroup',  values='surv')
+    pm.f[,agegroup := NULL]
+    pm.f <- as.matrix(pm.f)
+    
+    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
+    
+    pm[, surv := NULL]
+    
+    sire2 <- copy(sire)
+    sire2[, Tstop  := as.integer(ex_date - dg_date)]
+    sire2[, dg_age := as.integer(dg_date - bi_date)]
+    
+    
+    sire2[, agegr := cut(dg_age/365.25, breaks = c(0,45,70,Inf))]
+    x <- lexpand(sire2, birth  = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2,
+                breaks=list(fot=0:5), pophaz=pm)
+    rp <- relpois(x, formula = lex.Xst%in%1:2 ~ -1+FOT+agegr)
+    setDT(x)
+    setattr(x, "class", c("Lexis", "data.table", "data.frame"))
+    sire2[, sex := 2L]
+    sire2[, paste0(c("bi", "dg", "ex"), "_date") := lapply(.SD, as.Date),
+          .SDcols = paste0(c("bi", "dg", "ex"), "_date")]
+    
+    rs <- relsurv::rsadd(Surv(Tstop, event = status %in% 1:2) ~ ratetable(age=dg_age, sex=sex, year=dg_date) + agegr,
+                        int = 0:5,
+                        ratetable = popm, data = sire2, method = "glm.poi")
+    
+    
+    testthat::expect_equal(coef(rp)[1:5] ,  coef(rs)[3:7], tolerance = 0.055, scale=1, check.attributes=FALSE)
+  })
+
+
+
+  testthat::test_that("Ederer I expected survival curve agrees with survival::survexp", {
+    
+    
+    library(relsurv)
+    library(Epi)
+    
+    # male
+    pm <- data.table(popEpi::popmort)
+    # pm[, surv := 1L]
+    pm[, surv := exp(-haz)]
+    pm.m <- cast_simple(pm[sex == 0], columns = 'year', rows = 'agegroup', values = 'surv')
+    pm.m[,agegroup := NULL]
+    pm.m <- as.matrix(pm.m)
+    # female
+    pm.f <- cast_simple(pm[sex == 1], columns = 'year', rows = 'agegroup', values = 'surv')
+    pm.f[,agegroup := NULL]
+    pm.f <- as.matrix(pm.f)
+    
+    popm <- transrate(pm.m, pm.f, yearlim = c(1951, 2013), int.length = 1)
+    
+    pm[, surv := NULL]
+    
+    sire2 <- popEpi::sire[dg_date < ex_date, ]
+    sire2 <- sire2[1:500, ]
+    sire2[, "Tstop"  := as.integer(ex_date - dg_date)]
+    sire2[, "dg_age" := as.integer(dg_date - bi_date)]
+    
+    x <- Lexis(entry = list(age = dg_age, per = dg_date, fot = 0L),
+              exit = list(fot = Tstop), 
+              exit.status = as.integer(status %in% 1:2),
+              entry.status = 0L, data = sire2)
+    setDT(x)
+    setattr(x, "class", c("Lexis","data.table", "data.frame"))
+    x[, "per" := as.Date(per)]
+    
+    ## rs.surv
+    ## sex must be coded c(1,2) (male, female)
+    x[, "sex" := 2L]
+    
+    fb <- seq(0, 19, 1/24)
+    su <- survexp(~1, data = x, ratetab = popm, method = "ederer", 
+                  rmap = list(sex = "female", year = per, age = age),
+                  times = fb*365.242199)
+    
+    x[, "sex" := 1L]
+    x[, "lex.dur" := max(fb)] ## not really needed but illustrative
+    x[, "age" := age/365.242199]
+    x[, "per" := get.yrs(per, year.length = "approx")]
+    
+    setnames(pm, c("year", "agegroup"), c("per", "age"))
+    e1 <- comp_e1(x, breaks = list(fot = fb), pophaz = pm, survScale = "fot")
+    
+    ## rs.surv
+    fb <- fb[-1]
+    fbd <- fb*365.242199
+    
+    su <- data.table(time = su$time, surv.exp = su$surv)
+    su <- su[time != 0L]
+    su[, time := time / 365.242199]
+    setnames(su, "time", "fot")
+    
+    testthat::expect_equal(e1, su, tolerance = 0.000004575, scale = 1L)
+    testthat::expect_equal(max(abs(e1$surv.exp - su$surv.exp)), 0L, 
+                tolerance = 0.0000103, scale = 1L)
+    
+    
+  })
+
+}
+
diff --git a/tests/testthat/test_survtab_usage.R b/tests/testthat/test_survtab_usage.R
index 0940ea5..514debb 100644
--- a/tests/testthat/test_survtab_usage.R
+++ b/tests/testthat/test_survtab_usage.R
@@ -1,484 +1,484 @@
-testthat::context("survtab usage")
-
-
-
-testthat::test_that("Dates and frac. yrs produce congruent results", {
-  popEpi:::skip_normally()
-  library(Epi)
-  
-  x <- data.table(popEpi::sire)
-  x <- x[dg_date<ex_date]
-  
-  ## phony group variable
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-  
-  
-  ## yrs
-  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-              exit = list(CAL = get.yrs(ex_date)), 
-              data = x,
-              entry.status = 0L,
-              exit.status = as.integer(status %in% 1:2), 
-              merge = TRUE)
-  
-  ## dates
-  xd <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
-              exit = list(CAL = ex_date),
-              data = x,
-              entry.status = 0L,
-              exit.status = as.integer(status %in% 1:2), 
-              merge = TRUE)
-  yd <- 365.242199
-  BLy <- list(FUT = seq(0, 5, 1/4))
-  BLd <- lapply(BLy, function(el) el * yd)
-  
-  pmy <- data.table(popEpi::popmort)
-  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
-  
-  pmd <- data.table(pmy)
-  pmd[, CAL := as.Date(paste0(CAL, "-01-01"))]
-  pmd[, AGE := AGE * yd]
-  pmd[, haz := haz/yd]
-  
-  #### hazard method
-  ## observed survival & Ederer II
-  
-  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                     surv.type = "surv.rel", relsurv.method = "e2",
-                     surv.method = "hazard",
-                     breaks = BLy, pophaz = pmy)
-  
-  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                     surv.type = "surv.rel", relsurv.method = "e2",
-                     surv.method = "hazard",
-                     breaks = BLd, pophaz = pmd)    
-  
-  testthat::expect_equal(sty$surv.obs.lo, std$surv.obs.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$surv.obs, std$surv.obs, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$surv.obs.hi, std$surv.obs.hi, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2.lo, std$r.e2.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2, std$r.e2, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2.hi, std$r.e2.hi, scale = 1L, tolerance = 0.0005)
-  
-  ## pohar perme
-  
-  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                     surv.type = "surv.rel", relsurv.method = "pp",
-                     surv.method = "hazard",
-                     breaks = BLy, pophaz = pmy)
-  
-  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                     surv.type = "surv.rel", relsurv.method = "pp",
-                     surv.method = "hazard",
-                     breaks = BLd, pophaz = pmd)    
-  
-  testthat::expect_equal(sty$r.pp.lo, std$r.pp.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.pp, std$r.pp, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.pp.hi, std$r.pp.hi, scale = 1L, tolerance = 0.0005)
-  
-  #### lifetable method
-  ## observed survival & Ederer II
-  
-  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                     surv.type = "surv.rel", relsurv.method = "e2",
-                     surv.method = "lifetable",
-                     breaks = BLy, pophaz = pmy)
-  
-  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                     surv.type = "surv.rel", relsurv.method = "e2",
-                     surv.method = "lifetable",
-                     breaks = BLd, pophaz = pmd)    
-  
-  testthat::expect_equal(sty$surv.obs.lo, std$surv.obs.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$surv.obs, std$surv.obs, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$surv.obs.hi, std$surv.obs.hi, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2.lo, std$r.e2.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2, std$r.e2, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.e2.hi, std$r.e2.hi, scale = 1L, tolerance = 0.0005)
-  
-  ## pohar perme
-  
-  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
-                     surv.type = "surv.rel", relsurv.method = "pp",
-                     surv.method = "lifetable",
-                     breaks = BLy, pophaz = pmy)
-  
-  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
-                     surv.type = "surv.rel", relsurv.method = "pp",
-                     surv.method = "lifetable",
-                     breaks = BLd, pophaz = pmd)    
-  
-  testthat::expect_equal(sty$r.pp.lo, std$r.pp.lo, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.pp, std$r.pp, scale = 1L, tolerance = 0.0005)
-  testthat::expect_equal(sty$r.pp.hi, std$r.pp.hi, scale = 1L, tolerance = 0.0005)
-  
-})
-
-
-
-testthat::test_that("hazard and lifetable produce congruent results", {
-  popEpi:::skip_normally()
-  library(Epi)
-  
-  x <- data.table(popEpi::sire)
-  x <- x[dg_date<ex_date]
-  
-  ## phony group variable
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-  
-  
-  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-              exit = list(CAL = get.yrs(ex_date)), 
-              data = x,
-              entry.status = 0L,
-              exit.status = as.integer(status %in% 1:2), 
-              merge = TRUE)
-  
-  
-  pmy <- data.table(popEpi::popmort)
-  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
-  BL <- list(FUT = seq(0, 5, 1/12))
-  
-  sth <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy,
-                 pophaz = pmy, breaks = BL,
-                 surv.type = "surv.rel", surv.method = "hazard")
-  
-  stl <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy,
-                 pophaz = pmy, breaks = BL,
-                 surv.type = "surv.rel", surv.method = "lifetable")
-  
-  testthat::expect_equal(sth$r.e2, stl$r.e2, scale = 1, tol = 0.0003415)
-  testthat::expect_equal(sth$r.e2.lo, stl$r.e2.lo, scale = 1, tol = 0.000354)
-  testthat::expect_equal(sth$r.e2.hi, stl$r.e2.hi, scale = 1, tol = 0.00033)
-  
-  testthat::expect_equal(sth$surv.obs, stl$surv.obs, scale = 1, tol = 0.00002575)
-  testthat::expect_equal(sth$surv.obs.lo, stl$surv.obs.lo, scale = 1, tol = 0.000027)
-  testthat::expect_equal(sth$surv.obs.hi, stl$surv.obs.hi, scale = 1, tol = 0.000025)
-  
-})
-
-## test below even should not work, since period analysis
-## means that at.risk counts (subjects entering an interval in its beginning)
-## will not necessarily equal the counts of events.
-# test_that("lifetable counts work with period analysis", {
-#   library(Epi)
-#   x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-#              exit = list(CAL = get.yrs(ex_date)),
-#              data = sire[sire$dg_date < sire$ex_date, ],
-#              exit.status = as.integer(status %in% 1:2),
-#              entry.status = 0,
-#              merge = TRUE)
-# 
-#   ## phony group variable
-#   set.seed(1L)
-#   x$group <- rbinom(nrow(x), 1, 0.5)
-# 
-#   BL <- list(FUT = seq(0, 5, 1/12), CAL = c(2008,2013))
-#   x <- splitMulti(x, BL)
-# 
-#   st1 <- survtab(Surv(FUT, lex.Xst) ~ 1, data = x, 
-#                  surv.type = "surv.obs",
-#                  surv.method = "lifetable")
-#   
-#   a <- aggre(x, by = FUT)
-#   
-#   st2 <- survtab_ag(FUT ~ 1, data = a, surv.type = "surv.obs",
-#                     surv.method = "lifetable",
-#                     n = "at.risk", d = c("from0to1"),
-#                     n.cens = "from0to0", pyrs = "pyrs")
-#   
-#   expect_equal(st1$surv.obs, st2$surv.obs)
-#   
-# })
-
-
-testthat::test_that("its possible to pass dynamically created arguments", {
-  popEpi:::skip_normally()
-  library(Epi)
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = sire[sire$dg_date < sire$ex_date, ],
-             exit.status = as.integer(status %in% 1:2),
-             entry.status = 0,
-             merge = TRUE)
-  
-  TF <- environment()
-  
-  ## phony group variable
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-  agegr <- cut(x$dg_age, 4, right = FALSE)
-  x$agegr <- NULL
-  w <- as.data.frame(table(agegr))
-  names(w) <- c("agegr", "weights")
-  
-  BL <- list(FUT = seq(0, 5, 1/12), CAL = c(2008,2013))
-  
-  form <- Surv(FUT, lex.Xst) ~ group
-  
-  
-  st1 <- survtab(form, data = x,
-                 adjust = agegr,
-                 weights = w, breaks = BL,
-                 surv.type = "surv.obs",
-                 surv.method = "hazard")
-  
-  x <- splitMulti(x, breaks = BL)
-  x$agegr <- cut(x$dg_age, 4, right = FALSE)
-  a <- aggre(x, by = list(group, agegr, FUT))
-  
-  group <- a$group
-  set(a, j = "group", value = NULL)
-  
-  form <- FUT ~ group + adjust(agegr)
-  pyrs <- a$pyrs
-  set(a, j = "pyrs", value = NULL)
-  
-  st2 <- survtab_ag(form, data = a, surv.type = "surv.obs",
-                    surv.method = "hazard",
-                    d = c("from0to1"),
-                    weights = "internal",
-                    pyrs = pyrs)
-  
-  testthat::expect_equal(st1$surv.obs, st2$surv.obs)
-  
-})
-
-
-testthat::test_that("getCall & formula methods for survtab work", {
-  data("sire")
-  BL <- list(fot=seq(0, 5, by = 1/12),
-             per = c("2008-01-01", "2013-01-01"))
-  set.seed(1)
-  x <- lexpand(sire[sample(1:.N, 100)], 
-               birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks = BL,
-               pophaz = popmort,
-               aggre = list(sex, fot))
-  form <- fot ~ sex
-  e <- quote(survtab_ag(formula = form, data = x))
-  st <- eval(e)
-  
-  testthat::expect_equal(formula(st), form)
-  testthat::expect_equal(getCall(st), e)
-  
-})
-
-
-
-
-
-testthat::test_that("survtab_ag allows for certain arguments to be length > 1", {
-  data(sire)
-  set.seed(1)
-  sire <- sire[sample(1:.N, 100)]
-  
-  BL <- list(fot=0:5)
-  x <- lexpand(sire, 
-               birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status,
-               breaks = BL,
-               pophaz = popmort,
-               aggre = list(fot))
-  
-  st1 <- survtab_ag(fot ~ 1, data = x, 
-                    surv.method = "lifetable",
-                    n.cens = c("from0to0", "from0to2"), d = "from0to1")
-  st2 <- survtab_ag(fot ~ 1, data = x, 
-                    surv.method = "lifetable",
-                    relsurv.method = "pp",
-                    n.cens = c("from0to0", "from0to2"), 
-                    d = "from0to1",
-                    d.pp = "from0to1.pp",
-                    d.pp.2 = "from0to1.pp.2",
-                    n.pp = "at.risk.pp",
-                    n.cens.pp = c("from0to0.pp", "from0to2.pp"))
-  st3 <- survtab_ag(fot ~ 1, data = x, 
-                    surv.method = "lifetable",
-                    relsurv.method = "pp",
-                    d = c("from0to0", "from0to2"), 
-                    n.cens = "from0to1",
-                    n.cens.pp = "from0to1.pp",
-                    n.pp = "at.risk.pp",
-                    d.pp = c("from0to0.pp", "from0to2.pp"),
-                    d.pp.2 = c("from0to0.pp.2", "from0to2.pp.2"))
-  
-  testthat::expect_true("surv.obs" %in% names(st1))
-  testthat::expect_true("surv.obs" %in% names(st2))
-  testthat::expect_true("surv.obs" %in% names(st3))
-  testthat::expect_equal(st1[["surv_obs"]], st2[["surv_obs"]])
-  testthat::expect_equal(st2[["surv_obs"]], st3[["surv_obs"]])
-  
-})
-
-
-
-
-
-testthat::test_that("update() works with survtab objects", {
-  data(sire)
-  set.seed(1)
-  sire <- sire[sample(1:.N, 100)]
-  
-  BL <- list(fot=seq(0, 5, by = 1/12),
-             per = c("2008-01-01", "2013-01-01"))
-  x <- lexpand(sire, 
-               birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks = BL,
-               pophaz = popmort,
-               aggre = list(sex, fot))
-  
-  
-  st <- survtab_ag(fot ~ 1, data = x)
-  sts <- survtab_ag(fot ~ sex, data = x)
-  
-  testthat::expect_equal(sts, update(st, formula. = fot ~ sex))
-  
-  
-  library(Epi)
-  
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-             exit = list(CAL = get.yrs(ex_date)), 
-             data = sire[sire$dg_date < sire$ex_date, ],
-             entry.status = 0L,
-             exit.status = as.integer(status %in% 1:2), 
-             merge = TRUE)
-  
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-  
-  st <- survtab(FUT ~ group, data = x, 
-                surv.type = "surv.obs",
-                breaks = list(FUT = seq(0, 5, 1/12)))
-  
-  sts <- survtab(FUT ~ 1, data = x, 
-                 surv.type = "surv.obs",
-                 breaks = list(FUT = seq(0, 5, 1/12)))
-  
-  testthat::expect_equal(sts, update(st, . ~ -group))
-  
-})
-
-
-
-
-
-testthat::test_that("internal weights work as intended", {
-  library("data.table")
-  data("sire")
-  sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
-  BL <- list(fot=seq(0, 5, by = 1/12),
-             per = c("2008-01-01", "2013-01-01"))
-  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks = BL,
-               pophaz = popmort,
-               aggre = list(fot,agegr))
-
-  ## age standardisation using internal weights (age distribution of
-  ## patients diagnosed within the period window)
-  w <- x[fot == 0, .(weights = sum(at.risk)), keyby = agegr]
-
-  st <- survtab_ag(fot ~ adjust(agegr), data = x, weights=w)
-  
-  st2 <- survtab_ag(fot ~ adjust(agegr), data = x, weights = "internal")
-  
-  testthat::expect_equal(st$surv.obs.as.lo, st2$surv.obs.as.lo)
-  
-})
-
-
-
-
-
-testthat::test_that("survtab_ag works with bare data.frames", {
-  
-  data(sire)
-  
-  BL <- list(fot = 0:5,
-             per = c("2008-01-01", "2013-01-01"))
-  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2,
-               breaks = BL,
-               aggre = list(fot))
-  
-  e <- quote(survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs"))
-  eb <- quote(survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs", 
-                         surv.breaks = 0:5))
-  
-  la <- list(eval(e), eval(eb))
-  testthat::expect_equal(la[[1]]$surv.obs.hi, la[[2]]$surv.obs.hi)
-  
-  
-  x <- data.frame(x)
-  er <- paste0("Data did not contain breaks and no breaks ",
-               "were supplied by hand.")
-  testthat::expect_error(eval(e), regexp = er)
-  testthat::expect_equal(eval(eb)$surv.obs.hi, la[[2]]$surv.obs.hi)
-  
-})
-
-
-
-
-
-
-testthat::test_that("confidence intervals are as intended", {
-  popEpi:::skip_normally()
-  
-  library(Epi)
-  
-
-  ## NOTE: recommended to use factor status variable
-  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
-             exit = list(CAL = get.yrs(ex_date)),
-             data = popEpi::sire[sire$dg_date < sire$ex_date, ],
-             entry.status = 0L,
-             exit.status = as.integer(status %in% 1:2),
-             merge = TRUE)
-
-  ## phony group variable
-  set.seed(1L)
-  x$group <- rbinom(nrow(x), 1, 0.5)
-
-  ## log-log transformation
-  st <- survtab(FUT ~ group, data = x,
-                surv.type = "surv.obs",
-                breaks = list(FUT = seq(0, 5, 1/12)),
-                conf.type = "log-log", conf.level = 0.99)
-  
-  dt <- data.table(st)
-  dt[, "SE.A" := sqrt(SE.surv.obs^2*(1/(surv.obs*log(surv.obs)))^2)]
-  dt[, "s.lo" := surv.obs^exp(qnorm(0.995)*SE.A)]
-  dt[, "s.hi" := surv.obs^exp(qnorm(0.005)*SE.A)]
-  
-  testthat::expect_equal(dt[, .(lo = surv.obs.lo, hi = surv.obs.hi)], 
-               dt[, .(lo = s.lo, hi = s.hi)])
-  
-  ## log transformation
-  st <- survtab(FUT ~ group, data = x,
-                surv.type = "surv.obs",
-                breaks = list(FUT = seq(0, 5, 1/12)),
-                conf.type = "log", conf.level = 0.80)
-  
-  dt <- data.table(st)
-  dt[, "SE.A" := SE.surv.obs/surv.obs]
-  dt[, "s.lo" := surv.obs*exp(qnorm(0.10)*SE.A)]
-  dt[, "s.hi" := surv.obs*exp(qnorm(0.90)*SE.A)]
-  
-  testthat::expect_equal(dt[, .(lo = surv.obs.lo, hi = surv.obs.hi)], 
-               dt[, .(lo = s.lo, hi = s.hi)])
-  
-  
-})
-
-
-
-
-
-
+testthat::context("survtab usage")
+
+
+
+testthat::test_that("Dates and frac. yrs produce congruent results", {
+  popEpi:::skip_normally()
+  library(Epi)
+  
+  x <- data.table(popEpi::sire)
+  x <- x[dg_date<ex_date]
+  
+  ## phony group variable
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+  
+  
+  ## yrs
+  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+              exit = list(CAL = get.yrs(ex_date)), 
+              data = x,
+              entry.status = 0L,
+              exit.status = as.integer(status %in% 1:2), 
+              merge = TRUE)
+  
+  ## dates
+  xd <- Lexis(entry = list(FUT = 0L, AGE = dg_date-bi_date, CAL = dg_date),
+              exit = list(CAL = ex_date),
+              data = x,
+              entry.status = 0L,
+              exit.status = as.integer(status %in% 1:2), 
+              merge = TRUE)
+  yd <- 365.242199
+  BLy <- list(FUT = seq(0, 5, 1/4))
+  BLd <- lapply(BLy, function(el) el * yd)
+  
+  pmy <- data.table(popEpi::popmort)
+  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
+  
+  pmd <- data.table(pmy)
+  pmd[, CAL := as.Date(paste0(CAL, "-01-01"))]
+  pmd[, AGE := AGE * yd]
+  pmd[, haz := haz/yd]
+  
+  #### hazard method
+  ## observed survival & Ederer II
+  
+  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                     surv.type = "surv.rel", relsurv.method = "e2",
+                     surv.method = "hazard",
+                     breaks = BLy, pophaz = pmy)
+  
+  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                     surv.type = "surv.rel", relsurv.method = "e2",
+                     surv.method = "hazard",
+                     breaks = BLd, pophaz = pmd)    
+  
+  testthat::expect_equal(sty$surv.obs.lo, std$surv.obs.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$surv.obs, std$surv.obs, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$surv.obs.hi, std$surv.obs.hi, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2.lo, std$r.e2.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2, std$r.e2, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2.hi, std$r.e2.hi, scale = 1L, tolerance = 0.0005)
+  
+  ## pohar perme
+  
+  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                     surv.type = "surv.rel", relsurv.method = "pp",
+                     surv.method = "hazard",
+                     breaks = BLy, pophaz = pmy)
+  
+  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                     surv.type = "surv.rel", relsurv.method = "pp",
+                     surv.method = "hazard",
+                     breaks = BLd, pophaz = pmd)    
+  
+  testthat::expect_equal(sty$r.pp.lo, std$r.pp.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.pp, std$r.pp, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.pp.hi, std$r.pp.hi, scale = 1L, tolerance = 0.0005)
+  
+  #### lifetable method
+  ## observed survival & Ederer II
+  
+  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                     surv.type = "surv.rel", relsurv.method = "e2",
+                     surv.method = "lifetable",
+                     breaks = BLy, pophaz = pmy)
+  
+  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                     surv.type = "surv.rel", relsurv.method = "e2",
+                     surv.method = "lifetable",
+                     breaks = BLd, pophaz = pmd)    
+  
+  testthat::expect_equal(sty$surv.obs.lo, std$surv.obs.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$surv.obs, std$surv.obs, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$surv.obs.hi, std$surv.obs.hi, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2.lo, std$r.e2.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2, std$r.e2, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.e2.hi, std$r.e2.hi, scale = 1L, tolerance = 0.0005)
+  
+  ## pohar perme
+  
+  sty <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy, 
+                     surv.type = "surv.rel", relsurv.method = "pp",
+                     surv.method = "lifetable",
+                     breaks = BLy, pophaz = pmy)
+  
+  std <- survtab(Surv(FUT, lex.Xst) ~ group, data = xd, 
+                     surv.type = "surv.rel", relsurv.method = "pp",
+                     surv.method = "lifetable",
+                     breaks = BLd, pophaz = pmd)    
+  
+  testthat::expect_equal(sty$r.pp.lo, std$r.pp.lo, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.pp, std$r.pp, scale = 1L, tolerance = 0.0005)
+  testthat::expect_equal(sty$r.pp.hi, std$r.pp.hi, scale = 1L, tolerance = 0.0005)
+  
+})
+
+
+
+testthat::test_that("hazard and lifetable produce congruent results", {
+  popEpi:::skip_normally()
+  library(Epi)
+  
+  x <- data.table(popEpi::sire)
+  x <- x[dg_date<ex_date]
+  
+  ## phony group variable
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+  
+  
+  xy <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+              exit = list(CAL = get.yrs(ex_date)), 
+              data = x,
+              entry.status = 0L,
+              exit.status = as.integer(status %in% 1:2), 
+              merge = TRUE)
+  
+  
+  pmy <- data.table(popEpi::popmort)
+  setnames(pmy, c("year", "agegroup"), c("CAL", "AGE"))
+  BL <- list(FUT = seq(0, 5, 1/12))
+  
+  sth <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy,
+                 pophaz = pmy, breaks = BL,
+                 surv.type = "surv.rel", surv.method = "hazard")
+  
+  stl <- survtab(Surv(FUT, lex.Xst) ~ group, data = xy,
+                 pophaz = pmy, breaks = BL,
+                 surv.type = "surv.rel", surv.method = "lifetable")
+  
+  testthat::expect_equal(sth$r.e2, stl$r.e2, scale = 1, tol = 0.0003415)
+  testthat::expect_equal(sth$r.e2.lo, stl$r.e2.lo, scale = 1, tol = 0.000354)
+  testthat::expect_equal(sth$r.e2.hi, stl$r.e2.hi, scale = 1, tol = 0.00033)
+  
+  testthat::expect_equal(sth$surv.obs, stl$surv.obs, scale = 1, tol = 0.00002575)
+  testthat::expect_equal(sth$surv.obs.lo, stl$surv.obs.lo, scale = 1, tol = 0.000027)
+  testthat::expect_equal(sth$surv.obs.hi, stl$surv.obs.hi, scale = 1, tol = 0.000025)
+  
+})
+
+## test below even should not work, since period analysis
+## means that at.risk counts (subjects entering an interval in its beginning)
+## will not necessarily equal the counts of events.
+# test_that("lifetable counts work with period analysis", {
+#   library(Epi)
+#   x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+#              exit = list(CAL = get.yrs(ex_date)),
+#              data = sire[sire$dg_date < sire$ex_date, ],
+#              exit.status = as.integer(status %in% 1:2),
+#              entry.status = 0,
+#              merge = TRUE)
+# 
+#   ## phony group variable
+#   set.seed(1L)
+#   x$group <- rbinom(nrow(x), 1, 0.5)
+# 
+#   BL <- list(FUT = seq(0, 5, 1/12), CAL = c(2008,2013))
+#   x <- splitMulti(x, BL)
+# 
+#   st1 <- survtab(Surv(FUT, lex.Xst) ~ 1, data = x, 
+#                  surv.type = "surv.obs",
+#                  surv.method = "lifetable")
+#   
+#   a <- aggre(x, by = FUT)
+#   
+#   st2 <- survtab_ag(FUT ~ 1, data = a, surv.type = "surv.obs",
+#                     surv.method = "lifetable",
+#                     n = "at.risk", d = c("from0to1"),
+#                     n.cens = "from0to0", pyrs = "pyrs")
+#   
+#   expect_equal(st1$surv.obs, st2$surv.obs)
+#   
+# })
+
+
+testthat::test_that("its possible to pass dynamically created arguments", {
+  popEpi:::skip_normally()
+  library(Epi)
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = sire[sire$dg_date < sire$ex_date, ],
+             exit.status = as.integer(status %in% 1:2),
+             entry.status = 0,
+             merge = TRUE)
+  
+  TF <- environment()
+  
+  ## phony group variable
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+  agegr <- cut(x$dg_age, 4, right = FALSE)
+  x$agegr <- NULL
+  w <- as.data.frame(table(agegr))
+  names(w) <- c("agegr", "weights")
+  
+  BL <- list(FUT = seq(0, 5, 1/12), CAL = c(2008,2013))
+  
+  form <- Surv(FUT, lex.Xst) ~ group
+  
+  
+  st1 <- survtab(form, data = x,
+                 adjust = agegr,
+                 weights = w, breaks = BL,
+                 surv.type = "surv.obs",
+                 surv.method = "hazard")
+  
+  x <- splitMulti(x, breaks = BL)
+  x$agegr <- cut(x$dg_age, 4, right = FALSE)
+  a <- aggre(x, by = list(group, agegr, FUT))
+  
+  group <- a$group
+  set(a, j = "group", value = NULL)
+  
+  form <- FUT ~ group + adjust(agegr)
+  pyrs <- a$pyrs
+  set(a, j = "pyrs", value = NULL)
+  
+  st2 <- survtab_ag(form, data = a, surv.type = "surv.obs",
+                    surv.method = "hazard",
+                    d = c("from0to1"),
+                    weights = "internal",
+                    pyrs = pyrs)
+  
+  testthat::expect_equal(st1$surv.obs, st2$surv.obs)
+  
+})
+
+
+testthat::test_that("getCall & formula methods for survtab work", {
+  data("sire")
+  BL <- list(fot=seq(0, 5, by = 1/12),
+             per = c("2008-01-01", "2013-01-01"))
+  set.seed(1)
+  x <- lexpand(sire[sample(1:.N, 100)], 
+               birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks = BL,
+               pophaz = popmort,
+               aggre = list(sex, fot))
+  form <- fot ~ sex
+  e <- quote(survtab_ag(formula = form, data = x))
+  st <- eval(e)
+  
+  testthat::expect_equal(formula(st), form)
+  testthat::expect_equal(getCall(st), e)
+  
+})
+
+
+
+
+
+testthat::test_that("survtab_ag allows for certain arguments to be length > 1", {
+  data(sire)
+  set.seed(1)
+  sire <- sire[sample(1:.N, 100)]
+  
+  BL <- list(fot=0:5)
+  x <- lexpand(sire, 
+               birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status,
+               breaks = BL,
+               pophaz = popmort,
+               aggre = list(fot))
+  
+  st1 <- survtab_ag(fot ~ 1, data = x, 
+                    surv.method = "lifetable",
+                    n.cens = c("from0to0", "from0to2"), d = "from0to1")
+  st2 <- survtab_ag(fot ~ 1, data = x, 
+                    surv.method = "lifetable",
+                    relsurv.method = "pp",
+                    n.cens = c("from0to0", "from0to2"), 
+                    d = "from0to1",
+                    d.pp = "from0to1.pp",
+                    d.pp.2 = "from0to1.pp.2",
+                    n.pp = "at.risk.pp",
+                    n.cens.pp = c("from0to0.pp", "from0to2.pp"))
+  st3 <- survtab_ag(fot ~ 1, data = x, 
+                    surv.method = "lifetable",
+                    relsurv.method = "pp",
+                    d = c("from0to0", "from0to2"), 
+                    n.cens = "from0to1",
+                    n.cens.pp = "from0to1.pp",
+                    n.pp = "at.risk.pp",
+                    d.pp = c("from0to0.pp", "from0to2.pp"),
+                    d.pp.2 = c("from0to0.pp.2", "from0to2.pp.2"))
+  
+  testthat::expect_true("surv.obs" %in% names(st1))
+  testthat::expect_true("surv.obs" %in% names(st2))
+  testthat::expect_true("surv.obs" %in% names(st3))
+  testthat::expect_equal(st1[["surv_obs"]], st2[["surv_obs"]])
+  testthat::expect_equal(st2[["surv_obs"]], st3[["surv_obs"]])
+  
+})
+
+
+
+
+
+testthat::test_that("update() works with survtab objects", {
+  data(sire)
+  set.seed(1)
+  sire <- sire[sample(1:.N, 100)]
+  
+  BL <- list(fot=seq(0, 5, by = 1/12),
+             per = c("2008-01-01", "2013-01-01"))
+  x <- lexpand(sire, 
+               birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks = BL,
+               pophaz = popmort,
+               aggre = list(sex, fot))
+  
+  
+  st <- survtab_ag(fot ~ 1, data = x)
+  sts <- survtab_ag(fot ~ sex, data = x)
+  
+  testthat::expect_equal(sts, update(st, formula. = fot ~ sex))
+  
+  
+  library(Epi)
+  
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+             exit = list(CAL = get.yrs(ex_date)), 
+             data = sire[sire$dg_date < sire$ex_date, ],
+             entry.status = 0L,
+             exit.status = as.integer(status %in% 1:2), 
+             merge = TRUE)
+  
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+  
+  st <- survtab(FUT ~ group, data = x, 
+                surv.type = "surv.obs",
+                breaks = list(FUT = seq(0, 5, 1/12)))
+  
+  sts <- survtab(FUT ~ 1, data = x, 
+                 surv.type = "surv.obs",
+                 breaks = list(FUT = seq(0, 5, 1/12)))
+  
+  testthat::expect_equal(sts, update(st, . ~ -group))
+  
+})
+
+
+
+
+
+testthat::test_that("internal weights work as intended", {
+  library("data.table")
+  data("sire")
+  sire$agegr <- cut(sire$dg_age,c(0,45,55,65,75,Inf),right=FALSE)
+  BL <- list(fot=seq(0, 5, by = 1/12),
+             per = c("2008-01-01", "2013-01-01"))
+  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks = BL,
+               pophaz = popmort,
+               aggre = list(fot,agegr))
+
+  ## age standardisation using internal weights (age distribution of
+  ## patients diagnosed within the period window)
+  w <- x[fot == 0, .(weights = sum(at.risk)), keyby = agegr]
+
+  st <- survtab_ag(fot ~ adjust(agegr), data = x, weights=w)
+  
+  st2 <- survtab_ag(fot ~ adjust(agegr), data = x, weights = "internal")
+  
+  testthat::expect_equal(st$surv.obs.as.lo, st2$surv.obs.as.lo)
+  
+})
+
+
+
+
+
+testthat::test_that("survtab_ag works with bare data.frames", {
+  
+  data(sire)
+  
+  BL <- list(fot = 0:5,
+             per = c("2008-01-01", "2013-01-01"))
+  x <- lexpand(sire, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2,
+               breaks = BL,
+               aggre = list(fot))
+  
+  e <- quote(survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs"))
+  eb <- quote(survtab_ag(fot ~ 1, data = x, surv.type = "surv.obs", 
+                         surv.breaks = 0:5))
+  
+  la <- list(eval(e), eval(eb))
+  testthat::expect_equal(la[[1]]$surv.obs.hi, la[[2]]$surv.obs.hi)
+  
+  
+  x <- data.frame(x)
+  er <- paste0("Data did not contain breaks and no breaks ",
+               "were supplied by hand.")
+  testthat::expect_error(eval(e), regexp = er)
+  testthat::expect_equal(eval(eb)$surv.obs.hi, la[[2]]$surv.obs.hi)
+  
+})
+
+
+
+
+
+
+testthat::test_that("confidence intervals are as intended", {
+  popEpi:::skip_normally()
+  
+  library(Epi)
+  
+
+  ## NOTE: recommended to use factor status variable
+  x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)),
+             exit = list(CAL = get.yrs(ex_date)),
+             data = popEpi::sire[sire$dg_date < sire$ex_date, ],
+             entry.status = 0L,
+             exit.status = as.integer(status %in% 1:2),
+             merge = TRUE)
+
+  ## phony group variable
+  set.seed(1L)
+  x$group <- rbinom(nrow(x), 1, 0.5)
+
+  ## log-log transformation
+  st <- survtab(FUT ~ group, data = x,
+                surv.type = "surv.obs",
+                breaks = list(FUT = seq(0, 5, 1/12)),
+                conf.type = "log-log", conf.level = 0.99)
+  
+  dt <- data.table(st)
+  dt[, "SE.A" := sqrt(SE.surv.obs^2*(1/(surv.obs*log(surv.obs)))^2)]
+  dt[, "s.lo" := surv.obs^exp(qnorm(0.995)*SE.A)]
+  dt[, "s.hi" := surv.obs^exp(qnorm(0.005)*SE.A)]
+  
+  testthat::expect_equal(dt[, .(lo = surv.obs.lo, hi = surv.obs.hi)], 
+               dt[, .(lo = s.lo, hi = s.hi)])
+  
+  ## log transformation
+  st <- survtab(FUT ~ group, data = x,
+                surv.type = "surv.obs",
+                breaks = list(FUT = seq(0, 5, 1/12)),
+                conf.type = "log", conf.level = 0.80)
+  
+  dt <- data.table(st)
+  dt[, "SE.A" := SE.surv.obs/surv.obs]
+  dt[, "s.lo" := surv.obs*exp(qnorm(0.10)*SE.A)]
+  dt[, "s.hi" := surv.obs*exp(qnorm(0.90)*SE.A)]
+  
+  testthat::expect_equal(dt[, .(lo = surv.obs.lo, hi = surv.obs.hi)], 
+               dt[, .(lo = s.lo, hi = s.hi)])
+  
+  
+})
+
+
+
+
+
+
diff --git a/tests/testthat/test_utils.R b/tests/testthat/test_utils.R
index 78ba9c8..3df76bc 100644
--- a/tests/testthat/test_utils.R
+++ b/tests/testthat/test_utils.R
@@ -1,483 +1,483 @@
-testthat::context("utility functions")
-
-testthat::test_that("subsetting in ltable works and ltable has no side effects", {
-  popEpi:::skip_normally()
-  
-  sr <- popEpi::sire[1:100, ]
-  set.seed(1L)
-  sr[, sex := rbinom(.N, 1, prob = 0.5)]
-  sr[c(1, 50), sex := NA]
-  
-  setkeyv(sr, "bi_date")
-  old_sr <- copy(sr)
-  
-  lt1 <- ltable(sr, by = "sex", subset = sex == 0, na.rm = TRUE)
-  lt2 <- ltable(sr, by = "sex", subset = sex == 1, na.rm = TRUE)
-  lt3 <- ltable(sr, by = "sex", na.rm = TRUE)
-  
-  testthat::expect_equal(lt3$obs, c(lt1[1, ]$obs, lt2[2, ]$obs))
-  testthat::expect_true(all.equal(sr, old_sr))
-  
-})
-
-
-
-testthat::test_that("ltable works with NA values", {
-  
-  popEpi:::skip_normally()
-  
-  sr <- setDT(popEpi::sire[1:100, ])
-  set.seed(1L)
-  sr[, sex := rbinom(.N, 1, prob = 0.5)]
-  sr[c(1, 50), sex := NA]
-  
-  lt1 <- ltable(sr, by = "sex", na.rm = FALSE)
-  lt2 <- ltable(sr, by = "sex", na.rm = TRUE)
-  
-  testthat::expect_equal(lt1[!is.na(sex),], lt2)
-  
-})
-
-
-
-testthat::test_that("evalPopArg produces intended results",{
-  set.seed(1L)
-  dt <- data.table(a = rbinom(10, 100, 0.25), b = 1:2, c = 1:5)
-  
-  tf <- function(x=dt, arg) {
-    
-    as <- substitute(arg)
-    byTab <- evalPopArg(x, arg = as, enclos = parent.frame(1L))
-    
-    x[, list(sum = sum(a)), by = byTab]
-    
-  }
-  
-  ## symbol
-  t1 <- tf(arg=b)
-  
-  ## name string
-  t2 <- tf(arg="b")
-  
-  testthat::expect_equal(t1$sum, c(127, 131))
-  testthat::expect_equal(t1, t2)
-  
-  ## list of symbols / expressions
-  t3 <- tf(arg=list(b, c))
-  
-  ## name strings
-  t4 <- tf(arg=c("b", "c"))
-  
-  ## object containing name strings
-  byVars <- c("b", "c")
-  t5 <- tf(arg=byVars)
-  
-  testthat::expect_equal(t4$sum, c(22,24,26,31,21, 31,32,27,26,18))
-  testthat::expect_equal(t4, t3)
-  testthat::expect_equal(t4, t5)
-  
-  ## list of symbols / expressions
-  t6 <- tf(arg=list(var1 = b,c, cut(c,3)))
-  testthat::expect_equal(names(t6), c("var1", "c", "cut", "sum"))
-  
-  
-  ## NULL object
-  byVars <- NULL
-  t7 <- tf(arg=byVars)
-  t8 <- tf(arg=NULL)
-  testthat::expect_equal(t7, t8)
-  
-  ## a list of predetermined values
-  byList <- as.list(dt[, list(b, var1 = c)])
-  t9 <- tf(arg=byList)
-  
-  ## list without any names
-  byList <- list(dt$b, dt$c)
-  t10<- tf(arg=byList)
-  
-  ## partially named list
-  byList <- list(var1 = dt$b, dt$c)
-  t11<- tf(arg=byList)
-  
-  testthat::expect_equal(t9$sum, t10$sum)
-  testthat::expect_equal(t10$sum, t11$sum)
-  testthat::expect_equal(names(t11), c("var1", "BV2", "sum"))
-  
-  
-  t12 <- tf(arg=list(V0=dt$b, dt$c))
-  byList <- list(V0 = dt$b, dt$c)
-  t13 <- tf(arg=byList)
-  testthat::expect_equal(t12, t13)
-  
-  ## pre-substituted list
-  bl <- substitute(byList)
-  t14 <- tf(arg = bl)
-  testthat::expect_equal(t12, t14)
-  
-  ## pre-substituted vector of names
-  nv <- c("a", "b")
-  nvs <- substitute(nv)
-  t15a <- tf(arg = nv)
-  t15b <- tf(arg = nvs)
-  testthat::expect_equal(t15a, t15b)
-  
-  ## nested functions
-  tf2 <- function(a, x = dt) {
-    tf(x = x, arg = a)
-  }
-  
-  nv <- c("a", "b")
-  nvs <- substitute(nv)
-  t15a <- tf2(a = nv)
-  t15b <- tf2(a = nvs)
-  testthat::expect_equal(t15a, t15b)
-})
-
-
-testthat::test_that("cutLowMerge merges succesfully what is intended", {
-  popEpi:::skip_normally()
-  all_names_present(popEpi::popmort, c("sex", "year", "agegroup", "haz"))
-  all_names_present(popEpi::sire, c("sex", "bi_date", "dg_date", "ex_date", "status"))
-  
-  pm <- copy(popEpi::popmort)
-  pm[, haz := rbinom(.N, 100, 0.5)/1e5L]
-  
-  sr <- popEpi::sire[1:100,]
-  setDT(sr)
-  sr1 <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
-                 status = status, fot = seq(0, 5, 1/12))
-  sr1 <- data.table(sr1)
-  setattr(sr1, "class", c("Lexis", "data.table", "data.frame"))
-  
-  sr1[, year := per + 0.5*lex.dur]
-  sr1[, agegroup := age + 0.5*lex.dur]
-  
-  sr2 <- cutLowMerge(sr1, pm,
-                     by.x = c("sex", "per", "age"), 
-                     by.y = c("sex", "year", "agegroup"),
-                     all.x = TRUE, all.y = FALSE, old.nums = TRUE)
-  
-  sr3 <- copy(sr2)
-  sr3[, haz := NULL]
-  
-  sr4 <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
-                 status = status, fot = seq(0, 5, 1/12), pophaz = pm, pp = FALSE)
-  testthat::expect_equal(sr1, sr3, check.attributes = FALSE)
-  testthat::expect_equal(sr2$haz*1e5L, sr4$pop.haz*1e5L, check.attributes = FALSE)
-  
-  sr1[, year := popEpi:::cutLow(year, breaks = sort(unique(pm$year)))]
-  sr1[, agegroup := popEpi:::cutLow(agegroup, breaks = sort(unique(pm$agegroup)))]
-  
-  sr5 <- merge(sr1, pm, by = c("sex", "year", "agegroup"))
-  setDT(sr5)
-  setkey(sr5, lex.id, fot)
-  
-  testthat::expect_equal(sr4$haz*1e5L, sr5$pop.haz*1e5L, check.attributes = FALSE)
-})
-
-testthat::test_that("detectEvents works as intended", {
-  popEpi:::skip_normally()
-  x <- sire[dg_date<ex_date,]
-  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, pophaz = data.table(popEpi::popmort),
-               breaks = list(fot = seq(0,5,1/12), per = c(2007,2012), age = c(50,90)),
-               drop = TRUE)
-  ## this will only work with drop = TRUE.
-  setkeyv(x, c("lex.id", "fot", "per", "age"))
-  
-  ## this leaves observations cut short due to age or period censoring to
-  ## really be censoring.
-  x[, event := detectEvents(x, breaks = attr(x, "breaks")["fot"], by = "lex.id")]
-  
-  x[, alt.event := 0L]
-  x[!duplicated(lex.id, fromLast = TRUE), alt.event := 2L]
-  x[lex.Cst != lex.Xst, alt.event := 1L]
-  x[fot+lex.dur == 5L, alt.event := 0L]
-  
-  testthat::expect_equal(x$event, x$alt.event)
-  
-})
-
-testthat::test_that("comp_pp_weighted_figures produces intended results", {
-  set.seed(1L)
-  x <- sire[dg_date < ex_date,][sample(x = .N, size = 5L, replace = FALSE),]
-  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status %in% 1:2, pophaz = data.table(popEpi::popmort),
-               breaks = list(fot = seq(0,20,1/12), per = 1993:2013, age = 0:200))
-  x[, event := detectEvents(x, breaks = attr(x, "breaks"), by = "lex.id")]
-  
-  l <- comp_pp_weighted_figures(lex = x, haz = "pop.haz", pp = "pp", event.ind = "event", by = "lex.id")
-  x[, names(l) := l]
-  
-  testthat::expect_equal(x[event %in% 1:2, pp], x[event %in% 1:2, from0to0.pp + from0to1.pp])
-  testthat::expect_equal(x[event %in% 1:2, sum(pp)], x[, sum(from0to0.pp + from0to1.pp)])
-  testthat::expect_equal(x[, lex.dur*pp], x$ptime.pp)
-  testthat::expect_equal(x[, lex.dur*pp*pop.haz], x$d.exp.pp)
-  testthat::expect_equal(x[event == 1L, pp^2], x[event == 1L,]$from0to1.pp.2)
-  
-})
-
-
-testthat::test_that("evalPopFormula & usePopFormula output is stable", {
-  evalPopFormula <- popEpi:::evalPopFormula
-  usePopFormula <- popEpi:::usePopFormula
-  
-  x <- sire[1:5, ]
-  x[, "status" := c(1,1,0,1,1)]
-  x[, "sex" := c(1,0,1,0,1)]
-  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
-               status = status)
-  
-  f1a <- Surv(fot, lex.Xst) ~ 1
-  f1b <- Surv(fot, event = lex.Xst) ~ 1
-  f2 <- Surv(fot, lex.Xst) ~ sex
-  f3 <- Surv(fot, lex.Xst) ~ sex + adjust(factor(sex + 1))
-  f4 <- Surv(fot, lex.Xst) ~ adjust(factor(sex + 1))
-  f5 <- lex.Xst ~ 1
-  f6 <- lex.Xst ~ sex
-  f7 <- lex.Xst ~ sex + adjust(factor(sex + 1))
-  f8 <- lex.Xst ~ adjust(factor(sex + 1))
-  f9 <- lex.Xst ~ factor(sex)
-  
-  TF <- environment()
-  
-  res <- data.table(time = rep(0, 5), status = c(1,1,0,1,1), sex = c(1,0,1,0,1))
-  res[["factor(sex + 1)"]] <- factor(res$sex + 1)
-  res$lex.Xst <- as.integer(c(1,1,0,1,1))
-  
-  
-  ## evalPopFormula
-  r1a <- evalPopFormula(f1a, data = x, enclos = TF, Surv.response = TRUE)
-  r1b <- evalPopFormula(f1b, data = x, enclos = TF, Surv.response = TRUE)
-  setattr(r1a, "formula", attr(r1b, "formula"))
-  testthat::expect_equal(r1a, r1b)
-  testthat::expect_equal(data.table(r1a), res[, list(time, status)])
-  
-  r5 <- evalPopFormula(f5, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
-  r6 <- evalPopFormula(f6, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
-  testthat::expect_equal(r5$lex.Xst, c(1,1,0,1,1))
-  testthat::expect_equal(data.table(r6), res[, list(lex.Xst, sex)])
-  
-  ## model-type naming of columns
-  r9 <- evalPopFormula(f9, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
-  testthat::expect_equal(data.table(r9), res[, list(lex.Xst, "factor(sex)" = factor(sex))])
-  
-  ## multiple variables, model-type naming of columns with adjust
-  r3 <- evalPopFormula(f3, data = x, enclos = parent.frame(1L), Surv.response = TRUE)
-  testthat::expect_equivalent(r3, res[, .SD, .SDcols = names(r3)])
-  
-  
-  #### only the survival time scale as response
-  fa <- fot ~ 1
-  fb <- fot ~ sex
-  fc <- fot ~ sex + adjust(factor(sex + 1))
-  fd <- fot ~ adjust(factor(sex + 1))
-  
-  ra <- evalPopFormula(fa, data = x, enclos = TF, Surv.response = "either")
-  testthat::expect_equivalent(ra, x[, .(fot, lex.Xst)])
-  rb <- evalPopFormula(fb, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
-  testthat::expect_equivalent(rb, x[, .(fot, sex)])
-  rc <- evalPopFormula(fc, data = x, enclos = parent.frame(1L), Surv.response = "either")
-  testthat::expect_equivalent(rc, x[, .(fot, lex.Xst, sex, factor(sex + 1))])
-  rd <- evalPopFormula(fd, data = x, enclos = TF, Surv.response = FALSE)
-  testthat::expect_equivalent(rd, x[, .(fot, factor(sex + 1))])
-  
-  ## usePopFormula
-  
-  r3 <- usePopFormula(f3, data = x, enclos = parent.frame(2L), Surv.response = TRUE)
-  testthat::expect_equivalent(r3, 
-                    list(y = res[, list(time, status)], 
-                         print = res[, list(sex)], 
-                         adjust = res[, "factor(sex + 1)", with = FALSE],
-                         formula = f3) 
-  )
-  r4 <- usePopFormula(f4, data = x, enclos = parent.frame(2L), Surv.response = TRUE)
-  testthat::expect_equivalent(r4, 
-                    list(y = res[, list(time, status)], 
-                         print = NULL, 
-                         adjust = res[, "factor(sex + 1)", with = FALSE],
-                         formula = f4) 
-  )
-  r6 <- usePopFormula(f6, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
-  testthat::expect_equivalent(r6, 
-                    list(y = res[, list(lex.Xst)], 
-                         print = res[, list(sex)], 
-                         adjust = NULL,
-                         formula = f6) 
-  )
-  
-  r7 <- usePopFormula(f7, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
-  testthat::expect_equivalent(r7, 
-                    list(y = res[, list(lex.Xst)], 
-                         print = res[, list(sex)], 
-                         adjust = res[, "factor(sex + 1)", with = FALSE],
-                         formula = f7) 
-  )
-  r8 <- usePopFormula(f8, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
-  testthat::expect_equivalent(r8, 
-                    list(y = res[, list(lex.Xst)], 
-                         print = NULL, 
-                         adjust = res[, "factor(sex + 1)", with = FALSE],
-                         formula = f8) 
-  )
-  r9 <- usePopFormula(lex.Xst ~ sex, data = x, adjust = quote(factor(sex + 1)),
-                      enclos = parent.frame(2L), Surv.response = FALSE) 
-  testthat::expect_equivalent(r9, 
-                    list(y = res[, list(lex.Xst)], 
-                         print = res[, list(sex)], 
-                         adjust = res[, "factor(sex + 1)", with = FALSE],
-                         formula = lex.Xst ~ sex)
-  )
-  testthat::expect_equal(lapply(r9, names), list(y = "lex.Xst", print = "sex", adjust = "factor(sex + 1)", formula = NULL))
-  
-  r9 <- usePopFormula(lex.Xst ~ as.numeric(sex), data = x, adjust = quote(list(factor(sex + 1), factor(sex - 1))),
-                      enclos = parent.frame(2L), Surv.response = FALSE) 
-  
-  testthat::expect_equal(lapply(r9, names), list(y = "lex.Xst", print = "as.numeric(sex)", 
-                                       adjust = c("factor(sex + 1)", "factor(sex - 1)"), formula = NULL))
-  
-  
-  ra <- usePopFormula(fa, data = x, 
-                      adjust = quote(list(factor(sex + 1), factor(sex - 1))),
-                      enclos = parent.frame(2L), Surv.response = FALSE) 
-  
-  testthat::expect_equal(lapply(ra, names), 
-               list(y = "fot", print = NULL, 
-                    adjust = c("factor(sex + 1)", "factor(sex - 1)"), 
-                    formula = NULL))
-  
-  
-  rb <- usePopFormula(fb, data = x, 
-                      adjust = quote(list(factor(sex + 1), factor(sex - 1))),
-                      enclos = parent.frame(2L), Surv.response = FALSE) 
-  testthat::expect_equal(lapply(rb, names), 
-               list(y = "fot", print = "sex", 
-                    adjust = c("factor(sex + 1)", "factor(sex - 1)"), 
-                    formula = NULL))
-  
-  rc <- usePopFormula(fc, data = x, 
-                      adjust = NULL,
-                      enclos = parent.frame(2L), Surv.response = "either") 
-  testthat::expect_equal(lapply(rc, names), 
-               list(y = c("time", "status"), print = "sex", 
-                    adjust = "factor(sex + 1)", 
-                    formula = NULL))
-  
-  rd <- usePopFormula(fd, data = x, 
-                      adjust = NULL,
-                      enclos = parent.frame(2L), Surv.response = FALSE) 
-  testthat::expect_equal(lapply(rd, names), 
-               list(y = "fot", print = NULL, 
-                    adjust = "factor(sex + 1)", 
-                    formula = NULL))
-  
-  ## usePopFormula with "either" response
-  useForms <- paste0("f", 2:9)
-  useForms <- c("f1a", "f1a", useForms)
-  useForms <- intersect(useForms, ls())
-  TF <- environment()
-  l <- list()
-  for (k in seq_along(useForms)) {
-    l[[k]] <- usePopFormula(get(useForms[k], envir = TF), data = x, 
-                            adjust = NULL,
-                            enclos = TF, Surv.response = "either") 
-  }
-  
-  
-  
-  
-})
-
-
-
-
-
-testthat::test_that("fractional years computation works", {
-  library("data.table")
-  c <- paste0("2004-", c("01-01", "07-01", "12-31", "02-01"))
-  D <- as.Date(c)
-  
-  testthat::expect_equal(get.yrs(c), get.yrs(D))
-  
-  yl <- 365.242199
-  my <- year(D) + (yday(D) - 1L)/yl
-  
-  testthat::expect_equal(as.numeric(get.yrs(D)), my)
-  
-  yl <- ifelse(is_leap_year(year(D)), 366L, 365L)
-  my <- year(D) + (yday(D) - 1L)/yl
-  
-  testthat::expect_equal(as.numeric(get.yrs(D, year.length = "actual")), my)
-})
-
-
-
-
-
-testthat::test_that("subsetDTorDF works as intended", {
-  
-  dt <- data.table::data.table(a = 1:5, b = 5:1)
-  df <- data.frame(a = 1:5, b = 5:1)
-  
-  sub <- c(TRUE, TRUE, FALSE, FALSE, TRUE)
-  col_nms <- rep("a", 50)
-  settings <- list(
-    list(subset = sub, select = NULL),
-    list(subset = NULL, select = col_nms),
-    list(subset = NULL, select = NULL),
-    list(subset = sub, select = col_nms)
-  )
-  
-  expected_df <- list(
-    df[sub, ],
-    df[, col_nms],
-    df,
-    df[sub, col_nms]
-  )
-  expected_dt <- list(
-    dt[sub, ],
-    dt[, .SD, .SDcols = col_nms],
-    dt,
-    dt[sub, .SD, .SDcols = col_nms]
-  )
-  
-  unused <- lapply(seq_along(settings), function(i) {
-    set <- settings[[i]]
-    set[["data"]] <- dt
-    sub_dt <- do.call(subsetDTorDF, set)
-    set[["data"]] <- df
-    sub_df <- do.call(subsetDTorDF, set)
-    
-    exp_df <- expected_df[[i]]
-    exp_dt <- expected_dt[[i]]
-    
-    list(sub_dt = sub_dt, exp_dt = exp_dt, sub_df = sub_df, exp_df = exp_df)
-    testthat::expect_equal(
-      data.frame(sub_dt),
-      data.frame(exp_dt)
-    )
-    testthat::expect_equal(
-      sub_df,
-      exp_df
-    )
-  })
-  
-  
-})
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+testthat::context("utility functions")
+
+testthat::test_that("subsetting in ltable works and ltable has no side effects", {
+  popEpi:::skip_normally()
+  
+  sr <- popEpi::sire[1:100, ]
+  set.seed(1L)
+  sr[, sex := rbinom(.N, 1, prob = 0.5)]
+  sr[c(1, 50), sex := NA]
+  
+  setkeyv(sr, "bi_date")
+  old_sr <- copy(sr)
+  
+  lt1 <- ltable(sr, by = "sex", subset = sex == 0, na.rm = TRUE)
+  lt2 <- ltable(sr, by = "sex", subset = sex == 1, na.rm = TRUE)
+  lt3 <- ltable(sr, by = "sex", na.rm = TRUE)
+  
+  testthat::expect_equal(lt3$obs, c(lt1[1, ]$obs, lt2[2, ]$obs))
+  testthat::expect_true(all.equal(sr, old_sr))
+  
+})
+
+
+
+testthat::test_that("ltable works with NA values", {
+  
+  popEpi:::skip_normally()
+  
+  sr <- setDT(popEpi::sire[1:100, ])
+  set.seed(1L)
+  sr[, sex := rbinom(.N, 1, prob = 0.5)]
+  sr[c(1, 50), sex := NA]
+  
+  lt1 <- ltable(sr, by = "sex", na.rm = FALSE)
+  lt2 <- ltable(sr, by = "sex", na.rm = TRUE)
+  
+  testthat::expect_equal(lt1[!is.na(sex),], lt2)
+  
+})
+
+
+
+testthat::test_that("evalPopArg produces intended results",{
+  set.seed(1L)
+  dt <- data.table(a = rbinom(10, 100, 0.25), b = 1:2, c = 1:5)
+  
+  tf <- function(x=dt, arg) {
+    
+    as <- substitute(arg)
+    byTab <- evalPopArg(x, arg = as, enclos = parent.frame(1L))
+    
+    x[, list(sum = sum(a)), by = byTab]
+    
+  }
+  
+  ## symbol
+  t1 <- tf(arg=b)
+  
+  ## name string
+  t2 <- tf(arg="b")
+  
+  testthat::expect_equal(t1$sum, c(127, 131))
+  testthat::expect_equal(t1, t2)
+  
+  ## list of symbols / expressions
+  t3 <- tf(arg=list(b, c))
+  
+  ## name strings
+  t4 <- tf(arg=c("b", "c"))
+  
+  ## object containing name strings
+  byVars <- c("b", "c")
+  t5 <- tf(arg=byVars)
+  
+  testthat::expect_equal(t4$sum, c(22,24,26,31,21, 31,32,27,26,18))
+  testthat::expect_equal(t4, t3)
+  testthat::expect_equal(t4, t5)
+  
+  ## list of symbols / expressions
+  t6 <- tf(arg=list(var1 = b,c, cut(c,3)))
+  testthat::expect_equal(names(t6), c("var1", "c", "cut", "sum"))
+  
+  
+  ## NULL object
+  byVars <- NULL
+  t7 <- tf(arg=byVars)
+  t8 <- tf(arg=NULL)
+  testthat::expect_equal(t7, t8)
+  
+  ## a list of predetermined values
+  byList <- as.list(dt[, list(b, var1 = c)])
+  t9 <- tf(arg=byList)
+  
+  ## list without any names
+  byList <- list(dt$b, dt$c)
+  t10<- tf(arg=byList)
+  
+  ## partially named list
+  byList <- list(var1 = dt$b, dt$c)
+  t11<- tf(arg=byList)
+  
+  testthat::expect_equal(t9$sum, t10$sum)
+  testthat::expect_equal(t10$sum, t11$sum)
+  testthat::expect_equal(names(t11), c("var1", "BV2", "sum"))
+  
+  
+  t12 <- tf(arg=list(V0=dt$b, dt$c))
+  byList <- list(V0 = dt$b, dt$c)
+  t13 <- tf(arg=byList)
+  testthat::expect_equal(t12, t13)
+  
+  ## pre-substituted list
+  bl <- substitute(byList)
+  t14 <- tf(arg = bl)
+  testthat::expect_equal(t12, t14)
+  
+  ## pre-substituted vector of names
+  nv <- c("a", "b")
+  nvs <- substitute(nv)
+  t15a <- tf(arg = nv)
+  t15b <- tf(arg = nvs)
+  testthat::expect_equal(t15a, t15b)
+  
+  ## nested functions
+  tf2 <- function(a, x = dt) {
+    tf(x = x, arg = a)
+  }
+  
+  nv <- c("a", "b")
+  nvs <- substitute(nv)
+  t15a <- tf2(a = nv)
+  t15b <- tf2(a = nvs)
+  testthat::expect_equal(t15a, t15b)
+})
+
+
+testthat::test_that("cutLowMerge merges succesfully what is intended", {
+  popEpi:::skip_normally()
+  all_names_present(popEpi::popmort, c("sex", "year", "agegroup", "haz"))
+  all_names_present(popEpi::sire, c("sex", "bi_date", "dg_date", "ex_date", "status"))
+  
+  pm <- copy(popEpi::popmort)
+  pm[, haz := rbinom(.N, 100, 0.5)/1e5L]
+  
+  sr <- popEpi::sire[1:100,]
+  setDT(sr)
+  sr1 <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
+                 status = status, fot = seq(0, 5, 1/12))
+  sr1 <- data.table(sr1)
+  setattr(sr1, "class", c("Lexis", "data.table", "data.frame"))
+  
+  sr1[, year := per + 0.5*lex.dur]
+  sr1[, agegroup := age + 0.5*lex.dur]
+  
+  sr2 <- cutLowMerge(sr1, pm,
+                     by.x = c("sex", "per", "age"), 
+                     by.y = c("sex", "year", "agegroup"),
+                     all.x = TRUE, all.y = FALSE, old.nums = TRUE)
+  
+  sr3 <- copy(sr2)
+  sr3[, haz := NULL]
+  
+  sr4 <- lexpand(sr, birth = bi_date, entry = dg_date, exit = ex_date,
+                 status = status, fot = seq(0, 5, 1/12), pophaz = pm, pp = FALSE)
+  testthat::expect_equal(sr1, sr3, check.attributes = FALSE)
+  testthat::expect_equal(sr2$haz*1e5L, sr4$pop.haz*1e5L, check.attributes = FALSE)
+  
+  sr1[, year := popEpi:::cutLow(year, breaks = sort(unique(pm$year)))]
+  sr1[, agegroup := popEpi:::cutLow(agegroup, breaks = sort(unique(pm$agegroup)))]
+  
+  sr5 <- merge(sr1, pm, by = c("sex", "year", "agegroup"))
+  setDT(sr5)
+  setkey(sr5, lex.id, fot)
+  
+  testthat::expect_equal(sr4$haz*1e5L, sr5$pop.haz*1e5L, check.attributes = FALSE)
+})
+
+testthat::test_that("detectEvents works as intended", {
+  popEpi:::skip_normally()
+  x <- sire[dg_date<ex_date,]
+  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, pophaz = data.table(popEpi::popmort),
+               breaks = list(fot = seq(0,5,1/12), per = c(2007,2012), age = c(50,90)),
+               drop = TRUE)
+  ## this will only work with drop = TRUE.
+  setkeyv(x, c("lex.id", "fot", "per", "age"))
+  
+  ## this leaves observations cut short due to age or period censoring to
+  ## really be censoring.
+  x[, event := detectEvents(x, breaks = attr(x, "breaks")["fot"], by = "lex.id")]
+  
+  x[, alt.event := 0L]
+  x[!duplicated(lex.id, fromLast = TRUE), alt.event := 2L]
+  x[lex.Cst != lex.Xst, alt.event := 1L]
+  x[fot+lex.dur == 5L, alt.event := 0L]
+  
+  testthat::expect_equal(x$event, x$alt.event)
+  
+})
+
+testthat::test_that("comp_pp_weighted_figures produces intended results", {
+  set.seed(1L)
+  x <- sire[dg_date < ex_date,][sample(x = .N, size = 5L, replace = FALSE),]
+  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status %in% 1:2, pophaz = data.table(popEpi::popmort),
+               breaks = list(fot = seq(0,20,1/12), per = 1993:2013, age = 0:200))
+  x[, event := detectEvents(x, breaks = attr(x, "breaks"), by = "lex.id")]
+  
+  l <- comp_pp_weighted_figures(lex = x, haz = "pop.haz", pp = "pp", event.ind = "event", by = "lex.id")
+  x[, names(l) := l]
+  
+  testthat::expect_equal(x[event %in% 1:2, pp], x[event %in% 1:2, from0to0.pp + from0to1.pp])
+  testthat::expect_equal(x[event %in% 1:2, sum(pp)], x[, sum(from0to0.pp + from0to1.pp)])
+  testthat::expect_equal(x[, lex.dur*pp], x$ptime.pp)
+  testthat::expect_equal(x[, lex.dur*pp*pop.haz], x$d.exp.pp)
+  testthat::expect_equal(x[event == 1L, pp^2], x[event == 1L,]$from0to1.pp.2)
+  
+})
+
+
+testthat::test_that("evalPopFormula & usePopFormula output is stable", {
+  evalPopFormula <- popEpi:::evalPopFormula
+  usePopFormula <- popEpi:::usePopFormula
+  
+  x <- sire[1:5, ]
+  x[, "status" := c(1,1,0,1,1)]
+  x[, "sex" := c(1,0,1,0,1)]
+  x <- lexpand(x, birth = bi_date, entry = dg_date, exit = ex_date,
+               status = status)
+  
+  f1a <- Surv(fot, lex.Xst) ~ 1
+  f1b <- Surv(fot, event = lex.Xst) ~ 1
+  f2 <- Surv(fot, lex.Xst) ~ sex
+  f3 <- Surv(fot, lex.Xst) ~ sex + adjust(factor(sex + 1))
+  f4 <- Surv(fot, lex.Xst) ~ adjust(factor(sex + 1))
+  f5 <- lex.Xst ~ 1
+  f6 <- lex.Xst ~ sex
+  f7 <- lex.Xst ~ sex + adjust(factor(sex + 1))
+  f8 <- lex.Xst ~ adjust(factor(sex + 1))
+  f9 <- lex.Xst ~ factor(sex)
+  
+  TF <- environment()
+  
+  res <- data.table(time = rep(0, 5), status = c(1,1,0,1,1), sex = c(1,0,1,0,1))
+  res[["factor(sex + 1)"]] <- factor(res$sex + 1)
+  res$lex.Xst <- as.integer(c(1,1,0,1,1))
+  
+  
+  ## evalPopFormula
+  r1a <- evalPopFormula(f1a, data = x, enclos = TF, Surv.response = TRUE)
+  r1b <- evalPopFormula(f1b, data = x, enclos = TF, Surv.response = TRUE)
+  setattr(r1a, "formula", attr(r1b, "formula"))
+  testthat::expect_equal(r1a, r1b)
+  testthat::expect_equal(data.table(r1a), res[, list(time, status)])
+  
+  r5 <- evalPopFormula(f5, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
+  r6 <- evalPopFormula(f6, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
+  testthat::expect_equal(r5$lex.Xst, c(1,1,0,1,1))
+  testthat::expect_equal(data.table(r6), res[, list(lex.Xst, sex)])
+  
+  ## model-type naming of columns
+  r9 <- evalPopFormula(f9, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
+  testthat::expect_equal(data.table(r9), res[, list(lex.Xst, "factor(sex)" = factor(sex))])
+  
+  ## multiple variables, model-type naming of columns with adjust
+  r3 <- evalPopFormula(f3, data = x, enclos = parent.frame(1L), Surv.response = TRUE)
+  testthat::expect_equivalent(r3, res[, .SD, .SDcols = names(r3)])
+  
+  
+  #### only the survival time scale as response
+  fa <- fot ~ 1
+  fb <- fot ~ sex
+  fc <- fot ~ sex + adjust(factor(sex + 1))
+  fd <- fot ~ adjust(factor(sex + 1))
+  
+  ra <- evalPopFormula(fa, data = x, enclos = TF, Surv.response = "either")
+  testthat::expect_equivalent(ra, x[, .(fot, lex.Xst)])
+  rb <- evalPopFormula(fb, data = x, enclos = parent.frame(1L), Surv.response = FALSE)
+  testthat::expect_equivalent(rb, x[, .(fot, sex)])
+  rc <- evalPopFormula(fc, data = x, enclos = parent.frame(1L), Surv.response = "either")
+  testthat::expect_equivalent(rc, x[, .(fot, lex.Xst, sex, factor(sex + 1))])
+  rd <- evalPopFormula(fd, data = x, enclos = TF, Surv.response = FALSE)
+  testthat::expect_equivalent(rd, x[, .(fot, factor(sex + 1))])
+  
+  ## usePopFormula
+  
+  r3 <- usePopFormula(f3, data = x, enclos = parent.frame(2L), Surv.response = TRUE)
+  testthat::expect_equivalent(r3, 
+                    list(y = res[, list(time, status)], 
+                         print = res[, list(sex)], 
+                         adjust = res[, "factor(sex + 1)", with = FALSE],
+                         formula = f3) 
+  )
+  r4 <- usePopFormula(f4, data = x, enclos = parent.frame(2L), Surv.response = TRUE)
+  testthat::expect_equivalent(r4, 
+                    list(y = res[, list(time, status)], 
+                         print = NULL, 
+                         adjust = res[, "factor(sex + 1)", with = FALSE],
+                         formula = f4) 
+  )
+  r6 <- usePopFormula(f6, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
+  testthat::expect_equivalent(r6, 
+                    list(y = res[, list(lex.Xst)], 
+                         print = res[, list(sex)], 
+                         adjust = NULL,
+                         formula = f6) 
+  )
+  
+  r7 <- usePopFormula(f7, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
+  testthat::expect_equivalent(r7, 
+                    list(y = res[, list(lex.Xst)], 
+                         print = res[, list(sex)], 
+                         adjust = res[, "factor(sex + 1)", with = FALSE],
+                         formula = f7) 
+  )
+  r8 <- usePopFormula(f8, data = x, enclos = parent.frame(2L), Surv.response = FALSE)
+  testthat::expect_equivalent(r8, 
+                    list(y = res[, list(lex.Xst)], 
+                         print = NULL, 
+                         adjust = res[, "factor(sex + 1)", with = FALSE],
+                         formula = f8) 
+  )
+  r9 <- usePopFormula(lex.Xst ~ sex, data = x, adjust = quote(factor(sex + 1)),
+                      enclos = parent.frame(2L), Surv.response = FALSE) 
+  testthat::expect_equivalent(r9, 
+                    list(y = res[, list(lex.Xst)], 
+                         print = res[, list(sex)], 
+                         adjust = res[, "factor(sex + 1)", with = FALSE],
+                         formula = lex.Xst ~ sex)
+  )
+  testthat::expect_equal(lapply(r9, names), list(y = "lex.Xst", print = "sex", adjust = "factor(sex + 1)", formula = NULL))
+  
+  r9 <- usePopFormula(lex.Xst ~ as.numeric(sex), data = x, adjust = quote(list(factor(sex + 1), factor(sex - 1))),
+                      enclos = parent.frame(2L), Surv.response = FALSE) 
+  
+  testthat::expect_equal(lapply(r9, names), list(y = "lex.Xst", print = "as.numeric(sex)", 
+                                       adjust = c("factor(sex + 1)", "factor(sex - 1)"), formula = NULL))
+  
+  
+  ra <- usePopFormula(fa, data = x, 
+                      adjust = quote(list(factor(sex + 1), factor(sex - 1))),
+                      enclos = parent.frame(2L), Surv.response = FALSE) 
+  
+  testthat::expect_equal(lapply(ra, names), 
+               list(y = "fot", print = NULL, 
+                    adjust = c("factor(sex + 1)", "factor(sex - 1)"), 
+                    formula = NULL))
+  
+  
+  rb <- usePopFormula(fb, data = x, 
+                      adjust = quote(list(factor(sex + 1), factor(sex - 1))),
+                      enclos = parent.frame(2L), Surv.response = FALSE) 
+  testthat::expect_equal(lapply(rb, names), 
+               list(y = "fot", print = "sex", 
+                    adjust = c("factor(sex + 1)", "factor(sex - 1)"), 
+                    formula = NULL))
+  
+  rc <- usePopFormula(fc, data = x, 
+                      adjust = NULL,
+                      enclos = parent.frame(2L), Surv.response = "either") 
+  testthat::expect_equal(lapply(rc, names), 
+               list(y = c("time", "status"), print = "sex", 
+                    adjust = "factor(sex + 1)", 
+                    formula = NULL))
+  
+  rd <- usePopFormula(fd, data = x, 
+                      adjust = NULL,
+                      enclos = parent.frame(2L), Surv.response = FALSE) 
+  testthat::expect_equal(lapply(rd, names), 
+               list(y = "fot", print = NULL, 
+                    adjust = "factor(sex + 1)", 
+                    formula = NULL))
+  
+  ## usePopFormula with "either" response
+  useForms <- paste0("f", 2:9)
+  useForms <- c("f1a", "f1a", useForms)
+  useForms <- intersect(useForms, ls())
+  TF <- environment()
+  l <- list()
+  for (k in seq_along(useForms)) {
+    l[[k]] <- usePopFormula(get(useForms[k], envir = TF), data = x, 
+                            adjust = NULL,
+                            enclos = TF, Surv.response = "either") 
+  }
+  
+  
+  
+  
+})
+
+
+
+
+
+testthat::test_that("fractional years computation works", {
+  library("data.table")
+  c <- paste0("2004-", c("01-01", "07-01", "12-31", "02-01"))
+  D <- as.Date(c)
+  
+  testthat::expect_equal(get.yrs(c), get.yrs(D))
+  
+  yl <- 365.242199
+  my <- year(D) + (yday(D) - 1L)/yl
+  
+  testthat::expect_equal(as.numeric(get.yrs(D)), my)
+  
+  yl <- ifelse(is_leap_year(year(D)), 366L, 365L)
+  my <- year(D) + (yday(D) - 1L)/yl
+  
+  testthat::expect_equal(as.numeric(get.yrs(D, year.length = "actual")), my)
+})
+
+
+
+
+
+testthat::test_that("subsetDTorDF works as intended", {
+  
+  dt <- data.table::data.table(a = 1:5, b = 5:1)
+  df <- data.frame(a = 1:5, b = 5:1)
+  
+  sub <- c(TRUE, TRUE, FALSE, FALSE, TRUE)
+  col_nms <- rep("a", 50)
+  settings <- list(
+    list(subset = sub, select = NULL),
+    list(subset = NULL, select = col_nms),
+    list(subset = NULL, select = NULL),
+    list(subset = sub, select = col_nms)
+  )
+  
+  expected_df <- list(
+    df[sub, ],
+    df[, col_nms],
+    df,
+    df[sub, col_nms]
+  )
+  expected_dt <- list(
+    dt[sub, ],
+    dt[, .SD, .SDcols = col_nms],
+    dt,
+    dt[sub, .SD, .SDcols = col_nms]
+  )
+  
+  unused <- lapply(seq_along(settings), function(i) {
+    set <- settings[[i]]
+    set[["data"]] <- dt
+    sub_dt <- do.call(subsetDTorDF, set)
+    set[["data"]] <- df
+    sub_df <- do.call(subsetDTorDF, set)
+    
+    exp_df <- expected_df[[i]]
+    exp_dt <- expected_dt[[i]]
+    
+    list(sub_dt = sub_dt, exp_dt = exp_dt, sub_df = sub_df, exp_df = exp_df)
+    testthat::expect_equal(
+      data.frame(sub_dt),
+      data.frame(exp_dt)
+    )
+    testthat::expect_equal(
+      sub_df,
+      exp_df
+    )
+  })
+  
+  
+})
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tests/testthat/test_weighter.R b/tests/testthat/test_weighter.R
index 721ff49..5c851d4 100644
--- a/tests/testthat/test_weighter.R
+++ b/tests/testthat/test_weighter.R
@@ -1,140 +1,140 @@
-testthat::context("makeWeightsDT")
-
-
-testthat::test_that("makeWeightsDT works with various weights & adjust spesifications", {
-  popEpi:::skip_normally()
-  sibr <- popEpi::sibr[1:100]
-  sibr[1:50, sex := 0L]
-  
-  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, 
-                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
-  
-  print <- quote(list(gender = sex))
-  prdt <- evalPopArg(dt, print)
-  
-  adjust <- quote(list(agegr = popEpi:::cutLow(dg_age, c(0, seq(15,85,5), Inf))))
-  addt <- evalPopArg(dt, adjust)
-  
-  expr <- quote(list(pyrs = sum(pyrs), from0to1 = sum(from0to1)))
-  dt <- dt[, eval(expr), keyby = cbind(prdt, addt)]
-  
-  wdt <- ICSS[, list(agegr = c(0, seq(15,85,5)), weights = ICSS3)]
-  
-  
-  expr <- list(quote(list(pyrs = pyrs, from0to1 = from0to1)))
-  
-  ## works with weights data.frame?
-  ## Intention of test: wdt contains more levels of agegr than data;
-  ## weights need to be scaled to one correctly and only the original
-  ## rows in data kept.
-  DT <- makeWeightsDT(dt, values = expr, weights = wdt, 
-                      print = "gender", adjust = "agegr")
-  testthat::expect_equal(nrow(DT), 24L)
-  testthat::expect_equal(sum(dt$pyrs), sum(DT$pyrs))
-  DTW <- unique(DT, by = c("agegr"))$weights
-  wdt[, ww := weights/sum(weights)]
-  testthat::expect_equal(DTW, wdt[agegr %in% DT$agegr, ww])
-  testthat::expect_equal(DT[, sum(pyrs), keyby = agegr], dt[, sum(pyrs), keyby = agegr], check.attributes = FALSE)
-  testthat::expect_equal(DT[, sum(pyrs), keyby = gender], dt[, sum(pyrs), keyby = gender], check.attributes = FALSE)
-  
-  
-})
-
-
-
-testthat::test_that("internal weights are computed correctly", {
-  popEpi:::skip_normally()
-  sibr <- popEpi::sibr[1:100]
-  sibr[1:50, sex := 0L]
-  
-  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, 
-                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
-  
-  
-  adjust <- quote(popEpi:::cutLow(dg_age, c(0, seq(15,85,5), Inf)))
-  dt[, c("agegr") := eval(adjust)]
-  
-  vals <- list(c("pyrs", "from0to1"))
-  
-  testthat::expect_error({
-    DT <- makeWeightsDT(dt, values = vals, 
-                        weights = "internal", 
-                        print = NULL, adjust = c("sex", "agegr"))
-  })
-  
-  ## one adjust var
-  DT1 <- makeWeightsDT(dt, values = vals, 
-                       weights = "internal", 
-                       print = NULL, adjust = "agegr",
-                       internal.weights.values = "pyrs")
-  DT2 <- dt[, sum(pyrs)/sum(dt$pyrs), keyby = agegr]  
-  testthat::expect_equal(DT1$weights, DT2$V1)
-  
-  ## two adjust vars
-  DT1 <- makeWeightsDT(dt, values = vals, 
-                       weights = "internal", 
-                       print = NULL, adjust = c("sex", "agegr"),
-                       internal.weights.values = "pyrs")
-  DT2 <- dt[, sum(pyrs), keyby = agegr]  
-  DT3 <- dt[, sum(pyrs), keyby = sex] 
-  DT4 <- CJ(sex = 0:1, agegr = sort(unique(dt$agegr)))
-  DT4 <- merge(DT4, DT3, by = "sex", all.x = TRUE, all.y = TRUE)
-  setnames(DT4, "V1", "w.sex")
-  DT4 <- merge(DT4, DT2, by = "agegr", all.x = TRUE, all.y = TRUE)
-  setnames(DT4, "V1", "w.agegr")
-  DT4[, weight := w.agegr*w.sex]
-  DT4[, weight := weight/sum(weight)]
-  setkeyv(DT4, c("sex", "agegr"))
-  setkeyv(DT1, c("sex", "agegr"))
-  
-  testthat::expect_equal(DT1$weights, DT4$weight)
-  
-})
-
-
-testthat::test_that("weighter works with a list of values arguments", {
-  
-  popEpi:::skip_normally()
-  sibr <- popEpi::sibr[1:100]
-  sibr[1:50, sex := 0L]
-  
-  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, 
-                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
-  
-  vals <- list(c("pyrs", "from0to1"), quote(from0to0))
-  
-  DT1 <- makeWeightsDT(dt, values = vals, 
-                       print = "sex")
-  testthat::expect_equivalent(DT1, dt[, lapply(.SD, sum), by = sex, .SDcols = c("pyrs", "from0to1", "from0to0")])
-  
-  dt$agegr <- cut(dt$dg_age, c(0, 45, 75, Inf))
-  DT2 <- makeWeightsDT(dt, values = vals, 
-                       print = c("sex", "agegr"))
-  testthat::expect_equivalent(DT2, dt[, lapply(.SD, sum), by = list(sex,agegr), .SDcols = c("pyrs", "from0to1", "from0to0")])
-  
-})
-
-
-testthat::test_that("weighted NA checks work", {
-  popEpi:::skip_normally()
-  sibr <- popEpi::sibr[1:100]
-  sibr[1:50, sex := 0L]
-  
-  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
-                status = status %in% 1:2, 
-                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
-  dt[, agegr := cut(dg_age, c(0, 45, 75), right = FALSE)]
-  vals <- list(c("pyrs", "from0to1"), quote(from0to0))
-  
-  naTxt <- "A warning message with counts: %%NA_COUNT%%"
-  testthat::expect_warning({
-    DT1 <- makeWeightsDT(dt, values = list(c("pyrs", "from0to1"), quote(from0to0)), 
-                         adjust = "agegr", weights = "internal", 
-                         internal.weights.values = "pyrs", NA.text = naTxt)
-  }, regexp = "A warning message with counts: 15")
-
-})
-
+testthat::context("makeWeightsDT")
+
+
+testthat::test_that("makeWeightsDT works with various weights & adjust spesifications", {
+  popEpi:::skip_normally()
+  sibr <- popEpi::sibr[1:100]
+  sibr[1:50, sex := 0L]
+  
+  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, 
+                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
+  
+  print <- quote(list(gender = sex))
+  prdt <- evalPopArg(dt, print)
+  
+  adjust <- quote(list(agegr = popEpi:::cutLow(dg_age, c(0, seq(15,85,5), Inf))))
+  addt <- evalPopArg(dt, adjust)
+  
+  expr <- quote(list(pyrs = sum(pyrs), from0to1 = sum(from0to1)))
+  dt <- dt[, eval(expr), keyby = cbind(prdt, addt)]
+  
+  wdt <- ICSS[, list(agegr = c(0, seq(15,85,5)), weights = ICSS3)]
+  
+  
+  expr <- list(quote(list(pyrs = pyrs, from0to1 = from0to1)))
+  
+  ## works with weights data.frame?
+  ## Intention of test: wdt contains more levels of agegr than data;
+  ## weights need to be scaled to one correctly and only the original
+  ## rows in data kept.
+  DT <- makeWeightsDT(dt, values = expr, weights = wdt, 
+                      print = "gender", adjust = "agegr")
+  testthat::expect_equal(nrow(DT), 24L)
+  testthat::expect_equal(sum(dt$pyrs), sum(DT$pyrs))
+  DTW <- unique(DT, by = c("agegr"))$weights
+  wdt[, ww := weights/sum(weights)]
+  testthat::expect_equal(DTW, wdt[agegr %in% DT$agegr, ww])
+  testthat::expect_equal(DT[, sum(pyrs), keyby = agegr], dt[, sum(pyrs), keyby = agegr], check.attributes = FALSE)
+  testthat::expect_equal(DT[, sum(pyrs), keyby = gender], dt[, sum(pyrs), keyby = gender], check.attributes = FALSE)
+  
+  
+})
+
+
+
+testthat::test_that("internal weights are computed correctly", {
+  popEpi:::skip_normally()
+  sibr <- popEpi::sibr[1:100]
+  sibr[1:50, sex := 0L]
+  
+  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, 
+                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
+  
+  
+  adjust <- quote(popEpi:::cutLow(dg_age, c(0, seq(15,85,5), Inf)))
+  dt[, c("agegr") := eval(adjust)]
+  
+  vals <- list(c("pyrs", "from0to1"))
+  
+  testthat::expect_error({
+    DT <- makeWeightsDT(dt, values = vals, 
+                        weights = "internal", 
+                        print = NULL, adjust = c("sex", "agegr"))
+  })
+  
+  ## one adjust var
+  DT1 <- makeWeightsDT(dt, values = vals, 
+                       weights = "internal", 
+                       print = NULL, adjust = "agegr",
+                       internal.weights.values = "pyrs")
+  DT2 <- dt[, sum(pyrs)/sum(dt$pyrs), keyby = agegr]  
+  testthat::expect_equal(DT1$weights, DT2$V1)
+  
+  ## two adjust vars
+  DT1 <- makeWeightsDT(dt, values = vals, 
+                       weights = "internal", 
+                       print = NULL, adjust = c("sex", "agegr"),
+                       internal.weights.values = "pyrs")
+  DT2 <- dt[, sum(pyrs), keyby = agegr]  
+  DT3 <- dt[, sum(pyrs), keyby = sex] 
+  DT4 <- CJ(sex = 0:1, agegr = sort(unique(dt$agegr)))
+  DT4 <- merge(DT4, DT3, by = "sex", all.x = TRUE, all.y = TRUE)
+  setnames(DT4, "V1", "w.sex")
+  DT4 <- merge(DT4, DT2, by = "agegr", all.x = TRUE, all.y = TRUE)
+  setnames(DT4, "V1", "w.agegr")
+  DT4[, weight := w.agegr*w.sex]
+  DT4[, weight := weight/sum(weight)]
+  setkeyv(DT4, c("sex", "agegr"))
+  setkeyv(DT1, c("sex", "agegr"))
+  
+  testthat::expect_equal(DT1$weights, DT4$weight)
+  
+})
+
+
+testthat::test_that("weighter works with a list of values arguments", {
+  
+  popEpi:::skip_normally()
+  sibr <- popEpi::sibr[1:100]
+  sibr[1:50, sex := 0L]
+  
+  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, 
+                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
+  
+  vals <- list(c("pyrs", "from0to1"), quote(from0to0))
+  
+  DT1 <- makeWeightsDT(dt, values = vals, 
+                       print = "sex")
+  testthat::expect_equivalent(DT1, dt[, lapply(.SD, sum), by = sex, .SDcols = c("pyrs", "from0to1", "from0to0")])
+  
+  dt$agegr <- cut(dt$dg_age, c(0, 45, 75, Inf))
+  DT2 <- makeWeightsDT(dt, values = vals, 
+                       print = c("sex", "agegr"))
+  testthat::expect_equivalent(DT2, dt[, lapply(.SD, sum), by = list(sex,agegr), .SDcols = c("pyrs", "from0to1", "from0to0")])
+  
+})
+
+
+testthat::test_that("weighted NA checks work", {
+  popEpi:::skip_normally()
+  sibr <- popEpi::sibr[1:100]
+  sibr[1:50, sex := 0L]
+  
+  dt <- lexpand(sibr, birth = bi_date, entry = dg_date, exit = ex_date,
+                status = status %in% 1:2, 
+                aggre = list(sex, dg_age = popEpi:::cutLow(dg_age, 0:125)))
+  dt[, agegr := cut(dg_age, c(0, 45, 75), right = FALSE)]
+  vals <- list(c("pyrs", "from0to1"), quote(from0to0))
+  
+  naTxt <- "A warning message with counts: %%NA_COUNT%%"
+  testthat::expect_warning({
+    DT1 <- makeWeightsDT(dt, values = list(c("pyrs", "from0to1"), quote(from0to0)), 
+                         adjust = "agegr", weights = "internal", 
+                         internal.weights.values = "pyrs", NA.text = naTxt)
+  }, regexp = "A warning message with counts: 15")
+
+})
+
diff --git a/vignettes/sir.Rmd b/vignettes/sir.Rmd
index f125f05..97887c4 100644
--- a/vignettes/sir.Rmd
+++ b/vignettes/sir.Rmd
@@ -1,158 +1,158 @@
----
-title: "SMR Vignette"
-author: "Matti Rantanen"
-date: "`r Sys.Date()`"
-output: 
-  html_document:
-    fig_caption: yes
-    toc: true
-    toc_depth: 2
-vignette: >
-  %\VignetteEngine{knitr::rmarkdown}
-  %\VignetteIndexEntry{Standardised incidence and mortality ratios}
-  %\usepackage[utf8]{inputenc}
----
-
-```{r, echo=TRUE, warning=FALSE, message=FALSE}
-library(popEpi)
-library(Epi)
-```
-
-# Introduction
-
-Standardized incidence ratio (SIR) or mortality ratio (SMR) is a ratio of observed and expected cases. Observed cases is the absolute number of cases in the cohort. The expected cases are derived by multiplying the cohort person-years with reference populations rate. The rate should be stratified or adjusted by confounding factors. Usually these are age group, gender, calendar period and possibly a cancer type or other confounding variable. Also a social economic status or area variable can be used.
-
-In reference population the expected rate in strata $j$ is $\lambda_j = d_j$ / $n_j$, where $d_j$ is observed cases and $n_j$ is observed person years. Now the SIR can be written as a ratio
-$$
-SIR = \frac{ \sum d_j }{\sum n_j \lambda_j} = \frac{D}{E} 
-$$
-where $D$ is the observed cases in cohort population and $E$ is the expected number. Univariate confidence intervals are based on exact values of Poisson distribution and the formula for p-value is
-$$
-\chi^2 = \frac{ (|O - E| -0.5)^2 }{E}.
-$$
-Modelled SIR is a Poisson regression model with log-link and cohorts person-years as a offset.
-
-The homogeneity of SIR's can be tested using a likelihood ratio test in Poisson modelled SIRs.
-
-The same workflow applies for standardised mortality ratios.
-
-# Splines
-
-A continuous spline function can be fitted for time variables, e.g. age-group. Idea of the splines is to smooth the SMR estimates and do inference from the curve figure. This requires pre-defined knots/nodes that are used to fit the spline curve. Selecting the number of knots and knot places is a very subjective matter and there are three options to pass spline knots to function.
-
-It's good practice to try between different knot settings for realistic spline estimates. Overfitting might cause unintentional artefacts in the estimate and underfitting might smooth away interesting patterns.
-
-The spline variable should be as continuous as possible, say from 18 to 100 time points. But when splitting time in too narrow intervals, random variation might occur in the expected or population rate values. Therefore it's also possible to do two variables for age or period: first with wider intervals for standardisation and second with narrow intervals for the spline.
-
-## Knots
-
-There are three options to for assigning knots to the spline:
-
-1. A vector of numbers of knots for each spline variable. Number of knots includes the boundary knots, so that the minimum number of knots is 2, which is a log linear association. The knots are placed automatically using the quantiles of observed cases.
-
-2. A list of vectors of predefined knot places. Number of vectors needs to match the length of spline variables. And each vector has to have at least the minimum and maximum for boundary knots.
-
-3. NULL will automatically finds the optimal number of knots based on AIC. Knots are placed according the quantiles of observed cases. This is usually a good place to start the fitting process.
-
-Number of knots and knot places are always found in output. 
-
-# SMR
-
-## Mortality: External cohort and popmort data
-
-Estimate SMR of a simulated cohort of Finnish female rectal cancer patients, `sire`.
-Death rates for each age, period and sex is available in `popmort` dataset.
-
-For more information about the dataset see `help(popmort)` and `help(sire)`.
-
-```{r}
-data(sire)
-data(popmort)
-c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-```
-
-SMR's for other causes is 1 for both follow-up intervals. Also the p-value suggest that there is no heterogeneity between SMR estimates (p=0.735).
-
-
-The total mortality can be estimated by modifying the `status` argument. Now we want to account all deaths, i.e. status is 1 or 2.
-
-```{r}
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
-           ref.data = popmort, ref.rate = 'haz', 
-           adjust = c('agegroup','year','sex'), print ='fot')
-se
-```
-
-Now the estimates for follow-up intervals seems to differ significantly, p = 0. Plotting SMR (S3-method for `sir`-object) is easily done using default plot-function.
-
-```{r, fig.height=3, fig.width=6}
-plot(se, col = 2:3)
-title('SMR for follow-up categories')
-```
-
-
-## splines
-
-
-Lets fit splines for the follow-up time and age group using two different options: the splines are fitted in different model and in same model, `dependent.splines`.
-
-```{r, fig.height=5, fig.width=6}
-c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
-              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
-              aggre = list(fot, agegroup = age, year = per, sex) )
-
-sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines=FALSE)
-
-st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup','fot'), dependent.splines = TRUE)
-
-plot(sf, col=2, log=TRUE)
-title('Splines fitted in different models')
-
-plot(st, col=4, log=TRUE)
-title('Splines are dependent')
-```
-
-In dependent spline the `fot` is the ratio with zero time as reference point. Reference points can be altered. Here age group profile is assumed to be same for every follow-up time. SMR is 0.2 times from 0 to 10 years of follow-up. 
-
-
-Splines can also be stratified using the `print` argument. For example we split the death time in two time periods and test if the age group splines are equal.
-
-```{r, results='hide', fig.height=5, fig.width=6}
-c$year.cat <- ifelse(c$year < 2002, 1, 2)
-sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
-                 ref.data = popmort, ref.rate = 'haz', 
-                 adjust = c('agegroup','year','sex'),
-                 spline = c('agegroup'), print = 'year.cat')
-plot(sy, log=TRUE)
-legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
-```
-
-For category before 2002 the SMR seems to be higher after the age of 50. Also the p-value (<0.0001) indicates that there is a difference in age group trends before and after year 2002. P-value is a likelihood ratio test that compares models where splines are fitted together and separately.
-
-```{r}
-print(sy)
-```
-
-
-
-
-
-
-
+---
+title: "SMR Vignette"
+author: "Matti Rantanen"
+date: "`r Sys.Date()`"
+output: 
+  html_document:
+    fig_caption: yes
+    toc: true
+    toc_depth: 2
+vignette: >
+  %\VignetteEngine{knitr::rmarkdown}
+  %\VignetteIndexEntry{Standardised incidence and mortality ratios}
+  %\usepackage[utf8]{inputenc}
+---
+
+```{r, echo=TRUE, warning=FALSE, message=FALSE}
+library(popEpi)
+library(Epi)
+```
+
+# Introduction
+
+Standardized incidence ratio (SIR) or mortality ratio (SMR) is a ratio of observed and expected cases. Observed cases is the absolute number of cases in the cohort. The expected cases are derived by multiplying the cohort person-years with reference populations rate. The rate should be stratified or adjusted by confounding factors. Usually these are age group, gender, calendar period and possibly a cancer type or other confounding variable. Also a social economic status or area variable can be used.
+
+In reference population the expected rate in strata $j$ is $\lambda_j = d_j$ / $n_j$, where $d_j$ is observed cases and $n_j$ is observed person years. Now the SIR can be written as a ratio
+$$
+SIR = \frac{ \sum d_j }{\sum n_j \lambda_j} = \frac{D}{E} 
+$$
+where $D$ is the observed cases in cohort population and $E$ is the expected number. Univariate confidence intervals are based on exact values of Poisson distribution and the formula for p-value is
+$$
+\chi^2 = \frac{ (|O - E| -0.5)^2 }{E}.
+$$
+Modelled SIR is a Poisson regression model with log-link and cohorts person-years as a offset.
+
+The homogeneity of SIR's can be tested using a likelihood ratio test in Poisson modelled SIRs.
+
+The same workflow applies for standardised mortality ratios.
+
+# Splines
+
+A continuous spline function can be fitted for time variables, e.g. age-group. Idea of the splines is to smooth the SMR estimates and do inference from the curve figure. This requires pre-defined knots/nodes that are used to fit the spline curve. Selecting the number of knots and knot places is a very subjective matter and there are three options to pass spline knots to function.
+
+It's good practice to try between different knot settings for realistic spline estimates. Overfitting might cause unintentional artefacts in the estimate and underfitting might smooth away interesting patterns.
+
+The spline variable should be as continuous as possible, say from 18 to 100 time points. But when splitting time in too narrow intervals, random variation might occur in the expected or population rate values. Therefore it's also possible to do two variables for age or period: first with wider intervals for standardisation and second with narrow intervals for the spline.
+
+## Knots
+
+There are three options to for assigning knots to the spline:
+
+1. A vector of numbers of knots for each spline variable. Number of knots includes the boundary knots, so that the minimum number of knots is 2, which is a log linear association. The knots are placed automatically using the quantiles of observed cases.
+
+2. A list of vectors of predefined knot places. Number of vectors needs to match the length of spline variables. And each vector has to have at least the minimum and maximum for boundary knots.
+
+3. NULL will automatically finds the optimal number of knots based on AIC. Knots are placed according the quantiles of observed cases. This is usually a good place to start the fitting process.
+
+Number of knots and knot places are always found in output. 
+
+# SMR
+
+## Mortality: External cohort and popmort data
+
+Estimate SMR of a simulated cohort of Finnish female rectal cancer patients, `sire`.
+Death rates for each age, period and sex is available in `popmort` dataset.
+
+For more information about the dataset see `help(popmort)` and `help(sire)`.
+
+```{r}
+data(sire)
+data(popmort)
+c <- lexpand( sire, status = status, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to2', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+```
+
+SMR's for other causes is 1 for both follow-up intervals. Also the p-value suggest that there is no heterogeneity between SMR estimates (p=0.735).
+
+
+The total mortality can be estimated by modifying the `status` argument. Now we want to account all deaths, i.e. status is 1 or 2.
+
+```{r}
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = c(0,10,20,Inf)), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+se <- sir( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs',
+           ref.data = popmort, ref.rate = 'haz', 
+           adjust = c('agegroup','year','sex'), print ='fot')
+se
+```
+
+Now the estimates for follow-up intervals seems to differ significantly, p = 0. Plotting SMR (S3-method for `sir`-object) is easily done using default plot-function.
+
+```{r, fig.height=3, fig.width=6}
+plot(se, col = 2:3)
+title('SMR for follow-up categories')
+```
+
+
+## splines
+
+
+Lets fit splines for the follow-up time and age group using two different options: the splines are fitted in different model and in same model, `dependent.splines`.
+
+```{r, fig.height=5, fig.width=6}
+c <- lexpand( sire, status = status %in% 1:2, birth = bi_date, exit = ex_date, entry = dg_date,
+              breaks = list(per = 1950:2013, age = 1:100, fot = 0:50), 
+              aggre = list(fot, agegroup = age, year = per, sex) )
+
+sf <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines=FALSE)
+
+st <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup','fot'), dependent.splines = TRUE)
+
+plot(sf, col=2, log=TRUE)
+title('Splines fitted in different models')
+
+plot(st, col=4, log=TRUE)
+title('Splines are dependent')
+```
+
+In dependent spline the `fot` is the ratio with zero time as reference point. Reference points can be altered. Here age group profile is assumed to be same for every follow-up time. SMR is 0.2 times from 0 to 10 years of follow-up. 
+
+
+Splines can also be stratified using the `print` argument. For example we split the death time in two time periods and test if the age group splines are equal.
+
+```{r, results='hide', fig.height=5, fig.width=6}
+c$year.cat <- ifelse(c$year < 2002, 1, 2)
+sy <- sirspline( coh.data = c, coh.obs = 'from0to1', coh.pyrs = 'pyrs', 
+                 ref.data = popmort, ref.rate = 'haz', 
+                 adjust = c('agegroup','year','sex'),
+                 spline = c('agegroup'), print = 'year.cat')
+plot(sy, log=TRUE)
+legend('topright', c('before 2002','after 2002'), lty=1, col=c(1,2))
+```
+
+For category before 2002 the SMR seems to be higher after the age of 50. Also the p-value (<0.0001) indicates that there is a difference in age group trends before and after year 2002. P-value is a likelihood ratio test that compares models where splines are fitted together and separately.
+
+```{r}
+print(sy)
+```
+
+
+
+
+
+
+
diff --git a/vignettes/survtab_examples.Rmd b/vignettes/survtab_examples.Rmd
index 341f23c..ece43b6 100644
--- a/vignettes/survtab_examples.Rmd
+++ b/vignettes/survtab_examples.Rmd
@@ -1,295 +1,295 @@
----
-title: "Examples of using survtab"
-author: "Joonas Miettinen"
-date: "`r Sys.Date()`"
-output: 
-  html_document:
-    toc: true
-    toc_depth: 2
-    fig_width: 6
-    fig_height: 6
-vignette: >
-  %\VignetteEngine{knitr::rmarkdown}
-  %\VignetteIndexEntry{survtab examples}
-  %\usepackage[utf8]{inputenc}
----
-
-# Overview
-  
-This vignette aims to clarify the usage of the `survtab_ag` and `survtab` functions included in this package. `survtab_ag` estimates various survival functions and cumulative incidence functions (CIFs) non-parametrically using aggregated data, and `survtab` is a wrapper for `survtab_ag`, to which `Lexis` data is supplied. 
-  
-Two methods (`surv.method`) are currently supported: The `"lifetable"` (actuarial) method only makes use of counts when estimating any of the supported survival time functions. The default method (`"hazard"`}) estimates appropriate hazards and transforms them into survival function or CIF estimates.
-  
-For relative survival estimation we need also to enumerate the expected hazard levels for the subjects in the data. This is done by merging expected hazards to individuals' subintervals (which divide their survival time lines to a number of small intervals). For Pohar-Perme-weighted analyses one must additionally compute various weighted figures at the level of split subject data. 
-  
-If one has subject-level data, the simplest way of computing survival function estimates with `popEpi` is by defining a `Lexis` object and using `survtab`, which will do the rest. For pre-aggregated data one may use the `survtab_ag` function instead. One can also use the `lexpand` function to split, merge population hazards, and aggregate in a single function call and then use `survtab_ag` if that is convenient.
-  
-# Using `survtab`
-  
-It is straightforward to estimate various survival time functions with `survtab` once a `Lexis` object has been defined (see `?Lexis` in package `Epi` for details):
-  
-```{r pkgs, eval = TRUE, echo = TRUE, message = FALSE}
-library(popEpi)
-library(Epi)
-```
-
-```{r}
-data(sire)
-
-## NOTE: recommended to use factor status variable
-x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
-           exit = list(CAL = get.yrs(ex_date)), 
-           data = sire[sire$dg_date < sire$ex_date, ],
-           exit.status = factor(status, levels = 0:2, 
-                                labels = c("alive", "canD", "othD")), 
-           merge = TRUE)
-
-## pretend some are male
-set.seed(1L)
-x$sex <- rbinom(nrow(x), 1, 0.5)
-
-## observed survival - explicit method
-st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## observed survival - easy method (assumes lex.Xst in x is the status variable)
-st <- survtab(FUT ~ sex, data = x, 
-              surv.type = "surv.obs",
-              breaks = list(FUT = seq(0, 5, 1/12)))
-
-## printing gives the used settings and 
-## estimates at the middle and end of the estimated
-## curves; more information available using summary()
-st
-
-```
-
-Plotting by strata (men = blue, women = red):
-
-```{r}
-plot(st, col = c("blue", "red"))
-```
-
-Note that the correct usage of the `formula` argument in `survtab` specifies the time scale in the `Lexis` object over which survival is computed (here `"FUT"` for follow-up time). This is used to identify the appropriate time scale in the data. When only supplying the survival time scale as the right-hand-side of the formula, the column `lex.Xst` in the supplied `Lexis` object is assumed to be the (correctly formatted!) status variable. When using `Surv()` to be explicit, we effectively (and exceptionally) pass the starting times to the `time` argument in `Surv()`, and `time2` is ignored entirely. The function will fail if `time` does not match exactly with a time scale in data.
-
-When using `Surv()`, one must also pass the status variable, which can be something other than the `lex.Xst` variable created by `Lexis()`, though usually ``lex.Xst` is what you want to use (especially if the data has already been split using e.g. `splitLexis` or `splitMulti`, which is allowed). It is recommended to use a factor status variable to pass to `Surv()`, though a numeric variable will work in simple cases (0 = alive, 1 = dead; also `FALSE`  = alive, `TRUE` = dead). Using `Surv()` also allows easy passing of transformations of `lex.Xst`, e.g. `Surv(FUT, lex.Xst %in% 1:2)`.
-
-The argument `breaks` must be a named list of breaks by which to split the `Lexis` data (see `?splitMulti`). It is mandatory to assign breaks at least to the survival time scale (`"FUT"` in our example) so that `survtab` knows what intervals to use to estimate the requested survival time function(s). The breaks also determine the window used: It is therefore easy to compute so called period estimates by defining the roof and floor along the calendar time scale, e.g. 
-
-`breaks = list(FUT = seq(0, 5, 1/12), CAL = c(2000, 2005))`
-
-would cause `survtab` to compute period estimates for 2000-2004 (breaks given here as fractional years, so 2005 is effectively 2004.99999...).
-
-## Relative/net survival
-
-Relative/net survival estimation requires knowledge of the expected hazard levels for the individuals in the data. In `survtab` this is accomplished by passing a long-format `data.frame` of population hazards via the `pophaz` argument. E.g. the `popmort` dataset included in `popEpi` (Finnish overall mortality rates for men and women).
-
-```{r popmort}
-data(popmort)
-pm <- data.frame(popmort)
-names(pm) <- c("sex", "CAL", "AGE", "haz")
-head(pm)
-```
-
-The `data.frame` should contain a variable named `"haz"` indicating the population hazard at the level of one subject-year. Any other variables are considered to be variables, by which to merge population hazards to the (split) subject-level data within `survtab`. These merging variables may correspond to the time scales in the used `Lexis` object. This allows for e.g. merging in different population hazards for the same subject as they get older.
-
-The following causes `survtab` to estimate EdererII relative survival:
-
-```{r survtab_e2}
-st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-```{r}
-plot(st.e2, y = "r.e2", col = c("blue", "red"))
-```
-
-Note that the curves diverge due to merging in the "wrong" population hazards for some individuals which we randomized earlier to be male though all the individuals in data are actually female. Pohar-Perme-weighted estimates can be computed by
-
-```{r survtab_pp}
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-Compare with EdererII estimates:
-
-```{r}
-plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
-lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
-```
-
-## Adjusting estimates
-
-`survtab` also allows for adjusting the survival curves by categorical variables --- typically by age groups. The following demonstrates how:
-
-```{r survtab_adjust}
-## an age group variable
-x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
-
-## using "internal weights" - see ?ICSS for international weights standards
-w <- table(x$agegr)
-w
-
-w <- list(agegr = as.numeric(w))
-```
-
-```{r survtab_adjust_2}
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-```{r}
-plot(st.as, y = "r.e2.as", col = c("blue", "red"))
-```
-
-We now have age-adjusted EdererII relative/net survival estimates. The `weights` argument allows for either a list of weights (with one or multiple variables to adjust by) or a `data.frame` of weights. Examples:
-
-```{r weights_examples, eval = TRUE}
-list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
-
-wdf <- merge(0:1, 1:4)
-names(wdf) <- c("sex", "agegr")
-wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
-wdf
-```
-
-The weights do not have to sum to one when supplied as they are internally forced to do so within each stratum. In the `data.frame` of weights, the column of actual weights to use must be named "weights". When there are more than one variable to adjust by, and a list of weights has been supplied, the variable-specific weights are first multiplied together (cumulatively) and then scaled to sum to one.
-
-This adjusting can be done to any survival time function that `survtab` (and `survtab_ag`) estimates. One can also supply adjusting variables via the `adjust` argument if convenient:
-
-```{r survtab_adjust_3}
-st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
-                 adjust = "agegr",
-                 data = x, weights = w,
-                 surv.type = "surv.rel", relsurv.method = "e2",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-```
-
-Where `adjust` could also be `adjust = agegr`, `adjust = list(agegr)` or 
-
-`adjust = list(agegr = cut(dg_age, c(0, 60, 70, 80, Inf), right = FALSE))`
-
-for exactly the same results. When adjusting by multiple variables, one must supply a vector of variable names in data or a list of multiple elements (as in the base function `aggregate`).
-
-## Other survival time functions
-
-One can also estimate cause-specific survival functions, cumulative incidence functions (CIFs, a.k.a. crude risk a.k.a. absolute risk functions), and CIFs based on the excess numbers of events. Cause-specific survival is close to net survival as they are philosophically highly similar concepts: 
-
-```{r survtab_cause}
-st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                 data = x, 
-                 surv.type = "surv.cause",
-                 breaks = list(FUT = seq(0, 5, 1/12)))
-
-st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
-                 surv.type = "surv.rel", relsurv.method = "pp",
-                 breaks = list(FUT = seq(0, 5, 1/12)),
-                 pophaz = pm)
-
-plot(st.ca, y = "surv.obs.canD", col = "blue")
-lines(st.pp, y = "r.pp", col = "red")
-```
-
-Absolute risk:
-
-```{r survtab_cif}
-st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.obs",
-                  breaks = list(FUT = seq(0, 5, 1/12)))
-
-plot(st.cif, y = "CIF_canD", conf.int = FALSE)
-lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
-```
-
-The "relative CIF" attempts to be close to the true CIF without using knowledge about the types of events, e.g. causes of death:
-
-```{r survtab_relcif}
-st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
-                  data = x, 
-                  surv.type = "cif.rel",
-                  breaks = list(FUT = seq(0, 5, 1/12)),
-                  pophaz = pm)
-plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
-lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
-```
-
-
-# Using `survtab_ag`
-
-Arguments concerning the types and methods of estimating of survival time functions work the same in `survtab_ag` as in `survtab` (the latter uses the former). However, with aggregated data one must explicitly supply the various count and person-time variables. Also, usage of the `formula` argument is different.
-
-For demonstration purposes we form an aggregated data set using `lexpand`; see `?lexpand` for more information on that function.
-
-```{r}
-sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              aggre = list(sex, fot))
-head(ag)
-```
-
-Now simply do:
-
-```{r survtab_ag_example1}
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "hazard",
-                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
-```
-
-Or:
-
-```{r survtab_ag_example2}
-st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
-                 surv.method = "lifetable",
-                 d = c("from0to1", "from0to2"), n = "at.risk",
-                 n.cens = "from0to0")
-```
-
-Note that e.g. argument `d` could also have been supplied as 
-
-`list(from0to1, from0to2)`
-
-or
-
-`list(canD = from0to1, othD = from0to2)`
-
-for identical results. The last is convenient for e.g. `surv.cause` computations:
-
-```{r survtab_ag_cause}
-st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
-                    surv.method = "hazard",
-                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
-plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
-```
-
-One has to supply the most variables when computing Pohar-Perme estimates (though it is probably rare to have third-source aggregated data with Pohar-Perme weighted figures, it is implemented here to be used as a workhorse for `survtab`). For this we must aggregate again to get the Pohar-Perme weighted counts and subject-times:
-
-```{r}
-ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
-              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
-              pophaz = popmort, pp = TRUE,
-              aggre = list(sex, fot))
-
-st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
-                    surv.method = "hazard", relsurv.method = "pp",
-                    d = list(from0to1 + from0to2), pyrs = "pyrs",
-                    d.pp = list(from0to1.pp + from0to2.pp),
-                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
-                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
-plot(st.pp, y = "r.pp", col = c("blue", "red"))
-```
-
-Here it is best to supply only one column to each argument since Pohar-Perme estimates will not be computed for several types of events at the same time.
-
-
-
+---
+title: "Examples of using survtab"
+author: "Joonas Miettinen"
+date: "`r Sys.Date()`"
+output: 
+  html_document:
+    toc: true
+    toc_depth: 2
+    fig_width: 6
+    fig_height: 6
+vignette: >
+  %\VignetteEngine{knitr::rmarkdown}
+  %\VignetteIndexEntry{survtab examples}
+  %\usepackage[utf8]{inputenc}
+---
+
+# Overview
+  
+This vignette aims to clarify the usage of the `survtab_ag` and `survtab` functions included in this package. `survtab_ag` estimates various survival functions and cumulative incidence functions (CIFs) non-parametrically using aggregated data, and `survtab` is a wrapper for `survtab_ag`, to which `Lexis` data is supplied. 
+  
+Two methods (`surv.method`) are currently supported: The `"lifetable"` (actuarial) method only makes use of counts when estimating any of the supported survival time functions. The default method (`"hazard"`}) estimates appropriate hazards and transforms them into survival function or CIF estimates.
+  
+For relative survival estimation we need also to enumerate the expected hazard levels for the subjects in the data. This is done by merging expected hazards to individuals' subintervals (which divide their survival time lines to a number of small intervals). For Pohar-Perme-weighted analyses one must additionally compute various weighted figures at the level of split subject data. 
+  
+If one has subject-level data, the simplest way of computing survival function estimates with `popEpi` is by defining a `Lexis` object and using `survtab`, which will do the rest. For pre-aggregated data one may use the `survtab_ag` function instead. One can also use the `lexpand` function to split, merge population hazards, and aggregate in a single function call and then use `survtab_ag` if that is convenient.
+  
+# Using `survtab`
+  
+It is straightforward to estimate various survival time functions with `survtab` once a `Lexis` object has been defined (see `?Lexis` in package `Epi` for details):
+  
+```{r pkgs, eval = TRUE, echo = TRUE, message = FALSE}
+library(popEpi)
+library(Epi)
+```
+
+```{r}
+data(sire)
+
+## NOTE: recommended to use factor status variable
+x <- Lexis(entry = list(FUT = 0, AGE = dg_age, CAL = get.yrs(dg_date)), 
+           exit = list(CAL = get.yrs(ex_date)), 
+           data = sire[sire$dg_date < sire$ex_date, ],
+           exit.status = factor(status, levels = 0:2, 
+                                labels = c("alive", "canD", "othD")), 
+           merge = TRUE)
+
+## pretend some are male
+set.seed(1L)
+x$sex <- rbinom(nrow(x), 1, 0.5)
+
+## observed survival - explicit method
+st <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## observed survival - easy method (assumes lex.Xst in x is the status variable)
+st <- survtab(FUT ~ sex, data = x, 
+              surv.type = "surv.obs",
+              breaks = list(FUT = seq(0, 5, 1/12)))
+
+## printing gives the used settings and 
+## estimates at the middle and end of the estimated
+## curves; more information available using summary()
+st
+
+```
+
+Plotting by strata (men = blue, women = red):
+
+```{r}
+plot(st, col = c("blue", "red"))
+```
+
+Note that the correct usage of the `formula` argument in `survtab` specifies the time scale in the `Lexis` object over which survival is computed (here `"FUT"` for follow-up time). This is used to identify the appropriate time scale in the data. When only supplying the survival time scale as the right-hand-side of the formula, the column `lex.Xst` in the supplied `Lexis` object is assumed to be the (correctly formatted!) status variable. When using `Surv()` to be explicit, we effectively (and exceptionally) pass the starting times to the `time` argument in `Surv()`, and `time2` is ignored entirely. The function will fail if `time` does not match exactly with a time scale in data.
+
+When using `Surv()`, one must also pass the status variable, which can be something other than the `lex.Xst` variable created by `Lexis()`, though usually ``lex.Xst` is what you want to use (especially if the data has already been split using e.g. `splitLexis` or `splitMulti`, which is allowed). It is recommended to use a factor status variable to pass to `Surv()`, though a numeric variable will work in simple cases (0 = alive, 1 = dead; also `FALSE`  = alive, `TRUE` = dead). Using `Surv()` also allows easy passing of transformations of `lex.Xst`, e.g. `Surv(FUT, lex.Xst %in% 1:2)`.
+
+The argument `breaks` must be a named list of breaks by which to split the `Lexis` data (see `?splitMulti`). It is mandatory to assign breaks at least to the survival time scale (`"FUT"` in our example) so that `survtab` knows what intervals to use to estimate the requested survival time function(s). The breaks also determine the window used: It is therefore easy to compute so called period estimates by defining the roof and floor along the calendar time scale, e.g. 
+
+`breaks = list(FUT = seq(0, 5, 1/12), CAL = c(2000, 2005))`
+
+would cause `survtab` to compute period estimates for 2000-2004 (breaks given here as fractional years, so 2005 is effectively 2004.99999...).
+
+## Relative/net survival
+
+Relative/net survival estimation requires knowledge of the expected hazard levels for the individuals in the data. In `survtab` this is accomplished by passing a long-format `data.frame` of population hazards via the `pophaz` argument. E.g. the `popmort` dataset included in `popEpi` (Finnish overall mortality rates for men and women).
+
+```{r popmort}
+data(popmort)
+pm <- data.frame(popmort)
+names(pm) <- c("sex", "CAL", "AGE", "haz")
+head(pm)
+```
+
+The `data.frame` should contain a variable named `"haz"` indicating the population hazard at the level of one subject-year. Any other variables are considered to be variables, by which to merge population hazards to the (split) subject-level data within `survtab`. These merging variables may correspond to the time scales in the used `Lexis` object. This allows for e.g. merging in different population hazards for the same subject as they get older.
+
+The following causes `survtab` to estimate EdererII relative survival:
+
+```{r survtab_e2}
+st.e2 <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+```{r}
+plot(st.e2, y = "r.e2", col = c("blue", "red"))
+```
+
+Note that the curves diverge due to merging in the "wrong" population hazards for some individuals which we randomized earlier to be male though all the individuals in data are actually female. Pohar-Perme-weighted estimates can be computed by
+
+```{r survtab_pp}
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+Compare with EdererII estimates:
+
+```{r}
+plot(st.e2, y = "r.e2", col = c("blue", "red"), lty = 1)
+lines(st.pp, y = "r.pp", col = c("blue", "red"), lty = 2)
+```
+
+## Adjusting estimates
+
+`survtab` also allows for adjusting the survival curves by categorical variables --- typically by age groups. The following demonstrates how:
+
+```{r survtab_adjust}
+## an age group variable
+x$agegr <- cut(x$dg_age, c(0, 60, 70, 80, Inf), right = FALSE)
+
+## using "internal weights" - see ?ICSS for international weights standards
+w <- table(x$agegr)
+w
+
+w <- list(agegr = as.numeric(w))
+```
+
+```{r survtab_adjust_2}
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex + adjust(agegr), 
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+```{r}
+plot(st.as, y = "r.e2.as", col = c("blue", "red"))
+```
+
+We now have age-adjusted EdererII relative/net survival estimates. The `weights` argument allows for either a list of weights (with one or multiple variables to adjust by) or a `data.frame` of weights. Examples:
+
+```{r weights_examples, eval = TRUE}
+list(sex = c(0.4, 0.6), agegr = c(0.2, 0.2, 0.4, 0.2))
+
+wdf <- merge(0:1, 1:4)
+names(wdf) <- c("sex", "agegr")
+wdf$weights <- c(0.1, 0.1, 0.1, 0.1, 0.2, 0.2, 0.1, 0.1)
+wdf
+```
+
+The weights do not have to sum to one when supplied as they are internally forced to do so within each stratum. In the `data.frame` of weights, the column of actual weights to use must be named "weights". When there are more than one variable to adjust by, and a list of weights has been supplied, the variable-specific weights are first multiplied together (cumulatively) and then scaled to sum to one.
+
+This adjusting can be done to any survival time function that `survtab` (and `survtab_ag`) estimates. One can also supply adjusting variables via the `adjust` argument if convenient:
+
+```{r survtab_adjust_3}
+st.as <- survtab(Surv(time = FUT, event = lex.Xst) ~ sex, 
+                 adjust = "agegr",
+                 data = x, weights = w,
+                 surv.type = "surv.rel", relsurv.method = "e2",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+```
+
+Where `adjust` could also be `adjust = agegr`, `adjust = list(agegr)` or 
+
+`adjust = list(agegr = cut(dg_age, c(0, 60, 70, 80, Inf), right = FALSE))`
+
+for exactly the same results. When adjusting by multiple variables, one must supply a vector of variable names in data or a list of multiple elements (as in the base function `aggregate`).
+
+## Other survival time functions
+
+One can also estimate cause-specific survival functions, cumulative incidence functions (CIFs, a.k.a. crude risk a.k.a. absolute risk functions), and CIFs based on the excess numbers of events. Cause-specific survival is close to net survival as they are philosophically highly similar concepts: 
+
+```{r survtab_cause}
+st.ca <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                 data = x, 
+                 surv.type = "surv.cause",
+                 breaks = list(FUT = seq(0, 5, 1/12)))
+
+st.pp <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, data = x, 
+                 surv.type = "surv.rel", relsurv.method = "pp",
+                 breaks = list(FUT = seq(0, 5, 1/12)),
+                 pophaz = pm)
+
+plot(st.ca, y = "surv.obs.canD", col = "blue")
+lines(st.pp, y = "r.pp", col = "red")
+```
+
+Absolute risk:
+
+```{r survtab_cif}
+st.cif <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.obs",
+                  breaks = list(FUT = seq(0, 5, 1/12)))
+
+plot(st.cif, y = "CIF_canD", conf.int = FALSE)
+lines(st.cif, y = "CIF_othD", conf.int = FALSE, col = "red")
+```
+
+The "relative CIF" attempts to be close to the true CIF without using knowledge about the types of events, e.g. causes of death:
+
+```{r survtab_relcif}
+st.cir <- survtab(Surv(time = FUT, event = lex.Xst) ~ 1, 
+                  data = x, 
+                  surv.type = "cif.rel",
+                  breaks = list(FUT = seq(0, 5, 1/12)),
+                  pophaz = pm)
+plot(st.cif, y = "CIF_canD", conf.int = FALSE, col = "blue")
+lines(st.cir, y = "CIF.rel", conf.int = FALSE, col = "red")
+```
+
+
+# Using `survtab_ag`
+
+Arguments concerning the types and methods of estimating of survival time functions work the same in `survtab_ag` as in `survtab` (the latter uses the former). However, with aggregated data one must explicitly supply the various count and person-time variables. Also, usage of the `formula` argument is different.
+
+For demonstration purposes we form an aggregated data set using `lexpand`; see `?lexpand` for more information on that function.
+
+```{r}
+sire$sex <- rbinom(nrow(sire), size = 1, prob = 0.5)
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              aggre = list(sex, fot))
+head(ag)
+```
+
+Now simply do:
+
+```{r survtab_ag_example1}
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "hazard",
+                 d = c("from0to1", "from0to2"), pyrs = "pyrs")
+```
+
+Or:
+
+```{r survtab_ag_example2}
+st <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.obs",
+                 surv.method = "lifetable",
+                 d = c("from0to1", "from0to2"), n = "at.risk",
+                 n.cens = "from0to0")
+```
+
+Note that e.g. argument `d` could also have been supplied as 
+
+`list(from0to1, from0to2)`
+
+or
+
+`list(canD = from0to1, othD = from0to2)`
+
+for identical results. The last is convenient for e.g. `surv.cause` computations:
+
+```{r survtab_ag_cause}
+st.ca <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.cause",
+                    surv.method = "hazard",
+                    d = list(canD = from0to1, othD = from0to2), pyrs = "pyrs")
+plot(st.ca, y = "surv.obs.canD", col = c("blue", "red"))
+```
+
+One has to supply the most variables when computing Pohar-Perme estimates (though it is probably rare to have third-source aggregated data with Pohar-Perme weighted figures, it is implemented here to be used as a workhorse for `survtab`). For this we must aggregate again to get the Pohar-Perme weighted counts and subject-times:
+
+```{r}
+ag <- lexpand(sire, birth = "bi_date", entry = "dg_date", exit = "ex_date",
+              status = "status", breaks = list(fot = seq(0, 5, 1/12)), 
+              pophaz = popmort, pp = TRUE,
+              aggre = list(sex, fot))
+
+st.pp <- survtab_ag(fot ~ sex, data = ag, surv.type = "surv.rel",
+                    surv.method = "hazard", relsurv.method = "pp",
+                    d = list(from0to1 + from0to2), pyrs = "pyrs",
+                    d.pp = list(from0to1.pp + from0to2.pp),
+                    d.pp.2 = list(from0to1.pp.2 + from0to2.pp.2),
+                    pyrs.pp = "ptime.pp", d.exp.pp = "d.exp.pp")
+plot(st.pp, y = "r.pp", col = c("blue", "red"))
+```
+
+Here it is best to supply only one column to each argument since Pohar-Perme estimates will not be computed for several types of events at the same time.
+
+
+

Debdiff

[The following lists of changes regard files as different if they have different names, permissions or owners.]

Files in second set of .debs but not in first

-rw-r--r--  root/root   /usr/lib/R/site-library/popEpi/doc/sir.html
-rw-r--r--  root/root   /usr/lib/R/site-library/popEpi/doc/survtab_examples.html

Files in first set of .debs but not in second

-rw-r--r--  root/root   /usr/share/doc/r-cran-popepi/tests/testthat/Rplots.pdf

Control files: lines which differ (wdiff format)

  • Depends: r-base-core (>= 4.2.1-2), 4.2.2.20221110-1), r-api-4.0, r-cran-data.table (>= 1.10.4), r-cran-epi (>= 2.0), r-cran-survival

More details

Full run details