This repository has been archived by the owner on May 10, 2018. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathLNVL.lua
355 lines (315 loc) · 14.3 KB
/
LNVL.lua
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
--[[
LNVL: The LÖVE Visual Novel Engine
This is the only module your game must import in order to use LNVL.
Since the intent of LNVL is to act as a sub-module for a larger game
it cannot make assumptions about the paths to use in require()
statements below. Often a prefix will need to appear in each of those
statements. For that reason it is a two-step process to use LNVL, for
example:
local LNVL = require "LNVL"
LNVL.Initialize("prefix.to.LNVL.src")
Note well that the argument to Initialize does not end with a period.
It is acceptable for the argument to be an empty string or nil as
well, if no path prefix is necessary.
See the file README.md for more information and links to the official
website with documentation. See the file LICENSE for information on
the license for LNVL.
--]]
-- This table is the global namespace for all LNVL classes, functions,
-- and data. Some of the modules, i.e. the source code files we load
-- during LNVL.Initialize(), rely on access to this global table.
-- Therefore we cannot declare this table as 'local', even though that
-- would be a cleaner approach overall.
LNVL = {}
-- Version information.
LNVL.Version = setmetatable(
{
["Major"] = 0,
["Minor"] = 1,
["Patch"] = 0,
["Label"] = "-unstable",
},
{
__tostring = function()
return string.format("%d.%d.%d%s",
LNVL.Version.Major,
LNVL.Version.Minor,
LNVL.Version.Patch,
LNVL.Version.Label)
end
}
)
-- We sandbox all dialog scripts we load via LNVL.LoadScript() in
-- their own environment so that global variables in those scripts
-- cannot clobber existing global variables in any game using LNVL or
-- in LNVL itself. This table represents the blueprint for that
-- environment. Because of the ability to use Context objects with
-- LoadScript(), it is possibly that we will load a script with an
-- environment that is different from what is in this table by
-- default.
--
-- We explicitly define the 'LNVL' key so that scripts can access the
-- LNVL table. Without that key the scripts could not call any LNVL
-- functions and that would make it impossible to define scripts,
-- characters, or do anything meaningful.
LNVL.ScriptEnvironment = { ["LNVL"] = LNVL }
-- For debugging purposes we allow the use of Lua's print(), assert(),
-- error(), and tostring() functions within dialogue scripts.
--
-- The definitions for assert() and error() are deferred until we load
-- the LNVL.Debug module so that we can use the special versions of
-- those functions defined by that module.
LNVL.ScriptEnvironment["print"] = print
LNVL.ScriptEnvironment["tostring"] = tostring
-- This is a lookup table of functions which are essential to LNVL and
-- which we do not let the user overwrite, otherwise they may be able
-- to do something like rewrite the Character constructor. The keys
-- are strings naming the keywords and the values are strings naming
-- their type, i.e. return values from type(). For example, the value
-- of
--
-- ReservedKeywords["Character"]
--
-- should always be "function". After loading dialog scripts we check
-- to make sure all reserved keywords still have their expected type
-- as a way to see if the user overwrote their definitions. This is
-- not bullet-proof though, as a script could redefine "Character" as
-- another function and get away with it.
local ReservedKeywords = {}
-- This metatable changes the __newindex() of the script environment
-- so that we cannot add anything that conflicts with the names listed
-- in the ReservedKeywords table. Because the __newindex() metamethod
-- is only called when we are adding a new key this is a weak form of
-- protection, but still may help catch some errors.
setmetatable(LNVL.ScriptEnvironment, {
__newindex = function (table, key, value)
if ReservedKeywords[key] ~= nil then
error(key .. " is a reserved word.")
else
rawset(table, key, value)
end
end
})
-- This function creates a constructor alias in the script
-- environment. These aliases allow us to write more terse, readable
-- code in dialog scripts by providing shortcuts for common LNVL
-- constructors we use. For example, by calling
--
-- LNVL.CreateConstructorAlias("Scene", LNVL.Scene)
--
-- we can define scenes in our scripts by simply writing 'FOO =
-- Scene{...}' instead of 'FOO = LNVL.Scene:new{...}'.
--
-- This function expects the name of the alias to create as the first
-- argument, a string, and a reference to the class to instantiate as
-- the second argument. The function expects the class to have a
-- new() method for a constructor.
--
-- The constructor created by this function becomes a reserved
-- keyword in all LNVL dialogue scripts.
--
-- The function returns nothing.
function LNVL.CreateConstructorAlias(name, class)
LNVL.ScriptEnvironment[name] = function (...)
return class:new(...)
end
ReservedKeywords[name] = "function"
end
-- This function creates a function alias, i.e. a function we can use
-- in scripts as a short-cut for a more verbose function defined
-- within LNVL. The first argument must be the alias we want to
-- create, as a string, and the second argument a reference to the
-- actual function to call. This function returns no value.
--
-- Like the above, function aliases created this way become reserved
-- words in all LNVL dialogue scripts.
function LNVL.CreateFunctionAlias(name, implementation)
LNVL.ScriptEnvironment[name] = function (...)
return implementation(...)
end
ReservedKeywords[name] = "function"
end
-- This property represents the current Scene in use. We should
-- rarely change the value of this property directly. Instead we
-- should use the LNVL.ChangeToScene() function. LNVL.LoadScript()
-- will also change this value whenever we load a script that defines
-- a scene named 'START'.
LNVL.CurrentScene = nil
-- This table is a map of all scenes we have visited. That is, scenes
-- for which the user has progressed through their content. The keys
-- are the names of the scenes that we've shown, and the value for
-- each key is always 'true', allowing us to perform a simple look-up
-- to determine if we have already shown a scene or not.
--
-- N.B. This table does not represent the order in which we displayed
-- each scene.
LNVL.VisitedScenes = {}
-- This table is a stack respresenting the exact order in which the
-- player has traversed through the scenes. Using various gameplay
-- mechanics it may be possible to back up through old scenes, and see
-- the implementation of the 'set-scene' instruction for details.
--
-- The keys are numeric indicates and the values are LNVL.Scene's.
-- Developers may write code like
--
-- for index,scene in ipairs(LNVL.SceneHistory) do
-- ...
-- end
--
-- to access each LNVL.Scene (as 'scene' above) in the order in which
-- the player encountered those scenes (indicated by 'index' above).
LNVL.SceneHistory = {}
-- This function accepts the name of a scene as a string and changes
-- to that scene. Developers can provide a custom implementation by
-- defining a function for LNVL.Settings.Handlers.ChangeScene().
-- Otherwise LNVL will use a default implementation defined in the
-- `src/scene.lua` file.
LNVL.ChangeToScene = nil
-- This function loads all of the LNVL sub-modules, initializing the
-- engine. The argument, if given, must be a string that will be
-- treated a prefix to the paths for all require() statements we use
-- to load those sub-modules. The function also assigns the argument
-- value to 'LNVL.PathPrefix' so that sub-modules may use it for any
-- path operations.
--
-- The 'prefix' argument must not end in a period.
function LNVL.Initialize(prefix)
LNVL.PathPrefix = prefix or ""
local loadModule = function (name, path)
LNVL[name] = require(string.format("%s.%s", LNVL.PathPrefix, path))
end
-- Because all of the code in the 'src/' directory adds to the LNVL
-- table these require() statements must come after we declare the
-- LNVL table above. We must require() each module in a specific
-- order, so insertions or changes to this list must be careful.
-- First we must load any modules that define global values we may use
-- in the Settings module.
loadModule("Color", "src.color")
loadModule("Position", "src.position")
-- Next we need to load Settings as soon as possible so that other
-- modules can draw default values from there.
loadModule("Settings", "src.settings")
-- We want to load Debug after Settings and before other modules so
-- that they can have special behavior if debug mode is enabled, which
-- the Settings module controls.
loadModule("Debug", "src.debug")
-- Then we should load the Graphics module so that the rest have
-- access to primitive rendering functions.
loadModule("Graphics", "src.graphics")
-- Next come the Opcode and Instruction modules, in that order, since
-- the remaining modules may generate opcodes. And since opcodes
-- create instructions we load them in that sequence.
loadModule("Opcode", "src.opcode")
loadModule("Instruction", "src.instruction")
-- Next comes the Drawable module, which classes below may use for
-- certain properties.
loadModule("Drawable", "src.drawable")
-- The order of the remaining modules can come in any order as they do
-- not depend on each other.
--
-- Note that we load the LNVL.MenuChoice class inside of the LNVL.Menu
-- code, so it does not appear in the list below.
loadModule("Character", "src.character")
loadModule("Scene", "src.scene")
loadModule("Menu", "src.menu")
loadModule("Progress", "src.progress")
loadModule("Context", "src.context")
end
-- This function lets us advance through a dialog by pressing the
-- appropriate key declared in love.keypressed(). If the current dialog
-- has been fully displayed, it will move on to the next scene.
-- Otherwise, it will fully display the current dialog.
function LNVL.Advance()
if LNVL.Graphics.dialogProgress >= (#LNVL.Graphics.currentConversationText - LNVL.Graphics.displayLength)
and LNVL.CurrentScene.opcodeIndex < #LNVL.CurrentScene.opcodes then
LNVL.Graphics.dialogProgress = 0
LNVL.Graphics.displayLength = LNVL.Graphics.displaySpeedDefault
LNVL.CurrentScene:moveForward()
else
LNVL.Graphics.dialogProgress = #LNVL.Graphics.currentConversationText
end
end
-- The purpose of this function is to merge the data of Context
-- objects with LNVL.ScriptEnvironment whenever we call LoadScript().
-- It is not intended to be a general-purpose table merging function
-- and will break on such things as tables with circular values.
--
-- The function returns no value. It directly modifies the key-value
-- pairs of the LNVL.ScriptEnvironment table itself.
local function mergeContexts(...)
local contexts = {...}
if #contexts == 0 then return end
for _,context in ipairs(contexts) do
for key,value in pairs(context.data) do
LNVL.ScriptEnvironment[key] = value
end
end
end
-- We run this function after loading a script to make sure the
-- reserved keywords still have their required type. This is a way of
-- protecting dialog scripts from overwriting such things as the
-- 'Scene' constructor. However, it is not a bullet-proof solution.
-- If a dialog script redefines 'Scene' as another function then this
-- test will not see that as an error.
local function checkReservedKeywordTypes()
for key,value in pairs(LNVL.ScriptEnvironment) do
if ReservedKeywords[key] ~= nil then
local environmentType = type(LNVL.ScriptEnvironment[key])
if environmentType ~= ReservedKeywords[key] then
error(("Reserved keyword %s has the incorrect type %s."):format(key, environmentType))
end
end
end
end
-- This function loads an external LNVL script, i.e. one defining
-- scenes and story content. The argument is the path to the file;
-- the function assumes the caller has already ensured the file exists
-- and will crash with an error if the file is not found.
--
-- After the filename can come any number of LNVL.Context objects,
-- which will modify the script environment for the script we're
-- loading *AND* all future scripts we load.
--
-- The function returns no value.
function LNVL.LoadScript(filename, ...)
local script = love.filesystem.load(filename)
mergeContexts(...)
assert(script, "Could not load script " .. filename)
setfenv(script, LNVL.ScriptEnvironment)
-- The variable 'script' is a chunk reperesenting the code from
-- the file we loaded. In other words, 'script' is a function we
-- can execute to run the code from that file. If we are using
-- debug mode then we call script() like a regular function. This
-- will cause LNVL to crash if the code in that chunk causes any
-- errors. If we are running in debug mode that is what we want.
-- But if we are not using debug mode then we execute the chunck
-- in a protected mode and silently ignore any errors.
if LNVL.Settings.DebugModeEnabled == true then
script()
else
pcall(script)
end
-- Immediately after loading a script we try to ensure that the
-- script did not redefine any reserved keywords.
checkReservedKeywordTypes()
-- We always treat 'START' as the initial scene in any story so we
-- update the current scene if 'START' exists.
if LNVL.ScriptEnvironment["START"] ~= nil then
LNVL.ChangeToScene("START")
end
-- Once we have finished loading a script we loop through the
-- script environment looking for every Scene object. We then
-- take the key for that Scene from the environment table,
-- representing the variable name used when defining the scene,
-- and assign it to the Scene.id property for future debugging
-- output. There is some redundancy here in that we will perform
-- this action for scenes which we have already tagged earlier.
for key,value in pairs(LNVL.ScriptEnvironment) do
if getmetatable(value) == LNVL.Scene
or getmetatable(value) == LNVL.Character then
value.id = key
end
end
end
-- Return the LNVL module.
return LNVL