88 Commits

Author SHA1 Message Date
Luke aka SwissalpS
6e11868d1b Fix LBM filter injector crashing server when injector wasn't initialized properly (#168)
* check for existance of list param

I'm not sure this actually fixes the problem.
The issue was raised because of broken map data.

* actually fix the crash
2025-07-04 16:28:05 +02:00
The4codeblocks
2ebc4ac92d Replace minetest namespace with core (#158)
Co-authored-by: SX <50966843+S-S-X@users.noreply.github.com>
2025-06-26 17:41:04 +10:00
OgelGames
d39ff8a097 Check destination node before injecting item (#163) 2025-06-26 17:33:25 +10:00
OgelGames
1b77b64c08 Fix handling of split stacks (#167) 2025-06-26 17:32:25 +10:00
wsor4035
e23a2fda3d fix #161 - change digilines:receptor_send -> digilines.receptor_send 2025-05-18 21:18:43 -04:00
BuckarooBanzay
ad8984c72b fix/shim luacheck errors 2025-05-11 22:02:25 +02:00
Matt Chandler
e657d0476a change digilines:receptor_send -> digilines.receptor_send (#160) 2025-05-09 06:52:08 +02:00
The4codeblocks
8d34ff9079 fix #15 and #62 (process split stacks) (#154)
* fix #15

fixes #15

* fix luacheck stuff

* I mean here

* crash fix + allow arbitrary-count extraction

* crash fix

* wrong place

* missed an end
2025-05-07 20:22:49 +02:00
The4codeblocks
999b43baea digiline → digilines (#156) 2025-05-07 20:13:01 +02:00
The4codeblocks
c8bccec038 Update init.lua (#155) 2025-05-07 20:10:22 +02:00
The4codeblocks
b0496fcd41 fix issue #150 (prevent non-tubes / reinforced tubes from breaking) (#153)
* fix #150

(issue #150)

* tube not tube device

* tntd

* embedded tube is durable

* Update pane_embedded_tube.lua

* Update registration.lua

* Update item_transport.lua

* redundant

* one-way tube is tube

* pipe not tube

* built-in method

* more deduplication

* "=" → " = "
2025-05-07 20:05:54 +02:00
Luke aka SwissalpS
858154cb78 Fix #145 caused by #142 Make new feature opt-in (#146)
* Update default_settings.lua and settingtypes.txt
- Make new vertical behaviour opt-in
- crop comment
2024-12-17 20:43:19 +01:00
tour
dd660c3c1c fix crash from bad digiline msg (#144) 2024-12-01 20:00:02 -05:00
Deathwing777
dcc62eb231 Fixes Issue #64 (#142)
* Fixes Issue #64

Uses a simple boolean setting in default_settings.lua to decide which digiline rules to use in common.lua

* Reviewer Suggestion

Co-authored-by: SX <50966843+S-S-X@users.noreply.github.com>

* Changed Code Comment per Reviewer Suggestion

* Removed trailing whitespace from 2 comment lines.

* Modified settings comment per reviewer suggestion

* Changed variable name per reviewer suggestion.

Variable changed from use_default_digilines_rules to enable_vertical_digilines_connectivity.

---------

Co-authored-by: SX <50966843+S-S-X@users.noreply.github.com>
2024-11-03 19:00:04 -05:00
1F616EMO~nya
222a865e17 Allow digging injectors without clearing the filter (#141)
Thie PR allows diging injectors without clearing their filter inventory. I can't see a reason why this was nessessary.
2024-10-20 12:13:20 -04:00
OgelGames
7079fff85f remove hardcoded stack_max in tube registration
fixes #138
2024-09-07 20:19:46 +10:00
Vitaliy
3189da7c9b add Russian locale (#136) 2024-09-06 06:21:42 +02:00
OgelGames
a2ffaa9dc3 fix nodebreaker not ejecting replacement items
fixes #137
2024-09-05 12:56:33 +10:00
Buckaroo Banzai
63bc754889 luacheck with builtin minetest (#109)
* [PoC] luacheck with builtin minetest

* switch to `minetest+max` std for luacheck

* use branch in action

* use SX's docker image

* disable tty

* Update .github/workflows/luacheck.yml

Co-authored-by: SX <50966843+S-S-X@users.noreply.github.com>

* go back to upstream luacheck

* fix globals listing

---------

Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
Co-authored-by: SX <50966843+S-S-X@users.noreply.github.com>
2024-09-04 22:35:45 +03:00
1F616EMO~nya
171faec7e9 Optimize move_entities_globalstep_part1 (#134) 2024-08-24 12:05:27 -04:00
OgelGames
e9a9bd711e fix typo in pipeworks.override_chest
fixes #129
2024-07-07 11:09:37 +10:00
OgelGames
720f6003ba Add vacuum tube visualizations using vizlib (#128) 2024-07-05 11:17:19 +10:00
1F616EMO~nya
5919f432ae Remove extra print() calls and copy Luacontroller's print behavior (#127) 2024-07-03 15:34:18 +10:00
OgelGames
71fe60014f fix hard-coded inventory name in wielders 2024-05-27 23:19:46 +10:00
OgelGames
223c90e684 Fake player and wielder improvements (#126) 2024-05-27 00:31:04 +10:00
OgelGames
1225e4168b fix splitstacks toggle in default chests 2024-05-15 21:18:26 +10:00
Luke aka SwissalpS
94442e87bb patch for #124 (#125) async race condition fix
* restructure avoiding excessive indentation

* possibly fix #124

* async race condition fix
2024-05-13 12:00:17 +02:00
tour
1169cff163 de-duplicate chest code (#123)
* rewrite compat-chests.lua
add pipeworks.override_chest() to make a existing chest connecting to pipeworks

* move pipeworks.override_chest() to a new file
fix crash with after_place_node() there
let mcl_barrels make use of the new function

* some comments
2024-04-11 10:25:13 +02:00
OgelGames
21dbae9962 fix missing check for pipeworks.enable_item_tags
fixes #117 closes #121
2024-03-30 15:55:59 +11:00
Buckaroo Banzai
1b79084e6b fix chest duplication bug (#119)
closes #118

Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
2024-03-17 17:21:43 -04:00
Luke aka SwissalpS
1577af738f Autocrafter multi group ingredient (#115)
* check for non-zero group value

in theory groups can have negative values too.

* fix #114 multi group recipe items

Fix the bug that prevented crafting with recipes
that had ingredients that need to match multiple
groups.

* comments and whitespace changes

* fix faulty empty table check
2024-03-13 17:31:38 +01:00
OgelGames
8828183bef remove nan values from digiline messages 2024-03-12 14:17:29 +11:00
OgelGames
ce263da6d5 also accept strings of tags in digilines and lua
fixes #116
2024-03-07 14:56:19 +11:00
Slava Zanko
6c66a2f43c Tags support (#107)
Signed-off-by: Slava Zanko <slavazanko@gmail.com>
Co-authored-by: Slava Zanko <slava.zanko@godel.shellenergy.co.uk>
Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
Co-authored-by: OgelGames <olliverdc28@gmail.com>
2024-03-03 22:48:27 +11:00
wsor4035
8724c28939 switch to using xcompat (#113) 2024-03-03 12:37:46 +01:00
Luke aka SwissalpS
cb2a59131c Make wielder nodes not ground content (#112)
nodebreaker, deployer and dispenser
2024-03-03 14:52:23 +13:00
Luke aka SwissalpS
c7b153f1ef Is ground content (#110)
* trashcan isn't ground content

* pipes aren't ground content

* auto tree tap isn't ground content

* filter-injectors aren't ground content

* devices aren't ground content

also some whitespace indentation fixes

* is_ground_content

* autocrafter isn't ground content

* tubes aren't ground content

* pane_embedded_tube isn't ground content

* embedded_tube isn't ground content

* routing tubes aren't ground content

* whitespace fix indentation
2024-02-26 18:44:40 -05:00
Luke aka SwissalpS
96dca7e540 Fix autocrafter crafting bug (#108) 2024-02-08 21:01:54 +11:00
Buckaroo Banzai
c87522c526 export pipeworks.tptube.remove_tube (#105)
Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
2024-01-28 16:08:24 -05:00
OgelGames
6d824a318a Tube repair fixes (#102)
* restart vacuum tube node timers after repair

* clear metadata value after repair
2024-01-14 10:10:24 +01:00
OgelGames
04df87a6c2 Delete .editorconfig 2024-01-14 14:29:55 +11:00
Luke aka SwissalpS
360ed9bdde Autocrafter groups support (#74)
* Add support for recipes with groups to autocrafter

* fix autocrafter replacements

* remove unnecessary check and move some code

* reorder and change back var name

* only whitespace changes

* reuse hash that is asigned to variable anyway

* add get_matching_craft

function that looks for the best matching recipe according to what user
put in recipe inventory

some crafts use groups in multiple recipes

* return right away if there is a chached craft

gives us more horizontal space to work with

* indentation adjusting

* use get_matching_craft to get best craft

this line was buggy anyhow

* new craft format: decremented_input

craft.decremented_input now only holds the items. Extra parameters are
dropped

* bugfix: when item has no groups

"" and unknown items have no def and thus no groups

* add function has_room_for_output

moving the output space calculation into its own function
helps keep autocraft() short and overseeable

* use has_room_for_output

* remove double-check of available materials

* add warning to ensure double-check was not needed

* rebuild cache whenever recipe inv changes

no need to check if still current, this only happens when user is
interacting and can only be better to do so.
Otherwise we would have to check groups etc. again just to check, so
might as well reset cache.

* remove unused line

* use minetest.get_item_group()

* whitespace cleanup

make code easier to read and give it a more
"thoughtthrough" look than the "smash it in to make it work" look it has
had.

(I'm holding back, there would be more to do overall through this mod)

* add .editorconfig

this should also help keep consistency throughout this mod

* changelog updated

at least the most recent more important ones

* whitespace changelog (trailing spaces)

---------

Co-authored-by: rubenwardy <rw@rubenwardy.com>
Co-authored-by: OgelGames <olliverdc28@gmail.com>
Co-authored-by: wsor4035 <24964441+wsor4035@users.noreply.github.com>
2024-01-13 20:41:28 -05:00
Buckaroo Banzai
578e45257b code and registrations cleanup (#101)
* code and registrations cleanup

* don't expose materials to global env

---------

Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
2024-01-13 20:40:01 -05:00
OgelGames
dd6950f7b0 bump min_minetest_version to 5.5.0 for vector.zero()
closes #100
2024-01-05 16:11:23 +11:00
fluxionary
5b0dceb44a add dummy implementations of all current API calls to fake player (#98)
* add dummy implementations of all current API calls to fake player

* fakeplayer: move all stateless API into a metatable

* correctly set metatable
2023-12-29 14:01:08 -05:00
OgelGames
7c74d0bdf3 replace global_exists with get_modpath
related: #97
2023-11-21 14:35:51 +11:00
OgelGames
dc77d1056c fix escaping of ; in teleport tube formspec
closes #93 #94
2023-11-16 14:02:10 +11:00
debagos
ab0713d9a5 Fix item duplication bug with broken tubes. (#92)
It was possible to drag items from broken sorting-/mese-tubes.
2023-11-11 23:06:43 +01:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
b6ae50be87 fix MCL craft, chest item id is "mcl_chests:chest" (#89) 2023-10-09 18:22:14 -04:00
savilli
e7cba4071b Fix crash if add_entity fails (#88) 2023-09-16 18:19:03 +02:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
d90c17edee MineClone smoker and blast furnace support (#85) 2023-09-06 15:33:41 +02:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
4f55610df5 fix MCL crafts (#86) 2023-09-06 12:01:35 +02:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
500cce8668 MineClone containers support (#81)
* mcl_barrels compat

* open barrel is not in creative inv

* fix barrel drop content after dig

* fix barrel local var

* mcl_furnaces compat

* optional depends mcl_barrels and mcl_furnaces

* fixing luacheck

* .luacheckrc update

* renamed files to use _

* renamed files to init.lua

* format improvement
2023-09-02 15:17:14 -04:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
cb3e7757fc Mineclone dig times balanced (#84)
* mineclone dig times balanced

* fix pickaxey dupe
2023-09-01 15:00:50 +02:00
Github is a non-free platform owned by Microsoft. Reasonable alternatives exist, such as Gitea, Sourcehut. We need a federated, mastodon-like forge based on ForgeFed. See: https://forgefed.org
7f93e8aa0a MineClone formspec compat (#82)
* MineClone list backgrounds compat

* mineclone get inv func
2023-08-24 20:50:49 -04:00
Buckaroo Banzai
ebe2331619 remove facedir debugging logs (#73)
Co-authored-by: BuckarooBanzay <BuckarooBanzay@users.noreply.github.com>
2023-06-21 21:22:36 +02:00
OgelGames
2f97a367f3 Fix autocrafter destroying replacement items (#72) 2023-06-21 14:49:34 +10:00
Luke aka SwissalpS
9395013cde Support setting teleport tube can_receive by digilines (#68)
* allow setting can_receive flag by digilines

basically also allows sending tables to change
both or either the tp-channel and receive toggle

* apply suggestions

Co-authored-by: OgelGames <olliverdc28@gmail.com>

---------

Co-authored-by: OgelGames <olliverdc28@gmail.com>
2023-05-28 12:54:05 +10:00
OgelGames
2c880a3843 cache teleport tube logging setting
this gets checked every time an item is teleported
2023-05-19 14:17:04 +10:00
fluxionary
56bcc7dbb7 log when items teleport through tubes (#66)
* log when items teleport through tubes

* put teleport tube logging bethind a setting

* narrow scope of some variables
2023-05-18 19:49:12 +02:00
Awkanimus
bd5a42356b Use sane default i3 inventory settings when missing (#57)
- i3.settings.hotbar_len and i3.settings.inv_size were removed
   in commit 75fddf7 because they are now per-player settings
 - This just uses sane defaults when the settings are not
   present to avoid a few nils and allow the game to continue
2023-01-26 15:44:41 -05:00
wsor4035
b0b675f715 Fix deprecated call to player:get_look_pitch()
Fixes #53
2023-01-09 13:51:49 +11:00
unknown
855937aec3 make all nodes diggable in mcl* survival mode 2022-12-02 15:54:21 -05:00
unknown
fc0d4990b0 make minor bugfixes to restore compat with mineclone2 2022-11-20 19:47:14 -05:00
OgelGames
5e9bfa049b Add digilines to teleport tube and refactor code (#49) 2022-11-11 23:20:28 +11:00
OgelGames
5345fe7af3 Prevent vacuum tubes breaking themselves and refactor code (#51)
also bumps `min_minetest_version` to 5.4.0
2022-11-11 23:14:08 +11:00
Jude Melton-Houghton
8c251800db Store teleport tube DB more compactly (#45) 2022-09-21 17:53:42 -04:00
unknown
b190d29d21 fix https://github.com/mt-mods/pipeworks/issues/44 2022-09-19 20:21:34 -04:00
fluxionary
e45eca76e1 check if machine is protected before trying to break/place a node (#43)
* check if target is protected before trying to break/place a node, so as not to trigger protection violations

* check whether the machine is protected, not the target
2022-09-17 19:03:55 -04:00
fluxionary
65cea1e33c check if target is protected before trying to break/place a node, so as not to trigger protection violations (#42) 2022-09-16 19:16:58 -04:00
Luke aka SwissalpS
b6c02ac8de typo fix 2022-08-16 00:30:42 +02:00
wsor4035
54243764e0 prevent tubes from connecting to the front of furnaces to match chests 2022-08-13 23:59:46 -04:00
Jude Melton-Houghton
1823690dad Store teleport tube DB in mod storage (#40)
* Store teleport tube DB in mod storage

* Prevent overwriting of tube DB backup file

* Remove backup code

* Slightly improve storage reading code

* Log DB migration to mod storage

* Add migration note in README

* Improve pipeworks.logger
2022-08-13 16:22:53 -04:00
SX
75894355d4 Use on_repair when repairing tube (#37)
* Use on_repair when repairing tube

* Use tube.on_repair

* Verify was_node registration
2022-06-23 18:06:01 +02:00
Jude Melton-Houghton
7b15bdbd1f Add configurable globalstep interval (#32)
* Add configurable globalstep interval

* Add warning for high globalstep interval

* Lower maximum globalstep interval

With no accelerator tubes, 0.8 can handle item movement speeds even
with lag.
2022-05-24 19:13:41 -04:00
OgelGames
90425fde95 fix broken shift-clicking in wielder formspec
fixes #34
2022-05-23 02:28:21 +10:00
Jude Melton-Houghton
1349ff8dd8 Return early from on_step (#33) 2022-05-15 10:58:35 -04:00
unknown
3b31b6f9c0 fix https://github.com/mt-mods/pipeworks/issues/31 2022-05-13 22:10:40 -04:00
sfence
4fce320d9c Better Hades Revisited support (#30)
* Fix spaces vs tabs in crafts.lua

* Fix recipes, add telepoter device for teleporter tube recipe.

* Override chests and furnaces in Hades as well.
2022-05-12 22:52:17 -04:00
Jude Melton-Houghton
55ded7e569 Make tubes opaque when entities are not used (#29)
* Make tubes opaque when entities are not used
2022-05-11 18:41:28 -04:00
wsor4035
e3d94cb3a8 fix digiline formspec spacing (#27) 2022-05-07 17:04:06 -04:00
Jude Melton-Houghton
97903327a5 Add option to forego real entities (#26) 2022-05-06 11:25:02 -04:00
wsor4035
37eef73695 add I3 support + update to formspecs v2 (#22)
* add fs helper inv function

* add fs helper prepends function

* add trash can, vacuum tube, teleport tube. wielders, autocrafter, mesetube, injectors

* fixed missed texture to make game agnostic

* fix luacheck
2022-04-16 14:51:59 -04:00
fluxionary
3092ce771f Merge pull request #21 from mt-mods/issue_20
override (not replace) minetest_game furnace behaviors. fixes #20
2022-04-01 20:38:39 -07:00
unknown
96ccd29dd2 address flux review 2022-04-01 20:20:48 -04:00
unknown
8e0b25fb2c fix https://github.com/mt-mods/pipeworks/issues/20 2022-04-01 19:50:21 -04:00
wsor4035
8a29f8844c fix pipeworks:mese_tube_000000 craft 2022-03-20 12:47:45 -04:00
wsor4035
5aa0501872 make pipeworks Game agnostic (#16)
* make it boot in non mtg

* make water flow

* fix water texture warnings

* fix missing steel texture issues

* remove depreciated/unnessary default function calls

* make sounds work

* fix default formspec crashes

* fix wierd tab spacing

* additionally game dig group support

* move crafts to crafts.lua

* make crafts support various games
2022-03-14 21:39:58 -04:00
59 changed files with 3765 additions and 2751 deletions

View File

@@ -1,13 +1,10 @@
name: luacheck name: luacheck
on: [push, pull_request] on: [push, pull_request]
jobs: jobs:
build: luacheck:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@master - name: Checkout
- name: apt uses: actions/checkout@master
run: sudo apt-get install -y luarocks - name: Luacheck
- name: luacheck install uses: lunarmodules/luacheck@master
run: luarocks install --local luacheck
- name: luacheck run
run: $HOME/.luarocks/bin/luacheck ./

2
.gitignore vendored
View File

@@ -1,4 +1,4 @@
## Files related to minetest development cycle ## Files related to luanti development cycle
/*.patch /*.patch
# GNU Patch reject file # GNU Patch reject file
*.rej *.rej

View File

@@ -1,6 +1,7 @@
unused_args = false unused_args = false
max_line_length= 240 max_line_length= 240
redefined = false redefined = false
std = "minetest+max"
globals = { globals = {
"pipeworks", "pipeworks",
@@ -8,17 +9,11 @@ globals = {
} }
read_globals = { read_globals = {
-- Stdlib -- remove after luacheck release: https://github.com/lunarmodules/luacheck/issues/121
string = {fields = {"split"}}, "core",
table = {fields = {"copy", "getn"}},
-- Minetest
"vector", "ItemStack",
"dump", "minetest",
"VoxelManip", "VoxelArea",
-- mods -- mods
"default", "mesecon", "digiline", "default", "mesecon", "digilines",
"screwdriver", "unified_inventory" "screwdriver", "unified_inventory",
"i3", "mcl_experience", "awards",
"xcompat", "fakelib", "vizlib"
} }

4
README
View File

@@ -20,3 +20,7 @@ This mod is a work in progress.
Please note that owing to the nature of this mod, I have opted to use 64px Please note that owing to the nature of this mod, I have opted to use 64px
textures. Anything less just looks terrible. textures. Anything less just looks terrible.
The teleport tube database used to be kept in a file named 'teleport_tubes'.
The database is now kept in mod storage. The migration from 'teleport_tubes' is
automatic. The old file is then kept around but is not used at all.

View File

@@ -1,11 +1,15 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local autocrafterCache = {} -- caches some recipe data to avoid to call the slow function minetest.get_craft_result() every second -- cache some recipe data to avoid calling the slow function
-- core.get_craft_result() every second
local autocrafterCache = {}
local craft_time = 1 local craft_time = 1
local next = next
local function count_index(invlist) local function count_index(invlist)
local index = {} local index = {}
for _, stack in pairs(invlist) do for _, stack in pairs(invlist) do
stack = ItemStack(stack)
if not stack:is_empty() then if not stack:is_empty() then
local stack_name = stack:get_name() local stack_name = stack:get_name()
index[stack_name] = (index[stack_name] or 0) + stack:get_count() index[stack_name] = (index[stack_name] or 0) + stack:get_count()
@@ -16,67 +20,249 @@ end
local function get_item_info(stack) local function get_item_info(stack)
local name = stack:get_name() local name = stack:get_name()
local def = minetest.registered_items[name] local def = core.registered_items[name]
local description = def and def.description or S("Unknown item") local description = def and def.description or S("Unknown item")
return description, name return description, name
end end
local function get_craft(pos, inventory, hash) -- Get best matching recipe for what user has put in crafting grid.
local hash = hash or minetest.hash_node_position(pos) -- This function does not consider crafting method (mix vs craft)
local craft = autocrafterCache[hash] local function get_matching_craft(output_name, example_recipe)
if not craft then local recipes = core.get_all_craft_recipes(output_name)
local recipe = inventory:get_list("recipe") if not recipes then
local output, decremented_input = minetest.get_craft_result({method = "normal", width = 3, items = recipe}) return example_recipe
craft = {recipe = recipe, consumption=count_index(recipe), output = output, decremented_input = decremented_input}
autocrafterCache[hash] = craft
end end
if 1 == #recipes then
return recipes[1].items
end
local index_example = count_index(example_recipe)
local best_score = 0
local index_recipe, best_index, score, group
for i = 1, #recipes do
score = 0
index_recipe = count_index(recipes[i].items)
for recipe_item_name, _ in pairs(index_recipe) do
if index_example[recipe_item_name] then
score = score + 1
elseif recipe_item_name:sub(1, 6) == "group:" then
group = recipe_item_name:sub(7)
for example_item_name, _ in pairs(index_example) do
if core.get_item_group(
example_item_name, group) ~= 0
then
score = score + 1
break
end
end
end
end
if best_score < score then
best_index = i
best_score = score
end
end
return best_index and recipes[best_index].items or example_recipe
end
local function get_craft(pos, inventory, hash)
local hash = hash or core.hash_node_position(pos)
local craft = autocrafterCache[hash]
if craft then return craft end
local example_recipe = inventory:get_list("recipe")
local output, decremented_input = core.get_craft_result({
method = "normal", width = 3, items = example_recipe
})
local recipe = example_recipe
if output and not output.item:is_empty() then
recipe = get_matching_craft(output.item:get_name(), example_recipe)
end
craft = {
recipe = recipe,
consumption = count_index(recipe),
output = output,
decremented_input = decremented_input.items
}
autocrafterCache[hash] = craft
return craft return craft
end end
-- From a consumption table with groups and an inventory index,
-- build a consumption table without groups
local function calculate_consumption(inv_index, consumption_with_groups)
inv_index = table.copy(inv_index)
consumption_with_groups = table.copy(consumption_with_groups)
-- table of items to actually consume
local consumption = {}
-- table of ingredients defined as one or more groups each
local grouped_ingredients = {}
-- First consume all non-group requirements
-- This is done to avoid consuming a non-group item which
-- is also in a group
for key, count in pairs(consumption_with_groups) do
if key:sub(1, 6) == "group:" then
-- build table with group recipe items while looping
grouped_ingredients[key] = key:sub(7):split(',')
else
-- if the item to consume doesn't exist in inventory
-- or not enough of them, abort crafting
if not inv_index[key] or inv_index[key] < count then
return nil
end
consumption[key] = (consumption[key] or 0) + count
consumption_with_groups[key] = consumption_with_groups[key] - count
assert(consumption_with_groups[key] == 0)
consumption_with_groups[key] = nil
inv_index[key] = inv_index[key] - count
assert(inv_index[key] >= 0)
end
end
-- helper function to resolve matching ingredients with multiple group
-- requirements
local function ingredient_groups_match_item(ingredient_groups, name)
local found = 0
local count_ingredient_groups = #ingredient_groups
for i = 1, count_ingredient_groups do
if core.get_item_group(name,
ingredient_groups[i]) ~= 0
then
found = found + 1
end
end
return found == count_ingredient_groups
end
-- Next, resolve groups using the remaining items in the inventory
if next(grouped_ingredients) ~= nil then
local take
for itemname, count in pairs(inv_index) do
if count > 0 then
-- groupname is the string as defined by recipe.
-- e.g. group:dye,color_blue
-- groups holds the group names split into a list
-- ready to be passed to core.get_item_group()
for groupname, groups in pairs(grouped_ingredients) do
if consumption_with_groups[groupname] > 0
and ingredient_groups_match_item(groups, itemname)
then
take = math.min(count,
consumption_with_groups[groupname])
consumption_with_groups[groupname] =
consumption_with_groups[groupname] - take
assert(consumption_with_groups[groupname] >= 0)
consumption[itemname] =
(consumption[itemname] or 0) + take
inv_index[itemname] =
inv_index[itemname] - take
assert(inv_index[itemname] >= 0)
end
end
end
end
end
-- Finally, check everything has been consumed
for key, count in pairs(consumption_with_groups) do
if count > 0 then
return nil
end
end
return consumption
end
local function has_room_for_output(list_output, index_output)
local name
local empty_count = 0
for _, item in pairs(list_output) do
if item:is_empty() then
empty_count = empty_count + 1
else
name = item:get_name()
if index_output[name] then
index_output[name] = index_output[name] - item:get_free_space()
end
end
end
for _, count in pairs(index_output) do
if count > 0 then
empty_count = empty_count - 1
end
end
if empty_count < 0 then
return false
end
return true
end
local function autocraft(inventory, craft) local function autocraft(inventory, craft)
if not craft then return false end if not craft then return false end
local output_item = craft.output.item
-- check if we have enough room in dst -- check if output and all replacements fit in dst
if not inventory:room_for_item("dst", output_item) then return false end local output = craft.output.item
local consumption = craft.consumption local out_items = count_index(craft.decremented_input)
local inv_index = count_index(inventory:get_list("src")) out_items[output:get_name()] =
-- check if we have enough material available (out_items[output:get_name()] or 0) + output:get_count()
for itemname, number in pairs(consumption) do
if (not inv_index[itemname]) or inv_index[itemname] < number then return false end if not has_room_for_output(inventory:get_list("dst"), out_items) then
return false
end end
-- check if we have enough material available
local inv_index = count_index(inventory:get_list("src"))
local consumption = calculate_consumption(inv_index, craft.consumption)
if not consumption then
return false
end
-- consume material -- consume material
for itemname, number in pairs(consumption) do for itemname, number in pairs(consumption) do
for _ = 1, number do -- We have to do that since remove_item does not work if count > stack_max -- We have to do that since remove_item does not work if count > stack_max
for _ = 1, number do
inventory:remove_item("src", ItemStack(itemname)) inventory:remove_item("src", ItemStack(itemname))
end end
end end
-- craft the result into the dst inventory and add any "replacements" as well -- craft the result into the dst inventory and add any "replacements" as well
inventory:add_item("dst", output_item) inventory:add_item("dst", output)
local leftover
for i = 1, 9 do for i = 1, 9 do
inventory:add_item("dst", craft.decremented_input.items[i]) leftover = inventory:add_item("dst", craft.decremented_input[i])
if leftover and not leftover:is_empty() then
core.log("warning", "[pipeworks] autocrafter didn't " ..
"calculate output space correctly.")
end
end end
return true return true
end end
-- returns false to stop the timer, true to continue running -- returns false to stop the timer, true to continue running
-- is started only from start_autocrafter(pos) after sanity checks and cached recipe -- is started only from start_autocrafter(pos) after sanity checks and
-- recipe is cached
local function run_autocrafter(pos, elapsed) local function run_autocrafter(pos, elapsed)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inventory = meta:get_inventory() local inventory = meta:get_inventory()
local craft = get_craft(pos, inventory) local craft = get_craft(pos, inventory)
local output_item = craft.output.item local output_item = craft.output.item
-- NALC: existence de limitgroup ?
local limitcraft = minetest.get_item_group(output_item:get_name(), "limitcraft") or 0
-- only use crafts that have an actual result -- only use crafts that have an actual result
-- NALC: ou si l'item n'est pas dans le group limitcraft if output_item:is_empty() then
if output_item:is_empty() or limitcraft > 0 then
meta:set_string("infotext", S("unconfigured Autocrafter: unknown recipe")) meta:set_string("infotext", S("unconfigured Autocrafter: unknown recipe"))
return false return false
end end
for _ = 1, math.floor(elapsed/craft_time) do for _ = 1, math.floor(elapsed / craft_time) do
local continue = autocraft(inventory, craft) local continue = autocraft(inventory, craft)
if not continue then return false end if not continue then return false end
end end
@@ -84,9 +270,9 @@ local function run_autocrafter(pos, elapsed)
end end
local function start_crafter(pos) local function start_crafter(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
if meta:get_int("enabled") == 1 then if meta:get_int("enabled") == 1 then
local timer = minetest.get_node_timer(pos) local timer = core.get_node_timer(pos)
if not timer:is_started() then if not timer:is_started() then
timer:start(craft_time) timer:start(craft_time)
end end
@@ -100,33 +286,17 @@ end
-- note, that this function assumes allready being updated to virtual items -- note, that this function assumes allready being updated to virtual items
-- and doesn't handle recipes with stacksizes > 1 -- and doesn't handle recipes with stacksizes > 1
local function after_recipe_change(pos, inventory) local function after_recipe_change(pos, inventory)
local meta = minetest.get_meta(pos) local hash = core.hash_node_position(pos)
local meta = core.get_meta(pos)
autocrafterCache[hash] = nil
-- if we emptied the grid, there's no point in keeping it running or cached -- if we emptied the grid, there's no point in keeping it running or cached
if inventory:is_empty("recipe") then if inventory:is_empty("recipe") then
minetest.get_node_timer(pos):stop() core.get_node_timer(pos):stop()
autocrafterCache[minetest.hash_node_position(pos)] = nil
meta:set_string("infotext", S("unconfigured Autocrafter")) meta:set_string("infotext", S("unconfigured Autocrafter"))
inventory:set_stack("output", 1, "") inventory:set_stack("output", 1, "")
return return
end end
local recipe = inventory:get_list("recipe") local craft = get_craft(pos, inventory, hash)
local hash = minetest.hash_node_position(pos)
local craft = autocrafterCache[hash]
if craft then
-- check if it changed
local cached_recipe = craft.recipe
for i = 1, 9 do
if recipe[i]:get_name() ~= cached_recipe[i]:get_name() then
autocrafterCache[hash] = nil -- invalidate recipe
craft = nil
break
end
end
end
craft = craft or get_craft(pos, inventory, hash)
local output_item = craft.output.item local output_item = craft.output.item
local description, name = get_item_info(output_item) local description, name = get_item_info(output_item)
meta:set_string("infotext", S("'@1' Autocrafter (@2)", description, name)) meta:set_string("infotext", S("'@1' Autocrafter (@2)", description, name))
@@ -135,13 +305,14 @@ local function after_recipe_change(pos, inventory)
after_inventory_change(pos) after_inventory_change(pos)
end end
-- clean out unknown items and groups, which would be handled like unknown items in the crafting grid -- clean out unknown items and groups, which would be handled like unknown
-- if minetest supports query by group one day, this might replace them -- items in the crafting grid
-- if Luanti supports query by group one day, this might replace them
-- with a canonical version instead -- with a canonical version instead
local function normalize(item_list) local function normalize(item_list)
for i = 1, #item_list do for i = 1, #item_list do
local name = item_list[i] local name = item_list[i]
if not minetest.registered_items[name] then if not core.registered_items[name] then
item_list[i] = "" item_list[i] = ""
end end
end end
@@ -153,7 +324,7 @@ local function on_output_change(pos, inventory, stack)
inventory:set_list("output", {}) inventory:set_list("output", {})
inventory:set_list("recipe", {}) inventory:set_list("recipe", {})
else else
local input = minetest.get_craft_recipe(stack:get_name()) local input = core.get_craft_recipe(stack:get_name())
if not input.items or input.type ~= "normal" then return end if not input.items or input.type ~= "normal" then return end
local items, width = normalize(input.items), input.width local items, width = normalize(input.items), input.width
local item_idx, width_idx = 1, 1 local item_idx, width_idx = 1, 1
@@ -166,38 +337,65 @@ local function on_output_change(pos, inventory, stack)
end end
width_idx = (width_idx < 3) and (width_idx + 1) or 1 width_idx = (width_idx < 3) and (width_idx + 1) or 1
end end
-- we'll set the output slot in after_recipe_change to the actual result of the new recipe -- we'll set the output slot in after_recipe_change to the actual
-- result of the new recipe
end end
after_recipe_change(pos, inventory) after_recipe_change(pos, inventory)
end end
-- returns false if we shouldn't bother attempting to start the timer again after this -- returns false if we shouldn't bother attempting to start the timer again
-- after this
local function update_meta(meta, enabled) local function update_meta(meta, enabled)
local state = enabled and "on" or "off" local state = enabled and "on" or "off"
meta:set_int("enabled", enabled and 1 or 0) meta:set_int("enabled", enabled and 1 or 0)
local fs = "size[8,12]".. local list_backgrounds = ""
"list[context;recipe;0,0;3,3;]".. if core.get_modpath("i3") or core.get_modpath("mcl_formspec") then
"image[3,1;1,1;gui_hb_bg.png^[colorize:#141318:255]".. list_backgrounds = "style_type[box;colors=#666]"
"list[context;output;3,1;1,1;]".. for i = 0, 2 do
"image_button[3,2;1,0.6;pipeworks_button_" .. state .. ".png;" .. state .. ";;;false;pipeworks_button_interm.png]" .. for j = 0, 2 do
"list[context;src;0,4.5;8,3;]".. list_backgrounds = list_backgrounds .. "box[" ..
"list[context;dst;4,0;4,3;]".. 0.22 + (i * 1.25) .. "," .. 0.22 + (j * 1.25) .. ";1,1;]"
default.gui_bg.. end
default.gui_bg_img.. end
default.gui_slots.. for i = 0, 3 do
default.get_hotbar_bg(0,8) .. for j = 0, 2 do
"list[current_player;main;0,8;8,4;]" .. list_backgrounds = list_backgrounds .. "box[" ..
"listring[current_player;main]".. 5.28 + (i * 1.25) .. "," .. 0.22 + (j * 1.25) .. ";1,1;]"
"listring[context;src]" .. end
"listring[current_player;main]".. end
"listring[context;dst]" .. for i = 0, 7 do
"listring[current_player;main]" for j = 0, 2 do
if minetest.get_modpath("digilines") then list_backgrounds = list_backgrounds .. "box[" ..
fs = fs.."field[0.3,3.5;4.5,1;channel;"..S("Channel")..";${channel}]".. 0.22 + (i * 1.25) .. "," .. 5 + (j * 1.25) .. ";1,1;]"
"button[4.5,3.2;1.5,1;set_channel;"..S("Set").."]".. end
"button_exit[6,3.2;2,1;close;"..S("Close").."]" end
end end
meta:set_string("formspec",fs) local size = "10.2,14"
local fs =
"formspec_version[2]" ..
"size[" .. size .. "]" ..
pipeworks.fs_helpers.get_prepends(size) ..
list_backgrounds ..
"list[context;recipe;0.22,0.22;3,3;]" ..
"image[4,1.45;1,1;[combine:16x16^[noalpha^[colorize:#141318:255]" ..
"list[context;output;4,1.45;1,1;]" ..
"image_button[4,2.6;1,0.6;pipeworks_button_" .. state .. ".png;" ..
state .. ";;;false;pipeworks_button_interm.png]" ..
"list[context;dst;5.28,0.22;4,3;]" ..
"list[context;src;0.22,5;8,3;]" ..
pipeworks.fs_helpers.get_inv(9) ..
"listring[current_player;main]" ..
"listring[context;src]" ..
"listring[current_player;main]" ..
"listring[context;dst]" ..
"listring[current_player;main]"
if core.get_modpath("digilines") then
fs = fs .. "field[0.22,4.1;4.5,0.75;channel;" .. S("Channel") ..
";${channel}]" ..
"button[5,4.1;1.5,0.75;set_channel;" .. S("Set") .. "]" ..
"button_exit[6.8,4.1;2,0.75;close;" .. S("Close") .. "]"
end
meta:set_string("formspec", fs)
-- toggling the button doesn't quite call for running a recipe change check -- toggling the button doesn't quite call for running a recipe change check
-- so instead we run a minimal version for infotext setting only -- so instead we run a minimal version for infotext setting only
@@ -217,11 +415,14 @@ local function update_meta(meta, enabled)
end end
-- 1st version of the autocrafter had actual items in the crafting grid -- 1st version of the autocrafter had actual items in the crafting grid
-- the 2nd replaced these with virtual items, dropped the content on update and set "virtual_items" to string "1" -- the 2nd replaced these with virtual items, dropped the content on update and
-- the third added an output inventory, changed the formspec and added a button for enabling/disabling -- set "virtual_items" to string "1"
-- so we work out way backwards on this history and update each single case to the newest version -- the third added an output inventory, changed the formspec and added a button
-- for enabling/disabling
-- so we work out way backwards on this history and update each single case
-- to the newest version
local function upgrade_autocrafter(pos, meta) local function upgrade_autocrafter(pos, meta)
local meta = meta or minetest.get_meta(pos) local meta = meta or core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
if inv:get_size("output") == 0 then -- we are version 2 or 1 if inv:get_size("output") == 0 then -- we are version 2 or 1
@@ -230,14 +431,15 @@ local function upgrade_autocrafter(pos, meta)
update_meta(meta, true) update_meta(meta, true)
if meta:get_string("virtual_items") == "1" then -- we are version 2 if meta:get_string("virtual_items") == "1" then -- we are version 2
-- we already dropped stuff, so lets remove the metadatasetting (we are not being called again for this node) -- we already dropped stuff, so lets remove the metadatasetting
-- (we are not being called again for this node)
meta:set_string("virtual_items", "") meta:set_string("virtual_items", "")
else -- we are version 1 else -- we are version 1
local recipe = inv:get_list("recipe") local recipe = inv:get_list("recipe")
if not recipe then return end if not recipe then return end
for idx, stack in ipairs(recipe) do for idx, stack in ipairs(recipe) do
if not stack:is_empty() then if not stack:is_empty() then
minetest.add_item(pos, stack) core.add_item(pos, stack)
stack:set_count(1) stack:set_count(1)
stack:set_wear(0) stack:set_wear(0)
inv:set_stack("recipe", idx, stack) inv:set_stack("recipe", idx, stack)
@@ -246,47 +448,54 @@ local function upgrade_autocrafter(pos, meta)
end end
-- update the recipe, cache, and start the crafter -- update the recipe, cache, and start the crafter
autocrafterCache[minetest.hash_node_position(pos)] = nil autocrafterCache[core.hash_node_position(pos)] = nil
after_recipe_change(pos, inv) after_recipe_change(pos, inv)
end end
end end
minetest.register_node("pipeworks:autocrafter", { core.register_node("pipeworks:autocrafter", {
description = S("Autocrafter"), description = S("Autocrafter"),
drawtype = "normal", drawtype = "normal",
tiles = {"pipeworks_autocrafter.png"}, tiles = {"pipeworks_autocrafter.png"},
groups = {snappy = 3, tubedevice = 1, tubedevice_receiver = 1}, groups = {snappy = 3, tubedevice = 1, tubedevice_receiver = 1, dig_generic = 1, axey=1, handy=1, pickaxey=1},
is_ground_content = false,
_mcl_hardness=0.8,
tube = {insert_object = function(pos, node, stack, direction) tube = {insert_object = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
local added = inv:add_item("src", stack) local added = inv:add_item("src", stack)
after_inventory_change(pos) after_inventory_change(pos)
return added return added
end, end,
can_insert = function(pos, node, stack, direction) can_insert = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
return inv:room_for_item("src", stack) return inv:room_for_item("src", stack)
end, end,
input_inventory = "dst", input_inventory = "dst",
connect_sides = {left = 1, right = 1, front = 1, back = 1, top = 1, bottom = 1}}, connect_sides = {
left = 1, right = 1, front = 1, back = 1, top = 1, bottom = 1
}
},
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
inv:set_size("src", 3*8) inv:set_size("src", 3 * 8)
inv:set_size("recipe", 3*3) inv:set_size("recipe", 3 * 3)
inv:set_size("dst", 4*3) inv:set_size("dst", 4 * 3)
inv:set_size("output", 1) inv:set_size("output", 1)
update_meta(meta, false) update_meta(meta, false)
end, end,
on_receive_fields = function(pos, formname, fields, sender) on_receive_fields = function(pos, formname, fields, sender)
if (fields.quit and not fields.key_enter_field) or not pipeworks.may_configure(pos, sender) then if (fields.quit and not fields.key_enter_field)
or not pipeworks.may_configure(pos, sender)
then
return return
end end
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
if fields.on then if fields.on then
update_meta(meta, false) update_meta(meta, false)
minetest.get_node_timer(pos):stop() core.get_node_timer(pos):stop()
elseif fields.off then elseif fields.off then
if update_meta(meta, true) then if update_meta(meta, true) then
start_crafter(pos) start_crafter(pos)
@@ -298,7 +507,7 @@ minetest.register_node("pipeworks:autocrafter", {
end, end,
can_dig = function(pos, player) can_dig = function(pos, player)
upgrade_autocrafter(pos) upgrade_autocrafter(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
return (inv:is_empty("src") and inv:is_empty("dst")) return (inv:is_empty("src") and inv:is_empty("dst"))
end, end,
@@ -307,12 +516,12 @@ minetest.register_node("pipeworks:autocrafter", {
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end, end,
on_destruct = function(pos) on_destruct = function(pos)
autocrafterCache[minetest.hash_node_position(pos)] = nil autocrafterCache[core.hash_node_position(pos)] = nil
end, end,
allow_metadata_inventory_put = function(pos, listname, index, stack, player) allow_metadata_inventory_put = function(pos, listname, index, stack, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
upgrade_autocrafter(pos) upgrade_autocrafter(pos)
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
if listname == "recipe" then if listname == "recipe" then
stack:set_count(1) stack:set_count(1)
inv:set_stack(listname, index, stack) inv:set_stack(listname, index, stack)
@@ -327,11 +536,13 @@ minetest.register_node("pipeworks:autocrafter", {
end, end,
allow_metadata_inventory_take = function(pos, listname, index, stack, player) allow_metadata_inventory_take = function(pos, listname, index, stack, player)
if not pipeworks.may_configure(pos, player) then if not pipeworks.may_configure(pos, player) then
minetest.log("action", string.format("%s attempted to take from autocrafter at %s", player:get_player_name(), minetest.pos_to_string(pos))) core.log("action", string.format("%s attempted to take from " ..
"autocrafter at %s",
player:get_player_name(), core.pos_to_string(pos)))
return 0 return 0
end end
upgrade_autocrafter(pos) upgrade_autocrafter(pos)
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
if listname == "recipe" then if listname == "recipe" then
inv:set_stack(listname, index, ItemStack("")) inv:set_stack(listname, index, ItemStack(""))
after_recipe_change(pos, inv) after_recipe_change(pos, inv)
@@ -343,10 +554,12 @@ minetest.register_node("pipeworks:autocrafter", {
after_inventory_change(pos) after_inventory_change(pos)
return stack:get_count() return stack:get_count()
end, end,
allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player) allow_metadata_inventory_move = function(
pos, from_list, from_index, to_list, to_index, count, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
upgrade_autocrafter(pos) upgrade_autocrafter(pos)
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
local stack = inv:get_stack(from_list, from_index) local stack = inv:get_stack(from_list, from_index)
if to_list == "output" then if to_list == "output" then
@@ -375,41 +588,44 @@ minetest.register_node("pipeworks:autocrafter", {
return count return count
end, end,
on_timer = run_autocrafter, on_timer = run_autocrafter,
digiline = { digilines = {
receptor = {}, receptor = {},
effector = { effector = {
action = function(pos,node,channel,msg) action = function(pos,node,channel,msg)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
if channel ~= meta:get_string("channel") then return end if channel ~= meta:get_string("channel") then return end
if type(msg) == "table" then if type(msg) == "table" then
if #msg < 3 then return end if #msg < 3 then return end
local inv = meta:get_inventory() local inv = meta:get_inventory()
for y=0,2,1 do for y = 0, 2, 1 do
for x=1,3,1 do local row = msg[y + 1]
local slot = y*3+x for x = 1, 3, 1 do
if minetest.registered_items[msg[y+1][x]] then local slot = y * 3 + x
inv:set_stack("recipe",slot,ItemStack(msg[y+1][x])) if type(row) == "table" and core.registered_items[row[x]] then
inv:set_stack("recipe", slot, ItemStack(
row[x]))
else else
inv:set_stack("recipe",slot,ItemStack("")) inv:set_stack("recipe", slot, ItemStack(""))
end end
end end
end end
after_recipe_change(pos,inv) after_recipe_change(pos,inv)
elseif msg == "get_recipe" then elseif msg == "get_recipe" then
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
local recipe = {} local recipe = {}
for y=0,2,1 do for y = 0, 2, 1 do
local row = {} local row = {}
for x=1,3,1 do for x = 1, 3, 1 do
local slot = y*3+x local slot = y * 3 + x
table.insert(row, inv:get_stack("recipe",slot):get_name()) table.insert(row, inv:get_stack(
"recipe", slot):get_name())
end end
table.insert(recipe, row) table.insert(recipe, row)
end end
local setchan = meta:get_string("channel") local setchan = meta:get_string("channel")
local output = inv:get_stack("output", 1) local output = inv:get_stack("output", 1)
digiline:receptor_send(pos, digiline.rules.default, setchan, { digilines.receptor_send(pos, digilines.rules.default, setchan, {
recipe = recipe, recipe = recipe,
result = { result = {
name = output:get_name(), name = output:get_name(),
@@ -418,7 +634,7 @@ minetest.register_node("pipeworks:autocrafter", {
}) })
elseif msg == "off" then elseif msg == "off" then
update_meta(meta, false) update_meta(meta, false)
minetest.get_node_timer(pos):stop() core.get_node_timer(pos):stop()
elseif msg == "on" then elseif msg == "on" then
if update_meta(meta, true) then if update_meta(meta, true) then
start_crafter(pos) start_crafter(pos)
@@ -430,13 +646,4 @@ minetest.register_node("pipeworks:autocrafter", {
}, },
}, },
}) })
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:autocrafter" pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list + 1] = "pipeworks:autocrafter"
minetest.register_craft( {
output = "pipeworks:autocrafter 2",
recipe = {
{ "default:steel_ingot", "default:mese_crystal", "default:steel_ingot" },
{ "basic_materials:plastic_sheet", "default:steel_ingot", "basic_materials:plastic_sheet" },
{ "default:steel_ingot", "default:mese_crystal", "default:steel_ingot" }
},
})

View File

@@ -1,7 +1,7 @@
-- enable finite liquid in the presence of dynamic liquid to preserve water volume. -- enable finite liquid in the presence of dynamic liquid to preserve water volume.
local enable = false local enable = false
if minetest.get_modpath("dynamic_liquid") then if core.get_modpath("dynamic_liquid") then
pipeworks.logger("detected mod dynamic_liquid, enabling finite liquid flag") pipeworks.logger("detected mod dynamic_liquid, enabling finite liquid flag")
enable = true enable = true
end end

View File

@@ -31,32 +31,32 @@ function pipeworks.get_axis_dir(nodetable, pattern)
local pxm,pxp,pym,pyp,pzm,pzp local pxm,pxp,pym,pyp,pzm,pzp
if string.find(nodetable.nxm.name, pattern) if string.find(nodetable.nxm.name, pattern)
and minetest.facedir_to_dir(nodetable.nxm.param2).x ~= 0 then and core.facedir_to_dir(nodetable.nxm.param2).x ~= 0 then
pxm=1 pxm=1
end end
if string.find(nodetable.nxp.name, pattern) if string.find(nodetable.nxp.name, pattern)
and minetest.facedir_to_dir(nodetable.nxp.param2).x ~= 0 then and core.facedir_to_dir(nodetable.nxp.param2).x ~= 0 then
pxp=1 pxp=1
end end
if string.find(nodetable.nzm.name, pattern) if string.find(nodetable.nzm.name, pattern)
and minetest.facedir_to_dir(nodetable.nzm.param2).z ~= 0 then and core.facedir_to_dir(nodetable.nzm.param2).z ~= 0 then
pzm=1 pzm=1
end end
if string.find(nodetable.nzp.name, pattern) if string.find(nodetable.nzp.name, pattern)
and minetest.facedir_to_dir(nodetable.nzp.param2).z ~= 0 then and core.facedir_to_dir(nodetable.nzp.param2).z ~= 0 then
pzp=1 pzp=1
end end
if string.find(nodetable.nym.name, pattern) if string.find(nodetable.nym.name, pattern)
and minetest.facedir_to_dir(nodetable.nym.param2).y ~= 0 then and core.facedir_to_dir(nodetable.nym.param2).y ~= 0 then
pym=1 pym=1
end end
if string.find(nodetable.nyp.name, pattern) if string.find(nodetable.nyp.name, pattern)
and minetest.facedir_to_dir(nodetable.nyp.param2).y ~= 0 then and core.facedir_to_dir(nodetable.nyp.param2).y ~= 0 then
pyp=1 pyp=1
end end
local match = pxm or pxp or pym or pyp or pzm or pzp local match = pxm or pxp or pym or pyp or pzm or pzp
@@ -67,14 +67,14 @@ local tube_table = {[0] = 1, 2, 2, 4, 2, 4, 4, 5, 2, 3, 4, 6, 4, 6, 5, 7, 2, 4,
local tube_table_facedirs = {[0] = 0, 0, 5, 0, 3, 4, 3, 0, 2, 0, 2, 0, 6, 4, 3, 0, 7, 12, 5, 12, 7, 4, 5, 5, 18, 20, 16, 0, 7, 4, 7, 0, 1, 8, 1, 1, 1, 13, 1, 1, 10, 8, 2, 2, 17, 4, 3, 6, 9, 9, 9, 9, 21, 13, 1, 1, 10, 10, 11, 2, 19, 4, 3, 0} local tube_table_facedirs = {[0] = 0, 0, 5, 0, 3, 4, 3, 0, 2, 0, 2, 0, 6, 4, 3, 0, 7, 12, 5, 12, 7, 4, 5, 5, 18, 20, 16, 0, 7, 4, 7, 0, 1, 8, 1, 1, 1, 13, 1, 1, 10, 8, 2, 2, 17, 4, 3, 6, 9, 9, 9, 9, 21, 13, 1, 1, 10, 10, 11, 2, 19, 4, 3, 0}
local function autoroute_pipes(pos) local function autoroute_pipes(pos)
local nctr = minetest.get_node(pos) local nctr = core.get_node(pos)
local state = "_empty" local state = "_empty"
if (string.find(nctr.name, "pipeworks:pipe_") == nil) then return end if (string.find(nctr.name, "pipeworks:pipe_") == nil) then return end
if (string.find(nctr.name, "_loaded") ~= nil) then state = "_loaded" end if (string.find(nctr.name, "_loaded") ~= nil) then state = "_loaded" end
local nsurround = pipeworks.scan_pipe_surroundings(pos) local nsurround = pipeworks.scan_pipe_surroundings(pos)
if nsurround == 0 then nsurround = 9 end if nsurround == 0 then nsurround = 9 end
minetest.swap_node(pos, {name = "pipeworks:pipe_"..tube_table[nsurround]..state, core.swap_node(pos, {name = "pipeworks:pipe_"..tube_table[nsurround]..state,
param2 = tube_table_facedirs[nsurround]}) param2 = tube_table_facedirs[nsurround]})
end end
@@ -98,12 +98,12 @@ function pipeworks.scan_pipe_surroundings(pos)
local pzm=0 local pzm=0
local pzp=0 local pzp=0
local nxm = minetest.get_node({ x=pos.x-1, y=pos.y , z=pos.z }) local nxm = core.get_node({ x=pos.x-1, y=pos.y , z=pos.z })
local nxp = minetest.get_node({ x=pos.x+1, y=pos.y , z=pos.z }) local nxp = core.get_node({ x=pos.x+1, y=pos.y , z=pos.z })
local nym = minetest.get_node({ x=pos.x , y=pos.y-1, z=pos.z }) local nym = core.get_node({ x=pos.x , y=pos.y-1, z=pos.z })
local nyp = minetest.get_node({ x=pos.x , y=pos.y+1, z=pos.z }) local nyp = core.get_node({ x=pos.x , y=pos.y+1, z=pos.z })
local nzm = minetest.get_node({ x=pos.x , y=pos.y , z=pos.z-1 }) local nzm = core.get_node({ x=pos.x , y=pos.y , z=pos.z-1 })
local nzp = minetest.get_node({ x=pos.x , y=pos.y , z=pos.z+1 }) local nzp = core.get_node({ x=pos.x , y=pos.y , z=pos.z+1 })
local nodetable = { local nodetable = {
nxm = nxm, nxm = nxm,
@@ -173,12 +173,12 @@ function pipeworks.scan_pipe_surroundings(pos)
-- ... other nodes -- ... other nodes
local def_left = minetest.registered_nodes[nxp.name] -- the node that {pos} is to the left of (not the local def_left = core.registered_nodes[nxp.name] -- the node that {pos} is to the left of (not the
local def_right = minetest.registered_nodes[nxm.name] -- ...note that is AT the left!), etc. local def_right = core.registered_nodes[nxm.name] -- ...note that is AT the left!), etc.
local def_bottom = minetest.registered_nodes[nyp.name] local def_bottom = core.registered_nodes[nyp.name]
local def_top = minetest.registered_nodes[nym.name] local def_top = core.registered_nodes[nym.name]
local def_front = minetest.registered_nodes[nzp.name] local def_front = core.registered_nodes[nzp.name]
local def_back = minetest.registered_nodes[nzm.name] local def_back = core.registered_nodes[nzm.name]
if def_left and def_left.pipe_connections and def_left.pipe_connections.left if def_left and def_left.pipe_connections and def_left.pipe_connections.left
and (not def_left.pipe_connections.pattern or string.find(nxp.name, def_left.pipe_connections.pattern)) and (not def_left.pipe_connections.pattern or string.find(nxp.name, def_left.pipe_connections.pattern))
@@ -211,16 +211,16 @@ function pipeworks.scan_pipe_surroundings(pos)
pzm = 1 pzm = 1
end end
print("stage 2 returns "..pxm+8*pxp+2*pym+16*pyp+4*pzm+32*pzp.. core.log("info", "stage 2 returns "..pxm+8*pxp+2*pym+16*pyp+4*pzm+32*pzp..
" for nodes surrounding "..minetest.get_node(pos).name.." at "..minetest.pos_to_string(pos)) " for nodes surrounding "..core.get_node(pos).name.." at "..core.pos_to_string(pos))
return pxm+8*pxp+2*pym+16*pyp+4*pzm+32*pzp return pxm+8*pxp+2*pym+16*pyp+4*pzm+32*pzp
end end
function pipeworks.look_for_stackable_tanks(pos) function pipeworks.look_for_stackable_tanks(pos)
local tym = minetest.get_node({ x=pos.x , y=pos.y-1, z=pos.z }) local tym = core.get_node({ x=pos.x , y=pos.y-1, z=pos.z })
if string.find(tym.name, "pipeworks:storage_tank_") ~= nil or if string.find(tym.name, "pipeworks:storage_tank_") ~= nil or
string.find(tym.name, "pipeworks:expansion_tank_") ~= nil then string.find(tym.name, "pipeworks:expansion_tank_") ~= nil then
minetest.add_node(pos, { name = "pipeworks:expansion_tank_0", param2 = tym.param2}) core.add_node(pos, { name = "pipeworks:expansion_tank_0", param2 = tym.param2})
end end
end end

View File

@@ -10,8 +10,8 @@ local function nodeside(node, tubedir)
node.param2 = 0 node.param2 = 0
end end
local backdir = minetest.facedir_to_dir(node.param2) local backdir = core.facedir_to_dir(node.param2)
local back = pipeworks.vector_dot(backdir, tubedir) local back = vector.dot(backdir, tubedir)
if back == 1 then if back == 1 then
return "back" return "back"
elseif back == -1 then elseif back == -1 then
@@ -19,7 +19,7 @@ local function nodeside(node, tubedir)
end end
local topdir = pipeworks.facedir_to_top_dir(node.param2) local topdir = pipeworks.facedir_to_top_dir(node.param2)
local top = pipeworks.vector_dot(topdir, tubedir) local top = vector.dot(topdir, tubedir)
if top == 1 then if top == 1 then
return "top" return "top"
elseif top == -1 then elseif top == -1 then
@@ -27,7 +27,7 @@ local function nodeside(node, tubedir)
end end
local rightdir = pipeworks.facedir_to_right_dir(node.param2) local rightdir = pipeworks.facedir_to_right_dir(node.param2)
local right = pipeworks.vector_dot(rightdir, tubedir) local right = vector.dot(rightdir, tubedir)
if right == 1 then if right == 1 then
return "right" return "right"
else else
@@ -40,7 +40,7 @@ local tube_table = {[0] = 1, 2, 2, 4, 2, 4, 4, 5, 2, 3, 4, 6, 4, 6, 5, 7, 2, 4,
local tube_table_facedirs = {[0] = 0, 0, 5, 0, 3, 4, 3, 0, 2, 0, 2, 0, 6, 4, 3, 0, 7, 12, 5, 12, 7, 4, 5, 5, 18, 20, 16, 0, 7, 4, 7, 0, 1, 8, 1, 1, 1, 13, 1, 1, 10, 8, 2, 2, 17, 4, 3, 6, 9, 9, 9, 9, 21, 13, 1, 1, 10, 10, 11, 2, 19, 4, 3, 0} local tube_table_facedirs = {[0] = 0, 0, 5, 0, 3, 4, 3, 0, 2, 0, 2, 0, 6, 4, 3, 0, 7, 12, 5, 12, 7, 4, 5, 5, 18, 20, 16, 0, 7, 4, 7, 0, 1, 8, 1, 1, 1, 13, 1, 1, 10, 8, 2, 2, 17, 4, 3, 6, 9, 9, 9, 9, 21, 13, 1, 1, 10, 10, 11, 2, 19, 4, 3, 0}
local function tube_autoroute(pos) local function tube_autoroute(pos)
local active = {0, 0, 0, 0, 0, 0} local active = {0, 0, 0, 0, 0, 0}
local nctr = minetest.get_node(pos) local nctr = core.get_node(pos)
if not is_tube(nctr.name) then return end if not is_tube(nctr.name) then return end
local adjustments = { local adjustments = {
@@ -57,9 +57,9 @@ local function tube_autoroute(pos)
for i, adj in ipairs(adjustments) do for i, adj in ipairs(adjustments) do
local position = vector.add(pos, adj) local position = vector.add(pos, adj)
local node = minetest.get_node(position) local node = core.get_node(position)
local idef = minetest.registered_nodes[node.name] local idef = core.registered_nodes[node.name]
-- handle the tubes themselves -- handle the tubes themselves
if is_tube(node.name) then if is_tube(node.name) then
active[i] = 1 active[i] = 1
@@ -73,11 +73,11 @@ local function tube_autoroute(pos)
end end
end end
minetest.get_meta(pos):set_string("adjlist", minetest.serialize(adjlist)) core.get_meta(pos):set_string("adjlist", core.serialize(adjlist))
-- all sides checked, now figure which tube to use. -- all sides checked, now figure which tube to use.
local nodedef = minetest.registered_nodes[nctr.name] local nodedef = core.registered_nodes[nctr.name]
local basename = nodedef.basename local basename = nodedef.basename
if nodedef.style == "old" then if nodedef.style == "old" then
local nsurround = "" local nsurround = ""
@@ -95,7 +95,7 @@ local function tube_autoroute(pos)
nctr.name = basename.."_"..tube_table[s] nctr.name = basename.."_"..tube_table[s]
nctr.param2 = tube_table_facedirs[s] nctr.param2 = tube_table_facedirs[s]
end end
minetest.swap_node(pos, nctr) core.swap_node(pos, nctr)
end end
function pipeworks.scan_for_tube_objects(pos) function pipeworks.scan_for_tube_objects(pos)
@@ -123,12 +123,12 @@ end
-- when they are updated. -- when they are updated.
function pipeworks.on_rotate(pos, node, user, mode, new_param2) function pipeworks.on_rotate(pos, node, user, mode, new_param2)
node.param2 = new_param2 node.param2 = new_param2
minetest.swap_node(pos, node) core.swap_node(pos, node)
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
return true return true
end end
if minetest.get_modpath("mesecons_mvps") then if core.get_modpath("mesecons_mvps") then
mesecon.register_on_mvps_move(function(moved_nodes) mesecon.register_on_mvps_move(function(moved_nodes)
for _, n in ipairs(moved_nodes) do for _, n in ipairs(moved_nodes) do
pipeworks.scan_for_tube_objects(n.pos) pipeworks.scan_for_tube_objects(n.pos)

View File

@@ -3,6 +3,65 @@ Changelog
2024-02-26 (SwissalpS)
set is_ground_content to false for various nodes.
2023-06-22 (SwissalpS, rubenwardy)
groups support in recipe. Set recipe as usual via recipe formspec or digilines.
Autocrafter now resolves matching recipe using groups so that items in input
inventory are used, that match group and no longer only strictly what user
has in recipe-inventory
2023-06-21 (OgelGames, BuckarooBanzay)
fix autocrafter destroying replacement items (OG)
remove facedir debugging logs (BB)
2023-05-28 (SwissalpS)
support setting 'can_receive' for teleport tubes via digiline
2023-05-19 (fluxionary, OgelGames)
log items going through teleport tubes
2022-12-02 (wsor4035)
bring back compatibility with mineclone2
2022-11-11 (OgelGames)
prevent tp- and sand-tubes from breaking themselves and refactor part of code
2022-09-18 (fluxionary)
protection checks before break-/placeing a node -> less violation logs
2022-08-14 (wsor4035)
prevent tubes from connecting to furnace front
2022-08-13 (TurkeyMcMac)
moved teleport-tube database to mod-storage
2022-06-23 (S-S-X)
on_repair tweak, improved repairing tubes
* many updates not mentioned here *
2017-10-19 (thetaepsilon) 2017-10-19 (thetaepsilon)
Directional flowables are now implemented. Directional flowables are now implemented.
All devices for which it is relevant (valve, flow sensor etc.) have been converted so that they only flow on their connecting sides, so pressure propogation now works as expected for these devices when pressure logic is enabled. All devices for which it is relevant (valve, flow sensor etc.) have been converted so that they only flow on their connecting sides, so pressure propogation now works as expected for these devices when pressure logic is enabled.

90
chests.lua Normal file
View File

@@ -0,0 +1,90 @@
pipeworks.chests = {}
-- register a chest to connect with pipeworks tubes.
-- will autoconnect to tubes and add tube inlets to the textures
-- it is highly recommended to allow the user to change the "splitstacks" int (1 to enable) in the node meta
-- but that can't be done by this function
-- @param override: additional overrides, such as stuff to modify the node formspec
-- @param connect_sides: which directions the chests shall connect to
function pipeworks.override_chest(chestname, override, connect_sides)
local old_def = core.registered_nodes[chestname]
local tube_entry = "^pipeworks_tube_connection_wooden.png"
override.tiles = override.tiles or old_def.tiles
-- expand the tiles table if it has been shortened
if #override.tiles < 6 then
for i = #override.tiles, 6 do
override.tiles[i] = override.tiles[#override.tiles]
end
end
-- add inlets to the sides that connect to tubes
local tile_directions = {"top", "bottom", "right", "left", "back", "front"}
for i, direction in ipairs(tile_directions) do
if connect_sides[direction] then
if type(override.tiles[i]) == "string" then
override.tiles[i] = override.tiles[i] .. tube_entry
elseif type(override.tiles[i]) == "table" and not override.tiles[i].animation then
override.tiles[i].name = override.tiles[i].name .. tube_entry
end
end
end
local old_after_place_node = override.after_place_node or old_def.after_place_node or function() end
override.after_place_node = function(pos, placer, itemstack, pointed_thing)
old_after_place_node(pos, placer, itemstack, pointed_thing)
pipeworks.after_place(pos)
end
local old_after_dig = override.after_dig or old_def.after_dig_node or function() end
override.after_dig_node = function(pos, oldnode, oldmetadata, digger)
old_after_dig(pos, oldnode, oldmetadata, digger)
pipeworks.after_dig(pos, oldnode, oldmetadata, digger)
end
local old_on_rotate
if override.on_rotate ~= nil then
old_on_rotate = override.on_rotate
elseif old_def.on_rotate ~= nil then
old_on_rotate = old_def.on_rotate
else
old_on_rotate = function() end
end
-- on_rotate = false -> rotation disabled, no need to update tubes
-- everything else: undefined by the most common screwdriver mods
if type(old_on_rotate) == "function" then
override.on_rotate = function(pos, node, user, mode, new_param2)
if old_on_rotate(pos, node, user, mode, new_param2) ~= false then
return pipeworks.on_rotate(pos, node, user, mode, new_param2)
else
return false
end
end
end
override.tube = {
insert_object = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
return inv:add_item("main", stack)
end,
can_insert = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if meta:get_int("splitstacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("main", stack)
end,
input_inventory = "main",
connect_sides = connect_sides
}
-- Add the extra groups
override.groups = override.groups or old_def.groups or {}
override.groups.tubedevice = 1
override.groups.tubedevice_receiver = 1
core.override_item(chestname, override)
pipeworks.chests[chestname] = true
end

View File

@@ -1,17 +1,104 @@
---------------------- local S = core.get_translator("pipeworks")
-- Vector functions --
----------------------
function pipeworks.vector_cross(a, b) -- Random variables
return {
x = a.y * b.z - a.z * b.y, pipeworks.expect_infinite_stacks = true
y = a.z * b.x - a.x * b.z, if core.get_modpath("unified_inventory") or not core.settings:get_bool("creative_mode") then
z = a.x * b.y - a.y * b.x pipeworks.expect_infinite_stacks = false
}
end end
function pipeworks.vector_dot(a, b) pipeworks.meseadjlist={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=0,y=1,z=0},{x=0,y=-1,z=0},{x=1,y=0,z=0},{x=-1,y=0,z=0}}
return a.x * b.x + a.y * b.y + a.z * b.z
pipeworks.rules_all = {{x=0, y=0, z=1},{x=0, y=0, z=-1},{x=1, y=0, z=0},{x=-1, y=0, z=0},
{x=0, y=1, z=1},{x=0, y=1, z=-1},{x=1, y=1, z=0},{x=-1, y=1, z=0},
{x=0, y=-1, z=1},{x=0, y=-1, z=-1},{x=1, y=-1, z=0},{x=-1, y=-1, z=0},
{x=0, y=1, z=0}, {x=0, y=-1, z=0}}
pipeworks.mesecons_rules={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=1,y=0,z=0},{x=-1,y=0,z=0},{x=0,y=1,z=0},{x=0,y=-1,z=0}}
local digilines_enabled = core.get_modpath("digilines") ~= nil
if digilines_enabled and pipeworks.enable_vertical_digilines_connectivity then
pipeworks.digilines_rules=digilines.rules.default
else
-- These rules break vertical connectivity to deployers, node breakers, dispensers, and digiline filter injectors
-- via digiline conducting tubes. Changing them may break some builds on some servers, so the setting was added
-- for server admins to be able to revert to the old "broken" behavior as some builds may use it as a "feature".
-- See https://github.com/mt-mods/pipeworks/issues/64
pipeworks.digilines_rules={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=1,y=0,z=0},{x=-1,y=0,z=0},{x=0,y=1,z=0},{x=0,y=-1,z=0}}
end
pipeworks.liquid_texture = core.registered_nodes[pipeworks.liquids.water.flowing].tiles[1]
if type(pipeworks.liquid_texture) == "table" then pipeworks.liquid_texture = pipeworks.liquid_texture.name end
pipeworks.button_off = {text="", texture="pipeworks_button_off.png", addopts="false;false;pipeworks_button_interm.png"}
pipeworks.button_on = {text="", texture="pipeworks_button_on.png", addopts="false;false;pipeworks_button_interm.png"}
pipeworks.button_base = "image_button[0,4.3;1,0.6"
pipeworks.button_label = "label[0.9,4.31;"..S("Allow splitting incoming stacks from tubes").."]"
-- Helper functions
function pipeworks.fix_image_names(table, replacement)
local outtable={}
for i in ipairs(table) do
outtable[i]=string.gsub(table[i], "_XXXXX", replacement)
end
return outtable
end
local function overlay_tube_texture(texture)
-- The texture appears the first time to be colorized as the opaque background.
return ("(%s)^[noalpha^[colorize:#dadada^(%s)"):format(texture, texture)
end
function pipeworks.make_tube_tile(tile)
if pipeworks.use_real_entities then
return tile
elseif type(tile) == "string" then
return overlay_tube_texture(tile)
else
tile = table.copy(tile)
if tile.color then
-- Won't work 100% of the time, but good enough.
tile.name = tile.name .. "^[multiply:" .. core.colorspec_to_colorstring(tile.color)
tile.color = nil
end
tile.name = overlay_tube_texture(tile.name)
tile.backface_culling = nil -- The texture is opaque
return tile
end
end
function pipeworks.add_node_box(t, b)
if not t or not b then return end
for i in ipairs(b)
do table.insert(t, b[i])
end
end
function pipeworks.may_configure(pos, player)
local name = player:get_player_name()
local meta = core.get_meta(pos)
local owner = meta:get_string("owner")
if owner ~= "" and owner == name then -- wielders and filters
return true
end
return not core.is_protected(pos, name)
end
function pipeworks.replace_name(tbl,tr,name)
local ntbl={}
for key,i in pairs(tbl) do
if type(i)=="string" then
ntbl[key]=string.gsub(i,tr,name)
elseif type(i)=="table" then
ntbl[key]=pipeworks.replace_name(i,tr,name)
else
ntbl[key]=i
end
end
return ntbl
end end
----------------------- -----------------------
@@ -29,9 +116,9 @@ function pipeworks.facedir_to_top_dir(facedir)
end end
function pipeworks.facedir_to_right_dir(facedir) function pipeworks.facedir_to_right_dir(facedir)
return pipeworks.vector_cross( return vector.cross(
pipeworks.facedir_to_top_dir(facedir), pipeworks.facedir_to_top_dir(facedir),
minetest.facedir_to_dir(facedir) core.facedir_to_dir(facedir)
) )
end end
@@ -49,29 +136,10 @@ function directions.side_to_dir(side)
end end
function directions.dir_to_side(dir) function directions.dir_to_side(dir)
local c = pipeworks.vector_dot(dir, vector.new(1, 2, 3)) + 4 local c = vector.dot(dir, vector.new(1, 2, 3)) + 4
return ({6, 2, 4, 0, 3, 1, 5})[c] return ({6, 2, 4, 0, 3, 1, 5})[c]
end end
----------------------
-- String functions --
----------------------
--[[function pipeworks.string_split(str, sep)
local fields = {}
local index = 1
local expr = "([^"..sep.."])+"
string.gsub(str, expr, function(substring)
fields[index] = substring
index = index + 1
end)
return fields
end]]
function pipeworks.string_startswith(str, substr)
return str:sub(1, substr:len()) == substr
end
--------------------- ---------------------
-- Table functions -- -- Table functions --
--------------------- ---------------------
@@ -113,9 +181,9 @@ end
local fs_helpers = {} local fs_helpers = {}
pipeworks.fs_helpers = fs_helpers pipeworks.fs_helpers = fs_helpers
function fs_helpers.on_receive_fields(pos, fields) function fs_helpers.on_receive_fields(pos, fields)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
for field in pairs(fields) do for field in pairs(fields) do
if pipeworks.string_startswith(field, "fs_helpers_cycling:") then if field:match("^fs_helpers_cycling:") then
local l = field:split(":") local l = field:split(":")
local new_value = tonumber(l[2]) local new_value = tonumber(l[2])
local meta_name = l[3] local meta_name = l[3]
@@ -140,7 +208,82 @@ function fs_helpers.cycling_button(meta, base, meta_name, values)
text = val text = val
end end
local field = "fs_helpers_cycling:"..new_value..":"..meta_name local field = "fs_helpers_cycling:"..new_value..":"..meta_name
return base..";"..(texture_name and texture_name..";" or "")..field..";"..minetest.formspec_escape(text)..(addopts and ";"..addopts or "").."]" return base..";"..(texture_name and texture_name..";" or "")..field..";"..core.formspec_escape(text)..(addopts and ";"..addopts or "").."]"
end
function fs_helpers.get_inv(y)
local fs = {}
if core.get_modpath("i3") then
local inv_x = i3.settings.legacy_inventory and 0.75 or 0.22
local inv_y = (y + 0.4) or 6.9
local size, spacing = 1, 0.1
local hotbar_len = i3.settings.hotbar_len or (i3.settings.legacy_inventory and 8 or 9)
local inv_size = i3.settings.inv_size or (hotbar_len * 4)
table.insert(fs, "style_type[box;colors=#77777710,#77777710,#777,#777]")
for i = 0, hotbar_len - 1 do
table.insert(fs, "box["..(i * size + inv_x + (i * spacing))..","..inv_y..";"..size..","..size..";]")
end
table.insert(fs, "style_type[list;size="..size..";spacing="..spacing.."]")
table.insert(fs, "list[current_player;main;"..inv_x..","..inv_y..";"..hotbar_len..",1;]")
table.insert(fs, "style_type[box;colors=#666]")
for i=0, 2 do
for j=0, hotbar_len - 1 do
table.insert(fs, "box["..0.2+(j*0.1)+(j*size)..","..(inv_y+size+spacing+0.05)+(i*0.1)+(i*size)..";"..size..","..size..";]")
end
end
table.insert(fs, "style_type[list;size="..size..";spacing="..spacing.."]")
table.insert(fs, "list[current_player;main;"..inv_x..","..(inv_y + 1.15)..";"..hotbar_len..","..(inv_size / hotbar_len)..";"..hotbar_len.."]")
elseif core.get_modpath("mcl_formspec") then
local inv_x = 0.22
local inv_y = (y + 0.4) or 6.9
local size, spacing = 1, 0.1
local hotbar_len = 9
local inv_size = hotbar_len * 4
table.insert(fs, "style_type[box;colors=#77777710,#77777710,#777,#777]")
for i = 0, hotbar_len - 1 do
table.insert(fs, "box["..(i * size + inv_x + (i * spacing))..","..inv_y..";"..size..","..size..";]")
end
table.insert(fs, "style_type[list;size="..size..";spacing="..spacing.."]")
table.insert(fs, "list[current_player;main;"..inv_x..","..inv_y..";"..hotbar_len..",1;]")
table.insert(fs, "style_type[box;colors=#666]")
for i=0, 2 do
for j=0, hotbar_len - 1 do
table.insert(fs, "box["..0.2+(j*0.1)+(j*size)..","..(inv_y+size+spacing+0.05)+(i*0.1)+(i*size)..";"..size..","..size..";]")
end
end
table.insert(fs, "style_type[list;size="..size..";spacing="..spacing.."]")
table.insert(fs, "list[current_player;main;"..inv_x..","..(inv_y + 1.15)..";"..hotbar_len..","..(inv_size / hotbar_len)..";"..hotbar_len.."]")
else
table.insert(fs, "list[current_player;main;0.22,"..y..";8,4;]")
end
return table.concat(fs, "")
end
function fs_helpers.get_prepends(size)
local prepend = {}
if core.get_modpath("i3") then
prepend = {
"no_prepend[]",
"bgcolor[black;neither]",
"background9[0,0;"..size..";i3_bg_full.png;false;10]",
"style_type[button;border=false;bgimg=[combine:16x16^[noalpha^[colorize:#6b6b6b]",
"listcolors[#0000;#ffffff20]"
}
end
return table.concat(prepend, "")
end end
--------- ---------
@@ -150,141 +293,12 @@ end
function pipeworks.load_position(pos) function pipeworks.load_position(pos)
if pos.x < -30912 or pos.y < -30912 or pos.z < -30912 or if pos.x < -30912 or pos.y < -30912 or pos.z < -30912 or
pos.x > 30927 or pos.y > 30927 or pos.z > 30927 then return end pos.x > 30927 or pos.y > 30927 or pos.z > 30927 then return end
if minetest.get_node_or_nil(pos) then if core.get_node_or_nil(pos) then
return return
end end
local vm = minetest.get_voxel_manip() local vm = core.get_voxel_manip()
vm:read_from_map(pos, pos) vm:read_from_map(pos, pos)
end end
local function delay(...) -- Kept for compatibility with old mods
local args = {...} pipeworks.create_fake_player = fakelib.create_player
return (function() return unpack(args) end)
end
local function get_set_wrap(name, is_dynamic)
return (function(self)
return self["_" .. name]
end), (function(self, value)
if is_dynamic then
self["_" .. name] = type(value) == "table"
and table.copy(value) or value
end
end)
end
function pipeworks.create_fake_player(def, is_dynamic)
local wielded_item = ItemStack("")
if def.inventory and def.wield_list then
wielded_item = def.inventory:get_stack(def.wield_list, def.wield_index or 1)
end
local p = {
get_player_name = delay(def.name),
is_player = delay(true),
is_fake_player = true,
_formspec = def.formspec or default.gui_survival_form,
_hp = def.hp or 20,
_breath = 11,
_pos = def.position and table.copy(def.position) or vector.new(),
_properties = def.properties or { eye_height = def.eye_height or 1.47 },
_inventory = def.inventory,
_wield_index = def.wield_index or 1,
_wielded_item = wielded_item,
-- Model and view
_eye_offset1 = vector.new(),
_eye_offset3 = vector.new(),
set_eye_offset = function(self, first, third)
self._eye_offset1 = table.copy(first)
self._eye_offset3 = table.copy(third)
end,
get_eye_offset = function(self)
return self._eye_offset1, self._eye_offset3
end,
get_look_dir = delay(def.look_dir or {x=0, y=0, z=1}),
get_look_pitch = delay(def.look_pitch or 0),
get_look_yaw = delay(def.look_yaw or 0),
get_look_horizontal = delay(def.look_yaw or 0),
get_look_vertical = delay(-(def.look_pitch or 0)),
set_animation = delay(),
-- Controls
get_player_control = delay({
jump=false, right=false, left=false, LMB=false, RMB=false,
sneak=def.sneak, aux1=false, down=false, up=false
}),
get_player_control_bits = delay(def.sneak and 64 or 0),
-- Inventory and ItemStacks
get_inventory = delay(def.inventory),
set_wielded_item = function(self, item)
if self._inventory and def.wield_list then
return self._inventory:set_stack(def.wield_list,
self._wield_index, item)
end
self._wielded_item = ItemStack(item)
end,
get_wielded_item = function(self, item)
if self._inventory and def.wield_list then
return self._inventory:get_stack(def.wield_list,
self._wield_index)
end
return ItemStack(self._wielded_item)
end,
get_wield_list = delay(def.wield_list),
punch = delay(),
remove = delay(),
right_click = delay(),
set_attach = delay(),
set_detach = delay(),
set_bone_position = delay(),
hud_change = delay(),
}
-- Getter & setter functions
p.get_inventory_formspec, p.set_inventory_formspec
= get_set_wrap("formspec", is_dynamic)
p.get_breath, p.set_breath = get_set_wrap("breath", is_dynamic)
p.get_hp, p.set_hp = get_set_wrap("hp", is_dynamic)
p.get_pos, p.set_pos = get_set_wrap("pos", is_dynamic)
p.get_wield_index, p.set_wield_index = get_set_wrap("wield_index", true)
p.get_properties, p.set_properties = get_set_wrap("properties", false)
-- For players, move_to and get_pos do the same
p.move_to = p.get_pos
-- Backwards compatibilty
p.getpos = p.get_pos
p.setpos = p.set_pos
p.moveto = p.move_to
-- TODO "implement" all these
-- set_armor_groups
-- get_armor_groups
-- get_animation
-- get_bone_position
-- get_player_velocity
-- set_look_pitch
-- set_look_yaw
-- set_physics_override
-- get_physics_override
-- hud_add
-- hud_remove
-- hud_get
-- hud_set_flags
-- hud_get_flags
-- hud_set_hotbar_itemcount
-- hud_get_hotbar_itemcount
-- hud_set_hotbar_image
-- hud_get_hotbar_image
-- hud_set_hotbar_selected_image
-- hud_get_hotbar_selected_image
-- hud_replace_builtin
-- set_sky
-- get_sky
-- override_day_night_ratio
-- get_day_night_ratio
-- set_local_animation
return p
end

View File

@@ -1,243 +1,184 @@
-- this bit of code modifies the default chests and furnaces to be compatible -- this bit of code overrides the default chests from common games (mtg, hades, minclone*) to be
-- with pipeworks. -- compatible with pipeworks. Where possible, it overrides their formspec to add a splitstacks switch
--
-- the formspecs found here are basically copies of the ones from minetest_game
-- plus bits from pipeworks' sorting tubes
-- Pipeworks Specific
local fs_helpers = pipeworks.fs_helpers local fs_helpers = pipeworks.fs_helpers
local tube_entry = "^pipeworks_tube_connection_wooden.png"
-- Chest Locals -- formspec helper to add the splitstacks switch
local open_chests = {} local function add_pipeworks_switch(formspec, pos)
-- based on the sorting tubes
local function get_chest_formspec(pos)
local spos = pos.x .. "," .. pos.y .. "," .. pos.z
local formspec =
"size[8,9]" ..
default.gui_bg ..
default.gui_bg_img ..
default.gui_slots ..
"list[nodemeta:" .. spos .. ";main;0,0.3;8,4;]" ..
"list[current_player;main;0,4.85;8,1;]" ..
"list[current_player;main;0,6.08;8,3;8]" ..
"listring[nodemeta:" .. spos .. ";main]" ..
"listring[current_player;main]" ..
default.get_hotbar_bg(0,4.85)
-- Pipeworks Switch
formspec = formspec .. formspec = formspec ..
fs_helpers.cycling_button( fs_helpers.cycling_button(
minetest.get_meta(pos), core.get_meta(pos),
pipeworks.button_base, pipeworks.button_base,
"splitstacks", "splitstacks",
{ {
pipeworks.button_off, pipeworks.button_off,
pipeworks.button_on pipeworks.button_on
} }
)..pipeworks.button_label )..pipeworks.button_label
return formspec return formspec
end end
local function chest_lid_obstructed(pos) -- helper to add the splitstacks switch to a node-formspec
local above = { x = pos.x, y = pos.y + 1, z = pos.z } local function update_node_formspec(pos)
local def = minetest.registered_nodes[minetest.get_node(above).name] local meta = core.get_meta(pos)
-- allow ladders, signs, wallmounted things and torches to not obstruct local old_fs = meta:get_string("formspec")
if not def then return true end local new_fs = add_pipeworks_switch(old_fs, pos)
if def.drawtype == "airlike" or meta:set_string("formspec", new_fs)
def.drawtype == "signlike" or
def.drawtype == "torchlike" or
(def.drawtype == "nodebox" and def.paramtype2 == "wallmounted") then
return false
end
return true
end end
minetest.register_on_player_receive_fields(function(player, formname, fields)
if formname == "pipeworks:chest_formspec" and player then if core.get_modpath("default") then
-- add the pipeworks switch into the default chest formspec
local old_get_chest_formspec = default.chest.get_chest_formspec
-- luacheck: ignore 122
default.chest.get_chest_formspec = function(pos)
local old_fs = old_get_chest_formspec(pos)
local node = core.get_node(pos)
-- not all chests using this formspec necessary connect to pipeworks
if pipeworks.chests[node.name] then
local new_fs = add_pipeworks_switch(old_fs, pos)
return new_fs
else
return old_fs
end
end
-- get the fields from the chest formspec, we can do this bc. newest functions are called first
-- https://github.com/luanti-org/luanti/blob/d4b10db998ebeb689b3d27368e30952a42169d03/doc/lua_api.md?plain=1#L5840
core.register_on_player_receive_fields(function(player, formname, fields)
if fields.quit or formname ~= "default:chest" then
return
end
local pn = player:get_player_name() local pn = player:get_player_name()
if open_chests[pn] then local chest_open = default.chest.open_chests[pn]
local pos = open_chests[pn].pos if not chest_open or not chest_open.pos then
if fields.quit then -- chest already closed before formspec
local sound = open_chests[pn].sound return
local swap = open_chests[pn].swap end
local node = minetest.get_node(pos) local pos = chest_open.pos
local node = core.get_node(pos)
if pipeworks.chests[node.name] and pipeworks.may_configure(pos, player) then
-- Pipeworks Switch
fs_helpers.on_receive_fields(pos, fields)
core.show_formspec(pn,
"default:chest",
default.chest.get_chest_formspec(pos))
end
-- Do NOT return true here, the callback from default still needs to run
return false
end)
open_chests[pn] = nil local connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
for _, v in pairs(open_chests) do local connect_sides_open = {left = 1, right = 1, back = 1, bottom = 1}
if v.pos.x == pos.x and v.pos.y == pos.y and v.pos.z == pos.z then
return true pipeworks.override_chest("default:chest", {}, connect_sides)
end pipeworks.override_chest("default:chest_open", {}, connect_sides_open)
pipeworks.override_chest("default:chest_locked", {}, connect_sides)
pipeworks.override_chest("default:chest_locked_open", {}, connect_sides_open)
elseif core.get_modpath("hades_chests") then
local chest_colors = {"", "white", "grey", "dark_grey", "black", "blue", "cyan", "dark_green", "green", "magenta",
"orange", "pink", "red", "violet", "yellow"}
for _, color in ipairs(chest_colors) do
local chestname = (color == "" and "hades_chests:chest")
or "hades_chests:chest_" .. color
local chestname_protected = (color == "" and "hades_chests:chest_locked")
or "hades_chests:chest_" .. color .. "_locked"
local old_def = core.registered_nodes[chestname]
-- chest formspec-creation functions are local, we need to find other ways
-- normal chests use node formspecs, we can hack into these
local old_on_construct = old_def.on_construct
local override = {
on_construct = function(pos)
old_on_construct(pos)
update_node_formspec(pos)
end,
on_receive_fields = function(pos, formname, fields, player)
if not fields.quit and pipeworks.may_configure(pos, player) then
-- Pipeworks Switch
fs_helpers.on_receive_fields(pos, fields)
update_node_formspec(pos)
end end
minetest.after(0.2, function() end,
minetest.swap_node(pos, { name = "default:" .. swap, param2 = node.param2 }) -- chest's on_rotate is "simple", but we assumed the api from the mtg screwdriver mod
-- this will keep the same behavior, but supports the code above
on_rotate = screwdriver.rotate_simple
}
-- Pipeworks notification -- locked chests uses local functions to create their formspec - we need to copy these
pipeworks.after_place(pos) -- https://codeberg.org/Wuzzy/Hades_Revisited/src/branch/master/mods/hades_chests/init.lua
end) local function get_locked_chest_formspec(pos)
minetest.sound_play(sound, {gain = 0.3, pos = pos, max_hear_distance = 10}) local spos = pos.x .. "," .. pos.y .. "," ..pos.z
elseif pipeworks.may_configure(pos, player) then local formspec =
-- Pipeworks Switch "size[10,9]"..
fs_helpers.on_receive_fields(pos, fields) "list[nodemeta:".. spos .. ";main;0,0;10,4;]"..
minetest.show_formspec(player:get_player_name(), "pipeworks:chest_formspec", get_chest_formspec(pos)) "list[current_player;main;0,5;10,4;]"..
"listring[]"..
"background9[8,8;10,9;hades_chests_chestui.png;true;8]"
-- change from pipeworks
local new_fs = add_pipeworks_switch(formspec, pos)
return new_fs
end
local function has_locked_chest_privilege(meta, player)
local name = player:get_player_name()
if name ~= meta:get_string("owner") and not core.check_player_privs(name, "protection_bypass") then
return false
end end
return true return true
end end
-- store, which chest a formspec submission belongs to
-- {player1 = pos1, player2 = pos2, ...}
local open_chests = {}
core.register_on_leaveplayer(function(player)
open_chests[player:get_player_name()] = nil
end)
local override_protected = {
on_rightclick = function(pos, node, clicker)
local meta = core.get_meta(pos)
if has_locked_chest_privilege(meta, clicker) then
core.show_formspec(
clicker:get_player_name(),
"hades_chests:chest_locked",
get_locked_chest_formspec(pos)
)
open_chests[clicker:get_player_name()] = pos
else
core.sound_play({ name = "hades_chests_locked", gain = 0.3 }, { max_hear_distance = 10 }, true)
end
end,
on_rotate = screwdriver.rotate_simple
}
-- get the fields from the chest formspec, we can do this bc. newest functions are called first
-- https://github.com/luanti-org/luanti/blob/d4b10db998ebeb689b3d27368e30952a42169d03/doc/lua_api.md?plain=1#L5840
core.register_on_player_receive_fields(function(player, formname, fields)
if fields.quit or formname ~= "hades_chests:chest_locked" then
return
end
local pn = player:get_player_name()
local pos = open_chests[pn]
if pos and pipeworks.may_configure(pos, player) then
-- Pipeworks Switch
fs_helpers.on_receive_fields(pos, fields)
core.show_formspec(pn, "hades_chests:chest_locked", get_locked_chest_formspec(pos))
end
-- Do NOT return true here, the callback from hades still needs to run (if they add one)
return false
end)
local connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
pipeworks.override_chest(chestname, override, connect_sides)
pipeworks.override_chest(chestname_protected, override_protected, connect_sides)
end end
end) elseif core.get_modpath("mcl_barrels") then
-- TODO: bring splitstacks switch in the formspec
-- Original Definitions -- with the current implementation of mcl_barrels this would mean to duplicate a lot of code from there...
local old_chest_def = table.copy(minetest.registered_items["default:chest"]) local connect_sides = {left = 1, right = 1, back = 1, front = 1, bottom = 1}
local old_chest_open_def = table.copy(minetest.registered_items["default:chest_open"]) pipeworks.override_chest("mcl_barrels:barrel_closed", {}, connect_sides)
local old_chest_locked_def = table.copy(minetest.registered_items["default:chest_locked"]) pipeworks.override_chest("mcl_barrels:barrel_open", {}, connect_sides)
local old_chest_locked_open_def = table.copy(minetest.registered_items["default:chest_locked_open"])
-- Override Construction
local override_protected, override, override_open, override_protected_open
override_protected = {
tiles = {
"default_chest_top.png"..tube_entry,
"default_chest_top.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_lock.png",
"default_chest_inside.png"
},
after_place_node = function(pos, placer)
old_chest_locked_def.after_place_node(pos, placer)
pipeworks.after_place(pos)
end,
on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
if not default.can_interact_with_node(clicker, pos) then
return itemstack
end
minetest.sound_play(old_chest_locked_def.sound_open, {gain = 0.3,
pos = pos, max_hear_distance = 10})
if not chest_lid_obstructed(pos) then
minetest.swap_node(pos,
{ name = "default:" .. "chest_locked" .. "_open",
param2 = node.param2 })
end
minetest.after(0.2, minetest.show_formspec,
clicker:get_player_name(),
"pipeworks:chest_formspec", get_chest_formspec(pos))
open_chests[clicker:get_player_name()] = { pos = pos,
sound = old_chest_locked_def.sound_close, swap = "chest_locked" }
end,
groups = table.copy(old_chest_locked_def.groups),
tube = {
insert_object = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
return inv:add_item("main", stack)
end,
can_insert = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
if meta:get_int("splitstacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("main", stack)
end,
input_inventory = "main",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
},
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate
}
override = {
tiles = {
"default_chest_top.png"..tube_entry,
"default_chest_top.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_front.png",
"default_chest_inside.png"
},
on_rightclick = function(pos, node, clicker)
minetest.sound_play(old_chest_def.sound_open, {gain = 0.3, pos = pos,
max_hear_distance = 10})
if not chest_lid_obstructed(pos) then
minetest.swap_node(pos, {
name = "default:" .. "chest" .. "_open",
param2 = node.param2 })
end
minetest.after(0.2, minetest.show_formspec,
clicker:get_player_name(),
"pipeworks:chest_formspec", get_chest_formspec(pos))
open_chests[clicker:get_player_name()] = { pos = pos,
sound = old_chest_def.sound_close, swap = "chest" }
end,
groups = table.copy(old_chest_def.groups),
tube = {
insert_object = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
return inv:add_item("main", stack)
end,
can_insert = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
if meta:get_int("splitstacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("main", stack)
end,
input_inventory = "main",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
},
after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate
}
--[[local override_common = {
}
for k,v in pairs(override_common) do
override_protected[k] = v
override[k] = v
end]]
override_open = table.copy(override)
override_open.groups = table.copy(old_chest_open_def.groups)
override_open.tube = table.copy(override.tube)
override_open.tube.connect_sides = table.copy(override.tube.connect_sides)
override_open.tube.connect_sides.top = nil
override_protected_open = table.copy(override_protected)
override_protected_open.groups = table.copy(old_chest_locked_open_def.groups)
override_protected_open.tube = table.copy(override_protected.tube)
override_protected_open.tube.connect_sides = table.copy(override_protected.tube.connect_sides)
override_protected_open.tube.connect_sides.top = nil
override_protected.tiles = { -- Rearranged according to the chest registration in Minetest_Game.
"default_chest_top.png"..tube_entry,
"default_chest_top.png"..tube_entry,
"default_chest_side.png"..tube_entry.."^[transformFX",
"default_chest_side.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_lock.png",
}
override.tiles = {
"default_chest_top.png"..tube_entry,
"default_chest_top.png"..tube_entry,
"default_chest_side.png"..tube_entry.."^[transformFX",
"default_chest_side.png"..tube_entry,
"default_chest_side.png"..tube_entry,
"default_chest_front.png",
}
-- Add the extra groups
for _,v in ipairs({override_protected, override, override_open, override_protected_open}) do
v.groups.tubedevice = 1
v.groups.tubedevice_receiver = 1
end end
-- Override with the new modifications.
minetest.override_item("default:chest", override)
minetest.override_item("default:chest_open", override_open)
minetest.override_item("default:chest_locked", override_protected)
minetest.override_item("default:chest_locked_open", override_protected_open)

View File

@@ -1,297 +1,28 @@
-- this file is basically a modified copy of -- this file is basically a modified copy of
-- minetest_game/mods/default/furnaces.lua -- minetest_game/mods/default/furnaces.lua
-- translation support local def--, def_active
local S = minetest.get_translator("pipeworks") if core.get_modpath("default") then
local DS = minetest.get_translator("default") def = table.copy(core.registered_nodes["default:furnace"])
--def_active = table.copy(core.registered_nodes["default:furnace_active"])
local fs_helpers = pipeworks.fs_helpers elseif core.get_modpath("hades_furnaces") then
def = table.copy(core.registered_nodes["hades_furnaces:furnace"])
--def_active = table.copy(core.registered_nodes["hades_furnaces:furnace_active"])
end
local tube_entry = "^pipeworks_tube_connection_stony.png" local tube_entry = "^pipeworks_tube_connection_stony.png"
local function active_formspec(fuel_percent, item_percent, pos, meta) local groups = def.groups
local formspec = groups["tubedevice"] = 1
"size[8,8.5]".. groups["tubedevice_receiver"] = 1
default.gui_bg.. local groups_active = table.copy(groups)
default.gui_bg_img.. groups_active["not_in_creative_inventory"] = 1
default.gui_slots..
"list[current_name;src;2.75,0.5;1,1;]"..
"list[current_name;fuel;2.75,2.5;1,1;]"..
"image[2.75,1.5;1,1;default_furnace_fire_bg.png^[lowpart:"..
(100-fuel_percent)..":default_furnace_fire_fg.png]"..
"image[3.75,1.5;1,1;gui_furnace_arrow_bg.png^[lowpart:"..
(item_percent)..":gui_furnace_arrow_fg.png^[transformR270]"..
"list[current_name;dst;4.75,0.96;2,2;]"..
"list[current_player;main;0,4.25;8,1;]"..
"list[current_player;main;0,5.5;8,3;8]"..
"listring[current_name;dst]"..
"listring[current_player;main]"..
"listring[current_name;src]"..
"listring[current_player;main]"..
"listring[current_name;fuel]"..
"listring[current_player;main]"..
default.get_hotbar_bg(0, 4.25) ..
fs_helpers.cycling_button(
meta,
"image_button[0,3.5;1,0.6",
"split_material_stacks",
{
pipeworks.button_off,
pipeworks.button_on
}
).."label[0.9,3.51;"..S("Allow splitting incoming material (not fuel) stacks from tubes").."]"
return formspec
end
local function inactive_formspec(pos, meta)
local formspec = "size[8,8.5]"..
default.gui_bg..
default.gui_bg_img..
default.gui_slots..
"list[current_name;src;2.75,0.5;1,1;]"..
"list[current_name;fuel;2.75,2.5;1,1;]"..
"image[2.75,1.5;1,1;default_furnace_fire_bg.png]"..
"image[3.75,1.5;1,1;gui_furnace_arrow_bg.png^[transformR270]"..
"list[current_name;dst;4.75,0.96;2,2;]"..
"list[current_player;main;0,4.25;8,1;]"..
"list[current_player;main;0,5.5;8,3;8]"..
"listring[current_name;dst]"..
"listring[current_player;main]"..
"listring[current_name;src]"..
"listring[current_player;main]"..
"listring[current_name;fuel]"..
"listring[current_player;main]"..
default.get_hotbar_bg(0, 4.25) ..
fs_helpers.cycling_button(
meta,
"image_button[0,3.5;1,0.6",
"split_material_stacks",
{
pipeworks.button_off,
pipeworks.button_on
}
).."label[0.9,3.51;"..S("Allow splitting incoming material (not fuel) stacks from tubes").."]"
return formspec
end
--
-- Node callback functions that are the same for active and inactive furnace
--
local function can_dig(pos, player)
local meta = minetest.get_meta(pos);
local inv = meta:get_inventory()
return inv:is_empty("fuel") and inv:is_empty("dst") and inv:is_empty("src")
end
local function allow_metadata_inventory_put(pos, listname, index, stack, player)
if minetest.is_protected(pos, player:get_player_name()) then
return 0
end
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
if listname == "fuel" then
if minetest.get_craft_result({method="fuel", width=1, items={stack}}).time ~= 0 then
if inv:is_empty("src") then
meta:set_string("infotext", DS("Furnace is empty"))
end
return stack:get_count()
else
return 0
end
elseif listname == "src" then
return stack:get_count()
elseif listname == "dst" then
return 0
end
end
local function allow_metadata_inventory_move(pos, from_list, from_index, to_list, to_index, count, player)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
local stack = inv:get_stack(from_list, from_index)
return allow_metadata_inventory_put(pos, to_list, to_index, stack, player)
end
local function allow_metadata_inventory_take(pos, listname, index, stack, player)
if minetest.is_protected(pos, player:get_player_name()) then
return 0
end
return stack:get_count()
end
local function swap_node(pos, name)
local node = minetest.get_node(pos)
if node.name == name then
return
end
node.name = name
minetest.swap_node(pos, node)
end
local function furnace_node_timer(pos, elapsed)
--
-- Inizialize metadata
--
local meta = minetest.get_meta(pos)
local fuel_time = meta:get_float("fuel_time") or 0
local src_time = meta:get_float("src_time") or 0
local fuel_totaltime = meta:get_float("fuel_totaltime") or 0
local inv = meta:get_inventory()
local srclist, fuellist
local cookable, cooked
local fuel
local update = true
while update do
update = false
srclist = inv:get_list("src")
fuellist = inv:get_list("fuel")
--
-- Cooking
--
-- Check if we have cookable content
local aftercooked
cooked, aftercooked = minetest.get_craft_result({method = "cooking", width = 1, items = srclist})
cookable = cooked.time ~= 0
-- Check if we have enough fuel to burn
if fuel_time < fuel_totaltime then
-- The furnace is currently active and has enough fuel
fuel_time = fuel_time + elapsed
-- If there is a cookable item then check if it is ready yet
if cookable then
src_time = src_time + elapsed
if src_time >= cooked.time then
-- Place result in dst list if possible
if inv:room_for_item("dst", cooked.item) then
inv:add_item("dst", cooked.item)
inv:set_stack("src", 1, aftercooked.items[1])
src_time = src_time - cooked.time
update = true
end
end
end
else
-- Furnace ran out of fuel
if cookable then
-- We need to get new fuel
local afterfuel
fuel, afterfuel = minetest.get_craft_result({method = "fuel", width = 1, items = fuellist})
if fuel.time == 0 then
-- No valid fuel in fuel list
fuel_totaltime = 0
src_time = 0
else
-- prevent blocking of fuel inventory (for automatization mods)
local is_fuel = minetest.get_craft_result({method = "fuel", width = 1, items = {afterfuel.items[1]:to_string()}})
if is_fuel.time == 0 then
table.insert(fuel.replacements, afterfuel.items[1])
inv:set_stack("fuel", 1, "")
else
-- Take fuel from fuel list
inv:set_stack("fuel", 1, afterfuel.items[1])
end
-- Put replacements in dst list or drop them on the furnace.
local replacements = fuel.replacements
if replacements[1] then
local leftover = inv:add_item("dst", replacements[1])
if not leftover:is_empty() then
local above = vector.new(pos.x, pos.y + 1, pos.z)
local drop_pos = minetest.find_node_near(above, 1, {"air"}) or above
minetest.item_drop(replacements[1], nil, drop_pos)
end
end
update = true
fuel_totaltime = fuel.time + (fuel_time - fuel_totaltime)
src_time = src_time + elapsed
end
else
-- We don't need to get new fuel since there is no cookable item
fuel_totaltime = 0
src_time = 0
end
fuel_time = 0
end
elapsed = 0
end
if fuel and fuel_totaltime > fuel.time then
fuel_totaltime = fuel.time
end
if srclist[1]:is_empty() then
src_time = 0
end
--
-- Update formspec, infotext and node
--
local formspec = inactive_formspec(pos, meta)
local item_state
local item_percent = 0
if cookable then
item_percent = math.floor(src_time / cooked.time * 100)
if item_percent > 100 then
item_state = DS("100% (output full)")
else
item_state = DS("@1%", item_percent)
end
else
if srclist[1]:is_empty() then
item_state = DS("Empty")
else
item_state = DS("Not cookable")
end
end
local fuel_state = DS("Empty")
local active = DS("Furnace inactive")
local result = false
if fuel_totaltime ~= 0 then
active = DS("Furnace active")
local fuel_percent = math.floor(fuel_time / fuel_totaltime * 100)
fuel_state = DS("@1%", fuel_percent)
formspec = active_formspec(fuel_percent, item_percent, pos, meta)
swap_node(pos, "default:furnace_active")
-- make sure timer restarts automatically
result = true
else
if not fuellist[1]:is_empty() then
fuel_state = DS("@1%", "0")
end
swap_node(pos, "default:furnace")
-- stop timer on the inactive furnace
minetest.get_node_timer(pos):stop()
end
local infotext = active.." "..DS("(Item: @1; Fuel: @2)", item_state, fuel_state)
--
-- Set meta values
--
meta:set_float("fuel_totaltime", fuel_totaltime)
meta:set_float("fuel_time", fuel_time)
meta:set_float("src_time", src_time)
meta:set_string("formspec", formspec)
meta:set_string("infotext", infotext)
return result
end
-- --
-- Node definitions -- Node definitions
-- --
minetest.register_node(":default:furnace", { local override = {
description = DS("Furnace"),
tiles = { tiles = {
"default_furnace_top.png"..tube_entry, "default_furnace_top.png"..tube_entry,
"default_furnace_bottom.png"..tube_entry, "default_furnace_bottom.png"..tube_entry,
@@ -300,12 +31,12 @@ minetest.register_node(":default:furnace", {
"default_furnace_side.png"..tube_entry, "default_furnace_side.png"..tube_entry,
"default_furnace_front.png" "default_furnace_front.png"
}, },
groups = {cracky = 2, tubedevice = 1, tubedevice_receiver = 1}, groups = groups,
tube = { tube = {
insert_object = function(pos, node, stack, direction) insert_object = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
local timer = minetest.get_node_timer(pos) local timer = core.get_node_timer(pos)
if not timer:is_started() then if not timer:is_started() then
timer:start(1.0) timer:start(1.0)
end end
@@ -316,7 +47,7 @@ minetest.register_node(":default:furnace", {
end end
end, end,
can_insert = function(pos,node,stack,direction) can_insert = function(pos,node,stack,direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
if direction.y == 1 then if direction.y == 1 then
return inv:room_for_item("fuel", stack) return inv:room_for_item("fuel", stack)
@@ -328,59 +59,15 @@ minetest.register_node(":default:furnace", {
end end
end, end,
input_inventory = "dst", input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, front = 1, bottom = 1, top = 1} connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}, },
paramtype2 = "facedir",
legacy_facedir_simple = true,
is_ground_content = false,
sounds = default.node_sound_stone_defaults(),
can_dig = can_dig,
on_timer = furnace_node_timer,
on_construct = function(pos)
local meta = minetest.get_meta(pos)
meta:set_string("formspec", inactive_formspec(pos, meta))
local inv = meta:get_inventory()
inv:set_size('src', 1)
inv:set_size('fuel', 1)
inv:set_size('dst', 4)
end,
on_metadata_inventory_move = function(pos)
minetest.get_node_timer(pos):start(1.0)
end,
on_metadata_inventory_put = function(pos)
-- start timer function, it will sort out whether furnace can burn or not.
minetest.get_node_timer(pos):start(1.0)
end,
on_blast = function(pos)
local drops = {}
default.get_inventory_drops(pos, "src", drops)
default.get_inventory_drops(pos, "fuel", drops)
default.get_inventory_drops(pos, "dst", drops)
drops[#drops+1] = "default:furnace"
minetest.remove_node(pos)
return drops
end,
allow_metadata_inventory_put = allow_metadata_inventory_put,
allow_metadata_inventory_move = allow_metadata_inventory_move,
allow_metadata_inventory_take = allow_metadata_inventory_take,
on_receive_fields = function(pos, formname, fields, sender)
if not pipeworks.may_configure(pos, sender) then return end
fs_helpers.on_receive_fields(pos, fields)
local meta = minetest.get_meta(pos)
local formspec = inactive_formspec(pos, meta)
meta:set_string("formspec", formspec)
end,
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate on_rotate = pipeworks.on_rotate
}) }
minetest.register_node(":default:furnace_active", { local override_active = {
description = DS("Furnace"),
tiles = { tiles = {
"default_furnace_top.png"..tube_entry, "default_furnace_top.png"..tube_entry,
"default_furnace_bottom.png"..tube_entry, "default_furnace_bottom.png"..tube_entry,
@@ -398,12 +85,12 @@ minetest.register_node(":default:furnace_active", {
}, },
} }
}, },
groups = {cracky = 2, tubedevice = 1, tubedevice_receiver = 1, not_in_creative_inventory = 1}, groups = groups_active,
tube = { tube = {
insert_object = function(pos,node,stack,direction) insert_object = function(pos,node,stack,direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
local timer = minetest.get_node_timer(pos) local timer = core.get_node_timer(pos)
if not timer:is_started() then if not timer:is_started() then
timer:start(1.0) timer:start(1.0)
end end
@@ -414,42 +101,28 @@ minetest.register_node(":default:furnace_active", {
end end
end, end,
can_insert = function(pos, node, stack, direction) can_insert = function(pos, node, stack, direction)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
if direction.y == 1 then if direction.y == 1 then
return inv:room_for_item("fuel", stack) return inv:room_for_item("fuel", stack)
else else
if meta:get_int("split_material_stacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("src", stack) return inv:room_for_item("src", stack)
end end
end, end,
input_inventory = "dst", input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, front = 1, bottom = 1, top = 1} connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}, },
paramtype2 = "facedir",
light_source = 8,
drop = "default:furnace",
legacy_facedir_simple = true,
is_ground_content = false,
sounds = default.node_sound_stone_defaults(),
on_timer = furnace_node_timer,
can_dig = can_dig,
allow_metadata_inventory_put = allow_metadata_inventory_put,
allow_metadata_inventory_move = allow_metadata_inventory_move,
allow_metadata_inventory_take = allow_metadata_inventory_take,
on_receive_fields = function(pos, formname, fields, sender)
if not pipeworks.may_configure(pos, sender) then return end
fs_helpers.on_receive_fields(pos, fields)
local meta = minetest.get_meta(pos)
local formspec = active_formspec(0, 0, pos, meta)
meta:set_string("formspec", formspec)
end,
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate on_rotate = pipeworks.on_rotate
}) }
if core.get_modpath("default") then
core.override_item("default:furnace", override)
core.override_item("default:furnace_active", override_active)
elseif core.get_modpath("hades_furnaces") then
core.override_item("hades_furnaces:furnace", override)
core.override_item("hades_furnaces:furnace_active", override_active)
end

View File

@@ -1,87 +1,318 @@
-- Crafting recipes for pipes -- Crafting recipes for pipes
local materials = xcompat.materials
minetest.register_craft( { core.register_craft( {
output = "pipeworks:pipe_1_empty 12", output = "pipeworks:pipe_1_empty 12",
recipe = {
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" },
{ "", "", "" },
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" }
},
})
minetest.register_craft( {
output = "pipeworks:straight_pipe_empty 3",
recipe = {
{ "pipeworks:pipe_1_empty", "pipeworks:pipe_1_empty", "pipeworks:pipe_1_empty" },
},
})
minetest.register_craft( {
output = "pipeworks:spigot 3",
recipe = {
{ "pipeworks:pipe_1_empty", "" },
{ "", "pipeworks:pipe_1_empty" },
},
})
minetest.register_craft( {
output = "pipeworks:entry_panel_empty 2",
recipe = { recipe = {
{ "", "default:steel_ingot", "" }, { materials.steel_ingot, materials.steel_ingot, materials.steel_ingot },
{ "", "pipeworks:pipe_1_empty", "" }, { "", "", "" },
{ "", "default:steel_ingot", "" }, { materials.steel_ingot, materials.steel_ingot, materials.steel_ingot }
}, },
}) })
core.register_craft( {
output = "pipeworks:straight_pipe_empty 3",
recipe = {
{ "pipeworks:pipe_1_empty", "pipeworks:pipe_1_empty", "pipeworks:pipe_1_empty" },
},
})
core.register_craft( {
output = "pipeworks:spigot 3",
recipe = {
{ "pipeworks:pipe_1_empty", "" },
{ "", "pipeworks:pipe_1_empty" },
},
})
core.register_craft( {
output = "pipeworks:entry_panel_empty 2",
recipe = {
{ "", materials.steel_ingot, "" },
{ "", "pipeworks:pipe_1_empty", "" },
{ "", materials.steel_ingot, "" },
},
})
-- Various ancillary pipe devices -- Various ancillary pipe devices
minetest.register_craft( { core.register_craft( {
output = "pipeworks:pump_off 2", output = "pipeworks:pump_off 2",
recipe = { recipe = {
{ "default:stone", "default:steel_ingot", "default:stone" }, { materials.stone, materials.steel_ingot, materials.stone },
{ "default:copper_ingot", "default:mese_crystal_fragment", "default:copper_ingot" }, { materials.copper_ingot, materials.mese_crystal_fragment, materials.copper_ingot },
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" } { materials.steel_ingot, materials.steel_ingot, materials.steel_ingot }
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:valve_off_empty 2", output = "pipeworks:valve_off_empty 2",
recipe = { recipe = {
{ "", "group:stick", "" }, { "", "group:stick", "" },
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" }, { materials.steel_ingot, materials.steel_ingot, materials.steel_ingot },
{ "", "default:steel_ingot", "" } { "", materials.steel_ingot, "" }
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:storage_tank_0 2", output = "pipeworks:storage_tank_0 2",
recipe = { recipe = {
{ "", "default:steel_ingot", "default:steel_ingot" }, { "", materials.steel_ingot, materials.steel_ingot },
{ "default:steel_ingot", "default:glass", "default:steel_ingot" }, { materials.steel_ingot, materials.glass, materials.steel_ingot },
{ "default:steel_ingot", "default:steel_ingot", "" } { materials.steel_ingot, materials.steel_ingot, "" }
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:grating 2", output = "pipeworks:grating 2",
recipe = { recipe = {
{ "default:steel_ingot", "", "default:steel_ingot" }, { materials.steel_ingot, "", materials.steel_ingot },
{ "", "pipeworks:pipe_1_empty", "" }, { "", "pipeworks:pipe_1_empty", "" },
{ "default:steel_ingot", "", "default:steel_ingot" } { materials.steel_ingot, "", materials.steel_ingot }
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:flow_sensor_empty 2", output = "pipeworks:flow_sensor_empty 2",
recipe = { recipe = {
{ "pipeworks:pipe_1_empty", "mesecons:mesecon", "pipeworks:pipe_1_empty" }, { "pipeworks:pipe_1_empty", "mesecons:mesecon", "pipeworks:pipe_1_empty" },
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:fountainhead 2", output = "pipeworks:fountainhead 2",
recipe = { recipe = {
{ "pipeworks:pipe_1_empty" }, { "pipeworks:pipe_1_empty" },
{ "pipeworks:pipe_1_empty" } { "pipeworks:pipe_1_empty" }
}, },
}) })
-- injectors
core.register_craft( {
output = "pipeworks:filter 2",
recipe = {
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" },
{ "group:stick", materials.mese_crystal, "basic_materials:plastic_sheet" },
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" }
},
})
core.register_craft( {
output = "pipeworks:mese_filter 2",
recipe = {
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" },
{ "group:stick", materials.mese, "basic_materials:plastic_sheet" },
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" }
},
})
if core.get_modpath("digilines") then
core.register_craft( {
output = "pipeworks:digiline_filter 2",
recipe = {
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" },
{ "group:stick", "digilines:wire_std_00000000", "basic_materials:plastic_sheet" },
{ materials.steel_ingot, materials.steel_ingot, "basic_materials:plastic_sheet" }
},
})
end
-- other
core.register_craft( {
output = "pipeworks:autocrafter 2",
recipe = {
{ materials.steel_ingot, materials.mese_crystal, materials.steel_ingot },
{ "basic_materials:plastic_sheet", materials.steel_ingot, "basic_materials:plastic_sheet" },
{ materials.steel_ingot, materials.mese_crystal, materials.steel_ingot }
},
})
core.register_craft( {
output = "pipeworks:steel_pane_embedded_tube 1",
recipe = {
{ "", materials.steel_ingot, "" },
{ "", "pipeworks:tube_1", "" },
{ "", materials.steel_ingot, "" }
},
})
core.register_craft({
output = "pipeworks:trashcan",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ materials.steel_ingot, "", materials.steel_ingot },
{ materials.steel_ingot, materials.steel_ingot, materials.steel_ingot },
},
})
core.register_craft( {
output = "pipeworks:teleport_tube_1 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ materials.desert_stone, materials.mese, materials.desert_stone },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
if pipeworks.enable_priority_tube then
core.register_craft( {
output = "pipeworks:priority_tube_1 6",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ materials.gold_ingot, "", materials.gold_ingot },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end
if pipeworks.enable_accelerator_tube then
core.register_craft( {
output = "pipeworks:accelerator_tube_1 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ materials.mese_crystal_fragment, materials.steel_ingot, materials.mese_crystal_fragment },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end
if pipeworks.enable_crossing_tube then
core.register_craft( {
output = "pipeworks:crossing_tube_1 5",
recipe = {
{ "", "pipeworks:tube_1", "" },
{ "pipeworks:tube_1", "pipeworks:tube_1", "pipeworks:tube_1" },
{ "", "pipeworks:tube_1", "" }
},
})
end
if pipeworks.enable_one_way_tube then
core.register_craft({
output = "pipeworks:one_way_tube 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "group:stick", materials.mese_crystal, "basic_materials:plastic_sheet" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end
if pipeworks.enable_mese_tube then
core.register_craft( {
output = "pipeworks:mese_tube_000000 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "", materials.mese_crystal, "" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
core.register_craft( {
type = "shapeless",
output = "pipeworks:mese_tube_000000",
recipe = {
"pipeworks:tube_1",
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
},
})
end
if pipeworks.enable_item_tags and pipeworks.enable_tag_tube then
core.register_craft( {
output = "pipeworks:tag_tube_000000 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ materials.book, materials.mese_crystal, materials.book },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
core.register_craft( {
type = "shapeless",
output = "pipeworks:tag_tube_000000",
recipe = {
"pipeworks:mese_tube_000000",
materials.book,
},
})
end
if pipeworks.enable_sand_tube then
core.register_craft( {
output = "pipeworks:sand_tube_1 2",
recipe = {
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet"},
{"group:sand", "group:sand", "group:sand"},
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet"}
},
})
core.register_craft( {
output = "pipeworks:sand_tube_1",
recipe = {
{"group:sand", "pipeworks:tube_1", "group:sand"},
},
})
end
if pipeworks.enable_mese_sand_tube then
core.register_craft( {
output = "pipeworks:mese_sand_tube_1 2",
recipe = {
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{"group:sand", materials.mese_crystal, "group:sand" },
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
core.register_craft( {
type = "shapeless",
output = "pipeworks:mese_sand_tube_1",
recipe = {
"pipeworks:sand_tube_1",
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
materials.mese_crystal_fragment,
},
})
end
if pipeworks.enable_deployer then
core.register_craft({
output = "pipeworks:deployer_off",
recipe = {
{ "group:wood", materials.chest, "group:wood" },
{ materials.stone, "mesecons:piston", materials.stone },
{ materials.stone, "mesecons:mesecon", materials.stone },
}
})
end
if pipeworks.enable_dispenser then
core.register_craft({
output = "pipeworks:dispenser_off",
recipe = {
{ materials.desert_sand, materials.chest, materials.desert_sand },
{ materials.stone, "mesecons:piston", materials.stone },
{ materials.stone, "mesecons:mesecon", materials.stone },
}
})
end
if pipeworks.enable_node_breaker then
core.register_craft({
output = "pipeworks:nodebreaker_off",
recipe = {
{ "basic_materials:gear_steel", "basic_materials:gear_steel", "basic_materials:gear_steel" },
{ materials.stone, "mesecons:piston", materials.stone },
{ "group:wood", "mesecons:mesecon", "group:wood" },
}
})
end

View File

@@ -1,94 +0,0 @@
local S = minetest.get_translator("pipeworks")
local straight = function(pos, node, velocity, stack) return {velocity} end
minetest.register_node("pipeworks:steel_block_embedded_tube", {
description = S("Airtight steelblock embedded tube"),
tiles = {
"default_steel_block.png", "default_steel_block.png",
"default_steel_block.png", "default_steel_block.png",
"default_steel_block.png^pipeworks_tube_connection_metallic.png",
"default_steel_block.png^pipeworks_tube_connection_metallic.png",
},
paramtype = "light",
paramtype2 = "facedir",
groups = {cracky=1, oddly_breakable_by_hand = 1, tubedevice = 1},
legacy_facedir_simple = true,
sounds = default.node_sound_stone_defaults(),
tube = {
connect_sides = {front = 1, back = 1,},
priority = 50,
can_go = straight,
can_insert = function(pos, node, stack, direction)
local dir = minetest.facedir_to_dir(node.param2)
return vector.equals(dir, direction) or vector.equals(vector.multiply(dir, -1), direction)
end,
},
after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate,
})
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:steel_block_embedded_tube"
minetest.register_craft( {
output = "pipeworks:steel_block_embedded_tube 1",
recipe = {
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" },
{ "default:steel_ingot", "pipeworks:tube_1", "default:steel_ingot" },
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" }
},
})
local pane_box = {
type = "fixed",
fixed = {
{ -9/64, -9/64, -8/16, 9/64, 9/64, 8/16 }, -- tube
{ -8/16, -8/16, -1/16, 8/16, 8/16, 1/16 } -- pane
}
}
local texture_alpha_mode = minetest.features.use_texture_alpha_string_modes
and "clip" or true
minetest.register_node("pipeworks:steel_pane_embedded_tube", {
drawtype = "nodebox",
description = S("Airtight panel embedded tube"),
tiles = {
"pipeworks_pane_embedded_tube_sides.png^[transformR90",
"pipeworks_pane_embedded_tube_sides.png^[transformR90",
"pipeworks_pane_embedded_tube_sides.png",
"pipeworks_pane_embedded_tube_sides.png",
"pipeworks_pane_embedded_tube_ends.png", "pipeworks_pane_embedded_tube_ends.png",
},
use_texture_alpha = texture_alpha_mode,
node_box = pane_box,
selection_box = pane_box,
collision_box = pane_box,
paramtype = "light",
paramtype2 = "facedir",
groups = {cracky=1, oddly_breakable_by_hand = 1, tubedevice = 1},
legacy_facedir_simple = true,
sounds = default.node_sound_stone_defaults(),
tube = {
connect_sides = {front = 1, back = 1,},
priority = 50,
can_go = straight,
can_insert = function(pos, node, stack, direction)
local dir = minetest.facedir_to_dir(node.param2)
return vector.equals(dir, direction) or vector.equals(vector.multiply(dir, -1), direction)
end,
},
after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate,
})
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:steel_pane_embedded_tube"
minetest.register_craft( {
output = "pipeworks:steel_pane_embedded_tube 1",
recipe = {
{ "", "default:steel_ingot", "" },
{ "", "pipeworks:tube_1", "" },
{ "", "default:steel_ingot", "" }
},
})

View File

@@ -4,6 +4,8 @@ local prefix = "pipeworks_"
local settings = { local settings = {
enable_pipes = true, enable_pipes = true,
enable_item_tags = true,
enable_tag_tube = true,
enable_lowpoly = false, enable_lowpoly = false,
enable_autocrafter = true, enable_autocrafter = true,
enable_deployer = true, enable_deployer = true,
@@ -27,6 +29,9 @@ local settings = {
enable_cyclic_mode = true, enable_cyclic_mode = true,
drop_on_routing_fail = false, drop_on_routing_fail = false,
delete_item_on_clearobject = true, delete_item_on_clearobject = true,
use_real_entities = true,
entity_update_interval = 0,
enable_vertical_digilines_connectivity = false,
} }
pipeworks.toggles = {} pipeworks.toggles = {}
@@ -62,10 +67,9 @@ pipeworks.toggles.finite_water = nil
for name, value in pairs(settings) do for name, value in pairs(settings) do
local setting_type = type(value) local setting_type = type(value)
if setting_type == "boolean" then if setting_type == "boolean" then
pipeworks[name] = minetest.settings:get_bool(prefix..name) pipeworks[name] = core.settings:get_bool(prefix..name, value)
if pipeworks[name] == nil then elseif setting_type == "number" then
pipeworks[name] = value pipeworks[name] = tonumber(core.settings:get(prefix..name) or value)
end
else else
pipeworks[name] = value pipeworks[name] = value
end end

View File

@@ -1,7 +1,7 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local new_flow_logic_register = pipeworks.flowables.register local new_flow_logic_register = pipeworks.flowables.register
local texture_alpha_mode = minetest.features.use_texture_alpha_string_modes local texture_alpha_mode = core.features.use_texture_alpha_string_modes
local polys = "" local polys = ""
if pipeworks.enable_lowpoly then polys = "_lowpoly" end if pipeworks.enable_lowpoly then polys = "_lowpoly" end
@@ -13,7 +13,7 @@ function pipeworks.fix_after_rotation(pos, node, user, mode, new_param2)
if string.find(node.name, "spigot") then new_param2 = new_param2 % 4 end if string.find(node.name, "spigot") then new_param2 = new_param2 % 4 end
local newnode = string.gsub(node.name, "_on", "_off") local newnode = string.gsub(node.name, "_on", "_off")
minetest.swap_node(pos, { name = newnode, param2 = new_param2 }) core.swap_node(pos, { name = newnode, param2 = new_param2 })
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
return true return true
@@ -22,24 +22,24 @@ end
function pipeworks.rotate_on_place(itemstack, placer, pointed_thing) function pipeworks.rotate_on_place(itemstack, placer, pointed_thing)
local playername = placer:get_player_name() local playername = placer:get_player_name()
if not minetest.is_protected(pointed_thing.under, playername) if not core.is_protected(pointed_thing.under, playername)
and not minetest.is_protected(pointed_thing.above, playername) then and not core.is_protected(pointed_thing.above, playername) then
local node = minetest.get_node(pointed_thing.under) local node = core.get_node(pointed_thing.under)
if (not placer:get_player_control().sneak) if (not placer:get_player_control().sneak)
and minetest.registered_nodes[node.name] and core.registered_nodes[node.name]
and minetest.registered_nodes[node.name].on_rightclick then and core.registered_nodes[node.name].on_rightclick then
minetest.registered_nodes[node.name].on_rightclick(pointed_thing.under, core.registered_nodes[node.name].on_rightclick(pointed_thing.under,
node, placer, itemstack, pointed_thing) node, placer, itemstack, pointed_thing)
else else
local pitch = placer:get_look_pitch() local pitch = -placer:get_look_vertical()
local above = pointed_thing.above local above = pointed_thing.above
local under = pointed_thing.under local under = pointed_thing.under
local fdir = minetest.dir_to_facedir(placer:get_look_dir()) local fdir = core.dir_to_facedir(placer:get_look_dir())
local undernode = minetest.get_node(under) local undernode = core.get_node(under)
local uname = undernode.name local uname = undernode.name
local isabove = (above.x == under.x) and (above.z == under.z) and (pitch > 0) local isabove = (above.x == under.x) and (above.z == under.z) and (pitch > 0)
local pos1 = above local pos1 = above
@@ -58,24 +58,24 @@ function pipeworks.rotate_on_place(itemstack, placer, pointed_thing)
( string.find(uname, "pipeworks:valve") ( string.find(uname, "pipeworks:valve")
or string.find(uname, "pipeworks:entry_panel") or string.find(uname, "pipeworks:entry_panel")
or string.find(uname, "pipeworks:flow_sensor") ) or string.find(uname, "pipeworks:flow_sensor") )
and minetest.facedir_to_dir(undernode.param2).y ~= 0 ) and core.facedir_to_dir(undernode.param2).y ~= 0 )
) )
then then
fdir = 17 fdir = 17
end end
if minetest.registered_nodes[uname] if core.registered_nodes[uname]
and minetest.registered_nodes[uname]["buildable_to"] then and core.registered_nodes[uname]["buildable_to"] then
pos1 = under pos1 = under
end end
if minetest.registered_nodes[minetest.get_node(pos1).name] if core.registered_nodes[core.get_node(pos1).name]
and not minetest.registered_nodes[minetest.get_node(pos1).name]["buildable_to"] then return end and not core.registered_nodes[core.get_node(pos1).name]["buildable_to"] then return end
local placednode = string.gsub(itemstack:get_name(), "_loaded", "_empty") local placednode = string.gsub(itemstack:get_name(), "_loaded", "_empty")
placednode = string.gsub(placednode, "_on", "_off") placednode = string.gsub(placednode, "_on", "_off")
minetest.swap_node(pos1, {name = placednode, param2 = fdir }) core.swap_node(pos1, {name = placednode, param2 = fdir })
pipeworks.scan_for_pipe_objects(pos1) pipeworks.scan_for_pipe_objects(pos1)
if not pipeworks.expect_infinite_stacks then if not pipeworks.expect_infinite_stacks then
@@ -91,7 +91,7 @@ end
local pipereceptor_on = nil local pipereceptor_on = nil
local pipereceptor_off = nil local pipereceptor_off = nil
if minetest.get_modpath("mesecons") then if core.get_modpath("mesecons") then
pipereceptor_on = { pipereceptor_on = {
receptor = { receptor = {
state = mesecon.state.on, state = mesecon.state.on,
@@ -133,13 +133,13 @@ for s in ipairs(states) do
local dgroups local dgroups
if states[s] == "off" then if states[s] == "off" then
dgroups = {snappy=3, pipe=1} dgroups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
else else
dgroups = {snappy=3, pipe=1, not_in_creative_inventory=1} dgroups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
end end
local pumpname = "pipeworks:pump_"..states[s] local pumpname = "pipeworks:pump_"..states[s]
minetest.register_node(pumpname, { core.register_node(pumpname, {
description = S("Pump/Intake Module"), description = S("Pump/Intake Module"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_pump"..polys..".obj", mesh = "pipeworks_pump"..polys..".obj",
@@ -148,7 +148,11 @@ for s in ipairs(states) do
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = dgroups, groups = dgroups,
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { top = 1 }, pipe_connections = { top = 1 },
after_place_node = function(pos) after_place_node = function(pos)
@@ -160,17 +164,17 @@ for s in ipairs(states) do
drop = "pipeworks:pump_off", drop = "pipeworks:pump_off",
mesecons = {effector = { mesecons = {effector = {
action_on = function (pos, node) action_on = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:pump_on", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:pump_on", param2 = node.param2})
end, end,
action_off = function (pos, node) action_off = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:pump_off", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:pump_off", param2 = node.param2})
end end
}}, }},
on_rightclick = function(pos, node, clicker, itemstack, pointed_thing) on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
local fdir = node.param2 local fdir = node.param2
minetest.swap_node(pos, { name = "pipeworks:pump_"..states[3-s], param2 = fdir }) core.swap_node(pos, { name = "pipeworks:pump_"..states[3-s], param2 = fdir })
end, end,
on_rotate = screwdriver.rotate_simple on_rotate = screwdriver and screwdriver.rotate_simple or nil
}) })
-- FIXME: this currently assumes that pumps can only rotate around the fixed axis pointing Y+. -- FIXME: this currently assumes that pumps can only rotate around the fixed axis pointing Y+.
@@ -183,7 +187,7 @@ for s in ipairs(states) do
local nodename_valve_empty = "pipeworks:valve_"..states[s].."_empty" local nodename_valve_empty = "pipeworks:valve_"..states[s].."_empty"
minetest.register_node(nodename_valve_empty, { core.register_node(nodename_valve_empty, {
description = S("Valve"), description = S("Valve"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_valve_"..states[s]..polys..".obj", mesh = "pipeworks_valve_"..states[s]..polys..".obj",
@@ -200,7 +204,11 @@ for s in ipairs(states) do
fixed = { -5/16, -4/16, -8/16, 5/16, 5/16, 8/16 } fixed = { -5/16, -4/16, -8/16, 5/16, 5/16, 8/16 }
}, },
groups = dgroups, groups = dgroups,
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -209,15 +217,15 @@ for s in ipairs(states) do
drop = "pipeworks:valve_off_empty", drop = "pipeworks:valve_off_empty",
mesecons = {effector = { mesecons = {effector = {
action_on = function (pos, node) action_on = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:valve_on_empty", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:valve_on_empty", param2 = node.param2})
end, end,
action_off = function (pos, node) action_off = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:valve_off_empty", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:valve_off_empty", param2 = node.param2})
end end
}}, }},
on_rightclick = function(pos, node, clicker, itemstack, pointed_thing) on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
local fdir = node.param2 local fdir = node.param2
minetest.swap_node(pos, { name = "pipeworks:valve_"..states[3-s].."_empty", param2 = fdir }) core.swap_node(pos, { name = "pipeworks:valve_"..states[3-s].."_empty", param2 = fdir })
end, end,
on_rotate = pipeworks.fix_after_rotation on_rotate = pipeworks.fix_after_rotation
}) })
@@ -231,7 +239,7 @@ pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:pump_off"
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:valve_off_empty" pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:valve_off_empty"
local nodename_valve_loaded = "pipeworks:valve_on_loaded" local nodename_valve_loaded = "pipeworks:valve_on_loaded"
minetest.register_node(nodename_valve_loaded, { core.register_node(nodename_valve_loaded, {
description = S("Valve"), description = S("Valve"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_valve_on"..polys..".obj", mesh = "pipeworks_valve_on"..polys..".obj",
@@ -247,8 +255,12 @@ minetest.register_node(nodename_valve_loaded, {
type = "fixed", type = "fixed",
fixed = { -5/16, -4/16, -8/16, 5/16, 5/16, 8/16 } fixed = { -5/16, -4/16, -8/16, 5/16, 5/16, 8/16 }
}, },
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -257,15 +269,15 @@ minetest.register_node(nodename_valve_loaded, {
drop = "pipeworks:valve_off_empty", drop = "pipeworks:valve_off_empty",
mesecons = {effector = { mesecons = {effector = {
action_on = function (pos, node) action_on = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:valve_on_empty", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:valve_on_empty", param2 = node.param2})
end, end,
action_off = function (pos, node) action_off = function (pos, node)
minetest.swap_node(pos,{name="pipeworks:valve_off_empty", param2 = node.param2}) core.swap_node(pos,{name="pipeworks:valve_off_empty", param2 = node.param2})
end end
}}, }},
on_rightclick = function(pos, node, clicker, itemstack, pointed_thing) on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
local fdir = node.param2 local fdir = node.param2
minetest.swap_node(pos, { name = "pipeworks:valve_off_empty", param2 = fdir }) core.swap_node(pos, { name = "pipeworks:valve_off_empty", param2 = fdir })
end, end,
on_rotate = pipeworks.fix_after_rotation on_rotate = pipeworks.fix_after_rotation
}) })
@@ -279,7 +291,7 @@ new_flow_logic_register.directional_horizonal_rotate(nodename_valve_loaded, true
-- grating -- grating
-- FIXME: should this do anything useful in the new flow logic? -- FIXME: should this do anything useful in the new flow logic?
minetest.register_node("pipeworks:grating", { core.register_node("pipeworks:grating", {
description = S("Decorative grating"), description = S("Decorative grating"),
tiles = { tiles = {
"pipeworks_grating_top.png", "pipeworks_grating_top.png",
@@ -297,8 +309,12 @@ minetest.register_node("pipeworks:grating", {
}, },
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { top = 1 }, pipe_connections = { top = 1 },
after_place_node = function(pos) after_place_node = function(pos)
@@ -314,7 +330,7 @@ pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:grating"
-- outlet spigot -- outlet spigot
local nodename_spigot_empty = "pipeworks:spigot" local nodename_spigot_empty = "pipeworks:spigot"
minetest.register_node(nodename_spigot_empty, { core.register_node(nodename_spigot_empty, {
description = S("Spigot outlet"), description = S("Spigot outlet"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_spigot"..polys..".obj", mesh = "pipeworks_spigot"..polys..".obj",
@@ -322,8 +338,12 @@ minetest.register_node(nodename_spigot_empty, {
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { left=1, right=1, front=1, back=1, pipe_connections = { left=1, right=1, front=1, back=1,
left_param2 = 3, right_param2 = 1, front_param2 = 2, back_param2 = 0 }, left_param2 = 3, right_param2 = 1, front_param2 = 2, back_param2 = 0 },
@@ -345,33 +365,29 @@ minetest.register_node(nodename_spigot_empty, {
}) })
local nodename_spigot_loaded = "pipeworks:spigot_pouring" local nodename_spigot_loaded = "pipeworks:spigot_pouring"
minetest.register_node(nodename_spigot_loaded, { core.register_node(nodename_spigot_loaded, {
description = S("Spigot outlet"), description = S("Spigot outlet"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_spigot_pouring"..polys..".obj", mesh = "pipeworks_spigot_pouring"..polys..".obj",
tiles = { tiles = {
{ core.registered_nodes[pipeworks.liquids.water.source].tiles[1],
name = "default_water_flowing_animated.png",
animation = {
type = "vertical_frames",
aspect_w = 16,
aspect_h = 16,
length = 0.8,
},
},
{ name = "pipeworks_spigot.png" } { name = "pipeworks_spigot.png" }
}, },
use_texture_alpha = texture_alpha_mode and "blend" or true, use_texture_alpha = texture_alpha_mode and "blend" or true,
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { left=1, right=1, front=1, back=1, pipe_connections = { left=1, right=1, front=1, back=1,
left_param2 = 3, right_param2 = 1, front_param2 = 2, back_param2 = 0 }, left_param2 = 3, right_param2 = 1, front_param2 = 2, back_param2 = 0 },
after_place_node = function(pos) after_place_node = function(pos)
minetest.set_node(pos, { name = "pipeworks:spigot", param2 = minetest.get_node(pos).param2 }) core.set_node(pos, { name = "pipeworks:spigot", param2 = core.get_node(pos).param2 })
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
end, end,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -414,15 +430,19 @@ local panel_cbox = {
} }
local nodename_panel_empty = "pipeworks:entry_panel_empty" local nodename_panel_empty = "pipeworks:entry_panel_empty"
minetest.register_node(nodename_panel_empty, { core.register_node(nodename_panel_empty, {
description = S("Airtight Pipe entry/exit"), description = S("Airtight Pipe entry/exit"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_entry_panel"..polys..".obj", mesh = "pipeworks_entry_panel"..polys..".obj",
tiles = { "pipeworks_entry_panel.png" }, tiles = { "pipeworks_entry_panel.png" },
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -434,15 +454,19 @@ minetest.register_node(nodename_panel_empty, {
}) })
local nodename_panel_loaded = "pipeworks:entry_panel_loaded" local nodename_panel_loaded = "pipeworks:entry_panel_loaded"
minetest.register_node(nodename_panel_loaded, { core.register_node(nodename_panel_loaded, {
description = S("Airtight Pipe entry/exit"), description = S("Airtight Pipe entry/exit"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_entry_panel"..polys..".obj", mesh = "pipeworks_entry_panel"..polys..".obj",
tiles = { "pipeworks_entry_panel.png" }, tiles = { "pipeworks_entry_panel.png" },
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -463,7 +487,7 @@ new_flow_logic_register.directional_horizonal_rotate(nodename_panel_loaded, true
local nodename_sensor_empty = "pipeworks:flow_sensor_empty" local nodename_sensor_empty = "pipeworks:flow_sensor_empty"
minetest.register_node(nodename_sensor_empty, { core.register_node(nodename_sensor_empty, {
description = S("Flow Sensor"), description = S("Flow Sensor"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_flow_sensor"..polys..".obj", mesh = "pipeworks_flow_sensor"..polys..".obj",
@@ -471,8 +495,12 @@ minetest.register_node(nodename_sensor_empty, {
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -502,7 +530,7 @@ minetest.register_node(nodename_sensor_empty, {
}) })
local nodename_sensor_loaded = "pipeworks:flow_sensor_loaded" local nodename_sensor_loaded = "pipeworks:flow_sensor_loaded"
minetest.register_node(nodename_sensor_loaded, { core.register_node(nodename_sensor_loaded, {
description = S("Flow sensor (on)"), description = S("Flow sensor (on)"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_flow_sensor"..polys..".obj", mesh = "pipeworks_flow_sensor"..polys..".obj",
@@ -510,8 +538,12 @@ minetest.register_node(nodename_sensor_loaded, {
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -555,16 +587,16 @@ new_flow_logic_register.transition_simple_set(sensor_pressure_set, { mesecons=pi
-- TODO flow-logic-stub: these don't currently do anything under the new flow logic. -- TODO flow-logic-stub: these don't currently do anything under the new flow logic.
for fill = 0, 10 do for fill = 0, 10 do
local filldesc=S("empty") local filldesc=S("empty")
local sgroups = {snappy=3, pipe=1, tankfill=fill+1} local sgroups = {snappy=3, pipe=1, tankfill=fill+1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
local image = nil local image = nil
if fill ~= 0 then if fill ~= 0 then
filldesc=S("@1% full", 10*fill) filldesc=S("@1% full", 10*fill)
sgroups = {snappy=3, pipe=1, tankfill=fill+1, not_in_creative_inventory=1} sgroups = {snappy=3, pipe=1, tankfill=fill+1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
image = "pipeworks_storage_tank_fittings.png" image = "pipeworks_storage_tank_fittings.png"
end end
minetest.register_node("pipeworks:expansion_tank_"..fill, { core.register_node("pipeworks:expansion_tank_"..fill, {
description = S("Expansion Tank (@1)", filldesc), description = S("Expansion Tank (@1)", filldesc),
tiles = { tiles = {
"pipeworks_storage_tank_fittings.png", "pipeworks_storage_tank_fittings.png",
@@ -577,8 +609,12 @@ for fill = 0, 10 do
inventory_image = image, inventory_image = image,
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1, tankfill=fill+1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, tankfill=fill+1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
drop = "pipeworks:storage_tank_0", drop = "pipeworks:storage_tank_0",
pipe_connections = { top = 1, bottom = 1}, pipe_connections = { top = 1, bottom = 1},
@@ -592,7 +628,7 @@ for fill = 0, 10 do
on_rotate = false on_rotate = false
}) })
minetest.register_node("pipeworks:storage_tank_"..fill, { core.register_node("pipeworks:storage_tank_"..fill, {
description = S("Fluid Storage Tank (@1)", filldesc), description = S("Fluid Storage Tank (@1)", filldesc),
tiles = { tiles = {
"pipeworks_storage_tank_fittings.png", "pipeworks_storage_tank_fittings.png",
@@ -606,7 +642,11 @@ for fill = 0, 10 do
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = sgroups, groups = sgroups,
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
drop = "pipeworks:storage_tank_0", drop = "pipeworks:storage_tank_0",
pipe_connections = { top = 1, bottom = 1}, pipe_connections = { top = 1, bottom = 1},
@@ -625,15 +665,19 @@ pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:storage_t
-- fountainhead -- fountainhead
local nodename_fountain_empty = "pipeworks:fountainhead" local nodename_fountain_empty = "pipeworks:fountainhead"
minetest.register_node(nodename_fountain_empty, { core.register_node(nodename_fountain_empty, {
description = S("Fountainhead"), description = S("Fountainhead"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_fountainhead"..polys..".obj", mesh = "pipeworks_fountainhead"..polys..".obj",
tiles = { "pipeworks_fountainhead.png" }, tiles = { "pipeworks_fountainhead.png" },
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { bottom = 1 }, pipe_connections = { bottom = 1 },
after_place_node = function(pos) after_place_node = function(pos)
@@ -660,19 +704,23 @@ minetest.register_node(nodename_fountain_empty, {
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:fountainhead" pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:fountainhead"
local nodename_fountain_loaded = "pipeworks:fountainhead_pouring" local nodename_fountain_loaded = "pipeworks:fountainhead_pouring"
minetest.register_node(nodename_fountain_loaded, { core.register_node(nodename_fountain_loaded, {
description = S("Fountainhead"), description = S("Fountainhead"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_fountainhead"..polys..".obj", mesh = "pipeworks_fountainhead"..polys..".obj",
tiles = { "pipeworks_fountainhead.png" }, tiles = { "pipeworks_fountainhead.png" },
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
pipe_connections = { bottom = 1 }, pipe_connections = { bottom = 1 },
after_place_node = function(pos) after_place_node = function(pos)
minetest.set_node(pos, { name = "pipeworks:fountainhead", param2 = minetest.get_node(pos).param2 }) core.set_node(pos, { name = "pipeworks:fountainhead", param2 = core.get_node(pos).param2 })
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
end, end,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -710,15 +758,19 @@ local sp_cbox = {
} }
local nodename_sp_empty = "pipeworks:straight_pipe_empty" local nodename_sp_empty = "pipeworks:straight_pipe_empty"
minetest.register_node(nodename_sp_empty, { core.register_node(nodename_sp_empty, {
description = S("Straight-only Pipe"), description = S("Straight-only Pipe"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_straight_pipe"..polys..".obj", mesh = "pipeworks_straight_pipe"..polys..".obj",
tiles = { "pipeworks_straight_pipe_empty.png" }, tiles = { "pipeworks_straight_pipe_empty.png" },
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1}, groups = {snappy=3, pipe=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -732,15 +784,19 @@ minetest.register_node(nodename_sp_empty, {
}) })
local nodename_sp_loaded = "pipeworks:straight_pipe_loaded" local nodename_sp_loaded = "pipeworks:straight_pipe_loaded"
minetest.register_node(nodename_sp_loaded, { core.register_node(nodename_sp_loaded, {
description = S("Straight-only Pipe"), description = S("Straight-only Pipe"),
drawtype = "mesh", drawtype = "mesh",
mesh = "pipeworks_straight_pipe"..polys..".obj", mesh = "pipeworks_straight_pipe"..polys..".obj",
tiles = { "pipeworks_straight_pipe_loaded.png" }, tiles = { "pipeworks_straight_pipe_loaded.png" },
paramtype = "light", paramtype = "light",
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy=3, pipe=1, not_in_creative_inventory=1}, groups = {snappy=3, pipe=1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
on_place = pipeworks.rotate_on_place, on_place = pipeworks.rotate_on_place,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -760,6 +816,6 @@ new_flow_logic_register.directional_horizonal_rotate(nodename_sp_loaded, true)
-- Other misc stuff -- Other misc stuff
minetest.register_alias("pipeworks:valve_off_loaded", "pipeworks:valve_off_empty") core.register_alias("pipeworks:valve_off_loaded", "pipeworks:valve_off_empty")
minetest.register_alias("pipeworks:entry_panel", "pipeworks:entry_panel_empty") core.register_alias("pipeworks:entry_panel", "pipeworks:entry_panel_empty")

View File

@@ -1,4 +1,4 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local fs_helpers = pipeworks.fs_helpers local fs_helpers = pipeworks.fs_helpers
local function set_filter_infotext(data, meta) local function set_filter_infotext(data, meta)
@@ -14,59 +14,84 @@ local function set_filter_formspec(data, meta)
local formspec local formspec
if data.digiline then if data.digiline then
local form_height = 3
if pipeworks.enable_item_tags then
form_height = 4
end
formspec = formspec =
"size[8.5,3]".. ("size[8.5,%f]"):format(form_height) ..
"item_image[0.2,0;1,1;pipeworks:"..data.name.."]".. "item_image[0.2,0;1,1;pipeworks:"..data.name.."]"..
"label[1.2,0.2;"..minetest.formspec_escape(itemname).."]".. "label[1.2,0.2;"..core.formspec_escape(itemname).."]"..
"field[0.5,1.6;4.6,1;channel;"..S("Channel")..";${channel}]".. "field[0.5,1.6;4.6,1;channel;"..S("Channel")..";${channel}]"..
"button[4.8,1.3;1.5,1;set_channel;"..S("Set").."]".. "button[4.8,1.3;1.5,1;set_channel;"..S("Set").."]"..
fs_helpers.cycling_button(meta, "button[0.2,2.3;4.05,1", "slotseq_mode", fs_helpers.cycling_button(meta, ("button[0.2,%f;4.05,1"):format(form_height - 0.7), "slotseq_mode",
{S("Sequence slots by Priority"), {S("Sequence slots by Priority"),
S("Sequence slots Randomly"), S("Sequence slots Randomly"),
S("Sequence slots by Rotation")}).. S("Sequence slots by Rotation")})..
fs_helpers.cycling_button(meta, "button[4.25,2.3;4.05,1", "exmatch_mode", fs_helpers.cycling_button(meta, ("button[4.25,%f;4.05,1"):format(form_height - 0.7), "exmatch_mode",
{S("Exact match - off"), {S("Exact match - off"),
S("Exact match - on")}).. S("Exact match - on")})..
"button_exit[6.3,1.3;2,1;close;"..S("Close").."]" ("button_exit[6.3,%f;2,1;close;" .. S("Close") .. "]"):format(form_height - 1.7)
if pipeworks.enable_item_tags then
formspec = formspec ..
("field[0.5,%f;4.6,1;item_tags;"):format(form_height - 1.4) .. S("Item Tags") .. ";${item_tags}]" ..
("button[4.8,%f;1.5,1;set_item_tags;"):format(form_height - 1.7) .. S("Set") .. "]"
end
else else
local exmatch_button = "" local exmatch_button = ""
if data.stackwise then if data.stackwise then
exmatch_button = exmatch_button =
fs_helpers.cycling_button(meta, "button[4,3.5;4,1", "exmatch_mode", fs_helpers.cycling_button(meta, "button["..(10.2-(0.22)-4)..",4.5;4,1", "exmatch_mode",
{S("Exact match - off"), {S("Exact match - off"),
S("Exact match - on")}) S("Exact match - on")})
end end
local size = "10.2,11"
formspec = "size[8,8.5]".. local list_backgrounds = ""
"item_image[0,0;1,1;pipeworks:"..data.name.."]".. if core.get_modpath("i3") or core.get_modpath("mcl_formspec") then
"label[1,0;"..minetest.formspec_escape(itemname).."]".. list_backgrounds = "style_type[box;colors=#666]"
"label[0,1;"..S("Prefer item types:").."]".. for i=0, 7 do
"list[context;main;0,1.5;8,2;]".. for j=0, 1 do
fs_helpers.cycling_button(meta, "button[0,3.5;4,1", "slotseq_mode", list_backgrounds = list_backgrounds .. "box[".. 0.22+(i*1.25) ..",".. 1.75+(j*1.25) ..";1,1;]"
end
end
end
formspec =
"formspec_version[2]"..
"size["..size.."]"..
pipeworks.fs_helpers.get_prepends(size)..
"item_image[0.22,0.22;1,1;pipeworks:"..data.name.."]"..
"label[1.22,0.72;"..core.formspec_escape(itemname).."]"..
"label[0.22,1.5;"..S("Prefer item types:").."]"..
list_backgrounds..
"list[context;main;0.22,1.75;8,2;]"..
fs_helpers.cycling_button(meta, "button[0.22,4.5;4,1", "slotseq_mode",
{S("Sequence slots by Priority"), {S("Sequence slots by Priority"),
S("Sequence slots Randomly"), S("Sequence slots Randomly"),
S("Sequence slots by Rotation")}).. S("Sequence slots by Rotation")})..
exmatch_button.. exmatch_button..
"list[current_player;main;0,4.5;8,4;]" .. pipeworks.fs_helpers.get_inv(6)..
"listring[]" "listring[]"
if pipeworks.enable_item_tags then
formspec = formspec ..
"field[5.8,0.5;3,0.8;item_tags;" .. S("Item Tags") .. ";${item_tags}]" ..
"button[9,0.3;1,1.1;set_item_tags;" .. S("Set") .. "]"
end
end end
meta:set_string("formspec", formspec) meta:set_string("formspec", formspec)
end end
local function punch_filter(data, filtpos, filtnode, msg) local function punch_filter(data, filtpos, filtnode, msg)
local filtmeta = minetest.get_meta(filtpos) local filtmeta = core.get_meta(filtpos)
local filtinv = filtmeta:get_inventory() local filtinv = filtmeta:get_inventory()
local owner = filtmeta:get_string("owner") local owner = filtmeta:get_string("owner")
local fakePlayer = pipeworks.create_fake_player({ local fakeplayer = fakelib.create_player(owner)
name = owner
})
local dir = pipeworks.facedir_to_right_dir(filtnode.param2) local dir = pipeworks.facedir_to_right_dir(filtnode.param2)
local frompos = vector.subtract(filtpos, dir) local frompos = vector.subtract(filtpos, dir)
local fromnode = minetest.get_node(frompos) local fromnode = core.get_node(frompos)
if not fromnode then return end if not fromnode then return end
local fromdef = minetest.registered_nodes[fromnode.name] local fromdef = core.registered_nodes[fromnode.name]
if not fromdef then return end if not fromdef or not fromdef.tube then return end
local fromtube = fromdef.tube local fromtube = table.copy(fromdef.tube)
local input_special_cases = { local input_special_cases = {
["technic:mv_electric_furnace"] = "dst", ["technic:mv_electric_furnace"] = "dst",
["technic:mv_electric_furnace_active"] = "dst", ["technic:mv_electric_furnace_active"] = "dst",
@@ -92,15 +117,14 @@ local function punch_filter(data, filtpos, filtnode, msg)
} }
-- make sure there's something appropriate to inject the item into -- make sure there's something appropriate to inject the item into
local todir = pipeworks.facedir_to_right_dir(filtnode.param2) local topos = vector.add(filtpos, dir)
local topos = vector.add(filtpos, todir) local tonode = core.get_node(topos)
local tonode = minetest.get_node(topos) local todef = core.registered_nodes[tonode.name]
local todef = minetest.registered_nodes[tonode.name]
if not todef if not todef
or not (minetest.get_item_group(tonode.name, "tube") == 1 or not (core.get_item_group(tonode.name, "tube") == 1
or minetest.get_item_group(tonode.name, "tubedevice") == 1 or core.get_item_group(tonode.name, "tubedevice") == 1
or minetest.get_item_group(tonode.name, "tubedevice_receiver") == 1) then or core.get_item_group(tonode.name, "tubedevice_receiver") == 1) then
return return
end end
@@ -110,6 +134,7 @@ local function punch_filter(data, filtpos, filtnode, msg)
local slotseq_mode local slotseq_mode
local exmatch_mode local exmatch_mode
local item_tags = pipeworks.sanitize_tags(filtmeta:get_string("item_tags"))
local filters = {} local filters = {}
if data.digiline then if data.digiline then
local function add_filter(name, group, count, wear, metadata) local function add_filter(name, group, count, wear, metadata)
@@ -173,6 +198,14 @@ local function punch_filter(data, filtpos, filtnode, msg)
set_filter_formspec(data, filtmeta) set_filter_formspec(data, filtmeta)
end end
if pipeworks.enable_item_tags then
if type(msg.tags) == "table" or type(msg.tags) == "string" then
item_tags = pipeworks.sanitize_tags(msg.tags)
elseif type(msg.tag) == "string" then
item_tags = pipeworks.sanitize_tags({msg.tag})
end
end
if msg.nofire then if msg.nofire then
return return
end end
@@ -218,7 +251,7 @@ local function punch_filter(data, filtpos, filtnode, msg)
return return
end end
else else
local frommeta = minetest.get_meta(frompos) local frommeta = core.get_meta(frompos)
frominv = frommeta:get_inventory() frominv = frommeta:get_inventory()
end end
if fromtube.before_filter then fromtube.before_filter(frompos) end if fromtube.before_filter then fromtube.before_filter(frompos) end
@@ -240,7 +273,7 @@ local function punch_filter(data, filtpos, filtnode, msg)
and (not fgroup -- If there's a group filter, and (not fgroup -- If there's a group filter,
or (type(fgroup) == "string" -- it must be a string or (type(fgroup) == "string" -- it must be a string
and minetest.get_item_group( -- and it must match. and core.get_item_group( -- and it must match.
stack:get_name(), fgroup) ~= 0)) stack:get_name(), fgroup) ~= 0))
and (not fwear -- If there's a wear filter: and (not fwear -- If there's a wear filter:
@@ -280,13 +313,15 @@ local function punch_filter(data, filtpos, filtnode, msg)
return a < b return a < b
end) end)
end end
local available_items = {}
local available_count = 0
for _, spos in ipairs(sposes) do for _, spos in ipairs(sposes) do
local stack = frominv:get_stack(frominvname, spos) local stack = frominv:get_stack(frominvname, spos)
local doRemove = stack:get_count() local doRemove = stack:get_count()
if fromtube.can_remove then if fromtube.can_remove then
doRemove = fromtube.can_remove(frompos, fromnode, stack, dir, frominvname, spos) doRemove = fromtube.can_remove(frompos, fromnode, stack, dir, frominvname, spos)
elseif fromdef.allow_metadata_inventory_take then elseif fromdef.allow_metadata_inventory_take then
doRemove = fromdef.allow_metadata_inventory_take(frompos, frominvname,spos, stack, fakePlayer) doRemove = fromdef.allow_metadata_inventory_take(frompos, frominvname, spos, stack, fakeplayer)
end end
-- stupid lack of continue statements grumble -- stupid lack of continue statements grumble
if doRemove > 0 then if doRemove > 0 then
@@ -298,39 +333,63 @@ local function punch_filter(data, filtpos, filtnode, msg)
filtmeta:set_int("slotseq_index", nextpos) filtmeta:set_int("slotseq_index", nextpos)
set_filter_infotext(data, filtmeta) set_filter_infotext(data, filtmeta)
end end
local item
local count local count
if data.stackwise then if data.stackwise then
count = math.min(stack:get_count(), doRemove) count = math.min(stack:get_count(), doRemove)
if filterfor.count and (filterfor.count > 1 or data.digiline) then if filterfor.count and (filterfor.count > 1 or data.digiline) then
if exmatch_mode ~= 0 and filterfor.count > count then count = math.min(filterfor.count - available_count, count)
return false -- not enough, fail end
else table.insert(available_items, {spos = spos, count = count})
-- limit quantity to filter amount available_count = available_count + count
count = math.min(filterfor.count, count) if not filterfor.count or available_count >= filterfor.count then
end break
end end
else else
count = 1 table.insert(available_items, {spos = spos, count = 1})
available_count = 1
break -- only one item allowed so ignore other stacks
end end
if fromtube.remove_items then
-- it could be the entire stack...
item = fromtube.remove_items(frompos, fromnode, stack, dir, count, frominvname, spos)
else
item = stack:take_item(count)
frominv:set_stack(frominvname, spos, stack)
if fromdef.on_metadata_inventory_take then
fromdef.on_metadata_inventory_take(frompos, frominvname, spos, item, fakePlayer)
end
end
local pos = vector.add(frompos, vector.multiply(dir, 1.4))
local start_pos = vector.add(frompos, dir)
pipeworks.tube_inject_item(pos, start_pos, dir, item,
fakePlayer:get_player_name())
return true -- only fire one item, please
end end
end end
return false if available_count == 0 or (exmatch_mode ~= 0 and filterfor.count and available_count < filterfor.count) then
return false -- not enough, fail
end
local taken_stacks = {}
for _,item in ipairs(available_items) do
local stack = frominv:get_stack(frominvname, item.spos)
local taken
if fromtube.remove_items then
taken = fromtube.remove_items(frompos, fromnode, stack, dir, item.count, frominvname, item.spos)
else
taken = stack:take_item(item.count)
frominv:set_stack(frominvname, item.spos, stack)
if fromdef.on_metadata_inventory_take then
fromdef.on_metadata_inventory_take(frompos, frominvname, item.spos, taken, fakeplayer)
end
end
if not taken:is_empty() then
table.insert(taken_stacks, taken)
end
end
if #taken_stacks > 1 then
-- merge stacks if possible to reduce items in tubes
local merged = {}
for _,a in ipairs(taken_stacks) do
for _,b in ipairs(merged) do
a = b:add_item(a)
end
if not a:is_empty() then
table.insert(merged, a)
end
end
taken_stacks = merged
end
for _,stack in ipairs(taken_stacks) do
local pos = vector.add(frompos, vector.multiply(dir, 1.4))
local start_pos = vector.add(frompos, dir)
pipeworks.tube_inject_item(pos, start_pos, dir, stack, fakeplayer:get_player_name(), item_tags)
end
return true
end end
for _, frominvname in ipairs(type(fromtube.input_inventory) == "table" and fromtube.input_inventory or {fromtube.input_inventory}) do for _, frominvname in ipairs(type(fromtube.input_inventory) == "table" and fromtube.input_inventory or {fromtube.input_inventory}) do
@@ -375,18 +434,22 @@ for _, data in ipairs({
"pipeworks_"..data.name.."_top.png", "pipeworks_"..data.name.."_top.png",
}, },
paramtype2 = "facedir", paramtype2 = "facedir",
groups = {snappy = 2, choppy = 2, oddly_breakable_by_hand = 2, mesecon = 2}, groups = {snappy = 2, choppy = 2, oddly_breakable_by_hand = 2, mesecon = 2, axey=1, handy=1, pickaxey=1},
is_ground_content = false,
_mcl_hardness=0.8,
legacy_facedir_simple = true, legacy_facedir_simple = true,
sounds = default.node_sound_wood_defaults(), _sound_def = {
key = "node_sound_wood_defaults",
},
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
set_filter_formspec(data, meta) set_filter_formspec(data, meta)
set_filter_infotext(data, meta) set_filter_infotext(data, meta)
local inv = meta:get_inventory() local inv = meta:get_inventory()
inv:set_size("main", 8*2) inv:set_size("main", 8*2)
end, end,
after_place_node = function (pos, placer) after_place_node = function (pos, placer)
minetest.get_meta(pos):set_string("owner", placer:get_player_name()) core.get_meta(pos):set_string("owner", placer:get_player_name())
pipeworks.after_place(pos) pipeworks.after_place(pos)
end, end,
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
@@ -395,7 +458,7 @@ for _, data in ipairs({
if not pipeworks.may_configure(pos, player) then if not pipeworks.may_configure(pos, player) then
return 0 return 0
end end
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
inv:set_stack("main", index, stack) inv:set_stack("main", index, stack)
return 0 return 0
end, end,
@@ -403,7 +466,7 @@ for _, data in ipairs({
if not pipeworks.may_configure(pos, player) then if not pipeworks.may_configure(pos, player) then
return 0 return 0
end end
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
local fake_stack = inv:get_stack("main", index) local fake_stack = inv:get_stack("main", index)
fake_stack:take_item(stack:get_count()) fake_stack:take_item(stack:get_count())
inv:set_stack("main", index, fake_stack) inv:set_stack("main", index, fake_stack)
@@ -413,17 +476,12 @@ for _, data in ipairs({
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
return count return count
end, end,
can_dig = function(pos, player)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
return inv:is_empty("main")
end,
tube = {connect_sides = {right = 1}}, tube = {connect_sides = {right = 1}},
} }
if data.digiline then if data.digiline then
node.groups.mesecon = nil node.groups.mesecon = nil
if not minetest.get_modpath("digilines") then if not core.get_modpath("digilines") then
node.groups.not_in_creative_inventory = 1 node.groups.not_in_creative_inventory = 1
end end
@@ -436,18 +494,22 @@ for _, data in ipairs({
fs_helpers.on_receive_fields(pos, fields) fs_helpers.on_receive_fields(pos, fields)
if fields.channel and (fields.key_enter_field == "channel" or fields.set_channel) then if fields.channel and (fields.key_enter_field == "channel" or fields.set_channel) then
minetest.get_meta(pos):set_string("channel", fields.channel) core.get_meta(pos):set_string("channel", fields.channel)
end end
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
if pipeworks.enable_item_tags and fields.item_tags and (fields.key_enter_field == "item_tags" or fields.set_item_tags) then
local tags = pipeworks.sanitize_tags(fields.item_tags)
meta:set_string("item_tags", table.concat(tags, ","))
end
--meta:set_int("slotseq_index", 1) --meta:set_int("slotseq_index", 1)
set_filter_formspec(data, meta) set_filter_formspec(data, meta)
set_filter_infotext(data, meta) set_filter_infotext(data, meta)
end end
node.digiline = { node.digilines = {
effector = { effector = {
action = function(pos, node, channel, msg) action = function(pos, node, channel, msg)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local setchan = meta:get_string("channel") local setchan = meta:get_string("channel")
if setchan ~= channel then return end if setchan ~= channel then return end
@@ -459,8 +521,12 @@ for _, data in ipairs({
node.on_receive_fields = function(pos, formname, fields, sender) node.on_receive_fields = function(pos, formname, fields, sender)
if not pipeworks.may_configure(pos, sender) then return end if not pipeworks.may_configure(pos, sender) then return end
fs_helpers.on_receive_fields(pos, fields) fs_helpers.on_receive_fields(pos, fields)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
meta:set_int("slotseq_index", 1) meta:set_int("slotseq_index", 1)
if pipeworks.enable_item_tags and fields.item_tags and (fields.key_enter_field == "item_tags" or fields.set_item_tags) then
local tags = pipeworks.sanitize_tags(fields.item_tags)
meta:set_string("item_tags", table.concat(tags, ","))
end
set_filter_formspec(data, meta) set_filter_formspec(data, meta)
set_filter_infotext(data, meta) set_filter_infotext(data, meta)
end end
@@ -478,39 +544,10 @@ for _, data in ipairs({
minetest.register_node("pipeworks:"..data.name, node) core.register_node("pipeworks:"..data.name, node)
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:"..data.name pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:"..data.name
end end
minetest.register_craft( {
output = "pipeworks:filter 2",
recipe = {
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" },
{ "group:stick", "default:mese_crystal", "basic_materials:plastic_sheet" },
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" }
},
})
minetest.register_craft( {
output = "pipeworks:mese_filter 2",
recipe = {
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" },
{ "group:stick", "default:mese", "basic_materials:plastic_sheet" },
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" }
},
})
if minetest.get_modpath("digilines") then
minetest.register_craft( {
output = "pipeworks:digiline_filter 2",
recipe = {
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" },
{ "group:stick", "digilines:wire_std_00000000", "basic_materials:plastic_sheet" },
{ "default:steel_ingot", "default:steel_ingot", "basic_materials:plastic_sheet" }
},
})
end
--[[ --[[
In the past the filter-injectors had real items in their inventories. This code In the past the filter-injectors had real items in their inventories. This code
puts them to the input to the filter-injector if possible. Else the items are puts them to the input to the filter-injector if possible. Else the items are
@@ -519,11 +556,12 @@ dropped.
local function put_to_inputinv(pos, node, filtmeta, list) local function put_to_inputinv(pos, node, filtmeta, list)
local dir = pipeworks.facedir_to_right_dir(node.param2) local dir = pipeworks.facedir_to_right_dir(node.param2)
local frompos = vector.subtract(pos, dir) local frompos = vector.subtract(pos, dir)
local fromnode = minetest.get_node(frompos) local fromnode = core.get_node(frompos)
local fromdef = minetest.registered_nodes[fromnode.name] local fromdef = core.registered_nodes[fromnode.name]
if not fromdef or not fromdef.tube then if not (fromdef and fromdef.tube) then
return return
end end
local fromtube = fromdef.tube local fromtube = fromdef.tube
local frominv local frominv
if fromtube.return_input_invref then if fromtube.return_input_invref then
@@ -533,41 +571,48 @@ local function put_to_inputinv(pos, node, filtmeta, list)
return return
end end
else else
frominv = minetest.get_meta(frompos):get_inventory() frominv = core.get_meta(frompos):get_inventory()
end end
local listname = type(fromtube.input_inventory) == "table" and local listname = type(fromtube.input_inventory) == "table" and
fromtube.input_inventory[1] or fromtube.input_inventory fromtube.input_inventory[1] or fromtube.input_inventory
if not listname then if not listname then
return return
end end
for i = 1, #list do for i = 1, #list do
local item = list[i] local item = list[i]
if not item:is_empty() then if not item:is_empty() then
local leftover = frominv:add_item(listname, item) local leftover = frominv:add_item(listname, item)
if not leftover:is_empty() then if not leftover:is_empty() then
minetest.add_item(pos, leftover) core.add_item(pos, leftover)
end end
end end
end end
return true return true
end end
minetest.register_lbm({ core.register_lbm({
label = "Give back items of old filters that had real inventories", label = "Give back items of old filters that had real inventories",
name = "pipeworks:give_back_old_filter_items", name = "pipeworks:give_back_old_filter_items",
nodenames = {"pipeworks:filter", "pipeworks:mese_filter"}, nodenames = {"pipeworks:filter", "pipeworks:mese_filter"},
run_at_every_load = false, run_at_every_load = false,
action = function(pos, node) action = function(pos, node)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local list = meta:get_inventory():get_list("main") local list = meta:get_inventory():get_list("main")
if not list then
return
end
if put_to_inputinv(pos, node, meta, list) then if put_to_inputinv(pos, node, meta, list) then
return return
end end
pos.y = pos.y + 1 pos.y = pos.y + 1
for i = 1, #list do for i = 1, #list do
local item = list[i] local item = list[i]
if not item:is_empty() then if not item:is_empty() then
minetest.add_item(pos, item) core.add_item(pos, item)
end end
end end
end, end,
}) })

View File

@@ -4,7 +4,7 @@
-- Contributed by mauvebic, 2013-01-03, rewritten a bit by Vanessa Ezekowitz -- Contributed by mauvebic, 2013-01-03, rewritten a bit by Vanessa Ezekowitz
-- --
local finitewater = minetest.settings:get_bool("liquid_finite") local finitewater = core.settings:get_bool("liquid_finite")
pipeworks.check_for_liquids = function(pos) pipeworks.check_for_liquids = function(pos)
local coords = { local coords = {
@@ -15,9 +15,9 @@ pipeworks.check_for_liquids = function(pos)
{x=pos.x,y=pos.y,z=pos.z-1}, {x=pos.x,y=pos.y,z=pos.z-1},
{x=pos.x,y=pos.y,z=pos.z+1}, } {x=pos.x,y=pos.y,z=pos.z+1}, }
for i =1,6 do for i =1,6 do
local name = minetest.get_node(coords[i]).name local name = core.get_node(coords[i]).name
if name and string.find(name,"water") then if name and string.find(name,"water") then
if finitewater then minetest.remove_node(coords[i]) end if finitewater then core.remove_node(coords[i]) end
return true return true
end end
end end
@@ -37,19 +37,19 @@ pipeworks.check_for_inflows = function(pos,node)
local source = false local source = false
for i = 1, 6 do for i = 1, 6 do
if newnode then break end if newnode then break end
local testnode = minetest.get_node(coords[i]) local testnode = core.get_node(coords[i])
local name = testnode.name local name = testnode.name
if name and (name == "pipeworks:pump_on" and pipeworks.check_for_liquids(coords[i])) or string.find(name,"_loaded") then if name and (name == "pipeworks:pump_on" and pipeworks.check_for_liquids(coords[i])) or string.find(name,"_loaded") then
if string.find(name,"_loaded") then if string.find(name,"_loaded") then
source = minetest.get_meta(coords[i]):get_string("source") source = core.get_meta(coords[i]):get_string("source")
if source == minetest.pos_to_string(pos) then break end if source == core.pos_to_string(pos) then break end
end end
if string.find(name, "valve") or string.find(name, "sensor") if string.find(name, "valve") or string.find(name, "sensor")
or string.find(name, "straight_pipe") or string.find(name, "panel") then or string.find(name, "straight_pipe") or string.find(name, "panel") then
if ((i == 3 or i == 4) and minetest.facedir_to_dir(testnode.param2).x ~= 0) if ((i == 3 or i == 4) and core.facedir_to_dir(testnode.param2).x ~= 0)
or ((i == 5 or i == 6) and minetest.facedir_to_dir(testnode.param2).z ~= 0) or ((i == 5 or i == 6) and core.facedir_to_dir(testnode.param2).z ~= 0)
or ((i == 1 or i == 2) and minetest.facedir_to_dir(testnode.param2).y ~= 0) then or ((i == 1 or i == 2) and core.facedir_to_dir(testnode.param2).y ~= 0) then
newnode = string.gsub(node.name,"empty","loaded") newnode = string.gsub(node.name,"empty","loaded")
source = {x=coords[i].x,y=coords[i].y,z=coords[i].z} source = {x=coords[i].x,y=coords[i].y,z=coords[i].z}
@@ -61,30 +61,30 @@ pipeworks.check_for_inflows = function(pos,node)
end end
end end
if newnode then if newnode then
minetest.add_node(pos,{name=newnode, param2 = node.param2}) core.add_node(pos,{name=newnode, param2 = node.param2})
minetest.get_meta(pos):set_string("source",minetest.pos_to_string(source)) core.get_meta(pos):set_string("source",core.pos_to_string(source))
end end
end end
pipeworks.check_sources = function(pos,node) pipeworks.check_sources = function(pos,node)
local sourcepos = minetest.string_to_pos(minetest.get_meta(pos):get_string("source")) local sourcepos = core.string_to_pos(core.get_meta(pos):get_string("source"))
if not sourcepos then return end if not sourcepos then return end
local source = minetest.get_node(sourcepos).name local source = core.get_node(sourcepos).name
local newnode = false local newnode = false
if source and not ((source == "pipeworks:pump_on" and pipeworks.check_for_liquids(sourcepos)) or string.find(source,"_loaded") or source == "ignore" ) then if source and not ((source == "pipeworks:pump_on" and pipeworks.check_for_liquids(sourcepos)) or string.find(source,"_loaded") or source == "ignore" ) then
newnode = string.gsub(node.name,"loaded","empty") newnode = string.gsub(node.name,"loaded","empty")
end end
if newnode then if newnode then
minetest.add_node(pos,{name=newnode, param2 = node.param2}) core.add_node(pos,{name=newnode, param2 = node.param2})
minetest.get_meta(pos):set_string("source","") core.get_meta(pos):set_string("source","")
end end
end end
pipeworks.spigot_check = function(pos, node) pipeworks.spigot_check = function(pos, node)
local belowname = minetest.get_node({x=pos.x,y=pos.y-1,z=pos.z}).name local belowname = core.get_node({x=pos.x,y=pos.y-1,z=pos.z}).name
if belowname and (belowname == "air" or belowname == "default:water_flowing" or belowname == "default:water_source") then if belowname and (belowname == "air" or belowname == pipeworks.liquids.water.flowing or belowname == pipeworks.liquids.water.source) then
local spigotname = minetest.get_node(pos).name local spigotname = core.get_node(pos).name
local fdir=node.param2 % 4 local fdir=node.param2 % 4
local check = { local check = {
{x=pos.x,y=pos.y,z=pos.z+1}, {x=pos.x,y=pos.y,z=pos.z+1},
@@ -92,19 +92,19 @@ pipeworks.spigot_check = function(pos, node)
{x=pos.x,y=pos.y,z=pos.z-1}, {x=pos.x,y=pos.y,z=pos.z-1},
{x=pos.x-1,y=pos.y,z=pos.z} {x=pos.x-1,y=pos.y,z=pos.z}
} }
local near_node = minetest.get_node(check[fdir+1]) local near_node = core.get_node(check[fdir+1])
if near_node and string.find(near_node.name, "_loaded") then if near_node and string.find(near_node.name, "_loaded") then
if spigotname and spigotname == "pipeworks:spigot" then if spigotname and spigotname == "pipeworks:spigot" then
minetest.add_node(pos,{name = "pipeworks:spigot_pouring", param2 = fdir}) core.add_node(pos,{name = "pipeworks:spigot_pouring", param2 = fdir})
if finitewater or belowname ~= "default:water_source" then if finitewater or belowname ~= pipeworks.liquids.water.source then
minetest.add_node({x=pos.x,y=pos.y-1,z=pos.z},{name = "default:water_source"}) core.add_node({x=pos.x,y=pos.y-1,z=pos.z},{name = pipeworks.liquids.water.source})
end end
end end
else else
if spigotname == "pipeworks:spigot_pouring" then if spigotname == "pipeworks:spigot_pouring" then
minetest.add_node({x=pos.x,y=pos.y,z=pos.z},{name = "pipeworks:spigot", param2 = fdir}) core.add_node({x=pos.x,y=pos.y,z=pos.z},{name = "pipeworks:spigot", param2 = fdir})
if belowname == "default:water_source" and not finitewater then if belowname == pipeworks.liquids.water.source and not finitewater then
minetest.remove_node({x=pos.x,y=pos.y-1,z=pos.z}) core.remove_node({x=pos.x,y=pos.y-1,z=pos.z})
end end
end end
end end
@@ -112,22 +112,22 @@ pipeworks.spigot_check = function(pos, node)
end end
pipeworks.fountainhead_check = function(pos, node) pipeworks.fountainhead_check = function(pos, node)
local abovename = minetest.get_node({x=pos.x,y=pos.y+1,z=pos.z}).name local abovename = core.get_node({x=pos.x,y=pos.y+1,z=pos.z}).name
if abovename and (abovename == "air" or abovename == "default:water_flowing" or abovename == "default:water_source") then if abovename and (abovename == "air" or abovename == pipeworks.liquids.water.flowing or abovename == pipeworks.liquids.water.source) then
local fountainhead_name = minetest.get_node(pos).name local fountainhead_name = core.get_node(pos).name
local near_node = minetest.get_node({x=pos.x,y=pos.y-1,z=pos.z}) local near_node = core.get_node({x=pos.x,y=pos.y-1,z=pos.z})
if near_node and string.find(near_node.name, "_loaded") then if near_node and string.find(near_node.name, "_loaded") then
if fountainhead_name and fountainhead_name == "pipeworks:fountainhead" then if fountainhead_name and fountainhead_name == "pipeworks:fountainhead" then
minetest.add_node(pos,{name = "pipeworks:fountainhead_pouring"}) core.add_node(pos,{name = "pipeworks:fountainhead_pouring"})
if finitewater or abovename ~= "default:water_source" then if finitewater or abovename ~= pipeworks.liquids.water.source then
minetest.add_node({x=pos.x,y=pos.y+1,z=pos.z},{name = "default:water_source"}) core.add_node({x=pos.x,y=pos.y+1,z=pos.z},{name = pipeworks.liquids.water.source})
end end
end end
else else
if fountainhead_name == "pipeworks:fountainhead_pouring" then if fountainhead_name == "pipeworks:fountainhead_pouring" then
minetest.add_node({x=pos.x,y=pos.y,z=pos.z},{name = "pipeworks:fountainhead"}) core.add_node({x=pos.x,y=pos.y,z=pos.z},{name = "pipeworks:fountainhead"})
if abovename == "default:water_source" and not finitewater then if abovename == pipeworks.liquids.water.source and not finitewater then
minetest.remove_node({x=pos.x,y=pos.y+1,z=pos.z}) core.remove_node({x=pos.x,y=pos.y+1,z=pos.z})
end end
end end
end end

156
init.lua
View File

@@ -4,12 +4,21 @@
-- and devices that they can connect to. -- and devices that they can connect to.
-- --
pipeworks = {} pipeworks = {
pipeworks.ui_cat_tube_list = {} ui_cat_tube_list = {},
worldpath = core.get_worldpath(),
pipeworks.worldpath = minetest.get_worldpath() modpath = core.get_modpath("pipeworks"),
pipeworks.modpath = minetest.get_modpath("pipeworks") liquids = {
local S = minetest.get_translator("pipeworks") water = {
source = core.registered_nodes["mapgen_water_source"].name,
flowing = core.registered_nodes["mapgen_water_source"].liquid_alternative_flowing
},
river_water = {
source = core.registered_nodes["mapgen_river_water_source"].name,
flowing = core.registered_nodes["mapgen_river_water_source"].liquid_alternative_flowing
}
}
}
dofile(pipeworks.modpath.."/default_settings.lua") dofile(pipeworks.modpath.."/default_settings.lua")
-- Read the external config file if it exists. -- Read the external config file if it exists.
@@ -20,78 +29,14 @@ if worldsettingsfile then
dofile(worldsettingspath) dofile(worldsettingspath)
end end
if pipeworks.toggles.pipe_mode == "pressure" then if pipeworks.toggles.pipe_mode == "pressure" then
minetest.log("warning", "pipeworks pressure logic mode comes with caveats and differences in behaviour, you have been warned!") core.log("warning", "pipeworks pressure logic mode comes with caveats and differences in behaviour, you have been warned!")
end end
if pipeworks.entity_update_interval >= 0.2 and pipeworks.enable_accelerator_tube then
-- Random variables core.log("warning", "pipeworks accelerator tubes will not entirely work with an entity update interval 0.2 or above.")
pipeworks.expect_infinite_stacks = true
if minetest.get_modpath("unified_inventory") or not minetest.settings:get_bool("creative_mode") then
pipeworks.expect_infinite_stacks = false
end
pipeworks.meseadjlist={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=0,y=1,z=0},{x=0,y=-1,z=0},{x=1,y=0,z=0},{x=-1,y=0,z=0}}
pipeworks.rules_all = {{x=0, y=0, z=1},{x=0, y=0, z=-1},{x=1, y=0, z=0},{x=-1, y=0, z=0},
{x=0, y=1, z=1},{x=0, y=1, z=-1},{x=1, y=1, z=0},{x=-1, y=1, z=0},
{x=0, y=-1, z=1},{x=0, y=-1, z=-1},{x=1, y=-1, z=0},{x=-1, y=-1, z=0},
{x=0, y=1, z=0}, {x=0, y=-1, z=0}}
pipeworks.mesecons_rules={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=1,y=0,z=0},{x=-1,y=0,z=0},{x=0,y=1,z=0},{x=0,y=-1,z=0}}
pipeworks.digilines_rules={{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=1,y=0,z=0},{x=-1,y=0,z=0},{x=0,y=1,z=0},{x=0,y=-1,z=0}}
pipeworks.liquid_texture = "default_water.png"
pipeworks.button_off = {text="", texture="pipeworks_button_off.png", addopts="false;false;pipeworks_button_interm.png"}
pipeworks.button_on = {text="", texture="pipeworks_button_on.png", addopts="false;false;pipeworks_button_interm.png"}
pipeworks.button_base = "image_button[0,4.3;1,0.6"
pipeworks.button_label = "label[0.9,4.31;"..S("Allow splitting incoming stacks from tubes").."]"
-- Helper functions
function pipeworks.fix_image_names(table, replacement)
local outtable={}
for i in ipairs(table) do
outtable[i]=string.gsub(table[i], "_XXXXX", replacement)
end
return outtable
end
function pipeworks.add_node_box(t, b)
if not t or not b then return end
for i in ipairs(b)
do table.insert(t, b[i])
end
end
function pipeworks.may_configure(pos, player)
local name = player:get_player_name()
local meta = minetest.get_meta(pos)
local owner = meta:get_string("owner")
if owner ~= "" and owner == name then -- wielders and filters
return true
end
return not minetest.is_protected(pos, name)
end
function pipeworks.replace_name(tbl,tr,name)
local ntbl={}
for key,i in pairs(tbl) do
if type(i)=="string" then
ntbl[key]=string.gsub(i,tr,name)
elseif type(i)=="table" then
ntbl[key]=pipeworks.replace_name(i,tr,name)
else
ntbl[key]=i
end
end
return ntbl
end end
pipeworks.logger = function(msg) pipeworks.logger = function(msg)
print("[pipeworks] "..msg) core.log("action", "[pipeworks] "..msg)
end end
------------------------------------------- -------------------------------------------
@@ -102,7 +47,7 @@ if pipeworks.toggles.finite_water == nil then
dofile(pipeworks.modpath.."/autodetect-finite-water.lua") dofile(pipeworks.modpath.."/autodetect-finite-water.lua")
end end
if minetest.get_modpath("signs_lib") then if core.get_modpath("signs_lib") then
dofile(pipeworks.modpath.."/signs_compat.lua") dofile(pipeworks.modpath.."/signs_compat.lua")
end end
@@ -113,16 +58,27 @@ dofile(pipeworks.modpath.."/autoplace_tubes.lua")
dofile(pipeworks.modpath.."/luaentity.lua") dofile(pipeworks.modpath.."/luaentity.lua")
dofile(pipeworks.modpath.."/item_transport.lua") dofile(pipeworks.modpath.."/item_transport.lua")
dofile(pipeworks.modpath.."/flowing_logic.lua") dofile(pipeworks.modpath.."/flowing_logic.lua")
dofile(pipeworks.modpath.."/crafts.lua")
dofile(pipeworks.modpath.."/tube_registration.lua")
dofile(pipeworks.modpath.."/routing_tubes.lua")
dofile(pipeworks.modpath.."/sorting_tubes.lua")
dofile(pipeworks.modpath.."/vacuum_tubes.lua")
dofile(pipeworks.modpath.."/signal_tubes.lua")
dofile(pipeworks.modpath.."/decorative_tubes.lua")
dofile(pipeworks.modpath.."/filter-injector.lua") dofile(pipeworks.modpath.."/filter-injector.lua")
dofile(pipeworks.modpath.."/chests.lua")
dofile(pipeworks.modpath.."/trashcan.lua") dofile(pipeworks.modpath.."/trashcan.lua")
dofile(pipeworks.modpath.."/wielder.lua") dofile(pipeworks.modpath.."/wielder.lua")
dofile(pipeworks.modpath.."/tubes/registration.lua")
dofile(pipeworks.modpath.."/tubes/routing.lua")
dofile(pipeworks.modpath.."/tubes/sorting.lua")
dofile(pipeworks.modpath.."/tubes/signal.lua")
dofile(pipeworks.modpath.."/tubes/embedded_tube.lua")
dofile(pipeworks.modpath.."/tubes/pane_embedded_tube.lua")
dofile(pipeworks.modpath.."/tubes/tags.lua")
if pipeworks.enable_teleport_tube then
dofile(pipeworks.modpath.."/tubes/teleport.lua")
end
if pipeworks.enable_lua_tube and core.get_modpath("mesecons") then
dofile(pipeworks.modpath.."/tubes/lua.lua")
end
if pipeworks.enable_sand_tube or pipeworks.enable_mese_sand_tube then
dofile(pipeworks.modpath.."/tubes/vacuum.lua")
end
local logicdir = "/pressure_logic/" local logicdir = "/pressure_logic/"
@@ -134,31 +90,45 @@ dofile(pipeworks.modpath..logicdir.."abms.lua")
dofile(pipeworks.modpath..logicdir.."abm_register.lua") dofile(pipeworks.modpath..logicdir.."abm_register.lua")
dofile(pipeworks.modpath..logicdir.."flowable_node_registry_install.lua") dofile(pipeworks.modpath..logicdir.."flowable_node_registry_install.lua")
if pipeworks.enable_pipes then dofile(pipeworks.modpath.."/pipes.lua") end if pipeworks.enable_pipes then
if pipeworks.enable_teleport_tube then dofile(pipeworks.modpath.."/teleport_tube.lua") end dofile(pipeworks.modpath.."/pipes.lua")
if pipeworks.enable_pipe_devices then dofile(pipeworks.modpath.."/devices.lua") end end
if pipeworks.enable_pipe_devices then
dofile(pipeworks.modpath.."/devices.lua")
end
if pipeworks.enable_redefines then if pipeworks.enable_redefines then
dofile(pipeworks.modpath.."/compat-chests.lua") dofile(pipeworks.modpath.."/compat-chests.lua")
end
if pipeworks.enable_redefines and (core.get_modpath("default") or core.get_modpath("hades_core")) then
dofile(pipeworks.modpath.."/compat-furnaces.lua") dofile(pipeworks.modpath.."/compat-furnaces.lua")
end end
if pipeworks.enable_autocrafter then dofile(pipeworks.modpath.."/autocrafter.lua") end if pipeworks.enable_redefines and core.get_modpath("mcl_furnaces") then
if pipeworks.enable_lua_tube and dofile(pipeworks.modpath.."/mcl_furnaces.lua")
(minetest.get_modpath("mesecons") or minetest.get_modpath("digilines")) then end
dofile(pipeworks.modpath.."/lua_tube.lua") if pipeworks.enable_autocrafter then
dofile(pipeworks.modpath.."/autocrafter.lua")
end end
minetest.register_alias("pipeworks:pipe", "pipeworks:pipe_110000_empty") dofile(pipeworks.modpath.."/crafts.lua")
core.register_alias("pipeworks:pipe", "pipeworks:pipe_110000_empty")
-- Unified Inventory categories integration -- Unified Inventory categories integration
if minetest.global_exists("unified_inventory") and unified_inventory.registered_categories then if core.get_modpath("unified_inventory") and unified_inventory.registered_categories then
if not unified_inventory.registered_categories["automation"] then if not unified_inventory.registered_categories["automation"] then
local symbol
if pipeworks.enable_lua_tube then
symbol = "pipeworks:lua_tube000000"
else
symbol = "pipeworks:mese_filter" -- fallback when lua tube isn't registered
end
unified_inventory.register_category("automation", { unified_inventory.register_category("automation", {
symbol = "pipeworks:lua_tube000000", symbol = symbol,
label = "Automation components" label = "Automation components"
}) })
end end
unified_inventory.add_category_items("automation", pipeworks.ui_cat_tube_list) unified_inventory.add_category_items("automation", pipeworks.ui_cat_tube_list)
end end
minetest.log("info", "Pipeworks loaded!") core.log("info", "Pipeworks loaded!")

View File

@@ -1,22 +1,59 @@
local luaentity = pipeworks.luaentity local luaentity = pipeworks.luaentity
local enable_max_limit = minetest.settings:get_bool("pipeworks_enable_items_per_tube_limit") local enable_max_limit = core.settings:get_bool("pipeworks_enable_items_per_tube_limit")
local max_tube_limit = tonumber(minetest.settings:get("pipeworks_max_items_per_tube")) or 30 local max_tube_limit = tonumber(core.settings:get("pipeworks_max_items_per_tube")) or 30
if enable_max_limit == nil then enable_max_limit = true end if enable_max_limit == nil then enable_max_limit = true end
if pipeworks.enable_item_tags then
local max_tag_length = tonumber(core.settings:get("pipeworks_max_item_tag_length")) or 32
local max_tags = tonumber(core.settings:get("pipeworks_max_item_tags")) or 16
function pipeworks.sanitize_tags(tags)
if type(tags) == "string" then
tags = tags:split(",")
end
local sanitized = {}
for i, tag in ipairs(tags) do
if type(tag) == "string" then
tag = tag:gsub("[%s,]", "") -- Remove whitespace and commas
tag = tag:gsub("%$%b%{%}", "") -- Remove special ${key} values
if tag ~= "" then
table.insert(sanitized, tag:sub(1, max_tag_length))
end
end
if #sanitized >= max_tags then
break
end
end
return sanitized
end
end
function pipeworks.tube_item(pos, item) function pipeworks.tube_item(pos, item)
error("obsolete pipeworks.tube_item() called; change caller to use pipeworks.tube_inject_item() instead") error("obsolete pipeworks.tube_item() called; change caller to use pipeworks.tube_inject_item() instead")
end end
function pipeworks.tube_inject_item(pos, start_pos, velocity, item, owner) function pipeworks.tube_inject_item(pos, start_pos, velocity, item, owner, tags)
-- Take item in any format -- Take item in any format
local stack = ItemStack(item) local stack = ItemStack(item)
local to_pos = vector.add(pos, velocity)
local def = core.registered_nodes[core.get_node(to_pos).name]
if not def or not def.groups or not (def.groups.tube
or def.groups.tubedevice or def.groups.tubedevice_receiver) then
local dropped_item = core.add_item(pos, stack)
if dropped_item then
dropped_item:set_velocity(vector.multiply(velocity, 5))
end
return
end
local obj = luaentity.add_entity(pos, "pipeworks:tubed_item") local obj = luaentity.add_entity(pos, "pipeworks:tubed_item")
obj:set_item(stack:to_string()) if obj then
obj.start_pos = vector.new(start_pos) obj:set_item(stack:to_string())
obj:set_velocity(velocity) obj.start_pos = vector.new(start_pos)
obj.owner = owner obj:set_velocity(velocity)
--obj:set_color("red") -- todo: this is test-only code obj.owner = owner
return obj obj.tags = tags
return obj
end
end end
-- adding two tube functions -- adding two tube functions
@@ -37,14 +74,14 @@ end
local tube_item_count = {} local tube_item_count = {}
minetest.register_globalstep(function(dtime) core.register_globalstep(function(dtime)
if not luaentity.entities then if not luaentity.entities then
return return
end end
tube_item_count = {} tube_item_count = {}
for _, entity in pairs(luaentity.entities) do for _, entity in pairs(luaentity.entities) do
if entity.name == "pipeworks:tubed_item" then if entity.name == "pipeworks:tubed_item" then
local h = minetest.hash_node_position(vector.round(entity._pos)) local h = core.hash_node_position(vector.round(entity._pos))
tube_item_count[h] = (tube_item_count[h] or 0) + 1 tube_item_count[h] = (tube_item_count[h] or 0) + 1
end end
end end
@@ -57,19 +94,20 @@ end)
-- exceeds the limit configured per tube, replace it with a broken one. -- exceeds the limit configured per tube, replace it with a broken one.
function pipeworks.break_tube(pos) function pipeworks.break_tube(pos)
local node = minetest.get_node(pos) local node = core.get_node(pos)
local meta = minetest.get_meta(pos) if core.get_item_group(node.name, "tube") ~= 1 then return end
meta:set_string("the_tube_was", minetest.serialize(node)) local meta = core.get_meta(pos)
minetest.swap_node(pos, {name = "pipeworks:broken_tube_1"}) meta:set_string("the_tube_was", core.serialize(node))
core.swap_node(pos, {name = "pipeworks:broken_tube_1"})
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end end
local crunch_tube = function(pos, cnode, cmeta) local crunch_tube = function(pos, cnode, cmeta)
if enable_max_limit then if enable_max_limit then
local h = minetest.hash_node_position(pos) local h = core.hash_node_position(pos)
local itemcount = tube_item_count[h] or 0 local itemcount = tube_item_count[h] or 0
if itemcount > max_tube_limit then if itemcount > max_tube_limit then
pipeworks.logger("Warning - a tube at "..minetest.pos_to_string(pos).." broke due to too many items ("..itemcount..")") pipeworks.logger("Warning - a tube at "..core.pos_to_string(pos).." broke due to too many items ("..itemcount..")")
pipeworks.break_tube(pos) pipeworks.break_tube(pos)
end end
end end
@@ -79,16 +117,17 @@ end
-- compatibility behaviour for the existing can_go() callbacks, -- compatibility behaviour for the existing can_go() callbacks,
-- which can only specify a list of possible positions. -- which can only specify a list of possible positions.
local function go_next_compat(pos, cnode, cmeta, cycledir, vel, stack, owner) local function go_next_compat(pos, cnode, cmeta, cycledir, vel, stack, owner, tags)
local next_positions = {} local next_positions = {}
local max_priority = 0 local max_priority = 0
local can_go local can_go
if minetest.registered_nodes[cnode.name] and minetest.registered_nodes[cnode.name].tube and minetest.registered_nodes[cnode.name].tube.can_go then local def = core.registered_nodes[cnode.name]
can_go = minetest.registered_nodes[cnode.name].tube.can_go(pos, cnode, vel, stack) if def and def.tube and def.tube.can_go then
can_go = def.tube.can_go(pos, cnode, vel, stack, tags)
else else
local adjlist_string = minetest.get_meta(pos):get_string("adjlist") local adjlist_string = core.get_meta(pos):get_string("adjlist")
local adjlist = minetest.deserialize(adjlist_string) or default_adjlist -- backward compat: if not found, use old behavior: all directions local adjlist = core.deserialize(adjlist_string) or default_adjlist -- backward compat: if not found, use old behavior: all directions
can_go = pipeworks.notvel(adjlist, vel) can_go = pipeworks.notvel(adjlist, vel)
end end
@@ -101,11 +140,11 @@ local function go_next_compat(pos, cnode, cmeta, cycledir, vel, stack, owner)
for _, vect in ipairs(can_go) do for _, vect in ipairs(can_go) do
local npos = vector.add(pos, vect) local npos = vector.add(pos, vect)
pipeworks.load_position(npos) pipeworks.load_position(npos)
local node = minetest.get_node(npos) local node = core.get_node(npos)
local reg_node = minetest.registered_nodes[node.name] local reg_node = core.registered_nodes[node.name]
if reg_node then if reg_node then
local tube_def = reg_node.tube local tube_def = reg_node.tube
local tubedevice = minetest.get_item_group(node.name, "tubedevice") local tubedevice = core.get_item_group(node.name, "tubedevice")
local tube_priority = (tube_def and tube_def.priority) or 100 local tube_priority = (tube_def and tube_def.priority) or 100
if tubedevice > 0 and tube_priority >= max_priority then if tubedevice > 0 and tube_priority >= max_priority then
if not tube_def or not tube_def.can_insert or if not tube_def or not tube_def.can_insert or
@@ -144,9 +183,9 @@ end
-- * a "multi-mode" data table (or nil if N/A) where a stack was split apart. -- * a "multi-mode" data table (or nil if N/A) where a stack was split apart.
-- if this is not nil, the luaentity spawns new tubed items for each new fragment stack, -- if this is not nil, the luaentity spawns new tubed items for each new fragment stack,
-- then deletes itself (i.e. the original item stack). -- then deletes itself (i.e. the original item stack).
local function go_next(pos, velocity, stack, owner) local function go_next(pos, velocity, stack, owner, tags)
local cnode = minetest.get_node(pos) local cnode = core.get_node(pos)
local cmeta = minetest.get_meta(pos) local cmeta = core.get_meta(pos)
local speed = math.abs(velocity.x + velocity.y + velocity.z) local speed = math.abs(velocity.x + velocity.y + velocity.z)
if speed == 0 then if speed == 0 then
speed = 1 speed = 1
@@ -172,7 +211,7 @@ local function go_next(pos, velocity, stack, owner)
-- n is the new value of the cycle counter. -- n is the new value of the cycle counter.
-- XXX: this probably needs cleaning up after being split out, -- XXX: this probably needs cleaning up after being split out,
-- seven args is a bit too many -- seven args is a bit too many
local n, found, new_velocity, multimode = go_next_compat(pos, cnode, cmeta, cycledir, vel, stack, owner) local n, found, new_velocity, multimode = go_next_compat(pos, cnode, cmeta, cycledir, vel, stack, owner, tags)
-- if not using output cycling, -- if not using output cycling,
-- don't update the field so it stays the same for the next item. -- don't update the field so it stays the same for the next item.
@@ -184,7 +223,7 @@ end
minetest.register_entity("pipeworks:tubed_item", { core.register_entity("pipeworks:tubed_item", {
initial_properties = { initial_properties = {
hp_max = 1, hp_max = 1,
physical = false, physical = false,
@@ -209,9 +248,9 @@ minetest.register_entity("pipeworks:tubed_item", {
end end
local item_texture = nil local item_texture = nil
local item_type = "" local item_type = ""
if minetest.registered_items[itemname] then if core.registered_items[itemname] then
item_texture = minetest.registered_items[itemname].inventory_image item_texture = core.registered_items[itemname].inventory_image
item_type = minetest.registered_items[itemname].type item_type = core.registered_items[itemname].type
end end
--]] --]]
self.object:set_properties({ self.object:set_properties({
@@ -231,13 +270,13 @@ minetest.register_entity("pipeworks:tubed_item", {
self.object:remove() self.object:remove()
return return
end end
local item = minetest.deserialize(staticdata) local item = core.deserialize(staticdata)
pipeworks.tube_inject_item(self.object:get_pos(), item.start_pos, item.velocity, item.itemstring) pipeworks.tube_inject_item(self.object:get_pos(), item.start_pos, item.velocity, item.itemstring)
self.object:remove() self.object:remove()
end, end,
}) })
minetest.register_entity("pipeworks:color_entity", { core.register_entity("pipeworks:color_entity", {
initial_properties = { initial_properties = {
hp_max = 1, hp_max = 1,
physical = false, physical = false,
@@ -276,6 +315,7 @@ luaentity.register_entity("pipeworks:tubed_item", {
color_entity = nil, color_entity = nil,
color = nil, color = nil,
start_pos = nil, start_pos = nil,
tags = nil,
set_item = function(self, item) set_item = function(self, item)
local itemstring = ItemStack(item):to_string() -- Accept any input format local itemstring = ItemStack(item):to_string() -- Accept any input format
@@ -311,8 +351,6 @@ luaentity.register_entity("pipeworks:tubed_item", {
self:set_pos(pos) self:set_pos(pos)
end end
local stack = ItemStack(self.itemstring)
local velocity = self:get_velocity() local velocity = self:get_velocity()
local moved = false local moved = false
@@ -329,12 +367,19 @@ luaentity.register_entity("pipeworks:tubed_item", {
moved = true moved = true
end end
if not moved then
return
end
local stack = ItemStack(self.itemstring)
pipeworks.load_position(self.start_pos) pipeworks.load_position(self.start_pos)
local node = minetest.get_node(self.start_pos) local node = core.get_node(self.start_pos)
if moved and minetest.get_item_group(node.name, "tubedevice_receiver") == 1 then if core.get_item_group(node.name, "tubedevice_receiver") == 1 then
local leftover local leftover
if minetest.registered_nodes[node.name].tube and minetest.registered_nodes[node.name].tube.insert_object then local def = core.registered_nodes[node.name]
leftover = minetest.registered_nodes[node.name].tube.insert_object(self.start_pos, node, stack, vel, self.owner) if def.tube and def.tube.insert_object then
leftover = def.tube.insert_object(self.start_pos, node, stack, vel, self.owner)
else else
leftover = stack leftover = stack
end end
@@ -349,62 +394,67 @@ luaentity.register_entity("pipeworks:tubed_item", {
return return
end end
if moved then local tags
local found_next, new_velocity, multimode = go_next(self.start_pos, velocity, stack, self.owner) -- todo: color if pipeworks.enable_item_tags then
local rev_vel = vector.multiply(velocity, -1) tags = self.tags or {}
local rev_dir = vector.direction(self.start_pos,vector.add(self.start_pos,rev_vel)) end
local rev_node = minetest.get_node(vector.round(vector.add(self.start_pos,rev_dir))) local found_next, new_velocity, multimode = go_next(self.start_pos, velocity, stack, self.owner, tags) -- todo: color
local tube_present = minetest.get_item_group(rev_node.name,"tubedevice") == 1 if pipeworks.enable_item_tags then
if not found_next then self.tags = #tags > 0 and tags or nil
if pipeworks.drop_on_routing_fail or not tube_present or end
minetest.get_item_group(rev_node.name,"tube") ~= 1 then local rev_vel = vector.multiply(velocity, -1)
-- Using add_item instead of item_drop since this makes pipeworks backward local rev_dir = vector.direction(self.start_pos,vector.add(self.start_pos,rev_vel))
-- compatible with Minetest 0.4.13. local rev_node = core.get_node(vector.round(vector.add(self.start_pos,rev_dir)))
-- Using item_drop here makes Minetest 0.4.13 crash. local tube_present = core.get_item_group(rev_node.name,"tubedevice") == 1
local dropped_item = minetest.add_item(self.start_pos, stack) if not found_next then
if dropped_item then if pipeworks.drop_on_routing_fail or not tube_present or
dropped_item:set_velocity(vector.multiply(velocity, 5)) core.get_item_group(rev_node.name,"tube") ~= 1 then
self:remove() -- Using add_item instead of item_drop since this makes pipeworks backward
end -- compatible with Minetest 0.4.13.
return -- Using item_drop here makes Minetest 0.4.13 crash.
else local dropped_item = core.add_item(self.start_pos, stack)
velocity = vector.multiply(velocity, -1) if dropped_item then
self:set_pos(vector.subtract(self.start_pos, vector.multiply(vel, moved_by - 1))) dropped_item:set_velocity(vector.multiply(velocity, 5))
self:set_velocity(velocity) self:remove()
end end
elseif is_multimode(multimode) then
-- create new stacks according to returned data.
local s = self.start_pos
for _, split in ipairs(multimode) do
pipeworks.tube_inject_item(s, s, split.velocity, split.itemstack, self.owner)
end
-- remove ourself now the splits are sent
self:remove()
return return
else
velocity = vector.multiply(velocity, -1)
self:set_pos(vector.subtract(self.start_pos, vector.multiply(vel, moved_by - 1)))
self:set_velocity(velocity)
end end
elseif is_multimode(multimode) then
-- create new stacks according to returned data.
local s = self.start_pos
for _, split in ipairs(multimode) do
pipeworks.tube_inject_item(s, s, split.velocity, split.itemstack, self.owner)
end
-- remove ourself now the splits are sent
self:remove()
return
end
if new_velocity and not vector.equals(velocity, new_velocity) then if new_velocity and not vector.equals(velocity, new_velocity) then
local nvelr = math.abs(new_velocity.x + new_velocity.y + new_velocity.z) local nvelr = math.abs(new_velocity.x + new_velocity.y + new_velocity.z)
self:set_pos(vector.add(self.start_pos, vector.multiply(new_velocity, (moved_by - 1) / nvelr))) self:set_pos(vector.add(self.start_pos, vector.multiply(new_velocity, (moved_by - 1) / nvelr)))
self:set_velocity(new_velocity) self:set_velocity(new_velocity)
end
end end
end end
}) })
if minetest.get_modpath("mesecons_mvps") then if core.get_modpath("mesecons_mvps") then
mesecon.register_mvps_unmov("pipeworks:tubed_item") mesecon.register_mvps_unmov("pipeworks:tubed_item")
mesecon.register_mvps_unmov("pipeworks:color_entity") mesecon.register_mvps_unmov("pipeworks:color_entity")
mesecon.register_on_mvps_move(function(moved_nodes) mesecon.register_on_mvps_move(function(moved_nodes)
local moved = {} local moved = {}
for _, n in ipairs(moved_nodes) do for _, n in ipairs(moved_nodes) do
moved[minetest.hash_node_position(n.oldpos)] = vector.subtract(n.pos, n.oldpos) moved[core.hash_node_position(n.oldpos)] = vector.subtract(n.pos, n.oldpos)
end end
for _, entity in pairs(luaentity.entities) do for _, entity in pairs(luaentity.entities) do
if entity.name == "pipeworks:tubed_item" then if entity.name == "pipeworks:tubed_item" then
local pos = entity:get_pos() local pos = entity:get_pos()
local rpos = vector.round(pos) local rpos = vector.round(pos)
local dir = moved[minetest.hash_node_position(rpos)] local dir = moved[core.hash_node_position(rpos)]
if dir then if dir then
entity:set_pos(vector.add(pos, dir)) entity:set_pos(vector.add(pos, dir))
entity.start_pos = vector.add(entity.start_pos, dir) entity.start_pos = vector.add(entity.start_pos, dir)

View File

@@ -1,60 +0,0 @@
local S = minetest.get_translator("pipeworks")
if not minetest.get_modpath("auto_tree_tap") and
minetest.get_modpath("technic") then
minetest.register_abm({
nodenames = { "auto_tree_tap:off", "auto_tree_tap:on" },
chance = 1,
interval = 1,
action = function(pos, node, active_object_count, active_object_count_wider)
local fdir = node.param2
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
inv:set_size("pick", 1)
inv:set_size("ghost_pick", 1)
inv:set_size("main", 100)
minetest.set_node(pos, {name = "pipeworks:nodebreaker_off", param2 = fdir})
minetest.registered_nodes["pipeworks:nodebreaker_off"].on_punch(pos, node)
inv:set_stack("pick", 1, ItemStack("technic:treetap"))
end
})
minetest.register_node(":auto_tree_tap:off", {
description = S("Auto-Tap"),
tiles = {"pipeworks_nodebreaker_top_off.png","pipeworks_nodebreaker_bottom_off.png","pipeworks_nodebreaker_side2_off.png","pipeworks_nodebreaker_side1_off.png",
"pipeworks_nodebreaker_back.png","pipeworks_nodebreaker_front_off.png"},
is_ground_content = true,
paramtype2 = "facedir",
groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2, mesecon = 2,tubedevice=1, not_in_creative_inventory=1 },
sounds = default.node_sound_stone_defaults(),
tube = {connect_sides={back=1}},
on_construct = function(pos)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
inv:set_size("pick", 1)
inv:set_stack("pick", 1, ItemStack("default:pick_mese"))
end,
after_place_node = function (pos, placer)
pipeworks.scan_for_tube_objects(pos, placer)
local placer_pos = placer:get_pos()
--correct for the player's height
if placer:is_player() then placer_pos.y = placer_pos.y + 1.5 end
--correct for 6d facedir
if placer_pos then
local dir = {
x = pos.x - placer_pos.x,
y = pos.y - placer_pos.y,
z = pos.z - placer_pos.z
}
local node = minetest.get_node(pos)
node.param2 = minetest.dir_to_facedir(dir, true)
minetest.set_node(pos, node)
minetest.log("action", "real (6d) facedir: " .. node.param2)
end
end,
after_dig_node = pipeworks.scan_for_tube_objects,
})
end

121
locale/pipeworks.ru.tr Normal file
View File

@@ -0,0 +1,121 @@
# textdomain: pipeworks
# License: CC-by-SA 4.0
# Author: VinAdmin ovvitalik@gmail.com
## generic interaction
Set=Установить
Cancel=Отмена
## digilines interfacing
Channel=Канал
## init
Allow splitting incoming stacks from tubes=Разрешить разделение входящих стопок из трубок
## autocrafter
Unknown item=Неизвестный предмет
unconfigured Autocrafter: unknown recipe=ненастроенный автокрафтер: неизвестный рецепт
unconfigured Autocrafter=ненастроенный автокрафтер
'@1' Autocrafter (@2)='@1' Автокрафтер (@2)
Save=Сохранить
paused '@1' Autocrafter=приостановлено '@1' Автокрафтер
Autocrafter=Автокрафтер
## compat-furnaces
Allow splitting incoming material (not fuel) stacks from tubes=Разрешить разделение стопок поступающего материала (не топлива) из трубок.
## decorative tubes
Airtight steelblock embedded tube=Герметичная встроенная трубка из стального блока
Airtight panel embedded tube=Герметичная встроенная в панель трубка
## devices
Pump/Intake Module=Модуль насоса/впуска
Valve=Клапан
Decorative grating=Декоративная решетка
Spigot outlet=Выходной патрубок
Airtight Pipe entry/exit=Вход/выход герметичной трубы
Flow Sensor=Датчик потока
Flow sensor (on)=Датчик расхода (вкл.)
empty=пустой
@1% full=@1% заполнено
Expansion Tank (@1)=Расширительный бак (@1)
Fluid Storage Tank (@1)=Резервуар для хранения жидкости (@1)
Fountainhead=Источник
Straight-only Pipe=Только прямая труба
## filter-injector
(slot #@1 next)=(слот #@1 следующий)
@1 Filter-Injector=@1 фильтр-инжектор
Sequence slots by Priority=Последовательность слотов по приоритету
Sequence slots Randomly=Слоты последовательности Случайный
Sequence slots by Rotation=Последовательность слотов по вращению
Exact match - off=Точное совпадение выключено
Exact match - on=Точное совпадение - включено
Prefer item types:=Предпочитаете типы предметов:
Itemwise=По пунктам
Stackwise=Стекообразно
Digiline=Диджилайн
## legacy
Auto-Tap=Авто-нажатие
## pipes
Pipe Segment=Сегмент трубы
Pipe Segment (legacy)=Сегмент трубы (устаревший)
## routing tubes
Pneumatic tube segment=Сегмент пневматической трубки
Broken Tube=Сломанная трубка
High Priority Tube Segment=Сегмент трубы с высоким приоритетом
Accelerating Pneumatic Tube Segment=Ускорительный сегмент пневматической трубки
Crossing Pneumatic Tube Segment=Пересечение сегмента пневматической трубы
One way tube=Односторонняя трубка
## signal tubes
Detecting Pneumatic Tube Segment on=Обнаружение сегмента пневматической трубки включено
Detecting Pneumatic Tube Segment=Обнаружение сегмента пневматической трубки
Digiline Detecting Pneumatic Tube Segment=Digiline обнаруживает сегмент пневматической трубки
Digiline Detecting Tube=Детекторная трубка Digiline
Conducting Pneumatic Tube Segment=Проводящий сегмент пневматической трубки
Conducting Pneumatic Tube Segment on=Проводящий сегмент пневматической трубки на
Digiline Conducting Pneumatic Tube Segment=Сегмент проводящей пневматической трубки Digiline
Mesecon and Digiline Conducting Pneumatic Tube Segment=Сегмент токопроводящей пневматической трубки Mesecon и Digiline
Mesecon and Digiline Conducting Pneumatic Tube Segment on=Сегмент проводящей пневматической трубки Mesecon и Digiline на
Tag Sorting Pneumatic Tube Segment=Сегмент пневматической трубки для сортировки тегов
Lua controlled Tube=Трубка, управляемая Lua
## sorting tubes
Sorting Pneumatic Tube Segment=Сортировка сегментов пневматической трубки
Sorting pneumatic tube=Сортировочная пневматическая труба
## teleport tube
Receive=Получить
Channels are public by default=По умолчанию каналы являются общедоступными
Use <player>:<channel> for fully private channels=Используйте <игрок>:<канал> для полностью приватных каналов
Use <player>;<channel> for private receivers=Используйте <игрок>;<канал> для частных приемников
Teleporting Pneumatic Tube Segment=Сегмент пневматической трубы для телепортации
Teleporting Tube=Телепортационная труба
Unconfigured Teleportation Tube=Неконфигурированная телепортационная труба
Sorry, channel '@1' is reserved for exclusive use by @2=Извините, канал '@1' зарезервирован исключительно для использования @2
Sorry, receiving from channel '@1' is reserved for @2=Извините, прием с канала '@1' зарезервирован для @2
Teleportation Tube @1 on '@2'=Трубка телепортации @1 на '@2'
## trashcan
Trash Can=Мусорное ведро
## tube registration
Pneumatic tube segment (legacy)=Сегмент пневматической трубы (устаревший)
## vacuum tubes
Radius=Радиус
Vacuuming Pneumatic Tube Segment=Сегмент пневматической трубки для вакуумирования
Adjustable Vacuuming Tube=Регулируемая вакуумная трубка
Adjustable Vacuuming Pneumatic Tube Segment=Регулируемый сегмент вакуумной пневматической трубки
Adjustable Vacuuming Pneumatic Tube Segment (@1m)=Регулируемый сегмент вакуумной пневматической трубки (@1m)
## wielder
Node Breaker=Разрушитель узла
Deployer=Развертыватель
Dispenser=Распылитель

View File

@@ -5,19 +5,19 @@ pipeworks.luaentity = luaentity
luaentity.registered_entities = {} luaentity.registered_entities = {}
local filename = minetest.get_worldpath().."/luaentities" local filename = core.get_worldpath().."/luaentities"
local function read_file() local function read_file()
local f = io.open(filename, "r") local f = io.open(filename, "r")
if f == nil then return {} end if f == nil then return {} end
local t = f:read("*all") local t = f:read("*all")
f:close() f:close()
if t == "" or t == nil then return {} end if t == "" or t == nil then return {} end
return minetest.deserialize(t) or {} return core.deserialize(t) or {}
end end
local function write_file(tbl) local function write_file(tbl)
local f = io.open(filename, "w") local f = io.open(filename, "w")
f:write(minetest.serialize(tbl)) f:write(core.serialize(tbl))
f:close() f:close()
end end
@@ -64,40 +64,57 @@ local function write_entities()
write_file(luaentity.entities) write_file(luaentity.entities)
end end
minetest.register_on_shutdown(write_entities) core.register_on_shutdown(write_entities)
luaentity.entities_index = 0 luaentity.entities_index = 0
local function get_blockpos(pos) local move_entities_globalstep_part1
return {x = math.floor(pos.x / 16), local is_active
y = math.floor(pos.y / 16),
z = math.floor(pos.z / 16)}
end
local active_blocks = {} -- These only contain active blocks near players (i.e., not forceloaded ones) if pipeworks.use_real_entities then
local active_blocks = {} -- These only contain active blocks near players (i.e., not forceloaded ones)
local move_entities_globalstep_part1 = function(dtime) local function get_blockpos(pos)
local active_block_range = tonumber(minetest.settings:get("active_block_range")) or 2 return {x = math.floor(pos.x / 16),
local new_active_blocks = {} y = math.floor(pos.y / 16),
for _, player in ipairs(minetest.get_connected_players()) do z = math.floor(pos.z / 16)}
local blockpos = get_blockpos(player:get_pos())
local minp = vector.subtract(blockpos, active_block_range)
local maxp = vector.add(blockpos, active_block_range)
for x = minp.x, maxp.x do
for y = minp.y, maxp.y do
for z = minp.z, maxp.z do
local pos = {x = x, y = y, z = z}
new_active_blocks[minetest.hash_node_position(pos)] = pos
end
end
end
end end
active_blocks = new_active_blocks
-- todo: callbacks on block load/unload
end
local function is_active(pos) move_entities_globalstep_part1 = function(dtime)
return active_blocks[minetest.hash_node_position(get_blockpos(pos))] ~= nil local active_block_range = tonumber(core.settings:get("active_block_range")) or 2
for key in pairs(active_blocks) do
active_blocks[key] = nil
end
for _, player in ipairs(core.get_connected_players()) do
local blockpos = get_blockpos(player:get_pos())
local minpx = blockpos.x - active_block_range
local minpy = blockpos.y - active_block_range
local minpz = blockpos.z - active_block_range
local maxpx = blockpos.x + active_block_range
local maxpy = blockpos.y + active_block_range
local maxpz = blockpos.z + active_block_range
for x = minpx, maxpx do
for y = minpy, maxpy do
for z = minpz, maxpz do
local pos = {x = x, y = y, z = z}
active_blocks[core.hash_node_position(pos)] = true
end
end
end
end
-- todo: callbacks on block load/unload
end
is_active = function(pos)
return active_blocks[core.hash_node_position(get_blockpos(pos))] ~= nil
end
else
move_entities_globalstep_part1 = function()
end
is_active = function()
return false
end
end end
local entitydef_default = { local entitydef_default = {
@@ -151,11 +168,15 @@ local entitydef_default = {
if not is_active(entity_pos) then if not is_active(entity_pos) then
return return
end end
local ent = minetest.add_entity(entity_pos, entity.name):get_luaentity() local object = core.add_entity(entity_pos, entity.name)
if not object then
return
end
local ent = object:get_luaentity()
ent:from_data(entity.data) ent:from_data(entity.data)
ent.parent_id = self._id ent.parent_id = self._id
ent.attached_id = index ent.attached_id = index
entity.entity = ent.object entity.entity = object
local master = self._attached_entities_master local master = self._attached_entities_master
if master then if master then
self:_attach(index, master) self:_attach(index, master)
@@ -269,7 +290,7 @@ end
function luaentity.add_entity(pos, name) function luaentity.add_entity(pos, name)
if not luaentity.entities then if not luaentity.entities then
minetest.after(0, luaentity.add_entity, vector.new(pos), name) core.after(0, luaentity.add_entity, vector.new(pos), name)
return return
end end
local index = luaentity.entities_index local index = luaentity.entities_index
@@ -370,13 +391,36 @@ local move_entities_globalstep_part2 = function(dtime)
end end
end end
local handle_active_blocks_timer = 0.1 -- dtime after which there is an update (or skip).
local dtime_threshold = pipeworks.entity_update_interval
-- Accumulated dtime since last update (or skip).
local dtime_accum = 0
-- Delayed dtime accumulated due to skipped updates.
local dtime_delayed = 0
local skip_update = false
minetest.register_globalstep(function(dtime) core.register_globalstep(function(dtime)
handle_active_blocks_timer = handle_active_blocks_timer + dtime if dtime >= 0.2 and dtime_delayed < 1 then
if dtime < 0.2 or handle_active_blocks_timer >= (dtime * 3) then -- Reduce activity when the server is lagging.
handle_active_blocks_timer = 0.1 skip_update = true
move_entities_globalstep_part1(dtime)
move_entities_globalstep_part2(dtime)
end end
dtime_accum = dtime_accum + dtime
if dtime_accum < dtime_threshold then
return
end
if skip_update then
dtime_delayed = dtime_delayed + dtime_accum
skip_update = false
else
move_entities_globalstep_part1(dtime_accum + dtime_delayed)
move_entities_globalstep_part2(dtime_accum + dtime_delayed)
dtime_delayed = 0
end
-- Tune the threshold so that the average interval is pipeworks.entity_update_interval.
dtime_threshold = math.max(dtime_threshold + (pipeworks.entity_update_interval - dtime_accum) / 10, 0)
dtime_accum = 0
end) end)

376
mcl_furnaces.lua Normal file
View File

@@ -0,0 +1,376 @@
local old_furnace = table.copy(core.registered_nodes["mcl_furnaces:furnace"])
local old_blast_furnace = table.copy(core.registered_nodes["mcl_blast_furnace:blast_furnace"])
local old_smoker = table.copy(core.registered_nodes["mcl_smoker:smoker"])
local tube_entry = "^pipeworks_tube_connection_stony.png"
-- groups
local furnace_groups = old_furnace.groups
furnace_groups["tubedevice"] = 1
furnace_groups["tubedevice_receiver"] = 1
local furnace_groups_active = table.copy(furnace_groups)
furnace_groups_active["not_in_creative_inventory"] = 1
local blast_furnace_groups = old_blast_furnace.groups
blast_furnace_groups["tubedevice"] = 1
blast_furnace_groups["tubedevice_receiver"] = 1
local blast_furnace_groups_active = table.copy(blast_furnace_groups)
blast_furnace_groups_active["not_in_creative_inventory"] = 1
local smoker_groups = old_smoker.groups
smoker_groups["tubedevice"] = 1
smoker_groups["tubedevice_receiver"] = 1
local smoker_groups_active = table.copy(smoker_groups)
smoker_groups_active["not_in_creative_inventory"] = 1
-- a hack to give the exp to fake players it's be dropped instead added to (fake) player inv
local function give_xp(pos, player)
local meta = core.get_meta(pos)
local dir = vector.divide(core.facedir_to_dir(core.get_node(pos).param2), -1.95)
local xp = meta:get_int("xp")
if xp > 0 then
mcl_experience.throw_xp(vector.add(pos, dir), xp)
meta:set_int("xp", 0)
end
end
local override = {}
override.tiles = {
"default_furnace_top.png"..tube_entry,
"default_furnace_bottom.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_front.png"
}
override.groups = furnace_groups
override.tube = {
insert_object = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
if meta:get_int("split_material_stacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
override.after_place_node = function(pos, placer, itemstack, pointed_thing)
pipeworks.after_place(pos, placer, itemstack, pointed_thing)
end
override.after_dig_node = function(pos, oldnode, oldmetadata, digger)
old_furnace.after_dig_node(pos, oldnode, oldmetadata, digger)
pipeworks.after_dig(pos)
end
override.on_metadata_inventory_take = function(pos, listname, index, stack, player)
if listname == "dst" then
if stack:get_name() == "mcl_core:iron_ingot" then
awards.unlock(player:get_player_name(), "mcl:acquireIron")
elseif stack:get_name() == "mcl_fishing:fish_cooked" then
awards.unlock(player:get_player_name(), "mcl:cookFish")
end
give_xp(pos, player)
end
end
override.on_rotate = pipeworks.on_rotate
local override_active = table.copy(override)
override_active.tiles = {
"default_furnace_top.png"..tube_entry,
"default_furnace_bottom.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_side.png"..tube_entry,
"default_furnace_front_active.png",
}
override_active.groups = furnace_groups_active
override_active.tube = {
insert_object = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
--blast furnace
local override_blast_furnace = {}
override_blast_furnace.tiles = {
"blast_furnace_top.png"..tube_entry,
"blast_furnace_top.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
"blast_furnace_front.png"
}
override_blast_furnace.groups = blast_furnace_groups
override_blast_furnace.tube = {
insert_object = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
if meta:get_int("split_material_stacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
override_blast_furnace.after_place_node = function(pos, placer, itemstack, pointed_thing)
pipeworks.after_place(pos, placer, itemstack, pointed_thing)
end
override_blast_furnace.after_dig_node = function(pos, oldnode, oldmetadata, digger)
old_blast_furnace.after_dig_node(pos, oldnode, oldmetadata, digger)
pipeworks.after_dig(pos)
end
override_blast_furnace.on_metadata_inventory_take = function(pos, listname, index, stack, player)
-- Award smelting achievements
if listname == "dst" then
if stack:get_name() == "mcl_core:iron_ingot" then
awards.unlock(player:get_player_name(), "mcl:acquireIron")
end
give_xp(pos, player)
end
end
override_blast_furnace.on_rotate = pipeworks.on_rotate
local override_blast_active = table.copy(override)
override_blast_active.tiles = {
"blast_furnace_top.png"..tube_entry,
"blast_furnace_top.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
"blast_furnace_side.png"..tube_entry,
{
name = "blast_furnace_front_on.png",
animation = { type = "vertical_frames", aspect_w = 16, aspect_h = 16, length = 48 }
},
}
override_blast_active.groups = blast_furnace_groups_active
override_blast_active.tube = {
insert_object = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
-- smoker
local override_smoker = {}
override_smoker.tiles = {
"smoker_top.png"..tube_entry,
"smoker_bottom.png"..tube_entry,
"smoker_side.png"..tube_entry,
"smoker_side.png"..tube_entry,
"smoker_side.png"..tube_entry,
"smoker_front.png"
}
override_smoker.groups = smoker_groups
override_smoker.tube = {
insert_object = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
if meta:get_int("split_material_stacks") == 1 then
stack = stack:peek_item(1)
end
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
override_smoker.after_place_node = function(pos, placer, itemstack, pointed_thing)
pipeworks.after_place(pos, placer, itemstack, pointed_thing)
end
override_smoker.after_dig_node = function(pos, oldnode, oldmetadata, digger)
old_smoker.after_dig_node(pos, oldnode, oldmetadata, digger)
pipeworks.after_dig(pos)
end
override_smoker.on_metadata_inventory_take = function(pos, listname, index, stack, player)
-- Award fish achievements
if listname == "dst" then
if stack:get_name() == "mcl_fishing:fish_cooked" then
awards.unlock(player:get_player_name(), "mcl:cookFish")
end
give_xp(pos, player)
end
end
override_smoker.on_rotate = pipeworks.on_rotate
local override_smoker_active = table.copy(override)
override_smoker_active.tiles = {
"smoker_top.png"..tube_entry,
"smoker_bottom.png"..tube_entry,
"smoker_side.png"..tube_entry,
"smoker_side.png"..tube_entry,
"smoker_side.png"..tube_entry,
{
name = "smoker_front_on.png",
animation = { type = "vertical_frames", aspect_w = 16, aspect_h = 16, length = 48 }
},
}
override_smoker_active.groups = smoker_groups_active
override_smoker_active.tube = {
insert_object = function(pos,node,stack,direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
local timer = core.get_node_timer(pos)
if not timer:is_started() then
timer:start(1.0)
end
if direction.y == 1 then
return inv:add_item("fuel", stack)
else
return inv:add_item("src", stack)
end
end,
can_insert = function(pos, node, stack, direction)
local meta = core.get_meta(pos)
local inv = meta:get_inventory()
if direction.y == 1 then
return inv:room_for_item("fuel", stack)
else
return inv:room_for_item("src", stack)
end
end,
input_inventory = "dst",
connect_sides = {left = 1, right = 1, back = 1, bottom = 1, top = 1}
}
-- override
core.override_item("mcl_furnaces:furnace", override)
core.override_item("mcl_furnaces:furnace_active", override_active)
core.override_item("mcl_blast_furnace:blast_furnace", override_blast_furnace)
core.override_item("mcl_blast_furnace:blast_furnace_active", override_blast_active)
core.override_item("mcl_smoker:smoker", override_smoker)
core.override_item("mcl_smoker:smoker_active", override_smoker_active)

View File

@@ -1,5 +1,5 @@
name = pipeworks name = pipeworks
description = This mod uses mesh nodes and nodeboxes to supply a complete set of 3D pipes and tubes, along with devices that work with them. description = This mod uses mesh nodes and nodeboxes to supply a complete set of 3D pipes and tubes, along with devices that work with them.
depends = default, basic_materials, screwdriver depends = basic_materials, xcompat, fakelib
optional_depends = mesecons, mesecons_mvps, digilines, signs_lib, unified_inventory optional_depends = mesecons, mesecons_mvps, digilines, signs_lib, unified_inventory, default, screwdriver, fl_mapgen, sound_api, i3, hades_core, hades_furnaces, hades_chests, mcl_mapgen_core, mcl_barrels, mcl_furnaces, mcl_experience, vizlib
min_minetest_version = 5.2.0 min_minetest_version = 5.5.0

View File

@@ -1,5 +1,5 @@
-- This file supplies the steel pipes -- This file supplies the steel pipes
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local REGISTER_COMPATIBILITY = true local REGISTER_COMPATIBILITY = true
@@ -37,11 +37,11 @@ for index, connects in ipairs(cconnects) do
end end
--]] --]]
local pgroups = {snappy = 3, pipe = 1, not_in_creative_inventory = 1} local pgroups = {snappy = 3, pipe = 1, not_in_creative_inventory = 1, dig_generic = 4, axey = 1, handy = 1, pickaxey = 1}
local pipedesc = S("Pipe Segment").." "..dump(connects) local pipedesc = S("Pipe Segment").." "..dump(connects)
if #connects == 0 then if #connects == 0 then
pgroups = {snappy = 3, tube = 1} pgroups = {snappy = 3, pipe = 1, dig_generic = 4, axey = 1, handy = 1, pickaxey = 1}
pipedesc = S("Pipe Segment") pipedesc = S("Pipe Segment")
end end
@@ -59,7 +59,7 @@ for index, connects in ipairs(cconnects) do
mesh = "pipeworks_pipe_3"..polys..".obj" mesh = "pipeworks_pipe_3"..polys..".obj"
end end
minetest.register_node("pipeworks:pipe_"..index.."_empty", { core.register_node("pipeworks:pipe_"..index.."_empty", {
description = pipedesc, description = pipedesc,
drawtype = "mesh", drawtype = "mesh",
mesh = mesh, mesh = mesh,
@@ -76,7 +76,11 @@ for index, connects in ipairs(cconnects) do
fixed = outsel fixed = outsel
}, },
groups = pgroups, groups = pgroups,
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness = 0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
drop = "pipeworks:pipe_1_empty", drop = "pipeworks:pipe_1_empty",
after_place_node = function(pos) after_place_node = function(pos)
@@ -91,9 +95,9 @@ for index, connects in ipairs(cconnects) do
pipenumber = index pipenumber = index
}) })
local pgroups = {snappy = 3, pipe = 1, not_in_creative_inventory = 1} local pgroups = {snappy = 3, pipe = 1, not_in_creative_inventory = 1, dig_generic = 4, axey = 1, handy = 1, pickaxey = 1}
minetest.register_node("pipeworks:pipe_"..index.."_loaded", { core.register_node("pipeworks:pipe_"..index.."_loaded", {
description = pipedesc, description = pipedesc,
drawtype = "mesh", drawtype = "mesh",
mesh = mesh, mesh = mesh,
@@ -110,11 +114,15 @@ for index, connects in ipairs(cconnects) do
fixed = outsel fixed = outsel
}, },
groups = pgroups, groups = pgroups,
sounds = default.node_sound_metal_defaults(), is_ground_content = false,
_mcl_hardness = 0.8,
_sound_def = {
key = "node_sound_metal_defaults",
},
walkable = true, walkable = true,
drop = "pipeworks:pipe_1_empty", drop = "pipeworks:pipe_1_empty",
after_place_node = function(pos) after_place_node = function(pos)
minetest.set_node(pos, { name = "pipeworks:pipe_"..index.."_empty" }) core.set_node(pos, { name = "pipeworks:pipe_"..index.."_empty" })
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
end, end,
after_dig_node = function(pos) after_dig_node = function(pos)
@@ -139,12 +147,13 @@ pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:pipe_1_em
if REGISTER_COMPATIBILITY then if REGISTER_COMPATIBILITY then
local cempty = "pipeworks:pipe_compatibility_empty" local cempty = "pipeworks:pipe_compatibility_empty"
local cloaded = "pipeworks:pipe_compatibility_loaded" local cloaded = "pipeworks:pipe_compatibility_loaded"
minetest.register_node(cempty, { core.register_node(cempty, {
drawtype = "airlike", drawtype = "airlike",
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
description = S("Pipe Segment (legacy)"), description = S("Pipe Segment (legacy)"),
groups = {not_in_creative_inventory = 1, pipe_to_update = 1}, groups = {not_in_creative_inventory = 1, pipe_to_update = 1},
is_ground_content = false,
drop = "pipeworks:pipe_1_empty", drop = "pipeworks:pipe_1_empty",
after_place_node = function(pos) after_place_node = function(pos)
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
@@ -152,11 +161,12 @@ if REGISTER_COMPATIBILITY then
on_rotate = false on_rotate = false
}) })
minetest.register_node(cloaded, { core.register_node(cloaded, {
drawtype = "airlike", drawtype = "airlike",
sunlight_propagates = true, sunlight_propagates = true,
paramtype = "light", paramtype = "light",
groups = {not_in_creative_inventory = 1, pipe_to_update = 1}, groups = {not_in_creative_inventory = 1, pipe_to_update = 1},
is_ground_content = false,
drop = "pipeworks:pipe_1_empty", drop = "pipeworks:pipe_1_empty",
after_place_node = function(pos) after_place_node = function(pos)
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
@@ -171,22 +181,22 @@ if REGISTER_COMPATIBILITY then
for zm = 0, 1 do for zm = 0, 1 do
for zp = 0, 1 do for zp = 0, 1 do
local pname = xm..xp..ym..yp..zm..zp local pname = xm..xp..ym..yp..zm..zp
minetest.register_alias("pipeworks:pipe_"..pname.."_empty", cempty) core.register_alias("pipeworks:pipe_"..pname.."_empty", cempty)
minetest.register_alias("pipeworks:pipe_"..pname.."_loaded", cloaded) core.register_alias("pipeworks:pipe_"..pname.."_loaded", cloaded)
end end
end end
end end
end end
end end
end end
minetest.register_abm({ core.register_abm({
nodenames = {"group:pipe_to_update"}, nodenames = {"group:pipe_to_update"},
interval = 1, interval = 1,
chance = 1, chance = 1,
action = function(pos, node, active_object_count, active_object_count_wider) action = function(pos, node, active_object_count, active_object_count_wider)
local minp = {x = pos.x-1, y = pos.y-1, z = pos.z-1} local minp = {x = pos.x-1, y = pos.y-1, z = pos.z-1}
local maxp = {x = pos.x+1, y = pos.y+1, z = pos.z+1} local maxp = {x = pos.x+1, y = pos.y+1, z = pos.z+1}
if table.getn(minetest.find_nodes_in_area(minp, maxp, "ignore")) == 0 then if table.getn(core.find_nodes_in_area(minp, maxp, "ignore")) == 0 then
pipeworks.scan_for_pipe_objects(pos) pipeworks.scan_for_pipe_objects(pos)
end end
end end
@@ -219,7 +229,7 @@ pipeworks.pipes_empty_nodenames = pipes_empty_nodenames
if pipeworks.toggles.pipe_mode == "classic" then if pipeworks.toggles.pipe_mode == "classic" then
minetest.register_abm({ core.register_abm({
nodenames = pipes_empty_nodenames, nodenames = pipes_empty_nodenames,
interval = 1, interval = 1,
chance = 1, chance = 1,
@@ -228,7 +238,7 @@ minetest.register_abm({
end end
}) })
minetest.register_abm({ core.register_abm({
nodenames = pipes_full_nodenames, nodenames = pipes_full_nodenames,
interval = 1, interval = 1,
chance = 1, chance = 1,
@@ -237,7 +247,7 @@ minetest.register_abm({
end end
}) })
minetest.register_abm({ core.register_abm({
nodenames = {"pipeworks:spigot","pipeworks:spigot_pouring"}, nodenames = {"pipeworks:spigot","pipeworks:spigot_pouring"},
interval = 1, interval = 1,
chance = 1, chance = 1,
@@ -246,7 +256,7 @@ minetest.register_abm({
end end
}) })
minetest.register_abm({ core.register_abm({
nodenames = {"pipeworks:fountainhead","pipeworks:fountainhead_pouring"}, nodenames = {"pipeworks:fountainhead","pipeworks:fountainhead_pouring"},
interval = 1, interval = 1,
chance = 1, chance = 1,

View File

@@ -11,7 +11,7 @@ local flowlogic = pipeworks.flowlogic
local register_flowlogic_abm = function(nodename) local register_flowlogic_abm = function(nodename)
if pipeworks.toggles.pipe_mode == "pressure" then if pipeworks.toggles.pipe_mode == "pressure" then
minetest.register_abm({ core.register_abm({
label = "pipeworks new_flow_logic run", label = "pipeworks new_flow_logic run",
nodenames = { nodename }, nodenames = { nodename },
interval = 1, interval = 1,
@@ -21,7 +21,7 @@ local register_flowlogic_abm = function(nodename)
end end
}) })
else else
minetest.log("warning", "pipeworks pressure_logic not enabled but register.flowlogic() requested") core.log("warning", "pipeworks pressure_logic not enabled but register.flowlogic() requested")
end end
end end
register.flowlogic = register_flowlogic_abm register.flowlogic = register_flowlogic_abm

View File

@@ -41,9 +41,9 @@ local check_for_liquids_v2 = function(pos, limit)
local total = 0 local total = 0
for _, tpos in ipairs(coords) do for _, tpos in ipairs(coords) do
if total >= limit then break end if total >= limit then break end
local name = minetest.get_node(tpos).name local name = core.get_node(tpos).name
if name == "default:water_source" then if name == pipeworks.liquids.water.source then
minetest.remove_node(tpos) core.remove_node(tpos)
total = total + 1 total = total + 1
end end
end end
@@ -56,7 +56,7 @@ flowlogic.check_for_liquids_v2 = check_for_liquids_v2
local label_pressure = "pipeworks.water_pressure" local label_pressure = "pipeworks.water_pressure"
local get_pressure_access = function(pos) local get_pressure_access = function(pos)
local metaref = minetest.get_meta(pos) local metaref = core.get_meta(pos)
return { return {
get = function() get = function()
return metaref:get_float(label_pressure) return metaref:get_float(label_pressure)
@@ -114,7 +114,7 @@ flowlogic.run = function(pos, node)
if pipeworks.flowables.transitions.list[nodename] then if pipeworks.flowables.transitions.list[nodename] then
local newnode = flowlogic.run_transition(node, currentpressure) local newnode = flowlogic.run_transition(node, currentpressure)
--pipeworks.logger("flowlogic.run()@"..formatvec(pos).." transition, new node name = "..dump(newnode).." pressure "..tostring(currentpressure)) --pipeworks.logger("flowlogic.run()@"..formatvec(pos).." transition, new node name = "..dump(newnode).." pressure "..tostring(currentpressure))
minetest.swap_node(pos, newnode) core.swap_node(pos, newnode)
flowlogic.run_transition_post(pos, newnode) flowlogic.run_transition_post(pos, newnode)
end end
@@ -155,7 +155,7 @@ local get_neighbour_positions = function(pos, node)
local connections = {} local connections = {}
for _, offset in ipairs(candidates) do for _, offset in ipairs(candidates) do
local npos = vector.add(pos, offset) local npos = vector.add(pos, offset)
local neighbour = minetest.get_node(npos) local neighbour = core.get_node(npos)
local nodename = neighbour.name local nodename = neighbour.name
local is_simple = (pipeworks.flowables.list.simple[nodename]) local is_simple = (pipeworks.flowables.list.simple[nodename])
if is_simple then if is_simple then
@@ -241,15 +241,15 @@ flowlogic.helpers.make_neighbour_output_fixed = function(neighbours)
local taken = 0 local taken = 0
for _, offset in pairs(neighbours) do for _, offset in pairs(neighbours) do
local npos = vector.add(pos, offset) local npos = vector.add(pos, offset)
local name = minetest.get_node(npos).name local name = core.get_node(npos).name
if currentpressure < 1 then break end if currentpressure < 1 then break end
-- take pressure anyway in non-finite mode, even if node is water source already. -- take pressure anyway in non-finite mode, even if node is water source already.
-- in non-finite mode, pressure has to be sustained to keep the sources there. -- in non-finite mode, pressure has to be sustained to keep the sources there.
-- so in non-finite mode, placing water is dependent on the target node; -- so in non-finite mode, placing water is dependent on the target node;
-- draining pressure is not. -- draining pressure is not.
local canplace = (name == "air") or (name == "default:water_flowing") local canplace = (name == "air") or (name == pipeworks.liquids.water.flowing)
if canplace then if canplace then
minetest.swap_node(npos, {name="default:water_source"}) core.swap_node(npos, {name=pipeworks.liquids.water.source})
end end
if (not finitemode) or canplace then if (not finitemode) or canplace then
taken = taken + 1 taken = taken + 1
@@ -267,10 +267,10 @@ flowlogic.helpers.make_neighbour_cleanup_fixed = function(neighbours)
--pipeworks.logger("neighbour_cleanup_fixed@"..formatvec(pos)) --pipeworks.logger("neighbour_cleanup_fixed@"..formatvec(pos))
for _, offset in pairs(neighbours) do for _, offset in pairs(neighbours) do
local npos = vector.add(pos, offset) local npos = vector.add(pos, offset)
local name = minetest.get_node(npos).name local name = core.get_node(npos).name
if (name == "default:water_source") then if (name == pipeworks.liquids.water.source) then
--pipeworks.logger("neighbour_cleanup_fixed removing "..formatvec(npos)) --pipeworks.logger("neighbour_cleanup_fixed removing "..formatvec(npos))
minetest.remove_node(npos) core.remove_node(npos)
end end
end end
end end
@@ -351,9 +351,9 @@ end
-- among other things, updates mesecons if present. -- among other things, updates mesecons if present.
-- node here means the new node, returned from run_transition() above -- node here means the new node, returned from run_transition() above
flowlogic.run_transition_post = function(pos, node) flowlogic.run_transition_post = function(pos, node)
local mesecons_def = minetest.registered_nodes[node.name].mesecons local mesecons_def = core.registered_nodes[node.name].mesecons
local mesecons_rules = pipeworks.flowables.transitions.mesecons[node.name] local mesecons_rules = pipeworks.flowables.transitions.mesecons[node.name]
if minetest.global_exists("mesecon") and (mesecons_def ~= nil) and mesecons_rules then if core.get_modpath("mesecons") and (mesecons_def ~= nil) and mesecons_rules then
if type(mesecons_def) ~= "table" then if type(mesecons_def) ~= "table" then
pipeworks.logger("flowlogic.run_transition_post() BUG mesecons def for "..node.name.."not a table: got "..tostring(mesecons_def)) pipeworks.logger("flowlogic.run_transition_post() BUG mesecons def for "..node.name.."not a table: got "..tostring(mesecons_def))
else else

View File

@@ -75,3 +75,27 @@ pipeworks_drop_on_routing_fail (Drop On Routing Fail) bool false
#Delete item on clearobject. #Delete item on clearobject.
pipeworks_delete_item_on_clearobject (Delete Item On Clearobject) bool true pipeworks_delete_item_on_clearobject (Delete Item On Clearobject) bool true
#Use real visible entities in tubes within active areas.
#When disabled, tubes are made opaque.
pipeworks_use_real_entities (Use Real Entities) bool true
#Target interval between tube entity steps.
#A high value may cause issues with tube entity visuals.
#A value 0.2 or above may cause issues with accelerator tubes.
pipeworks_entity_update_interval (Entity Update Interval) float 0 0 0.8
# Use the default rules from the digilines mod.
# If enabled the following devices will connect to digiline networks in the vertical direction:
# digiline filter injector, deployer, dispenser, node breaker, autocrafter
# This breaks expected behavior with digiline conducting tubes.
# If disabled, the devices will not be able to send or recieve digiline signals from the top
# or bottom faces, regardless of the node rotation.
enable_vertical_digilines_connectivity (Use the default rules from the digilines mod) bool false
# if set to true, items passing through teleport tubes will log log where they came from and where they went.
pipeworks_log_teleport_tubes (Log Teleport Tubes) bool false
# Behavior of print() inside a lua tube. By default, this emits a message into actionstream.
# Set it to noop if you wish to disable that behavior.
pipeworks_lua_tube_print_behavior (Behavior of print in Lua Tube) enum log log,noop

View File

@@ -1,293 +0,0 @@
local S = minetest.get_translator("pipeworks")
local filename=minetest.get_worldpath() .. "/teleport_tubes"
local tp_tube_db = nil -- nil forces a read
local tp_tube_db_version = 2.0
-- cached rceiver list: hash(pos) => {receivers}
local cache = {}
local function hash(pos)
return string.format("%.30g", minetest.hash_node_position(pos))
end
local function save_tube_db()
local file, err = io.open(filename, "w")
if file then
tp_tube_db.version = tp_tube_db_version
file:write(minetest.serialize(tp_tube_db))
tp_tube_db.version = nil
io.close(file)
else
error(err)
end
-- reset tp-tube cache
cache = {}
end
local function migrate_tube_db()
local tmp_db = {}
tp_tube_db.version = nil
for _, val in pairs(tp_tube_db) do
if(val.channel ~= "") then -- skip unconfigured tubes
tmp_db[hash(val)] = val
end
end
tp_tube_db = tmp_db
save_tube_db()
end
local function read_tube_db()
local file = io.open(filename, "r")
if file ~= nil then
local file_content = file:read("*all")
io.close(file)
if file_content and file_content ~= "" then
tp_tube_db = minetest.deserialize(file_content)
if(not tp_tube_db.version or tonumber(tp_tube_db.version) < tp_tube_db_version) then
migrate_tube_db()
end
tp_tube_db.version = nil -- we add it back when saving
return tp_tube_db -- we read sucessfully
end
end
tp_tube_db = {}
return tp_tube_db
end
-- debug formatter for coordinates used below
local fmt = function(pos)
return pos.x..", "..pos.y..", "..pos.z
end
-- updates or adds a tube
local function set_tube(pos, channel, can_receive)
local tubes = tp_tube_db or read_tube_db()
local hash = hash(pos)
local tube = tubes[hash]
if tube then
tube.channel = channel
tube.cr = can_receive
save_tube_db()
return
end
-- we haven't found any tp tube to update, so lets add it
-- but sanity check that the hash has not already been inserted.
-- if so, complain very loudly and refuse the update so the player knows something is amiss.
-- to catch regressions of https://github.com/minetest-mods/pipeworks/issues/166
local existing = tp_tube_db[hash]
if existing ~= nil then
local e = "error"
minetest.log(e, "pipeworks teleport tube update refused due to position hash collision")
minetest.log(e, "collided hash: "..hash)
minetest.log(e, "tried-to-place tube: "..fmt(pos))
minetest.log(e, "existing tube: "..fmt(existing))
return
end
tp_tube_db[hash] = {x=pos.x,y=pos.y,z=pos.z,channel=channel,cr=can_receive}
save_tube_db()
end
local function remove_tube(pos)
local tubes = tp_tube_db or read_tube_db()
tubes[hash(pos)] = nil
save_tube_db()
end
local function read_node_with_vm(pos)
local vm = VoxelManip()
local MinEdge, MaxEdge = vm:read_from_map(pos, pos)
local data = vm:get_data()
local area = VoxelArea:new({MinEdge = MinEdge, MaxEdge = MaxEdge})
return minetest.get_name_from_content_id(data[area:index(pos.x, pos.y, pos.z)])
end
local function get_receivers(pos, channel)
local hash = minetest.hash_node_position(pos)
if cache[hash] then
-- re-use cached result
return cache[hash]
end
local tubes = tp_tube_db or read_tube_db()
local receivers = {}
local dirty = false
for key, val in pairs(tubes) do
-- skip all non-receivers and the tube that it came from as early as possible, as this is called often
if (val.cr == 1 and val.channel == channel and (val.x ~= pos.x or val.y ~= pos.y or val.z ~= pos.z)) then
local is_loaded = (minetest.get_node_or_nil(val) ~= nil)
local node_name = is_loaded and minetest.get_node(pos).name or read_node_with_vm(val)
if minetest.registered_nodes[node_name] and minetest.registered_nodes[node_name].is_teleport_tube then
table.insert(receivers, val)
else
tp_tube_db[key] = nil
dirty = true
end
end
end
if dirty then
save_tube_db()
end
-- cache the result for next time
cache[hash] = receivers
return receivers
end
local function update_meta(meta, can_receive)
meta:set_int("can_receive", can_receive and 1 or 0)
local cr_state = can_receive and "on" or "off"
local itext = S("Channels are public by default").."\n"..
S("Use <player>:<channel> for fully private channels").."\n"..
S("Use <player>\\;<channel> for private receivers")
meta:set_string("formspec",
"size[8.5,3.5]"..
"image[0.2,o;1,1;pipeworks_teleport_tube_inv.png]"..
"label[1.2,0.2;"..S("Teleporting Tube").."]"..
"field[0.5,1.6;4.6,1;channel;"..S("Channel")..";${channel}]"..
"button[4.8,1.3;1.5,1;set_channel;"..S("Set").."]"..
"label[7.0,0;"..S("Receive").."]"..
"image_button[7.0,0.5;1,0.6;pipeworks_button_" .. cr_state .. ".png;cr" .. (can_receive and 0 or 1) .. ";;;false;pipeworks_button_interm.png]"..
"button_exit[6.3,1.3;2,1;close;"..S("Close").."]"..
"label[0.2,2.3;"..itext.."]"..
default.gui_bg..
default.gui_bg_img)
end
pipeworks.register_tube("pipeworks:teleport_tube", {
description = S("Teleporting Pneumatic Tube Segment"),
inventory_image = "pipeworks_teleport_tube_inv.png",
noctr = { "pipeworks_teleport_tube_noctr.png" },
plain = { "pipeworks_teleport_tube_plain.png" },
ends = { "pipeworks_teleport_tube_end.png" },
short = "pipeworks_teleport_tube_short.png",
node_def = {
is_teleport_tube = true,
tube = {
can_go = function(pos,node,velocity,stack)
velocity.x = 0
velocity.y = 0
velocity.z = 0
local channel = minetest.get_meta(pos):get_string("channel")
if channel == "" then return {} end
local target = get_receivers(pos, channel)
if target[1] == nil then return {} end
local d = math.random(1,#target)
pos.x = target[d].x
pos.y = target[d].y
pos.z = target[d].z
return pipeworks.meseadjlist
end
},
on_construct = function(pos)
local meta = minetest.get_meta(pos)
update_meta(meta, true)
meta:set_string("infotext", S("Unconfigured Teleportation Tube"))
end,
on_receive_fields = function(pos,formname,fields,sender)
if not fields.channel -- ignore escaping or clientside manipulation of the form
or (fields.quit and not fields.key_enter_field)
or not pipeworks.may_configure(pos, sender) then
return
end
local new_channel = tostring(fields.channel):trim()
local meta = minetest.get_meta(pos)
local can_receive = meta:get_int("can_receive")
-- check for private channels each time before actually changing anything
-- to not even allow switching between can_receive states of private channels
if new_channel ~= "" then
local sender_name = sender:get_player_name()
local name, mode = new_channel:match("^([^:;]+)([:;])")
if name and mode and name ~= sender_name then
--channels starting with '[name]:' can only be used by the named player
if mode == ":" then
minetest.chat_send_player(sender_name, S("Sorry, channel '@1' is reserved for exclusive use by @2",
new_channel, name))
return
--channels starting with '[name];' can be used by other players, but cannot be received from
elseif mode == ";" and (fields.cr1 or (can_receive ~= 0 and not fields.cr0)) then
minetest.chat_send_player(sender_name, S("Sorry, receiving from channel '@1' is reserved for @2",
new_channel, name))
return
end
end
end
local dirty = false
-- was the channel changed?
local channel = meta:get_string("channel")
if new_channel ~= channel and (fields.key_enter_field == "channel" or fields.set_channel) then
channel = new_channel
meta:set_string("channel", channel)
dirty = true
end
-- test if a can_receive button was pressed
if fields.cr0 and can_receive ~= 0 then
can_receive = 0
update_meta(meta, false)
dirty = true
elseif fields.cr1 and can_receive ~= 1 then
can_receive = 1
update_meta(meta, true)
dirty = true
end
-- save if we changed something, handle the empty channel while we're at it
if dirty then
if channel ~= "" then
set_tube(pos, channel, can_receive)
local cr_description = (can_receive == 1) and "sending and receiving" or "sending"
meta:set_string("infotext", S("Teleportation Tube @1 on '@2'", cr_description, channel))
else
-- remove empty channel tubes, to not have to search through them
remove_tube(pos)
meta:set_string("infotext", S("Unconfigured Teleportation Tube"))
end
end
end,
on_destruct = function(pos)
remove_tube(pos)
end
},
})
minetest.register_craft( {
output = "pipeworks:teleport_tube_1 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "default:desert_stone", "default:mese", "default:desert_stone" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
if minetest.get_modpath("mesecons_mvps") ~= nil then
mesecon.register_on_mvps_move(function(moved_nodes)
for _, n in ipairs(moved_nodes) do
if string.find(n.node.name, "pipeworks:teleport_tube") ~= nil then
local meta = minetest.get_meta(n.pos)
set_tube(n.pos, meta:get_string("channel"), meta:get_int("can_receive"))
end
end
end)
end
-- Expose teleport tube database API for other mods
pipeworks.tptube = {
hash = hash,
save_tube_db = save_tube_db,
get_db = function() return tp_tube_db or read_tube_db() end,
set_tube = set_tube,
update_meta = update_meta,
tp_tube_db_version = tp_tube_db_version
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 875 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 496 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 768 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 744 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 743 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 747 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 769 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 753 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 864 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 865 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 863 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 864 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 871 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 860 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 457 B

View File

@@ -1,5 +1,5 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
minetest.register_node("pipeworks:trashcan", { core.register_node("pipeworks:trashcan", {
description = S("Trash Can"), description = S("Trash Can"),
drawtype = "normal", drawtype = "normal",
tiles = { tiles = {
@@ -10,7 +10,9 @@ minetest.register_node("pipeworks:trashcan", {
"pipeworks_trashcan_side.png", "pipeworks_trashcan_side.png",
"pipeworks_trashcan_side.png", "pipeworks_trashcan_side.png",
}, },
groups = {snappy = 3, tubedevice = 1, tubedevice_receiver = 1}, groups = {snappy = 3, tubedevice = 1, tubedevice_receiver = 1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
is_ground_content = false,
_mcl_hardness=0.8,
tube = { tube = {
insert_object = function(pos, node, stack, direction) insert_object = function(pos, node, stack, direction)
return ItemStack("") return ItemStack("")
@@ -19,34 +21,32 @@ minetest.register_node("pipeworks:trashcan", {
priority = 1, -- Lower than anything else priority = 1, -- Lower than anything else
}, },
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local size = "10.2,9"
local list_background = ""
if core.get_modpath("i3") or core.get_modpath("mcl_formspec") then
list_background = "style_type[box;colors=#666]box[4.5,2;1,1;]"
end
meta:set_string("formspec", meta:set_string("formspec",
"size[8,7]".. "formspec_version[2]" ..
"item_image[0,0;1,1;pipeworks:trashcan]".. "size["..size.."]"..
"label[1,0;"..S("Trash Can").."]".. pipeworks.fs_helpers.get_prepends(size) ..
"list[context;trash;3.5,1;1,1;]".. "item_image[0.5,0.5;1,1;pipeworks:trashcan]"..
default.gui_bg.. "label[1.5,1;"..S("Trash Can").."]"..
default.gui_bg_img.. list_background..
default.gui_slots.. "list[context;trash;4.5,2;1,1;]"..
default.get_hotbar_bg(0,3) .. --"list[current_player;main;0,3;8,4;]" ..
"list[current_player;main;0,3;8,4;]" .. pipeworks.fs_helpers.get_inv(4)..
"listring[]") "listring[context;trash]"..
"listring[current_player;main]"
)
meta:set_string("infotext", S("Trash Can")) meta:set_string("infotext", S("Trash Can"))
meta:get_inventory():set_size("trash", 1) meta:get_inventory():set_size("trash", 1)
end, end,
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
on_metadata_inventory_put = function(pos, listname, index, stack, player) on_metadata_inventory_put = function(pos, listname, index, stack, player)
minetest.get_meta(pos):get_inventory():set_stack(listname, index, ItemStack("")) core.get_meta(pos):get_inventory():set_stack(listname, index, ItemStack(""))
end, end,
}) })
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:trashcan" pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:trashcan"
minetest.register_craft({
output = "pipeworks:trashcan",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "default:steel_ingot", "", "default:steel_ingot" },
{ "default:steel_ingot", "default:steel_ingot", "default:steel_ingot" },
},
})

69
tubes/embedded_tube.lua Normal file
View File

@@ -0,0 +1,69 @@
local materials = xcompat.materials
local S = core.get_translator("pipeworks")
local straight = function(pos, node, velocity, stack) return {velocity} end
local steel_tex = "[combine:16x16^[noalpha^[colorize:#D3D3D3"
if core.get_modpath("default") then steel_tex = "default_steel_block.png" end
-- register an embedded tube
function pipeworks.register_embedded_tube(nodename, opts)
core.register_node(nodename, {
description = opts.description,
tiles = {
opts.base_texture,
opts.base_texture,
opts.base_texture,
opts.base_texture,
opts.base_texture .. "^pipeworks_tube_connection_metallic.png",
opts.base_texture .. "^pipeworks_tube_connection_metallic.png",
},
paramtype = "light",
paramtype2 = "facedir",
groups = {
cracky = 1,
oddly_breakable_by_hand = 1,
tubedevice = 1,
dig_glass = 2,
pickaxey=1,
handy=1
},
is_ground_content = false,
_mcl_hardness = 0.8,
legacy_facedir_simple = true,
_sound_def = {
key = "node_sound_stone_defaults",
},
tube = {
connect_sides = {
front = 1,
back = 1
},
priority = 50,
can_go = straight,
can_insert = function(pos, node, stack, direction)
local dir = core.facedir_to_dir(node.param2)
return vector.equals(dir, direction) or vector.equals(vector.multiply(dir, -1), direction)
end
},
after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate,
})
core.register_craft( {
output = nodename .. " 1",
recipe = {
{ opts.base_ingredient, opts.base_ingredient, opts.base_ingredient },
{ opts.base_ingredient, "pipeworks:tube_1", opts.base_ingredient },
{ opts.base_ingredient, opts.base_ingredient, opts.base_ingredient }
},
})
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = nodename
end
-- steelblock embedded tube
pipeworks.register_embedded_tube("pipeworks:steel_block_embedded_tube", {
description = S("Airtight steelblock embedded tube"),
base_texture = steel_tex,
base_ingredient = materials.steel_ingot
})

View File

@@ -26,10 +26,12 @@
-- use too much memory from the sandbox. -- use too much memory from the sandbox.
-- You can add more functions to the environment -- You can add more functions to the environment
-- (see where local env is defined) -- (see where local env is defined)
-- Something nice to play is appending minetest.env to it. -- Something nice to play is appending core.env to it.
local BASENAME = "pipeworks:lua_tube" local BASENAME = "pipeworks:lua_tube"
local has_digilines = core.get_modpath("digilines")
local rules = { local rules = {
red = {x = -1, y = 0, z = 0, name = "red"}, red = {x = -1, y = 0, z = 0, name = "red"},
blue = {x = 1, y = 0, z = 0, name = "blue"}, blue = {x = 1, y = 0, z = 0, name = "blue"},
@@ -63,7 +65,7 @@ local digiline_rules_luatube = {
-- These helpers are required to set the port states of the lua_tube -- These helpers are required to set the port states of the lua_tube
local function update_real_port_states(pos, rule_name, new_state) local function update_real_port_states(pos, rule_name, new_state)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
if rule_name == nil then if rule_name == nil then
meta:set_int("real_portstates", 1) meta:set_int("real_portstates", 1)
return return
@@ -100,7 +102,7 @@ local port_names = {"red", "blue", "yellow", "green", "black", "white"}
local function get_real_port_states(pos) local function get_real_port_states(pos)
-- Determine if ports are powered (by itself or from outside) -- Determine if ports are powered (by itself or from outside)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local L = {} local L = {}
local n = meta:get_int("real_portstates") - 1 local n = meta:get_int("real_portstates") - 1
for _, name in ipairs(port_names) do for _, name in ipairs(port_names) do
@@ -153,10 +155,10 @@ end
local function set_port_states(pos, ports) local function set_port_states(pos, ports)
local node = minetest.get_node(pos) local node = core.get_node(pos)
local name = node.name local name = node.name
clean_port_states(ports) clean_port_states(ports)
local vports = minetest.registered_nodes[name].virtual_portstates local vports = core.registered_nodes[name].virtual_portstates
local new_name = generate_name(ports) local new_name = generate_name(ports)
if name ~= new_name and vports then if name ~= new_name and vports then
@@ -169,17 +171,17 @@ local function set_port_states(pos, ports)
-- its output off. -- its output off.
-- Solution / Workaround: -- Solution / Workaround:
-- Remember which output was turned off and ignore next "off" event. -- Remember which output was turned off and ignore next "off" event.
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local ign = minetest.deserialize(meta:get_string("ignore_offevents"), true) or {} local ign = core.deserialize(meta:get_string("ignore_offevents"), true) or {}
if ports.red and not vports.red and not mesecon.is_powered(pos, rules.red) then ign.red = true end if ports.red and not vports.red and not mesecon.is_powered(pos, rules.red) then ign.red = true end
if ports.blue and not vports.blue and not mesecon.is_powered(pos, rules.blue) then ign.blue = true end if ports.blue and not vports.blue and not mesecon.is_powered(pos, rules.blue) then ign.blue = true end
if ports.yellow and not vports.yellow and not mesecon.is_powered(pos, rules.yellow) then ign.yellow = true end if ports.yellow and not vports.yellow and not mesecon.is_powered(pos, rules.yellow) then ign.yellow = true end
if ports.green and not vports.green and not mesecon.is_powered(pos, rules.green) then ign.green = true end if ports.green and not vports.green and not mesecon.is_powered(pos, rules.green) then ign.green = true end
if ports.black and not vports.black and not mesecon.is_powered(pos, rules.black) then ign.black = true end if ports.black and not vports.black and not mesecon.is_powered(pos, rules.black) then ign.black = true end
if ports.white and not vports.white and not mesecon.is_powered(pos, rules.white) then ign.white = true end if ports.white and not vports.white and not mesecon.is_powered(pos, rules.white) then ign.white = true end
meta:set_string("ignore_offevents", minetest.serialize(ign)) meta:set_string("ignore_offevents", core.serialize(ign))
minetest.swap_node(pos, {name = new_name, param2 = node.param2}) core.swap_node(pos, {name = new_name, param2 = node.param2})
if ports.red ~= vports.red then set_port(pos, rules.red, ports.red) end if ports.red ~= vports.red then set_port(pos, rules.red, ports.red) end
if ports.blue ~= vports.blue then set_port(pos, rules.blue, ports.blue) end if ports.blue ~= vports.blue then set_port(pos, rules.blue, ports.blue) end
@@ -195,12 +197,12 @@ end
-- Overheating -- -- Overheating --
----------------- -----------------
local function burn_controller(pos) local function burn_controller(pos)
local node = minetest.get_node(pos) local node = core.get_node(pos)
node.name = BASENAME.."_burnt" node.name = BASENAME.."_burnt"
minetest.swap_node(pos, node) core.swap_node(pos, node)
minetest.get_meta(pos):set_string("lc_memory", ""); core.get_meta(pos):set_string("lc_memory", "");
-- Wait for pending operations -- Wait for pending operations
minetest.after(0.2, mesecon.receptor_off, pos, mesecon.rules.flat) core.after(0.2, mesecon.receptor_off, pos, mesecon.rules.flat)
end end
local function overheat(pos, meta) local function overheat(pos, meta)
@@ -216,10 +218,10 @@ end
local function ignore_event(event, meta) local function ignore_event(event, meta)
if event.type ~= "off" then return false end if event.type ~= "off" then return false end
local ignore_offevents = minetest.deserialize(meta:get_string("ignore_offevents"), true) or {} local ignore_offevents = core.deserialize(meta:get_string("ignore_offevents"), true) or {}
if ignore_offevents[event.pin.name] then if ignore_offevents[event.pin.name] then
ignore_offevents[event.pin.name] = nil ignore_offevents[event.pin.name] = nil
meta:set_string("ignore_offevents", minetest.serialize(ignore_offevents)) meta:set_string("ignore_offevents", core.serialize(ignore_offevents))
return true return true
end end
end end
@@ -229,11 +231,13 @@ end
------------------------- -------------------------
local function safe_print(param) local function safe_print(param)
local string_meta = getmetatable("") if (core.settings:get("pipeworks_lua_tube_print_behavior") or "log") == "log" then
local sandbox = string_meta.__index local string_meta = getmetatable("")
string_meta.__index = string -- Leave string sandbox temporarily local sandbox = string_meta.__index
print(dump(param)) string_meta.__index = string -- Leave string sandbox temporarily
string_meta.__index = sandbox -- Restore string sandbox core.log("action", string.format("[pipeworks.tubes.lua] print(%s)", dump(param)))
string_meta.__index = sandbox -- Restore string sandbox
end
end end
local function safe_date() local function safe_date()
@@ -305,7 +309,7 @@ if mesecon.setting("luacontroller_lightweight_interrupts", false) then
return (function(time, iid) return (function(time, iid)
if type(time) ~= "number" then error("Delay must be a number") end if type(time) ~= "number" then error("Delay must be a number") end
if iid ~= nil then send_warning("Interrupt IDs are disabled on this server") end if iid ~= nil then send_warning("Interrupt IDs are disabled on this server") end
table.insert(itbl, function() minetest.get_node_timer(pos):start(time) end) table.insert(itbl, function() core.get_node_timer(pos):start(time) end)
end) end)
end end
else else
@@ -319,12 +323,12 @@ else
if type(time) ~= "number" then error("Delay must be a number") end if type(time) ~= "number" then error("Delay must be a number") end
table.insert(itbl, function () table.insert(itbl, function ()
-- Outside string metatable sandbox, can safely run this now -- Outside string metatable sandbox, can safely run this now
local luac_id = minetest.get_meta(pos):get_int("luac_id") local luac_id = core.get_meta(pos):get_int("luac_id")
-- Check if IID is dodgy, so you can't use interrupts to store an infinite amount of data. -- Check if IID is dodgy, so you can't use interrupts to store an infinite amount of data.
-- Note that this is safe from alter-after-free because this code gets run after the sandbox has ended. -- Note that this is safe from alter-after-free because this code gets run after the sandbox has ended.
-- This runs outside of the timer and *shouldn't* harm perf. unless dodgy data is being sent in the first place -- This runs outside of the timer and *shouldn't* harm perf. unless dodgy data is being sent in the first place
iid = remove_functions(iid) iid = remove_functions(iid)
local msg_ser = minetest.serialize(iid) local msg_ser = core.serialize(iid)
if #msg_ser <= mesecon.setting("luacontroller_interruptid_maxlen", 256) then if #msg_ser <= mesecon.setting("luacontroller_interruptid_maxlen", 256) then
mesecon.queue:add_action(pos, "pipeworks:lc_tube_interrupt", {luac_id, iid}, time, iid, 1) mesecon.queue:add_action(pos, "pipeworks:lc_tube_interrupt", {luac_id, iid}, time, iid, 1)
else else
@@ -370,7 +374,10 @@ local function clean_and_weigh_digiline_message(msg, back_references)
return msg, #msg + 25 return msg, #msg + 25
elseif t == "number" then elseif t == "number" then
-- Numbers are passed by value so need not be touched, and cost 8 bytes -- Numbers are passed by value so need not be touched, and cost 8 bytes
-- as all numbers in Lua are doubles. -- as all numbers in Lua are doubles. NaN values are removed.
if msg ~= msg then
return nil, 0
end
return msg, 8 return msg, 8
elseif t == "boolean" then elseif t == "boolean" then
-- Booleans are passed by value so need not be touched, and cost 1 -- Booleans are passed by value so need not be touched, and cost 1
@@ -425,7 +432,7 @@ end
-- itbl: Flat table of functions to run after sandbox cleanup, used to prevent various security hazards -- itbl: Flat table of functions to run after sandbox cleanup, used to prevent various security hazards
local function get_digiline_send(pos, itbl, send_warning) local function get_digiline_send(pos, itbl, send_warning)
if not minetest.global_exists("digilines") then return end if not has_digilines then return end
local chan_maxlen = mesecon.setting("luacontroller_digiline_channel_maxlen", 256) local chan_maxlen = mesecon.setting("luacontroller_digiline_channel_maxlen", 256)
local maxlen = mesecon.setting("luacontroller_digiline_maxlen", 50000) local maxlen = mesecon.setting("luacontroller_digiline_maxlen", 50000)
return function(channel, msg) return function(channel, msg)
@@ -451,7 +458,7 @@ local function get_digiline_send(pos, itbl, send_warning)
table.insert(itbl, function () table.insert(itbl, function ()
-- Runs outside of string metatable sandbox -- Runs outside of string metatable sandbox
local luac_id = minetest.get_meta(pos):get_int("luac_id") local luac_id = core.get_meta(pos):get_int("luac_id")
mesecon.queue:add_action(pos, "pipeworks:lt_digiline_relay", {channel, luac_id, msg}) mesecon.queue:add_action(pos, "pipeworks:lt_digiline_relay", {channel, luac_id, msg})
end) end)
return true return true
@@ -466,7 +473,7 @@ local safe_globals = {
local function create_environment(pos, mem, event, itbl, send_warning) local function create_environment(pos, mem, event, itbl, send_warning)
-- Make sure the tube hasn't broken. -- Make sure the tube hasn't broken.
local vports = minetest.registered_nodes[minetest.get_node(pos).name].virtual_portstates local vports = core.registered_nodes[core.get_node(pos).name].virtual_portstates
if not vports then return {} end if not vports then return {} end
-- Gather variables for the environment -- Gather variables for the environment
@@ -588,19 +595,19 @@ end
local function load_memory(meta) local function load_memory(meta)
return minetest.deserialize(meta:get_string("lc_memory"), true) or {} return core.deserialize(meta:get_string("lc_memory"), true) or {}
end end
local function save_memory(pos, meta, mem) local function save_memory(pos, meta, mem)
local memstring = minetest.serialize(remove_functions(mem)) local memstring = core.serialize(remove_functions(mem))
local memsize_max = mesecon.setting("luacontroller_memsize", 100000) local memsize_max = mesecon.setting("luacontroller_memsize", 100000)
if (#memstring <= memsize_max) then if (#memstring <= memsize_max) then
meta:set_string("lc_memory", memstring) meta:set_string("lc_memory", memstring)
meta:mark_as_private("lc_memory") meta:mark_as_private("lc_memory")
else else
print("Error: lua_tube memory overflow. "..memsize_max.." bytes available, " core.log("info", "lua_tube memory overflow. "..memsize_max.." bytes available, "
..#memstring.." required. Controller overheats.") ..#memstring.." required. Controller overheats.")
burn_controller(pos) burn_controller(pos)
end end
@@ -609,7 +616,7 @@ end
-- Returns success (boolean), errmsg (string), retval(any, return value of the user supplied code) -- Returns success (boolean), errmsg (string), retval(any, return value of the user supplied code)
-- run (as opposed to run_inner) is responsible for setting up meta according to this output -- run (as opposed to run_inner) is responsible for setting up meta according to this output
local function run_inner(pos, code, event) local function run_inner(pos, code, event)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
-- Note: These return success, presumably to avoid changing LC ID. -- Note: These return success, presumably to avoid changing LC ID.
if overheat(pos) then return true, "", nil end if overheat(pos) then return true, "", nil end
if ignore_event(event, meta) then return true, "", nil end if ignore_event(event, meta) then return true, "", nil end
@@ -663,8 +670,8 @@ end
local function reset_formspec(meta, code, errmsg) local function reset_formspec(meta, code, errmsg)
meta:set_string("code", code) meta:set_string("code", code)
meta:mark_as_private("code") meta:mark_as_private("code")
code = minetest.formspec_escape(code or "") code = core.formspec_escape(code or "")
errmsg = minetest.formspec_escape(tostring(errmsg or "")) errmsg = core.formspec_escape(tostring(errmsg or ""))
meta:set_string("formspec", "size[12,10]" meta:set_string("formspec", "size[12,10]"
.."style_type[label,textarea;font=mono]" .."style_type[label,textarea;font=mono]"
.."background[-0.2,-0.25;12.4,10.75;jeija_luac_background.png]" .."background[-0.2,-0.25;12.4,10.75;jeija_luac_background.png]"
@@ -676,14 +683,14 @@ local function reset_formspec(meta, code, errmsg)
end end
local function reset_meta(pos, code, errmsg) local function reset_meta(pos, code, errmsg)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
reset_formspec(meta, code, errmsg) reset_formspec(meta, code, errmsg)
meta:set_int("luac_id", math.random(1, 65535)) meta:set_int("luac_id", math.random(1, 65535))
end end
-- Wraps run_inner with LC-reset-on-error -- Wraps run_inner with LC-reset-on-error
local function run(pos, event) local function run(pos, event)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local code = meta:get_string("code") local code = meta:get_string("code")
local ok, errmsg, retval = run_inner(pos, code, event) local ok, errmsg, retval = run_inner(pos, code, event)
if not ok then if not ok then
@@ -700,7 +707,7 @@ local function reset(pos)
end end
local function node_timer(pos) local function node_timer(pos)
if minetest.registered_nodes[minetest.get_node(pos).name].is_burnt then if core.registered_nodes[core.get_node(pos).name].is_burnt then
return false return false
end end
run(pos, {type="interrupt"}) run(pos, {type="interrupt"})
@@ -713,18 +720,18 @@ end
mesecon.queue:add_function("pipeworks:lc_tube_interrupt", function (pos, luac_id, iid) mesecon.queue:add_function("pipeworks:lc_tube_interrupt", function (pos, luac_id, iid)
-- There is no lua_tube anymore / it has been reprogrammed / replaced / burnt -- There is no lua_tube anymore / it has been reprogrammed / replaced / burnt
if (minetest.get_meta(pos):get_int("luac_id") ~= luac_id) then return end if (core.get_meta(pos):get_int("luac_id") ~= luac_id) then return end
if (minetest.registered_nodes[minetest.get_node(pos).name].is_burnt) then return end if (core.registered_nodes[core.get_node(pos).name].is_burnt) then return end
run(pos, {type="interrupt", iid = iid}) run(pos, {type="interrupt", iid = iid})
end) end)
mesecon.queue:add_function("pipeworks:lt_digiline_relay", function (pos, channel, luac_id, msg) mesecon.queue:add_function("pipeworks:lt_digiline_relay", function (pos, channel, luac_id, msg)
if not digiline then return end if not has_digilines then return end
-- This check is only really necessary because in case of server crash, old actions can be thrown into the future -- This check is only really necessary because in case of server crash, old actions can be thrown into the future
if (minetest.get_meta(pos):get_int("luac_id") ~= luac_id) then return end if (core.get_meta(pos):get_int("luac_id") ~= luac_id) then return end
if (minetest.registered_nodes[minetest.get_node(pos).name].is_burnt) then return end if (core.registered_nodes[core.get_node(pos).name].is_burnt) then return end
-- The actual work -- The actual work
digiline:receptor_send(pos, digiline_rules_luatube, channel, msg) digilines.receptor_send(pos, digiline_rules_luatube, channel, msg)
end) end)
----------------------- -----------------------
@@ -765,7 +772,7 @@ local digiline = {
} }
local function get_program(pos) local function get_program(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
return meta:get_string("code") return meta:get_string("code")
end end
@@ -780,14 +787,14 @@ local function on_receive_fields(pos, form_name, fields, sender)
return return
end end
local name = sender:get_player_name() local name = sender:get_player_name()
if minetest.is_protected(pos, name) and not minetest.check_player_privs(name, {protection_bypass=true}) then if core.is_protected(pos, name) and not core.check_player_privs(name, {protection_bypass=true}) then
minetest.record_protection_violation(pos, name) core.record_protection_violation(pos, name)
return return
end end
local ok, err = set_program(pos, fields.code) local ok, err = set_program(pos, fields.code)
if not ok then if not ok then
-- it's not an error from the server perspective -- it's not an error from the server perspective
minetest.log("action", "Lua controller programming error: " .. tostring(err)) core.log("action", "Lua controller programming error: " .. tostring(err))
end end
end end
@@ -821,7 +828,7 @@ local tiles_on_off = {
R270 = "^(pipeworks_lua_tube_port_%s.png^[transformR270)" R270 = "^(pipeworks_lua_tube_port_%s.png^[transformR270)"
} }
local texture_alpha_mode = minetest.features.use_texture_alpha_string_modes local texture_alpha_mode = core.features.use_texture_alpha_string_modes
and "clip" or true and "clip" or true
for red = 0, 1 do -- 0 = off 1 = on for red = 0, 1 do -- 0 = off 1 = on
@@ -865,7 +872,7 @@ for white = 0, 1 do
tiles[3] = tiles[3]..tiles_on_off.R270:format(white == 1 and "on" or "off"); tiles[3] = tiles[3]..tiles_on_off.R270:format(white == 1 and "on" or "off");
tiles[4] = tiles[4]..tiles_on_off.R_90:format(white == 1 and "on" or "off"); tiles[4] = tiles[4]..tiles_on_off.R_90:format(white == 1 and "on" or "off");
local groups = {snappy = 3, tube = 1, tubedevice = 1, overheat = 1} local groups = {snappy = 3, tube = 1, tubedevice = 1, overheat = 1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
if red + blue + yellow + green + black + white ~= 0 then if red + blue + yellow + green + black + white ~= 0 then
groups.not_in_creative_inventory = 1 groups.not_in_creative_inventory = 1
end end
@@ -904,7 +911,7 @@ for white = 0, 1 do
}, },
} }
minetest.register_node(node_name, { core.register_node(node_name, {
description = "Lua controlled Tube", description = "Lua controlled Tube",
drawtype = "nodebox", drawtype = "nodebox",
tiles = tiles, tiles = tiles,
@@ -912,15 +919,18 @@ for white = 0, 1 do
paramtype = "light", paramtype = "light",
is_ground_content = false, is_ground_content = false,
groups = groups, groups = groups,
_mcl_hardness=0.8,
drop = BASENAME.."000000", drop = BASENAME.."000000",
sunlight_propagates = true, sunlight_propagates = true,
selection_box = selection_box, selection_box = selection_box,
node_box = node_box, node_box = node_box,
on_construct = reset_meta, on_construct = reset_meta,
on_receive_fields = on_receive_fields, on_receive_fields = on_receive_fields,
sounds = default.node_sound_wood_defaults(), _sound_def = {
key = "node_sound_wood_defaults",
},
mesecons = mesecons, mesecons = mesecons,
digiline = digiline, digilines = digiline,
-- Virtual portstates are the ports that -- Virtual portstates are the ports that
-- the node shows as powered up (light up). -- the node shows as powered up (light up).
virtual_portstates = { virtual_portstates = {
@@ -942,7 +952,7 @@ for white = 0, 1 do
tube = { tube = {
connect_sides = {front = 1, back = 1, left = 1, right = 1, top = 1, bottom = 1}, connect_sides = {front = 1, back = 1, left = 1, right = 1, top = 1, bottom = 1},
priority = 50, priority = 50,
can_go = function(pos, node, velocity, stack) can_go = function(pos, node, velocity, stack, tags)
local src = {name = nil} local src = {name = nil}
-- add color of the incoming tube explicitly; referring to rules, in case they change later -- add color of the incoming tube explicitly; referring to rules, in case they change later
for _, rule in pairs(rules) do for _, rule in pairs(rules) do
@@ -957,21 +967,42 @@ for white = 0, 1 do
itemstring = stack:to_string(), itemstring = stack:to_string(),
item = stack:to_table(), item = stack:to_table(),
velocity = velocity, velocity = velocity,
tags = table.copy(tags),
side = src.name,
}) })
if not succ or type(msg) ~= "string" then if not succ then
return go_back(velocity) return go_back(velocity)
end end
local r = rules[msg] if type(msg) == "string" then
return r and {r} or go_back(velocity) local side = rules[msg]
return side and {side} or go_back(velocity)
elseif type(msg) == "table" then
if pipeworks.enable_item_tags then
local new_tags
if type(msg.tags) == "table" or type(msg.tags) == "string" then
new_tags = pipeworks.sanitize_tags(msg.tags)
elseif type(msg.tag) == "string" then
new_tags = pipeworks.sanitize_tags({msg.tag})
end
if new_tags then
for i=1, math.max(#tags, #new_tags) do
tags[i] = new_tags[i]
end
end
end
local side = rules[msg.side]
return side and {side} or go_back(velocity)
end
return go_back(velocity)
end, end,
}, },
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
on_blast = function(pos, intensity) on_blast = function(pos, intensity)
if not intensity or intensity > 1 + 3^0.5 then if not intensity or intensity > 1 + 3^0.5 then
minetest.remove_node(pos) core.remove_node(pos)
return return
end end
minetest.swap_node(pos, {name = "pipeworks:broken_tube_1"}) core.swap_node(pos, {name = "pipeworks:broken_tube_1"})
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end, end,
}) })
@@ -1014,21 +1045,24 @@ tiles_burnt[2] = tiles_burnt[2].."^(pipeworks_lua_tube_port_burnt.png^[transform
tiles_burnt[3] = tiles_burnt[3].."^(pipeworks_lua_tube_port_burnt.png^[transformR270)" tiles_burnt[3] = tiles_burnt[3].."^(pipeworks_lua_tube_port_burnt.png^[transformR270)"
tiles_burnt[4] = tiles_burnt[4].."^(pipeworks_lua_tube_port_burnt.png^[transformR90)" tiles_burnt[4] = tiles_burnt[4].."^(pipeworks_lua_tube_port_burnt.png^[transformR90)"
minetest.register_node(BASENAME .. "_burnt", { core.register_node(BASENAME .. "_burnt", {
drawtype = "nodebox", drawtype = "nodebox",
tiles = tiles_burnt, tiles = tiles_burnt,
use_texture_alpha = texture_alpha_mode, use_texture_alpha = texture_alpha_mode,
is_burnt = true, is_burnt = true,
paramtype = "light", paramtype = "light",
is_ground_content = false, is_ground_content = false,
groups = {snappy = 3, tube = 1, tubedevice = 1, not_in_creative_inventory=1}, groups = {snappy = 3, tube = 1, tubedevice = 1, not_in_creative_inventory=1, dig_generic = 4, axey=1, handy=1, pickaxey=1},
_mcl_hardness=0.8,
drop = BASENAME.."000000", drop = BASENAME.."000000",
sunlight_propagates = true, sunlight_propagates = true,
selection_box = selection_box, selection_box = selection_box,
node_box = node_box, node_box = node_box,
on_construct = reset_meta, on_construct = reset_meta,
on_receive_fields = on_receive_fields, on_receive_fields = on_receive_fields,
sounds = default.node_sound_wood_defaults(), _sound_def = {
key = "node_sound_wood_defaults",
},
virtual_portstates = {red = false, blue = false, yellow = false, virtual_portstates = {red = false, blue = false, yellow = false,
green = false, black = false, white = false}, green = false, black = false, white = false},
mesecons = { mesecons = {
@@ -1048,10 +1082,10 @@ minetest.register_node(BASENAME .. "_burnt", {
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
on_blast = function(pos, intensity) on_blast = function(pos, intensity)
if not intensity or intensity > 1 + 3^0.5 then if not intensity or intensity > 1 + 3^0.5 then
minetest.remove_node(pos) core.remove_node(pos)
return return
end end
minetest.swap_node(pos, {name = "pipeworks:broken_tube_1"}) core.swap_node(pos, {name = "pipeworks:broken_tube_1"})
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end, end,
}) })
@@ -1060,7 +1094,7 @@ minetest.register_node(BASENAME .. "_burnt", {
-- Craft Registration -- -- Craft Registration --
------------------------ ------------------------
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = BASENAME.."000000", output = BASENAME.."000000",
recipe = {"pipeworks:mese_tube_000000", "mesecons_luacontroller:luacontroller0000"}, recipe = {"pipeworks:mese_tube_000000", "mesecons_luacontroller:luacontroller0000"},

View File

@@ -0,0 +1,53 @@
local S = core.get_translator("pipeworks")
local straight = function(pos, node, velocity, stack) return {velocity} end
local pane_box = {
type = "fixed",
fixed = {
{ -9/64, -9/64, -8/16, 9/64, 9/64, 8/16 }, -- tube
{ -8/16, -8/16, -1/16, 8/16, 8/16, 1/16 } -- pane
}
}
local texture_alpha_mode = core.features.use_texture_alpha_string_modes
and "clip" or true
core.register_node("pipeworks:steel_pane_embedded_tube", {
drawtype = "nodebox",
description = S("Airtight panel embedded tube"),
tiles = {
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_sides.png^[transformR90"),
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_sides.png^[transformR90"),
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_sides.png"),
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_sides.png"),
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_ends.png"),
pipeworks.make_tube_tile("pipeworks_pane_embedded_tube_ends.png"),
},
use_texture_alpha = texture_alpha_mode,
node_box = pane_box,
selection_box = pane_box,
collision_box = pane_box,
paramtype = "light",
paramtype2 = "facedir",
groups = {cracky=1, oddly_breakable_by_hand = 1, tubedevice = 1, tube = 1, dig_glass = 2, pickaxey=1, handy=1},
is_ground_content = false,
_mcl_hardness=0.8,
legacy_facedir_simple = true,
_sound_def = {
key = "node_sound_stone_defaults",
},
tube = {
connect_sides = {front = 1, back = 1,},
priority = 50,
can_go = straight,
can_insert = function(pos, node, stack, direction)
local dir = core.facedir_to_dir(node.param2)
return vector.equals(dir, direction) or vector.equals(vector.multiply(dir, -1), direction)
end,
},
after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig,
on_rotate = pipeworks.on_rotate,
})
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:steel_pane_embedded_tube"

View File

@@ -1,10 +1,10 @@
-- This file supplies the various kinds of pneumatic tubes -- This file supplies the various kinds of pneumatic tubes
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local tubenodes = {} local tubenodes = {}
pipeworks.tubenodes = tubenodes pipeworks.tubenodes = tubenodes
minetest.register_alias("pipeworks:tube", "pipeworks:tube_000000") core.register_alias("pipeworks:tube", "pipeworks:tube_000000")
-- now, a function to define the tubes -- now, a function to define the tubes
@@ -27,8 +27,8 @@ local texture_mt = {
} }
-- This will remove any semi-transparent pixels -- This will remove any semi-transparent pixels
-- because that is still buggy in Minetest, force this as default -- because that is still buggy in Luanti, force this as default
local texture_alpha_mode = minetest.features.use_texture_alpha_string_modes local texture_alpha_mode = core.features.use_texture_alpha_string_modes
and "clip" or true and "clip" or true
local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, ends, short, inv, special, connects, style) local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, ends, short, inv, special, connects, style)
@@ -61,13 +61,13 @@ local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, e
outimgs[vti[v]] = ends[v] outimgs[vti[v]] = ends[v]
end end
local tgroups = {snappy = 3, tube = 1, tubedevice = 1, not_in_creative_inventory = 1} local tgroups = {snappy = 3, tube = 1, tubedevice = 1, not_in_creative_inventory = 1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
local tubedesc = string.format("%s %s", desc, dump(connects)) local tubedesc = string.format("%s %s", desc, dump(connects))
local iimg = type(plain[1]) == "table" and plain[1].name or plain[1] local iimg = type(plain[1]) == "table" and plain[1].name or plain[1]
local wscale = {x = 1, y = 1, z = 1} local wscale = {x = 1, y = 1, z = 1}
if #connects == 0 then if #connects == 0 then
tgroups = {snappy = 3, tube = 1, tubedevice = 1} tgroups = {snappy = 3, tube = 1, tubedevice = 1, dig_generic = 4, axey=1, handy=1, pickaxey=1}
tubedesc = desc tubedesc = desc
iimg=inv iimg=inv
outimgs = { outimgs = {
@@ -80,6 +80,10 @@ local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, e
wscale = {x = 1, y = 1, z = 0.01} wscale = {x = 1, y = 1, z = 0.01}
end end
for i, tile in ipairs(outimgs) do
outimgs[i] = pipeworks.make_tube_tile(tile)
end
local rname = string.format("%s_%s", name, tname) local rname = string.format("%s_%s", name, tname)
table.insert(tubenodes, rname) table.insert(tubenodes, rname)
@@ -102,9 +106,12 @@ local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, e
fixed = outboxes fixed = outboxes
}, },
groups = tgroups, groups = tgroups,
sounds = default.node_sound_wood_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_wood_defaults",
},
walkable = true, walkable = true,
stack_max = 99,
basename = name, basename = name,
style = style, style = style,
drop = string.format("%s_%s", name, dropname), drop = string.format("%s_%s", name, dropname),
@@ -115,25 +122,25 @@ local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, e
}, },
on_punch = function(pos, node, player, pointed) on_punch = function(pos, node, player, pointed)
local playername = player:get_player_name() local playername = player:get_player_name()
if minetest.is_protected(pos, playername) and not minetest.check_player_privs(playername, {protection_bypass=true}) then if core.is_protected(pos, playername) and not core.check_player_privs(playername, {protection_bypass=true}) then
return minetest.node_punch(pos, node, player, pointed) return core.node_punch(pos, node, player, pointed)
end end
if pipeworks.check_and_wear_hammer(player) then if pipeworks.check_and_wear_hammer(player) then
local wieldname = player:get_wielded_item():get_name() local wieldname = player:get_wielded_item():get_name()
pipeworks.logger(string.format("%s struck a tube at %s with %s to break it.", playername, minetest.pos_to_string(pos), wieldname)) pipeworks.logger(string.format("%s struck a tube at %s with %s to break it.", playername, core.pos_to_string(pos), wieldname))
pipeworks.break_tube(pos) pipeworks.break_tube(pos)
end end
return minetest.node_punch(pos, node, player, pointed) return core.node_punch(pos, node, player, pointed)
end, end,
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
after_dig_node = pipeworks.after_dig, after_dig_node = pipeworks.after_dig,
on_rotate = false, on_rotate = false,
on_blast = function(pos, intensity) on_blast = function(pos, intensity)
if not intensity or intensity > 1 + 3^0.5 then if not intensity or intensity > 1 + 3^0.5 then
minetest.remove_node(pos) core.remove_node(pos)
return {string.format("%s_%s", name, dropname)} return {string.format("%s_%s", name, dropname)}
end end
minetest.swap_node(pos, {name = "pipeworks:broken_tube_1"}) core.swap_node(pos, {name = "pipeworks:broken_tube_1"})
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end, end,
check_for_pole = pipeworks.check_for_vert_tube, check_for_pole = pipeworks.check_for_vert_tube,
@@ -162,7 +169,7 @@ local register_one_tube = function(name, tname, dropname, desc, plain, noctrs, e
end end
end end
minetest.register_node(rname, nodedef) core.register_node(rname, nodedef)
end end
local register_all_tubes = function(name, desc, plain, noctrs, ends, short, inv, special, old_registration) local register_all_tubes = function(name, desc, plain, noctrs, ends, short, inv, special, old_registration)
@@ -209,7 +216,7 @@ local register_all_tubes = function(name, desc, plain, noctrs, ends, short, inv,
end end
if REGISTER_COMPATIBILITY then if REGISTER_COMPATIBILITY then
local cname = name.."_compatibility" local cname = name.."_compatibility"
minetest.register_node(cname, { core.register_node(cname, {
drawtype = "airlike", drawtype = "airlike",
style = "6d", style = "6d",
basename = name, basename = name,
@@ -220,6 +227,7 @@ local register_all_tubes = function(name, desc, plain, noctrs, ends, short, inv,
description = S("Pneumatic tube segment (legacy)"), description = S("Pneumatic tube segment (legacy)"),
after_place_node = pipeworks.after_place, after_place_node = pipeworks.after_place,
groups = {not_in_creative_inventory = 1, tube_to_update = 1, tube = 1}, groups = {not_in_creative_inventory = 1, tube_to_update = 1, tube = 1},
is_ground_content = false,
tube = {connect_sides = {front = 1, back = 1, left = 1, right = 1, top = 1, bottom = 1}}, tube = {connect_sides = {front = 1, back = 1, left = 1, right = 1, top = 1, bottom = 1}},
drop = name.."_1", drop = name.."_1",
}) })
@@ -232,7 +240,7 @@ local register_all_tubes = function(name, desc, plain, noctrs, ends, short, inv,
for zm = 0, 1 do for zm = 0, 1 do
for zp = 0, 1 do for zp = 0, 1 do
local tname = xm..xp..ym..yp..zm..zp local tname = xm..xp..ym..yp..zm..zp
minetest.register_alias(name.."_"..tname, cname) core.register_alias(name.."_"..tname, cname)
end end
end end
end end
@@ -258,14 +266,14 @@ end
if REGISTER_COMPATIBILITY then if REGISTER_COMPATIBILITY then
minetest.register_abm({ core.register_abm({
nodenames = {"group:tube_to_update"}, nodenames = {"group:tube_to_update"},
interval = 1, interval = 1,
chance = 1, chance = 1,
action = function(pos, node, active_object_count, active_object_count_wider) action = function(pos, node, active_object_count, active_object_count_wider)
local minp = vector.subtract(pos, 1) local minp = vector.subtract(pos, 1)
local maxp = vector.add(pos, 1) local maxp = vector.add(pos, 1)
if table.getn(minetest.find_nodes_in_area(minp, maxp, "ignore")) == 0 then if table.getn(core.find_nodes_in_area(minp, maxp, "ignore")) == 0 then
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end end
end end

View File

@@ -1,7 +1,7 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
-- the default tube and default textures -- the default tube and default textures
pipeworks.register_tube("pipeworks:tube", S("Pneumatic tube segment")) pipeworks.register_tube("pipeworks:tube", S("Pneumatic tube segment"))
minetest.register_craft( { core.register_craft( {
output = "pipeworks:tube_1 6", output = "pipeworks:tube_1 6",
recipe = { recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }, { "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
@@ -50,9 +50,10 @@ pipeworks.register_tube("pipeworks:broken_tube", {
node_def = { node_def = {
drop = "pipeworks:tube_1", drop = "pipeworks:tube_1",
groups = {not_in_creative_inventory = 1, tubedevice_receiver = 1}, groups = {not_in_creative_inventory = 1, tubedevice_receiver = 1},
is_ground_content = false,
tube = { tube = {
insert_object = function(pos, node, stack, direction) insert_object = function(pos, node, stack, direction)
minetest.item_drop(stack, nil, pos) core.item_drop(stack, nil, pos)
return ItemStack("") return ItemStack("")
end, end,
can_insert = function(pos,node,stack,direction) can_insert = function(pos,node,stack,direction)
@@ -64,29 +65,50 @@ pipeworks.register_tube("pipeworks:broken_tube", {
local itemstack = puncher:get_wielded_item() local itemstack = puncher:get_wielded_item()
local wieldname = itemstack:get_name() local wieldname = itemstack:get_name()
local playername = puncher:get_player_name() local playername = puncher:get_player_name()
local log_msg = playername.." struck a broken tube at "..minetest.pos_to_string(pos).."\n" local log_msg = playername.." struck a broken tube at "..core.pos_to_string(pos).."\n "
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local was_node = minetest.deserialize(meta:get_string("the_tube_was")) local was_node = core.deserialize(meta:get_string("the_tube_was"))
if not was_node then if not was_node then
pipeworks.logger(log_msg.." but it can't be repaired.") pipeworks.logger(log_msg.."but it can't be repaired.")
return return
end end
if not pipeworks.check_and_wear_hammer(puncher) then if not pipeworks.check_and_wear_hammer(puncher) then
if wieldname == "" then if wieldname == "" then
pipeworks.logger(log_msg.." by hand. It's not very effective.") pipeworks.logger(log_msg.."by hand. It's not very effective.")
if minetest.settings:get_bool("enable_damage") then if core.settings:get_bool("enable_damage") then
minetest.chat_send_player(playername,S("Broken tubes may be a bit sharp. Perhaps try with a hammer?")) core.chat_send_player(playername,S("Broken tubes may be a bit sharp. Perhaps try with a hammer?"))
puncher:set_hp(puncher:get_hp()-1) puncher:set_hp(puncher:get_hp()-1)
end end
else else
pipeworks.logger(log_msg.." with "..wieldname.." but that tool is too weak.") pipeworks.logger(log_msg.."with "..wieldname.." but that tool is too weak.")
end end
return return
end end
pipeworks.logger(log_msg.." with "..wieldname.." to repair it.") log_msg = log_msg.."with "..wieldname.." to repair it"
minetest.swap_node(pos, { name = was_node.name, param2 = was_node.param2 }) local nodedef = core.registered_nodes[was_node.name]
pipeworks.scan_for_tube_objects(pos) if nodedef then
end pipeworks.logger(log_msg..".")
if nodedef.tube and nodedef.tube.on_repair then
nodedef.tube.on_repair(pos, was_node)
else
core.swap_node(pos, { name = was_node.name, param2 = was_node.param2 })
pipeworks.scan_for_tube_objects(pos)
end
meta:set_string("the_tube_was", "")
else
pipeworks.logger(log_msg.." but original node "..was_node.name.." is not registered anymore.")
core.chat_send_player(playername, S("This tube cannot be repaired."))
end
end,
allow_metadata_inventory_put = function()
return 0
end,
allow_metadata_inventory_move = function()
return 0
end,
allow_metadata_inventory_take = function()
return 0
end,
} }
}) })
@@ -106,14 +128,6 @@ if pipeworks.enable_priority_tube then
tube = { priority = 150 } -- higher than tubedevices (100) tube = { priority = 150 } -- higher than tubedevices (100)
}, },
}) })
minetest.register_craft( {
output = "pipeworks:priority_tube_1 6",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "default:gold_ingot", "", "default:gold_ingot" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end end
if pipeworks.enable_accelerator_tube then if pipeworks.enable_accelerator_tube then
@@ -131,14 +145,6 @@ if pipeworks.enable_accelerator_tube then
end} end}
}, },
}) })
minetest.register_craft( {
output = "pipeworks:accelerator_tube_1 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "default:mese_crystal_fragment", "default:steel_ingot", "default:mese_crystal_fragment" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end end
if pipeworks.enable_crossing_tube then if pipeworks.enable_crossing_tube then
@@ -153,32 +159,32 @@ if pipeworks.enable_crossing_tube then
tube = {can_go = function(pos, node, velocity, stack) return {velocity} end } tube = {can_go = function(pos, node, velocity, stack) return {velocity} end }
}, },
}) })
minetest.register_craft( {
output = "pipeworks:crossing_tube_1 5",
recipe = {
{ "", "pipeworks:tube_1", "" },
{ "pipeworks:tube_1", "pipeworks:tube_1", "pipeworks:tube_1" },
{ "", "pipeworks:tube_1", "" }
},
})
end end
local texture_alpha_mode = minetest.features.use_texture_alpha_string_modes local texture_alpha_mode = core.features.use_texture_alpha_string_modes
and "clip" or true and "clip" or true
if pipeworks.enable_one_way_tube then if pipeworks.enable_one_way_tube then
minetest.register_node("pipeworks:one_way_tube", { local tiles = {"pipeworks_one_way_tube_top.png", "pipeworks_one_way_tube_top.png", "pipeworks_one_way_tube_output.png",
"pipeworks_one_way_tube_input.png", "pipeworks_one_way_tube_side.png", "pipeworks_one_way_tube_top.png"}
for i, tile in ipairs(tiles) do
tiles[i] = pipeworks.make_tube_tile(tile)
end
core.register_node("pipeworks:one_way_tube", {
description = S("One way tube"), description = S("One way tube"),
tiles = {"pipeworks_one_way_tube_top.png", "pipeworks_one_way_tube_top.png", "pipeworks_one_way_tube_output.png", tiles = tiles,
"pipeworks_one_way_tube_input.png", "pipeworks_one_way_tube_side.png", "pipeworks_one_way_tube_top.png"},
use_texture_alpha = texture_alpha_mode, use_texture_alpha = texture_alpha_mode,
paramtype2 = "facedir", paramtype2 = "facedir",
drawtype = "nodebox", drawtype = "nodebox",
paramtype = "light", paramtype = "light",
node_box = {type = "fixed", node_box = {type = "fixed",
fixed = {{-1/2, -9/64, -9/64, 1/2, 9/64, 9/64}}}, fixed = {{-1/2, -9/64, -9/64, 1/2, 9/64, 9/64}}},
groups = {snappy = 2, choppy = 2, oddly_breakable_by_hand = 2, tubedevice = 1}, groups = {snappy = 2, choppy = 2, oddly_breakable_by_hand = 2, tubedevice = 1, tube = 1, axey=1, handy=1, pickaxey=1},
sounds = default.node_sound_wood_defaults(), is_ground_content = false,
_mcl_hardness=0.8,
_sound_def = {
key = "node_sound_wood_defaults",
},
tube = { tube = {
connect_sides = {left = 1, right = 1}, connect_sides = {left = 1, right = 1},
can_go = function(pos, node, velocity, stack) can_go = function(pos, node, velocity, stack)
@@ -197,12 +203,4 @@ if pipeworks.enable_one_way_tube then
check_for_horiz_pole = pipeworks.check_for_horiz_tube check_for_horiz_pole = pipeworks.check_for_horiz_tube
}) })
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:one_way_tube" pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:one_way_tube"
minetest.register_craft({
output = "pipeworks:one_way_tube 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "group:stick", "default:mese_crystal", "basic_materials:plastic_sheet" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
end end

View File

@@ -1,51 +1,51 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
-- the minetest.after() calls below can sometimes trigger after a tube -- the core.after() calls below can sometimes trigger after a tube
-- breaks, at which point item_exit() is no longer valid, so we have to make -- breaks, at which point item_exit() is no longer valid, so we have to make
-- sure that there even IS a callback to run, first. -- sure that there even IS a callback to run, first.
local function after_break(pos) local function after_break(pos)
local name = minetest.get_node(pos).name local name = core.get_node(pos).name
if minetest.registered_nodes[name].item_exit then if core.registered_nodes[name].item_exit then
minetest.registered_nodes[name].item_exit(pos) core.registered_nodes[name].item_exit(pos)
end end
end end
if pipeworks.enable_detector_tube then if core.get_modpath("mesecons") and pipeworks.enable_detector_tube then
local detector_tube_step = 5 * tonumber(minetest.settings:get("dedicated_server_step")) local detector_tube_step = 5 * (tonumber(core.settings:get("dedicated_server_step")) or 0.09)
pipeworks.register_tube("pipeworks:detector_tube_on", { pipeworks.register_tube("pipeworks:detector_tube_on", {
description = S("Detecting Pneumatic Tube Segment on"), description = S("Detecting Pneumatic Tube Segment on"),
inventory_image = "pipeworks_detector_tube_inv.png", inventory_image = "pipeworks_detector_tube_inv.png",
plain = { "pipeworks_detector_tube_plain.png" }, plain = { "pipeworks_detector_tube_plain.png" },
node_def = { node_def = {
tube = {can_go = function(pos, node, velocity, stack) tube = {can_go = function(pos, node, velocity, stack)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local nitems = meta:get_int("nitems")+1 local nitems = meta:get_int("nitems")+1
meta:set_int("nitems", nitems) meta:set_int("nitems", nitems)
local saved_pos = vector.new(pos) local saved_pos = vector.new(pos)
minetest.after(detector_tube_step, after_break, saved_pos) core.after(detector_tube_step, after_break, saved_pos)
return pipeworks.notvel(pipeworks.meseadjlist,velocity) return pipeworks.notvel(pipeworks.meseadjlist,velocity)
end}, end},
groups = {mesecon = 2, not_in_creative_inventory = 1}, groups = {mesecon = 2, not_in_creative_inventory = 1},
drop = "pipeworks:detector_tube_off_1", drop = "pipeworks:detector_tube_off_1",
mesecons = {receptor = {state = "on", rules = pipeworks.mesecons_rules}}, mesecons = {receptor = {state = "on", rules = pipeworks.mesecons_rules}},
item_exit = function(pos) item_exit = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local nitems = meta:get_int("nitems")-1 local nitems = meta:get_int("nitems")-1
local node = minetest.get_node(pos) local node = core.get_node(pos)
local name = node.name local name = node.name
local fdir = node.param2 local fdir = node.param2
if nitems == 0 then if nitems == 0 then
minetest.set_node(pos, {name = string.gsub(name, "on", "off"), param2 = fdir}) core.set_node(pos, {name = string.gsub(name, "on", "off"), param2 = fdir})
mesecon.receptor_off(pos, pipeworks.mesecons_rules) mesecon.receptor_off(pos, pipeworks.mesecons_rules)
else else
meta:set_int("nitems", nitems) meta:set_int("nitems", nitems)
end end
end, end,
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
meta:set_int("nitems", 1) meta:set_int("nitems", 1)
minetest.after(detector_tube_step, after_break, pos) core.after(detector_tube_step, after_break, pos)
end, end,
}, },
}) })
@@ -55,10 +55,10 @@ if pipeworks.enable_detector_tube then
plain = { "pipeworks_detector_tube_plain.png" }, plain = { "pipeworks_detector_tube_plain.png" },
node_def = { node_def = {
tube = {can_go = function(pos, node, velocity, stack) tube = {can_go = function(pos, node, velocity, stack)
local node = minetest.get_node(pos) local node = core.get_node(pos)
local name = node.name local name = node.name
local fdir = node.param2 local fdir = node.param2
minetest.set_node(pos,{name = string.gsub(name, "off", "on"), param2 = fdir}) core.set_node(pos,{name = string.gsub(name, "off", "on"), param2 = fdir})
mesecon.receptor_on(pos, pipeworks.mesecons_rules) mesecon.receptor_on(pos, pipeworks.mesecons_rules)
return pipeworks.notvel(pipeworks.meseadjlist, velocity) return pipeworks.notvel(pipeworks.meseadjlist, velocity)
end}, end},
@@ -67,7 +67,7 @@ if pipeworks.enable_detector_tube then
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:detector_tube_off_1 2", output = "pipeworks:detector_tube_off_1 2",
recipe = { recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }, { "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
@@ -77,7 +77,7 @@ if pipeworks.enable_detector_tube then
}) })
end end
local digiline_enabled = minetest.get_modpath("digilines") ~= nil local digiline_enabled = core.get_modpath("digilines") ~= nil
if digiline_enabled and pipeworks.enable_digiline_detector_tube then if digiline_enabled and pipeworks.enable_digiline_detector_tube then
pipeworks.register_tube("pipeworks:digiline_detector_tube", { pipeworks.register_tube("pipeworks:digiline_detector_tube", {
description = S("Digiline Detecting Pneumatic Tube Segment"), description = S("Digiline Detecting Pneumatic Tube Segment"),
@@ -85,16 +85,16 @@ if digiline_enabled and pipeworks.enable_digiline_detector_tube then
plain = { "pipeworks_digiline_detector_tube_plain.png" }, plain = { "pipeworks_digiline_detector_tube_plain.png" },
node_def = { node_def = {
tube = {can_go = function(pos, node, velocity, stack) tube = {can_go = function(pos, node, velocity, stack)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local setchan = meta:get_string("channel") local setchan = meta:get_string("channel")
digiline:receptor_send(pos, digiline.rules.default, setchan, stack:to_table()) digilines.receptor_send(pos, digilines.rules.default, setchan, stack:to_table())
return pipeworks.notvel(pipeworks.meseadjlist, velocity) return pipeworks.notvel(pipeworks.meseadjlist, velocity)
end}, end},
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
meta:set_string("formspec", meta:set_string("formspec",
"size[8.5,2.2]".. "size[8.5,2.2]"..
"image[0.2,0;1,1;pipeworks_digiline_detector_tube_inv.png]".. "image[0.2,0;1,1;pipeworks_digiline_detector_tube_inv.png]"..
@@ -111,11 +111,11 @@ if digiline_enabled and pipeworks.enable_digiline_detector_tube then
return return
end end
if fields.channel then if fields.channel then
minetest.get_meta(pos):set_string("channel", fields.channel) core.get_meta(pos):set_string("channel", fields.channel)
end end
end, end,
groups = {}, groups = {},
digiline = { digilines = {
receptor = {}, receptor = {},
effector = { effector = {
action = function(pos,node,channel,msg) end action = function(pos,node,channel,msg) end
@@ -127,7 +127,7 @@ if digiline_enabled and pipeworks.enable_digiline_detector_tube then
}, },
}) })
minetest.register_craft( { core.register_craft( {
output = "pipeworks:digiline_detector_tube_1 2", output = "pipeworks:digiline_detector_tube_1 2",
recipe = { recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }, { "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
@@ -137,7 +137,7 @@ if digiline_enabled and pipeworks.enable_digiline_detector_tube then
}) })
end end
if pipeworks.enable_conductor_tube then if core.get_modpath("mesecons") and pipeworks.enable_conductor_tube then
pipeworks.register_tube("pipeworks:conductor_tube_off", { pipeworks.register_tube("pipeworks:conductor_tube_off", {
description = S("Conducting Pneumatic Tube Segment"), description = S("Conducting Pneumatic Tube Segment"),
inventory_image = "pipeworks_conductor_tube_inv.png", inventory_image = "pipeworks_conductor_tube_inv.png",
@@ -168,7 +168,7 @@ if pipeworks.enable_conductor_tube then
}, },
}) })
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = "pipeworks:conductor_tube_off_1", output = "pipeworks:conductor_tube_off_1",
recipe = {"pipeworks:tube_1", "mesecons:mesecon"} recipe = {"pipeworks:tube_1", "mesecons:mesecon"}
@@ -183,9 +183,9 @@ if digiline_enabled and pipeworks.enable_digiline_conductor_tube then
plain = {"pipeworks_tube_plain.png^pipeworks_digiline_conductor_tube_plain.png"}, plain = {"pipeworks_tube_plain.png^pipeworks_digiline_conductor_tube_plain.png"},
noctr = {"pipeworks_tube_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"}, noctr = {"pipeworks_tube_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"},
ends = {"pipeworks_tube_end.png^pipeworks_digiline_conductor_tube_end.png"}, ends = {"pipeworks_tube_end.png^pipeworks_digiline_conductor_tube_end.png"},
node_def = {digiline = {wire = {rules = pipeworks.digilines_rules}}}, node_def = {digilines = {wire = {rules = pipeworks.digilines_rules}}},
}) })
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = "pipeworks:digiline_conductor_tube_1", output = "pipeworks:digiline_conductor_tube_1",
recipe = {"pipeworks:tube_1", "digilines:wire_std_00000000"} recipe = {"pipeworks:tube_1", "digilines:wire_std_00000000"}
@@ -202,7 +202,7 @@ if digiline_enabled and pipeworks.enable_digiline_conductor_tube and
noctr = {"pipeworks_conductor_tube_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"}, noctr = {"pipeworks_conductor_tube_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"},
ends = {"pipeworks_conductor_tube_end.png^pipeworks_digiline_conductor_tube_end.png"}, ends = {"pipeworks_conductor_tube_end.png^pipeworks_digiline_conductor_tube_end.png"},
node_def = { node_def = {
digiline = {wire = {rules = pipeworks.digilines_rules}}, digilines = {wire = {rules = pipeworks.digilines_rules}},
groups = {mesecon = 2}, groups = {mesecon = 2},
mesecons = {conductor = { mesecons = {conductor = {
state = "off", state = "off",
@@ -219,7 +219,7 @@ if digiline_enabled and pipeworks.enable_digiline_conductor_tube and
noctr = {"pipeworks_conductor_tube_on_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"}, noctr = {"pipeworks_conductor_tube_on_noctr.png^pipeworks_digiline_conductor_tube_noctr.png"},
ends = {"pipeworks_conductor_tube_on_end.png^pipeworks_digiline_conductor_tube_end.png"}, ends = {"pipeworks_conductor_tube_on_end.png^pipeworks_digiline_conductor_tube_end.png"},
node_def = { node_def = {
digiline = {wire = {rules = pipeworks.digilines_rules}}, digilines = {wire = {rules = pipeworks.digilines_rules}},
groups = {mesecon = 2, not_in_creative_inventory = 1}, groups = {mesecon = 2, not_in_creative_inventory = 1},
drop = "pipeworks:mesecon_and_digiline_conductor_tube_off_1", drop = "pipeworks:mesecon_and_digiline_conductor_tube_off_1",
mesecons = {conductor = { mesecons = {conductor = {
@@ -229,17 +229,17 @@ if digiline_enabled and pipeworks.enable_digiline_conductor_tube and
}, },
}, },
}) })
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1", output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1",
recipe = {"pipeworks:tube_1", "mesecons:mesecon", "digilines:wire_std_00000000"} recipe = {"pipeworks:tube_1", "mesecons:mesecon", "digilines:wire_std_00000000"}
}) })
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1", output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1",
recipe = {"pipeworks:conductor_tube_off_1", "digilines:wire_std_00000000"} recipe = {"pipeworks:conductor_tube_off_1", "digilines:wire_std_00000000"}
}) })
minetest.register_craft({ core.register_craft({
type = "shapeless", type = "shapeless",
output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1", output = "pipeworks:mesecon_and_digiline_conductor_tube_off_1",
recipe = {"pipeworks:digiline_conductor_tube_1", "mesecons:mesecon"} recipe = {"pipeworks:digiline_conductor_tube_1", "mesecons:mesecon"}

View File

@@ -1,44 +1,58 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local fs_helpers = pipeworks.fs_helpers local fs_helpers = pipeworks.fs_helpers
if pipeworks.enable_mese_tube then if pipeworks.enable_mese_tube then
local function update_formspec(pos) local function update_formspec(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local old_formspec = meta:get_string("formspec") local old_formspec = meta:get_string("formspec")
if string.find(old_formspec, "button1") then -- Old version if string.find(old_formspec, "button1") then -- Old version
local inv = meta:get_inventory() local inv = meta:get_inventory()
for i = 1, 6 do for i = 1, 6 do
for _, stack in ipairs(inv:get_list("line"..i)) do for _, stack in ipairs(inv:get_list("line"..i)) do
minetest.add_item(pos, stack) core.add_item(pos, stack)
end end
end end
end end
local buttons_formspec = "" local buttons_formspec = ""
for i = 0, 5 do for i = 0, 5 do
buttons_formspec = buttons_formspec .. fs_helpers.cycling_button(meta, buttons_formspec = buttons_formspec .. fs_helpers.cycling_button(meta,
"image_button[7,"..(i+0.2)..";1,0.6", "l"..(i+1).."s", "image_button[9,"..(i+(i*0.25)+0.5)..";1,0.6", "l"..(i+1).."s",
{ {
pipeworks.button_off, pipeworks.button_off,
pipeworks.button_on pipeworks.button_on
} }
) )
end end
local list_backgrounds = ""
if core.get_modpath("i3") or core.get_modpath("mcl_formspec") then
list_backgrounds = "style_type[box;colors=#666]"
for i=0, 5 do
for j=0, 5 do
list_backgrounds = list_backgrounds .. "box[".. 1.5+(i*1.25) ..",".. 0.25+(j*1.25) ..";1,1;]"
end
end
end
local size = "10.2,13"
meta:set_string("formspec", meta:set_string("formspec",
"size[8,11]".. "formspec_version[2]"..
"list[context;line1;1,0;6,1;]".. "size["..size.."]"..
"list[context;line2;1,1;6,1;]".. pipeworks.fs_helpers.get_prepends(size)..
"list[context;line3;1,2;6,1;]".. "list[context;line1;1.5,0.25;6,1;]"..
"list[context;line4;1,3;6,1;]".. "list[context;line2;1.5,1.50;6,1;]"..
"list[context;line5;1,4;6,1;]".. "list[context;line3;1.5,2.75;6,1;]"..
"list[context;line6;1,5;6,1;]".. "list[context;line4;1.5,4.00;6,1;]"..
"image[0,0;1,1;pipeworks_white.png]".. "list[context;line5;1.5,5.25;6,1;]"..
"image[0,1;1,1;pipeworks_black.png]".. "list[context;line6;1.5,6.50;6,1;]"..
"image[0,2;1,1;pipeworks_green.png]".. list_backgrounds..
"image[0,3;1,1;pipeworks_yellow.png]".. "image[0.22,0.25;1,1;pipeworks_white.png]"..
"image[0,4;1,1;pipeworks_blue.png]".. "image[0.22,1.50;1,1;pipeworks_black.png]"..
"image[0,5;1,1;pipeworks_red.png]".. "image[0.22,2.75;1,1;pipeworks_green.png]"..
"image[0.22,4.00;1,1;pipeworks_yellow.png]"..
"image[0.22,5.25;1,1;pipeworks_blue.png]"..
"image[0.22,6.50;1,1;pipeworks_red.png]"..
buttons_formspec.. buttons_formspec..
"list[current_player;main;0,7;8,4;]" .. --"list[current_player;main;0,8;8,4;]" ..
pipeworks.fs_helpers.get_inv(8)..
"listring[current_player;main]" .. "listring[current_player;main]" ..
"listring[current_player;main]" .. "listring[current_player;main]" ..
"listring[context;line1]" .. "listring[context;line1]" ..
@@ -69,13 +83,13 @@ if pipeworks.enable_mese_tube then
tube = {can_go = function(pos, node, velocity, stack) tube = {can_go = function(pos, node, velocity, stack)
local tbl, tbln = {}, 0 local tbl, tbln = {}, 0
local found, foundn = {}, 0 local found, foundn = {}, 0
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
local name = stack:get_name() local name = stack:get_name()
for i, vect in ipairs(pipeworks.meseadjlist) do for i, vect in ipairs(pipeworks.meseadjlist) do
local npos = vector.add(pos, vect) local npos = vector.add(pos, vect)
local node = minetest.get_node(npos) local node = core.get_node(npos)
local reg_node = minetest.registered_nodes[node.name] local reg_node = core.registered_nodes[node.name]
if meta:get_int("l"..i.."s") == 1 and reg_node then if meta:get_int("l"..i.."s") == 1 and reg_node then
local tube_def = reg_node.tube local tube_def = reg_node.tube
if not tube_def or not tube_def.can_insert or if not tube_def or not tube_def.can_insert or
@@ -101,7 +115,7 @@ if pipeworks.enable_mese_tube then
return (foundn > 0) and found or tbl return (foundn > 0) and found or tbl
end}, end},
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
local inv = meta:get_inventory() local inv = meta:get_inventory()
for i = 1, 6 do for i = 1, 6 do
meta:set_int("l"..tostring(i).."s", 1) meta:set_int("l"..tostring(i).."s", 1)
@@ -112,7 +126,7 @@ if pipeworks.enable_mese_tube then
end, end,
after_place_node = function(pos, placer, itemstack, pointed_thing) after_place_node = function(pos, placer, itemstack, pointed_thing)
if placer and placer:is_player() and placer:get_player_control().aux1 then if placer and placer:is_player() and placer:get_player_control().aux1 then
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
for i = 1, 6 do for i = 1, 6 do
meta:set_int("l"..tostring(i).."s", 0) meta:set_int("l"..tostring(i).."s", 0)
end end
@@ -136,7 +150,7 @@ if pipeworks.enable_mese_tube then
allow_metadata_inventory_put = function(pos, listname, index, stack, player) allow_metadata_inventory_put = function(pos, listname, index, stack, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
update_formspec(pos) -- For old tubes update_formspec(pos) -- For old tubes
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
local stack_copy = ItemStack(stack) local stack_copy = ItemStack(stack)
stack_copy:set_count(1) stack_copy:set_count(1)
inv:set_stack(listname, index, stack_copy) inv:set_stack(listname, index, stack_copy)
@@ -145,14 +159,14 @@ if pipeworks.enable_mese_tube then
allow_metadata_inventory_take = function(pos, listname, index, stack, player) allow_metadata_inventory_take = function(pos, listname, index, stack, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
update_formspec(pos) -- For old tubes update_formspec(pos) -- For old tubes
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
inv:set_stack(listname, index, ItemStack("")) inv:set_stack(listname, index, ItemStack(""))
return 0 return 0
end, end,
allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player) allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
update_formspec(pos) -- For old tubes update_formspec(pos) -- For old tubes
local inv = minetest.get_meta(pos):get_inventory() local inv = core.get_meta(pos):get_inventory()
if from_list:match("line%d") and to_list:match("line%d") then if from_list:match("line%d") and to_list:match("line%d") then
return count return count
@@ -163,25 +177,4 @@ if pipeworks.enable_mese_tube then
end, end,
}, },
}) })
minetest.register_craft( {
output = "pipeworks:mese_tube_000000 2",
recipe = {
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{ "", "default:mese_crystal", "" },
{ "basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
minetest.register_craft( {
type = "shapeless",
output = "pipeworks:mese_tube_000000",
recipe = {
"pipeworks:tube_1",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment"
},
})
end end

139
tubes/tags.lua Normal file
View File

@@ -0,0 +1,139 @@
local S = core.get_translator("pipeworks")
local fs_helpers = pipeworks.fs_helpers
if not pipeworks.enable_item_tags or not pipeworks.enable_tag_tube then return end
local help_text = core.formspec_escape(
S("Separate multiple tags using commas.").."\n"..
S("Use \"<none>\" to match items without tags.")
)
local update_formspec = function(pos)
local meta = core.get_meta(pos)
local buttons_formspec = ""
for i = 0, 5 do
buttons_formspec = buttons_formspec .. fs_helpers.cycling_button(meta,
"image_button[9," .. (i + (i * 0.25) + 0.5) .. ";1,0.6", "l" .. (i + 1) .. "s",
{
pipeworks.button_off,
pipeworks.button_on
}
)
end
local size = "10.2,9"
meta:set_string("formspec",
"formspec_version[2]" ..
"size[" .. size .. "]" ..
pipeworks.fs_helpers.get_prepends(size) ..
"field[1.5,0.25;7.25,1;tags1;;${tags1}]" ..
"field[1.5,1.5;7.25,1;tags2;;${tags2}]" ..
"field[1.5,2.75;7.25,1;tags3;;${tags3}]" ..
"field[1.5,4.0;7.25,1;tags4;;${tags4}]" ..
"field[1.5,5.25;7.25,1;tags5;;${tags5}]" ..
"field[1.5,6.5;7.25,1;tags6;;${tags6}]" ..
"image[0.22,0.25;1,1;pipeworks_white.png]" ..
"image[0.22,1.50;1,1;pipeworks_black.png]" ..
"image[0.22,2.75;1,1;pipeworks_green.png]" ..
"image[0.22,4.00;1,1;pipeworks_yellow.png]" ..
"image[0.22,5.25;1,1;pipeworks_blue.png]" ..
"image[0.22,6.50;1,1;pipeworks_red.png]" ..
buttons_formspec ..
"label[0.22,7.9;"..help_text.."]"..
"button[7.25,7.8;1.5,0.8;set_item_tags;" .. S("Set") .. "]"
)
end
pipeworks.register_tube("pipeworks:tag_tube", {
description = S("Tag Sorting Pneumatic Tube Segment"),
inventory_image = "pipeworks_tag_tube_inv.png",
noctr = { "pipeworks_tag_tube_noctr_1.png", "pipeworks_tag_tube_noctr_2.png", "pipeworks_tag_tube_noctr_3.png",
"pipeworks_tag_tube_noctr_4.png", "pipeworks_tag_tube_noctr_5.png", "pipeworks_tag_tube_noctr_6.png" },
plain = { "pipeworks_tag_tube_plain_1.png", "pipeworks_tag_tube_plain_2.png", "pipeworks_tag_tube_plain_3.png",
"pipeworks_tag_tube_plain_4.png", "pipeworks_tag_tube_plain_5.png", "pipeworks_tag_tube_plain_6.png" },
ends = { "pipeworks_tag_tube_end.png" },
short = "pipeworks_tag_tube_short.png",
no_facedir = true, -- Must use old tubes, since the textures are rotated with 6d ones
node_def = {
tube = {
can_go = function(pos, node, velocity, stack, tags)
local tbl, tbln = {}, 0
local found, foundn = {}, 0
local meta = core.get_meta(pos)
local tag_hash = {}
if #tags > 0 then
for _,tag in ipairs(tags) do
tag_hash[tag] = true
end
else
tag_hash["<none>"] = true -- Matches items without tags
end
for i, vect in ipairs(pipeworks.meseadjlist) do
local npos = vector.add(pos, vect)
local node = core.get_node(npos)
local reg_node = core.registered_nodes[node.name]
if meta:get_int("l" .. i .. "s") == 1 and reg_node then
local tube_def = reg_node.tube
if not tube_def or not tube_def.can_insert or
tube_def.can_insert(npos, node, stack, vect) then
local side_tags = meta:get_string("tags" .. i)
if side_tags ~= "" then
side_tags = pipeworks.sanitize_tags(side_tags)
for _,tag in ipairs(side_tags) do
if tag_hash[tag] then
foundn = foundn + 1
found[foundn] = vect
break
end
end
else
tbln = tbln + 1
tbl[tbln] = vect
end
end
end
end
return (foundn > 0) and found or tbl
end
},
on_construct = function(pos)
local meta = core.get_meta(pos)
for i = 1, 6 do
meta:set_int("l" .. tostring(i) .. "s", 1)
end
update_formspec(pos)
meta:set_string("infotext", S("Tag sorting pneumatic tube"))
end,
after_place_node = function(pos, placer, itemstack, pointed_thing)
if placer and placer:is_player() and placer:get_player_control().aux1 then
local meta = core.get_meta(pos)
for i = 1, 6 do
meta:set_int("l" .. tostring(i) .. "s", 0)
end
update_formspec(pos)
end
return pipeworks.after_place(pos, placer, itemstack, pointed_thing)
end,
on_receive_fields = function(pos, formname, fields, sender)
if (fields.quit and not fields.key_enter_field)
or not pipeworks.may_configure(pos, sender) then
return
end
local meta = core.get_meta(pos)
for i = 1, 6 do
local field_name = "tags" .. tostring(i)
if fields[field_name] then
local tags = pipeworks.sanitize_tags(fields[field_name])
meta:set_string(field_name, table.concat(tags, ","))
end
end
fs_helpers.on_receive_fields(pos, fields)
update_formspec(pos)
end,
can_dig = function(pos, player)
return true
end,
},
})

374
tubes/teleport.lua Normal file
View File

@@ -0,0 +1,374 @@
local S = core.get_translator("pipeworks")
local filename = core.get_worldpath().."/teleport_tubes" -- Only used for backward-compat
local storage = core.get_mod_storage()
local enable_logging = core.settings:get_bool("pipeworks_log_teleport_tubes", false)
local has_digilines = core.get_modpath("digilines")
-- V1: Serialized text file indexed by vector position.
-- V2: Serialized text file indexed by hash position.
-- V3: Mod storage using serialized tables.
-- V4: Mod storage using "<can_receive>:<channel>" format.
local tube_db_version = 4
local tube_db = {}
local receiver_cache = {}
local function hash_pos(pos)
vector.round(pos)
return string.format("%.0f", core.hash_node_position(pos))
end
local function serialize_tube(tube)
return string.format("%d:%s", tube.cr, tube.channel)
end
local function deserialize_tube(hash, str)
local sep = str:sub(2, 2) == ":"
local cr = tonumber(str:sub(1, 1))
local channel = str:sub(3)
if sep and cr and channel then
local pos = core.get_position_from_hash(tonumber(hash))
return {x = pos.x, y = pos.y, z = pos.z, cr = cr, channel = channel}
end
end
local function save_tube_db()
receiver_cache = {}
local fields = {version = tube_db_version}
for key, val in pairs(tube_db) do
fields[key] = serialize_tube(val)
end
storage:from_table({fields = fields})
end
local function save_tube(hash)
local tube = tube_db[hash]
receiver_cache[tube.channel] = nil
storage:set_string(hash, serialize_tube(tube))
end
local function remove_tube(pos)
local hash = hash_pos(pos)
if tube_db[hash] then
receiver_cache[tube_db[hash].channel] = nil
tube_db[hash] = nil
storage:set_string(hash, "")
end
end
local function migrate_tube_db()
if storage:get_int("version") == 3 then
for key, val in pairs(storage:to_table().fields) do
if tonumber(key) then
tube_db[key] = core.deserialize(val)
elseif key ~= "version" then
error("Unknown field in teleport tube database: "..key)
end
end
save_tube_db()
return
end
local file = io.open(filename, "r")
if file then
local content = file:read("*all")
io.close(file)
if content and content ~= "" then
tube_db = core.deserialize(content)
end
end
local version = tube_db.version or 0
tube_db.version = nil
if version < 2 then
local tmp_db = {}
for _, val in pairs(tube_db) do
if val.channel ~= "" then -- Skip unconfigured tubes
tmp_db[hash_pos(val)] = val
end
end
tube_db = tmp_db
end
save_tube_db()
end
local function read_tube_db()
local version = storage:get_int("version")
if version < tube_db_version then
migrate_tube_db()
elseif version > tube_db_version then
error("Cannot read teleport tube database of version "..version)
else
for key, val in pairs(storage:to_table().fields) do
if tonumber(key) then
tube_db[key] = deserialize_tube(key, val)
elseif key ~= "version" then
error("Unknown field in teleport tube database: "..key)
end
end
end
tube_db.version = nil
end
local function set_tube(pos, channel, cr)
local hash = hash_pos(pos)
local tube = tube_db[hash]
if tube then
if tube.channel ~= channel or tube.cr ~= cr then
tube.channel = channel
tube.cr = cr
save_tube(hash)
end
else
tube_db[hash] = {x = pos.x, y = pos.y, z = pos.z, channel = channel, cr = cr}
save_tube(hash)
end
end
local function get_receivers(pos, channel)
local hash = hash_pos(pos)
local cache = receiver_cache[channel] or {}
if cache[hash] then
return cache[hash]
end
local receivers = {}
for key, val in pairs(tube_db) do
if val.cr == 1 and val.channel == channel and not vector.equals(val, pos) then
core.load_area(val)
local node_name = core.get_node(val).name
if node_name:find("pipeworks:teleport_tube") then
table.insert(receivers, val)
else
remove_tube(val)
end
end
end
cache[hash] = receivers
receiver_cache[channel] = cache
return receivers
end
local help_text = core.formspec_escape(
S("Channels are public by default").."\n"..
S("Use <player>:<channel> for fully private channels").."\n"..
S("Use <player>;<channel> for private receivers")
)
local size = has_digilines and "8,5.9" or "8,4.4"
local formspec = "formspec_version[2]size["..size.."]"..
pipeworks.fs_helpers.get_prepends(size)..
"image[0.5,0.3;1,1;pipeworks_teleport_tube_inv.png]"..
"label[1.75,0.8;"..S("Teleporting Tube").."]"..
"field[0.5,1.7;5,0.8;channel;"..S("Channel")..";${channel}]"..
"button_exit[5.5,1.7;2,0.8;save;"..S("Save").."]"..
"label[6.5,0.6;"..S("Receive").."]"..
"label[0.5,2.8;"..help_text.."]"
if has_digilines then
formspec = formspec..
"field[0.5,4.6;5,0.8;digiline_channel;"..S("Digiline Channel")..";${digiline_channel}]"..
"button_exit[5.5,4.6;2,0.8;save;"..S("Save").."]"
end
local function update_meta(meta)
local channel = meta:get_string("channel")
local cr = meta:get_int("can_receive") == 1
if channel == "" then
meta:set_string("infotext", S("Unconfigured Teleportation Tube"))
else
local desc = cr and "sending and receiving" or "sending"
meta:set_string("infotext", S("Teleportation Tube @1 on '@2'", desc, channel))
end
local state = cr and "on" or "off"
meta:set_string("formspec", formspec..
"image_button[6.4,0.8;1,0.6;pipeworks_button_"..state..
".png;cr_"..state..";;;false;pipeworks_button_interm.png]")
end
local function update_tube(pos, channel, cr, player_name)
local meta = core.get_meta(pos)
if meta:get_string("channel") == channel and meta:get_int("can_receive") == cr then
return
end
if channel == "" then
meta:set_string("channel", "")
meta:set_int("can_receive", cr)
remove_tube(pos)
return
end
local name, mode = channel:match("^([^:;]+)([:;])")
if name and mode and name ~= player_name then
if mode == ":" then
core.chat_send_player(player_name,
S("Sorry, channel '@1' is reserved for exclusive use by @2", channel, name))
return
elseif mode == ";" and cr ~= 0 then
core.chat_send_player(player_name,
S("Sorry, receiving from channel '@1' is reserved for @2", channel, name))
return
end
end
meta:set_string("channel", channel)
meta:set_int("can_receive", cr)
set_tube(pos, channel, cr)
end
local function receive_fields(pos, _, fields, sender)
if not fields.channel or not pipeworks.may_configure(pos, sender) then
return
end
local meta = core.get_meta(pos)
local channel = fields.channel:trim()
local cr = meta:get_int("can_receive")
if fields.cr_on then
cr = 0
elseif fields.cr_off then
cr = 1
end
if has_digilines and fields.digiline_channel then
meta:set_string("digiline_channel", fields.digiline_channel)
end
update_tube(pos, channel, cr, sender:get_player_name())
update_meta(meta)
end
local function can_go(pos, node, velocity, stack)
velocity.x = 0
velocity.y = 0
velocity.z = 0
local src_meta = core.get_meta(pos)
local channel = src_meta:get_string("channel")
if channel == "" then
return {}
end
local receivers = get_receivers(pos, channel)
if #receivers == 0 then
return {}
end
local target = receivers[math.random(1, #receivers)]
if enable_logging then
local src_owner = src_meta:get_string("owner")
local dst_meta = core.get_meta(pos)
local dst_owner = dst_meta:get_string("owner")
core.log("action", string.format("[pipeworks] %s teleported from %s (owner=%s) to %s (owner=%s) via %s",
stack:to_string(), core.pos_to_string(pos), src_owner, core.pos_to_string(target), dst_owner, channel
))
end
pos.x = target.x
pos.y = target.y
pos.z = target.z
return pipeworks.meseadjlist
end
local function repair_tube(pos, node)
core.swap_node(pos, {name = node.name, param2 = node.param2})
pipeworks.scan_for_tube_objects(pos)
local meta = core.get_meta(pos)
local channel = meta:get_string("channel")
if channel ~= "" then
set_tube(pos, channel, meta:get_int("can_receive"))
end
update_meta(meta)
end
local function digiline_action(pos, _, digiline_channel, msg)
local meta = core.get_meta(pos)
if digiline_channel ~= meta:get_string("digiline_channel") then
return
end
local channel = meta:get_string("channel")
local can_receive = meta:get_int("can_receive")
if type(msg) == "string" then
channel = msg
elseif type(msg) == "table" then
if type(msg.channel) == "string" then
channel = msg.channel
end
if msg.can_receive == 1 or msg.can_receive == true then
can_receive = 1
elseif msg.can_receive == 0 or msg.can_receive == false then
can_receive = 0
end
else
return
end
local player_name = meta:get_string("owner")
update_tube(pos, channel, can_receive, player_name)
update_meta(meta)
end
local def = {
tube = {
can_go = can_go,
on_repair = repair_tube,
},
on_construct = function(pos)
local meta = core.get_meta(pos)
meta:set_int("can_receive", 1) -- Enabled by default
update_meta(meta)
end,
on_receive_fields = receive_fields,
on_destruct = remove_tube,
}
if has_digilines then
def.after_place_node = function(pos, placer)
-- Set owner for digilines
core.get_meta(pos):set_string("owner", placer:get_player_name())
pipeworks.after_place(pos)
end
def.digilines = {
receptor = {
rules = pipeworks.digilines_rules,
},
effector = {
rules = pipeworks.digilines_rules,
action = digiline_action,
}
}
end
pipeworks.register_tube("pipeworks:teleport_tube", {
description = S("Teleporting Pneumatic Tube Segment"),
inventory_image = "pipeworks_teleport_tube_inv.png",
noctr = { "pipeworks_teleport_tube_noctr.png" },
plain = { "pipeworks_teleport_tube_plain.png" },
ends = { "pipeworks_teleport_tube_end.png" },
short = "pipeworks_teleport_tube_short.png",
node_def = def,
})
if core.get_modpath("mesecons_mvps") then
-- Update tubes when moved by pistons
mesecon.register_on_mvps_move(function(moved_nodes)
for _, n in ipairs(moved_nodes) do
if n.node.name:find("pipeworks:teleport_tube") then
local meta = core.get_meta(n.pos)
set_tube(n.pos, meta:get_string("channel"), meta:get_int("can_receive"))
end
end
end)
end
-- Expose teleport tube database API for other mods
pipeworks.tptube = {
version = tube_db_version,
hash = hash_pos,
get_db = function() return tube_db end,
save_tube_db = save_tube_db,
remove_tube = remove_tube,
set_tube = set_tube,
save_tube = save_tube,
update_tube = update_tube,
update_meta = function(meta, cr)
-- Legacy behaviour
if cr ~= nil then
meta:set_int("can_receive", cr and 1 or 0)
end
update_meta(meta)
end,
}
-- Load the database
read_tube_db()

127
tubes/vacuum.lua Normal file
View File

@@ -0,0 +1,127 @@
local S = core.get_translator("pipeworks")
local has_vislib = core.get_modpath("vizlib")
local enable_max = core.settings:get_bool("pipeworks_enable_items_per_tube_limit", true)
local max_items = tonumber(core.settings:get("pipeworks_max_items_per_tube")) or 30
max_items = math.ceil(max_items / 2) -- Limit vacuuming to half the max limit
local function vacuum(pos, radius)
radius = radius + 0.5
local min_pos = vector.subtract(pos, radius)
local max_pos = vector.add(pos, radius)
local count = 0
for _, obj in pairs(core.get_objects_in_area(min_pos, max_pos)) do
local entity = obj:get_luaentity()
if entity and entity.name == "__builtin:item" then
if entity.itemstring ~= "" then
pipeworks.tube_inject_item(pos, pos, vector.new(0, 0, 0), entity.itemstring)
entity.itemstring = ""
count = count + 1
end
obj:remove()
if enable_max and count >= max_items then
return -- Don't break tube by vacuuming too many items
end
end
end
end
local function set_timer(pos)
local timer = core.get_node_timer(pos)
-- Randomize timer so not all tubes vacuum at the same time
timer:start(math.random(10, 20) * 0.1)
end
local function repair_tube(pos, was_node)
core.swap_node(pos, {name = was_node.name, param2 = was_node.param2})
pipeworks.scan_for_tube_objects(pos)
set_timer(pos)
end
local function show_area(pos, node, player)
if not player or player:get_wielded_item():get_name() ~= "" then
-- Only show area when using an empty hand
return
end
local radius = tonumber(core.get_meta(pos):get("dist")) or 2
vizlib.draw_cube(pos, radius + 0.5, {player = player})
end
if pipeworks.enable_sand_tube then
pipeworks.register_tube("pipeworks:sand_tube", {
description = S("Vacuuming Pneumatic Tube Segment"),
inventory_image = "pipeworks_sand_tube_inv.png",
short = "pipeworks_sand_tube_short.png",
noctr = {"pipeworks_sand_tube_noctr.png"},
plain = {"pipeworks_sand_tube_plain.png"},
ends = {"pipeworks_sand_tube_end.png"},
node_def = {
groups = {vacuum_tube = 1},
tube = {
on_repair = repair_tube,
},
on_construct = set_timer,
on_timer = function(pos, elapsed)
vacuum(pos, 2)
set_timer(pos)
end,
on_punch = has_vislib and show_area or nil,
},
})
end
if pipeworks.enable_mese_sand_tube then
local formspec = "formspec_version[2]size[8,3]"..
pipeworks.fs_helpers.get_prepends("8,3")..
"image[0.5,0.3;1,1;pipeworks_mese_sand_tube_inv.png]"..
"label[1.75,0.8;"..S("Adjustable Vacuuming Tube").."]"..
"field[0.5,1.7;5,0.8;dist;"..S("Radius")..";${dist}]"..
"button_exit[5.5,1.7;2,0.8;save;"..S("Save").."]"
pipeworks.register_tube("pipeworks:mese_sand_tube", {
description = S("Adjustable Vacuuming Tube"),
inventory_image = "pipeworks_mese_sand_tube_inv.png",
short = "pipeworks_mese_sand_tube_short.png",
noctr = {"pipeworks_mese_sand_tube_noctr.png"},
plain = {"pipeworks_mese_sand_tube_plain.png"},
ends = {"pipeworks_mese_sand_tube_end.png"},
node_def = {
groups = {vacuum_tube = 1},
tube = {
on_repair = repair_tube,
},
on_construct = function(pos)
local meta = core.get_meta(pos)
meta:set_int("dist", 2)
meta:set_string("formspec", formspec)
meta:set_string("infotext", S("Adjustable Vacuuming Tube (@1m)", 2))
set_timer(pos)
end,
on_timer = function(pos, elapsed)
local radius = core.get_meta(pos):get_int("dist")
vacuum(pos, radius)
set_timer(pos)
end,
on_receive_fields = function(pos, _, fields, sender)
if not fields.dist or not pipeworks.may_configure(pos, sender) then
return
end
local meta = core.get_meta(pos)
local dist = math.min(math.max(tonumber(fields.dist) or 0, 0), 8)
meta:set_int("dist", dist)
meta:set_string("infotext", S("Adjustable Vacuuming Tube (@1m)", dist))
end,
on_punch = has_vislib and show_area or nil,
},
})
end
core.register_lbm({
label = "Vacuum tube node timer starter",
name = "pipeworks:vacuum_tube_start",
nodenames = {"group:vacuum_tube"},
run_at_every_load = false,
action = set_timer,
})

View File

@@ -1,128 +0,0 @@
local S = minetest.get_translator("pipeworks")
if pipeworks.enable_sand_tube then
pipeworks.register_tube("pipeworks:sand_tube", {
description = S("Vacuuming Pneumatic Tube Segment"),
inventory_image = "pipeworks_sand_tube_inv.png",
short = "pipeworks_sand_tube_short.png",
noctr = {"pipeworks_sand_tube_noctr.png"},
plain = {"pipeworks_sand_tube_plain.png"},
ends = {"pipeworks_sand_tube_end.png"},
node_def = {groups = {vacuum_tube = 1}},
})
minetest.register_craft( {
output = "pipeworks:sand_tube_1 2",
recipe = {
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet"},
{"group:sand", "group:sand", "group:sand"},
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet"}
},
})
minetest.register_craft( {
output = "pipeworks:sand_tube_1",
recipe = {
{"group:sand", "pipeworks:tube_1", "group:sand"},
},
})
end
if pipeworks.enable_mese_sand_tube then
pipeworks.register_tube("pipeworks:mese_sand_tube", {
description = S("Adjustable Vacuuming Pneumatic Tube Segment"),
inventory_image = "pipeworks_mese_sand_tube_inv.png",
short = "pipeworks_mese_sand_tube_short.png",
noctr = {"pipeworks_mese_sand_tube_noctr.png"},
plain = {"pipeworks_mese_sand_tube_plain.png"},
ends = {"pipeworks_mese_sand_tube_end.png"},
node_def = {
groups = {vacuum_tube = 1},
on_construct = function(pos)
local meta = minetest.get_meta(pos)
meta:set_int("dist", 0)
meta:set_string("formspec",
"size[6.0,2.2]"..
"image[0.2,0;1,1;pipeworks_mese_sand_tube_inv.png]"..
"label[1.2,0.2;"..S("Adjustable Vacuuming Tube").."]"..
"field[0.5,1.6;2.1,1;dist;"..S("Radius")..";${dist}]"..
"button[2.3,1.3;1.5,1;set_dist;"..S("Set").."]"..
"button_exit[3.8,1.3;2,1;close;"..S("Close").."]"..
default.gui_bg..
default.gui_bg_img)
meta:set_string("infotext", S("Adjustable Vacuuming Pneumatic Tube Segment"))
end,
on_receive_fields = function(pos,formname,fields,sender)
if (fields.quit and not fields.key_enter_field)
or (fields.key_enter_field ~= "dist" and not fields.set_dist)
or not pipeworks.may_configure(pos, sender) then
return
end
local meta = minetest.get_meta(pos)
local dist = tonumber(fields.dist)
if dist then
dist = math.max(0, dist)
dist = math.min(8, dist)
meta:set_int("dist", dist)
meta:set_string("infotext", S("Adjustable Vacuuming Pneumatic Tube Segment (@1m)", dist))
end
end,
},
})
minetest.register_craft( {
output = "pipeworks:mese_sand_tube_1 2",
recipe = {
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" },
{"group:sand", "default:mese_crystal", "group:sand" },
{"basic_materials:plastic_sheet", "basic_materials:plastic_sheet", "basic_materials:plastic_sheet" }
},
})
minetest.register_craft( {
type = "shapeless",
output = "pipeworks:mese_sand_tube_1",
recipe = {
"pipeworks:sand_tube_1",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment",
"default:mese_crystal_fragment"
},
})
end
local function vacuum(pos, radius)
radius = radius + 0.5
for _, object in pairs(minetest.get_objects_inside_radius(pos, math.sqrt(3) * radius)) do
local lua_entity = object:get_luaentity()
if not object:is_player() and lua_entity and lua_entity.name == "__builtin:item" then
local obj_pos = object:get_pos()
local minpos = vector.subtract(pos, radius)
local maxpos = vector.add(pos, radius)
if obj_pos.x >= minpos.x and obj_pos.x <= maxpos.x
and obj_pos.y >= minpos.y and obj_pos.y <= maxpos.y
and obj_pos.z >= minpos.z and obj_pos.z <= maxpos.z then
if lua_entity.itemstring ~= "" then
pipeworks.tube_inject_item(pos, pos, vector.new(0, 0, 0), lua_entity.itemstring)
lua_entity.itemstring = ""
end
object:remove()
end
end
end
end
minetest.register_abm({nodenames = {"group:vacuum_tube"},
interval = 1,
chance = 1,
label = "Vacuum tubes",
action = function(pos, node, active_object_count, active_object_count_wider)
if node.name:find("pipeworks:sand_tube") then
vacuum(pos, 2)
else
local radius = minetest.get_meta(pos):get_int("dist")
vacuum(pos, radius)
end
end
})

View File

@@ -1,238 +1,208 @@
local S = minetest.get_translator("pipeworks") local S = core.get_translator("pipeworks")
local assumed_eye_pos = vector.new(0, 1.5, 0) local has_digilines = core.get_modpath("digilines")
local function delay(x) local function set_wielder_formspec(def, meta)
return (function() return x end) local width, height = def.wield_inv.width, def.wield_inv.height
end local offset = 5.22 - width * 0.625
local size = "10.2,"..(6.5 + height * 1.25 + (has_digilines and 1.25 or 0))
local function set_wielder_formspec(data, meta) local list_bg = ""
meta:set_string("formspec", if core.get_modpath("i3") or core.get_modpath("mcl_formspec") then
"size[8,"..(6+data.wield_inv_height)..";]".. list_bg = "style_type[box;colors=#666]"
"item_image[0,0;1,1;"..data.name_base.."_off]".. for i=0, height-1 do
"label[1,0;"..minetest.formspec_escape(data.description).."]".. for j=0, width-1 do
"list[current_name;"..minetest.formspec_escape(data.wield_inv_name)..";"..((8-data.wield_inv_width)*0.5)..",1;"..data.wield_inv_width..","..data.wield_inv_height..";]".. list_bg = list_bg.."box["..offset+(i*1.25)..","..1.25+(j*1.25)..";1,1;]"
"list[current_player;main;0,"..(2+data.wield_inv_height)..";8,4;]" .. end
"listring[]")
meta:set_string("infotext", data.description)
end
local can_tool_dig_node = function(nodename, toolcaps, toolname)
--pipeworks.logger("can_tool_dig_node() STUB nodename="..tostring(nodename).." toolname="..tostring(toolname).." toolcaps: "..dump(toolcaps))
-- brief documentation of minetest.get_dig_params() as it's not yet documented in lua_api.txt:
-- takes two arguments, a node's block groups and a tool's capabilities,
-- both as they appear in their respective definitions.
-- returns a table with the following fields:
-- diggable: boolean, can this tool dig this node at all
-- time: float, time needed to dig with this tool
-- wear: int, number of wear points to inflict on the item
local nodedef = minetest.registered_nodes[nodename]
-- don't explode due to nil def in event of unknown node!
if (nodedef == nil) then return false end
local nodegroups = nodedef.groups
local diggable = minetest.get_dig_params(nodegroups, toolcaps).diggable
if not diggable then
-- a pickaxe can't actually dig leaves based on it's groups alone,
-- but a player holding one can - the game seems to fall back to the hand.
-- fall back to checking the hand's properties if the tool isn't the correct one.
local hand_caps = minetest.registered_items[""].tool_capabilities
diggable = minetest.get_dig_params(nodegroups, hand_caps).diggable
end
return diggable
end
local function wielder_on(data, wielder_pos, wielder_node)
data.fixup_node(wielder_pos, wielder_node)
if wielder_node.name ~= data.name_base.."_off" then return end
wielder_node.name = data.name_base.."_on"
minetest.swap_node(wielder_pos, wielder_node)
minetest.check_for_falling(wielder_pos)
local wielder_meta = minetest.get_meta(wielder_pos)
local inv = wielder_meta:get_inventory()
local wield_inv_name = data.wield_inv_name
local wieldindex
for i, stack in ipairs(inv:get_list(wield_inv_name)) do
if not stack:is_empty() then
wieldindex = i
break
end end
end end
if not wieldindex then local inv_offset = 1.5 + height * 1.25
if not data.ghost_inv_name then return end local fs = "formspec_version[2]size["..size.."]"..
wield_inv_name = data.ghost_inv_name pipeworks.fs_helpers.get_prepends(size)..list_bg..
inv:set_stack(wield_inv_name, 1, ItemStack(data.ghost_tool)) "item_image[0.5,0.3;1,1;"..def.name.."_off]"..
wieldindex = 1 "label[1.75,0.8;"..core.formspec_escape(def.description).."]"..
end "list[context;"..def.wield_inv.name..";"..offset..",1.25;"..width..","..height..";]"
local dir = minetest.facedir_to_dir(wielder_node.param2) if has_digilines then
-- under/above is currently intentionally left switched fs = fs.."field[1.5,"..inv_offset..";5,0.8;channel;"..S("Channel")..";${channel}]"..
-- even though this causes some problems with deployers and e.g. seeds "button_exit[6.5,"..inv_offset..";2,0.8;save;"..S("Save").."]"..
-- as there are some issues related to nodebreakers otherwise breaking 2 nodes afar. pipeworks.fs_helpers.get_inv(inv_offset + 1.25).."listring[]"
-- solidity would have to be checked as well,
-- but would open a whole can of worms related to difference in nodebreaker/deployer behavior
-- and the problems of wielders acting on themselves if below is solid
local under_pos = vector.subtract(wielder_pos, dir)
local above_pos = vector.subtract(under_pos, dir)
local pitch
local yaw
if dir.z < 0 then
yaw = 0
pitch = 0
elseif dir.z > 0 then
yaw = math.pi
pitch = 0
elseif dir.x < 0 then
yaw = 3*math.pi/2
pitch = 0
elseif dir.x > 0 then
yaw = math.pi/2
pitch = 0
elseif dir.y > 0 then
yaw = 0
pitch = -math.pi/2
else else
yaw = 0 fs = fs..pipeworks.fs_helpers.get_inv(inv_offset).."listring[]"
pitch = math.pi/2
end end
local virtplayer = pipeworks.create_fake_player({ meta:set_string("formspec", fs)
name = data.masquerade_as_owner and wielder_meta:get_string("owner") meta:set_string("infotext", def.description)
or ":pipeworks:" .. minetest.pos_to_string(wielder_pos), end
formspec = wielder_meta:get_string("formspec"),
look_dir = vector.multiply(dir, -1),
look_pitch = pitch,
look_yaw = yaw,
sneak = data.sneak,
position = vector.subtract(wielder_pos, assumed_eye_pos),
inventory = inv,
wield_index = wieldindex,
wield_list = wield_inv_name
})
local pointed_thing = { type="node", under=under_pos, above=above_pos } local function wielder_action(def, pos, node, index)
data.act(virtplayer, pointed_thing) local meta = core.get_meta(pos)
if data.eject_drops then local inv = meta:get_inventory()
local list = inv:get_list(def.wield_inv.name)
local wield_index
if index then
if list[index] and (def.wield_hand or not list[index]:is_empty()) then
wield_index = index
end
else
for i, stack in ipairs(list) do
if not stack:is_empty() then
wield_index = i
break
end
end
end
if not wield_index and not def.wield_hand then
return
end
local dir = core.facedir_to_dir(node.param2)
local fakeplayer = fakelib.create_player({
name = meta:get_string("owner"),
direction = vector.multiply(dir, -1),
position = pos,
inventory = inv,
wield_index = wield_index or 1,
wield_list = def.wield_inv.name,
})
-- Under and above positions are intentionally switched.
local pointed = {
type = "node",
under = vector.subtract(pos, dir),
above = vector.subtract(pos, vector.multiply(dir, 2)),
}
def.action(fakeplayer, pointed)
if def.eject_drops then
for i, stack in ipairs(inv:get_list("main")) do for i, stack in ipairs(inv:get_list("main")) do
if not stack:is_empty() then if not stack:is_empty() then
pipeworks.tube_inject_item(wielder_pos, wielder_pos, dir, stack) pipeworks.tube_inject_item(pos, pos, dir, stack)
inv:set_stack("main", i, ItemStack("")) inv:set_stack("main", i, ItemStack(""))
end end
end end
end end
end end
local function wielder_off(data, pos, node) local function wielder_on(def, pos, node)
if node.name == data.name_base.."_on" then if node.name ~= def.name.."_off" then
node.name = data.name_base.."_off" return
minetest.swap_node(pos, node) end
minetest.check_for_falling(pos) node.name = def.name.."_on"
core.swap_node(pos, node)
wielder_action(def, pos, node)
end
local function wielder_off(def, pos, node)
if node.name == def.name.."_on" then
node.name = def.name.."_off"
core.swap_node(pos, node)
end end
end end
local function register_wielder(data) local function wielder_digiline_action(def, pos, channel, msg)
data.fixup_node = data.fixup_node or function (pos, node) end local meta = core.get_meta(pos)
data.fixup_oldmetadata = data.fixup_oldmetadata or function (m) return m end local set_channel = meta:get_string("channel")
for _, state in ipairs({ "off", "on" }) do if channel ~= set_channel then
local groups = { snappy=2, choppy=2, oddly_breakable_by_hand=2, mesecon=2, tubedevice=1, tubedevice_receiver=1 } return
if state == "on" then groups.not_in_creative_inventory = 1 end end
local tile_images = {} if type(msg) ~= "table" then
for _, face in ipairs({ "top", "bottom", "side2", "side1", "back", "front" }) do if type(msg) == "string" then
table.insert(tile_images, data.texture_base.."_"..face..(data.texture_stateful[face] and "_"..state or "")..".png") if msg:sub(1, 8) == "activate" then
msg = {command = "activate", slot = tonumber(msg:sub(9))}
end
else
return
end end
minetest.register_node(data.name_base.."_"..state, { end
description = data.description, if msg.command == "activate" then
tiles = tile_images, local node = core.get_node(pos)
local index = type(msg.slot) == "number" and msg.slot or nil
wielder_action(def, pos, node, index)
end
end
function pipeworks.register_wielder(def)
for _,state in ipairs({"off", "on"}) do
local groups = {
snappy = 2, choppy = 2, oddly_breakable_by_hand = 2,
mesecon = 2, tubedevice = 1, tubedevice_receiver = 1,
axey = 1, handy = 1, pickaxey = 1,
not_in_creative_inventory = state == "on" and 1 or nil
}
core.register_node(def.name.."_"..state, {
description = def.description,
tiles = def.tiles[state],
paramtype2 = "facedir",
groups = groups,
is_ground_content = false,
_mcl_hardness = 0.6,
_sound_def = {
key = "node_sound_stone_defaults",
},
drop = def.name.."_off",
mesecons = { mesecons = {
effector = { effector = {
rules = pipeworks.rules_all, rules = pipeworks.rules_all,
action_on = function (pos, node) action_on = function(pos, node)
wielder_on(data, pos, node) wielder_on(def, pos, node)
end, end,
action_off = function (pos, node) action_off = function(pos, node)
wielder_off(data, pos, node) wielder_off(def, pos, node)
end,
},
},
digilines = {
receptor = {},
effector = {
action = function(pos, _, channel, msg)
wielder_digiline_action(def, pos, channel, msg)
end, end,
}, },
}, },
tube = { tube = {
can_insert = function(pos, node, stack, tubedir) can_insert = function(pos, node, stack, direction)
if not data.tube_permit_anteroposterior_insert then if def.eject_drops then
local nodedir = minetest.facedir_to_dir(node.param2) -- Prevent ejected items from being inserted
if vector.equals(tubedir, nodedir) or vector.equals(tubedir, vector.multiply(nodedir, -1)) then local dir = vector.multiply(core.facedir_to_dir(node.param2), -1)
if vector.equals(direction, dir) then
return false return false
end end
end end
local meta = minetest.get_meta(pos) local inv = core.get_meta(pos):get_inventory()
local inv = meta:get_inventory() return inv:room_for_item(def.wield_inv.name, stack)
return inv:room_for_item(data.wield_inv_name, stack)
end, end,
insert_object = function(pos, node, stack, tubedir) insert_object = function(pos, node, stack)
if not data.tube_permit_anteroposterior_insert then local inv = core.get_meta(pos):get_inventory()
local nodedir = minetest.facedir_to_dir(node.param2) return inv:add_item(def.wield_inv.name, stack)
if vector.equals(tubedir, nodedir) or vector.equals(tubedir, vector.multiply(nodedir, -1)) then
return stack
end
end
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
return inv:add_item(data.wield_inv_name, stack)
end, end,
input_inventory = data.wield_inv_name, input_inventory = def.wield_inv.name,
connect_sides = data.tube_connect_sides, connect_sides = def.connect_sides,
can_remove = function(pos, node, stack, tubedir) can_remove = function(pos, node, stack)
return stack:get_count() return stack:get_count()
end, end,
}, },
is_ground_content = true,
paramtype2 = "facedir",
tubelike = 1,
groups = groups,
sounds = default.node_sound_stone_defaults(),
drop = data.name_base.."_off",
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = core.get_meta(pos)
set_wielder_formspec(data, meta)
local inv = meta:get_inventory() local inv = meta:get_inventory()
inv:set_size(data.wield_inv_name, data.wield_inv_width*data.wield_inv_height) inv:set_size(def.wield_inv.name, def.wield_inv.width * def.wield_inv.height)
if data.ghost_inv_name then if def.eject_drops then
inv:set_size(data.ghost_inv_name, 1) inv:set_size("main", 32)
end
if data.eject_drops then
inv:set_size("main", 100)
end end
set_wielder_formspec(def, meta)
end, end,
after_place_node = function (pos, placer) after_place_node = function(pos, placer)
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
local placer_pos = placer:get_pos() if not placer then
if placer_pos and placer:is_player() then placer_pos = vector.add(placer_pos, assumed_eye_pos) end return
if placer_pos then
local dir = vector.subtract(pos, placer_pos)
local node = minetest.get_node(pos)
node.param2 = minetest.dir_to_facedir(dir, true)
minetest.set_node(pos, node)
minetest.log("action", "real (6d) facedir: " .. node.param2)
end end
minetest.get_meta(pos):set_string("owner", placer:get_player_name()) local node = core.get_node(pos)
node.param2 = core.dir_to_facedir(placer:get_look_dir(), true)
core.set_node(pos, node)
core.get_meta(pos):set_string("owner", placer:get_player_name())
end, end,
can_dig = (data.can_dig_nonempty_wield_inv and delay(true) or function(pos, player)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
return inv:is_empty(data.wield_inv_name)
end),
after_dig_node = function(pos, oldnode, oldmetadata, digger) after_dig_node = function(pos, oldnode, oldmetadata, digger)
-- The legacy-node fixup is done here in a for _,stack in ipairs(oldmetadata.inventory[def.wield_inv.name] or {}) do
-- different form from the standard fixup,
-- rather than relying on a standard fixup
-- in an on_dig callback, because some
-- non-standard diggers (such as technic's
-- mining drill) don't respect on_dig.
oldmetadata = data.fixup_oldmetadata(oldmetadata)
for _, stack in ipairs(oldmetadata.inventory[data.wield_inv_name] or {}) do
if not stack:is_empty() then if not stack:is_empty() then
minetest.add_item(pos, stack) core.add_item(pos, stack)
end end
end end
pipeworks.scan_for_tube_objects(pos) pipeworks.scan_for_tube_objects(pos)
end, end,
on_rotate = pipeworks.on_rotate, on_rotate = pipeworks.on_rotate,
on_punch = data.fixup_node,
allow_metadata_inventory_put = function(pos, listname, index, stack, player) allow_metadata_inventory_put = function(pos, listname, index, stack, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
return stack:get_count() return stack:get_count()
@@ -244,228 +214,141 @@ local function register_wielder(data)
allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player) allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player)
if not pipeworks.may_configure(pos, player) then return 0 end if not pipeworks.may_configure(pos, player) then return 0 end
return count return count
end end,
on_receive_fields = function(pos, _, fields, sender)
if not fields.channel or not pipeworks.may_configure(pos, sender) then
return
end
core.get_meta(pos):set_string("channel", fields.channel)
end,
}) })
end end
table.insert(pipeworks.ui_cat_tube_list, def.name.."_off")
end
local function get_tiles(name, stateful)
local tiles = {on = {}, off = {}}
for _,state in ipairs({"off", "on"}) do
for _,side in ipairs({"top", "bottom", "side2", "side1", "back", "front"}) do
local suffix = stateful[side] and "_"..state or ""
table.insert(tiles[state], "pipeworks_"..name.."_"..side..suffix..".png")
end
end
return tiles
end end
if pipeworks.enable_node_breaker then if pipeworks.enable_node_breaker then
local data pipeworks.register_wielder({
-- see after end of data table for other use of these variables name = "pipeworks:nodebreaker",
local name_base = "pipeworks:nodebreaker"
local wield_inv_name = "pick"
data = {
name_base = name_base,
description = S("Node Breaker"), description = S("Node Breaker"),
texture_base = "pipeworks_nodebreaker", tiles = get_tiles("nodebreaker", {top = 1, bottom = 1, side2 = 1, side1 = 1, front = 1}),
texture_stateful = { top = true, bottom = true, side2 = true, side1 = true, front = true }, connect_sides = {top = 1, bottom = 1, left = 1, right = 1, back = 1},
tube_connect_sides = { top=1, bottom=1, left=1, right=1, back=1 }, wield_inv = {name = "pick", width = 1, height = 1},
tube_permit_anteroposterior_insert = false, wield_hand = true,
wield_inv_name = wield_inv_name, eject_drops = true,
wield_inv_width = 1, action = function(fakeplayer, pointed)
wield_inv_height = 1, local stack = fakeplayer:get_wielded_item()
can_dig_nonempty_wield_inv = true, local old_stack = ItemStack(stack)
ghost_inv_name = "ghost_pick", local item_def = core.registered_items[stack:get_name()]
ghost_tool = ":", -- hand by default if item_def.on_use then
fixup_node = function (pos, node) stack = item_def.on_use(stack, fakeplayer, pointed) or stack
local meta = minetest.get_meta(pos) fakeplayer:set_wielded_item(stack)
local inv = meta:get_inventory()
-- Node breakers predating the visible pick slot
-- may have been partially updated. This code
-- fully updates them. Some have been observed
-- to have no pick slot at all; first add one.
if inv:get_size("pick") ~= 1 then
inv:set_size("pick", 1)
end
-- Originally, they had a ghost pick in a "pick"
-- inventory, no other inventory, and no form.
-- The partial update of early with-form node
-- breaker code gives them "ghost_pick" and "main"
-- inventories, but leaves the old ghost pick in
-- the "pick" inventory, and doesn't add a form.
-- First perform that partial update.
if inv:get_size("ghost_pick") ~= 1 then
inv:set_size("ghost_pick", 1)
inv:set_size("main", 100)
end
-- If the node breaker predates the visible pick
-- slot, which we can detect by it not having a
-- form, then the pick slot needs to be cleared
-- of the old ghost pick.
if (meta:get_string("formspec") or "") == "" then
inv:set_stack("pick", 1, ItemStack(""))
end
-- Finally, unconditionally set the formspec
-- and infotext. This not only makes the
-- pick slot visible for node breakers where
-- it wasn't before; it also updates the form
-- for node breakers that had an older version
-- of the form, and sets infotext where it was
-- missing for early with-form node breakers.
set_wielder_formspec(data, meta)
end,
fixup_oldmetadata = function (oldmetadata)
-- Node breakers predating the visible pick slot,
-- with node form, kept their ghost pick in an
-- inventory named "pick", the same name as the
-- later visible pick slot. The pick must be
-- removed to avoid spilling it.
if not oldmetadata.fields.formspec then
return { inventory = { pick = {} }, fields = oldmetadata.fields }
else else
return oldmetadata local node = core.get_node(pointed.under)
end local node_def = core.registered_nodes[node.name]
end, if not node_def or not node_def.on_dig then
masquerade_as_owner = true,
sneak = false,
act = function(virtplayer, pointed_thing)
--local dname = "nodebreaker.act() "
local wieldstack = virtplayer:get_wielded_item()
local oldwieldstack = ItemStack(wieldstack)
local on_use = (minetest.registered_items[wieldstack:get_name()] or {}).on_use
if on_use then
--pipeworks.logger(dname.."invoking on_use "..tostring(on_use))
wieldstack = on_use(wieldstack, virtplayer, pointed_thing) or wieldstack
virtplayer:set_wielded_item(wieldstack)
else
local under_node = minetest.get_node(pointed_thing.under)
local def = minetest.registered_nodes[under_node.name]
if not def then
-- do not dig an unknown node
return return
end end
-- check that the current tool is capable of destroying the -- Check if the tool can dig the node
-- target node. local tool = stack:get_tool_capabilities()
-- if we can't, don't dig, and leave the wield stack unchanged. if not core.get_dig_params(node_def.groups, tool).diggable then
-- note that wieldstack:get_tool_capabilities() returns hand -- Try using hand if tool can't dig the node
-- properties if the item has none of it's own. local hand = ItemStack():get_tool_capabilities()
if can_tool_dig_node(under_node.name, if not core.get_dig_params(node_def.groups, hand).diggable then
wieldstack:get_tool_capabilities(), return
wieldstack:get_name()) then
def.on_dig(pointed_thing.under, under_node, virtplayer)
local sound = def.sounds and def.sounds.dug
if sound then
minetest.sound_play(sound.name,
{pos=pointed_thing.under, gain=sound.gain})
end end
wieldstack = virtplayer:get_wielded_item()
--~ else
--pipeworks.logger(dname.."couldn't dig node!")
end end
-- This must only check for false, because `on_dig` returning nil is the same as returning true.
if node_def.on_dig(pointed.under, node, fakeplayer) == false then
return
end
local sound = node_def.sounds and node_def.sounds.dug
if sound then
core.sound_play(sound, {pos = pointed.under}, true)
end
stack = fakeplayer:get_wielded_item()
end end
local wieldname = wieldstack:get_name() if stack:get_name() == old_stack:get_name() then
if wieldname == oldwieldstack:get_name() then -- Don't mechanically wear out tool
-- don't mechanically wear out tool if stack:get_wear() ~= old_stack:get_wear() and stack:get_count() == old_stack:get_count()
if wieldstack:get_count() == oldwieldstack:get_count() and and (item_def.wear_represents == nil or item_def.wear_represents == "mechanical_wear") then
wieldstack:get_metadata() == oldwieldstack:get_metadata() and fakeplayer:set_wielded_item(old_stack)
((minetest.registered_items[wieldstack:get_name()] or {}).wear_represents or "mechanical_wear") == "mechanical_wear" then
virtplayer:set_wielded_item(oldwieldstack)
end end
elseif wieldname ~= "" then elseif not stack:is_empty() then
-- tool got replaced by something else: -- Tool got replaced by something else, treat it as a drop.
-- treat it as a drop fakeplayer:get_inventory():add_item("main", stack)
virtplayer:get_inventory():add_item("main", wieldstack) fakeplayer:set_wielded_item("")
virtplayer:set_wielded_item(ItemStack(""))
end end
end, end,
eject_drops = true,
}
register_wielder(data)
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:nodebreaker_off"
minetest.register_craft({
output = "pipeworks:nodebreaker_off",
recipe = {
{ "basic_materials:gear_steel", "basic_materials:gear_steel", "basic_materials:gear_steel" },
{ "default:stone", "mesecons:piston", "default:stone" },
{ "group:wood", "mesecons:mesecon", "group:wood" },
}
})
-- aliases for when someone had technic installed, but then uninstalled it but not pipeworks
minetest.register_alias("technic:nodebreaker_off", "pipeworks:nodebreaker_off")
minetest.register_alias("technic:nodebreaker_on", "pipeworks:nodebreaker_on")
minetest.register_alias("technic:node_breaker_off", "pipeworks:nodebreaker_off")
minetest.register_alias("technic:node_breaker_on", "pipeworks:nodebreaker_on")
-- turn legacy auto-tree-taps into node breakers
dofile(pipeworks.modpath.."/legacy.lua")
-- register LBM for transition to cheaper node breakers
local lbm_id = "pipeworks:refund_node_breaker_pick"
minetest.register_lbm({
name = lbm_id,
label = "Give back mese pick for pre-transition node breakers",
run_at_every_load = false,
nodenames = { name_base.."_on", name_base.."_off" },
action = function(pos, node)
pipeworks.logger(lbm_id.." entry, nodename="..node.name)
local invref = minetest.get_meta(pos):get_inventory()
invref:add_item(wield_inv_name, ItemStack("default:pick_mese"))
end
}) })
core.register_alias("technic:nodebreaker_off", "pipeworks:nodebreaker_off")
core.register_alias("technic:nodebreaker_on", "pipeworks:nodebreaker_on")
core.register_alias("technic:node_breaker_off", "pipeworks:nodebreaker_off")
core.register_alias("technic:node_breaker_on", "pipeworks:nodebreaker_on")
core.register_alias("auto_tree_tap:off", "pipeworks:nodebreaker_off")
core.register_alias("auto_tree_tap:on", "pipeworks:nodebreaker_on")
end end
if pipeworks.enable_deployer then if pipeworks.enable_deployer then
register_wielder({ pipeworks.register_wielder({
name_base = "pipeworks:deployer", name = "pipeworks:deployer",
description = S("Deployer"), description = S("Deployer"),
texture_base = "pipeworks_deployer", tiles = get_tiles("deployer", {front = 1}),
texture_stateful = { front = true }, connect_sides = {back = 1},
tube_connect_sides = { back=1 }, wield_inv = {name = "main", width = 3, height = 3},
tube_permit_anteroposterior_insert = true, action = function(fakeplayer, pointed)
wield_inv_name = "main", local stack = fakeplayer:get_wielded_item()
wield_inv_width = 3, local def = core.registered_items[stack:get_name()]
wield_inv_height = 3, if def and def.on_place then
can_dig_nonempty_wield_inv = false, local new_stack, placed_pos = def.on_place(stack, fakeplayer, pointed)
masquerade_as_owner = true, fakeplayer:set_wielded_item(new_stack or stack)
sneak = false, -- core.item_place_node doesn't play sound to the placer
act = function(virtplayer, pointed_thing) local sound = placed_pos and def.sounds and def.sounds.place
local wieldstack = virtplayer:get_wielded_item() local name = fakeplayer:get_player_name()
virtplayer:set_wielded_item((minetest.registered_items[wieldstack:get_name()] or {on_place=minetest.item_place}).on_place(wieldstack, virtplayer, pointed_thing) or wieldstack) if sound and name ~= "" then
core.sound_play(sound, {pos = placed_pos, to_player = name}, true)
end
end
end, end,
eject_drops = false,
}) })
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:deployer_off" core.register_alias("technic:deployer_off", "pipeworks:deployer_off")
minetest.register_craft({ core.register_alias("technic:deployer_on", "pipeworks:deployer_on")
output = "pipeworks:deployer_off",
recipe = {
{ "group:wood", "default:chest", "group:wood" },
{ "default:stone", "mesecons:piston", "default:stone" },
{ "default:stone", "mesecons:mesecon", "default:stone" },
}
})
-- aliases for when someone had technic installed, but then uninstalled it but not pipeworks
minetest.register_alias("technic:deployer_off", "pipeworks:deployer_off")
minetest.register_alias("technic:deployer_on", "pipeworks:deployer_on")
end end
if pipeworks.enable_dispenser then if pipeworks.enable_dispenser then
register_wielder({ -- Override core.item_drop to negate its hardcoded offset
name_base = "pipeworks:dispenser", -- when the dropper is a fake player.
local item_drop = core.item_drop
-- luacheck: ignore 122
function core.item_drop(stack, dropper, pos)
if dropper and dropper.is_fake_player then
pos = vector.new(pos.x, pos.y - 1.2, pos.z)
end
return item_drop(stack, dropper, pos)
end
pipeworks.register_wielder({
name = "pipeworks:dispenser",
description = S("Dispenser"), description = S("Dispenser"),
texture_base = "pipeworks_dispenser", tiles = get_tiles("dispenser", {front = 1}),
texture_stateful = { front = true }, connect_sides = {back = 1},
tube_connect_sides = { back=1 }, wield_inv = {name = "main", width = 3, height = 3},
tube_permit_anteroposterior_insert = true, action = function(fakeplayer)
wield_inv_name = "main", local stack = fakeplayer:get_wielded_item()
wield_inv_width = 3, local def = core.registered_items[stack:get_name()]
wield_inv_height = 3, if def and def.on_drop then
can_dig_nonempty_wield_inv = false, local pos = fakeplayer:get_pos()
masquerade_as_owner = false, fakeplayer:set_wielded_item(def.on_drop(stack, fakeplayer, pos) or stack)
sneak = true, end
act = function(virtplayer, pointed_thing)
local wieldstack = virtplayer:get_wielded_item()
virtplayer:set_wielded_item((minetest.registered_items[wieldstack:get_name()] or
{on_drop=minetest.item_drop}).on_drop(wieldstack, virtplayer, virtplayer:get_pos()) or
wieldstack)
end, end,
eject_drops = false,
})
pipeworks.ui_cat_tube_list[#pipeworks.ui_cat_tube_list+1] = "pipeworks:dispenser_off"
minetest.register_craft({
output = "pipeworks:dispenser_off",
recipe = {
{ "default:desert_sand", "default:chest", "default:desert_sand" },
{ "default:stone", "mesecons:piston", "default:stone" },
{ "default:stone", "mesecons:mesecon", "default:stone" },
}
}) })
end end