Compare commits
	
		
			1 Commits
		
	
	
		
			2019-06-18
			...
			unique_ids
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 767d3e5ef7 | 
							
								
								
									
										17
									
								
								.luacheckrc
									
									
									
									
									
								
							
							
						
						| @@ -1,17 +0,0 @@ | ||||
| unused_args = false | ||||
| allow_defined_top = true | ||||
|  | ||||
| read_globals = { | ||||
| 	"DIR_DELIM", | ||||
| 	"minetest", "core", | ||||
| 	"dump", | ||||
| 	"vector", "nodeupdate", | ||||
| 	"VoxelManip", "VoxelArea", | ||||
| 	"PseudoRandom", "ItemStack", | ||||
| 	"intllib", | ||||
| 	"default", | ||||
| 	table = { fields = { "copy", "getn" } }, | ||||
| 	"biome_lib", | ||||
| 	"stairs", "stairsplus", | ||||
| } | ||||
|  | ||||
							
								
								
									
										960
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						| @@ -3,618 +3,390 @@ Minetest mod moretrees | ||||
|  | ||||
| All source code: | ||||
| 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | ||||
| Date & cocos palm code (date_palm.lua, cocos_palm.lua) | ||||
| 	© 2016, Rogier <rogier777@gmail.com> | ||||
| All date & date palm textures, date-based food, cocos flower & green coconuts, | ||||
| and all poplar textures: | ||||
| 	© 2016, Rogier <rogier777@gmail.com> | ||||
| 	- Three of the date palm textures are modifications of existing moretrees textures | ||||
| 	- The green coconuts are a modification of the brown coconut | ||||
| 	- The date cake batter is a modification of the acorn muffin batter | ||||
| All other sapling textures (textures/*_sapling.png): | ||||
| 	Published under the terms and conditions of the WTFPL. | ||||
| All sapling textures (textures/*_sapling.png): | ||||
| 	© 2013, Tim Huppertz <mitroman@naturalnet.de> | ||||
| 	Published under the terms and conditions of CC-BY-SA-3.0 Unported. | ||||
| All other textures: | ||||
| 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | ||||
| 	Published under the terms and conditions of CC-BY-SA-3.0 Unported. | ||||
|  | ||||
| ############################################################################### | ||||
| ------------------------------------------------------------------------------- | ||||
|  | ||||
| License for all code: LGPL 3.0 | ||||
| License for all media and all other assets:  CC-by-SA 4.0 | ||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE | ||||
|                     Version 2, December 2004 | ||||
|  | ||||
| ############################################################################### | ||||
|  Copyright (C) 2004 Sam Hocevar <sam@hocevar.net> | ||||
|  | ||||
|                    GNU LESSER GENERAL PUBLIC LICENSE | ||||
|                        Version 3, 29 June 2007 | ||||
|  Everyone is permitted to copy and distribute verbatim or modified | ||||
|  copies of this license document, and changing it is allowed as long | ||||
|  as the name is changed. | ||||
|  | ||||
|  Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> | ||||
|  Everyone is permitted to copy and distribute verbatim copies | ||||
|  of this license document, but changing it is not allowed. | ||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE | ||||
|    TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION | ||||
|  | ||||
|   0. You just DO WHAT THE FUCK YOU WANT TO. | ||||
|  | ||||
|   This version of the GNU Lesser General Public License incorporates | ||||
| the terms and conditions of version 3 of the GNU General Public | ||||
| License, supplemented by the additional permissions listed below. | ||||
| This license is commonly known as "WTFPL". | ||||
|  | ||||
|   0. Additional Definitions. | ||||
| ------------------------------------------------------------------------------- | ||||
|  | ||||
|   As used herein, "this License" refers to version 3 of the GNU Lesser | ||||
| General Public License, and the "GNU GPL" refers to version 3 of the GNU | ||||
| General Public License. | ||||
| Creative Commons Legal Code | ||||
|  | ||||
|   "The Library" refers to a covered work governed by this License, | ||||
| other than an Application or a Combined Work as defined below. | ||||
| Attribution-ShareAlike 3.0 Unported | ||||
|  | ||||
|   An "Application" is any work that makes use of an interface provided | ||||
| by the Library, but which is not otherwise based on the Library. | ||||
| Defining a subclass of a class defined by the Library is deemed a mode | ||||
| of using an interface provided by the Library. | ||||
|     CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE | ||||
|     LEGAL SERVICES. DISTRIBUTION OF THIS LICENSE DOES NOT CREATE AN | ||||
|     ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS | ||||
|     INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES | ||||
|     REGARDING THE INFORMATION PROVIDED, AND DISCLAIMS LIABILITY FOR | ||||
|     DAMAGES RESULTING FROM ITS USE. | ||||
|  | ||||
|   A "Combined Work" is a work produced by combining or linking an | ||||
| Application with the Library.  The particular version of the Library | ||||
| with which the Combined Work was made is also called the "Linked | ||||
| Version". | ||||
|  | ||||
|   The "Minimal Corresponding Source" for a Combined Work means the | ||||
| Corresponding Source for the Combined Work, excluding any source code | ||||
| for portions of the Combined Work that, considered in isolation, are | ||||
| based on the Application, and not on the Linked Version. | ||||
|  | ||||
|   The "Corresponding Application Code" for a Combined Work means the | ||||
| object code and/or source code for the Application, including any data | ||||
| and utility programs needed for reproducing the Combined Work from the | ||||
| Application, but excluding the System Libraries of the Combined Work. | ||||
|  | ||||
|   1. Exception to Section 3 of the GNU GPL. | ||||
|  | ||||
|   You may convey a covered work under sections 3 and 4 of this License | ||||
| without being bound by section 3 of the GNU GPL. | ||||
|  | ||||
|   2. Conveying Modified Versions. | ||||
|  | ||||
|   If you modify a copy of the Library, and, in your modifications, a | ||||
| facility refers to a function or data to be supplied by an Application | ||||
| that uses the facility (other than as an argument passed when the | ||||
| facility is invoked), then you may convey a copy of the modified | ||||
| version: | ||||
|  | ||||
|    a) under this License, provided that you make a good faith effort to | ||||
|    ensure that, in the event an Application does not supply the | ||||
|    function or data, the facility still operates, and performs | ||||
|    whatever part of its purpose remains meaningful, or | ||||
|  | ||||
|    b) under the GNU GPL, with none of the additional permissions of | ||||
|    this License applicable to that copy. | ||||
|  | ||||
|   3. Object Code Incorporating Material from Library Header Files. | ||||
|  | ||||
|   The object code form of an Application may incorporate material from | ||||
| a header file that is part of the Library.  You may convey such object | ||||
| code under terms of your choice, provided that, if the incorporated | ||||
| material is not limited to numerical parameters, data structure | ||||
| layouts and accessors, or small macros, inline functions and templates | ||||
| (ten or fewer lines in length), you do both of the following: | ||||
|  | ||||
|    a) Give prominent notice with each copy of the object code that the | ||||
|    Library is used in it and that the Library and its use are | ||||
|    covered by this License. | ||||
|  | ||||
|    b) Accompany the object code with a copy of the GNU GPL and this license | ||||
|    document. | ||||
|  | ||||
|   4. Combined Works. | ||||
|  | ||||
|   You may convey a Combined Work under terms of your choice that, | ||||
| taken together, effectively do not restrict modification of the | ||||
| portions of the Library contained in the Combined Work and reverse | ||||
| engineering for debugging such modifications, if you also do each of | ||||
| the following: | ||||
|  | ||||
|    a) Give prominent notice with each copy of the Combined Work that | ||||
|    the Library is used in it and that the Library and its use are | ||||
|    covered by this License. | ||||
|  | ||||
|    b) Accompany the Combined Work with a copy of the GNU GPL and this license | ||||
|    document. | ||||
|  | ||||
|    c) For a Combined Work that displays copyright notices during | ||||
|    execution, include the copyright notice for the Library among | ||||
|    these notices, as well as a reference directing the user to the | ||||
|    copies of the GNU GPL and this license document. | ||||
|  | ||||
|    d) Do one of the following: | ||||
|  | ||||
|        0) Convey the Minimal Corresponding Source under the terms of this | ||||
|        License, and the Corresponding Application Code in a form | ||||
|        suitable for, and under terms that permit, the user to | ||||
|        recombine or relink the Application with a modified version of | ||||
|        the Linked Version to produce a modified Combined Work, in the | ||||
|        manner specified by section 6 of the GNU GPL for conveying | ||||
|        Corresponding Source. | ||||
|  | ||||
|        1) Use a suitable shared library mechanism for linking with the | ||||
|        Library.  A suitable mechanism is one that (a) uses at run time | ||||
|        a copy of the Library already present on the user's computer | ||||
|        system, and (b) will operate properly with a modified version | ||||
|        of the Library that is interface-compatible with the Linked | ||||
|        Version. | ||||
|  | ||||
|    e) Provide Installation Information, but only if you would otherwise | ||||
|    be required to provide such information under section 6 of the | ||||
|    GNU GPL, and only to the extent that such information is | ||||
|    necessary to install and execute a modified version of the | ||||
|    Combined Work produced by recombining or relinking the | ||||
|    Application with a modified version of the Linked Version. (If | ||||
|    you use option 4d0, the Installation Information must accompany | ||||
|    the Minimal Corresponding Source and Corresponding Application | ||||
|    Code. If you use option 4d1, you must provide the Installation | ||||
|    Information in the manner specified by section 6 of the GNU GPL | ||||
|    for conveying Corresponding Source.) | ||||
|  | ||||
|   5. Combined Libraries. | ||||
|  | ||||
|   You may place library facilities that are a work based on the | ||||
| Library side by side in a single library together with other library | ||||
| facilities that are not Applications and are not covered by this | ||||
| License, and convey such a combined library under terms of your | ||||
| choice, if you do both of the following: | ||||
|  | ||||
|    a) Accompany the combined library with a copy of the same work based | ||||
|    on the Library, uncombined with any other library facilities, | ||||
|    conveyed under the terms of this License. | ||||
|  | ||||
|    b) Give prominent notice with the combined library that part of it | ||||
|    is a work based on the Library, and explaining where to find the | ||||
|    accompanying uncombined form of the same work. | ||||
|  | ||||
|   6. Revised Versions of the GNU Lesser General Public License. | ||||
|  | ||||
|   The Free Software Foundation may publish revised and/or new versions | ||||
| of the GNU Lesser General Public License from time to time. Such new | ||||
| versions will be similar in spirit to the present version, but may | ||||
| differ in detail to address new problems or concerns. | ||||
|  | ||||
|   Each version is given a distinguishing version number. If the | ||||
| Library as you received it specifies that a certain numbered version | ||||
| of the GNU Lesser General Public License "or any later version" | ||||
| applies to it, you have the option of following the terms and | ||||
| conditions either of that published version or of any later version | ||||
| published by the Free Software Foundation. If the Library as you | ||||
| received it does not specify a version number of the GNU Lesser | ||||
| General Public License, you may choose any version of the GNU Lesser | ||||
| General Public License ever published by the Free Software Foundation. | ||||
|  | ||||
|   If the Library as you received it specifies that a proxy can decide | ||||
| whether future versions of the GNU Lesser General Public License shall | ||||
| apply, that proxy's public statement of acceptance of any version is | ||||
| permanent authorization for you to choose that version for the | ||||
| Library. | ||||
|  | ||||
| ############################################################################### | ||||
|  | ||||
| Attribution-ShareAlike 4.0 International | ||||
|  | ||||
| ======================================================================= | ||||
|  | ||||
| Creative Commons Corporation ("Creative Commons") is not a law firm and | ||||
| does not provide legal services or legal advice. Distribution of | ||||
| Creative Commons public licenses does not create a lawyer-client or | ||||
| other relationship. Creative Commons makes its licenses and related | ||||
| information available on an "as-is" basis. Creative Commons gives no | ||||
| warranties regarding its licenses, any material licensed under their | ||||
| terms and conditions, or any related information. Creative Commons | ||||
| disclaims all liability for damages resulting from their use to the | ||||
| fullest extent possible. | ||||
|  | ||||
| Using Creative Commons Public Licenses | ||||
|  | ||||
| Creative Commons public licenses provide a standard set of terms and | ||||
| conditions that creators and other rights holders may use to share | ||||
| original works of authorship and other material subject to copyright | ||||
| and certain other rights specified in the public license below. The | ||||
| following considerations are for informational purposes only, are not | ||||
| exhaustive, and do not form part of our licenses. | ||||
|  | ||||
|      Considerations for licensors: Our public licenses are | ||||
|      intended for use by those authorized to give the public | ||||
|      permission to use material in ways otherwise restricted by | ||||
|      copyright and certain other rights. Our licenses are | ||||
|      irrevocable. Licensors should read and understand the terms | ||||
|      and conditions of the license they choose before applying it. | ||||
|      Licensors should also secure all rights necessary before | ||||
|      applying our licenses so that the public can reuse the | ||||
|      material as expected. Licensors should clearly mark any | ||||
|      material not subject to the license. This includes other CC- | ||||
|      licensed material, or material used under an exception or | ||||
|      limitation to copyright. More considerations for licensors: | ||||
| 	wiki.creativecommons.org/Considerations_for_licensors | ||||
|  | ||||
|      Considerations for the public: By using one of our public | ||||
|      licenses, a licensor grants the public permission to use the | ||||
|      licensed material under specified terms and conditions. If | ||||
|      the licensor's permission is not necessary for any reason--for | ||||
|      example, because of any applicable exception or limitation to | ||||
|      copyright--then that use is not regulated by the license. Our | ||||
|      licenses grant only permissions under copyright and certain | ||||
|      other rights that a licensor has authority to grant. Use of | ||||
|      the licensed material may still be restricted for other | ||||
|      reasons, including because others have copyright or other | ||||
|      rights in the material. A licensor may make special requests, | ||||
|      such as asking that all changes be marked or described. | ||||
|      Although not required by our licenses, you are encouraged to | ||||
|      respect those requests where reasonable. More considerations | ||||
|      for the public:  | ||||
| 	wiki.creativecommons.org/Considerations_for_licensees | ||||
|  | ||||
| ======================================================================= | ||||
|  | ||||
| Creative Commons Attribution-ShareAlike 4.0 International Public | ||||
| License | ||||
|  | ||||
| By exercising the Licensed Rights (defined below), You accept and agree | ||||
| to be bound by the terms and conditions of this Creative Commons | ||||
| Attribution-ShareAlike 4.0 International Public License ("Public | ||||
| License"). To the extent this Public License may be interpreted as a | ||||
| contract, You are granted the Licensed Rights in consideration of Your | ||||
| acceptance of these terms and conditions, and the Licensor grants You | ||||
| such rights in consideration of benefits the Licensor receives from | ||||
| making the Licensed Material available under these terms and | ||||
| conditions. | ||||
| THE WORK (AS DEFINED BELOW) IS PROVIDED UNDER THE TERMS OF THIS CREATIVE | ||||
| COMMONS PUBLIC LICENSE ("CCPL" OR "LICENSE"). THE WORK IS PROTECTED BY | ||||
| COPYRIGHT AND/OR OTHER APPLICABLE LAW. ANY USE OF THE WORK OTHER THAN AS | ||||
| AUTHORIZED UNDER THIS LICENSE OR COPYRIGHT LAW IS PROHIBITED. | ||||
|  | ||||
| BY EXERCISING ANY RIGHTS TO THE WORK PROVIDED HERE, YOU ACCEPT AND AGREE | ||||
| TO BE BOUND BY THE TERMS OF THIS LICENSE. TO THE EXTENT THIS LICENSE MAY | ||||
| BE CONSIDERED TO BE A CONTRACT, THE LICENSOR GRANTS YOU THE RIGHTS | ||||
| CONTAINED HERE IN CONSIDERATION OF YOUR ACCEPTANCE OF SUCH TERMS AND | ||||
| CONDITIONS. | ||||
|  | ||||
| Section 1 -- Definitions. | ||||
| 1. Definitions | ||||
|  | ||||
|   a. Adapted Material means material subject to Copyright and Similar | ||||
|      Rights that is derived from or based upon the Licensed Material | ||||
|      and in which the Licensed Material is translated, altered, | ||||
|      arranged, transformed, or otherwise modified in a manner requiring | ||||
|      permission under the Copyright and Similar Rights held by the | ||||
|      Licensor. For purposes of this Public License, where the Licensed | ||||
|      Material is a musical work, performance, or sound recording, | ||||
|      Adapted Material is always produced where the Licensed Material is | ||||
|      synched in timed relation with a moving image. | ||||
|  | ||||
|   b. Adapter's License means the license You apply to Your Copyright | ||||
|      and Similar Rights in Your contributions to Adapted Material in | ||||
|      accordance with the terms and conditions of this Public License. | ||||
|  | ||||
|   c. BY-SA Compatible License means a license listed at | ||||
|      creativecommons.org/compatiblelicenses, approved by Creative | ||||
|      Commons as essentially the equivalent of this Public License. | ||||
|  | ||||
|   d. Copyright and Similar Rights means copyright and/or similar rights | ||||
|      closely related to copyright including, without limitation, | ||||
|      performance, broadcast, sound recording, and Sui Generis Database | ||||
|      Rights, without regard to how the rights are labeled or | ||||
|      categorized. For purposes of this Public License, the rights | ||||
|      specified in Section 2(b)(1)-(2) are not Copyright and Similar | ||||
|      Rights. | ||||
|  | ||||
|   e. Effective Technological Measures means those measures that, in the | ||||
|      absence of proper authority, may not be circumvented under laws | ||||
|      fulfilling obligations under Article 11 of the WIPO Copyright | ||||
|      Treaty adopted on December 20, 1996, and/or similar international | ||||
|      agreements. | ||||
|  | ||||
|   f. Exceptions and Limitations means fair use, fair dealing, and/or | ||||
|      any other exception or limitation to Copyright and Similar Rights | ||||
|      that applies to Your use of the Licensed Material. | ||||
|  | ||||
|   g. License Elements means the license attributes listed in the name | ||||
|      of a Creative Commons Public License. The License Elements of this | ||||
|      Public License are Attribution and ShareAlike. | ||||
|  | ||||
|   h. Licensed Material means the artistic or literary work, database, | ||||
|      or other material to which the Licensor applied this Public | ||||
|  a. "Adaptation" means a work based upon the Work, or upon the Work and | ||||
|     other pre-existing works, such as a translation, adaptation, | ||||
|     derivative work, arrangement of music or other alterations of a | ||||
|     literary or artistic work, or phonogram or performance and includes | ||||
|     cinematographic adaptations or any other form in which the Work may be | ||||
|     recast, transformed, or adapted including in any form recognizably | ||||
|     derived from the original, except that a work that constitutes a | ||||
|     Collection will not be considered an Adaptation for the purpose of | ||||
|     this License. For the avoidance of doubt, where the Work is a musical | ||||
|     work, performance or phonogram, the synchronization of the Work in | ||||
|     timed-relation with a moving image ("synching") will be considered an | ||||
|     Adaptation for the purpose of this License. | ||||
|  b. "Collection" means a collection of literary or artistic works, such as | ||||
|     encyclopedias and anthologies, or performances, phonograms or | ||||
|     broadcasts, or other works or subject matter other than works listed | ||||
|     in Section 1(f) below, which, by reason of the selection and | ||||
|     arrangement of their contents, constitute intellectual creations, in | ||||
|     which the Work is included in its entirety in unmodified form along | ||||
|     with one or more other contributions, each constituting separate and | ||||
|     independent works in themselves, which together are assembled into a | ||||
|     collective whole. A work that constitutes a Collection will not be | ||||
|     considered an Adaptation (as defined below) for the purposes of this | ||||
|     License. | ||||
|  | ||||
|   i. Licensed Rights means the rights granted to You subject to the | ||||
|      terms and conditions of this Public License, which are limited to | ||||
|      all Copyright and Similar Rights that apply to Your use of the | ||||
|      Licensed Material and that the Licensor has authority to license. | ||||
|  | ||||
|   j. Licensor means the individual(s) or entity(ies) granting rights | ||||
|      under this Public License. | ||||
|  | ||||
|   k. Share means to provide material to the public by any means or | ||||
|      process that requires permission under the Licensed Rights, such | ||||
|      as reproduction, public display, public performance, distribution, | ||||
|      dissemination, communication, or importation, and to make material | ||||
|      available to the public including in ways that members of the | ||||
|      public may access the material from a place and at a time | ||||
|      individually chosen by them. | ||||
|  | ||||
|   l. Sui Generis Database Rights means rights other than copyright | ||||
|      resulting from Directive 96/9/EC of the European Parliament and of | ||||
|      the Council of 11 March 1996 on the legal protection of databases, | ||||
|      as amended and/or succeeded, as well as other essentially | ||||
|      equivalent rights anywhere in the world. | ||||
|  | ||||
|   m. You means the individual or entity exercising the Licensed Rights | ||||
|      under this Public License. Your has a corresponding meaning. | ||||
|  | ||||
|  | ||||
| Section 2 -- Scope. | ||||
|  | ||||
|   a. License grant. | ||||
|  | ||||
|        1. Subject to the terms and conditions of this Public License, | ||||
|           the Licensor hereby grants You a worldwide, royalty-free, | ||||
|           non-sublicensable, non-exclusive, irrevocable license to | ||||
|           exercise the Licensed Rights in the Licensed Material to: | ||||
|  | ||||
|             a. reproduce and Share the Licensed Material, in whole or | ||||
|                in part; and | ||||
|  | ||||
|             b. produce, reproduce, and Share Adapted Material. | ||||
|  | ||||
|        2. Exceptions and Limitations. For the avoidance of doubt, where | ||||
|           Exceptions and Limitations apply to Your use, this Public | ||||
|           License does not apply, and You do not need to comply with | ||||
|           its terms and conditions. | ||||
|  | ||||
|        3. Term. The term of this Public License is specified in Section | ||||
|           6(a). | ||||
|  | ||||
|        4. Media and formats; technical modifications allowed. The | ||||
|           Licensor authorizes You to exercise the Licensed Rights in | ||||
|           all media and formats whether now known or hereafter created, | ||||
|           and to make technical modifications necessary to do so. The | ||||
|           Licensor waives and/or agrees not to assert any right or | ||||
|           authority to forbid You from making technical modifications | ||||
|           necessary to exercise the Licensed Rights, including | ||||
|           technical modifications necessary to circumvent Effective | ||||
|           Technological Measures. For purposes of this Public License, | ||||
|           simply making modifications authorized by this Section 2(a) | ||||
|           (4) never produces Adapted Material. | ||||
|  | ||||
|        5. Downstream recipients. | ||||
|  | ||||
|             a. Offer from the Licensor -- Licensed Material. Every | ||||
|                recipient of the Licensed Material automatically | ||||
|                receives an offer from the Licensor to exercise the | ||||
|                Licensed Rights under the terms and conditions of this | ||||
|                Public License. | ||||
|  | ||||
|             b. Additional offer from the Licensor -- Adapted Material. | ||||
|                Every recipient of Adapted Material from You | ||||
|                automatically receives an offer from the Licensor to | ||||
|                exercise the Licensed Rights in the Adapted Material | ||||
|                under the conditions of the Adapter's License You apply. | ||||
|  | ||||
|             c. No downstream restrictions. You may not offer or impose | ||||
|                any additional or different terms or conditions on, or | ||||
|                apply any Effective Technological Measures to, the | ||||
|                Licensed Material if doing so restricts exercise of the | ||||
|                Licensed Rights by any recipient of the Licensed | ||||
|                Material. | ||||
|  | ||||
|        6. No endorsement. Nothing in this Public License constitutes or | ||||
|           may be construed as permission to assert or imply that You | ||||
|           are, or that Your use of the Licensed Material is, connected | ||||
|           with, or sponsored, endorsed, or granted official status by, | ||||
|           the Licensor or others designated to receive attribution as | ||||
|           provided in Section 3(a)(1)(A)(i). | ||||
|  | ||||
|   b. Other rights. | ||||
|  | ||||
|        1. Moral rights, such as the right of integrity, are not | ||||
|           licensed under this Public License, nor are publicity, | ||||
|           privacy, and/or other similar personality rights; however, to | ||||
|           the extent possible, the Licensor waives and/or agrees not to | ||||
|           assert any such rights held by the Licensor to the limited | ||||
|           extent necessary to allow You to exercise the Licensed | ||||
|           Rights, but not otherwise. | ||||
|  | ||||
|        2. Patent and trademark rights are not licensed under this | ||||
|           Public License. | ||||
|  | ||||
|        3. To the extent possible, the Licensor waives any right to | ||||
|           collect royalties from You for the exercise of the Licensed | ||||
|           Rights, whether directly or through a collecting society | ||||
|           under any voluntary or waivable statutory or compulsory | ||||
|           licensing scheme. In all other cases the Licensor expressly | ||||
|           reserves any right to collect such royalties. | ||||
|  | ||||
|  | ||||
| Section 3 -- License Conditions. | ||||
|  | ||||
| Your exercise of the Licensed Rights is expressly made subject to the | ||||
| following conditions. | ||||
|  | ||||
|   a. Attribution. | ||||
|  | ||||
|        1. If You Share the Licensed Material (including in modified | ||||
|           form), You must: | ||||
|  | ||||
|             a. retain the following if it is supplied by the Licensor | ||||
|                with the Licensed Material: | ||||
|  | ||||
|                  i. identification of the creator(s) of the Licensed | ||||
|                     Material and any others designated to receive | ||||
|                     attribution, in any reasonable manner requested by | ||||
|                     the Licensor (including by pseudonym if | ||||
|                     designated); | ||||
|  | ||||
|                 ii. a copyright notice; | ||||
|  | ||||
|                iii. a notice that refers to this Public License; | ||||
|  | ||||
|                 iv. a notice that refers to the disclaimer of | ||||
|                     warranties; | ||||
|  | ||||
|                  v. a URI or hyperlink to the Licensed Material to the | ||||
|                     extent reasonably practicable; | ||||
|  | ||||
|             b. indicate if You modified the Licensed Material and | ||||
|                retain an indication of any previous modifications; and | ||||
|  | ||||
|             c. indicate the Licensed Material is licensed under this | ||||
|                Public License, and include the text of, or the URI or | ||||
|                hyperlink to, this Public License. | ||||
|  | ||||
|        2. You may satisfy the conditions in Section 3(a)(1) in any | ||||
|           reasonable manner based on the medium, means, and context in | ||||
|           which You Share the Licensed Material. For example, it may be | ||||
|           reasonable to satisfy the conditions by providing a URI or | ||||
|           hyperlink to a resource that includes the required | ||||
|           information. | ||||
|  | ||||
|        3. If requested by the Licensor, You must remove any of the | ||||
|           information required by Section 3(a)(1)(A) to the extent | ||||
|           reasonably practicable. | ||||
|  | ||||
|   b. ShareAlike. | ||||
|  | ||||
|      In addition to the conditions in Section 3(a), if You Share | ||||
|      Adapted Material You produce, the following conditions also apply. | ||||
|  | ||||
|        1. The Adapter's License You apply must be a Creative Commons | ||||
|           license with the same License Elements, this version or | ||||
|           later, or a BY-SA Compatible License. | ||||
|  | ||||
|        2. You must include the text of, or the URI or hyperlink to, the | ||||
|           Adapter's License You apply. You may satisfy this condition | ||||
|           in any reasonable manner based on the medium, means, and | ||||
|           context in which You Share Adapted Material. | ||||
|  | ||||
|        3. You may not offer or impose any additional or different terms | ||||
|           or conditions on, or apply any Effective Technological | ||||
|           Measures to, Adapted Material that restrict exercise of the | ||||
|           rights granted under the Adapter's License You apply. | ||||
|  | ||||
|  | ||||
| Section 4 -- Sui Generis Database Rights. | ||||
|  | ||||
| Where the Licensed Rights include Sui Generis Database Rights that | ||||
| apply to Your use of the Licensed Material: | ||||
|  | ||||
|   a. for the avoidance of doubt, Section 2(a)(1) grants You the right | ||||
|      to extract, reuse, reproduce, and Share all or a substantial | ||||
|      portion of the contents of the database; | ||||
|  | ||||
|   b. if You include all or a substantial portion of the database | ||||
|      contents in a database in which You have Sui Generis Database | ||||
|      Rights, then the database in which You have Sui Generis Database | ||||
|      Rights (but not its individual contents) is Adapted Material, | ||||
|  | ||||
|      including for purposes of Section 3(b); and | ||||
|   c. You must comply with the conditions in Section 3(a) if You Share | ||||
|      all or a substantial portion of the contents of the database. | ||||
|  | ||||
| For the avoidance of doubt, this Section 4 supplements and does not | ||||
| replace Your obligations under this Public License where the Licensed | ||||
| Rights include other Copyright and Similar Rights. | ||||
|  | ||||
|  | ||||
| Section 5 -- Disclaimer of Warranties and Limitation of Liability. | ||||
|  | ||||
|   a. UNLESS OTHERWISE SEPARATELY UNDERTAKEN BY THE LICENSOR, TO THE | ||||
|      EXTENT POSSIBLE, THE LICENSOR OFFERS THE LICENSED MATERIAL AS-IS | ||||
|      AND AS-AVAILABLE, AND MAKES NO REPRESENTATIONS OR WARRANTIES OF | ||||
|      ANY KIND CONCERNING THE LICENSED MATERIAL, WHETHER EXPRESS, | ||||
|      IMPLIED, STATUTORY, OR OTHER. THIS INCLUDES, WITHOUT LIMITATION, | ||||
|      WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR | ||||
|      PURPOSE, NON-INFRINGEMENT, ABSENCE OF LATENT OR OTHER DEFECTS, | ||||
|      ACCURACY, OR THE PRESENCE OR ABSENCE OF ERRORS, WHETHER OR NOT | ||||
|      KNOWN OR DISCOVERABLE. WHERE DISCLAIMERS OF WARRANTIES ARE NOT | ||||
|      ALLOWED IN FULL OR IN PART, THIS DISCLAIMER MAY NOT APPLY TO YOU. | ||||
|  | ||||
|   b. TO THE EXTENT POSSIBLE, IN NO EVENT WILL THE LICENSOR BE LIABLE | ||||
|      TO YOU ON ANY LEGAL THEORY (INCLUDING, WITHOUT LIMITATION, | ||||
|      NEGLIGENCE) OR OTHERWISE FOR ANY DIRECT, SPECIAL, INDIRECT, | ||||
|      INCIDENTAL, CONSEQUENTIAL, PUNITIVE, EXEMPLARY, OR OTHER LOSSES, | ||||
|      COSTS, EXPENSES, OR DAMAGES ARISING OUT OF THIS PUBLIC LICENSE OR | ||||
|      USE OF THE LICENSED MATERIAL, EVEN IF THE LICENSOR HAS BEEN | ||||
|      ADVISED OF THE POSSIBILITY OF SUCH LOSSES, COSTS, EXPENSES, OR | ||||
|      DAMAGES. WHERE A LIMITATION OF LIABILITY IS NOT ALLOWED IN FULL OR | ||||
|      IN PART, THIS LIMITATION MAY NOT APPLY TO YOU. | ||||
|  | ||||
|   c. The disclaimer of warranties and limitation of liability provided | ||||
|      above shall be interpreted in a manner that, to the extent | ||||
|      possible, most closely approximates an absolute disclaimer and | ||||
|      waiver of all liability. | ||||
|  | ||||
|  | ||||
| Section 6 -- Term and Termination. | ||||
|  | ||||
|   a. This Public License applies for the term of the Copyright and | ||||
|      Similar Rights licensed here. However, if You fail to comply with | ||||
|      this Public License, then Your rights under this Public License | ||||
|      terminate automatically. | ||||
|  | ||||
|   b. Where Your right to use the Licensed Material has terminated under | ||||
|      Section 6(a), it reinstates: | ||||
|  | ||||
|        1. automatically as of the date the violation is cured, provided | ||||
|           it is cured within 30 days of Your discovery of the | ||||
|           violation; or | ||||
|  | ||||
|        2. upon express reinstatement by the Licensor. | ||||
|  | ||||
|      For the avoidance of doubt, this Section 6(b) does not affect any | ||||
|      right the Licensor may have to seek remedies for Your violations | ||||
|      of this Public License. | ||||
|  | ||||
|   c. For the avoidance of doubt, the Licensor may also offer the | ||||
|      Licensed Material under separate terms or conditions or stop | ||||
|      distributing the Licensed Material at any time; however, doing so | ||||
|      will not terminate this Public License. | ||||
|  | ||||
|   d. Sections 1, 5, 6, 7, and 8 survive termination of this Public | ||||
|      License. | ||||
|  | ||||
|  | ||||
| Section 7 -- Other Terms and Conditions. | ||||
|  | ||||
|   a. The Licensor shall not be bound by any additional or different | ||||
|      terms or conditions communicated by You unless expressly agreed. | ||||
|  | ||||
|   b. Any arrangements, understandings, or agreements regarding the | ||||
|      Licensed Material not stated herein are separate from and | ||||
|      independent of the terms and conditions of this Public License. | ||||
|  | ||||
|  | ||||
| Section 8 -- Interpretation. | ||||
|  | ||||
|   a. For the avoidance of doubt, this Public License does not, and | ||||
|      shall not be interpreted to, reduce, limit, restrict, or impose | ||||
|      conditions on any use of the Licensed Material that could lawfully | ||||
|      be made without permission under this Public License. | ||||
|  | ||||
|   b. To the extent possible, if any provision of this Public License is | ||||
|      deemed unenforceable, it shall be automatically reformed to the | ||||
|      minimum extent necessary to make it enforceable. If the provision | ||||
|      cannot be reformed, it shall be severed from this Public License | ||||
|      without affecting the enforceability of the remaining terms and | ||||
|      conditions. | ||||
|  | ||||
|   c. No term or condition of this Public License will be waived and no | ||||
|      failure to comply consented to unless expressly agreed to by the | ||||
|      Licensor. | ||||
|  | ||||
|   d. Nothing in this Public License constitutes or may be interpreted | ||||
|      as a limitation upon, or waiver of, any privileges and immunities | ||||
|      that apply to the Licensor or You, including from the legal | ||||
|      processes of any jurisdiction or authority. | ||||
|  | ||||
|  | ||||
| ======================================================================= | ||||
|  | ||||
| Creative Commons is not a party to its public | ||||
| licenses. Notwithstanding, Creative Commons may elect to apply one of | ||||
| its public licenses to material it publishes and in those instances | ||||
| will be considered the “Licensor.” The text of the Creative Commons | ||||
| public licenses is dedicated to the public domain under the CC0 Public | ||||
| Domain Dedication. Except for the limited purpose of indicating that | ||||
| material is shared under a Creative Commons public license or as | ||||
| otherwise permitted by the Creative Commons policies published at | ||||
| creativecommons.org/policies, Creative Commons does not authorize the | ||||
| use of the trademark "Creative Commons" or any other trademark or logo | ||||
| of Creative Commons without its prior written consent including, | ||||
| without limitation, in connection with any unauthorized modifications | ||||
| to any of its public licenses or any other arrangements, | ||||
| understandings, or agreements concerning use of licensed material. For | ||||
| the avoidance of doubt, this paragraph does not form part of the | ||||
| public licenses. | ||||
|  | ||||
| Creative Commons may be contacted at creativecommons.org. | ||||
|  c. "Creative Commons Compatible License" means a license that is listed | ||||
|     at http://creativecommons.org/compatiblelicenses that has been | ||||
|     approved by Creative Commons as being essentially equivalent to this | ||||
|     License, including, at a minimum, because that license: (i) contains | ||||
|     terms that have the same purpose, meaning and effect as the License | ||||
|     Elements of this License; and, (ii) explicitly permits the relicensing | ||||
|     of adaptations of works made available under that license under this | ||||
|     License or a Creative Commons jurisdiction license with the same | ||||
|     License Elements as this License. | ||||
|  d. "Distribute" means to make available to the public the original and | ||||
|     copies of the Work or Adaptation, as appropriate, through sale or | ||||
|     other transfer of ownership. | ||||
|  e. "License Elements" means the following high-level license attributes | ||||
|     as selected by Licensor and indicated in the title of this License: | ||||
|     Attribution, ShareAlike. | ||||
|  f. "Licensor" means the individual, individuals, entity or entities that | ||||
|     offer(s) the Work under the terms of this License. | ||||
|  g. "Original Author" means, in the case of a literary or artistic work, | ||||
|     the individual, individuals, entity or entities who created the Work | ||||
|     or if no individual or entity can be identified, the publisher; and in | ||||
|     addition (i) in the case of a performance the actors, singers, | ||||
|     musicians, dancers, and other persons who act, sing, deliver, declaim, | ||||
|     play in, interpret or otherwise perform literary or artistic works or | ||||
|     expressions of folklore; (ii) in the case of a phonogram the producer | ||||
|     being the person or legal entity who first fixes the sounds of a | ||||
|     performance or other sounds; and, (iii) in the case of broadcasts, the | ||||
|     organization that transmits the broadcast. | ||||
|  h. "Work" means the literary and/or artistic work offered under the terms | ||||
|     of this License including without limitation any production in the | ||||
|     literary, scientific and artistic domain, whatever may be the mode or | ||||
|     form of its expression including digital form, such as a book, | ||||
|     pamphlet and other writing; a lecture, address, sermon or other work | ||||
|     of the same nature; a dramatic or dramatico-musical work; a | ||||
|     choreographic work or entertainment in dumb show; a musical | ||||
|     composition with or without words; a cinematographic work to which are | ||||
|     assimilated works expressed by a process analogous to cinematography; | ||||
|     a work of drawing, painting, architecture, sculpture, engraving or | ||||
|     lithography; a photographic work to which are assimilated works | ||||
|     expressed by a process analogous to photography; a work of applied | ||||
|     art; an illustration, map, plan, sketch or three-dimensional work | ||||
|     relative to geography, topography, architecture or science; a | ||||
|     performance; a broadcast; a phonogram; a compilation of data to the | ||||
|     extent it is protected as a copyrightable work; or a work performed by | ||||
|     a variety or circus performer to the extent it is not otherwise | ||||
|     considered a literary or artistic work. | ||||
|  i. "You" means an individual or entity exercising rights under this | ||||
|     License who has not previously violated the terms of this License with | ||||
|     respect to the Work, or who has received express permission from the | ||||
|     Licensor to exercise rights under this License despite a previous | ||||
|     violation. | ||||
|  j. "Publicly Perform" means to perform public recitations of the Work and | ||||
|     to communicate to the public those public recitations, by any means or | ||||
|     process, including by wire or wireless means or public digital | ||||
|     performances; to make available to the public Works in such a way that | ||||
|     members of the public may access these Works from a place and at a | ||||
|     place individually chosen by them; to perform the Work to the public | ||||
|     by any means or process and the communication to the public of the | ||||
|     performances of the Work, including by public digital performance; to | ||||
|     broadcast and rebroadcast the Work by any means including signs, | ||||
|     sounds or images. | ||||
|  k. "Reproduce" means to make copies of the Work by any means including | ||||
|     without limitation by sound or visual recordings and the right of | ||||
|     fixation and reproducing fixations of the Work, including storage of a | ||||
|     protected performance or phonogram in digital form or other electronic | ||||
|     medium. | ||||
|  | ||||
| 2. Fair Dealing Rights. Nothing in this License is intended to reduce, | ||||
| limit, or restrict any uses free from copyright or rights arising from | ||||
| limitations or exceptions that are provided for in connection with the | ||||
| copyright protection under copyright law or other applicable laws. | ||||
|  | ||||
| 3. License Grant. Subject to the terms and conditions of this License, | ||||
| Licensor hereby grants You a worldwide, royalty-free, non-exclusive, | ||||
| perpetual (for the duration of the applicable copyright) license to | ||||
| exercise the rights in the Work as stated below: | ||||
|  | ||||
|  a. to Reproduce the Work, to incorporate the Work into one or more | ||||
|     Collections, and to Reproduce the Work as incorporated in the | ||||
|     Collections; | ||||
|  b. to create and Reproduce Adaptations provided that any such Adaptation, | ||||
|     including any translation in any medium, takes reasonable steps to | ||||
|     clearly label, demarcate or otherwise identify that changes were made | ||||
|     to the original Work. For example, a translation could be marked "The | ||||
|     original work was translated from English to Spanish," or a | ||||
|     modification could indicate "The original work has been modified."; | ||||
|  c. to Distribute and Publicly Perform the Work including as incorporated | ||||
|     in Collections; and, | ||||
|  d. to Distribute and Publicly Perform Adaptations. | ||||
|  e. For the avoidance of doubt: | ||||
|  | ||||
|      i. Non-waivable Compulsory License Schemes. In those jurisdictions in | ||||
|         which the right to collect royalties through any statutory or | ||||
|         compulsory licensing scheme cannot be waived, the Licensor | ||||
|         reserves the exclusive right to collect such royalties for any | ||||
|         exercise by You of the rights granted under this License; | ||||
|     ii. Waivable Compulsory License Schemes. In those jurisdictions in | ||||
|         which the right to collect royalties through any statutory or | ||||
|         compulsory licensing scheme can be waived, the Licensor waives the | ||||
|         exclusive right to collect such royalties for any exercise by You | ||||
|         of the rights granted under this License; and, | ||||
|    iii. Voluntary License Schemes. The Licensor waives the right to | ||||
|         collect royalties, whether individually or, in the event that the | ||||
|         Licensor is a member of a collecting society that administers | ||||
|         voluntary licensing schemes, via that society, from any exercise | ||||
|         by You of the rights granted under this License. | ||||
|  | ||||
| The above rights may be exercised in all media and formats whether now | ||||
| known or hereafter devised. The above rights include the right to make | ||||
| such modifications as are technically necessary to exercise the rights in | ||||
| other media and formats. Subject to Section 8(f), all rights not expressly | ||||
| granted by Licensor are hereby reserved. | ||||
|  | ||||
| 4. Restrictions. The license granted in Section 3 above is expressly made | ||||
| subject to and limited by the following restrictions: | ||||
|  | ||||
|  a. You may Distribute or Publicly Perform the Work only under the terms | ||||
|     of this License. You must include a copy of, or the Uniform Resource | ||||
|     Identifier (URI) for, this License with every copy of the Work You | ||||
|     Distribute or Publicly Perform. You may not offer or impose any terms | ||||
|     on the Work that restrict the terms of this License or the ability of | ||||
|     the recipient of the Work to exercise the rights granted to that | ||||
|     recipient under the terms of the License. You may not sublicense the | ||||
|     Work. You must keep intact all notices that refer to this License and | ||||
|     to the disclaimer of warranties with every copy of the Work You | ||||
|     Distribute or Publicly Perform. When You Distribute or Publicly | ||||
|     Perform the Work, You may not impose any effective technological | ||||
|     measures on the Work that restrict the ability of a recipient of the | ||||
|     Work from You to exercise the rights granted to that recipient under | ||||
|     the terms of the License. This Section 4(a) applies to the Work as | ||||
|     incorporated in a Collection, but this does not require the Collection | ||||
|     apart from the Work itself to be made subject to the terms of this | ||||
|     License. If You create a Collection, upon notice from any Licensor You | ||||
|     must, to the extent practicable, remove from the Collection any credit | ||||
|     as required by Section 4(c), as requested. If You create an | ||||
|     Adaptation, upon notice from any Licensor You must, to the extent | ||||
|     practicable, remove from the Adaptation any credit as required by | ||||
|     Section 4(c), as requested. | ||||
|  b. You may Distribute or Publicly Perform an Adaptation only under the | ||||
|     terms of: (i) this License; (ii) a later version of this License with | ||||
|     the same License Elements as this License; (iii) a Creative Commons | ||||
|     jurisdiction license (either this or a later license version) that | ||||
|     contains the same License Elements as this License (e.g., | ||||
|     Attribution-ShareAlike 3.0 US)); (iv) a Creative Commons Compatible | ||||
|     License. If you license the Adaptation under one of the licenses | ||||
|     mentioned in (iv), you must comply with the terms of that license. If | ||||
|     you license the Adaptation under the terms of any of the licenses | ||||
|     mentioned in (i), (ii) or (iii) (the "Applicable License"), you must | ||||
|     comply with the terms of the Applicable License generally and the | ||||
|     following provisions: (I) You must include a copy of, or the URI for, | ||||
|     the Applicable License with every copy of each Adaptation You | ||||
|     Distribute or Publicly Perform; (II) You may not offer or impose any | ||||
|     terms on the Adaptation that restrict the terms of the Applicable | ||||
|     License or the ability of the recipient of the Adaptation to exercise | ||||
|     the rights granted to that recipient under the terms of the Applicable | ||||
|     License; (III) You must keep intact all notices that refer to the | ||||
|     Applicable License and to the disclaimer of warranties with every copy | ||||
|     of the Work as included in the Adaptation You Distribute or Publicly | ||||
|     Perform; (IV) when You Distribute or Publicly Perform the Adaptation, | ||||
|     You may not impose any effective technological measures on the | ||||
|     Adaptation that restrict the ability of a recipient of the Adaptation | ||||
|     from You to exercise the rights granted to that recipient under the | ||||
|     terms of the Applicable License. This Section 4(b) applies to the | ||||
|     Adaptation as incorporated in a Collection, but this does not require | ||||
|     the Collection apart from the Adaptation itself to be made subject to | ||||
|     the terms of the Applicable License. | ||||
|  c. If You Distribute, or Publicly Perform the Work or any Adaptations or | ||||
|     Collections, You must, unless a request has been made pursuant to | ||||
|     Section 4(a), keep intact all copyright notices for the Work and | ||||
|     provide, reasonable to the medium or means You are utilizing: (i) the | ||||
|     name of the Original Author (or pseudonym, if applicable) if supplied, | ||||
|     and/or if the Original Author and/or Licensor designate another party | ||||
|     or parties (e.g., a sponsor institute, publishing entity, journal) for | ||||
|     attribution ("Attribution Parties") in Licensor's copyright notice, | ||||
|     terms of service or by other reasonable means, the name of such party | ||||
|     or parties; (ii) the title of the Work if supplied; (iii) to the | ||||
|     extent reasonably practicable, the URI, if any, that Licensor | ||||
|     specifies to be associated with the Work, unless such URI does not | ||||
|     refer to the copyright notice or licensing information for the Work; | ||||
|     and (iv) , consistent with Ssection 3(b), in the case of an | ||||
|     Adaptation, a credit identifying the use of the Work in the Adaptation | ||||
|     (e.g., "French translation of the Work by Original Author," or | ||||
|     "Screenplay based on original Work by Original Author"). The credit | ||||
|     required by this Section 4(c) may be implemented in any reasonable | ||||
|     manner; provided, however, that in the case of a Adaptation or | ||||
|     Collection, at a minimum such credit will appear, if a credit for all | ||||
|     contributing authors of the Adaptation or Collection appears, then as | ||||
|     part of these credits and in a manner at least as prominent as the | ||||
|     credits for the other contributing authors. For the avoidance of | ||||
|     doubt, You may only use the credit required by this Section for the | ||||
|     purpose of attribution in the manner set out above and, by exercising | ||||
|     Your rights under this License, You may not implicitly or explicitly | ||||
|     assert or imply any connection with, sponsorship or endorsement by the | ||||
|     Original Author, Licensor and/or Attribution Parties, as appropriate, | ||||
|     of You or Your use of the Work, without the separate, express prior | ||||
|     written permission of the Original Author, Licensor and/or Attribution | ||||
|     Parties. | ||||
|  d. Except as otherwise agreed in writing by the Licensor or as may be | ||||
|     otherwise permitted by applicable law, if You Reproduce, Distribute or | ||||
|     Publicly Perform the Work either by itself or as part of any | ||||
|     Adaptations or Collections, You must not distort, mutilate, modify or | ||||
|     take other derogatory action in relation to the Work which would be | ||||
|     prejudicial to the Original Author's honor or reputation. Licensor | ||||
|     agrees that in those jurisdictions (e.g. Japan), in which any exercise | ||||
|     of the right granted in Section 3(b) of this License (the right to | ||||
|     make Adaptations) would be deemed to be a distortion, mutilation, | ||||
|     modification or other derogatory action prejudicial to the Original | ||||
|     Author's honor and reputation, the Licensor will waive or not assert, | ||||
|     as appropriate, this Section, to the fullest extent permitted by the | ||||
|     applicable national law, to enable You to reasonably exercise Your | ||||
|     right under Section 3(b) of this License (right to make Adaptations) | ||||
|     but not otherwise. | ||||
|  | ||||
| 5. Representations, Warranties and Disclaimer | ||||
|  | ||||
| UNLESS OTHERWISE MUTUALLY AGREED TO BY THE PARTIES IN WRITING, LICENSOR | ||||
| OFFERS THE WORK AS-IS AND MAKES NO REPRESENTATIONS OR WARRANTIES OF ANY | ||||
| KIND CONCERNING THE WORK, EXPRESS, IMPLIED, STATUTORY OR OTHERWISE, | ||||
| INCLUDING, WITHOUT LIMITATION, WARRANTIES OF TITLE, MERCHANTIBILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE, NONINFRINGEMENT, OR THE ABSENCE OF | ||||
| LATENT OR OTHER DEFECTS, ACCURACY, OR THE PRESENCE OF ABSENCE OF ERRORS, | ||||
| WHETHER OR NOT DISCOVERABLE. SOME JURISDICTIONS DO NOT ALLOW THE EXCLUSION | ||||
| OF IMPLIED WARRANTIES, SO SUCH EXCLUSION MAY NOT APPLY TO YOU. | ||||
|  | ||||
| 6. Limitation on Liability. EXCEPT TO THE EXTENT REQUIRED BY APPLICABLE | ||||
| LAW, IN NO EVENT WILL LICENSOR BE LIABLE TO YOU ON ANY LEGAL THEORY FOR | ||||
| ANY SPECIAL, INCIDENTAL, CONSEQUENTIAL, PUNITIVE OR EXEMPLARY DAMAGES | ||||
| ARISING OUT OF THIS LICENSE OR THE USE OF THE WORK, EVEN IF LICENSOR HAS | ||||
| BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. | ||||
|  | ||||
| 7. Termination | ||||
|  | ||||
|  a. This License and the rights granted hereunder will terminate | ||||
|     automatically upon any breach by You of the terms of this License. | ||||
|     Individuals or entities who have received Adaptations or Collections | ||||
|     from You under this License, however, will not have their licenses | ||||
|     terminated provided such individuals or entities remain in full | ||||
|     compliance with those licenses. Sections 1, 2, 5, 6, 7, and 8 will | ||||
|     survive any termination of this License. | ||||
|  b. Subject to the above terms and conditions, the license granted here is | ||||
|     perpetual (for the duration of the applicable copyright in the Work). | ||||
|     Notwithstanding the above, Licensor reserves the right to release the | ||||
|     Work under different license terms or to stop distributing the Work at | ||||
|     any time; provided, however that any such election will not serve to | ||||
|     withdraw this License (or any other license that has been, or is | ||||
|     required to be, granted under the terms of this License), and this | ||||
|     License will continue in full force and effect unless terminated as | ||||
|     stated above. | ||||
|  | ||||
| 8. Miscellaneous | ||||
|  | ||||
|  a. Each time You Distribute or Publicly Perform the Work or a Collection, | ||||
|     the Licensor offers to the recipient a license to the Work on the same | ||||
|     terms and conditions as the license granted to You under this License. | ||||
|  b. Each time You Distribute or Publicly Perform an Adaptation, Licensor | ||||
|     offers to the recipient a license to the original Work on the same | ||||
|     terms and conditions as the license granted to You under this License. | ||||
|  c. If any provision of this License is invalid or unenforceable under | ||||
|     applicable law, it shall not affect the validity or enforceability of | ||||
|     the remainder of the terms of this License, and without further action | ||||
|     by the parties to this agreement, such provision shall be reformed to | ||||
|     the minimum extent necessary to make such provision valid and | ||||
|     enforceable. | ||||
|  d. No term or provision of this License shall be deemed waived and no | ||||
|     breach consented to unless such waiver or consent shall be in writing | ||||
|     and signed by the party to be charged with such waiver or consent. | ||||
|  e. This License constitutes the entire agreement between the parties with | ||||
|     respect to the Work licensed here. There are no understandings, | ||||
|     agreements or representations with respect to the Work not specified | ||||
|     here. Licensor shall not be bound by any additional provisions that | ||||
|     may appear in any communication from You. This License may not be | ||||
|     modified without the mutual written agreement of the Licensor and You. | ||||
|  f. The rights granted under, and the subject matter referenced, in this | ||||
|     License were drafted utilizing the terminology of the Berne Convention | ||||
|     for the Protection of Literary and Artistic Works (as amended on | ||||
|     September 28, 1979), the Rome Convention of 1961, the WIPO Copyright | ||||
|     Treaty of 1996, the WIPO Performances and Phonograms Treaty of 1996 | ||||
|     and the Universal Copyright Convention (as revised on July 24, 1971). | ||||
|     These rights and subject matter take effect in the relevant | ||||
|     jurisdiction in which the License terms are sought to be enforced | ||||
|     according to the corresponding provisions of the implementation of | ||||
|     those treaty provisions in the applicable national law. If the | ||||
|     standard suite of rights granted under applicable copyright law | ||||
|     includes additional rights not granted under this License, such | ||||
|     additional rights are deemed to be included in the License; this | ||||
|     License is not intended to restrict the license of any rights under | ||||
|     applicable law. | ||||
|  | ||||
|  | ||||
| Creative Commons Notice | ||||
|  | ||||
|     Creative Commons is not a party to this License, and makes no warranty | ||||
|     whatsoever in connection with the Work. Creative Commons will not be | ||||
|     liable to You or any party on any legal theory for any damages | ||||
|     whatsoever, including without limitation any general, special, | ||||
|     incidental or consequential damages arising in connection to this | ||||
|     license. Notwithstanding the foregoing two (2) sentences, if Creative | ||||
|     Commons has expressly identified itself as the Licensor hereunder, it | ||||
|     shall have all rights and obligations of Licensor. | ||||
|  | ||||
|     Except for the limited purpose of indicating to the public that the | ||||
|     Work is licensed under the CCPL, Creative Commons does not authorize | ||||
|     the use by either party of the trademark "Creative Commons" or any | ||||
|     related trademark or logo of Creative Commons without the prior | ||||
|     written consent of Creative Commons. Any permitted use will be in | ||||
|     compliance with Creative Commons' then-current trademark usage | ||||
|     guidelines, as may be published on its website or otherwise made | ||||
|     available upon request from time to time. For the avoidance of doubt, | ||||
|     this trademark restriction does not form part of the License. | ||||
|  | ||||
|     Creative Commons may be contacted at http://creativecommons.org/. | ||||
|   | ||||
| @@ -8,4 +8,4 @@ jungle trees mod, and big contributions by RealBadAngel. | ||||
| Brought together into one mod and made L-systems compatible by Vanessa | ||||
| Ezekowitz. | ||||
|  | ||||
| Dependencies: <a href="https://forum.minetest.net/viewtopic.php?f=11&t=12999">biome_lib</a> and default | ||||
| Dependencies: <a href="https://github.com/VanessaE/plantlife">plants_lib</a> and default | ||||
|   | ||||
							
								
								
									
										168
									
								
								biome_defs.lua
									
									
									
									
									
								
							
							
						
						| @@ -24,40 +24,6 @@ moretrees.palm_biome = { | ||||
| 	max_count = 10, | ||||
| } | ||||
|  | ||||
| moretrees.date_palm_biome = { | ||||
| 	surface = "default:desert_sand", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 10, | ||||
| 	seed_diff = 339, | ||||
| 	min_elevation = -1, | ||||
| 	max_elevation = 10, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 20, | ||||
| 	near_nodes_count = 100, | ||||
| 	near_nodes_vertical = 20, | ||||
| 	temp_min = -0.20, | ||||
| 	humidity_max = 0.20, | ||||
| 	rarity = 10, | ||||
| 	max_count = 30, | ||||
| } | ||||
|  | ||||
| moretrees.date_palm_biome_2 = { | ||||
| 	surface = "default:desert_sand", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 10, | ||||
| 	seed_diff = 340, | ||||
| 	min_elevation = 11, | ||||
| 	max_elevation = 30, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 1, | ||||
| 	near_nodes_count = 1, | ||||
| 	near_nodes_vertical = 30, | ||||
| 	temp_min = -0.20, | ||||
| 	humidity_max = 0.20, | ||||
| 	rarity = 10, | ||||
| 	max_count = 30, | ||||
| } | ||||
|  | ||||
| moretrees.apple_tree_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| @@ -124,6 +90,18 @@ moretrees.willow_biome = { | ||||
| 	max_count = 5, | ||||
| } | ||||
|  | ||||
| moretrees.acacia_biome = { | ||||
| 	surface = { "default:dirt_with_grass", "default:desert_sand" }, | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 15, | ||||
| 	seed_diff = 1, | ||||
| 	rarity = 50, | ||||
| 	max_count = 15, | ||||
| 	plantlife_limit = -1, | ||||
| 	humidity_min = 0.3, | ||||
| 	humidity_max = 0, | ||||
| } | ||||
|  | ||||
| moretrees.rubber_tree_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| @@ -140,24 +118,18 @@ moretrees.rubber_tree_biome = { | ||||
| } | ||||
|  | ||||
| moretrees.jungletree_biome = { | ||||
| 	surface = { | ||||
| 		"default:dirt", | ||||
| 		"default:dirt_with_grass", | ||||
| 		"woodsoils:dirt_with_leaves_1", | ||||
| 		"woodsoils:grass_with_leaves_1", | ||||
| 		"woodsoils:grass_with_leaves_2" | ||||
| 	}, | ||||
| 	avoid_nodes = {"moretrees:jungletree_trunk"}, | ||||
| 	max_count = 12, | ||||
| 	avoid_radius = 3, | ||||
| 	rarity = 85, | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 5, | ||||
| 	seed_diff = 329, | ||||
| 	min_elevation = 1, | ||||
| 	near_nodes = {"default:jungletree"}, | ||||
| 	near_nodes_size = 6, | ||||
| 	near_nodes_vertical = 2, | ||||
| 	near_nodes_count = 1, | ||||
| 	plantlife_limit = -0.9, | ||||
| 	min_elevation = -5, | ||||
| 	max_elevation = 10, | ||||
| 	temp_min = 0.25, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 20, | ||||
| 	near_nodes_count = 7, | ||||
| 	rarity = 10, | ||||
| 	max_count = 10, | ||||
| } | ||||
|  | ||||
| moretrees.spruce_biome = { | ||||
| @@ -172,7 +144,7 @@ moretrees.spruce_biome = { | ||||
| 	max_count = 5, | ||||
| } | ||||
|  | ||||
| moretrees.cedar_biome = { | ||||
| moretrees.pine_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 10, | ||||
| @@ -184,97 +156,6 @@ moretrees.cedar_biome = { | ||||
| 	max_count = 10, | ||||
| } | ||||
|  | ||||
|  | ||||
| -- Poplar requires a lot of water. | ||||
| moretrees.poplar_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 6, | ||||
| 	seed_diff = 341, | ||||
| 	min_elevation = 0, | ||||
| 	max_elevation = 50, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 15, | ||||
| 	near_nodes_vertical = 5, | ||||
| 	near_nodes_count = 1, | ||||
| 	humidity_min = -0.7, | ||||
| 	humidity_max = -1, | ||||
| 	rarity = 50, | ||||
| 	max_count = 15, | ||||
| } | ||||
|  | ||||
| -- The humidity requirement it quite restrictive (apparently). | ||||
| -- Spawn an occasional poplar elsewhere. | ||||
| moretrees.poplar_biome_2 = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 6, | ||||
| 	seed_diff = 341, | ||||
| 	min_elevation = 0, | ||||
| 	max_elevation = 50, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 15, | ||||
| 	near_nodes_vertical = 4, | ||||
| 	near_nodes_count = 10, | ||||
| 	humidity_min = 0.1, | ||||
| 	humidity_max = -0.6, | ||||
| 	rarity = 50, | ||||
| 	max_count = 1, | ||||
| } | ||||
|  | ||||
| -- Subterranean lakes provide enough water for poplars to grow | ||||
| moretrees.poplar_biome_3 = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 6, | ||||
| 	seed_diff = 342, | ||||
| 	min_elevation = 0, | ||||
| 	max_elevation = 50, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 1, | ||||
| 	near_nodes_vertical = 25, | ||||
| 	near_nodes_count = 1, | ||||
| 	humidity_min = -0.5, | ||||
| 	humidity_max = -1, | ||||
| 	rarity = 0, | ||||
| 	max_count = 30, | ||||
| } | ||||
|  | ||||
| moretrees.poplar_small_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 4, | ||||
| 	seed_diff = 343, | ||||
| 	min_elevation = 0, | ||||
| 	max_elevation = 50, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 10, | ||||
| 	near_nodes_vertical = 5, | ||||
| 	near_nodes_count = 1, | ||||
| 	humidity_min = -0.7, | ||||
| 	humidity_max = -1, | ||||
| 	rarity = 50, | ||||
| 	max_count = 10, | ||||
| } | ||||
|  | ||||
| moretrees.poplar_small_biome_2 = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| 	avoid_radius = 4, | ||||
| 	seed_diff = 343, | ||||
| 	min_elevation = 0, | ||||
| 	max_elevation = 50, | ||||
| 	near_nodes = {"default:water_source"}, | ||||
| 	near_nodes_size = 10, | ||||
| 	near_nodes_vertical = 4, | ||||
| 	near_nodes_count = 5, | ||||
| 	humidity_min = 0.1, | ||||
| 	humidity_max = -0.6, | ||||
| 	rarity = 50, | ||||
| 	max_count = 3, | ||||
| } | ||||
|  | ||||
|  | ||||
| moretrees.fir_biome = { | ||||
| 	surface = "default:dirt_with_grass", | ||||
| 	avoid_nodes = moretrees.avoidnodes, | ||||
| @@ -299,3 +180,4 @@ moretrees.fir_biome_snow = { | ||||
| 	delete_above = true, | ||||
| 	spawn_replace_node = true | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										283
									
								
								cocos_palm.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,283 +0,0 @@ | ||||
| local S = moretrees.intllib | ||||
|  | ||||
| -- © 2016, Rogier <rogier777@gmail.com> | ||||
|  | ||||
| -- Some constants | ||||
|  | ||||
| local coconut_drop_ichance = 8 | ||||
|  | ||||
| -- Make the cocos palm fruit trunk a real trunk (it is generated as a fruit) | ||||
| local trunk = minetest.registered_nodes["moretrees:palm_trunk"] | ||||
| local ftrunk = {} | ||||
| local gftrunk = {} | ||||
| for k,v in pairs(trunk) do | ||||
| 	ftrunk[k] = v | ||||
| 	gftrunk[k] = v | ||||
| end | ||||
| ftrunk.tiles = {} | ||||
| gftrunk.tiles = {} | ||||
| for k,v in pairs(trunk.tiles) do | ||||
| 	ftrunk.tiles[k] = v | ||||
| 	gftrunk.tiles[k] = v | ||||
| end | ||||
| ftrunk.drop = "moretrees:palm_trunk" | ||||
| gftrunk.drop = "moretrees:palm_trunk" | ||||
| ftrunk.after_destruct = function(pos, oldnode) | ||||
| 	local coconuts = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, {"group:moretrees_coconut"}) | ||||
| 	for _,coconutpos in pairs(coconuts) do | ||||
| 		-- minetest.dig_node(coconutpos) does not cause nearby coconuts to be dropped :-( ... | ||||
| 		--minetest.dig_node(coconutpos) | ||||
| 		local items = minetest.get_node_drops(minetest.get_node(coconutpos).name) | ||||
| 		minetest.swap_node(coconutpos, biome_lib.air) | ||||
| 		for _, itemname in pairs(items) do | ||||
| 			minetest.add_item(coconutpos, itemname) | ||||
| 		end | ||||
| 	end | ||||
| end | ||||
| -- Make the different trunk types distinguishable (but barely) | ||||
| ftrunk.tiles[1] = "moretrees_palm_trunk_top.png^[transformR90" | ||||
| gftrunk.tiles[1] = "moretrees_palm_trunk_top.png^[transformR180" | ||||
| gftrunk.description = gftrunk.description.." (gen)" | ||||
| minetest.register_node("moretrees:palm_fruit_trunk", ftrunk) | ||||
| minetest.register_node("moretrees:palm_fruit_trunk_gen", gftrunk) | ||||
|  | ||||
| local coconut_regrow_abm_spec = { | ||||
| 	nodenames = { "moretrees:palm_fruit_trunk" }, | ||||
| 	interval = moretrees.coconut_flower_interval, | ||||
| 	chance = moretrees.coconut_flower_chance, | ||||
| 	action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 		local coconuts = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, "group:moretrees_coconut") | ||||
| 		-- Expected growth interval increases exponentially with number of coconuts already hanging. | ||||
| 		-- Also: if more coconuts are hanging, the chance of picking an empty spot decreases as well... | ||||
| 		if math.random(2^#coconuts) <= 2 then | ||||
| 			-- Grow in area of 3x3 round trunk | ||||
| 			local dx=math.floor(math.random(3)-2) | ||||
| 			local dz=math.floor(math.random(3)-2) | ||||
| 			local coconutpos = {x=pos.x+dx, y=pos.y, z=pos.z+dz} | ||||
| 			local coconutnode = minetest.get_node(coconutpos) | ||||
| 			if coconutnode.name == "air" then | ||||
| 				minetest.swap_node(coconutpos, {name="moretrees:coconut_0"}) | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
| } | ||||
| if moretrees.coconuts_regrow then | ||||
| 	minetest.register_abm(coconut_regrow_abm_spec) | ||||
| end | ||||
|  | ||||
| -- Spawn initial coconuts | ||||
|  | ||||
| -- Spawn initial coconuts | ||||
| -- (Instead of coconuts, a generated-palm fruit trunk is generated with the tree. This | ||||
| --  ABM converts the trunk to a regular fruit trunk, and spawns some coconuts) | ||||
| minetest.register_abm({ | ||||
| 	nodenames = { "moretrees:palm_fruit_trunk_gen" }, | ||||
| 	interval = 1, | ||||
| 	chance = 1, | ||||
| 	action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 		minetest.swap_node(pos, {name="moretrees:palm_fruit_trunk"}) | ||||
| 		local poslist = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, "air") | ||||
| 		local genlist = {} | ||||
| 		for k,v in pairs(poslist) do | ||||
| 			genlist[k] = {x = math.random(100), pos = v} | ||||
| 		end | ||||
| 		table.sort(genlist, function(a, b) return a.x < b.x; end) | ||||
| 		local gen | ||||
| 		local count = 0 | ||||
| 		for _,gen in pairs(genlist) do | ||||
| 			minetest.swap_node(gen.pos, {name = "moretrees:coconut_3"}) | ||||
| 			count = count + 1 | ||||
| 			if count == 4 then | ||||
| 				break | ||||
| 			end | ||||
| 		end | ||||
| 	end, | ||||
| }) | ||||
|  | ||||
| -- Register coconuts, and make them regrow | ||||
|  | ||||
| local coconut_growfn = function(pos, elapsed) | ||||
| 	local node = minetest.get_node(pos) | ||||
| 	local delay = moretrees.coconut_grow_interval | ||||
| 	if not node then | ||||
| 		return | ||||
| 	elseif not moretrees.coconuts_regrow then | ||||
| 		-- Regrowing has been turned off. Make coconust grow instantly | ||||
| 		minetest.swap_node(pos, {name="moretrees:coconut_3"}) | ||||
| 		return | ||||
| 	elseif node.name == "moretrees:coconut_3" then | ||||
| 		-- Drop coconuts (i.e. remove them), so that new coconuts can grow. | ||||
| 		-- Coconuts will drop as items with a small chance | ||||
| 		if math.random(coconut_drop_ichance) == 1 then | ||||
| 			if moretrees.coconut_item_drop_ichance > 0 and math.random(moretrees.coconut_item_drop_ichance) == 1 then | ||||
| 				local items = minetest.get_node_drops(minetest.get_node(pos).name) | ||||
| 				for _, itemname in pairs(items) do | ||||
| 					minetest.add_item(pos, itemname) | ||||
| 				end | ||||
| 			end | ||||
| 			minetest.swap_node(pos, biome_lib.air) | ||||
| 		end | ||||
| 	else | ||||
| 		-- Grow coconuts to the next stage | ||||
| 		local offset = string.len("moretrees:coconut_x") | ||||
| 		local n = string.sub(node.name, offset) | ||||
| 		minetest.swap_node(pos, {name=string.sub(node.name, 1, offset-1)..n+1}) | ||||
| 	end | ||||
| 	-- Don't catch up when elapsed time is large. Regular visits are needed for growth... | ||||
| 	local timer = minetest.get_node_timer(pos) | ||||
| 	timer:start(delay + math.random(moretrees.coconut_grow_interval)) | ||||
| end | ||||
|  | ||||
| local coconut_starttimer = function(pos, elapsed) | ||||
| 	local timer = minetest.get_node_timer(pos) | ||||
| 	local base_interval = moretrees.coconut_grow_interval * 2 / 3 | ||||
| 	timer:set(base_interval + math.random(base_interval), elapsed or 0) | ||||
| end | ||||
|  | ||||
| for _,suffix in ipairs({"_0", "_1", "_2", "_3", ""}) do | ||||
| 	local name | ||||
| 	if suffix == "_0" then | ||||
| 		name = S("Coconut Flower") | ||||
| 	else | ||||
| 		name = S("Coconut") | ||||
| 	end | ||||
| 	local drop = "" | ||||
| 	local coco_group = 1 | ||||
| 	local tile = "moretrees_coconut"..suffix..".png" | ||||
| 	local timerfn = coconut_growfn | ||||
| 	local constructfn = coconut_starttimer | ||||
| 	if suffix == "_3" then | ||||
| 		drop = "moretrees:coconut" | ||||
| 		tile = "moretrees_coconut.png" | ||||
| 	elseif suffix == "" then | ||||
| 		drop = nil | ||||
| 		coco_group = nil | ||||
| 		timerfn = nil | ||||
| 		constructfn = nil | ||||
| 	end | ||||
| 	local coconutdef = { | ||||
| 		description = name, | ||||
| 		tiles = {tile}, | ||||
| 		drawtype = "plantlike", | ||||
| 		paramtype = "light", | ||||
| 		sunlight_propagates = true, | ||||
| 		walkable = false, | ||||
| 		groups = { fleshy=3, dig_immediate=3, flammable=2, moretrees_coconut=coco_group }, | ||||
| 		inventory_image = tile.."^[transformR180", | ||||
| 		wield_image = tile.."^[transformR180", | ||||
| 		sounds = default.node_sound_defaults(), | ||||
| 		drop = drop, | ||||
| 		selection_box = { | ||||
| 			type = "fixed", | ||||
| 			fixed = {-0.3, -0.3, -0.3, 0.3, 0.3, 0.3} | ||||
| 		}, | ||||
| 		on_timer = timerfn, | ||||
| 		on_construct = constructfn, | ||||
|  | ||||
| 	} | ||||
| 	minetest.register_node("moretrees:coconut"..suffix, coconutdef) | ||||
| end | ||||
|  | ||||
| -- convert exisiting cocos palms. This is a bit tricky... | ||||
| -- Try to make sure that this is indeed a generated tree, and not manually-placed trunks and/or coconuts | ||||
| if moretrees.coconuts_convert_existing_palms then | ||||
| 	local spec = { | ||||
| 		name = "moretrees:convert_existing_cocos_palms_to_regrow_coconuts", | ||||
| 		nodenames = "moretrees:coconut", | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			local trunks | ||||
| 			local cvtrunks | ||||
| 			local leaves | ||||
| 			local coconuts | ||||
| 			-- One regular trunk must be adjacent to  the coconut | ||||
| 			trunks = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, "moretrees:palm_trunk") | ||||
| 			if #trunks ~= 1 then | ||||
| 				return | ||||
| 			end | ||||
| 			local tpos = trunks[1] | ||||
| 			-- 1 or 2 other trunks must be one level below to the trunk being converted. | ||||
| 			trunks = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y-1, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y-1, z=tpos.z+1}, "moretrees:palm_trunk") | ||||
| 			if #trunks < 1 or #trunks > 2 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- 1 or 2 other trunks must be two levels below to the trunk being converted. | ||||
| 			trunks = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y-2, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y-2, z=tpos.z+1}, "moretrees:palm_trunk") | ||||
| 			if #trunks < 1 or #trunks > 2 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- 1 or 2 trunks must at the level of the trunk being converted. | ||||
| 			cvtrunks = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y, z=tpos.z+1}, "moretrees:palm_trunk") | ||||
| 			if #cvtrunks < 1 or #cvtrunks > 2 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- No trunks may be one level above the trunk being converted. | ||||
| 			trunks = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y+1, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y+1, z=tpos.z+1}, "moretrees:palm_trunk") | ||||
| 			if #trunks ~= 0 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- Leaves must be one level above the trunk being converted. | ||||
| 			leaves = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y+1, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y+1, z=tpos.z+1}, "moretrees:palm_leaves") | ||||
| 			if #leaves == 0 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- Leaves must be two levels above the trunk being converted. | ||||
| 			leaves = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y+2, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y+2, z=tpos.z+1}, "moretrees:palm_leaves") | ||||
| 			if #leaves == 0 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- No cocos fruit trunk may already be adjacent to the coconut | ||||
| 			trunks = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, "moretrees:palm_fruit_trunk") | ||||
| 			if #trunks ~= 0 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- No cocos fruit trunk may be adjacent to or below the trunk being converted. | ||||
| 			trunks = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y-2, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y, z=tpos.z+1}, "moretrees:palm_fruit_trunk") | ||||
| 			if #trunks ~= 0 then | ||||
| 				return | ||||
| 			end | ||||
| 			-- Convert trunk and all coconuts nearby. Maybe convert 2 trunks, just in case... | ||||
| 			for _, tpos in pairs(cvtrunks) do | ||||
| 				minetest.swap_node(tpos, {name = "moretrees:palm_fruit_trunk"}) | ||||
| 				coconuts = minetest.find_nodes_in_area({x=tpos.x-1, y=tpos.y, z=tpos.z-1}, {x=tpos.x+1, y=tpos.y, z=tpos.z+1}, "moretrees:coconut") | ||||
| 				for _, coconutpos in pairs(coconuts) do | ||||
| 					minetest.swap_node(coconutpos, {name = "moretrees:coconut_3"}) | ||||
| 				end | ||||
| 			end | ||||
| 		end, | ||||
| 	} | ||||
| 	if minetest.register_lbm then | ||||
| 		minetest.register_lbm(spec) | ||||
| 	else | ||||
| 		spec.interval = 3691 | ||||
| 		spec.chance = 10 | ||||
| 		minetest.register_abm(spec) | ||||
| 	end | ||||
| end | ||||
|  | ||||
| -- If regrowing was previously disabled, but is enabled now, make sure timers are started for existing coconuts | ||||
| if moretrees.coconuts_regrow then | ||||
| 	local spec = { | ||||
| 		name = "moretrees:restart_coconut_regrow_timer", | ||||
| 		nodenames = "group:moretrees_coconut", | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			local timer = minetest.get_node_timer(pos) | ||||
| 			if not timer:is_started() then | ||||
| 				coconut_starttimer(pos) | ||||
| 			else | ||||
| 				local timeout = timer:get_timeout() | ||||
| 				local elapsed = timer:get_elapsed() | ||||
| 				if timeout - elapsed > moretrees.coconut_grow_interval * 4/3 then | ||||
| 					coconut_starttimer(pos, math.random(moretrees.coconut_grow_interval * 4/3)) | ||||
| 				end | ||||
| 			end | ||||
| 		end, | ||||
| 	} | ||||
| 	if minetest.register_lbm then | ||||
| 		minetest.register_lbm(spec) | ||||
| 	else | ||||
| 		spec.interval = 3659 | ||||
| 		spec.chance = 10 | ||||
| 		minetest.register_abm(spec) | ||||
| 	end | ||||
| end | ||||
|  | ||||
							
								
								
									
										109
									
								
								crafts.lua
									
									
									
									
									
								
							
							
						
						| @@ -3,6 +3,14 @@ local S = moretrees.intllib | ||||
| for i in ipairs(moretrees.treelist) do | ||||
| 	local treename = moretrees.treelist[i][1] | ||||
|  | ||||
| 	minetest.register_craft({ | ||||
| 		output = "moretrees:"..treename.."_trunk 2", | ||||
| 		recipe = { | ||||
| 			{"moretrees:"..treename.."_trunk_sideways"}, | ||||
| 			{"moretrees:"..treename.."_trunk_sideways"} | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	minetest.register_craft({ | ||||
| 		type = "shapeless", | ||||
| 		output = "moretrees:"..treename.."_planks 4", | ||||
| @@ -11,6 +19,14 @@ for i in ipairs(moretrees.treelist) do | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	minetest.register_craft({ | ||||
| 		type = "shapeless", | ||||
| 		output = "moretrees:"..treename.."_planks 4", | ||||
| 		recipe = { | ||||
| 			"moretrees:"..treename.."_trunk_sideways" | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	minetest.register_craft({ | ||||
| 		type = "fuel", | ||||
| 		recipe = "moretrees:"..treename.."_sapling", | ||||
| @@ -47,35 +63,6 @@ minetest.register_craftitem("moretrees:raw_coconut", { | ||||
| 	on_use = minetest.item_eat(4), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:date", { | ||||
| 	description = S("Date"), | ||||
| 	inventory_image = "moretrees_date.png", | ||||
| 	on_use = minetest.item_eat(1), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:date_nut_snack", { | ||||
| 	description = S("Date & nut snack"), | ||||
| 	inventory_image = "moretrees_date_nut_snack.png", | ||||
| 	on_use = minetest.item_eat(4), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:date_nut_batter", { | ||||
| 	description = S("Date-nut cake batter"), | ||||
| 	inventory_image = "moretrees_date_nut_batter.png", | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:date_nut_cake", { | ||||
| 	description = S("Date-nut cake"), | ||||
| 	inventory_image = "moretrees_date_nut_cake.png", | ||||
| 	on_use = minetest.item_eat(32), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:date_nut_bar", { | ||||
| 	description = S("Date-nut energy bar"), | ||||
| 	inventory_image = "moretrees_date_nut_bar.png", | ||||
| 	on_use = minetest.item_eat(4), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:acorn_muffin_batter", { | ||||
| 	description = S("Acorn Muffin batter"), | ||||
| 	inventory_image = "moretrees_acorn_muffin_batter.png", | ||||
| @@ -93,9 +80,9 @@ minetest.register_craftitem("moretrees:spruce_nuts", { | ||||
| 	on_use = minetest.item_eat(1), | ||||
| }) | ||||
|  | ||||
| minetest.register_craftitem("moretrees:cedar_nuts", { | ||||
| 	description = S("Roasted Cedar Cone Nuts"), | ||||
| 	inventory_image = "moretrees_cedar_nuts.png", | ||||
| minetest.register_craftitem("moretrees:pine_nuts", { | ||||
| 	description = S("Roasted Pine Cone Nuts"), | ||||
| 	inventory_image = "moretrees_pine_nuts.png", | ||||
| 	on_use = minetest.item_eat(1), | ||||
| }) | ||||
|  | ||||
| @@ -122,60 +109,6 @@ for i in ipairs(moretrees.cutting_tools) do | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| minetest.register_craft({ | ||||
| 	type = "shapeless", | ||||
| 	output = "moretrees:date_nut_snack", | ||||
| 	recipe = { | ||||
| 		"moretrees:date", | ||||
| 		"moretrees:date", | ||||
| 		"moretrees:date", | ||||
| 		"moretrees:spruce_nuts", | ||||
| 		"moretrees:cedar_nuts", | ||||
| 		"moretrees:fir_nuts", | ||||
| 	} | ||||
| }) | ||||
|  | ||||
| -- The date-nut cake is an exceptional food item due to its highly | ||||
| -- concentrated nature (32 food units). Because of that, it requires | ||||
| -- many different ingredients, and, starting from the base ingredients | ||||
| -- found or harvested in nature, it requires many steps to prepare. | ||||
| local flour | ||||
| if minetest.registered_nodes["farming:flour"] then | ||||
| 	flour = "farming:flour" | ||||
| else | ||||
| 	flour = "moretrees:acorn_muffin_batter" | ||||
| end | ||||
| minetest.register_craft({ | ||||
| 	type = "shapeless", | ||||
| 	output = "moretrees:date_nut_batter", | ||||
| 	recipe = { | ||||
| 		"moretrees:date_nut_snack", | ||||
| 		"moretrees:date_nut_snack", | ||||
| 		"moretrees:date_nut_snack", | ||||
| 		"moretrees:coconut_milk", | ||||
| 		"moretrees:date_nut_snack", | ||||
| 		"moretrees:raw_coconut", | ||||
| 		"moretrees:coconut_milk", | ||||
| 		flour, | ||||
| 		"moretrees:raw_coconut", | ||||
| 	}, | ||||
| 	replacements = { | ||||
| 		{ "moretrees:coconut_milk", "vessels:drinking_glass 2" } | ||||
| 	} | ||||
| }) | ||||
|  | ||||
| minetest.register_craft({ | ||||
| 	type = "cooking", | ||||
| 	output = "moretrees:date_nut_cake", | ||||
| 	recipe = "moretrees:date_nut_batter", | ||||
| }) | ||||
|  | ||||
| minetest.register_craft({ | ||||
| 	type = "shapeless", | ||||
| 	output = "moretrees:date_nut_bar 8", | ||||
| 	recipe = {"moretrees:date_nut_cake"}, | ||||
| }) | ||||
|  | ||||
| minetest.register_craft({ | ||||
| 	type = "shapeless", | ||||
| 	output = "moretrees:acorn_muffin_batter", | ||||
| @@ -205,8 +138,8 @@ minetest.register_craft({ | ||||
|  | ||||
| minetest.register_craft({ | ||||
| 	type = "cooking", | ||||
| 	output = "moretrees:cedar_nuts 4", | ||||
| 	recipe = "moretrees:cedar_cone", | ||||
| 	output = "moretrees:pine_nuts 4", | ||||
| 	recipe = "moretrees:pine_cone", | ||||
| }) | ||||
|  | ||||
| minetest.register_craft({ | ||||
|   | ||||
							
								
								
									
										750
									
								
								date_palm.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,750 +0,0 @@ | ||||
| -- Date palms. | ||||
| -- | ||||
| -- Date palms grow in hot and dry desert, but they require water. This makes them | ||||
| -- a bit harder to find. If found in the middle of the desert, their presence | ||||
| -- indicates a water source below the surface. | ||||
| -- | ||||
| -- As an additional feature (which can be disabled), dates automatically regrow after | ||||
| -- harvesting (provided a male tree is sufficiently nearby). | ||||
| -- If regrowing is enabled, then ripe dates will not hang forever. Most will disappear | ||||
| -- (e.g. eaten by birds, ...), and a small fraction will drop as items. | ||||
|  | ||||
| -- © 2016, Rogier <rogier777@gmail.com> | ||||
|  | ||||
| local S = moretrees.intllib | ||||
|  | ||||
| -- Some constants | ||||
|  | ||||
| local dates_drop_ichance = 4 | ||||
| local stems_drop_ichance = 4 | ||||
| local flowers_wither_ichance = 3 | ||||
|  | ||||
| -- implementation | ||||
|  | ||||
| local dates_regrow_prob | ||||
| if moretrees.dates_regrow_unpollinated_percent <= 0 then | ||||
| 	dates_regrow_prob = 0 | ||||
| elseif moretrees.dates_regrow_unpollinated_percent >= 100 then | ||||
| 	dates_regrow_prob = 1 | ||||
| else | ||||
| 	dates_regrow_prob = 1 - math.pow(moretrees.dates_regrow_unpollinated_percent/100, 1/flowers_wither_ichance) | ||||
| end | ||||
|  | ||||
| -- Make the date palm fruit trunk a real trunk (it is generated as a fruit) | ||||
| local trunk = minetest.registered_nodes["moretrees:date_palm_trunk"] | ||||
| local ftrunk = {} | ||||
| local fftrunk = {} | ||||
| local mftrunk = {} | ||||
| for k,v in pairs(trunk) do | ||||
| 	ftrunk[k] = v | ||||
| end | ||||
| ftrunk.tiles = {} | ||||
| for k,v in pairs(trunk.tiles) do | ||||
| 	ftrunk.tiles[k] = v | ||||
| end | ||||
| ftrunk.drop = "moretrees:date_palm_trunk" | ||||
| ftrunk.after_destruct = function(pos, oldnode) | ||||
| 	local dates = minetest.find_nodes_in_area({x=pos.x-2, y=pos.y, z=pos.z-2}, {x=pos.x+2, y=pos.y, z=pos.z+2}, {"group:moretrees_dates"}) | ||||
| 	for _,datespos in pairs(dates) do | ||||
| 		-- minetest.dig_node(datespos) does not cause nearby dates to be dropped :-( ... | ||||
| 		local items = minetest.get_node_drops(minetest.get_node(datespos).name) | ||||
| 		minetest.swap_node(datespos, biome_lib.air) | ||||
| 		for _, itemname in pairs(items) do | ||||
| 			minetest.add_item(datespos, itemname) | ||||
| 		end | ||||
| 	end | ||||
| end | ||||
| for k,v in pairs(ftrunk) do | ||||
| 	mftrunk[k] = v | ||||
| 	fftrunk[k] = v | ||||
| end | ||||
| fftrunk.tiles = {} | ||||
| mftrunk.tiles = {} | ||||
| for k,v in pairs(trunk.tiles) do | ||||
| 	fftrunk.tiles[k] = v | ||||
| 	mftrunk.tiles[k] = v | ||||
| end | ||||
| -- Make the different types of trunk distinguishable (but not too easily) | ||||
| ftrunk.tiles[1] = "moretrees_date_palm_trunk_top.png^[transformR180" | ||||
| ftrunk.description = ftrunk.description.." (gen)" | ||||
| fftrunk.tiles[1] = "moretrees_date_palm_trunk_top.png^[transformR90" | ||||
| mftrunk.tiles[1] = "moretrees_date_palm_trunk_top.png^[transformR-90" | ||||
| minetest.register_node("moretrees:date_palm_fruit_trunk", ftrunk) | ||||
| minetest.register_node("moretrees:date_palm_ffruit_trunk", fftrunk) | ||||
| minetest.register_node("moretrees:date_palm_mfruit_trunk", mftrunk) | ||||
|  | ||||
| -- ABM to grow new date blossoms | ||||
| local date_regrow_abm_spec = { | ||||
| 	nodenames = { "moretrees:date_palm_ffruit_trunk", "moretrees:date_palm_mfruit_trunk" }, | ||||
| 	interval = moretrees.dates_flower_interval, | ||||
| 	chance = moretrees.dates_flower_chance, | ||||
| 	action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 		local dates = minetest.find_nodes_in_area({x=pos.x-2, y=pos.y, z=pos.z-2}, {x=pos.x+2, y=pos.y, z=pos.z+2}, "group:moretrees_dates") | ||||
|  | ||||
| 		-- New blossom interval increases exponentially with number of dates already hanging | ||||
| 		-- In addition: if more dates are hanging, the chance of picking an empty spot decreases as well... | ||||
| 		if math.random(2^#dates) <= 2 then | ||||
| 			-- Grow in area of 5x5 round trunk; higher probability in 3x3 area close to trunk | ||||
| 			local dx=math.floor((math.random(50)-18)/16) | ||||
| 			local dz=math.floor((math.random(50)-18)/16) | ||||
| 			local datepos = {x=pos.x+dx, y=pos.y, z=pos.z+dz} | ||||
| 			local datenode = minetest.get_node(datepos) | ||||
| 			if datenode.name == "air" then | ||||
| 				if node.name == "moretrees:date_palm_ffruit_trunk" then | ||||
| 					minetest.swap_node(datepos, {name="moretrees:dates_f0"}) | ||||
| 				else | ||||
| 					minetest.swap_node(datepos, {name="moretrees:dates_m0"}) | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
| } | ||||
| if moretrees.dates_regrow_pollinated or moretrees.dates_regrow_unpollinated_percent > 0 then | ||||
| 	minetest.register_abm(date_regrow_abm_spec) | ||||
| end | ||||
|  | ||||
| -- Choose male or female palm, and spawn initial dates | ||||
| -- (Instead of dates, a dates fruit trunk is generated with the tree. This | ||||
| --  ABM converts the trunk to a female or male fruit trunk, and spawns some | ||||
| --  hanging dates) | ||||
| minetest.register_abm({ | ||||
| 	nodenames = { "moretrees:date_palm_fruit_trunk" }, | ||||
| 	interval = 1, | ||||
| 	chance = 1, | ||||
| 	action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 		local type | ||||
| 		if math.random(100) <= moretrees.dates_female_percent then | ||||
| 			type = "f" | ||||
| 			minetest.swap_node(pos, {name="moretrees:date_palm_ffruit_trunk"}) | ||||
| 		else | ||||
| 			type = "m" | ||||
| 			minetest.swap_node(pos, {name="moretrees:date_palm_mfruit_trunk"}) | ||||
| 		end | ||||
| 		local dates1 = minetest.find_nodes_in_area({x=pos.x-1, y=pos.y, z=pos.z-1}, {x=pos.x+1, y=pos.y, z=pos.z+1}, "air") | ||||
| 		local genpos | ||||
| 		for _,genpos in pairs(dates1) do | ||||
| 			if math.random(100) <= 20 then | ||||
| 				if type == "m" then | ||||
| 					minetest.swap_node(genpos, {name = "moretrees:dates_n"}) | ||||
| 				else | ||||
| 					minetest.swap_node(genpos, {name = "moretrees:dates_f4"}) | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 		local dates2 = minetest.find_nodes_in_area({x=pos.x-2, y=pos.y, z=pos.z-2}, {x=pos.x+2, y=pos.y, z=pos.z+2}, "air") | ||||
| 		for _,genpos in pairs(dates2) do | ||||
| 			if math.random(100) <= 5 then | ||||
| 				if type == "m" then | ||||
| 					minetest.swap_node(genpos, {name = "moretrees:dates_n"}) | ||||
| 				else | ||||
| 					minetest.swap_node(genpos, {name = "moretrees:dates_f4"}) | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 	end, | ||||
| }) | ||||
|  | ||||
| -- Dates growing functions. | ||||
|  | ||||
| -- This is a bit complex, as the purpose is to find male flowers at horizontal distances of over | ||||
| -- 100 nodes. As searching such a large area is time consuming, this is optimized in four ways: | ||||
| -- - The search result (the locations of male trees) is cached, so that it can be used again | ||||
| -- - Only 1/9th of the desired area is searched at a time. A new search is only performed if no male | ||||
| --   flowers are found in the previously searched parts. | ||||
| -- - Search results are shared with other female palms nearby. | ||||
| -- - If previous searches for male palms have consumed too much CPU time, the search is skipped | ||||
| --   (This means no male palms will be found, and the pollination of the flowers affected will be | ||||
| --   delayed. If this happens repeatedly, eventually, the female flowers will wither...) | ||||
| -- A caching method was selected that is suited for the case where most date trees are long-lived, | ||||
| -- and where the number of trees nearby is limited: | ||||
| -- - Locations of male palms are stored as metadata for every female palm. This means that a player | ||||
| --   visiting a remote area with some date palms will not cause extensive searches for male palms as | ||||
| --   long overdue blossoming ABMs are triggered for every date palm. | ||||
| -- - Even when male palms *are* cut down, a cache refill will only be performed if the cached results do not | ||||
| --   contain a male palm with blossoms. | ||||
| -- The method will probably perform suboptimally: | ||||
| -- - If female palms are frequently chopped down and replanted. | ||||
| --   Freshly grown palms will need to search for male palms again | ||||
| --   (this is mitigated by the long blossoming interval, which increases the chance that search | ||||
| --    results have already been shared) | ||||
| -- - If an area contains a large number of male and female palms. | ||||
| --   In this area, every female palm will have an almost identical list of male palm locations | ||||
| --   as metadata. | ||||
| -- - If all male palms within range of a number of female palms have been chopped down (with possibly | ||||
| --   new ones planted). Although an attempt was made to share search results in this case as well, | ||||
| --   a number of similar searches will unavoidably be performed by the different female palms. | ||||
| -- - If no male palms are in range of a female palm. In that case, there will be frequent searches | ||||
| --   for newly-grown male palms. | ||||
|  | ||||
| -- Search statistics - used to limit the search load. | ||||
| local sect_search_stats = {}		-- Search statistics - server-wide | ||||
| local function reset_sect_search_stats() | ||||
| 	sect_search_stats.count = 0		-- # of searches | ||||
| 	sect_search_stats.skip = 0		-- # of times skipped | ||||
| 	sect_search_stats.sum = 0		-- total time spent | ||||
| 	sect_search_stats.min = 999999999	-- min time spent | ||||
| 	sect_search_stats.max = 0		-- max time spent | ||||
| end | ||||
| reset_sect_search_stats() | ||||
| sect_search_stats.last_us = 0			-- last time a search was done (microseconds, max: 2^32) | ||||
| sect_search_stats.last_s = 0			-- last time a search was done (system time in seconds) | ||||
|  | ||||
| -- Find male trunks in one section (=1/9 th) of the searchable area. | ||||
| -- sect is -4 to 4, where 0 is the center section | ||||
| local function find_fruit_trunks_near(ftpos, sect) | ||||
| 	local r = moretrees.dates_pollination_distance + 2 * math.sqrt(2) | ||||
| 	local sect_hr = math.floor(r / 3 + 0.9999) | ||||
| 	local sect_vr = math.floor(r / 2 + 0.9999) | ||||
| 	local t0us = core.get_us_time() | ||||
| 	local t0s = os.time() | ||||
|  | ||||
| 	-- Compute elapsed time since last search. | ||||
| 	-- Unfortunately, the time value wraps after about 71 minutes (2^32 microseconds), | ||||
| 	-- so it must be corrected to obtain the actual elapsed time. | ||||
| 	if t0us < sect_search_stats.last_us then | ||||
| 		-- Correct a simple wraparound. | ||||
| 		-- This is not sufficient, as the time value may have wrapped more than once... | ||||
| 		sect_search_stats.last_us = sect_search_stats.last_us - 2^32 | ||||
| 	end | ||||
| 	if t0s - sect_search_stats.last_s > 2^32/1000000 then | ||||
| 		-- One additional correction is enough for our purposes. | ||||
| 		-- For exact results, more corrections may be needed though... | ||||
| 		-- (and even not applying this correction at all would still only yield | ||||
| 		--  a minimal risk of a non-serious miscalculation...) | ||||
| 		sect_search_stats.last_us = sect_search_stats.last_us - 2^32 | ||||
| 	end | ||||
|  | ||||
| 	-- Skip the search if it is consuming too much CPU time | ||||
| 	if sect_search_stats.count > 0 and moretrees.dates_blossom_search_iload > 0 | ||||
| 			and sect_search_stats.sum / sect_search_stats.count > moretrees.dates_blossom_search_time_treshold | ||||
| 			and t0us - sect_search_stats.last_us < moretrees.dates_blossom_search_iload * (sect_search_stats.sum / sect_search_stats.count) then | ||||
| 		sect_search_stats.skip = sect_search_stats.skip + 1 | ||||
| 		return nil | ||||
| 	end | ||||
|  | ||||
| 	local basevec = { x = ftpos.x + 2 * sect.x * sect_hr, | ||||
| 			  y = ftpos.y, | ||||
| 			  z = ftpos.z + 2 * sect.z * sect_hr} | ||||
| 	-- find_nodes_in_area is limited to 82^3, make sure to not overrun it | ||||
| 	local sizevec = { x = sect_hr, y = sect_vr, z = sect_hr } | ||||
| 	if sect_hr * sect_hr * sect_vr > 41^3 then | ||||
| 		sizevec = vector.apply(sizevec, function(a) return math.min(a, 41) end) | ||||
| 	end | ||||
|  | ||||
| 	local all_palms = minetest.find_nodes_in_area( | ||||
| 				vector.subtract(basevec, sizevec), | ||||
| 				vector.add(basevec, sizevec), | ||||
| 				{"moretrees:date_palm_mfruit_trunk", "moretrees:date_palm_ffruit_trunk"}) | ||||
|  | ||||
| 	-- Collect different palms in separate lists. | ||||
| 	local female_palms = {} | ||||
| 	local male_palms = {} | ||||
| 	local all_male_palms = {} | ||||
| 	for _, pos in pairs(all_palms) do | ||||
| 		if pos.x ~= ftpos.x or pos.y ~= ftpos.y or pos.z ~= ftpos.z then | ||||
| 			local node = minetest.get_node(pos) | ||||
| 			if node and node.name == "moretrees:date_palm_ffruit_trunk" then | ||||
| 				table.insert(female_palms,pos) | ||||
| 			elseif node then | ||||
| 				table.insert(all_male_palms,pos) | ||||
| 				-- In sector 0, all palms are of interest. | ||||
| 				-- In other sectors, forget about palms that are too far away. | ||||
| 				if sect == 0 then | ||||
| 					table.insert(male_palms,pos) | ||||
| 				else | ||||
| 					local ssq = 0 | ||||
| 					for _, c in pairs({"x", "z"}) do | ||||
| 						local dc = pos[c] - ftpos[c] | ||||
| 						ssq = ssq + dc * dc | ||||
| 					end | ||||
| 					if math.sqrt(ssq) <= r then | ||||
| 						table.insert(male_palms,pos) | ||||
| 					end | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| 	-- Update search statistics | ||||
| 	local t1us = core.get_us_time() | ||||
| 	if t1us < t0us then | ||||
| 		-- Wraparound. Assume the search lasted less than 2^32 microseconds (~71 min) | ||||
| 		-- (so no need to apply another correction) | ||||
| 		t0us = t0us - 2^32 | ||||
| 	end | ||||
| 	sect_search_stats.last_us = t0us | ||||
| 	sect_search_stats.last_s = t0s | ||||
| 	sect_search_stats.count = sect_search_stats.count + 1 | ||||
| 	sect_search_stats.sum = sect_search_stats.sum + t1us-t0us | ||||
| 	if t1us - t0us < sect_search_stats.min then | ||||
| 		sect_search_stats.min = t1us - t0us | ||||
| 	end | ||||
| 	if t1us - t0us > sect_search_stats.max then | ||||
| 		sect_search_stats.max = t1us - t0us | ||||
| 	end | ||||
|  | ||||
| 	return male_palms, female_palms, all_male_palms | ||||
| end | ||||
|  | ||||
| local function dates_print_search_stats(log) | ||||
| 	local stats | ||||
| 	if sect_search_stats.count > 0 then | ||||
| 		stats = string.format("Male date tree searching stats: searches: %d/%d:  average: %d µs  (%d..%d)", | ||||
| 			sect_search_stats.count, sect_search_stats.count + sect_search_stats.skip, | ||||
| 			sect_search_stats.sum/sect_search_stats.count, sect_search_stats.min, sect_search_stats.max) | ||||
| 	else | ||||
| 		stats = string.format("Male date tree searching stats: searches: 0/0:  average: (no searches yet)") | ||||
| 	end | ||||
| 	if log then | ||||
| 		minetest.log("action", "[moretrees] " .. stats) | ||||
| 	end | ||||
| 	return true, stats | ||||
| end | ||||
|  | ||||
| minetest.register_chatcommand("dates_stats", { | ||||
| 	description = "Print male date palm search statistics", | ||||
| 	params = "|chat|log|reset", | ||||
| 	privs = { server = true }, | ||||
| 	func = function(name, param) | ||||
| 		param = string.lower(string.trim(param)) | ||||
| 		if param == "" or param == "chat" then | ||||
| 			return dates_print_search_stats(false) | ||||
| 		elseif param == "log" then | ||||
| 			return dates_print_search_stats(true) | ||||
| 		elseif param == "reset" then | ||||
| 			reset_sect_search_stats() | ||||
| 			return true | ||||
| 		else | ||||
| 			return false, "Invalid subcommand; expected: '' or 'chat' or 'log' or 'reset'" | ||||
| 		end | ||||
| 	end, | ||||
| }) | ||||
|  | ||||
| -- Find the female trunk near the female flowers to be pollinated | ||||
| local function find_female_trunk(fbpos) | ||||
| 	local trunks = minetest.find_nodes_in_area({x=fbpos.x-2, y=fbpos.y, z=fbpos.z-2}, | ||||
| 						{x=fbpos.x+2, y=fbpos.y, z=fbpos.z+2}, | ||||
| 						"moretrees:date_palm_ffruit_trunk") | ||||
| 	local ftpos | ||||
| 	local d = 99 | ||||
| 	for x, pos in pairs(trunks) do | ||||
| 		local ssq = 0 | ||||
| 		for _, c in pairs({"x", "z"}) do | ||||
| 			local dc = pos[c] - fbpos[c] | ||||
| 			ssq = ssq + dc * dc | ||||
| 		end | ||||
| 		if math.sqrt(ssq) < d then | ||||
| 			ftpos = pos | ||||
| 			d = math.sqrt(ssq) | ||||
| 		end | ||||
| 	end | ||||
| 	return ftpos | ||||
| end | ||||
|  | ||||
| -- Find male blossom near a male trunk, | ||||
| -- the male blossom must be in range of a specific female blossom as well | ||||
| local function find_male_blossom_near_trunk(fbpos, mtpos) | ||||
| 	local r = moretrees.dates_pollination_distance | ||||
| 	local blossoms = minetest.find_nodes_in_area({x=mtpos.x-2, y=mtpos.y, z=mtpos.z-2}, | ||||
| 						{x=mtpos.x+2, y=mtpos.y, z=mtpos.z+2}, | ||||
| 						"moretrees:dates_m0") | ||||
| 	for x, mbpos in pairs(blossoms) do | ||||
| 		local ssq = 0 | ||||
| 		for _, c in pairs({"x", "z"}) do | ||||
| 			local dc = mbpos[c] - fbpos[c] | ||||
| 			ssq = ssq + dc * dc | ||||
| 		end | ||||
| 		if math.sqrt(ssq) <= r then | ||||
| 			return mbpos | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| end | ||||
|  | ||||
| -- Find a male blossom in range of a specific female blossom, | ||||
| -- using a nested list of male blossom positions | ||||
| local function find_male_blossom_in_mpalms(ftpos, fbpos, mpalms) | ||||
| 	-- Process the elements of mpalms.sect (index -4 .. 4) in random order | ||||
| 	-- First, compute the order in which the sectors will be searched | ||||
| 	local sect_index = {} | ||||
| 	local sect_rnd = {} | ||||
| 	for i = -4,4 do | ||||
| 		local n = math.random(1023) | ||||
| 		sect_index[n] =  i | ||||
| 		table.insert(sect_rnd, n) | ||||
| 	end | ||||
| 	table.sort(sect_rnd) | ||||
|  | ||||
| 	-- Search the sectors | ||||
| 	local sect_old = 0 | ||||
| 	local sect_time = minetest.get_gametime() | ||||
| 	for _, n in pairs(sect_rnd) do | ||||
| 		-- Record the oldest sector, so that it can be searched if no male | ||||
| 		-- blossoms were found | ||||
| 		if not mpalms.sect_time[sect_index[n]] then | ||||
| 			sect_old = sect_index[n] | ||||
| 			sect_time = 0 | ||||
| 		elseif mpalms.sect_time[sect_index[n]] < sect_time then | ||||
| 			sect_old = sect_index[n] | ||||
| 			sect_time = mpalms.sect_time[sect_index[n]] | ||||
| 		end | ||||
| 		if mpalms.sect[sect_index[n]] and #mpalms.sect[sect_index[n]] then | ||||
| 			for px, mtpos in pairs(mpalms.sect[sect_index[n]]) do | ||||
| 				local node = minetest.get_node(mtpos) | ||||
| 				if node and node.name == "moretrees:date_palm_mfruit_trunk" then | ||||
| 					local mbpos = find_male_blossom_near_trunk(fbpos, mtpos) | ||||
| 					if mbpos then | ||||
| 						return mbpos | ||||
| 					end | ||||
| 				elseif node and node.name ~= "ignore" then | ||||
| 					-- no more male trunk here. | ||||
| 					mpalms.sect[sect_index[n]][px] = nil | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
| 	return nil, sect_old | ||||
| end | ||||
|  | ||||
| -- Find a male blossom in range of a specific female blossom, | ||||
| -- using the cache associated with the given female trunk | ||||
| -- If necessary, recompute part of the cache | ||||
| local last_search_result = {} | ||||
| local function find_male_blossom_with_ftrunk(fbpos,ftpos) | ||||
| 	local meta = minetest.get_meta(ftpos) | ||||
| 	local mpalms | ||||
| 	local cache_changed = true | ||||
|  | ||||
| 	-- Load cache. If distance has changed, start with empty cache instead. | ||||
| 	local mpalms_dist = meta:get_int("male_palms_dist") | ||||
| 	if mpalms_dist and mpalms_dist == moretrees.dates_pollination_distance then | ||||
| 		mpalms = meta:get_string("male_palms") | ||||
| 		if mpalms and mpalms ~= "" then | ||||
| 			mpalms = minetest.deserialize(mpalms) | ||||
| 			cache_changed = false | ||||
| 		end | ||||
| 	end | ||||
| 	if not mpalms or not mpalms.sect then | ||||
| 		mpalms = {} | ||||
| 		mpalms.sect = {} | ||||
| 		mpalms.sect_time = {} | ||||
| 		meta:set_int("male_palms_dist", moretrees.dates_pollination_distance) | ||||
| 		cache_changed = true | ||||
| 	end | ||||
| 	local fpalms_list | ||||
| 	local all_mpalms_list | ||||
| 	local sector0_searched = false | ||||
|  | ||||
| 	-- Always make sure that sector 0 is cached | ||||
| 	if not mpalms.sect[0] then | ||||
| 		mpalms.sect[0], fpalms_list, all_mpalms_list = find_fruit_trunks_near(ftpos, {x = 0, z = 0}) | ||||
| 		mpalms.sect_time[0] = minetest.get_gametime() | ||||
| 		sector0_searched = true | ||||
| 		cache_changed = true | ||||
| 		last_search_result.female = fpalms_list | ||||
| 		last_search_result.male = all_mpalms_list | ||||
| 	end | ||||
|  | ||||
| 	-- Find male palms | ||||
| 	local mbpos, sect_old = find_male_blossom_in_mpalms(ftpos, fbpos, mpalms) | ||||
|  | ||||
| 	-- If not found, (re)generate the cache for an additional sector. But don't search it yet (for performance reasons) | ||||
| 	-- (Use the globally cached results if possible) | ||||
| 	if not mbpos and not sector0_searched then | ||||
| 		if not mpalms.sect_time[0] or mpalms.sect_time[0] == 0 or math.random(3) == 1 then | ||||
| 			-- Higher probability of re-searching the center sector | ||||
| 			sect_old = 0 | ||||
| 		end | ||||
| 		-- Use globally cached result if possible | ||||
| 		mpalms.sect[sect_old] = nil | ||||
| 		if sect_old == 0 and mpalms.sect_time[0] and mpalms.sect_time[0] > 0 | ||||
| 				and last_search_result.male and #last_search_result.male then | ||||
| 			for _, pos in pairs(last_search_result.female) do | ||||
| 				if pos.x == ftpos.x and pos.y == ftpos.y and pos.z == ftpos.z then | ||||
| 					mpalms.sect[sect_old] = last_search_result.male | ||||
| 					-- Next time, don't use the cached result | ||||
| 					mpalms.sect_time[sect_old] = nil | ||||
| 					cache_changed = true | ||||
| 				end | ||||
| 			end | ||||
| 		end | ||||
| 		-- Else do a new search | ||||
| 		if not mpalms.sect[sect_old] then | ||||
| 			mpalms.sect[sect_old], fpalms_list, all_mpalms_list = find_fruit_trunks_near(ftpos, {x = (sect_old + 4) % 3 - 1, z = (sect_old + 4) / 3 - 1}) | ||||
| 			cache_changed = true | ||||
| 			if sect_old == 0 then | ||||
| 				-- Save the results if it is sector 0 | ||||
| 				-- (chance of reusing results from another sector are smaller) | ||||
| 				last_search_result.female = fpalms_list | ||||
| 				last_search_result.male = all_mpalms_list | ||||
| 			end | ||||
| 			if mpalms.sect[sect_old] then | ||||
| 				mpalms.sect_time[sect_old] = minetest.get_gametime() | ||||
| 			else | ||||
| 				mpalms.sect_time[sect_old] = nil | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| 	-- Share search results with other female trunks in the same area | ||||
| 	-- Note that the list of female trunks doesn't (shouldn't :-) contain the current female trunk. | ||||
| 	if fpalms_list and #fpalms_list and #all_mpalms_list then | ||||
| 		local all_mpalms = {} | ||||
| 		all_mpalms.sect = {} | ||||
| 		all_mpalms.sect_time = {} | ||||
| 		all_mpalms.sect[0] = all_mpalms_list | ||||
| 		-- Don't set sect_time[0], so that the cached sector will be re-searched soon (if necessary) | ||||
| 		local all_mpalms_serialized = minetest.serialize(all_mpalms) | ||||
| 		for _, pos in pairs(fpalms_list) do | ||||
| 			local fmeta = minetest.get_meta(pos) | ||||
| 			local fdist = fmeta:get_int("male_palms_dist") | ||||
| 			if not fdist or fdist ~= moretrees.dates_pollination_distance then | ||||
| 				fmeta:set_string("male_palms", all_mpalms_serialized) | ||||
| 				fmeta:set_int("male_palms_dist", moretrees.dates_pollination_distance) | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| 	-- Save cache. | ||||
| 	if cache_changed then | ||||
| 		meta:set_string("male_palms", minetest.serialize(mpalms)) | ||||
| 	end | ||||
|  | ||||
| 	return mbpos | ||||
| end | ||||
|  | ||||
| -- Find a male blossom in range of a specific female blossom | ||||
| local function find_male_blossom(fbpos) | ||||
| 	local ftpos = find_female_trunk(fbpos) | ||||
| 	if ftpos then | ||||
| 		return find_male_blossom_with_ftrunk(fbpos, ftpos) | ||||
| 	end | ||||
| 	return nil | ||||
| end | ||||
|  | ||||
| -- Growing function for dates | ||||
| local dates_growfn = function(pos, elapsed) | ||||
| 	local node = minetest.get_node(pos) | ||||
| 	local delay = moretrees.dates_grow_interval | ||||
| 	local r = moretrees.dates_pollination_distance | ||||
| 	local action | ||||
| 	if not node then | ||||
| 		return | ||||
| 	elseif not moretrees.dates_regrow_pollinated and dates_regrow_prob == 0 then | ||||
| 		-- Regrowing of dates is disabled. | ||||
| 		if string.find(node.name, "moretrees:dates_f") then | ||||
| 			minetest.swap_node(pos, {name="moretrees:dates_f4"}) | ||||
| 		elseif string.find(node.name, "moretrees:dates_m") then | ||||
| 			minetest.swap_node(pos, {name="moretrees:dates_n"}) | ||||
| 		else | ||||
| 			minetest.swap_node(pos, biome_lib.air) | ||||
| 		end | ||||
| 		return | ||||
| 	elseif node.name == "moretrees:dates_f0" and math.random(100) <= 100 * dates_regrow_prob then | ||||
| 		-- Dates grow unpollinated | ||||
| 		minetest.swap_node(pos, {name="moretrees:dates_f1"}) | ||||
| 		action = "nopollinate" | ||||
| 	elseif node.name == "moretrees:dates_f0" and moretrees.dates_regrow_pollinated and find_male_blossom(pos) then | ||||
| 		-- Pollinate flowers | ||||
| 		minetest.swap_node(pos, {name="moretrees:dates_f1"}) | ||||
| 		action = "pollinate" | ||||
| 	elseif string.match(node.name, "0$") then | ||||
| 		-- Make female unpollinated and male flowers last a bit longer | ||||
| 		if math.random(flowers_wither_ichance) == 1 then | ||||
| 			if node.name == "moretrees:dates_f0" then | ||||
| 				minetest.swap_node(pos, {name="moretrees:dates_fn"}) | ||||
| 			else | ||||
| 				minetest.swap_node(pos, {name="moretrees:dates_n"}) | ||||
| 			end | ||||
| 			action = "wither" | ||||
| 		else | ||||
| 			action = "nowither" | ||||
| 		end | ||||
| 	elseif node.name == "moretrees:dates_f4" then | ||||
| 		-- Remove dates, and optionally drop them as items | ||||
| 		if math.random(dates_drop_ichance) == 1 then | ||||
| 			if moretrees.dates_item_drop_ichance > 0 and math.random(moretrees.dates_item_drop_ichance) == 1 then | ||||
| 				local items = minetest.get_node_drops(minetest.get_node(pos).name) | ||||
| 				for _, itemname in pairs(items) do | ||||
| 					minetest.add_item(pos, itemname) | ||||
| 				end | ||||
| 			end | ||||
| 			minetest.swap_node(pos, {name="moretrees:dates_n"}) | ||||
| 			action = "drop" | ||||
| 		else | ||||
| 			action = "nodrop" | ||||
| 		end | ||||
| 	elseif string.match(node.name, "n$") then | ||||
| 		-- Remove stems. | ||||
| 		if math.random(stems_drop_ichance) == 1 then | ||||
| 			minetest.swap_node(pos, biome_lib.air) | ||||
| 			return "stemdrop" | ||||
| 		end | ||||
| 		action = "nostemdrop" | ||||
| 	else | ||||
| 		-- Grow dates | ||||
| 		local offset = 18 | ||||
| 		local n = string.sub(node.name, offset) | ||||
| 		minetest.swap_node(pos, {name=string.sub(node.name, 1, offset-1)..n+1}) | ||||
| 		action = "grow" | ||||
| 	end | ||||
| 	-- Don't catch up when elapsed time is large. Regular visits are needed for growth... | ||||
| 	local timer = minetest.get_node_timer(pos) | ||||
| 	timer:start(delay + math.random(moretrees.dates_grow_interval)) | ||||
| 	return action | ||||
| end | ||||
|  | ||||
| -- Alternate growth function for dates. | ||||
| -- It calls the primary growth function, but also measures CPU time consumed. | ||||
| -- Use this function to analyze date growing performance. | ||||
| local stat = {} | ||||
| stat.count = 0 | ||||
| local dates_growfn_profiling = function(pos, elapsed) | ||||
| 	local t0 = core.get_us_time() | ||||
| 	local action = dates_growfn(pos, elapsed) | ||||
| 	local t1 = core.get_us_time() | ||||
| 	if t1 < t0 then | ||||
| 		t1 = t1 + 2^32 | ||||
| 	end | ||||
| 	stat.count = stat.count + 1 | ||||
| 	if not stat[action] then | ||||
| 		stat[action] = {} | ||||
| 		stat[action].count = 0 | ||||
| 		stat[action].sum = 0 | ||||
| 		stat[action].min = 9999999999 | ||||
| 		stat[action].max = 0 | ||||
| 	end | ||||
| 	stat[action].count = stat[action].count + 1 | ||||
| 	stat[action].sum = stat[action].sum + t1-t0 | ||||
| 	if t1-t0 < stat[action].min then | ||||
| 		stat[action].min = t1-t0 | ||||
| 	end | ||||
| 	if t1-t0 > stat[action].max then | ||||
| 		stat[action].max = t1-t0 | ||||
| 	end | ||||
|  | ||||
| 	if stat.count % 10 == 0 then | ||||
| 		io.write(".") | ||||
| 		io.flush() | ||||
| 	end | ||||
| 	if stat.count % 100 == 0 then | ||||
| 		print(string.format("Date grow statistics %5d:", stat.count)) | ||||
| 		local sum = 0 | ||||
| 		local count = 0 | ||||
| 		if sect_search_stats.count > 0 and stat.pollinate and stat.pollinate.count > 0 then | ||||
| 			print(string.format("\t%-12s: %6d (%4.1f%%): %6dus (%d..%d)", | ||||
| 				"search", sect_search_stats.count, | ||||
| 				100*sect_search_stats.count/stat.pollinate.count, | ||||
| 				sect_search_stats.sum/sect_search_stats.count, | ||||
| 				sect_search_stats.min, sect_search_stats.max)) | ||||
| 		else | ||||
| 			print(string.format("\t%-12s: %6d (%4.1f%%): %6dus (%d..%d)", | ||||
| 				"search", sect_search_stats.count, | ||||
| 				0, 0, 0, 0)) | ||||
| 		end | ||||
| 		for action,data in pairs(stat) do | ||||
| 			if action ~= "count" then | ||||
| 				count = count + data.count | ||||
| 				sum = sum + data.sum | ||||
| 				print(string.format("\t%-12s: %6d (%4.1f%%): %6dus (%d..%d)", | ||||
| 					action, data.count, | ||||
| 					100*data.count/stat.count, data.sum/data.count, | ||||
| 					data.min, data.max)) | ||||
| 			end | ||||
| 		end | ||||
| 		print(string.format("\t%-12s: %6d ( 100%%): %6dus", | ||||
| 			"TOTAL", count, sum/count)) | ||||
| 	end | ||||
| end | ||||
|  | ||||
| -- Register dates | ||||
|  | ||||
| local dates_starttimer = function(pos, elapsed) | ||||
| 	local timer = minetest.get_node_timer(pos) | ||||
| 	local base_interval = moretrees.dates_grow_interval * 2 / 3 | ||||
| 	timer:set(base_interval + math.random(base_interval), elapsed or 0) | ||||
| end | ||||
|  | ||||
| local dates_drop = { | ||||
| 	items = { | ||||
| 		{items = { "moretrees:date" }}, | ||||
| 		{items = { "moretrees:date" }}, | ||||
| 		{items = { "moretrees:date" }}, | ||||
| 		{items = { "moretrees:date" }}, | ||||
| 		{items = { "moretrees:date" }, rarity = 2 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 2 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 2 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 2 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 5 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 5 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 5 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 5 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 20 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 20 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 20 }, | ||||
| 		{items = { "moretrees:date" }, rarity = 20 }, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| for _,suffix in ipairs({"f0", "f1", "f2", "f3", "f4", "m0", "fn", "n"}) do | ||||
| 	local name | ||||
| 	if suffix == "f0" or suffix == "m0" then | ||||
| 		name = S("Date Flowers") | ||||
| 	elseif suffix == "n" or suffix == "fn" then | ||||
| 		name = S("Date Stem") | ||||
| 	else | ||||
| 		name = S("Dates") | ||||
| 	end | ||||
| 	local dropfn = suffix == "f4" and dates_drop or "" | ||||
| 	local datedef = { | ||||
| 		description = name, | ||||
| 		tiles = {"moretrees_dates_"..suffix..".png"}, | ||||
| 		visual_scale = 2, | ||||
| 		drawtype = "plantlike", | ||||
| 		paramtype = "light", | ||||
| 		sunlight_propagates = true, | ||||
| 		walkable = false, | ||||
| 		groups = { fleshy=3, dig_immediate=3, flammable=2, moretrees_dates=1 }, | ||||
| 		inventory_image = "moretrees_dates_"..suffix..".png^[transformR0", | ||||
| 		wield_image = "moretrees_dates_"..suffix..".png^[transformR90", | ||||
| 		sounds = default.node_sound_defaults(), | ||||
| 		drop = dropfn, | ||||
| 		selection_box = { | ||||
| 			type = "fixed", | ||||
| 			fixed = {-0.3, -0.3, -0.3, 0.3, 3.5, 0.3} | ||||
| 		}, | ||||
| 		on_timer = dates_growfn, | ||||
| 		on_construct = (moretrees.dates_regrow_pollinated or moretrees.dates_regrow_unpollinated_percent > 0) | ||||
| 				and dates_starttimer, | ||||
|  | ||||
| 	} | ||||
| 	minetest.register_node("moretrees:dates_"..suffix, datedef) | ||||
| end | ||||
|  | ||||
| -- If regrowing was previously disabled, but is enabled now, make sure timers are started for existing dates | ||||
| if moretrees.dates_regrow_pollinated or moretrees.dates_regrow_unpollinated_percent > 0 then | ||||
| 	local spec = { | ||||
| 		name = "moretrees:restart_dates_regrow_timer", | ||||
| 		nodenames = "group:moretrees_dates", | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			local timer = minetest.get_node_timer(pos) | ||||
| 			if not timer:is_started() then | ||||
| 				dates_starttimer(pos) | ||||
| 			else | ||||
| 				local timeout = timer:get_timeout() | ||||
| 				local elapsed = timer:get_elapsed() | ||||
| 				if timeout - elapsed > moretrees.dates_grow_interval * 4/3 then | ||||
| 					dates_starttimer(pos, math.random(moretrees.dates_grow_interval * 4/3)) | ||||
| 				end | ||||
| 			end | ||||
| 		end, | ||||
| 	} | ||||
| 	if minetest.register_lbm then | ||||
| 		minetest.register_lbm(spec) | ||||
| 	else | ||||
| 		spec.interval = 3557 | ||||
| 		spec.chance = 10 | ||||
| 		minetest.register_abm(spec) | ||||
| 	end | ||||
| end | ||||
|  | ||||
| @@ -6,29 +6,25 @@ moretrees.enable_apple_tree		= true | ||||
| moretrees.enable_oak			= true | ||||
| moretrees.enable_sequoia		= true | ||||
| moretrees.enable_palm			= true | ||||
| moretrees.enable_date_palm		= true | ||||
| moretrees.enable_cedar			= true | ||||
| moretrees.enable_pine			= true | ||||
| moretrees.enable_rubber_tree	= true | ||||
| moretrees.enable_willow			= true | ||||
| moretrees.enable_acacia			= true | ||||
| moretrees.enable_birch			= true | ||||
| moretrees.enable_spruce			= true | ||||
| moretrees.enable_jungle_tree	= true | ||||
| moretrees.enable_fir			= true | ||||
| moretrees.enable_poplar			= true | ||||
| moretrees.enable_beech			= false | ||||
|  | ||||
| -- set this to true to make moretrees spawn saplings at mapgen time instead | ||||
| -- of fully-grown trees, which will grow into full trees after a very short | ||||
| -- delay.  This reduces mapgen lag in some situations, and fixes situations | ||||
| -- where the mapgen conflicts with the tree generator. | ||||
| -- of fully-grown trees, which will grow into full trees very quickly.  With | ||||
| -- older versions of plants_lib, doing this will reduce mapgen lag. | ||||
|  | ||||
| moretrees.spawn_saplings		= true | ||||
| moretrees.spawn_saplings		= false | ||||
|  | ||||
| -- Set this to true to allow defining stairs/slabs/etc.  If Moreblocks is | ||||
| -- installed, this will use that mod's Stairs Plus component.  Otherwise, it  | ||||
| -- will use the default stairs mod in minetest_game, if present | ||||
| -- Set this to true to allow usage of the stairsplus mod in moreblocks | ||||
|  | ||||
| moretrees.enable_stairs			= true | ||||
| moretrees.enable_stairsplus		= true | ||||
|  | ||||
| -- Set this to true if you want the plantlike drawtype for leaves, which  | ||||
| -- improves some peoples' framerates without resorting to making leaf nodes opaque. | ||||
| @@ -36,21 +32,42 @@ moretrees.enable_stairs			= true | ||||
|  | ||||
| moretrees.plantlike_leaves		= false | ||||
|  | ||||
| -- Set this to true to enable leaf decay of all trees except the default ones. | ||||
|  | ||||
| moretrees.enable_leafdecay		= true | ||||
|  | ||||
| -- Enable this one if you want this mod's leafdecay code to affect the old | ||||
| -- default trees too; this setting is independent of the one above.  You'll | ||||
| -- want to manually disable the default leafdecay code in minetest_game if | ||||
| -- you enable this, otherwise you'll have two sets of leaf decay code running | ||||
| -- at the same time, which will just waste CPU for no benefit. | ||||
|  | ||||
| moretrees.enable_default_leafdecay	= true | ||||
|  | ||||
| -- Enable this one for default *jungle* leaves | ||||
|  | ||||
| moretrees.enable_default_jungle_leafdecay	= true | ||||
|  | ||||
| -- Enable this if you want moretrees to redefine default apples so that they | ||||
| -- fall when leaves decay/are dug. | ||||
|  | ||||
| moretrees.enable_redefine_apple		= true | ||||
|  | ||||
| -- Set this to true to enable leaf decay of all trees except the default ones. | ||||
|  | ||||
| moretrees.enable_leafdecay		= true | ||||
|  | ||||
| -- various related settings to configure leaf decay. | ||||
| -- various settings to configure default and default-jungle leaf decay. | ||||
|  | ||||
| moretrees.leafdecay_delay		= 2 | ||||
| moretrees.leafdecay_chance		= 5 | ||||
| moretrees.leafdecay_chance		= 100 | ||||
| moretrees.leafdecay_radius		= 5 | ||||
| moretrees.palm_leafdecay_radius		= 10 | ||||
|  | ||||
| moretrees.default_jungle_leafdecay_delay = 2 | ||||
| moretrees.default_jungle_leafdecay_chance = 100 | ||||
| moretrees.default_jungle_leafdecay_radius = 5 | ||||
|  | ||||
| moretrees.palm_leafdecay_radius		= 15 | ||||
|  | ||||
| moretrees.default_leafdecay_delay	= 3 | ||||
| moretrees.default_leafdecay_chance	= 50 | ||||
| moretrees.default_leafdecay_radius	= 4 | ||||
|  | ||||
| -- Change these settings if you want default trees to be gradually cut down | ||||
| -- above the elevation where firs normally generate. | ||||
| @@ -59,71 +76,10 @@ moretrees.firs_remove_default_trees	= false | ||||
| moretrees.firs_remove_interval		= 2 | ||||
| moretrees.firs_remove_chance		= 150 | ||||
|  | ||||
| -- Cocos palm settings | ||||
|  | ||||
| moretrees.coconuts_regrow		= true | ||||
| moretrees.coconuts_convert_existing_palms = true	-- Converting existing palm trees will make coconuts regrow on them as well | ||||
| 							-- Else, they will only regrow on newly-spawned palms | ||||
| 							-- However, conversion is not an exact science, and although an attempt is | ||||
| 							-- made to detect whether a trunk belongs to an actual palm, some coconut trunks | ||||
| 							-- and some coconuts may be incorrectly converted. | ||||
| moretrees.coconut_flower_interval	= 59 | ||||
| moretrees.coconut_flower_chance		= 67 | ||||
| moretrees.coconut_grow_interval		= 2 * moretrees.coconut_flower_interval * moretrees.coconut_flower_chance | ||||
| 							-- Actual interval will randomly vary between 67% and 133% of this value | ||||
| 							-- 2 * 59 * 67 ~ 2 hours. So flowers become coconuts in about 6 hours | ||||
| moretrees.coconut_item_drop_ichance	= 10 		-- inverse probability of ripe coconuts dropping as items (instead of disappearing) | ||||
|  | ||||
| -- Date palm settings | ||||
|  | ||||
| -- Suggested configuration alternatives: | ||||
| -- - Dates grow only when pollinated: | ||||
| --   - Set dates_regrow_pollinated to true | ||||
| --   - Set dates_regrow_unpollinated_percent to 0 | ||||
| -- - Dates grow without pollination. Pollination disabled: | ||||
| --   - Set dates_regrow_pollinated to false | ||||
| --   - Set dates_regrow_unpollinated_percent to some larger positive value, e.g. 95 | ||||
| -- - Dates grow, but more and faster if male flowers are nearby | ||||
| --   - Set dates_regrow_pollinated to true | ||||
| --   - Set dates_regrow_unpollinated_percent to some small positive value, e.g. 33 | ||||
| --   - Optional but recommended: Reduce the pollination distance, e.g. to 30 | ||||
|  | ||||
| -- Note that it should not be necessary to disable pollination for performance | ||||
| -- reasons. A lot of effort has gone into ensuring that date growing will not cause lag. | ||||
| -- | ||||
| -- If lag is suspected, use the chat command '/dates_stats' to obtain the male dates | ||||
| -- search time, as well as the counts of total number of searches requested and the | ||||
| -- number of searches actually performed. | ||||
|  | ||||
| moretrees.dates_regrow_pollinated	= true		-- Enable pollination. If enabled, male trees are required for dates to grow. | ||||
| 							-- If disabled, dates_regrow_unpollinated_percent must be non-zero for dates to regrow. | ||||
| moretrees.dates_regrow_unpollinated_percent	= 0	-- Percentage of female dates becoming dates without being pollinated. | ||||
| 							-- If 0, dates_regrow_pollinated must be enabled for dates to grow. | ||||
| moretrees.dates_female_percent		= 57		-- Ratio of female to male trees - tune this to improve # of generated trees that actually bear fruit | ||||
| 							-- ~57% gives near optimal results for groups of 3 random trees, while it is only slightly suboptimal | ||||
| 							-- for groups of 2 and 4 random trees (~2% less fruit than optimal). | ||||
| 							-- Optimal values per group size: 2: 50%, 3: 57.78%, 4: 63%, 5: 66.9%, 6: 69.9%, [...], 12: 79.8% | ||||
| 							-- So 57% is optimal for small groups of trees. As larger groups have more female palms anyway, a | ||||
| 							-- less than optimal proportion of female to male trees is not a problem. | ||||
| moretrees.dates_pollination_distance	= 120 | ||||
| moretrees.dates_blossom_search_time_treshold = 1000	-- If average male blossom search time (in microseconds) exceeds this, start limiting the search load. | ||||
| moretrees.dates_blossom_search_iload = 10		-- Inverse fraction of CPU time that male blossom searching search may consume. | ||||
| 							-- As searching a large area (radius: dates_pollination_distance/3 per attempt) can cause lag, | ||||
| 							-- this limits the search frequency server-wide so that the impact on server lag is minimised | ||||
| 							-- For personal servers, this can be set lower, or even to 1 or 0 (0 disables load limiting). | ||||
| 							-- Obtain the current average search time using /dates_stats | ||||
| moretrees.dates_flower_interval		= 59 | ||||
| moretrees.dates_flower_chance		= 181 | ||||
| moretrees.dates_grow_interval		= 2 * moretrees.dates_flower_interval * moretrees.dates_flower_chance | ||||
| 							-- As date palms have a high yield, don't grow dates too fast | ||||
| 							-- The actual interval will vary randomly between 67% and 133% of this value. | ||||
| 							-- 2 * 59 * 181 ~ 6 hours. So by default flowers become dates in about one (human) day. | ||||
| moretrees.dates_item_drop_ichance	= 10 		-- inverse probability of ripe dates dropping as items (instead of disappearing) | ||||
|  | ||||
| -- Sapling settings | ||||
|  | ||||
| moretrees.sapling_interval		= 100 | ||||
| moretrees.sapling_chance		= 5 | ||||
| moretrees.sapling_interval		= 500 | ||||
| moretrees.sapling_chance		= 20 | ||||
|  | ||||
| -- If this variable is set to true, drop leaves out as entities during leaf | ||||
| -- decay, rather than just disappearing them. | ||||
|   | ||||
| @@ -1,9 +1,5 @@ | ||||
| default | ||||
| biome_lib | ||||
| vessels | ||||
| doors? | ||||
| stairs? | ||||
| plants_lib | ||||
| moreblocks? | ||||
| intllib? | ||||
| farming? | ||||
|  | ||||
|   | ||||
| @@ -1 +0,0 @@ | ||||
| This mod adds a whole bunch of new types of trees to the game | ||||
							
								
								
									
										163
									
								
								init.lua
									
									
									
									
									
								
							
							
						
						| @@ -8,48 +8,16 @@ | ||||
| -- Brought together into one mod and made L-systems compatible by Vanessa | ||||
| -- Ezekowitz. | ||||
| -- | ||||
| -- Firs and Jungle tree axioms/rules by Vanessa Dannenberg, with the | ||||
| -- Firs and Jungle tree axioms/rules by Vanessa Ezekowitz, with the  | ||||
| -- latter having been tweaked by RealBadAngel, most other axioms/rules written | ||||
| -- by RealBadAngel. | ||||
| -- | ||||
| -- License: WTFPL for all parts (code and textures, including those copied | ||||
| -- from the jungletree and conifers mods) except the default jungle tree trunk | ||||
| -- texture, which is CC-By-SA. | ||||
|  | ||||
| moretrees = {} | ||||
|  | ||||
| minetest.override_item("default:sapling", { | ||||
| 	description = "Sapling" | ||||
| }) | ||||
|  | ||||
| minetest.override_item("default:tree", { | ||||
| 	description = "Tree" | ||||
| }) | ||||
|  | ||||
| minetest.override_item("default:wood", { | ||||
| 	description = "Wooden Planks" | ||||
| }) | ||||
|  | ||||
| minetest.override_item("default:leaves", { | ||||
| 	description = "Leaves" | ||||
| }) | ||||
|  | ||||
| minetest.override_item("default:fence_wood", { | ||||
| 	description = "Wooden Fence" | ||||
| }) | ||||
|  | ||||
| minetest.override_item("default:fence_rail_wood", { | ||||
| 	description = "Wooden Fence Rail" | ||||
| }) | ||||
|  | ||||
| if minetest.get_modpath("doors") then | ||||
| 	minetest.override_item("doors:gate_wood_closed", { | ||||
| 		description = "Wooden Fence Gate" | ||||
| 	}) | ||||
|  | ||||
| 	minetest.override_item("doors:gate_wood_open", { | ||||
| 		description = "Wooden Fence Gate" | ||||
| 	}) | ||||
| end | ||||
|  | ||||
|  | ||||
| -- Read the default config file (and if necessary, copy it to the world folder). | ||||
|  | ||||
| local worldpath=minetest.get_worldpath() | ||||
| @@ -57,8 +25,21 @@ local modpath=minetest.get_modpath("moretrees") | ||||
|  | ||||
| dofile(modpath.."/default_settings.txt") | ||||
|  | ||||
| if io.open(worldpath.."/moretrees_settings.txt","r") then | ||||
| if io.open(worldpath.."/moretrees_settings.txt","r") == nil then | ||||
|  | ||||
| 	io.input(modpath.."/default_settings.txt") | ||||
| 	io.output(worldpath.."/moretrees_settings.txt") | ||||
|  | ||||
| 	local size = 2^13      -- good buffer size (8K) | ||||
| 	while true do | ||||
| 		local block = io.read(size) | ||||
| 		if not block then | ||||
| 			io.close() | ||||
| 			break | ||||
| 		end | ||||
| 		io.write(block) | ||||
| 	end | ||||
| else | ||||
| 	dofile(worldpath.."/moretrees_settings.txt") | ||||
| end | ||||
|  | ||||
| @@ -73,8 +54,7 @@ moretrees.intllib = S | ||||
|  | ||||
| -- infinite stacks checking | ||||
|  | ||||
| if minetest.get_modpath("unified_inventory") or not | ||||
| 		minetest.settings:get_bool("creative_mode") then | ||||
| if minetest.get_modpath("unified_inventory") or not minetest.setting_getbool("creative_mode") then | ||||
| 	moretrees.expect_infinite_stacks = false | ||||
| else | ||||
| 	moretrees.expect_infinite_stacks = true | ||||
| @@ -96,11 +76,10 @@ moretrees.cutting_tools = { | ||||
|  | ||||
| dofile(modpath.."/tree_models.lua") | ||||
| dofile(modpath.."/node_defs.lua") | ||||
| dofile(modpath.."/date_palm.lua") | ||||
| dofile(modpath.."/cocos_palm.lua") | ||||
| dofile(modpath.."/biome_defs.lua") | ||||
| dofile(modpath.."/saplings.lua") | ||||
| dofile(modpath.."/crafts.lua") | ||||
| dofile(modpath.."/leafdecay.lua") | ||||
|  | ||||
| -- tree spawning setup | ||||
|  | ||||
| @@ -110,104 +89,92 @@ if moretrees.spawn_saplings then | ||||
| 	moretrees.spawn_oak_object = "moretrees:oak_sapling_ongen" | ||||
| 	moretrees.spawn_sequoia_object = "moretrees:sequoia_sapling_ongen" | ||||
| 	moretrees.spawn_palm_object = "moretrees:palm_sapling_ongen" | ||||
| 	moretrees.spawn_date_palm_object = "moretrees:date_palm_sapling_ongen" | ||||
| 	moretrees.spawn_cedar_object = "moretrees:cedar_sapling_ongen" | ||||
| 	moretrees.spawn_pine_object = "moretrees:pine_sapling_ongen" | ||||
| 	moretrees.spawn_rubber_tree_object = "moretrees:rubber_tree_sapling_ongen" | ||||
| 	moretrees.spawn_willow_object = "moretrees:willow_sapling_ongen" | ||||
| 	moretrees.spawn_acacia_object = "moretrees:acacia_sapling_ongen" | ||||
| 	moretrees.spawn_birch_object = "moretrees:birch_sapling_ongen" | ||||
| 	moretrees.spawn_spruce_object = "moretrees:spruce_sapling_ongen" | ||||
| 	moretrees.spawn_jungletree_object = "moretrees:jungletree_sapling_ongen" | ||||
| 	moretrees.spawn_fir_object = "moretrees:fir_sapling_ongen" | ||||
| 	moretrees.spawn_fir_snow_object = "snow:sapling_pine" | ||||
| 	moretrees.spawn_poplar_object = "moretrees:poplar_sapling_ongen" | ||||
| 	moretrees.spawn_poplar_small_object = "moretrees:poplar_small_sapling_ongen" | ||||
| else | ||||
| 	moretrees.spawn_beech_object = moretrees.beech_model | ||||
| 	moretrees.spawn_apple_tree_object = moretrees.apple_tree_model | ||||
| 	moretrees.spawn_oak_object = moretrees.oak_model | ||||
| 	moretrees.spawn_sequoia_object = moretrees.sequoia_model | ||||
| 	moretrees.spawn_palm_object = moretrees.palm_model | ||||
| 	moretrees.spawn_date_palm_object = moretrees.date_palm_model | ||||
| 	moretrees.spawn_cedar_object = moretrees.cedar_model | ||||
| 	moretrees.spawn_pine_object = moretrees.pine_model | ||||
| 	moretrees.spawn_rubber_tree_object = moretrees.rubber_tree_model | ||||
| 	moretrees.spawn_willow_object = moretrees.willow_model | ||||
| 	moretrees.spawn_birch_object = "moretrees.grow_birch" | ||||
| 	moretrees.spawn_spruce_object = "moretrees.grow_spruce" | ||||
| 	moretrees.spawn_jungletree_object = "moretrees.grow_jungletree" | ||||
| 	moretrees.spawn_fir_object = "moretrees.grow_fir" | ||||
| 	moretrees.spawn_fir_snow_object = "moretrees.grow_fir_snow" | ||||
| 	moretrees.spawn_poplar_object = moretrees.poplar_model | ||||
| 	moretrees.spawn_poplar_small_object = moretrees.poplar_small_model | ||||
| 	moretrees.spawn_acacia_object = moretrees.acacia_model | ||||
| 	moretrees.spawn_birch_object = "moretrees:grow_birch" | ||||
| 	moretrees.spawn_spruce_object = "moretrees:grow_spruce" | ||||
| 	moretrees.spawn_jungletree_object = "moretrees:grow_jungletree" | ||||
| 	moretrees.spawn_fir_object = "moretrees:grow_fir" | ||||
| 	moretrees.spawn_fir_snow_object = "moretrees:grow_fir_snow" | ||||
| end | ||||
|  | ||||
|  | ||||
| if moretrees.enable_beech then | ||||
| 	biome_lib:register_generate_plant(moretrees.beech_biome, moretrees.spawn_beech_object) | ||||
| 	plantslib:register_generate_plant(moretrees.beech_biome, moretrees.spawn_beech_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_apple_tree then | ||||
| 	biome_lib:register_generate_plant(moretrees.apple_tree_biome, moretrees.spawn_apple_tree_object) | ||||
| 	plantslib:register_generate_plant(moretrees.apple_tree_biome, moretrees.spawn_apple_tree_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_oak then | ||||
| 	biome_lib:register_generate_plant(moretrees.oak_biome, moretrees.spawn_oak_object) | ||||
| 	plantslib:register_generate_plant(moretrees.oak_biome, moretrees.spawn_oak_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_sequoia then | ||||
| 	biome_lib:register_generate_plant(moretrees.sequoia_biome, moretrees.spawn_sequoia_object) | ||||
| 	plantslib:register_generate_plant(moretrees.sequoia_biome, moretrees.spawn_sequoia_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_palm then | ||||
| 	biome_lib:register_generate_plant(moretrees.palm_biome, moretrees.spawn_palm_object) | ||||
| 	plantslib:register_generate_plant(moretrees.palm_biome, moretrees.spawn_palm_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_date_palm then | ||||
| 	biome_lib:register_generate_plant(moretrees.date_palm_biome, moretrees.spawn_date_palm_object) | ||||
| 	biome_lib:register_generate_plant(moretrees.date_palm_biome_2, moretrees.spawn_date_palm_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_cedar then | ||||
| 	biome_lib:register_generate_plant(moretrees.cedar_biome, moretrees.spawn_cedar_object) | ||||
| if moretrees.enable_pine then | ||||
| 	plantslib:register_generate_plant(moretrees.pine_biome, moretrees.spawn_pine_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_rubber_tree then | ||||
| 	biome_lib:register_generate_plant(moretrees.rubber_tree_biome, moretrees.spawn_rubber_tree_object) | ||||
| 	plantslib:register_generate_plant(moretrees.rubber_tree_biome, moretrees.spawn_rubber_tree_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_willow then | ||||
| 	biome_lib:register_generate_plant(moretrees.willow_biome, moretrees.spawn_willow_object) | ||||
| 	plantslib:register_generate_plant(moretrees.willow_biome, moretrees.spawn_willow_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_acacia then | ||||
| 	plantslib:register_generate_plant(moretrees.acacia_biome, moretrees.spawn_acacia_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_birch then | ||||
| 	biome_lib:register_generate_plant(moretrees.birch_biome, moretrees.spawn_birch_object) | ||||
| 	plantslib:register_generate_plant(moretrees.birch_biome, moretrees.spawn_birch_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_spruce then | ||||
| 	biome_lib:register_generate_plant(moretrees.spruce_biome, moretrees.spawn_spruce_object) | ||||
| 	plantslib:register_generate_plant(moretrees.spruce_biome, moretrees.spawn_spruce_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_jungle_tree then | ||||
| 	biome_lib:register_generate_plant(moretrees.jungletree_biome, moretrees.spawn_jungletree_object) | ||||
| 	plantslib:register_generate_plant(moretrees.jungletree_biome, moretrees.spawn_jungletree_object) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_fir then | ||||
| 	biome_lib:register_generate_plant(moretrees.fir_biome, moretrees.spawn_fir_object) | ||||
| 	plantslib:register_generate_plant(moretrees.fir_biome, moretrees.spawn_fir_object) | ||||
| 	if minetest.get_modpath("snow") then | ||||
| 		biome_lib:register_generate_plant(moretrees.fir_biome_snow, moretrees.spawn_fir_snow_object) | ||||
| 		plantslib:register_generate_plant(moretrees.fir_biome_snow, moretrees.spawn_fir_snow_object) | ||||
| 	end | ||||
| end | ||||
|  | ||||
| if moretrees.enable_poplar then | ||||
| 	biome_lib:register_generate_plant(moretrees.poplar_biome, moretrees.spawn_poplar_object) | ||||
| 	biome_lib:register_generate_plant(moretrees.poplar_biome_2, moretrees.spawn_poplar_object) | ||||
| 	biome_lib:register_generate_plant(moretrees.poplar_biome_3, moretrees.spawn_poplar_object) | ||||
| 	biome_lib:register_generate_plant(moretrees.poplar_small_biome, moretrees.spawn_poplar_small_object) | ||||
| 	biome_lib:register_generate_plant(moretrees.poplar_small_biome_2, moretrees.spawn_poplar_small_object) | ||||
| end | ||||
|  | ||||
| -- Code to spawn a birch tree | ||||
|  | ||||
| function moretrees.grow_birch(pos) | ||||
| 	minetest.swap_node(pos, biome_lib.air) | ||||
| function moretrees:grow_birch(pos) | ||||
| 	minetest.remove_node(pos) | ||||
| 	if math.random(1,2) == 1 then | ||||
| 		minetest.spawn_tree(pos, moretrees.birch_model1) | ||||
| 	else | ||||
| @@ -217,8 +184,8 @@ end | ||||
|  | ||||
| -- Code to spawn a spruce tree | ||||
|  | ||||
| function moretrees.grow_spruce(pos) | ||||
| 	minetest.swap_node(pos, biome_lib.air) | ||||
| function moretrees:grow_spruce(pos) | ||||
| 	minetest.remove_node(pos) | ||||
| 	if math.random(1,2) == 1 then | ||||
| 		minetest.spawn_tree(pos, moretrees.spruce_model1) | ||||
| 	else | ||||
| @@ -242,7 +209,7 @@ moretrees.ct_rules_b1 = "[-FBf][+FBf]" | ||||
| moretrees.ct_rules_a2 = "FF[FF][&&-FBF][&&+FBF][&&---FBF][&&+++FBF]F/A" | ||||
| moretrees.ct_rules_b2 = "[-fB][+fB]" | ||||
|  | ||||
| function moretrees.grow_jungletree(pos) | ||||
| function moretrees:grow_jungletree(pos) | ||||
| 	local r1 = math.random(2) | ||||
| 	local r2 = math.random(3) | ||||
| 	if r1 == 1 then | ||||
| @@ -272,17 +239,19 @@ function moretrees.grow_jungletree(pos) | ||||
| 		moretrees.jungletree_model.rules_b = moretrees.jt_rules_b2 | ||||
| 	end | ||||
|  | ||||
| 	minetest.swap_node(pos, biome_lib.air) | ||||
| 	moretrees.jungletree_model.enable_unique_ids = true | ||||
|  | ||||
| 	minetest.remove_node(pos) | ||||
| 	local leaves = minetest.find_nodes_in_area({x = pos.x-1, y = pos.y, z = pos.z-1}, {x = pos.x+1, y = pos.y+10, z = pos.z+1}, "default:leaves") | ||||
| 	for leaf in ipairs(leaves) do | ||||
| 			minetest.swap_node(leaves[leaf], biome_lib.air) | ||||
| 			minetest.remove_node(leaves[leaf]) | ||||
| 	end | ||||
| 	minetest.spawn_tree(pos, moretrees.jungletree_model) | ||||
| end | ||||
|  | ||||
| -- code to spawn fir trees | ||||
|  | ||||
| function moretrees.grow_fir(pos) | ||||
| function moretrees:grow_fir(pos) | ||||
| 	if math.random(2) == 1 then | ||||
| 		moretrees.fir_model.leaves="moretrees:fir_leaves" | ||||
| 	else | ||||
| @@ -296,20 +265,21 @@ function moretrees.grow_fir(pos) | ||||
| 		moretrees.fir_model.rules_b = moretrees.ct_rules_b2 | ||||
| 	end | ||||
|  | ||||
| 	moretrees.fir_model.enable_unique_ids = true | ||||
| 	moretrees.fir_model.iterations = 7 | ||||
| 	moretrees.fir_model.random_level = 5 | ||||
|  | ||||
| 	minetest.swap_node(pos, biome_lib.air) | ||||
| 	minetest.remove_node(pos) | ||||
| 	local leaves = minetest.find_nodes_in_area({x = pos.x, y = pos.y, z = pos.z}, {x = pos.x, y = pos.y+5, z = pos.z}, "default:leaves") | ||||
| 	for leaf in ipairs(leaves) do | ||||
| 		minetest.swap_node(leaves[leaf], biome_lib.air) | ||||
| 			minetest.remove_node(leaves[leaf]) | ||||
| 	end | ||||
| 	minetest.spawn_tree(pos,moretrees.fir_model) | ||||
| end | ||||
|  | ||||
| -- same thing, but a smaller version that grows only in snow biomes | ||||
|  | ||||
| function moretrees.grow_fir_snow(pos) | ||||
| function moretrees:grow_fir_snow(pos) | ||||
| 	if math.random(2) == 1 then | ||||
| 		moretrees.fir_model.leaves="moretrees:fir_leaves" | ||||
| 	else | ||||
| @@ -323,15 +293,20 @@ function moretrees.grow_fir_snow(pos) | ||||
| 		moretrees.fir_model.rules_b = moretrees.ct_rules_b2 | ||||
| 	end | ||||
|  | ||||
| 	moretrees.fir_model.enable_unique_ids = true | ||||
| 	moretrees.fir_model.iterations = 2 | ||||
| 	moretrees.fir_model.random_level = 2 | ||||
|  | ||||
| 	minetest.swap_node(pos, biome_lib.air) | ||||
| 	minetest.remove_node(pos) | ||||
| 	local leaves = minetest.find_nodes_in_area({x = pos.x, y = pos.y, z = pos.z}, {x = pos.x, y = pos.y+5, z = pos.z}, "default:leaves") | ||||
| 	for leaf in ipairs(leaves) do | ||||
| 			minetest.swap_node(leaves[leaf], biome_lib.air) | ||||
| 			minetest.remove_node(leaves[leaf]) | ||||
| 	end | ||||
| 	minetest.spawn_tree(pos,moretrees.fir_model) | ||||
| end | ||||
|  | ||||
| minetest.register_on_punchnode(function(pos, node, puncher, pointed_thing) | ||||
| 	print(dump(node.name)..": param1 = "..dump(node.param1)..", param2 = "..dump(node.param2)) | ||||
| end) | ||||
|  | ||||
| print(S("[Moretrees] Loaded (2013-02-11)")) | ||||
|   | ||||
							
								
								
									
										117
									
								
								leafdecay.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,117 @@ | ||||
| -- leaf decay | ||||
|  | ||||
| -- this function is based on the default leafdecay code | ||||
| local process_drops = function(pos, name) | ||||
| 	local drops = minetest.get_node_drops(name) | ||||
| 	for _,dropitem in ipairs(drops) do | ||||
| 		if dropitem ~= name | ||||
| 		  or (string.find(name, "leaves") and moretrees.decay_leaves_as_items) then | ||||
| 			local newpos = { | ||||
| 						x=pos.x + math.random() - 0.5, | ||||
| 						y=pos.y + math.random() - 0.5, | ||||
| 						z=pos.z + math.random() - 0.5 | ||||
| 					} | ||||
| 			minetest.add_item(newpos, dropitem) | ||||
| 		end | ||||
| 	end | ||||
| end | ||||
|  | ||||
| if moretrees.enable_leafdecay then | ||||
| 	for i in ipairs(moretrees.treelist) do | ||||
| 		local treename = moretrees.treelist[i][1] | ||||
| 		if treename ~= "jungletree" and treename ~= "fir" and treename ~= "palm" then | ||||
| 			minetest.register_abm({ | ||||
| 				nodenames = "moretrees:"..treename.."_leaves", | ||||
| 				interval = moretrees.leafdecay_delay, | ||||
| 				chance = moretrees.leafdecay_chance, | ||||
| 				action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 					if minetest.find_node_near(pos, moretrees.leafdecay_radius, { "ignore", "moretrees:"..treename.."_trunk" }) then return end | ||||
| 					process_drops(pos, node.name) | ||||
| 					minetest.remove_node(pos) | ||||
| 					nodeupdate(pos) | ||||
| 				end | ||||
| 			}) | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = {"moretrees:jungletree_leaves_red","moretrees:jungletree_leaves_green","moretrees:jungletree_leaves_yellow"}, | ||||
| 		interval = moretrees.leafdecay_delay, | ||||
| 		chance = moretrees.leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			if minetest.find_node_near(pos, moretrees.leafdecay_radius, {"ignore", "default:jungletree", "moretrees:jungletree_trunk"}) then return end | ||||
| 			process_drops(pos, node.name) | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	}) | ||||
|  | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = {"moretrees:fir_leaves", "moretrees:fir_leaves_bright"}, | ||||
| 		interval = moretrees.leafdecay_delay, | ||||
| 		chance = moretrees.leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			if minetest.find_node_near(pos, moretrees.leafdecay_radius, { "ignore", "moretrees:fir_trunk" }) then return end | ||||
| 			process_drops(pos, node.name) | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	}) | ||||
|  | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = "moretrees:palm_leaves", | ||||
| 		interval = moretrees.leafdecay_delay, | ||||
| 		chance = moretrees.leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			if minetest.find_node_near(pos, moretrees.palm_leafdecay_radius, { "ignore", "moretrees:palm_trunk" }) then return end | ||||
| 			process_drops(pos, node.name) | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_default_leafdecay then | ||||
|  | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = "default:leaves", | ||||
| 		interval = moretrees.default_leafdecay_delay, | ||||
| 		chance = moretrees.default_leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			if minetest.find_node_near(pos, moretrees.default_leafdecay_radius, { "ignore", "default:tree" }) then return end | ||||
| 			process_drops(pos, node.name) | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	})	 | ||||
| end | ||||
|  | ||||
| -- Decay apple tree blossoms from nature_classic mod  | ||||
| if minetest.get_modpath("nature_classic") then | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = "moretrees:apple_blossoms", | ||||
| 		interval = moretrees.default_leafdecay_delay, | ||||
| 		chance = moretrees.default_leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 		if minetest.find_node_near(pos, moretrees.leafdecay_radius, { "ignore", "moretrees:apple_tree_trunk" }) then return end | ||||
| 			process_drops(pos, "moretrees:apple_tree_leaves") | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| if moretrees.enable_default_jungle_leafdecay then | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = "default:jungleleaves", | ||||
| 		interval = moretrees.default_jungle_leafdecay_delay, | ||||
| 		chance = moretrees.default_jungle_leafdecay_chance, | ||||
| 		action = function(pos, node, active_object_count, active_object_count_wider) | ||||
| 			if minetest.find_node_near(pos, moretrees.default_jungle_leafdecay_radius, { "ignore", "default:jungletree" }) then return end | ||||
| 			process_drops(pos, node.name) | ||||
| 			minetest.remove_node(pos) | ||||
| 			nodeupdate(pos) | ||||
| 		end | ||||
| 	})	 | ||||
| end | ||||
|  | ||||
| @@ -65,6 +65,7 @@ Spruce Cone = Fichtenzapfen | ||||
| Pine Cone = Kiefernzapfen | ||||
| Fir Cone = Tannenzapfen | ||||
| Jungle Sapling = Tropenbaumsetzling | ||||
| Jungle Tree Leaves (Green) = Tropenbaumlaub (gruen) | ||||
| Jungle Tree Leaves (Yellow) = Tropenbaumlaub (gelb) | ||||
| Jungle Tree Leaves (Red) = Tropenbaumlaub (rot) | ||||
| Douglas Fir Leaves (Bright) = Douglasiennadeln (breit) | ||||
|   | ||||
							
								
								
									
										100
									
								
								locale/es.txt
									
									
									
									
									
								
							
							
						
						| @@ -1,100 +0,0 @@ | ||||
| # Traducido por Carlos Barraza | ||||
|  | ||||
| ### crafts.lua ### | ||||
| Coconut Milk = Leche de Coco | ||||
| Raw Coconut = Coco crudo | ||||
| Acorn Muffin batter =  Masa de Mollete de Bellota | ||||
| Acorn Muffin = Mollete de Bellota | ||||
| Roasted Spruce Cone Nuts = Cono de Picea Tostado | ||||
| Roasted Pine Cone Nuts = Cono de Pino Tostado | ||||
| Roasted Fir Cone Nuts = Cono de Abeto Tostado | ||||
| Roasted Cedar Cone Nuts = Coco de Alamo Tostado | ||||
| Date = Datilera | ||||
| Date & nut snack = Datilera y Nueces | ||||
| Date-nut cake batter = Pasta de torta de Datilera | ||||
| Date-nut cake = Pastel de Datilera | ||||
| Date-nut energy bar = Barra energetica de Datilera | ||||
|  | ||||
| ### date_palm.lua ### | ||||
| Dates = Datilera | ||||
| Date Stem = Tallo de Datilera | ||||
| Date Flowers = Flores de Datilera | ||||
|  | ||||
| ### node_defs.lua ### | ||||
| Beech Tree Trunk = Tronco de Arbol de Haya | ||||
| Apple Tree Trunk = Tronco de Arbol de Manzana | ||||
| Oak Tree Trunk = Tronco de Arbol de Roble | ||||
| Giant Sequoia Trunk = Tronco de Sequoia Gigante | ||||
| Birch Tree Trunk = Tronco de Arbol de Abedul | ||||
| Palm Tree Trunk = Tronco de Palmera | ||||
| Spruce Tree Trunk = Tronco de Arbol de Abeto | ||||
| Willow Tree Trunk = Tronco de Arbol de Sauce | ||||
| Rubber Tree Trunk = Tronco de Arbol de Arbol de Goma | ||||
| Jungle Tree = Tronco de Arbol de Arbol de la Selva | ||||
| Douglas Fir Trunk = Tronco de Arbol de Abeto de Douglas | ||||
| Cedar Tree Trunk = Tronco de Arbol de Cedro | ||||
| Date Palm Tree Trunk = Tronco de Palmera Datilera | ||||
| Acacia Tree Trunk = Tronco de Arbol de Acacia | ||||
| Poplar Tree Trunk = Tronco de Arbol de Alamo | ||||
|  | ||||
| Beech Tree Planks = Madera de Arbol de Haya | ||||
| Apple Tree Planks = Madera de Arbol de Manzana | ||||
| Oak Tree Planks = Madera de Arbol de Roble | ||||
| Giant Sequoia Planks = Madera de Sequoia Gigante | ||||
| Birch Tree Planks = Madera de Arbol de Abedul | ||||
| Palm Tree Planks = Madera de Palmera | ||||
| Spruce Tree Planks = Madera de Arbol de Abeto | ||||
| Willow Tree Planks = Madera de Arbol de Sauce | ||||
| Rubber Tree Planks = Madera de Arbol de Arbol de Goma | ||||
| Jungle Planks = Madera de Arbol de Arbol de la Selva | ||||
| Douglas Fir Planks = Madera de Arbol de Abeto de Douglas | ||||
| Cedar Tree Planks = Madera de Arbol de Cedro | ||||
| Date Palm Tree Planks = Madera de Palmera Datilera | ||||
| Acacia Tree Planks = Madera de Arbol de Acacia | ||||
| Poplar Tree Planks = Madera de Arbol de Alamo | ||||
|  | ||||
| Beech Tree Sapling = Retoño de Arbol de Haya | ||||
| Apple Tree Sapling = Retoño de Arbol de Manzana | ||||
| Oak Tree Sapling = Retoño de Arbol de Roble | ||||
| Giant Sequoia Sapling = Retoño de Arbol de Sequoia Gigante | ||||
| Birch Tree Sapling = Retoño de Arbol de Abedul | ||||
| Palm Tree Sapling = Retoño de Palmera | ||||
| Spruce Tree Sapling = Retoño de Arbol de Abeto | ||||
| Willow Tree Sapling = Retoño de Arbol de Sauce | ||||
| Rubber Tree Sapling = Retoño de Arbol de Arbol de Goma | ||||
| Jungle Sapling = Retoño de Arbol de Arbol de la Selva | ||||
| Douglas Fir Sapling = Retoño de Arbol de Abeto de Douglas | ||||
| Cedar Tree Sapling = Retoño de Arbol de Cedro | ||||
| Date Palm Tree Tree Sapling = Retoño de Palmera Datilera | ||||
| Acacia Tree Sapling = Retoño de Arbol de Acacia | ||||
| Poplar Tree Sapling = Retoño de Arbol de Alamo | ||||
|  | ||||
| Beech Tree Leaves = Hojas de Arbol de Haya | ||||
| Apple Tree Leaves = Hojas de Arbol de Manzana | ||||
| Oak Tree Leaves = Hojas de Arbol de Roble | ||||
| Giant Sequoia Leaves = Hojas de Sequoia Gigante | ||||
| Birch Tree Leaves = Hojas de Arbol de Abedul | ||||
| Palm Tree Leaves = Hojas de Palmera | ||||
| Spruce Tree Leaves = Hojas de Arbol de Abeto | ||||
| Willow Tree Leaves = Hojas de Arbol de Sauce | ||||
| Rubber Tree Leaves = Hojas de Arbol de Arbol de Goma | ||||
| Jungle Leaves = Hojas de Arbol de Arbol de la Selva | ||||
| Douglas Fir Leaves = Hojas de Arbol de Abeto de Douglas | ||||
| Cedar Tree Leaves = Hojas de Arbol de Cedro | ||||
| Date Palm Tree Tree Leaves = Hojas de Palmera Datilera | ||||
| Acacia Tree Leaves = Hojas de Arbol de Acacia | ||||
| Poplar Tree Leaves = Hojas de Arbol de Alamo | ||||
|  | ||||
| Acorn = Bellota | ||||
| Coconut = Coco | ||||
| Spruce Cone = Coco de Picea | ||||
| Pine Cone = Coco de Pino | ||||
| Fir Cone = Coco de Abeto | ||||
| Cedar Cone = Coco de Alamo | ||||
| Jungle Sapling = Retoño de Arbol de la Selva | ||||
| Jungle Tree Leaves (Yellow) = Hojas de Arbol de la Selva (Amarilla) | ||||
| Jungle Tree Leaves (Red) = Hojas de Arbol de la Selva (Roja) | ||||
| Douglas Fir Leaves (Bright) = Hojas de Arbol de Abeto de Douglas (Brillante) | ||||
| Rubber Tree Trunk (Empty) = Tronco de Arbol de Goma (Vacio) | ||||
|  | ||||
| [Moretrees] Loaded (2013-02-11) = [Masarboles] Cargado | ||||
| @@ -1,74 +0,0 @@ | ||||
| # Traduction par Yoan31 | ||||
| #  | ||||
|  | ||||
| ### crafts.lua ### | ||||
| Coconut Milk = Lait de Coco | ||||
| Raw Coconut = Noix de Coco Crue | ||||
| Acorn Muffin batter = Pate à Muffins au Gland | ||||
| Acorn Muffin = Muffins au Gland | ||||
| Roasted Spruce Cone Nuts = Noix de Cône de Sapin Roties | ||||
| Roasted Pine Cone Nuts = Noix de Pomme de pin Roties | ||||
| Roasted Fir Cone Nuts = Noix de Cône de Sapin Roties | ||||
|  | ||||
| ### node_defs.lua ### | ||||
| Beech Tree Trunk = Tronc d'Arbre de Hêtre | ||||
| Apple Tree Trunk = Tronc d'Arbre de Pommier | ||||
| Oak Tree Trunk = Tronc d'Arbre de chêne | ||||
| Giant Sequoia Trunk = Tronc d'Arbre de Séquoia Géant | ||||
| Birch Tree Trunk = Tronc d'Arbre de bouleau | ||||
| Palm Tree Trunk = Tronc d'Arbre de Palmier | ||||
| Spruce Tree Trunk = Tronc d'Arbre d'Epicéa | ||||
| Pine Tree Trunk = Tronc d'Arbre de Pin | ||||
| Willow Tree Trunk = Tronc d'Arbre de Saule | ||||
| Rubber Tree Trunk = Tronc d'Arbre a Caoutchouc | ||||
| Jungle Tree Trunk = Tronc d'Arbre de la Jungle | ||||
| Douglas Fir Trunk = Tronc de Sapin de Douglas | ||||
| Beech Tree Planks = Planches de Hêtre | ||||
| Apple Tree Planks = Planches de Pommier | ||||
| Oak Tree Planks = Planches de Chêne | ||||
| Giant Sequoia Planks = Planches de d'Arbre de Séquoia Géant | ||||
| Birch Tree Planks = Planches d'Arbre de bouleau | ||||
| Palm Tree Planks = Planches d'Arbre de Palmier | ||||
| Spruce Tree Planks = Planches d'Arbre d'Epicéa | ||||
| Pine Tree Planks = Planches d'Arbre de Pin | ||||
| Willow Tree Planks = Planches d'Arbre de Saule | ||||
| Rubber Tree Planks = Planches d'Arbre a Caoutchouc | ||||
| Jungle Tree Planks = Planches d'Arbre de la Jungle | ||||
| Douglas Fir Planks = Planches de Sapin de Douglas | ||||
| Beech Tree Sapling = Pousse d'Arbre de Hêtre | ||||
| Apple Tree Sapling = Pousse d'Arbre de Pommier | ||||
| Oak Tree Sapling = Pousse d'Arbre de chêne | ||||
| Giant Sequoia Sapling = Pousse d'Arbre de Séquoia Géant | ||||
| Birch Tree Sapling = Pousse d'Arbre de bouleau | ||||
| Palm Tree Sapling = Pousse d'Arbre de Palmier | ||||
| Spruce Tree Sapling = Pousse d'Arbre d'Epicéa | ||||
| Pine Tree Sapling = Pousse d'Arbre de Pin | ||||
| Willow Tree Sapling = Pousse d'Arbre de Saule | ||||
| Rubber Tree Sapling = Pousse d'Arbre a Caoutchouc | ||||
| Jungle Tree Sapling = Pousse d'Arbre de la Jungle | ||||
| Douglas Fir Sapling = Pousse de Sapin de Douglas | ||||
| Beech Tree Leaves = Feuilles d'Arbre de Hêtre | ||||
| Apple Tree Leaves = Feuilles d'Arbre de Pommier | ||||
| Oak Tree Leaves = Feuilles d'Arbre de chêne | ||||
| Giant Sequoia Leaves = Feuilles d'Arbre de Séquoia Géant | ||||
| Birch Tree Leaves = Feuilles d'Arbre de bouleau | ||||
| Palm Tree Leaves = Feuilles d'Arbre de Palmier | ||||
| Spruce Tree Leaves = Feuilles d'Arbre d'Epicéa | ||||
| Pine Tree Leaves = Feuilles d'Arbre de Pin | ||||
| Willow Tree Leaves = Feuilles d'Arbre de Saule | ||||
| Rubber Tree Leaves = Feuilles d'Arbre a Caoutchouc | ||||
| Jungle Tree Leaves = Feuilles d'Arbre de la Jungle | ||||
| Douglas Fir Leaves = Feuilles de Sapin de Douglas | ||||
|  | ||||
| Acorn = Gland | ||||
| Coconut = Noix de Coco | ||||
| Spruce Cone = Cône de Sapin | ||||
| Pine Cone = Pomme de Pin | ||||
| Fir Cone = Pomme de Sapin | ||||
| Jungle Sapling = Pousse d'Arbre de la Jungle | ||||
| Jungle Tree Leaves (Yellow) = Feuille d'Arbre de la Jungle (Jaune) | ||||
| Jungle Tree Leaves (Red) =  Feuille d'Arbre de la Jungle (Rouge) | ||||
| Douglas Fir Leaves (Bright) = Feuille de Sapin de Douglas (Brillant) | ||||
| Rubber Tree Trunk (Empty) = Tronc d'Arbre en Caoutchouc (Vide) | ||||
|  | ||||
| [Moretrees] Loaded (2013-02-11) =  | ||||
| @@ -65,6 +65,7 @@ Spruce Cone = | ||||
| Pine Cone =  | ||||
| Fir Cone =  | ||||
| Jungle Sapling =  | ||||
| Jungle Tree Leaves (Green) =  | ||||
| Jungle Tree Leaves (Yellow) =  | ||||
| Jungle Tree Leaves (Red) =  | ||||
| Douglas Fir Leaves (Bright) =  | ||||
|   | ||||
							
								
								
									
										314
									
								
								node_defs.lua
									
									
									
									
									
								
							
							
						
						| @@ -8,16 +8,14 @@ moretrees.treelist = { | ||||
| 	{"oak",			"Oak Tree",			"acorn",		"Acorn",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | ||||
| 	{"sequoia",		"Giant Sequoia"}, | ||||
| 	{"birch",		"Birch Tree"}, | ||||
| 	{"palm",			"Palm Tree",		"palm_fruit_trunk_gen",		"Palm Tree",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2},	1.0 }, | ||||
| 	{"date_palm",		"Date Palm Tree",	"date_palm_fruit_trunk",	"Date Palm Tree",	{0, 0, 0, 0, 0, 0},	0.0 }, | ||||
| 	{"palm",		"Palm Tree",		"coconut",		"Coconut",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 1.0 }, | ||||
| 	{"spruce",		"Spruce Tree",		"spruce_cone",	"Spruce Cone",	{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | ||||
| 	{"cedar",			"Cedar Tree",		"cedar_cone",				"Cedar Cone",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2},	0.8 }, | ||||
| 	{"poplar",			"Poplar Tree"}, | ||||
| 	{"poplar_small",	"Poplar Tree"}, | ||||
| 	{"pine",		"Pine Tree",		"pine_cone",	"Pine Cone",	{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | ||||
| 	{"willow",		"Willow Tree"}, | ||||
| 	{"acacia",		"Acacia Tree"}, | ||||
| 	{"rubber_tree",	"Rubber Tree"}, | ||||
| 	{"jungletree",	"Jungle Tree"}, | ||||
| 	{"fir",			"Douglas Fir",		"fir_cone",		"Fir Cone",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | ||||
| 	{"jungletree",		"Jungle Tree",		nil,						nil,				nil,								nil, "default_junglesapling.png"  }, | ||||
| } | ||||
|  | ||||
| local dirs1 = { 21, 20, 23, 22, 21 } | ||||
| @@ -29,26 +27,44 @@ local moretrees_plantlike_leaves_visual_scale = 1 | ||||
|  | ||||
| if moretrees.plantlike_leaves then  | ||||
| 	moretrees_new_leaves_drawtype = "plantlike" | ||||
| 	moretrees_plantlike_leaves_visual_scale = math.sqrt(2) | ||||
| 	moretrees_plantlike_leaves_visual_scale = 1.189 | ||||
| end | ||||
|  | ||||
| -- redefine default leaves to handle plantlike and/or leaf decay options | ||||
|  | ||||
| if moretrees.enable_default_leafdecay then | ||||
| 	minetest.override_item("default:leaves", { | ||||
| 		groups = { snappy = 3, flammable = 2, leaves = 1 } | ||||
| 	}) | ||||
| end | ||||
| if moretrees.plantlike_leaves then | ||||
| 	minetest.override_item("default:leaves", { | ||||
| 		inventory_image = minetest.inventorycube("default_leaves.png"), | ||||
| 		drawtype = "plantlike", | ||||
| 		visual_scale = math.sqrt(2) | ||||
| 		visual_scale = 1.189 | ||||
| 	}) | ||||
| else | ||||
| 	minetest.override_item("default:leaves", { | ||||
| 		waving = 1 | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| -- redefine default jungle leaves for same | ||||
|  | ||||
| if moretrees.enable_default_leafdecay then | ||||
| 	minetest.override_item("default:jungleleaves", { | ||||
| 		groups = { snappy = 3, flammable = 2, leaves = 1 } | ||||
| 	}) | ||||
| end | ||||
| if moretrees.plantlike_leaves then | ||||
| 	minetest.override_item("default:jungleleaves", { | ||||
| 		inventory_image = minetest.inventorycube("default_jungleleaves.png"), | ||||
| 		drawtype = "plantlike", | ||||
| 		visual_scale = math.sqrt(2) | ||||
| 		visual_scale = 1.189 | ||||
| 	}) | ||||
| else | ||||
| 	minetest.override_item("default:jungleleaves", { | ||||
| 		waving = 1 | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| @@ -60,27 +76,7 @@ for i in ipairs(moretrees.treelist) do | ||||
| 	local selbox = moretrees.treelist[i][5] | ||||
| 	local vscale = moretrees.treelist[i][6] | ||||
|  | ||||
| 	local saptex = moretrees.treelist[i][7] | ||||
|  | ||||
| 	-- player will get a sapling with 1/100 chance | ||||
| 	-- player will get leaves only if he/she gets no saplings, | ||||
| 	-- this is because max_items is 1 | ||||
|  | ||||
| 	local droprarity = 100 | ||||
| 	local decay = moretrees.leafdecay_radius | ||||
|  | ||||
| 	if treename == "palm" then | ||||
| 		droprarity = 20 | ||||
| 		decay = moretrees.palm_leafdecay_radius | ||||
| 	elseif treename == "date_palm" then | ||||
| 		decay = moretrees.palm_leafdecay_radius | ||||
| 	end | ||||
|  | ||||
| 	if treename ~= "jungletree"  -- the default game provides jungle tree and pine trunk/planks nodes. | ||||
| 		and treename ~= "poplar_small" | ||||
| 		and treename ~= "pine" then | ||||
|  | ||||
| 		saptex = "moretrees_"..treename.."_sapling.png" | ||||
| 	if treename ~= "jungletree" then -- the default game provides jungle tree trunk/planks nodes. | ||||
|  | ||||
| 		minetest.register_node("moretrees:"..treename.."_trunk", { | ||||
| 			description = S(treedesc.." Trunk"), | ||||
| @@ -90,7 +86,7 @@ for i in ipairs(moretrees.treelist) do | ||||
| 				"moretrees_"..treename.."_trunk.png" | ||||
| 			}, | ||||
| 			paramtype2 = "facedir", | ||||
| 			is_ground_content = false, | ||||
| 			is_ground_content = true, | ||||
| 			groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | ||||
| 			sounds = default.node_sound_wood_defaults(), | ||||
| 			on_place = minetest.rotate_node, | ||||
| @@ -99,7 +95,7 @@ for i in ipairs(moretrees.treelist) do | ||||
| 		minetest.register_node("moretrees:"..treename.."_planks", { | ||||
| 			description = S(treedesc.." Planks"), | ||||
| 			tiles = {"moretrees_"..treename.."_wood.png"}, | ||||
| 			is_ground_content = false, | ||||
| 			is_ground_content = true, | ||||
| 			groups = {snappy=1,choppy=2,oddly_breakable_by_hand=2,flammable=3,wood=1}, | ||||
| 			sounds = default.node_sound_wood_defaults(), | ||||
| 		}) | ||||
| @@ -107,12 +103,10 @@ for i in ipairs(moretrees.treelist) do | ||||
| 		minetest.register_node("moretrees:"..treename.."_sapling", { | ||||
| 			description = S(treedesc.." Sapling"), | ||||
| 			drawtype = "plantlike", | ||||
| 			tiles = {saptex}, | ||||
| 			inventory_image = saptex, | ||||
| 			tiles = {"moretrees_"..treename.."_sapling.png"}, | ||||
| 			inventory_image = "moretrees_"..treename.."_sapling.png", | ||||
| 			paramtype = "light", | ||||
| 			paramtype2 = "waving", | ||||
| 			walkable = false, | ||||
| 			is_ground_content = true, | ||||
| 			selection_box = { | ||||
| 				type = "fixed", | ||||
| 				fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||
| @@ -121,6 +115,33 @@ for i in ipairs(moretrees.treelist) do | ||||
| 			sounds = default.node_sound_defaults(), | ||||
| 		}) | ||||
| 	 | ||||
| 		minetest.register_node("moretrees:"..treename.."_sapling_ongen", { | ||||
| 			description = S(treedesc.." Sapling"), | ||||
| 			drawtype = "plantlike", | ||||
| 			tiles = {"moretrees_"..treename.."_sapling.png"}, | ||||
| 			inventory_image = "moretrees_"..treename.."_sapling.png", | ||||
| 			paramtype = "light", | ||||
| 			walkable = false, | ||||
| 			selection_box = { | ||||
| 				type = "fixed", | ||||
| 				fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||
| 			}, | ||||
| 			groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,not_in_creative_inventory=1,sapling=1}, | ||||
| 			sounds = default.node_sound_defaults(), | ||||
| 			drop = "moretrees:"..treename.."_sapling" | ||||
| 		}) | ||||
|  | ||||
| 		-- player will get a sapling with 1/100 chance | ||||
| 		-- player will get leaves only if he/she gets no saplings, | ||||
| 		-- this is because max_items is 1 | ||||
|  | ||||
|  | ||||
| 		local droprarity = 100 | ||||
|  | ||||
| 		if treename == "palm" then | ||||
| 			droprarity = 20 | ||||
| 		end | ||||
|  | ||||
| 		local moretrees_leaves_inventory_image = nil | ||||
| 		local moretrees_new_leaves_waving = nil | ||||
|  | ||||
| @@ -138,8 +159,7 @@ for i in ipairs(moretrees.treelist) do | ||||
| 			tiles = { "moretrees_"..treename.."_leaves.png" }, | ||||
| 			inventory_image = moretrees_leaves_inventory_image, | ||||
| 			paramtype = "light", | ||||
| 			is_ground_content = false, | ||||
| 			groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 1}, | ||||
| 			groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1}, | ||||
| 			sounds = default.node_sound_leaves_defaults(), | ||||
|  | ||||
| 			drop = { | ||||
| @@ -151,10 +171,9 @@ for i in ipairs(moretrees.treelist) do | ||||
| 			}, | ||||
| 		}) | ||||
|  | ||||
| 		if moretrees.enable_stairs then | ||||
| 			if minetest.get_modpath("moreblocks") then | ||||
| 		if minetest.get_modpath("moreblocks") and moretrees.enable_stairsplus then | ||||
|  | ||||
| 	--			stairsplus:register_all(modname, subname, recipeitem, {fields}) | ||||
| --			stairsplus:register_all(modname, subname, recipeitem, {fields}) | ||||
|  | ||||
| 			stairsplus:register_all( | ||||
| 				"moretrees", | ||||
| @@ -183,85 +202,28 @@ for i in ipairs(moretrees.treelist) do | ||||
| 					drop = treename.."_planks", | ||||
| 				} | ||||
| 			) | ||||
| 			elseif minetest.get_modpath("stairs") then | ||||
| 				stairs.register_stair_and_slab( | ||||
| 					"moretrees_"..treename.."_trunk", | ||||
| 					"moretrees:"..treename.."_trunk", | ||||
| 					{ snappy=1, choppy=2, oddly_breakable_by_hand=1, flammable=2 }, | ||||
| 					{	"moretrees_"..treename.."_trunk_top.png", | ||||
| 						"moretrees_"..treename.."_trunk_top.png", | ||||
| 						"moretrees_"..treename.."_trunk.png" | ||||
| 					}, | ||||
| 					S(treedesc.." Trunk Stair"), | ||||
| 					S(treedesc.." Trunk Slab"), | ||||
| 					default.node_sound_wood_defaults() | ||||
| 				) | ||||
|  | ||||
| 				stairs.register_stair_and_slab( | ||||
| 					"moretrees_"..treename.."_planks", | ||||
| 					"moretrees:"..treename.."_planks", | ||||
| 					{ snappy=1, choppy=2, oddly_breakable_by_hand=2, flammable=3 }, | ||||
| 					{ "moretrees_"..treename.."_wood.png" }, | ||||
| 					S(treedesc.." Planks Stair"), | ||||
| 					S(treedesc.." Planks Slab"), | ||||
| 					default.node_sound_wood_defaults() | ||||
| 				) | ||||
|  | ||||
| 			end | ||||
| 		end | ||||
| 	end | ||||
|  | ||||
| 	minetest.register_node("moretrees:"..treename.."_sapling_ongen", { | ||||
| 		description = S(treedesc.." Sapling (fast growth)"), | ||||
| 		drawtype = "plantlike", | ||||
| 		tiles = {saptex}, | ||||
| 		inventory_image = saptex, | ||||
| 		paramtype = "light", | ||||
| 		paramtype2 = "waving", | ||||
| 		walkable = false, | ||||
| 		is_ground_content = true, | ||||
| 		selection_box = { | ||||
| 			type = "fixed", | ||||
| 			fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||
| 		}, | ||||
| 		groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,sapling=1}, | ||||
| 		sounds = default.node_sound_defaults(), | ||||
| 		drop = "moretrees:"..treename.."_sapling" | ||||
| 	}) | ||||
|  | ||||
| 	local fruitname = nil | ||||
| 	if fruit then | ||||
| 		fruitname = "moretrees:"..fruit | ||||
| 		minetest.register_node(fruitname, { | ||||
| 		minetest.register_node("moretrees:"..fruit, { | ||||
| 			description = S(fruitdesc), | ||||
| 			drawtype = "plantlike", | ||||
| 			tiles = { "moretrees_"..fruit..".png" }, | ||||
| 			inventory_image = "moretrees_"..fruit..".png^[transformR180", | ||||
| 			wield_image = "moretrees_"..fruit..".png^[transformR180", | ||||
| 			visual_scale = vscale, | ||||
| 			paramtype = "light", | ||||
| 			sunlight_propagates = true, | ||||
| 			is_ground_content = false, | ||||
| 			walkable = false, | ||||
| 			paramtype = "light", | ||||
| 			selection_box = { | ||||
| 				type = "fixed", | ||||
| 					fixed = selbox | ||||
| 				}, | ||||
| 			groups = {fleshy=3,dig_immediate=3,flammable=2, attached_node=1, leafdecay = 1, leafdecay_drop = 1}, | ||||
| 			groups = {fleshy=3,dig_immediate=3,flammable=2, attached_node=1}, | ||||
| 			sounds = default.node_sound_defaults(), | ||||
| 		}) | ||||
| 	end | ||||
|  | ||||
| 	if treename ~= "jungletree" | ||||
| 		and treename ~= "poplar_small" | ||||
| 		and treename ~= "pine" then | ||||
| 			default.register_leafdecay({ | ||||
| 				trunks = { "moretrees:"..treename.."_trunk" }, | ||||
| 				leaves = { "moretrees:"..treename.."_leaves", fruitname }, | ||||
| 				radius = decay, | ||||
| 			}) | ||||
| 	end | ||||
|  | ||||
| 	minetest.register_abm({ | ||||
| 		nodenames = { "moretrees:"..treename.."_trunk_sideways" }, | ||||
| 		interval = 1, | ||||
| @@ -281,50 +243,46 @@ for i in ipairs(moretrees.treelist) do | ||||
| 	end | ||||
| end | ||||
|  | ||||
| -- Add small poplar saplings | ||||
|  | ||||
| local poplar_sapling = minetest.registered_nodes["moretrees:poplar_sapling"] | ||||
| local poplar_sapling_ongen = minetest.registered_nodes["moretrees:poplar_sapling_ongen"] | ||||
| local poplar_small_sapling = {} | ||||
| local poplar_small_sapling_ongen = {} | ||||
| for k,v in pairs(poplar_sapling) do | ||||
| 	poplar_small_sapling[k] = v | ||||
| end | ||||
| for k,v in pairs(poplar_sapling_ongen) do | ||||
| 	poplar_small_sapling_ongen[k] = v | ||||
| end | ||||
| poplar_small_sapling.tiles = {"moretrees_poplar_small_sapling.png"} | ||||
| poplar_small_sapling.inventory_image = "moretrees_poplar_small_sapling.png" | ||||
| poplar_small_sapling.is_ground_content = true | ||||
| poplar_small_sapling_ongen.tiles_ongen = {"moretrees_poplar_small_sapling.png"} | ||||
| poplar_small_sapling_ongen.inventory_image_ongen = "moretrees_poplar_small_sapling.png" | ||||
| poplar_small_sapling_ongen.drop = "moretrees:poplar_small_sapling" | ||||
| poplar_small_sapling_ongen.is_ground_content = true | ||||
|  | ||||
| minetest.register_node("moretrees:poplar_small_sapling", poplar_small_sapling) | ||||
| minetest.register_node("moretrees:poplar_small_sapling_ongen", poplar_small_sapling_ongen) | ||||
| if moretrees.spawn_saplings then | ||||
| 	table.insert(moretrees.avoidnodes, "moretrees:poplar_sapling") | ||||
| 	table.insert(moretrees.avoidnodes, "moretrees:poplar_small_sapling_ongen") | ||||
| end | ||||
|  | ||||
| local poplar_leaves_drop = minetest.registered_nodes["moretrees:poplar_leaves"].drop | ||||
| minetest.override_item("moretrees:poplar_leaves", { | ||||
| 	drop = { | ||||
| 		max_items = poplar_leaves_drop.maxitems, | ||||
| 		items = { | ||||
| 			{items = {"moretrees:poplar_sapling"}, rarity = 1.33 * poplar_leaves_drop.items[1].rarity }, | ||||
| 			{items = {"moretrees:poplar_small_sapling"}, rarity = 1.33 * poplar_leaves_drop.items[1].rarity }, | ||||
| 			{items = {"moretrees:poplar_leaves"} } | ||||
| 		} | ||||
| 	} | ||||
| }) | ||||
|  | ||||
| -- Extra nodes for jungle trees: | ||||
|  | ||||
| local jungleleaves = {"yellow","red"} | ||||
| local jungleleavesnames = {"Yellow", "Red"} | ||||
| for color = 1, #jungleleaves do | ||||
| minetest.register_node("moretrees:jungletree_sapling", { | ||||
| 	description = S("Jungle Sapling"), | ||||
| 	drawtype = "plantlike", | ||||
| 	visual_scale = 1.0, | ||||
| 	tiles = {"default_junglesapling.png"}, | ||||
| 	inventory_image = "default_junglesapling.png", | ||||
| 	wield_image = "default_junglesapling.png", | ||||
| 	paramtype = "light", | ||||
| 	walkable = false, | ||||
| 	selection_box = { | ||||
| 		type = "fixed", | ||||
| 		fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||
| 	}, | ||||
| 	groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,sapling=1}, | ||||
| 	sounds = default.node_sound_leaves_defaults(), | ||||
| }) | ||||
|  | ||||
| minetest.register_node("moretrees:jungletree_sapling_ongen", { | ||||
| 	description = S("Jungle Sapling"), | ||||
| 	drawtype = "plantlike", | ||||
| 	visual_scale = 1.0, | ||||
| 	tiles = {"default_junglesapling.png"}, | ||||
| 	inventory_image = "default_junglesapling.png", | ||||
| 	wield_image = "default_junglesapling.png", | ||||
| 	paramtype = "light", | ||||
| 	walkable = false, | ||||
| 	selection_box = { | ||||
| 		type = "fixed", | ||||
| 		fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||
| 	}, | ||||
| 	groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,not_in_creative_inventory=1,sapling=1}, | ||||
| 	sounds = default.node_sound_leaves_defaults(), | ||||
| 	drop = "moretrees:jungletree_sapling" | ||||
| }) | ||||
|  | ||||
| local jungleleaves = {"green","yellow","red"} | ||||
| local jungleleavesnames = {"Green", "Yellow", "Red"} | ||||
| for color = 1, 3 do | ||||
| 	local leave_name = "moretrees:jungletree_leaves_"..jungleleaves[color] | ||||
|  | ||||
| 	local moretrees_leaves_inventory_image = nil | ||||
| @@ -343,12 +301,11 @@ for color = 1, #jungleleaves do | ||||
| 		tiles = {"moretrees_jungletree_leaves_"..jungleleaves[color]..".png"}, | ||||
| 		inventory_image = moretrees_leaves_inventory_image, | ||||
| 		paramtype = "light", | ||||
| 		is_ground_content = false, | ||||
| 		groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 3 }, | ||||
| 		groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1}, | ||||
| 		drop = { | ||||
| 			max_items = 1, | ||||
| 			items = { | ||||
| 				{items = {"default:junglesapling"}, rarity = 100 }, | ||||
| 				{items = {'moretrees:jungletree_sapling'}, rarity = 100 }, | ||||
| 				{items = {"moretrees:jungletree_leaves_"..jungleleaves[color]} } | ||||
| 			} | ||||
| 		}, | ||||
| @@ -356,18 +313,6 @@ for color = 1, #jungleleaves do | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| -- To get Moretrees to generate its own jungle trees among the default mapgen | ||||
| -- we need our own copy of that node, which moretrees will match against. | ||||
|  | ||||
| local jungle_tree = table.copy(minetest.registered_nodes["default:jungletree"]) | ||||
| minetest.register_node("moretrees:jungletree_trunk", jungle_tree) | ||||
|  | ||||
| default.register_leafdecay({ | ||||
| 	trunks = { "default:jungletree", "moretrees:jungletree_trunk" }, | ||||
| 	leaves = { "default:jungleleaves", "moretrees:jungletree_leaves_yellow", "moretrees:jungletree_leaves_red" }, | ||||
| 	radius = moretrees.leafdecay_radius, | ||||
| }) | ||||
|  | ||||
| -- Extra needles for firs | ||||
|  | ||||
| local moretrees_leaves_inventory_image = nil | ||||
| @@ -384,8 +329,7 @@ minetest.register_node("moretrees:fir_leaves_bright", { | ||||
| 	tiles = { "moretrees_fir_leaves_bright.png" }, | ||||
| 	inventory_image = moretrees_leaves_inventory_image, | ||||
| 	paramtype = "light", | ||||
| 	is_ground_content = false, | ||||
| 	groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 3 }, | ||||
| 	groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1 }, | ||||
| 	drop = { | ||||
| 		max_items = 1, | ||||
| 		items = { | ||||
| @@ -396,29 +340,22 @@ minetest.register_node("moretrees:fir_leaves_bright", { | ||||
| 	sounds = default.node_sound_leaves_defaults() | ||||
| }) | ||||
|  | ||||
| default.register_leafdecay({ | ||||
| 	trunks = { "moretrees:fir_trunk" }, | ||||
| 	leaves = { "moretrees:fir_leaves", "moretrees:fir_leaves_bright" }, | ||||
| 	radius = moretrees.leafdecay_radius, | ||||
| }) | ||||
|  | ||||
|  | ||||
| if moretrees.enable_redefine_apple then | ||||
| 	local appledef = table.copy(minetest.registered_nodes["default:apple"]) | ||||
| 	appledef.groups.attached_node = 1 | ||||
| 	minetest.register_node(":default:apple", appledef) | ||||
| 	minetest.override_item("default:apple", | ||||
| 		{groups = { fleshy=3, dig_immediate=3, flammable=2, leafdecay=3, leafdecay_drop=1, attached_node = 1} | ||||
| 	}) | ||||
| end | ||||
|  | ||||
| table.insert(moretrees.avoidnodes, "default:jungletree") | ||||
| table.insert(moretrees.avoidnodes, "default:pine_tree") | ||||
| table.insert(moretrees.avoidnodes, "default:acacia_tree") | ||||
| table.insert(moretrees.avoidnodes, "moretrees:jungletree_trunk") | ||||
| table.insert(moretrees.avoidnodes, "moretrees:fir_trunk") | ||||
| table.insert(moretrees.avoidnodes, "default:tree") | ||||
|  | ||||
| if moretrees.spawn_saplings then | ||||
| 	table.insert(moretrees.avoidnodes, "snow:sapling_pine") | ||||
| 	table.insert(moretrees.avoidnodes, "default:junglesapling") | ||||
| 	table.insert(moretrees.avoidnodes, "default:pine_sapling") | ||||
| 	table.insert(moretrees.avoidnodes, "moretrees:jungle_tree_sapling") | ||||
| 	table.insert(moretrees.avoidnodes, "moretrees:jungle_tree_sapling_ongen") | ||||
| end | ||||
|  | ||||
| -- "empty" (tapped) rubber tree nodes | ||||
| @@ -434,7 +371,6 @@ minetest.register_node("moretrees:rubber_tree_trunk_empty", { | ||||
| 	groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | ||||
| 	sounds = default.node_sound_wood_defaults(), | ||||
| 	paramtype2 = "facedir", | ||||
| 	is_ground_content = false, | ||||
| 	on_place = minetest.rotate_node, | ||||
| }) | ||||
|  | ||||
| @@ -449,7 +385,7 @@ minetest.register_abm({ | ||||
| 	end, | ||||
| }) | ||||
|  | ||||
| -- For compatibility with old nodes, recently-changed nodes, and default nodes | ||||
| -- For compatibility with old nodes and recently-changed nodes. | ||||
|  | ||||
| minetest.register_alias("technic:rubber_tree_full",      "moretrees:rubber_tree_trunk") | ||||
| minetest.register_alias("farming_plus:rubber_tree_full", "moretrees:rubber_tree_trunk") | ||||
| @@ -463,6 +399,15 @@ minetest.register_alias("technic:rubber_tree_sapling",			"moretrees:rubber_tree_ | ||||
| minetest.register_alias("farming_plus:rubber_sapling", "moretrees:rubber_tree_sapling") | ||||
| minetest.register_alias("farming:rubber_sapling", "moretrees:rubber_tree_sapling") | ||||
|  | ||||
| minetest.register_alias("default:junglesapling","moretrees:jungletree_sapling") | ||||
| minetest.register_alias("moretrees:jungletree_trunk_sideways", "moreblocks:horizontal_jungle_tree") | ||||
| minetest.register_alias("moretrees:jungletree_trunk", "default:jungletree") | ||||
| minetest.register_alias("moretrees:jungletree_planks", "default:junglewood") | ||||
|  | ||||
| minetest.register_alias("jungletree:leaves_green", "moretrees:jungletree_leaves_green") | ||||
| minetest.register_alias("jungletree:leaves_red", "moretrees:jungletree_leaves_red") | ||||
| minetest.register_alias("jungletree:leaves_yellow", "moretrees:jungletree_leaves_yellow") | ||||
|  | ||||
| minetest.register_alias("moretrees:conifer_trunk", "moretrees:fir_trunk") | ||||
| minetest.register_alias("moretrees:conifer_trunk_sideways", "moretrees:fir_trunk_sideways") | ||||
| minetest.register_alias("moretrees:conifer_leaves", "moretrees:fir_leaves") | ||||
| @@ -475,22 +420,3 @@ minetest.register_alias("conifers:leaves",						"moretrees:fir_leaves") | ||||
| minetest.register_alias("conifers:leaves_special", "moretrees:fir_leaves_bright") | ||||
| minetest.register_alias("conifers:sapling", "moretrees:fir_sapling") | ||||
|  | ||||
| minetest.register_alias("moretrees:jungletree_sapling",			"default:junglesapling") | ||||
| minetest.register_alias("moretrees:jungletree_trunk_sideways",	"moreblocks:horizontal_jungle_tree") | ||||
| minetest.register_alias("moretrees:jungletree_planks",			"default:junglewood") | ||||
| minetest.register_alias("moretrees:jungletree_leaves_green",	"default:jungleleaves") | ||||
|  | ||||
| minetest.register_alias("moretrees:acacia_trunk",				"default:acacia_tree") | ||||
| minetest.register_alias("moretrees:acacia_planks",				"default:acacia_wood") | ||||
| minetest.register_alias("moretrees:acacia_sapling",				"default:acacia_sapling") | ||||
| minetest.register_alias("moretrees:acacia_leaves",				"default:acacia_leaves") | ||||
|  | ||||
| minetest.register_alias("moretrees:pine_trunk",					"moretrees:cedar_trunk") | ||||
| minetest.register_alias("moretrees:pine_planks",				"moretrees:cedar_planks") | ||||
| minetest.register_alias("moretrees:pine_sapling",				"moretrees:cedar_sapling") | ||||
| minetest.register_alias("moretrees:pine_leaves",				"moretrees:cedar_leaves") | ||||
| minetest.register_alias("moretrees:pine_cone",					"moretrees:cedar_cone") | ||||
| minetest.register_alias("moretrees:pine_nuts",					"moretrees:cedar_nuts") | ||||
| minetest.register_alias("moretrees:pine_sapling_ongen",			"moretrees:cedar_sapling_ongen") | ||||
|  | ||||
| minetest.register_alias("moretrees:dates",					"moretrees:dates_f4") | ||||
|   | ||||
							
								
								
									
										143
									
								
								saplings.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,84 +1,93 @@ | ||||
| -- sapling growth | ||||
| -- these tables only affect hand-placed saplings | ||||
| -- mapgen-placed always use their biome def settings, which are much more | ||||
| -- limited, in the interest of speed. | ||||
|  | ||||
| local dirt_surfaces = { | ||||
| 	set = true, | ||||
| 	["default:dirt"] = true, | ||||
| 	["default:dirt_with_grass"] = true, | ||||
| 	["default:dirt_with_dry_grass"] = true, | ||||
| 	["default:dirt_with_coniferous_litter"] = true, | ||||
| 	["default:dirt_with_rainforest_litter"] = true, | ||||
| 	["woodsoils:dirt_with_leaves_1"] = true, | ||||
| 	["woodsoils:dirt_with_leaves_2"] = true, | ||||
| 	["woodsoils:grass_with_leaves_1"] = true, | ||||
| 	["woodsoils:grass_with_leaves_2"] = true | ||||
| } | ||||
|  | ||||
| local conifer_surfaces =  { | ||||
| 	set = true, | ||||
| 	["default:dirt"] = true, | ||||
| 	["default:dirt_with_grass"] = true, | ||||
| 	["default:dirt_with_dry_grass"] = true, | ||||
| 	["default:dirt_with_coniferous_litter"] = true, | ||||
| 	["default:dirt_with_rainforest_litter"] = true, | ||||
| 	["woodsoils:dirt_with_leaves_1"] = true, | ||||
| 	["woodsoils:dirt_with_leaves_2"] = true, | ||||
| 	["woodsoils:grass_with_leaves_1"] = true, | ||||
| 	["woodsoils:grass_with_leaves_2"] = true, | ||||
| 	["default:dirt_with_snow"] = true | ||||
| } | ||||
|  | ||||
| local sand_surfaces = { | ||||
| 	set = true, | ||||
| 	["default:sand"] = true, | ||||
| 	["default:desert_sand"] = true, | ||||
| 	["cottages:loam"] = true, | ||||
| 	-- note, no silver sand here. | ||||
| 	-- too cold for a palm, too... well... sandy for anything else. | ||||
| } | ||||
|  | ||||
| for i in ipairs(moretrees.treelist) do | ||||
| 	local treename = moretrees.treelist[i][1] | ||||
| 	local tree_model = treename.."_model" | ||||
| 	local tree_biome = treename.."_biome" | ||||
| 	local surfaces | ||||
| 	local grow_function = moretrees[tree_model] | ||||
|  | ||||
| 	if treename == "spruce" | ||||
| 	  or treename == "fir" | ||||
| 	  or treename == "cedar" | ||||
| 	  or treename == "pine" then | ||||
| 		surfaces = conifer_surfaces | ||||
| 	elseif string.find(treename, "palm") then | ||||
| 		surfaces = sand_surfaces | ||||
| 	else | ||||
| 		surfaces = dirt_surfaces | ||||
| 	end | ||||
| 	if treename ~= "birch" and treename ~= "spruce" and treename ~= "fir" and treename ~= "jungletree" then | ||||
|  | ||||
| 	if treename == "spruce" | ||||
| 	  or treename == "fir" | ||||
| 	  or treename == "birch" | ||||
| 	  or treename == "jungletree" then | ||||
| 		grow_function = "moretrees.grow_"..treename | ||||
| 	end | ||||
|  | ||||
| 	biome_lib:dbg(dump(moretrees[tree_biome].surface)) | ||||
|  | ||||
| 	biome_lib:grow_plants({ | ||||
| 		plantslib:dbg(dump(moretrees[tree_biome].surface)) | ||||
| 		plantslib:grow_plants({ | ||||
| 			grow_delay = moretrees.sapling_interval, | ||||
| 			grow_chance = moretrees.sapling_chance, | ||||
| 			grow_plant = "moretrees:"..treename.."_sapling", | ||||
| 		grow_nodes = surfaces, | ||||
| 		grow_function = grow_function, | ||||
| 			grow_nodes = moretrees[tree_biome].surface, | ||||
| 			grow_function = moretrees[tree_model], | ||||
| 		}) | ||||
|  | ||||
| 	biome_lib:grow_plants({ | ||||
| 		plantslib:grow_plants({ | ||||
| 			grow_delay = 2, | ||||
| 		grow_chance = 1, | ||||
| 			grow_chance = 30, | ||||
| 			grow_plant = "moretrees:"..treename.."_sapling_ongen", | ||||
| 		grow_nodes = surfaces, | ||||
| 		grow_function = grow_function, | ||||
| 			grow_nodes = moretrees[tree_biome].surface, | ||||
| 			grow_function = moretrees[tree_model], | ||||
| 		}) | ||||
|  | ||||
| 	end | ||||
| end | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = moretrees.sapling_interval, | ||||
| 	grow_chance = moretrees.sapling_chance, | ||||
| 	grow_plant = "moretrees:birch_sapling", | ||||
| 	grow_nodes = moretrees.birch_biome.surface, | ||||
| 	grow_function = "moretrees:grow_birch" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = 2, | ||||
| 	grow_chance = 30, | ||||
| 	grow_plant = "moretrees:birch_sapling_ongen", | ||||
| 	grow_nodes = moretrees.birch_biome.surface, | ||||
| 	grow_function = "moretrees:grow_birch" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = moretrees.sapling_interval, | ||||
| 	grow_chance = moretrees.sapling_chance, | ||||
| 	grow_plant = "moretrees:spruce_sapling", | ||||
| 	grow_nodes = moretrees.spruce_biome.surface, | ||||
| 	grow_function = "moretrees:grow_spruce" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = 2, | ||||
| 	grow_chance = 30, | ||||
| 	grow_plant = "moretrees:spruce_sapling_ongen", | ||||
| 	grow_nodes = moretrees.spruce_biome.surface, | ||||
| 	grow_function = "moretrees:grow_spruce" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = moretrees.sapling_interval, | ||||
| 	grow_chance = moretrees.sapling_chance, | ||||
| 	grow_plant = "moretrees:fir_sapling", | ||||
| 	grow_nodes = moretrees.fir_biome.surface, | ||||
| 	grow_function = "moretrees:grow_fir" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = 2, | ||||
| 	grow_chance = 30, | ||||
| 	grow_plant = "moretrees:fir_sapling_ongen", | ||||
| 	grow_nodes = moretrees.fir_biome.surface, | ||||
| 	grow_function = "moretrees:grow_fir" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = moretrees.sapling_interval, | ||||
| 	grow_chance = moretrees.sapling_chance, | ||||
| 	grow_plant = "moretrees:jungletree_sapling", | ||||
| 	grow_nodes = moretrees.jungletree_biome.surface, | ||||
| 	grow_function = "moretrees:grow_jungletree" | ||||
| }) | ||||
|  | ||||
| plantslib:grow_plants({ | ||||
| 	grow_delay = 2, | ||||
| 	grow_chance = 30, | ||||
| 	grow_plant = "moretrees:jungletree_sapling_ongen", | ||||
| 	grow_nodes = moretrees.jungletree_biome.surface, | ||||
| 	grow_function = "moretrees:grow_jungletree" | ||||
| }) | ||||
|  | ||||
|   | ||||
							
								
								
									
										168
									
								
								screenshot.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,168 +0,0 @@ | ||||
| -- Usage: | ||||
| -- - Create a new world | ||||
| -- - Set world mapgen: v6 | ||||
| -- - Set world seed: 2625051331357512570 | ||||
| -- - Enable the moretrees mod | ||||
| -- - Edit the moretrees source | ||||
| --   - Disable all trees in default_settings.lua | ||||
| --   - Recommended: make saplings grow fast in default_settings.lua: | ||||
| --     sapling_interval = 5 | ||||
| --     sapling_chance = 1 | ||||
| --   - Apply the patch below to moretrees | ||||
| --     (so that jungle trees are always large, and differently-colored): | ||||
| --     use 'git apply --ignore-space-change' | ||||
| --   - Make sure this file (you are reading) will be loaded when minetest starts ! | ||||
| --     (e.g. add 'dofile(modpath.."/screenshot.lua")' to init.lua) | ||||
| -- - Start minetest | ||||
| -- - Goto 700,y,-280 (approximately) | ||||
| -- - Make sure the world is loaded between x = 650 .. 780 and z = -350 .. -180 | ||||
| -- - Give the chat command '/make-scene' | ||||
| -- - Wait & walk/fly around until all trees have grown | ||||
| -- - goto the platform at 780, 30, -277 | ||||
| -- - Set the viewing range to 300, with fog enabled | ||||
| -- - Take a screenshot. | ||||
|  | ||||
| -- Patch to apply to moretrees | ||||
| --[[ | ||||
| diff --git a/init.lua b/init.lua | ||||
| index 8189ffd..afd4644 100644 | ||||
| --- a/init.lua | ||||
| +++ b/init.lua | ||||
| @@ -225,9 +225,12 @@ moretrees.ct_rules_b1 = "[-FBf][+FBf]" | ||||
|  moretrees.ct_rules_a2 = "FF[FF][&&-FBF][&&+FBF][&&---FBF][&&+++FBF]F/A" | ||||
|  moretrees.ct_rules_b2 = "[-fB][+fB]" | ||||
|   | ||||
| +local jleaves = 1 | ||||
|  function moretrees.grow_jungletree(pos) | ||||
|         local r1 = math.random(2) | ||||
|         local r2 = math.random(3) | ||||
| +       r1 = jleaves | ||||
| +       jleaves = jleaves % 2 + 1 | ||||
|         if r1 == 1 then | ||||
|                 moretrees.jungletree_model.leaves2 = "moretrees:jungletree_leaves_red" | ||||
|         else  | ||||
| @@ -235,6 +238,7 @@ function moretrees.grow_jungletree(pos) | ||||
|         end | ||||
|         moretrees.jungletree_model.leaves2_chance = math.random(25, 75) | ||||
|   | ||||
| +       r2=3 | ||||
|         if r2 == 1 then | ||||
|                 moretrees.jungletree_model.trunk_type = "single" | ||||
|                 moretrees.jungletree_model.iterations = 2 | ||||
| ]] | ||||
|  | ||||
|  | ||||
| minetest.register_chatcommand("make-scene", { | ||||
| 	func = function() | ||||
| 		minetest.place_node({x=780, y=30, z=-277}, {name="default:obsidian"}) | ||||
| 		minetest.place_node({x=780, y=30, z=-278}, {name="default:obsidian"}) | ||||
| 		minetest.place_node({x=781, y=30, z=-277}, {name="default:obsidian"}) | ||||
| 		minetest.place_node({x=781, y=30, z=-278}, {name="default:obsidian"}) | ||||
| 		minetest.place_node({x=781, y=30, z=-276}, {name="default:obsidian"}) | ||||
| 		minetest.place_node({x=780, y=30, z=-276}, {name="default:obsidian"}) | ||||
|  | ||||
| 		for z = -360, -300 do | ||||
| 			dy=2 | ||||
| 			for x = 630 + (-z - 360)/3, 660 + (-z - 300)/3 do | ||||
| 				for y = 5, 22 do | ||||
| 					minetest.place_node({x=x, y=y, z=z}, {name="default:desert_stone"}) | ||||
| 				end | ||||
| 				for y = 23, 25 + dy do | ||||
| 					minetest.place_node({x=x, y=y, z=z}, {name="default:desert_sand"}) | ||||
| 				end | ||||
| 				dy = 0 | ||||
| 			end | ||||
| 		end | ||||
|  | ||||
| 		minetest.place_node({x=717, y=2, z=-298}, {name = "moretrees:palm_sapling"}) | ||||
| 		minetest.place_node({x=713, y=2, z=-302}, {name = "moretrees:palm_sapling"}) | ||||
| 		minetest.place_node({x=713, y=2, z=-307}, {name = "moretrees:palm_sapling"}) | ||||
| 		minetest.place_node({x=717, y=2, z=-318}, {name = "moretrees:palm_sapling"}) | ||||
| 		minetest.place_node({x=723, y=2, z=-320}, {name = "moretrees:palm_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=645, y=26, z=-314}, {name="moretrees:date_palm_sapling"}) | ||||
| 		minetest.place_node({x=653, y=26, z=-322}, {name="moretrees:date_palm_sapling"}) | ||||
| 		minetest.place_node({x=649, y=26, z=-334}, {name="moretrees:date_palm_sapling"}) | ||||
| 		minetest.place_node({x=662, y=26, z=-342}, {name="moretrees:date_palm_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=672, y=5, z=-305}, {name="moretrees:oak_sapling"}) | ||||
| 		minetest.place_node({x=690, y=6, z=-322}, {name="moretrees:oak_sapling"}) | ||||
| 		minetest.place_node({x=695, y=7, z=-335}, {name="moretrees:oak_sapling"}) | ||||
| 		minetest.place_node({x=699, y=4, z=-301}, {name="moretrees:oak_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=751, y=5, z=-254}, {name="moretrees:apple_tree_sapling"}) | ||||
| 		minetest.place_node({x=729, y=3, z=-275}, {name="moretrees:apple_tree_sapling"}) | ||||
| 		minetest.place_node({x=747, y=4, z=-270}, {name="moretrees:apple_tree_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=671, y=5, z=-283}, {name="default:junglesapling"}) | ||||
| 		minetest.place_node({x=680, y=4, z=-287}, {name="default:junglesapling"}) | ||||
| 		minetest.place_node({x=702, y=4, z=-288}, {name="default:junglesapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=646, y=12, z=-199}, {name="moretrees:spruce_sapling"}) | ||||
| 		minetest.place_node({x=644, y=14, z=-177}, {name="moretrees:spruce_sapling"}) | ||||
| 		minetest.place_node({x=678, y=9, z=-211}, {name="moretrees:spruce_sapling"}) | ||||
| 		minetest.place_node({x=663, y=10, z=-215}, {name="moretrees:spruce_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=637, y=3, z=-263}, {name="moretrees:sequoia_sapling"}) | ||||
| 		minetest.place_node({x=625, y=3, z=-250}, {name="moretrees:sequoia_sapling"}) | ||||
| 		minetest.place_node({x=616, y=3, z=-233}, {name="moretrees:sequoia_sapling"}) | ||||
| 		minetest.place_node({x=635, y=3, z=-276}, {name="moretrees:sequoia_sapling"}) | ||||
| 		minetest.place_node({x=681, y=11, z=-260}, {name="moretrees:sequoia_sapling"}) | ||||
| 		minetest.place_node({x=682, y=10, z=-247}, {name="moretrees:sequoia_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=737, y=7, z=-195}, {name="moretrees:cedar_sapling"}) | ||||
| 		minetest.place_node({x=720, y=8, z=-189}, {name="moretrees:cedar_sapling"}) | ||||
| 		minetest.place_node({x=704, y=7, z=-187}, {name="moretrees:cedar_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=731, y=2, z=-227}, {name="moretrees:poplar_sapling"}) | ||||
| 		minetest.place_node({x=721, y=2, z=-233}, {name="moretrees:poplar_sapling"}) | ||||
| 		minetest.place_node({x=712, y=1, z=-237}, {name="moretrees:poplar_sapling"}) | ||||
| 		minetest.place_node({x=743, y=3, z=-228}, {name="moretrees:poplar_small_sapling"}) | ||||
| 		minetest.place_node({x=750, y=3, z=-230}, {name="moretrees:poplar_small_sapling"}) | ||||
| 		minetest.place_node({x=731, y=5, z=-233}, {name="moretrees:poplar_small_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=702, y=2, z=-274}, {name="moretrees:birch_sapling"}) | ||||
| 		minetest.place_node({x=697, y=2, z=-271}, {name="moretrees:birch_sapling"}) | ||||
| 		minetest.place_node({x=696, y=2, z=-264}, {name="moretrees:birch_sapling"}) | ||||
| 		minetest.place_node({x=710, y=2, z=-265}, {name="moretrees:birch_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=707, y=8, z=-247}, {name="moretrees:fir_sapling"}) | ||||
| 		minetest.place_node({x=699, y=10, z=-254}, {name="moretrees:fir_sapling"}) | ||||
| 		minetest.place_node({x=729, y=5, z=-261}, {name="moretrees:fir_sapling"}) | ||||
| 		minetest.place_node({x=732, y=5, z=-252}, {name="moretrees:fir_sapling"}) | ||||
| 		minetest.place_node({x=741, y=4, z=-262}, {name="moretrees:fir_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=751, y=2, z=-286}, {name="moretrees:willow_sapling"}) | ||||
|  | ||||
| 		minetest.place_node({x=760, y=5, z=-223}, {name="moretrees:rubber_tree_sapling"}) | ||||
| 		minetest.place_node({x=762, y=5, z=-230}, {name="moretrees:rubber_tree_sapling"}) | ||||
| 		minetest.place_node({x=766, y=5, z=-243}, {name="moretrees:rubber_tree_sapling"}) | ||||
| 		minetest.place_node({x=764, y=6, z=-252}, {name="moretrees:rubber_tree_sapling"}) | ||||
| 	end | ||||
| }) | ||||
|  | ||||
| --[[ | ||||
| The following is a search/replace command suitable for vi (/vim) or sed, to convert minetest log | ||||
| messages to equivalent lua commands: | ||||
|  | ||||
| s/.*\(\(moretrees\|default\)[^ ]*\) at (\([-0-9]\+\),\([-0-9]\+\),\([-0-9]\+\)).*/\t\tminetest.place_node({x=\3, y=\4, z=\5}, {name="\1"})/ | ||||
|  | ||||
| E.g. a minetest log line of the following kind: | ||||
| 		2016-07-03 11:30:50: ACTION[Server]: singleplayer places node moretrees:rubber_tree_sapling at  (760,5,-223) | ||||
| Becomes: | ||||
| 		minetest.place_node({x=760, y=5, z=-223}, {name="moretrees:rubber_tree_sapling"}) | ||||
| (Except that the example log line above has an extra space added, so it won't be converted) | ||||
|  | ||||
| vi/vim users: Add the minetest log lines to this file, then enter the following command, with | ||||
| <expression> replaced with the search/replace expression above. | ||||
| 	:%<expression> | ||||
|  | ||||
| sed users: Add the minetest log lines to this file, then execute the following command at the shell | ||||
| prompt with <expression> replaced by the search/replace expression above. Don't forget the | ||||
| single-quotes. | ||||
| 	sed '<expression>' < screenshot.lua > screenshot-new.lua | ||||
|  | ||||
| Windows users: You're probably out of luck. And the effort of doing such a thing is probably | ||||
| larger anyway than the effort of copying an existing line and typing things manually. | ||||
| ]] | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								screenshot.png
									
									
									
									
									
								
							
							
						
						| Before Width: | Height: | Size: 101 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_acacia_leaves.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 150 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_acacia_sapling.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 205 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_acacia_trunk.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 599 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_acacia_trunk_top.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 651 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_acacia_wood.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 175 B | 
