aboutsummaryrefslogtreecommitdiff
path: root/data/scripts/objects/test/server.lua
blob: 1213e2fe64d55fd21de2f8936e0faad4e3675156 (plain)
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
-- Server-side code of the test lua object

--
-- Some helper functions and classes
--

-- For debugging
function dump(o)
    if type(o) == 'table' then
        local s = '{ '
        for k,v in pairs(o) do
                if type(k) ~= 'number' then k = '"'..k..'"' end
                s = s .. '['..k..'] = ' .. dump(v) .. ','
        end
        return s .. '} '
    else
        return tostring(o)
    end
end

function table.copy(t)
	local t2 = {}
	for k,v in pairs(t) do
		t2[k] = v
	end
	return t2
end

function vector_zero()
	return {X=0,Y=0,Z=0}
end

function vector_subtract(a, b)
	return {X=a.X-b.X, Y=a.Y-b.Y, Z=a.Z-b.Z}
end

function vector_add(a, b)
	return {X=a.X+b.X, Y=a.Y+b.Y, Z=a.Z+b.Z}
end

function vector_multiply(a, d)
	return {X=a.X*d, Y=a.Y*d, Z=a.Z*d}
end

function vector_copy(a)
	return {X=a.X, Y=a.Y, Z=a.Z}
end

function vector_length(a)
	return math.sqrt(a.X*a.X + a.Y*a.Y + a.Z*a.Z)
end

function vector_eq(a, b)
	return (a.X==b.X and a.Y==b.Y and a.Z==b.Z)
end

function round(num, idp)
	local mult = 10^(idp or 0)
	return math.floor(num * mult + 0.5) / mult
end

function vector_snap(a)
	return {X=round(a.X, 0), Y=round(a.Y, 0), Z=round(a.Z, 0)}
end

--
-- Actual code
--

CONTENT_STONE = 0

is_digger = false
counter = 0
counter2 = 0
counter3 = 0
counter4 = 0
counter_move = 0
death_counter = 0
-- This is set in on_initialize()
position = {X=0,Y=0,Z=0}
starting_position = {X=0,Y=0,Z=0}
rotation = {X=0, Y=math.random(0,360), Z=0}
y_dir = 1
temp1 = 0
speed = 1.5
main_dir = {X=0,Y=0,Z=0}

function dir_goodness(env, pos, dir)
	if vector_eq(dir, vector_zero()) then
		return -1
	end
	p = vector_add(pos, dir)
	n = env_get_node(env, p)
	f = get_content_features(n.content)
	if f.walkable then
		p.Y = p.Y + 1
		n = env_get_node(env, p)
		f = get_content_features(n.content)
		if f.walkable then
			-- Too high
			return -1
		end
		-- Hill
		return 2
	end
	p.Y = p.Y - 1
	n = env_get_node(env, p)
	f = get_content_features(n.content)
	if f.walkable then
		-- Flat
		return 1
	end
	-- Drop
	return 0
end

