Skip to content

Storing Data⚓︎

MWSE offers various ways to store data persistently. Be that across save sessions for the same character or even persistently across different player characters. To store data that is meant to persist between different characters, you can use provided json API. It provides a set of functions needed to save Lua's values to json files and load the data to Lua's tables. This process is called serialization. One requirement is that the value can be serialized. Even the functions used to work with configuration files are implemented with this json-based API (those functions being mwse.saveConfig and mwse.loadConfig).

Serialization⚓︎

Serializible values are Lua's primitive types:

  • boolean
  • number
  • string
  • table's made of above values

You can't serialize MWSE classes such as tes3reference, tes3mobilePlayer, etc.

Persistent Storage for the Same Player Character⚓︎

Besides saving your data to files, you can save your data to some of the MWSE's classes that have a data or tempData property. Those are:

  • tes3combatSession.data
  • tes3itemData.data
  • tes3reference.data
  • tes3itemData.tempData
  • tes3reference.tempData

Note

Both data and tempData tables can only be used to store serializible data.

Data stored in the data table on a certain object will persist between savegame sessions, while data stored in tempData table will be cleared on game reload. There are some peculiarities when working with these tables:

  • Each of the table fields needs to be declared one by one.
  • Not every object can have Lua data. You can check that with myRef.supportsLuaData property.
  • The data field on a tes3reference won't persist between savegames if the reference isn't marked as modified (NPCs and creatures are usually marked as modified by other parts of the engine). Make sure to mark your reference as modified explicitly: myRef.modified = true

Example: creating a table inside data table on the player's reference

-- Correct way
tes3.player.data.myMod = {}
tes3.player.data.myMod.var1 = {}
tes3.player.data.myMod.var2 = 32

-- Wrong way, this would rise an error
tes3.player.data.myMod = {
    var1 = {},
    var2 = 32
}


-- On the other hand we can save our created
-- table to a local variable for easy access
-- Let's create our table first
tes3.player.data.myMod = {}

-- Now let's save it to a local variable
local myData = tes3.player.data.myMod

-- Let's save something now
myData.var1 = {}
myData.var2 = 32

The field data on the player's reference could be the perfect way to store some mod-related player statistics. For example, a mod implementing a karma system could save the player's current karma level inside that table, so that it can persist between save sessions. On the other hand, the data that the mod's user would want to persist between whichever character the player or savegame is played would be saved to a configuration file instead. For instance, keybindings for new abilities are a good candidate for that.

Usage of data table⚓︎

Here an example of a simple mod is given which stores some variables to tes3.player.data and tes3.player.tempData.

local shrineIds = {
    ["fields of kummu"] = true,
    ["vivec, temple"] = true,
    ["vivec, puzzle canal, center"] = true,
    ["gnisis, temple"] = true,
    ["koal cave"] = true,
    ["ghostgate, temple"] = true
}
local regions = {
    ["balmora"] = true,
    ["seyda neen"] = true,
    ["ald-ruhn"] = true
}

-- This is the default layout for the table
-- stored on tes3.player.data
---@class myData
local defaults = {
    regionalBounties = {
        -- This table is unused in this example, but
        -- it shows how to  initialize a sub-table.
        ["balmora"] = 0,
        ["seyda neen"] = 0,
        ["ald-ruhn"] = 0
    },
    karma = 0,
    shrinesVisited = {}
}

--- This function will recursively set all the fields on our
--- tes3.player.data table if they don't exist already
---@param data table
---@param t table
local function initTableValues(data, t)
    for k, v in pairs(t) do
        -- If a field already exists - we initialized the data
        -- table for this character before. Don't do anything.
        if data[k] == nil then
            if type(v) ~= "table" then
                data[k] = v
            elseif v == {} then
                data[k] = {}
            else
                -- Fill out the sub-tables
                data[k] = {}
                initTableValues(data[k], v)
            end
        end
    end
end

--- This is a standard function that will create
--- a table for our mod's storage in tes3.player.data
local function initializeData()
    local data = tes3.player.data
    data.myMod = data.myMod or {}
    local myData = data.myMod
    initTableValues(myData, defaults)