| Before Width: | Height: | Size: 877 B | 
| Before Width: | Height: | Size: 429 B | 
| Before Width: | Height: | Size: 1009 B | 
| Before Width: | Height: | Size: 2.1 KiB | 
| Before Width: | Height: | Size: 685 B | 
| Before Width: | Height: | Size: 608 B | 
| Before Width: | Height: | Size: 695 B | 
| Before Width: | Height: | Size: 450 B | 
| Before Width: | Height: | Size: 757 B | 
| Before Width: | Height: | Size: 261 B | 
| Before Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 673 B | 
| Before Width: | Height: | Size: 735 B | 
| Before Width: | Height: | Size: 9.2 KiB | 
| Before Width: | Height: | Size: 3.8 KiB | 
| Before Width: | Height: | Size: 4.0 KiB | 
| Before Width: | Height: | Size: 8.9 KiB | 
| Before Width: | Height: | Size: 8.1 KiB | 
| Before Width: | Height: | Size: 9.0 KiB | 
| Before Width: | Height: | Size: 2.5 KiB | 
| Before Width: | Height: | Size: 4.0 KiB | 
| Before Width: | Height: | Size: 2.1 KiB | 
| Before Width: | Height: | Size: 315 B After Width: | Height: | Size: 315 B | 
| Before Width: | Height: | Size: 205 B After Width: | Height: | Size: 205 B | 
| Before Width: | Height: | Size: 369 B After Width: | Height: | Size: 369 B | 
| Before Width: | Height: | Size: 202 B After Width: | Height: | Size: 202 B | 
| Before Width: | Height: | Size: 1.0 KiB After Width: | Height: | Size: 1.0 KiB | 
| Before Width: | Height: | Size: 451 B After Width: | Height: | Size: 451 B | 
| Before Width: | Height: | Size: 648 B After Width: | Height: | Size: 648 B | 
| Before Width: | Height: | Size: 563 B | 
| Before Width: | Height: | Size: 447 B | 
| Before Width: | Height: | Size: 347 B | 
| Before Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 980 B | 
| Before Width: | Height: | Size: 751 B | 
| Before Width: | Height: | Size: 726 B | 
| @@ -1,22 +1,21 @@ | ||||
|  | ||||
| 		Elevation	Temperature	Nearness to	Nearby	What nodes		Perlin		Avoid | ||||
| Tree type	   (m)		(approx., °C)	some node	water	to spawn on		seed diff	radius | ||||
| -------------------------------------------------------------------------------------------------------------- | ||||
| ----------------------------------------------------------------------------------------------------------------------- | ||||
| jungle tree	- 5 to +10	 above +15	water, 20	  10	dirt_with_grass		329		 5 | ||||
| fir		 above +25	-20 to +10	  n/a		n/a	dirt_with_grass		359		 8 | ||||
| firs on snow	 above +15	-20 to +10	  n/a		n/a	snow:snow		359		 8 | ||||
| palm		- 1 to + 1	+15 to +32	water, 15	  10	sand			330		 5 | ||||
| date palm		- 1 to +10		 above +39		water, 20h,20v	100		desert_sand			339			10 | ||||
| date palm		+11 to +30		 above +39		water, 1h,30v	  1		desert_sand			340			10 | ||||
| apple		+ 1 to +10	+23 to +32	  n/a		n/a	dirt_with grass		331		15 | ||||
| oak		  0 to +10	+ 4 to +16	  n/a		n/a	dirt_with grass		332		15 | ||||
| sequoia		  0 to +10	-30 to +50	  n/a		n/a	dirt_with grass		333		10 | ||||
| birch		+10 to +15	-20 to +10	  n/a		n/a	dirt_with grass		334		 5 | ||||
| spruce		 above +20	-20 to +10	  n/a		n/a	dirt_with grass		335		10 | ||||
| cedar			    n/a			    n/a			water, 15		  5		dirt_with grass		336			10 | ||||
| pine		    n/a		    n/a		water, 15	   5	dirt_with grass		336		10 | ||||
| willow		- 5 to + 5	    n/a		water, 15	   5	dirt_with grass		337		20 | ||||
| acacia		    n/a		    n/a		  n/a		n/a	dirt_with_grass, | ||||
| 									desert_sand		n/a		15 | ||||
| rubber		- 5 to + 5	 above +32	water, 15	  10	dirt_with_grass		338		20 | ||||
| poplar			    n/a			-10 to +26		water, 15h,5v	  1		dirt_with_grass		341,342,343	10 | ||||
|  | ||||
| beech		    n/a		    n/a		  n/a		n/a	dirt_with_grass		2		10 | ||||
|  | ||||
| @@ -27,8 +26,9 @@ Notes: | ||||
| Beech trees are meant to replace default trees, but are themselves disabled by default.  They grow in the same areas | ||||
| as the default ones do, and under the same conditions. | ||||
|  | ||||
| Acacia trees depend on humidity as their primary control, and they don't use the fertile ground option. They grow near | ||||
| the edges of a desert biome, occasionally into the grass beyond, and anywhere else on desert sand or dirt with grass, | ||||
| where the humidity is low (but not bone dry). | ||||
|  | ||||
| Fir trees appear in a snow biome only with older versions of SPlizard's Snow Biomes mod.  In more recent versions, | ||||
| these trees will not grow, due to an engine bug. | ||||
|  | ||||
| Cedar trees replace, and are identical to, the original pine trees, as the minetest default game now has (a completely | ||||
| different type of) pine trees. | ||||
|   | ||||
							
								
								
									
										147
									
								
								tree_models.lua
									
									
									
									
									
								
							
							
						
						| @@ -10,7 +10,8 @@ moretrees.beech_model={ | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.apple_tree_model={ | ||||
| @@ -26,6 +27,7 @@ moretrees.apple_tree_model={ | ||||
| 	thin_branches=true, | ||||
| 	fruit="default:apple", | ||||
| 	fruit_chance=15, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.oak_model={ | ||||
| @@ -41,37 +43,7 @@ moretrees.oak_model={ | ||||
| 	thin_branches=false, | ||||
| 	fruit="moretrees:acorn", | ||||
| 	fruit_chance=3, | ||||
| } | ||||
|  | ||||
| moretrees.poplar_model={ | ||||
| 	axiom="TTTaaBCCCCCCCCCCCcccBBB[[f]&&G++f++Gf++Gf++Gf++G--]G[[f]&&G++f++Gf++Gf++Gf++G--]Gff", | ||||
| 	rules_a="T", | ||||
| 	rules_b="[[T]&&G++f++ff++ff++ff++f--]G", | ||||
| 	rules_c="[[T]&&G++f++ff++ff++ff++f--G++[d]G[d]G++G[d]G[d]G[d]G++G[d]G[d]G[d]G++G[d]G[d]G[d]G++G[d]G]G", | ||||
| 	rules_d="f", | ||||
| 	trunk="air", | ||||
| 	trunk="moretrees:poplar_trunk", | ||||
| 	leaves="moretrees:poplar_leaves", | ||||
| 	angle=45, | ||||
| 	iterations=0, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=false, | ||||
| } | ||||
|  | ||||
| moretrees.poplar_small_model={ | ||||
| 	axiom="TT[T]BCCCCccBBff", | ||||
| 	rules_a="T", | ||||
| 	rules_b="[[f]&&G++f++Gf++Gf++Gf++G--]G", | ||||
| 	rules_c="[[T]&&G++f++[d]Gf++[d]Gf++[d]Gf++[d]G--]G", | ||||
| 	rules_d="f", | ||||
| 	trunk="moretrees:poplar_trunk", | ||||
| 	leaves="moretrees:poplar_leaves", | ||||
| 	angle=45, | ||||
| 	iterations=0, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=false, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.sequoia_model={ | ||||
| @@ -86,7 +58,8 @@ moretrees.sequoia_model={ | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="crossed", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.birch_model1={ | ||||
| @@ -101,7 +74,8 @@ moretrees.birch_model1={ | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.birch_model2={ | ||||
| @@ -116,16 +90,12 @@ moretrees.birch_model2={ | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| -- Coconuts can't be generated as fruit, because there is no support for the | ||||
| -- special fruit trunks that allow coconuts to regrow at the correct position | ||||
| -- in the tree. | ||||
| -- So, a placeholder fruit trunk is spawned. An ABM will convert it to the final | ||||
| -- fruit trunk, and generate the actual coconuts. | ||||
| moretrees.palm_model={ | ||||
| 	axiom="FFcccccc&FFFFFdddRA//A//A//A//A//A", | ||||
| 	axiom="FFcccccc&FFFFFddd[^&&&GR][^///&&&GR][^//////&&&GR][^***&&&GR]FA//A//A//A//A//A", | ||||
| 	rules_a="[&fb&bbb[++f--&ffff&ff][--f++&ffff&ff]&ffff&bbbb&b]", | ||||
| 	rules_b="f", | ||||
| 	rules_c="/", | ||||
| @@ -137,33 +107,9 @@ moretrees.palm_model={ | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true, | ||||
| 	fruit="moretrees:palm_fruit_trunk_gen", | ||||
| 	fruit_chance=0 | ||||
| } | ||||
|  | ||||
| -- Dates can't be generated as fruit, because there is no support for the | ||||
| -- special (male and female) fruit trunks that allow dates to regrow at the | ||||
| -- correct position in the tree. | ||||
| -- So, a generic fruit trunk is spawned. An ABM will convert it to a male | ||||
| -- or female fruit trunk, and generate the actual dates. | ||||
| moretrees.date_palm_model={ | ||||
| 	axiom="TTTTddddddddddccccccccccRT[TGGGGT]".. | ||||
| 		"ccccc[&&a]ccccc[&&a]ccccc[&&a]ccccc[&&a]ccccc[&&a]ccccc[&&a]".. | ||||
| 		"GGccccc[&a]ccccc[&a]ccccc[&a]ccccc[&a]ccccc[&a]ccccc[&a]".. | ||||
| 		"GGccccc[a]ccccc[a]ccccc[a]ccccc[a]ccccc[a]ccccc[a]", | ||||
| 	rules_a="Gffb&bbb[++f--&ffff&ff][--f++&ffff&ff]&ff&ff&bb&bb&bb", | ||||
| 	rules_b="f", | ||||
| 	rules_c="/", | ||||
| 	rules_d="F", | ||||
| 	trunk="moretrees:date_palm_trunk", | ||||
| 	leaves="moretrees:date_palm_leaves", | ||||
| 	angle=18, | ||||
| 	iterations=1, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=false, | ||||
| 	fruit="moretrees:date_palm_fruit_trunk", | ||||
| 	fruit_chance=0 | ||||
| 	fruit="moretrees:coconut", | ||||
| 	fruit_chance=0, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.spruce_model1={ | ||||
| @@ -180,7 +126,8 @@ moretrees.spruce_model1={ | ||||
| 	trunk_type="crossed", | ||||
| 	thin_branches=true, | ||||
| 	fruit="moretrees:spruce_cone", | ||||
| 	fruit_chance=8 | ||||
| 	fruit_chance=8, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.spruce_model2={ | ||||
| @@ -196,24 +143,26 @@ moretrees.spruce_model2={ | ||||
| 	trunk_type="crossed", | ||||
| 	thin_branches=true, | ||||
| 	fruit="moretrees:spruce_cone", | ||||
| 	fruit_chance=8 | ||||
| 	fruit_chance=8, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.cedar_model={ | ||||
| moretrees.pine_model={ | ||||
| 	axiom="FFFFFcccdddB///cFdFB////cFdFB///cFdFB///cFdFA///cFdFA///cFdFB[FF]f", | ||||
| 	rules_a="[&&&TTTT[++^TFdd][--&TFd]//Tdd[+^Fd][--&Fdd]]", | ||||
| 	rules_b="[&&&TTT[++^Fdd][--&Fdd]//dd[+^d][--&Fd]]", | ||||
| 	rules_c="/", | ||||
| 	rules_d="F", | ||||
| 	trunk="moretrees:cedar_trunk", | ||||
| 	leaves="moretrees:cedar_leaves", | ||||
| 	trunk="moretrees:pine_trunk", | ||||
| 	leaves="moretrees:pine_leaves", | ||||
| 	angle=30, | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true, | ||||
| 	fruit="moretrees:cedar_cone", | ||||
| 	fruit_chance=8 | ||||
| 	fruit="moretrees:pine_cone", | ||||
| 	fruit_chance=8, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.willow_model={ | ||||
| @@ -227,7 +176,40 @@ moretrees.willow_model={ | ||||
| 	iterations=2, | ||||
| 	random_level=0, | ||||
| 	trunk_type="crossed", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.acacia_model={ | ||||
| 	axiom="FFFFFFccccA", | ||||
| 	rules_a = "[B]//[B]//[B]//[B]", | ||||
| 	rules_b = "&TTTT&TT^^G&&----GGGGGG++GGG++"	-- line up with the "canvas" edge | ||||
| 			.."fffffffGG++G++"					-- first layer, drawn in a zig-zag raster pattern | ||||
| 			.."Gffffffff--G--" | ||||
| 			.."ffffffffG++G++" | ||||
| 			.."fffffffff--G--" | ||||
| 			.."fffffffff++G++" | ||||
| 			.."fffffffff--G--" | ||||
| 			.."ffffffffG++G++" | ||||
| 			.."Gffffffff--G--" | ||||
| 			.."fffffffGG" | ||||
| 			.."^^G&&----GGGGGGG++GGGGGG++"		-- re-align to second layer canvas edge | ||||
| 			.."ffffGGG++G++"					-- second layer | ||||
| 			.."GGfffff--G--" | ||||
| 			.."ffffffG++G++" | ||||
| 			.."fffffff--G--" | ||||
| 			.."ffffffG++G++" | ||||
| 			.."GGfffff--G--" | ||||
| 			.."ffffGGG", | ||||
| 	rules_c = "/", | ||||
| 	trunk="moretrees:acacia_trunk", | ||||
| 	leaves="moretrees:acacia_leaves", | ||||
| 	angle=45, | ||||
| 	iterations=3, | ||||
| 	random_level=0, | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.rubber_tree_model={ | ||||
| @@ -240,15 +222,16 @@ moretrees.rubber_tree_model={ | ||||
| 	iterations=3, | ||||
| 	random_level=1, | ||||
| 	trunk_type="double", | ||||
| 	thin_branches=true | ||||
| 	thin_branches=true, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.jungletree_model={ | ||||
| 	axiom=nil, | ||||
| 	rules_a=nil, | ||||
| 	rules_b=nil, | ||||
| 	trunk="moretrees:jungletree_trunk", | ||||
| 	leaves="default:jungleleaves", | ||||
| 	trunk="default:jungletree", | ||||
| 	leaves="moretrees:jungletree_leaves_green", | ||||
| 	leaves2=nil, | ||||
| 	leaves2_chance=nil, | ||||
| 	angle=45, | ||||
| @@ -257,7 +240,8 @@ moretrees.jungletree_model={ | ||||
| 	trunk_type=nil, | ||||
| 	thin_branches=true, | ||||
| 	fruit_chance=15, | ||||
| 	fruit="vines:vine" | ||||
| 	fruit="vines:vine", | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|  | ||||
| moretrees.fir_model={ | ||||
| @@ -272,5 +256,6 @@ moretrees.fir_model={ | ||||
| 	trunk_type="single", | ||||
| 	thin_branches=true, | ||||
| 	fruit="moretrees:fir_cone", | ||||
| 	fruit_chance=8 | ||||
| 	fruit_chance=8, | ||||
| 	enable_unique_ids = true, | ||||
| } | ||||
|   | ||||