local S = minetest.get_translator(minetest.get_current_modname()) local F = minetest.formspec_escape local C = minetest.colorize local max_text_length = 4500 -- TODO: Increase to 12800 when scroll bar was added to written book local max_title_length = 64 local bookshelf_inv = minetest.settings:get_bool("mcl_bookshelf_inventories", true) local header = "" if minetest.get_modpath("mcl_init") then header = "no_prepend[]" .. mcl_vars.gui_nonbg .. mcl_vars.gui_bg_color .. "style_type[button;border=false;bgimg=mcl_books_button9.png;bgimg_pressed=mcl_books_button9_pressed.png;bgimg_middle=2,2]" end -- Book minetest.register_craftitem("mcl_books:book", { description = S("Book"), _doc_items_longdesc = S("Books are used to make bookshelves and book and quills."), inventory_image = "default_book.png", stack_max = 64, groups = { book = 1, craftitem = 1, enchantability = 1 }, _mcl_enchanting_enchanted_tool = "mcl_enchanting:book_enchanted", }) if minetest.get_modpath("mcl_core") and minetest.get_modpath("mcl_mobitems") then minetest.register_craft({ type = "shapeless", output = "mcl_books:book", recipe = { "mcl_core:paper", "mcl_core:paper", "mcl_core:paper", "mcl_mobitems:leather", } }) end -- Get the included text out of the book item -- itemstack: Book item -- meta: Meta of book (optional) local function get_text(itemstack) -- Grab the text local meta = itemstack:get_meta() local text = meta:get_string("text") -- Backwards-compability with MCL2 0.21.0 -- Remember that get_metadata is deprecated since MT 0.4.16! if text == nil or text == "" then local meta_legacy = itemstack:get_metadata() if itemstack:get_name() == "mcl_books:written_book" then local des = minetest.deserialize(meta_legacy) if des then text = des.text end else text = meta_legacy end end -- Security check if not text then text = "" end return text end local function make_description(title, author, generation) local desc if generation == 0 then desc = S("“@1”", title) elseif generation == 1 then desc = S("Copy of “@1”", title) elseif generation == 2 then desc = S("Copy of Copy of “@1”", title) else desc = S("Tattered Book") end desc = desc .. "\n" .. minetest.colorize(mcl_colors.GRAY, S("by @1", author)) return desc end local function cap_text_length(text, max_length) return string.sub(text, 1, max_length) end local function write(itemstack, user, pointed_thing) -- Call on_rightclick if the pointed node defines it if pointed_thing.type == "node" then local node = minetest.get_node(pointed_thing.under) if user and not user:get_player_control().sneak then if minetest.registered_nodes[node.name] and minetest.registered_nodes[node.name].on_rightclick then return minetest.registered_nodes[node.name].on_rightclick(pointed_thing.under, node, user, itemstack) or itemstack end end end local text = get_text(itemstack) local formspec = "size[8,9]" .. header .. "background[-0.5,-0.5;9,10;mcl_books_book_bg.png]" .. "textarea[0.75,0.1;7.25,9;text;;" .. minetest.formspec_escape(text) .. "]" .. "button[0.75,7.95;3,1;sign;" .. minetest.formspec_escape(S("Sign")) .. "]" .. "button_exit[4.25,7.95;3,1;ok;" .. minetest.formspec_escape(S("Done")) .. "]" minetest.show_formspec(user:get_player_name(), "mcl_books:writable_book", formspec) end local function read(itemstack, user, pointed_thing) -- Call on_rightclick if the pointed node defines it if pointed_thing.type == "node" then local node = minetest.get_node(pointed_thing.under) if user and not user:get_player_control().sneak then if minetest.registered_nodes[node.name] and minetest.registered_nodes[node.name].on_rightclick then return minetest.registered_nodes[node.name].on_rightclick(pointed_thing.under, node, user, itemstack) or itemstack end end end local text = get_text(itemstack) local formspec = "size[8,9]" .. header .. "background[-0.5,-0.5;9,10;mcl_books_book_bg.png]" .. "textarea[0.75,0.1;7.25,9;;" .. minetest.formspec_escape(text) .. ";]" .. "button_exit[2.25,7.95;3,1;ok;" .. minetest.formspec_escape(S("Done")) .. "]" minetest.show_formspec(user:get_player_name(), "mcl_books:written_book", formspec) end -- Book and Quill minetest.register_craftitem("mcl_books:writable_book", { description = S("Book and Quill"), _tt_help = S("Write down some notes"), _doc_items_longdesc = S("This item can be used to write down some notes."), _doc_items_usagehelp = S( "Hold it in the hand, then rightclick to read the current notes and edit then. You can edit the text as often as you like. You can also sign the book which turns it into a written book which you can stack, but it can't be edited anymore.") .. "\n" .. S("A book can hold up to 4500 characters. The title length is limited to 64 characters."), inventory_image = "mcl_books_book_writable.png", groups = { book = 1 }, stack_max = 1, on_place = write, on_secondary_use = write, }) minetest.register_on_player_receive_fields(function(player, formname, fields) if ((formname == "mcl_books:writable_book") and fields and fields.text) then local stack = player:get_wielded_item() if (stack:get_name() and (stack:get_name() == "mcl_books:writable_book")) then local meta = stack:get_meta() local text = cap_text_length(fields.text, max_text_length) if fields.ok then meta:set_string("text", text) player:set_wielded_item(stack) elseif fields.sign then meta:set_string("text", text) player:set_wielded_item(stack) local name = player:get_player_name() local formspec = "size[8,9]" .. header .. "background[-0.5,-0.5;9,10;mcl_books_book_bg.png]" .. "field[0.75,1;7.25,1;title;" .. minetest.formspec_escape(minetest.colorize("#000000", S("Enter book title:"))) .. ";]" .. "label[0.75,1.5;" .. minetest.formspec_escape(minetest.colorize("#404040", S("by @1", name))) .. "]" .. "button_exit[0.75,7.95;3,1;sign;" .. minetest.formspec_escape(S("Sign and Close")) .. "]" .. "tooltip[sign;" .. minetest.formspec_escape(S("Note: The book will no longer be editable after signing")) .. "]" .. "button[4.25,7.95;3,1;cancel;" .. minetest.formspec_escape(S("Cancel")) .. "]" minetest.show_formspec(player:get_player_name(), "mcl_books:signing", formspec) end end elseif ((formname == "mcl_books:signing") and fields and fields.sign and fields.title) then local newbook = ItemStack("mcl_books:written_book") local book = player:get_wielded_item() local name = player:get_player_name() if book:get_name() == "mcl_books:writable_book" then local title = fields.title if string.len(title) == 0 then title = S("Nameless Book") end title = cap_text_length(title, max_title_length) local meta = newbook:get_meta() local text = cap_text_length(get_text(book), max_text_length) meta:set_string("title", title) meta:set_string("author", name) meta:set_string("text", text) meta:set_string("description", make_description(title, name, 0)) -- The book copy counter. 0 = original, 1 = copy of original, 2 = copy of copy of original, … meta:set_int("generation", 0) player:set_wielded_item(newbook) else minetest.log("error", "[mcl_books] " .. name .. " failed to sign a book!") end elseif ((formname == "mcl_books:signing") and fields and fields.cancel) then local book = player:get_wielded_item() if book:get_name() == "mcl_books:writable_book" then write(book, player, { type = "nothing" }) end end end) if minetest.get_modpath("mcl_dye") and minetest.get_modpath("mcl_mobitems") then minetest.register_craft({ type = "shapeless", output = "mcl_books:writable_book", recipe = { "mcl_books:book", "mcl_dye:black", "mcl_mobitems:feather" }, }) end -- Written Book minetest.register_craftitem("mcl_books:written_book", { description = S("Written Book"), _doc_items_longdesc = S( "Written books contain some text written by someone. They can be read and copied, but not edited." ), _doc_items_usagehelp = S("Hold it in your hand, then rightclick to read the book.") .. "\n\n" .. S( "To copy the text of the written book, place it into the crafting grid together with a book and quill (or multiple of those) and craft. The written book will not be consumed. Copies of copies can not be copied." ), inventory_image = "mcl_books_book_written.png", groups = { not_in_creative_inventory = 1, book = 1, no_rename = 1 }, stack_max = 16, on_place = read, on_secondary_use = read }) -- Copy books -- This adds 8 recipes local baq = "mcl_books:writable_book" local wb = "mcl_books:written_book" local recipes = { { wb, baq }, { baq, baq, wb }, { baq, baq, wb, baq }, { baq, baq, baq, baq, wb }, { baq, baq, baq, baq, wb, baq }, { baq, baq, baq, baq, wb, baq, baq }, { baq, baq, baq, baq, wb, baq, baq, baq }, { baq, baq, baq, baq, wb, baq, baq, baq, baq }, } for r = #recipes, 1, -1 do minetest.register_craft({ type = "shapeless", output = "mcl_books:written_book " .. r, recipe = recipes[r], }) end minetest.register_craft_predict(function(itemstack, player, old_craft_grid, craft_inv) if itemstack:get_name() ~= "mcl_books:written_book" then return end local original for i = 1, player:get_inventory():get_size("craft") do if old_craft_grid[i]:get_name() == "mcl_books:written_book" then original = old_craft_grid[i] end end if not original then return end local ometa = original:get_meta() local generation = ometa:get_int("generation") -- Check generation, don't allow crafting with copy of copy of book if generation >= 2 then return ItemStack("") else -- Valid copy. Let's update the description field of the result item -- so it is properly displayed in the crafting grid. local imeta = itemstack:get_meta() local title = cap_text_length(ometa:get_string("title"), max_title_length) local author = ometa:get_string("author") -- Increase book generation and update description generation = generation + 1 if generation < 1 then generation = 1 end local desc = make_description(title, author, generation) imeta:set_string("description", desc) return itemstack end end) minetest.register_on_craft(function(itemstack, player, old_craft_grid, craft_inv) if itemstack:get_name() ~= "mcl_books:written_book" then return end local original local index for i = 1, player:get_inventory():get_size("craft") do if old_craft_grid[i]:get_name() == "mcl_books:written_book" then original = old_craft_grid[i] index = i end end if not original then return end -- copy of the book local text = get_text(original) if not text or text == "" then local copymeta = original:get_metadata() itemstack:set_metadata(copymeta) else local ometa = original:get_meta() local generation = ometa:get_int("generation") -- No copy of copy of copy of book allowed if generation >= 2 then return ItemStack("") end -- Copy metadata local imeta = itemstack:get_meta() local title = cap_text_length(ometa:get_string("title"), max_title_length) local author = ometa:get_string("author") imeta:set_string("title", title) imeta:set_string("author", author) imeta:set_string("text", cap_text_length(text, max_text_length)) -- Increase book generation and update description generation = generation + 1 if generation < 1 then generation = 1 end local desc = make_description(title, author, generation) imeta:set_string("description", desc) imeta:set_int("generation", generation) end -- put the book with metadata back in the craft grid craft_inv:set_stack("craft", index, original) end) local wood_sound if minetest.get_modpath("mcl_sounds") then wood_sound = mcl_sounds.node_sound_wood_defaults() end -- Bookshelf GUI local drop_content = mcl_util.drop_items_from_meta_container("main") local function on_blast(pos) local node = minetest.get_node(pos) drop_content(pos, node) minetest.remove_node(pos) end -- Simple protection checking functions local function protection_check_move(pos, from_list, from_index, to_list, to_index, count, player) local name = player:get_player_name() if minetest.is_protected(pos, name) then minetest.record_protection_violation(pos, name) return 0 else return count end end local function protection_check_put_take(pos, listname, index, stack, player) local name = player:get_player_name() if minetest.is_protected(pos, name) then minetest.record_protection_violation(pos, name) return 0 elseif minetest.get_item_group(stack:get_name(), "book") ~= 0 or stack:get_name() == "mcl_enchanting:book_enchanted" then return stack:get_count() else return 0 end end ---@param pos Vector ---@param node node ---@param clicker ObjectRef local function bookshelf_gui(pos, node, clicker) if not bookshelf_inv then return end local name = minetest.get_meta(pos):get_string("name") if name == "" then name = S("Bookshelf") end local playername = clicker:get_player_name() minetest.show_formspec(playername, "mcl_books:bookshelf_" .. pos.x .. "_" .. pos.y .. "_" .. pos.z, table.concat({ "formspec_version[4]", "size[11.75,10.425]", "label[0.375,0.375;" .. F(C(mcl_formspec.label_color, name)) .. "]", mcl_formspec.get_itemslot_bg_v4(0.375, 0.75, 9, 3), mcl_formspec.get_itemslot_bg_v4(0.375, 0.75, 9, 3, 0, "mcl_book_book_empty_slot.png"), "list[nodemeta:" .. pos.x .. "," .. pos.y .. "," .. pos.z .. ";main;0.375,0.75;9,3;]", "label[0.375,4.7;" .. F(C(mcl_formspec.label_color, S("Inventory"))) .. "]", mcl_formspec.get_itemslot_bg_v4(0.375, 5.1, 9, 3), "list[current_player;main;0.375,5.1;9,3;9]", mcl_formspec.get_itemslot_bg_v4(0.375, 9.05, 9, 1), "list[current_player;main;0.375,9.05;9,1;]", "listring[nodemeta:" .. pos.x .. "," .. pos.y .. "," .. pos.z .. ";main]", "listring[current_player;main]", }) ) end local function close_forms(pos) local players = minetest.get_connected_players() local formname = "mcl_books:bookshelf_" .. pos.x .. "_" .. pos.y .. "_" .. pos.z for p = 1, #players do if vector.distance(players[p]:get_pos(), pos) <= 30 then minetest.close_formspec(players[p]:get_player_name(), formname) end end end -- Bookshelf minetest.register_node("mcl_books:bookshelf", { description = S("Bookshelf"), _doc_items_longdesc = S("Bookshelves are used for decoration."), tiles = { "mcl_books_bookshelf_top.png", "mcl_books_bookshelf_top.png", "default_bookshelf.png" }, stack_max = 64, is_ground_content = false, groups = { handy = 1, axey = 1, deco_block = 1, material_wood = 1, flammable = 3, fire_encouragement = 30, fire_flammability = 20, container = 2 }, drop = "mcl_books:book 3", sounds = wood_sound, _mcl_blast_resistance = 1.5, _mcl_hardness = 1.5, _mcl_silk_touch_drop = true, on_construct = function(pos) local meta = minetest.get_meta(pos) local inv = meta:get_inventory() inv:set_size("main", 9 * 3) end, after_place_node = function(pos, placer, itemstack, pointed_thing) minetest.get_meta(pos):set_string("name", itemstack:get_meta():get_string("name")) end, allow_metadata_inventory_move = protection_check_move, allow_metadata_inventory_take = protection_check_put_take, allow_metadata_inventory_put = protection_check_put_take, on_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player) minetest.log("action", player:get_player_name() .. " moves stuff in bookshelf at " .. minetest.pos_to_string(pos)) end, on_metadata_inventory_put = function(pos, listname, index, stack, player) minetest.log("action", player:get_player_name() .. " moves stuff to bookshelf at " .. minetest.pos_to_string(pos)) end, on_metadata_inventory_take = function(pos, listname, index, stack, player) minetest.log("action", player:get_player_name() .. " takes stuff from bookshelf at " .. minetest.pos_to_string(pos)) end, after_dig_node = drop_content, on_blast = on_blast, on_rightclick = bookshelf_gui, on_destruct = close_forms, _mcl_hoppers_on_try_push = function(pos, hop_pos, hop_inv, hop_list) local meta = minetest.get_meta(pos) local inv = meta:get_inventory() local function filter(stack) return minetest.get_item_group(stack:get_name(), "book") == 1 or stack:get_name() == "mcl_enchanting:book_enchanted" end return inv, "main", mcl_util.select_stack(hop_inv, hop_list, inv, "main", filter, 1) end, }) minetest.register_craft({ output = "mcl_books:bookshelf", recipe = { { "group:wood", "group:wood", "group:wood" }, { "mcl_books:book", "mcl_books:book", "mcl_books:book" }, { "group:wood", "group:wood", "group:wood" }, } }) minetest.register_craft({ type = "fuel", recipe = "mcl_books:bookshelf", burntime = 15, })