end
event.register(tes3.event.loaded, initializeData)

--- This is a convinience function to get our storage
---@return myData
local function getData()
    return tes3.player.data.myMod
end

--- This function will handle updating the player's Current
--- karma level, and handle if it passes over -100 or 100
---@param delta integer
local function modKarma(delta)
    local myData = getData()
    local oldKarma = myData.karma
    myData.karma = myData.karma + delta

    local karma = myData.karma
    local absOldKarma = math.abs(oldKarma)
    local absKarma = math.abs(karma)
    local barrierCrossed = ((absOldKarma >= 100) and (absKarma < 100)) or
                           ((absKarma >= 100) and (absOldKarma < 100))

    if barrierCrossed then
        -- Let's log to see what's happening
        tes3.messageBox("Current karma: %s", karma)
        if karma < -100 then
            tes3.messageBox("Because of your deeds your karma now reached evil level.")

            -- Now let's store that to tes3.player.tempData for later use
            -- We can use table.getset to create a table for our mod in
            -- tes3.player.tempData if it doesn't exist yet.
            local temp = table.getset(tes3.player.tempData, "myMod", {})

            -- Now actually store player's karma range
            temp.encounter = "bad"

        elseif karma < 100 then
            -- -100 < karma < 100
            tes3.messageBox("Because of your deeds your karma now reached neutral level.")
            local temp = table.getset(tes3.player.tempData, "myMod", {})
            temp.encounter = "neutral"

        elseif karma >= 100 then
            tes3.messageBox("Through your deeds your karma now reached good level.")
            local temp = table.getset(tes3.player.tempData, "myMod", {})
            temp.encounter = "good"
        end
    end
end

---@param e cellChangedEventData
local function onCellChange(e)
    local cell = e.cell
    local cellId = cell.id:lower()

    -- Check if visited cell is on
    -- the list of valid shrines
    if shrineIds[cellId] then
        local data = getData()

        -- Make sure player didn't already visit this shrine,
        -- not to award the karma boost twice.
        if not data.shrinesVisited[cellId] then
            -- Player visited this shrine for the first time,
            -- award some karma points.
            modKarma(50)

            -- Let's save that to the list of visited shrines in
            -- a table on tes3.player.data, so it can persist
            -- between savegames.
            data.shrinesVisited[cellId] = true
        end
    end
end
event.register(tes3.event.cellChanged, onCellChange)


---@param e lockPickEventData|trapDisarmEventData
local function onPick(e)
    local doorReference = e.reference
    local hasAccess = tes3.hasOwnershipAccess({ target = doorReference })

    if not hasAccess then
        tes3.messageBox("That isn't allowed!")
        modKarma(-25)
        local cellId = tes3.player.cell.id:lower()
        if regions[cellId] then
            local myData = getData()
            myData.regionalBounties[cellId] = myData.regionalBounties[cellId] + 25
        end

    end
end
event.register(tes3.event.lockPick, onPick)
event.register(tes3.event.trapDisarm, onPick)

---@param e calcRestInterruptEventData
local function onCalcRestInterrupt(e)
    local temp = tes3.player.tempData.myMod
    if not temp then return end

    local roll = math.random(100)
    if roll > 60 then return end

    local encounterType = temp.encounter
    if encounterType == "good" then
        -- The player has good karma, let's block
        -- the rest interruption
        e.count = 0

        -- Log what happened. This kind of message wouldn't
        -- be in the final release of the mod.
        tes3.messageBox("Blocking encounter!")
    elseif encounterType == "neutral" then
        local roll = math.random(100)
        if roll < 50 then
            e.count = 0
            tes3.messageBox("Blocking encounter!")
        end
    else
        -- Player is a bad guy, let's increase the amount
        -- of spawned creatures by a random number.
        local mod = math.random(10)
        e.count = e.count + mod
        tes3.messageBox("Increased the spawned enemy count!")
    end
end
event.register(tes3.event.calcRestInterrupt, onCalcRestInterrupt)