function on_step(self, dtime)
	-- Limit step to a sane value; it jumps a lot while the map generator
	-- is in action
	if dtime > 0.5 then
		dtime = 0.5
	end

	env = object_get_environment(self)
	
	--[[
	-- Returned value has these fields:
	-- * int content
	-- * int param1
	-- * int param2
	p = {X=position.X, Y=position.Y-0.35, Z=position.Z}
	n = env_get_node(env, p)
	f = get_content_features(n.content)
	if f.walkable then
		y_dir = 1
	else
		y_dir = -1
	end
	-- Keep the object approximately at ground level
	position.Y = position.Y + dtime * 2.0 * y_dir

	-- Move the object around
	position.X = position.X + math.cos(math.pi+rotation.Y/180*math.pi)
			* dtime * speed
	position.Z = position.Z + math.sin(math.pi+rotation.Y/180*math.pi)
			* dtime * speed
	
	-- Rotate the object if it is too far from the starting point
	counter3 = counter3 - dtime
	if counter3 < 0 then
		counter3 = counter3 + 1
		diff = vector_subtract(position, starting_position)
		d = vector_length(diff)
		--print("pos="..dump(position).." starting="..dump(starting_position))
		--print("diff=" .. dump(diff))
		--print("d=" .. d)
		if d > 3 then
			rotation.Y = rotation.Y + 90
			--rotation.Y = rotation.Y + math.random(-180, 180)
		end
	end

	-- This value has to be set; it determines to which player the
	-- object is near to and such
	object_set_base_position(self, position)

	counter4 = counter4 - dtime
	if counter4 < 0 then
		--counter4 = counter4 + math.random(0.5,8)
		counter4 = counter4 + 0.6/speed
		-- Mess around with the map
		if is_digger == true then
			np = vector_add(position, {X=0,Y=-0.6,Z=0})
			env_dig_node(env, np)
		else
			np = vector_add(position, {X=0,Y=0,Z=0})
			env_place_node(env, np, {content=0})
		end
	end
	--]]
	
	counter_move = counter_move - dtime
	if counter_move < 0 then
		counter_move = counter_move + 1/speed
		if counter_move < 0 then counter_move = 0 end

		old_position = vector_copy(position)

		dirs = {
			{X=1, Y=0, Z=0},
			{X=-1, Y=0, Z=0},
			{X=0, Y=0, Z=1},
			{X=0, Y=0, Z=-1}
		}
		
		best_dir = main_dir
		best_goodness = dir_goodness(env, position, main_dir)

		for k,v in ipairs(dirs) do
			-- Don't go directly backwards
			if not vector_eq(vector_subtract(vector_zero(), v), main_dir) then
				goodness = dir_goodness(env, position, v)
				if goodness > best_goodness then
					best_dir = v
					goodness = best_goodness
				end
			end
		end

		-- Place stone block when dir changed
		if not vector_eq(main_dir, best_dir) then
			np = vector_add(position, {X=0,Y=0,Z=0})
			env_place_node(env, np, {content=CONTENT_STONE})
		end

		main_dir = best_dir

		position = vector_add(position, main_dir)
		
		pos_diff = vector_subtract(position, old_position)
		rotation.Y = math.atan2(pos_diff.Z, pos_diff.X)/math.pi*180-180
		
		-- Returned value has these fields:
		-- * int content
		-- * int param1
		-- * int param2
		p = {X=position.X, Y=position.Y, Z=position.Z}
		n = env_get_node(env, p)
		f = get_content_features(n.content)
		if f.walkable then
			position.Y = position.Y + 1
		end
		p = {X=position.X, Y=position.Y-1, Z=position.Z}
		n = env_get_node(env, p)
		f = get_content_features(n.content)
		if not f.walkable then
			position.Y = position.Y - 1
		end
		
		-- Center in the middle of the node
		position = vector_snap(position)
	end

	-- This value has to be set; it determines to which player the
	-- object is near to and such
	object_set_base_position(self, position)
	
	--[[
	counter4 = counter4 - dtime
	if counter4 < 0 then
		--counter4 = counter4 + math.random(0.5,8)
		counter4 = counter4 + 0.6/speed
		-- Mess around with the map
		if is_digger == true then
			np = vector_add(position, {X=0,Y=-0.6,Z=0})
			env_dig_node(env, np)
		else
			np = vector_add(position, {X=0,Y=0,Z=0})
			env_place_node(env, np, {content=0})
		end
	end
	--]]
	
	---[[
	-- Send some custom messages at a custom interval
	
	counter = counter - dtime
	if counter < 0 then
		counter = counter + 0.25
		if counter < 0 then
			counter = 0
		end

		message = "pos " .. position.X .. " " .. position.Y .. " " .. position.Z
		object_add_message(self, message)

		message = "rot " .. rotation.X .. " " .. rotation.Y .. " " .. rotation.Z
		object_add_message(self, message)
	end
	--]]

	-- Remove the object after some time
	death_counter = death_counter + dtime
	if death_counter > 40 then
		object_remove(self)
	end

end

-- This stuff is passed to a newly created client-side counterpart of this object
function on_get_client_init_data(self)
	-- Just return some data for testing
	return "result of get_client_init_data"
end

-- This should return some data that mostly saves the state of this object
-- Not completely implemented yet
function on_get_server_init_data(self)
	-- Just return some data for testing
	return "result of get_server_init_data"
end

-- When the object is loaded from scratch, this is called before the first
-- on_step(). Data is an empty string or the output of an object spawner
-- hook, but such things are not yet implemented.
--
-- At reload time, the last output of get_server_init_data is passed as data.
--
-- This should initialize the position of the object to the value returned
--   by object_get_base_position(self)
--
-- Not completely implemented yet
--
function on_initialize(self, data)
	print("server object got initialization: " .. data)
	position = object_get_base_position(self)
	starting_position = vector_copy(position);
	if math.random() < 0.5 then
		is_digger = true
	end
end