Compare commits
	
		
			55 Commits
		
	
	
		
			unique_ids
			...
			2018-11-09
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | d521e34758 | ||
|  | 109b2009fb | ||
|  | 90a2d33707 | ||
|  | 61d1c54faf | ||
|  | 4e8cfdcd8c | ||
|  | f08f2d46f8 | ||
|  | 324155f805 | ||
|  | e5524c007d | ||
|  | ae53dc2e9b | ||
|  | ba2cd3d9b8 | ||
|  | af6d08414b | ||
|  | 1c0e3c7215 | ||
|  | aac8386b5a | ||
|  | c9add9b24d | ||
|  | f9d12a9def | ||
|  | 82744f4a96 | ||
|  | e5d30fed3d | ||
|  | 4e4636d6fe | ||
|  | 922058700d | ||
|  | cccfb1953b | ||
|  | f3a740f07e | ||
|  | 37e39a9ba7 | ||
|  | 3d7bcca134 | ||
|  | 2967c107d8 | ||
|  | f83f167d18 | ||
|  | 471a11f92d | ||
|  | b00d4f4a18 | ||
|  | 5735554d6a | ||
|  | 23ac37fc4c | ||
|  | d72574d288 | ||
|  | 8aa4888dee | ||
|  | 9028366407 | ||
|  | 9dd610e0e6 | ||
|  | 75e8027bf7 | ||
|  | c87a169fdd | ||
|  | afab0692df | ||
|  | 9ced339ce5 | ||
|  | 47980f5c86 | ||
|  | cd9926f9e5 | ||
|  | ee5bc1dfb3 | ||
|  | 62eb522948 | ||
|  | 8e0eeb5747 | ||
|  | fa3c3c6004 | ||
|  | 0c596f332e | ||
|  | 1cd2216731 | ||
|  | 2aad14a883 | ||
|  | f59fd56257 | ||
|  | 6694b8da61 | ||
|  | 2ae8a32aa3 | ||
|  | 7a105a0841 | ||
|  | 10ce68961f | ||
|  | ba170ee66e | ||
|  | 8276d82bac | ||
|  | ad0cbbc0fa | ||
|  | adecd4b1ea | 
							
								
								
									
										17
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | |||||||
|  | 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", | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										958
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						| @@ -3,390 +3,618 @@ Minetest mod moretrees | |||||||
|  |  | ||||||
| All source code: | All source code: | ||||||
| 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | ||||||
| 	Published under the terms and conditions of the WTFPL. | Date & cocos palm code (date_palm.lua, cocos_palm.lua) | ||||||
| All sapling textures (textures/*_sapling.png): | 	© 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): | ||||||
| 	© 2013, Tim Huppertz <mitroman@naturalnet.de> | 	© 2013, Tim Huppertz <mitroman@naturalnet.de> | ||||||
| 	Published under the terms and conditions of CC-BY-SA-3.0 Unported. |  | ||||||
| All other textures: | All other textures: | ||||||
| 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | 	© 2013, Vanessa Ezekowitz <vanessaezekowitz@gmail.com> | ||||||
| 	Published under the terms and conditions of CC-BY-SA-3.0 Unported. |  | ||||||
|  |  | ||||||
| ------------------------------------------------------------------------------- | ############################################################################### | ||||||
|  |  | ||||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE | License for all code: LGPL 3.0 | ||||||
|                     Version 2, December 2004 | License for all media and all other assets:  CC-by-SA 4.0 | ||||||
|  |  | ||||||
|  Copyright (C) 2004 Sam Hocevar <sam@hocevar.net> | ############################################################################### | ||||||
|  |  | ||||||
|  Everyone is permitted to copy and distribute verbatim or modified |                    GNU LESSER GENERAL PUBLIC LICENSE | ||||||
|  copies of this license document, and changing it is allowed as long |                        Version 3, 29 June 2007 | ||||||
|  as the name is changed. |  | ||||||
|  |  | ||||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE |  Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> | ||||||
|    TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION |  Everyone is permitted to copy and distribute verbatim copies | ||||||
|  |  of this license document, but changing it is not allowed. | ||||||
|  |  | ||||||
|   0. You just DO WHAT THE FUCK YOU WANT TO. |  | ||||||
|  |  | ||||||
| This license is commonly known as "WTFPL". |   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. | ||||||
|  |  | ||||||
| ------------------------------------------------------------------------------- |   0. Additional Definitions. | ||||||
|  |  | ||||||
| Creative Commons Legal Code |   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. | ||||||
|  |  | ||||||
| Attribution-ShareAlike 3.0 Unported |   "The Library" refers to a covered work governed by this License, | ||||||
|  | other than an Application or a Combined Work as defined below. | ||||||
|  |  | ||||||
|     CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE |   An "Application" is any work that makes use of an interface provided | ||||||
|     LEGAL SERVICES. DISTRIBUTION OF THIS LICENSE DOES NOT CREATE AN | by the Library, but which is not otherwise based on the Library. | ||||||
|     ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS | Defining a subclass of a class defined by the Library is deemed a mode | ||||||
|     INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES | of using an interface provided by the Library. | ||||||
|     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 | License | ||||||
|  |  | ||||||
| THE WORK (AS DEFINED BELOW) IS PROVIDED UNDER THE TERMS OF THIS CREATIVE | By exercising the Licensed Rights (defined below), You accept and agree | ||||||
| COMMONS PUBLIC LICENSE ("CCPL" OR "LICENSE"). THE WORK IS PROTECTED BY | to be bound by the terms and conditions of this Creative Commons | ||||||
| COPYRIGHT AND/OR OTHER APPLICABLE LAW. ANY USE OF THE WORK OTHER THAN AS | Attribution-ShareAlike 4.0 International Public License ("Public | ||||||
| AUTHORIZED UNDER THIS LICENSE OR COPYRIGHT LAW IS PROHIBITED. | License"). To the extent this Public License may be interpreted as a | ||||||
|  | contract, You are granted the Licensed Rights in consideration of Your | ||||||
| BY EXERCISING ANY RIGHTS TO THE WORK PROVIDED HERE, YOU ACCEPT AND AGREE | acceptance of these terms and conditions, and the Licensor grants You | ||||||
| TO BE BOUND BY THE TERMS OF THIS LICENSE. TO THE EXTENT THIS LICENSE MAY | such rights in consideration of benefits the Licensor receives from | ||||||
| BE CONSIDERED TO BE A CONTRACT, THE LICENSOR GRANTS YOU THE RIGHTS | making the Licensed Material available under these terms and | ||||||
| CONTAINED HERE IN CONSIDERATION OF YOUR ACCEPTANCE OF SUCH TERMS AND | conditions. | ||||||
| CONDITIONS. |  | ||||||
|  |  | ||||||
| 1. Definitions |  | ||||||
|  |  | ||||||
|  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. |  | ||||||
|  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 | Section 1 -- Definitions. | ||||||
|  |  | ||||||
|     Creative Commons is not a party to this License, and makes no warranty |   a. Adapted Material means material subject to Copyright and Similar | ||||||
|     whatsoever in connection with the Work. Creative Commons will not be |      Rights that is derived from or based upon the Licensed Material | ||||||
|     liable to You or any party on any legal theory for any damages |      and in which the Licensed Material is translated, altered, | ||||||
|     whatsoever, including without limitation any general, special, |      arranged, transformed, or otherwise modified in a manner requiring | ||||||
|     incidental or consequential damages arising in connection to this |      permission under the Copyright and Similar Rights held by the | ||||||
|     license. Notwithstanding the foregoing two (2) sentences, if Creative |      Licensor. For purposes of this Public License, where the Licensed | ||||||
|     Commons has expressly identified itself as the Licensor hereunder, it |      Material is a musical work, performance, or sound recording, | ||||||
|     shall have all rights and obligations of Licensor. |      Adapted Material is always produced where the Licensed Material is | ||||||
|  |      synched in timed relation with a moving image. | ||||||
|  |  | ||||||
|     Except for the limited purpose of indicating to the public that the |   b. Adapter's License means the license You apply to Your Copyright | ||||||
|     Work is licensed under the CCPL, Creative Commons does not authorize |      and Similar Rights in Your contributions to Adapted Material in | ||||||
|     the use by either party of the trademark "Creative Commons" or any |      accordance with the terms and conditions of this Public License. | ||||||
|     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/. |   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 | ||||||
|  |      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. | ||||||
|   | |||||||
| @@ -8,4 +8,4 @@ jungle trees mod, and big contributions by RealBadAngel. | |||||||
| Brought together into one mod and made L-systems compatible by Vanessa | Brought together into one mod and made L-systems compatible by Vanessa | ||||||
| Ezekowitz. | Ezekowitz. | ||||||
|  |  | ||||||
| Dependencies: <a href="https://github.com/VanessaE/plantlife">plants_lib</a> and default | Dependencies: <a href="https://forum.minetest.net/viewtopic.php?f=11&t=12999">biome_lib</a> and default | ||||||
|   | |||||||
							
								
								
									
										168
									
								
								biome_defs.lua
									
									
									
									
									
								
							
							
						
						| @@ -24,6 +24,40 @@ moretrees.palm_biome = { | |||||||
| 	max_count = 10, | 	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 = { | moretrees.apple_tree_biome = { | ||||||
| 	surface = "default:dirt_with_grass", | 	surface = "default:dirt_with_grass", | ||||||
| 	avoid_nodes = moretrees.avoidnodes, | 	avoid_nodes = moretrees.avoidnodes, | ||||||
| @@ -90,18 +124,6 @@ moretrees.willow_biome = { | |||||||
| 	max_count = 5, | 	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 = { | moretrees.rubber_tree_biome = { | ||||||
| 	surface = "default:dirt_with_grass", | 	surface = "default:dirt_with_grass", | ||||||
| 	avoid_nodes = moretrees.avoidnodes, | 	avoid_nodes = moretrees.avoidnodes, | ||||||
| @@ -118,18 +140,24 @@ moretrees.rubber_tree_biome = { | |||||||
| } | } | ||||||
|  |  | ||||||
| moretrees.jungletree_biome = { | moretrees.jungletree_biome = { | ||||||
| 	surface = "default:dirt_with_grass", | 	surface = { | ||||||
| 	avoid_nodes = moretrees.avoidnodes, | 		"default:dirt", | ||||||
| 	avoid_radius = 5, | 		"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, | ||||||
| 	seed_diff = 329, | 	seed_diff = 329, | ||||||
| 	min_elevation = -5, | 	min_elevation = 1, | ||||||
| 	max_elevation = 10, | 	near_nodes = {"default:jungletree"}, | ||||||
| 	temp_min = 0.25, | 	near_nodes_size = 6, | ||||||
| 	near_nodes = {"default:water_source"}, | 	near_nodes_vertical = 2, | ||||||
| 	near_nodes_size = 20, | 	near_nodes_count = 1, | ||||||
| 	near_nodes_count = 7, | 	plantlife_limit = -0.9, | ||||||
| 	rarity = 10, |  | ||||||
| 	max_count = 10, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| moretrees.spruce_biome = { | moretrees.spruce_biome = { | ||||||
| @@ -144,7 +172,7 @@ moretrees.spruce_biome = { | |||||||
| 	max_count = 5, | 	max_count = 5, | ||||||
| } | } | ||||||
|  |  | ||||||
| moretrees.pine_biome = { | moretrees.cedar_biome = { | ||||||
| 	surface = "default:dirt_with_grass", | 	surface = "default:dirt_with_grass", | ||||||
| 	avoid_nodes = moretrees.avoidnodes, | 	avoid_nodes = moretrees.avoidnodes, | ||||||
| 	avoid_radius = 10, | 	avoid_radius = 10, | ||||||
| @@ -156,6 +184,97 @@ moretrees.pine_biome = { | |||||||
| 	max_count = 10, | 	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 = { | moretrees.fir_biome = { | ||||||
| 	surface = "default:dirt_with_grass", | 	surface = "default:dirt_with_grass", | ||||||
| 	avoid_nodes = moretrees.avoidnodes, | 	avoid_nodes = moretrees.avoidnodes, | ||||||
| @@ -180,4 +299,3 @@ moretrees.fir_biome_snow = { | |||||||
| 	delete_above = true, | 	delete_above = true, | ||||||
| 	spawn_replace_node = true | 	spawn_replace_node = true | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										283
									
								
								cocos_palm.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,283 @@ | |||||||
|  | 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,14 +3,6 @@ local S = moretrees.intllib | |||||||
| for i in ipairs(moretrees.treelist) do | for i in ipairs(moretrees.treelist) do | ||||||
| 	local treename = moretrees.treelist[i][1] | 	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({ | 	minetest.register_craft({ | ||||||
| 		type = "shapeless", | 		type = "shapeless", | ||||||
| 		output = "moretrees:"..treename.."_planks 4", | 		output = "moretrees:"..treename.."_planks 4", | ||||||
| @@ -19,14 +11,6 @@ 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({ | 	minetest.register_craft({ | ||||||
| 		type = "fuel", | 		type = "fuel", | ||||||
| 		recipe = "moretrees:"..treename.."_sapling", | 		recipe = "moretrees:"..treename.."_sapling", | ||||||
| @@ -63,6 +47,35 @@ minetest.register_craftitem("moretrees:raw_coconut", { | |||||||
| 	on_use = minetest.item_eat(4), | 	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", { | minetest.register_craftitem("moretrees:acorn_muffin_batter", { | ||||||
| 	description = S("Acorn Muffin batter"), | 	description = S("Acorn Muffin batter"), | ||||||
| 	inventory_image = "moretrees_acorn_muffin_batter.png", | 	inventory_image = "moretrees_acorn_muffin_batter.png", | ||||||
| @@ -80,9 +93,9 @@ minetest.register_craftitem("moretrees:spruce_nuts", { | |||||||
| 	on_use = minetest.item_eat(1), | 	on_use = minetest.item_eat(1), | ||||||
| }) | }) | ||||||
|  |  | ||||||
| minetest.register_craftitem("moretrees:pine_nuts", { | minetest.register_craftitem("moretrees:cedar_nuts", { | ||||||
| 	description = S("Roasted Pine Cone Nuts"), | 	description = S("Roasted Cedar Cone Nuts"), | ||||||
| 	inventory_image = "moretrees_pine_nuts.png", | 	inventory_image = "moretrees_cedar_nuts.png", | ||||||
| 	on_use = minetest.item_eat(1), | 	on_use = minetest.item_eat(1), | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -109,6 +122,60 @@ for i in ipairs(moretrees.cutting_tools) do | |||||||
| 	}) | 	}) | ||||||
| end | 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({ | minetest.register_craft({ | ||||||
| 	type = "shapeless", | 	type = "shapeless", | ||||||
| 	output = "moretrees:acorn_muffin_batter", | 	output = "moretrees:acorn_muffin_batter", | ||||||
| @@ -138,8 +205,8 @@ minetest.register_craft({ | |||||||
|  |  | ||||||
| minetest.register_craft({ | minetest.register_craft({ | ||||||
| 	type = "cooking", | 	type = "cooking", | ||||||
| 	output = "moretrees:pine_nuts 4", | 	output = "moretrees:cedar_nuts 4", | ||||||
| 	recipe = "moretrees:pine_cone", | 	recipe = "moretrees:cedar_cone", | ||||||
| }) | }) | ||||||
|  |  | ||||||
| minetest.register_craft({ | minetest.register_craft({ | ||||||
|   | |||||||
							
								
								
									
										750
									
								
								date_palm.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,750 @@ | |||||||
|  | -- 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,25 +6,29 @@ moretrees.enable_apple_tree		= true | |||||||
| moretrees.enable_oak			= true | moretrees.enable_oak			= true | ||||||
| moretrees.enable_sequoia		= true | moretrees.enable_sequoia		= true | ||||||
| moretrees.enable_palm			= true | moretrees.enable_palm			= true | ||||||
| moretrees.enable_pine			= true | moretrees.enable_date_palm		= true | ||||||
|  | moretrees.enable_cedar			= true | ||||||
| moretrees.enable_rubber_tree	= true | moretrees.enable_rubber_tree	= true | ||||||
| moretrees.enable_willow			= true | moretrees.enable_willow			= true | ||||||
| moretrees.enable_acacia			= true |  | ||||||
| moretrees.enable_birch			= true | moretrees.enable_birch			= true | ||||||
| moretrees.enable_spruce			= true | moretrees.enable_spruce			= true | ||||||
| moretrees.enable_jungle_tree	= true | moretrees.enable_jungle_tree	= true | ||||||
| moretrees.enable_fir			= true | moretrees.enable_fir			= true | ||||||
|  | moretrees.enable_poplar			= true | ||||||
| moretrees.enable_beech			= false | moretrees.enable_beech			= false | ||||||
|  |  | ||||||
| -- set this to true to make moretrees spawn saplings at mapgen time instead | -- set this to true to make moretrees spawn saplings at mapgen time instead | ||||||
| -- of fully-grown trees, which will grow into full trees very quickly.  With | -- of fully-grown trees, which will grow into full trees after a very short | ||||||
| -- older versions of plants_lib, doing this will reduce mapgen lag. | -- delay.  This reduces mapgen lag in some situations, and fixes situations | ||||||
|  | -- where the mapgen conflicts with the tree generator. | ||||||
|  |  | ||||||
| moretrees.spawn_saplings		= false | moretrees.spawn_saplings		= true | ||||||
|  |  | ||||||
| -- Set this to true to allow usage of the stairsplus mod in moreblocks | -- 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 | ||||||
|  |  | ||||||
| moretrees.enable_stairsplus		= true | moretrees.enable_stairs			= true | ||||||
|  |  | ||||||
| -- Set this to true if you want the plantlike drawtype for leaves, which  | -- Set this to true if you want the plantlike drawtype for leaves, which  | ||||||
| -- improves some peoples' framerates without resorting to making leaf nodes opaque. | -- improves some peoples' framerates without resorting to making leaf nodes opaque. | ||||||
| @@ -32,42 +36,21 @@ moretrees.enable_stairsplus		= true | |||||||
|  |  | ||||||
| moretrees.plantlike_leaves		= false | 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 | -- Enable this if you want moretrees to redefine default apples so that they | ||||||
| -- fall when leaves decay/are dug. | -- fall when leaves decay/are dug. | ||||||
|  |  | ||||||
| moretrees.enable_redefine_apple		= true | moretrees.enable_redefine_apple		= true | ||||||
|  |  | ||||||
| -- various settings to configure default and default-jungle leaf decay. | -- 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. | ||||||
|  |  | ||||||
| moretrees.leafdecay_delay		= 2 | moretrees.leafdecay_delay		= 2 | ||||||
| moretrees.leafdecay_chance		= 100 | moretrees.leafdecay_chance		= 5 | ||||||
| moretrees.leafdecay_radius		= 5 | 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 | -- Change these settings if you want default trees to be gradually cut down | ||||||
| -- above the elevation where firs normally generate. | -- above the elevation where firs normally generate. | ||||||
| @@ -76,6 +59,67 @@ moretrees.firs_remove_default_trees	= false | |||||||
| moretrees.firs_remove_interval		= 2 | moretrees.firs_remove_interval		= 2 | ||||||
| moretrees.firs_remove_chance		= 150 | 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 | -- Sapling settings | ||||||
|  |  | ||||||
| moretrees.sapling_interval		= 500 | moretrees.sapling_interval		= 500 | ||||||
|   | |||||||
| @@ -1,5 +1,8 @@ | |||||||
| default | default | ||||||
| plants_lib | biome_lib | ||||||
|  | vessels | ||||||
|  | stairs? | ||||||
| moreblocks? | moreblocks? | ||||||
| intllib? | intllib? | ||||||
|  | farming? | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								description.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | This mod adds a whole bunch of new types of trees to the game | ||||||
							
								
								
									
										137
									
								
								init.lua
									
									
									
									
									
								
							
							
						
						| @@ -2,19 +2,16 @@ | |||||||
| -- | -- | ||||||
| -- This mod adds more types of trees to the game | -- This mod adds more types of trees to the game | ||||||
| -- | -- | ||||||
| -- Some of the node definitions and textures came from cisoun's conifers mod  | -- Some of the node definitions and textures came from cisoun's conifers mod | ||||||
| -- and bas080's jungle trees mod. | -- and bas080's jungle trees mod. | ||||||
| -- | -- | ||||||
| -- Brought together into one mod and made L-systems compatible by Vanessa | -- Brought together into one mod and made L-systems compatible by Vanessa | ||||||
| -- Ezekowitz. | -- Ezekowitz. | ||||||
| -- | -- | ||||||
| -- Firs and Jungle tree axioms/rules by Vanessa Ezekowitz, with the  | -- Firs and Jungle tree axioms/rules by Vanessa Dannenberg, with the | ||||||
| -- latter having been tweaked by RealBadAngel, most other axioms/rules written | -- latter having been tweaked by RealBadAngel, most other axioms/rules written | ||||||
| -- by RealBadAngel. | -- 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 = {} | moretrees = {} | ||||||
|  |  | ||||||
| @@ -25,21 +22,8 @@ local modpath=minetest.get_modpath("moretrees") | |||||||
|  |  | ||||||
| dofile(modpath.."/default_settings.txt") | dofile(modpath.."/default_settings.txt") | ||||||
|  |  | ||||||
| if io.open(worldpath.."/moretrees_settings.txt","r") == nil then | if io.open(worldpath.."/moretrees_settings.txt","r") then | ||||||
|  | 	io.close() | ||||||
| 	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") | 	dofile(worldpath.."/moretrees_settings.txt") | ||||||
| end | end | ||||||
|  |  | ||||||
| @@ -52,9 +36,21 @@ else | |||||||
| end | end | ||||||
| moretrees.intllib = S | moretrees.intllib = S | ||||||
|  |  | ||||||
|  | -- clone node | ||||||
|  |  | ||||||
|  | function moretrees.clone_node(name) | ||||||
|  | 	local node2 = {} | ||||||
|  | 	local node = minetest.registered_nodes[name] | ||||||
|  | 	for k,v in pairs(node) do | ||||||
|  | 		node2[k]=v | ||||||
|  | 	end | ||||||
|  | 	return node2 | ||||||
|  | end | ||||||
|  |  | ||||||
| -- infinite stacks checking | -- infinite stacks checking | ||||||
|  |  | ||||||
| if minetest.get_modpath("unified_inventory") or not minetest.setting_getbool("creative_mode") then | if minetest.get_modpath("unified_inventory") or not | ||||||
|  | 		minetest.settings:get_bool("creative_mode") then | ||||||
| 	moretrees.expect_infinite_stacks = false | 	moretrees.expect_infinite_stacks = false | ||||||
| else | else | ||||||
| 	moretrees.expect_infinite_stacks = true | 	moretrees.expect_infinite_stacks = true | ||||||
| @@ -76,10 +72,11 @@ moretrees.cutting_tools = { | |||||||
|  |  | ||||||
| dofile(modpath.."/tree_models.lua") | dofile(modpath.."/tree_models.lua") | ||||||
| dofile(modpath.."/node_defs.lua") | dofile(modpath.."/node_defs.lua") | ||||||
|  | dofile(modpath.."/date_palm.lua") | ||||||
|  | dofile(modpath.."/cocos_palm.lua") | ||||||
| dofile(modpath.."/biome_defs.lua") | dofile(modpath.."/biome_defs.lua") | ||||||
| dofile(modpath.."/saplings.lua") | dofile(modpath.."/saplings.lua") | ||||||
| dofile(modpath.."/crafts.lua") | dofile(modpath.."/crafts.lua") | ||||||
| dofile(modpath.."/leafdecay.lua") |  | ||||||
|  |  | ||||||
| -- tree spawning setup | -- tree spawning setup | ||||||
|  |  | ||||||
| @@ -89,92 +86,104 @@ if moretrees.spawn_saplings then | |||||||
| 	moretrees.spawn_oak_object = "moretrees:oak_sapling_ongen" | 	moretrees.spawn_oak_object = "moretrees:oak_sapling_ongen" | ||||||
| 	moretrees.spawn_sequoia_object = "moretrees:sequoia_sapling_ongen" | 	moretrees.spawn_sequoia_object = "moretrees:sequoia_sapling_ongen" | ||||||
| 	moretrees.spawn_palm_object = "moretrees:palm_sapling_ongen" | 	moretrees.spawn_palm_object = "moretrees:palm_sapling_ongen" | ||||||
| 	moretrees.spawn_pine_object = "moretrees:pine_sapling_ongen" | 	moretrees.spawn_date_palm_object = "moretrees:date_palm_sapling_ongen" | ||||||
|  | 	moretrees.spawn_cedar_object = "moretrees:cedar_sapling_ongen" | ||||||
| 	moretrees.spawn_rubber_tree_object = "moretrees:rubber_tree_sapling_ongen" | 	moretrees.spawn_rubber_tree_object = "moretrees:rubber_tree_sapling_ongen" | ||||||
| 	moretrees.spawn_willow_object = "moretrees:willow_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_birch_object = "moretrees:birch_sapling_ongen" | ||||||
| 	moretrees.spawn_spruce_object = "moretrees:spruce_sapling_ongen" | 	moretrees.spawn_spruce_object = "moretrees:spruce_sapling_ongen" | ||||||
| 	moretrees.spawn_jungletree_object = "moretrees:jungletree_sapling_ongen" | 	moretrees.spawn_jungletree_object = "moretrees:jungletree_sapling_ongen" | ||||||
| 	moretrees.spawn_fir_object = "moretrees:fir_sapling_ongen" | 	moretrees.spawn_fir_object = "moretrees:fir_sapling_ongen" | ||||||
| 	moretrees.spawn_fir_snow_object = "snow:sapling_pine" | 	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 | else | ||||||
| 	moretrees.spawn_beech_object = moretrees.beech_model | 	moretrees.spawn_beech_object = moretrees.beech_model | ||||||
| 	moretrees.spawn_apple_tree_object = moretrees.apple_tree_model | 	moretrees.spawn_apple_tree_object = moretrees.apple_tree_model | ||||||
| 	moretrees.spawn_oak_object = moretrees.oak_model | 	moretrees.spawn_oak_object = moretrees.oak_model | ||||||
| 	moretrees.spawn_sequoia_object = moretrees.sequoia_model | 	moretrees.spawn_sequoia_object = moretrees.sequoia_model | ||||||
| 	moretrees.spawn_palm_object = moretrees.palm_model | 	moretrees.spawn_palm_object = moretrees.palm_model | ||||||
| 	moretrees.spawn_pine_object = moretrees.pine_model | 	moretrees.spawn_date_palm_object = moretrees.date_palm_model | ||||||
|  | 	moretrees.spawn_cedar_object = moretrees.cedar_model | ||||||
| 	moretrees.spawn_rubber_tree_object = moretrees.rubber_tree_model | 	moretrees.spawn_rubber_tree_object = moretrees.rubber_tree_model | ||||||
| 	moretrees.spawn_willow_object = moretrees.willow_model | 	moretrees.spawn_willow_object = moretrees.willow_model | ||||||
| 	moretrees.spawn_acacia_object = moretrees.acacia_model | 	moretrees.spawn_birch_object = "moretrees.grow_birch" | ||||||
| 	moretrees.spawn_birch_object = "moretrees:grow_birch" | 	moretrees.spawn_spruce_object = "moretrees.grow_spruce" | ||||||
| 	moretrees.spawn_spruce_object = "moretrees:grow_spruce" | 	moretrees.spawn_jungletree_object = "moretrees.grow_jungletree" | ||||||
| 	moretrees.spawn_jungletree_object = "moretrees:grow_jungletree" | 	moretrees.spawn_fir_object = "moretrees.grow_fir" | ||||||
| 	moretrees.spawn_fir_object = "moretrees:grow_fir" | 	moretrees.spawn_fir_snow_object = "moretrees.grow_fir_snow" | ||||||
| 	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 | ||||||
| end | end | ||||||
|  |  | ||||||
|  |  | ||||||
| if moretrees.enable_beech then | if moretrees.enable_beech then | ||||||
| 	plantslib:register_generate_plant(moretrees.beech_biome, moretrees.spawn_beech_object) | 	biome_lib:register_generate_plant(moretrees.beech_biome, moretrees.spawn_beech_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_apple_tree then | if moretrees.enable_apple_tree then | ||||||
| 	plantslib:register_generate_plant(moretrees.apple_tree_biome, moretrees.spawn_apple_tree_object) | 	biome_lib:register_generate_plant(moretrees.apple_tree_biome, moretrees.spawn_apple_tree_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_oak then | if moretrees.enable_oak then | ||||||
| 	plantslib:register_generate_plant(moretrees.oak_biome, moretrees.spawn_oak_object) | 	biome_lib:register_generate_plant(moretrees.oak_biome, moretrees.spawn_oak_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_sequoia then | if moretrees.enable_sequoia then | ||||||
| 	plantslib:register_generate_plant(moretrees.sequoia_biome, moretrees.spawn_sequoia_object) | 	biome_lib:register_generate_plant(moretrees.sequoia_biome, moretrees.spawn_sequoia_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_palm then | if moretrees.enable_palm then | ||||||
| 	plantslib:register_generate_plant(moretrees.palm_biome, moretrees.spawn_palm_object) | 	biome_lib:register_generate_plant(moretrees.palm_biome, moretrees.spawn_palm_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_pine then | if moretrees.enable_date_palm then | ||||||
| 	plantslib:register_generate_plant(moretrees.pine_biome, moretrees.spawn_pine_object) | 	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) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_rubber_tree then | if moretrees.enable_rubber_tree then | ||||||
| 	plantslib:register_generate_plant(moretrees.rubber_tree_biome, moretrees.spawn_rubber_tree_object) | 	biome_lib:register_generate_plant(moretrees.rubber_tree_biome, moretrees.spawn_rubber_tree_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_willow then | if moretrees.enable_willow then | ||||||
| 	plantslib:register_generate_plant(moretrees.willow_biome, moretrees.spawn_willow_object) | 	biome_lib: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 | end | ||||||
|  |  | ||||||
| if moretrees.enable_birch then | if moretrees.enable_birch then | ||||||
| 	plantslib:register_generate_plant(moretrees.birch_biome, moretrees.spawn_birch_object) | 	biome_lib:register_generate_plant(moretrees.birch_biome, moretrees.spawn_birch_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_spruce then | if moretrees.enable_spruce then | ||||||
| 	plantslib:register_generate_plant(moretrees.spruce_biome, moretrees.spawn_spruce_object) | 	biome_lib:register_generate_plant(moretrees.spruce_biome, moretrees.spawn_spruce_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_jungle_tree then | if moretrees.enable_jungle_tree then | ||||||
| 	plantslib:register_generate_plant(moretrees.jungletree_biome, moretrees.spawn_jungletree_object) | 	biome_lib:register_generate_plant(moretrees.jungletree_biome, moretrees.spawn_jungletree_object) | ||||||
| end | end | ||||||
|  |  | ||||||
| if moretrees.enable_fir then | if moretrees.enable_fir then | ||||||
| 	plantslib:register_generate_plant(moretrees.fir_biome, moretrees.spawn_fir_object) | 	biome_lib:register_generate_plant(moretrees.fir_biome, moretrees.spawn_fir_object) | ||||||
| 	if minetest.get_modpath("snow") then | 	if minetest.get_modpath("snow") then | ||||||
| 		plantslib:register_generate_plant(moretrees.fir_biome_snow, moretrees.spawn_fir_snow_object) | 		biome_lib:register_generate_plant(moretrees.fir_biome_snow, moretrees.spawn_fir_snow_object) | ||||||
| 	end | 	end | ||||||
| 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 | -- Code to spawn a birch tree | ||||||
|  |  | ||||||
| function moretrees:grow_birch(pos) | function moretrees.grow_birch(pos) | ||||||
| 	minetest.remove_node(pos) | 	minetest.swap_node(pos, biome_lib.air) | ||||||
| 	if math.random(1,2) == 1 then | 	if math.random(1,2) == 1 then | ||||||
| 		minetest.spawn_tree(pos, moretrees.birch_model1) | 		minetest.spawn_tree(pos, moretrees.birch_model1) | ||||||
| 	else | 	else | ||||||
| @@ -184,8 +193,8 @@ end | |||||||
|  |  | ||||||
| -- Code to spawn a spruce tree | -- Code to spawn a spruce tree | ||||||
|  |  | ||||||
| function moretrees:grow_spruce(pos) | function moretrees.grow_spruce(pos) | ||||||
| 	minetest.remove_node(pos) | 	minetest.swap_node(pos, biome_lib.air) | ||||||
| 	if math.random(1,2) == 1 then | 	if math.random(1,2) == 1 then | ||||||
| 		minetest.spawn_tree(pos, moretrees.spruce_model1) | 		minetest.spawn_tree(pos, moretrees.spruce_model1) | ||||||
| 	else | 	else | ||||||
| @@ -209,12 +218,12 @@ moretrees.ct_rules_b1 = "[-FBf][+FBf]" | |||||||
| moretrees.ct_rules_a2 = "FF[FF][&&-FBF][&&+FBF][&&---FBF][&&+++FBF]F/A" | moretrees.ct_rules_a2 = "FF[FF][&&-FBF][&&+FBF][&&---FBF][&&+++FBF]F/A" | ||||||
| moretrees.ct_rules_b2 = "[-fB][+fB]" | moretrees.ct_rules_b2 = "[-fB][+fB]" | ||||||
|  |  | ||||||
| function moretrees:grow_jungletree(pos) | function moretrees.grow_jungletree(pos) | ||||||
| 	local r1 = math.random(2) | 	local r1 = math.random(2) | ||||||
| 	local r2 = math.random(3) | 	local r2 = math.random(3) | ||||||
| 	if r1 == 1 then | 	if r1 == 1 then | ||||||
| 		moretrees.jungletree_model.leaves2 = "moretrees:jungletree_leaves_red" | 		moretrees.jungletree_model.leaves2 = "moretrees:jungletree_leaves_red" | ||||||
| 	else  | 	else | ||||||
| 		moretrees.jungletree_model.leaves2 = "moretrees:jungletree_leaves_yellow" | 		moretrees.jungletree_model.leaves2 = "moretrees:jungletree_leaves_yellow" | ||||||
| 	end | 	end | ||||||
| 	moretrees.jungletree_model.leaves2_chance = math.random(25, 75) | 	moretrees.jungletree_model.leaves2_chance = math.random(25, 75) | ||||||
| @@ -239,17 +248,17 @@ function moretrees:grow_jungletree(pos) | |||||||
| 		moretrees.jungletree_model.rules_b = moretrees.jt_rules_b2 | 		moretrees.jungletree_model.rules_b = moretrees.jt_rules_b2 | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
| 	minetest.remove_node(pos) | 	minetest.swap_node(pos, biome_lib.air) | ||||||
| 	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") | 	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 | 	for leaf in ipairs(leaves) do | ||||||
| 			minetest.remove_node(leaves[leaf]) | 			minetest.swap_node(leaves[leaf], biome_lib.air) | ||||||
| 	end | 	end | ||||||
| 	minetest.spawn_tree(pos, moretrees.jungletree_model) | 	minetest.spawn_tree(pos, moretrees.jungletree_model) | ||||||
| end | end | ||||||
|  |  | ||||||
| -- code to spawn fir trees | -- code to spawn fir trees | ||||||
|  |  | ||||||
| function moretrees:grow_fir(pos) | function moretrees.grow_fir(pos) | ||||||
| 	if math.random(2) == 1 then | 	if math.random(2) == 1 then | ||||||
| 		moretrees.fir_model.leaves="moretrees:fir_leaves" | 		moretrees.fir_model.leaves="moretrees:fir_leaves" | ||||||
| 	else | 	else | ||||||
| @@ -266,17 +275,17 @@ function moretrees:grow_fir(pos) | |||||||
| 	moretrees.fir_model.iterations = 7 | 	moretrees.fir_model.iterations = 7 | ||||||
| 	moretrees.fir_model.random_level = 5 | 	moretrees.fir_model.random_level = 5 | ||||||
|  |  | ||||||
| 	minetest.remove_node(pos) | 	minetest.swap_node(pos, biome_lib.air) | ||||||
| 	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") | 	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 | 	for leaf in ipairs(leaves) do | ||||||
| 			minetest.remove_node(leaves[leaf]) | 		minetest.swap_node(leaves[leaf], biome_lib.air) | ||||||
| 	end | 	end | ||||||
| 	minetest.spawn_tree(pos,moretrees.fir_model) | 	minetest.spawn_tree(pos,moretrees.fir_model) | ||||||
| end | end | ||||||
|  |  | ||||||
| -- same thing, but a smaller version that grows only in snow biomes | -- 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 | 	if math.random(2) == 1 then | ||||||
| 		moretrees.fir_model.leaves="moretrees:fir_leaves" | 		moretrees.fir_model.leaves="moretrees:fir_leaves" | ||||||
| 	else | 	else | ||||||
| @@ -293,10 +302,10 @@ function moretrees:grow_fir_snow(pos) | |||||||
| 	moretrees.fir_model.iterations = 2 | 	moretrees.fir_model.iterations = 2 | ||||||
| 	moretrees.fir_model.random_level = 2 | 	moretrees.fir_model.random_level = 2 | ||||||
|  |  | ||||||
| 	minetest.remove_node(pos) | 	minetest.swap_node(pos, biome_lib.air) | ||||||
| 	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") | 	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 | 	for leaf in ipairs(leaves) do | ||||||
| 			minetest.remove_node(leaves[leaf]) | 			minetest.swap_node(leaves[leaf], biome_lib.air) | ||||||
| 	end | 	end | ||||||
| 	minetest.spawn_tree(pos,moretrees.fir_model) | 	minetest.spawn_tree(pos,moretrees.fir_model) | ||||||
| end | end | ||||||
|   | |||||||
							
								
								
									
										117
									
								
								leafdecay.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,117 +0,0 @@ | |||||||
| -- 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,7 +65,6 @@ Spruce Cone = Fichtenzapfen | |||||||
| Pine Cone = Kiefernzapfen | Pine Cone = Kiefernzapfen | ||||||
| Fir Cone = Tannenzapfen | Fir Cone = Tannenzapfen | ||||||
| Jungle Sapling = Tropenbaumsetzling | Jungle Sapling = Tropenbaumsetzling | ||||||
| Jungle Tree Leaves (Green) = Tropenbaumlaub (gruen) |  | ||||||
| Jungle Tree Leaves (Yellow) = Tropenbaumlaub (gelb) | Jungle Tree Leaves (Yellow) = Tropenbaumlaub (gelb) | ||||||
| Jungle Tree Leaves (Red) = Tropenbaumlaub (rot) | Jungle Tree Leaves (Red) = Tropenbaumlaub (rot) | ||||||
| Douglas Fir Leaves (Bright) = Douglasiennadeln (breit) | Douglas Fir Leaves (Bright) = Douglasiennadeln (breit) | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								locale/es.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,100 @@ | |||||||
|  | # 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 | ||||||
							
								
								
									
										74
									
								
								locale/fr.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,74 @@ | |||||||
|  | # 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,7 +65,6 @@ Spruce Cone = | |||||||
| Pine Cone =  | Pine Cone =  | ||||||
| Fir Cone =  | Fir Cone =  | ||||||
| Jungle Sapling =  | Jungle Sapling =  | ||||||
| Jungle Tree Leaves (Green) =  |  | ||||||
| Jungle Tree Leaves (Yellow) =  | Jungle Tree Leaves (Yellow) =  | ||||||
| Jungle Tree Leaves (Red) =  | Jungle Tree Leaves (Red) =  | ||||||
| Douglas Fir Leaves (Bright) =  | Douglas Fir Leaves (Bright) =  | ||||||
|   | |||||||
							
								
								
									
										419
									
								
								node_defs.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,21 +1,23 @@ | |||||||
| local S = moretrees.intllib | local S = moretrees.intllib | ||||||
|  |  | ||||||
| moretrees.avoidnodes = {} | moretrees.avoidnodes = {} | ||||||
| 	 |  | ||||||
| moretrees.treelist = { | moretrees.treelist = { | ||||||
| 	{"beech",		"Beech Tree"}, | 	{"beech",			"Beech Tree"}, | ||||||
| 	{"apple_tree",	"Apple Tree"}, | 	{"apple_tree",		"Apple Tree"}, | ||||||
| 	{"oak",			"Oak Tree",			"acorn",		"Acorn",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | 	{"oak",				"Oak Tree",			"acorn",					"Acorn",			{-0.2, -0.5, -0.2, 0.2, 0, 0.2},	0.8 }, | ||||||
| 	{"sequoia",		"Giant Sequoia"}, | 	{"sequoia",			"Giant Sequoia"}, | ||||||
| 	{"birch",		"Birch Tree"}, | 	{"birch",			"Birch Tree"}, | ||||||
| 	{"palm",		"Palm Tree",		"coconut",		"Coconut",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 1.0 }, | 	{"palm",			"Palm Tree",		"palm_fruit_trunk_gen",		"Palm Tree",		{-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 }, | 	{"date_palm",		"Date Palm Tree",	"date_palm_fruit_trunk",	"Date Palm Tree",	{0, 0, 0, 0, 0, 0},	0.0 }, | ||||||
| 	{"pine",		"Pine Tree",		"pine_cone",	"Pine Cone",	{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | 	{"spruce",			"Spruce Tree",		"spruce_cone",				"Spruce Cone",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2},	0.8 }, | ||||||
| 	{"willow",		"Willow Tree"}, | 	{"cedar",			"Cedar Tree",		"cedar_cone",				"Cedar Cone",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2},	0.8 }, | ||||||
| 	{"acacia",		"Acacia Tree"}, | 	{"poplar",			"Poplar Tree"}, | ||||||
| 	{"rubber_tree",	"Rubber Tree"}, | 	{"poplar_small",	"Poplar Tree"}, | ||||||
| 	{"jungletree",	"Jungle Tree"}, | 	{"willow",			"Willow Tree"}, | ||||||
| 	{"fir",			"Douglas Fir",		"fir_cone",		"Fir Cone",		{-0.2, -0.5, -0.2, 0.2, 0, 0.2}, 0.8 }, | 	{"rubber_tree",		"Rubber 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 } | local dirs1 = { 21, 20, 23, 22, 21 } | ||||||
| @@ -25,46 +27,28 @@ local dirs3 = { 14, 11, 16, 5, 14 } | |||||||
| local moretrees_new_leaves_drawtype = "allfaces_optional" | local moretrees_new_leaves_drawtype = "allfaces_optional" | ||||||
| local moretrees_plantlike_leaves_visual_scale = 1 | local moretrees_plantlike_leaves_visual_scale = 1 | ||||||
|  |  | ||||||
| if moretrees.plantlike_leaves then  | if moretrees.plantlike_leaves then | ||||||
| 	moretrees_new_leaves_drawtype = "plantlike" | 	moretrees_new_leaves_drawtype = "plantlike" | ||||||
| 	moretrees_plantlike_leaves_visual_scale = 1.189 | 	moretrees_plantlike_leaves_visual_scale = math.sqrt(2) | ||||||
| end | end | ||||||
|  |  | ||||||
| -- redefine default leaves to handle plantlike and/or leaf decay options | -- 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 | if moretrees.plantlike_leaves then | ||||||
| 	minetest.override_item("default:leaves", { | 	minetest.override_item("default:leaves", { | ||||||
| 		inventory_image = minetest.inventorycube("default_leaves.png"), | 		inventory_image = minetest.inventorycube("default_leaves.png"), | ||||||
| 		drawtype = "plantlike", | 		drawtype = "plantlike", | ||||||
| 		visual_scale = 1.189 | 		visual_scale = math.sqrt(2) | ||||||
| 	}) |  | ||||||
| else |  | ||||||
| 	minetest.override_item("default:leaves", { |  | ||||||
| 		waving = 1 |  | ||||||
| 	}) | 	}) | ||||||
| end | end | ||||||
|  |  | ||||||
| -- redefine default jungle leaves for same | -- 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 | if moretrees.plantlike_leaves then | ||||||
| 	minetest.override_item("default:jungleleaves", { | 	minetest.override_item("default:jungleleaves", { | ||||||
| 		inventory_image = minetest.inventorycube("default_jungleleaves.png"), | 		inventory_image = minetest.inventorycube("default_jungleleaves.png"), | ||||||
| 		drawtype = "plantlike", | 		drawtype = "plantlike", | ||||||
| 		visual_scale = 1.189 | 		visual_scale = math.sqrt(2) | ||||||
| 	}) |  | ||||||
| else |  | ||||||
| 	minetest.override_item("default:jungleleaves", { |  | ||||||
| 		waving = 1 |  | ||||||
| 	}) | 	}) | ||||||
| end | end | ||||||
|  |  | ||||||
| @@ -76,7 +60,27 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 	local selbox = moretrees.treelist[i][5] | 	local selbox = moretrees.treelist[i][5] | ||||||
| 	local vscale = moretrees.treelist[i][6] | 	local vscale = moretrees.treelist[i][6] | ||||||
|  |  | ||||||
| 	if treename ~= "jungletree" then -- the default game provides jungle tree trunk/planks nodes. | 	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" | ||||||
|  |  | ||||||
| 		minetest.register_node("moretrees:"..treename.."_trunk", { | 		minetest.register_node("moretrees:"..treename.."_trunk", { | ||||||
| 			description = S(treedesc.." Trunk"), | 			description = S(treedesc.." Trunk"), | ||||||
| @@ -86,7 +90,7 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 				"moretrees_"..treename.."_trunk.png" | 				"moretrees_"..treename.."_trunk.png" | ||||||
| 			}, | 			}, | ||||||
| 			paramtype2 = "facedir", | 			paramtype2 = "facedir", | ||||||
| 			is_ground_content = true, | 			is_ground_content = false, | ||||||
| 			groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | 			groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | ||||||
| 			sounds = default.node_sound_wood_defaults(), | 			sounds = default.node_sound_wood_defaults(), | ||||||
| 			on_place = minetest.rotate_node, | 			on_place = minetest.rotate_node, | ||||||
| @@ -95,7 +99,7 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 		minetest.register_node("moretrees:"..treename.."_planks", { | 		minetest.register_node("moretrees:"..treename.."_planks", { | ||||||
| 			description = S(treedesc.." Planks"), | 			description = S(treedesc.." Planks"), | ||||||
| 			tiles = {"moretrees_"..treename.."_wood.png"}, | 			tiles = {"moretrees_"..treename.."_wood.png"}, | ||||||
| 			is_ground_content = true, | 			is_ground_content = false, | ||||||
| 			groups = {snappy=1,choppy=2,oddly_breakable_by_hand=2,flammable=3,wood=1}, | 			groups = {snappy=1,choppy=2,oddly_breakable_by_hand=2,flammable=3,wood=1}, | ||||||
| 			sounds = default.node_sound_wood_defaults(), | 			sounds = default.node_sound_wood_defaults(), | ||||||
| 		}) | 		}) | ||||||
| @@ -103,11 +107,12 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 		minetest.register_node("moretrees:"..treename.."_sapling", { | 		minetest.register_node("moretrees:"..treename.."_sapling", { | ||||||
| 			description = S(treedesc.." Sapling"), | 			description = S(treedesc.." Sapling"), | ||||||
| 			drawtype = "plantlike", | 			drawtype = "plantlike", | ||||||
| 			tiles = {"moretrees_"..treename.."_sapling.png"}, | 			tiles = {saptex}, | ||||||
| 			inventory_image = "moretrees_"..treename.."_sapling.png", | 			inventory_image = saptex, | ||||||
| 			paramtype = "light", | 			paramtype = "light", | ||||||
| 			paramtype2 = "waving", | 			paramtype2 = "waving", | ||||||
| 			walkable = false, | 			walkable = false, | ||||||
|  | 			is_ground_content = true, | ||||||
| 			selection_box = { | 			selection_box = { | ||||||
| 				type = "fixed", | 				type = "fixed", | ||||||
| 				fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | 				fixed = {-0.3, -0.5, -0.3, 0.3, 0.35, 0.3} | ||||||
| @@ -115,34 +120,6 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 			groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,sapling=1}, | 			groups = {snappy=2,dig_immediate=3,flammable=2,attached_node=1,sapling=1}, | ||||||
| 			sounds = default.node_sound_defaults(), | 			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", |  | ||||||
| 			paramtype2 = "waving", |  | ||||||
| 			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_leaves_inventory_image = nil | ||||||
| 		local moretrees_new_leaves_waving = nil | 		local moretrees_new_leaves_waving = nil | ||||||
| @@ -161,7 +138,8 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 			tiles = { "moretrees_"..treename.."_leaves.png" }, | 			tiles = { "moretrees_"..treename.."_leaves.png" }, | ||||||
| 			inventory_image = moretrees_leaves_inventory_image, | 			inventory_image = moretrees_leaves_inventory_image, | ||||||
| 			paramtype = "light", | 			paramtype = "light", | ||||||
| 			groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1}, | 			is_ground_content = false, | ||||||
|  | 			groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 1}, | ||||||
| 			sounds = default.node_sound_leaves_defaults(), | 			sounds = default.node_sound_leaves_defaults(), | ||||||
|  |  | ||||||
| 			drop = { | 			drop = { | ||||||
| @@ -173,42 +151,88 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 			}, | 			}, | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
| 		if minetest.get_modpath("moreblocks") and moretrees.enable_stairsplus then | 		if moretrees.enable_stairs then | ||||||
|  | 			if minetest.get_modpath("moreblocks") then | ||||||
|  |  | ||||||
| --			stairsplus:register_all(modname, subname, recipeitem, {fields}) | 	--			stairsplus:register_all(modname, subname, recipeitem, {fields}) | ||||||
|  |  | ||||||
| 			stairsplus:register_all( | 				stairsplus:register_all( | ||||||
| 				"moretrees", | 					"moretrees", | ||||||
| 				treename.."_trunk", | 					treename.."_trunk", | ||||||
| 				"moretrees:"..treename.."_trunk",  | 					"moretrees:"..treename.."_trunk", | ||||||
| 				{ | 					{ | ||||||
| 					groups = { snappy=1, choppy=2, oddly_breakable_by_hand=1, flammable=2, not_in_creative_inventory=1 }, | 						groups = { snappy=1, choppy=2, oddly_breakable_by_hand=1, flammable=2, not_in_creative_inventory=1 }, | ||||||
| 					tiles =	{ | 						tiles =	{ | ||||||
| 						"moretrees_"..treename.."_trunk_top.png", | 							"moretrees_"..treename.."_trunk_top.png", | ||||||
|  | 							"moretrees_"..treename.."_trunk_top.png", | ||||||
|  | 							"moretrees_"..treename.."_trunk.png" | ||||||
|  | 						}, | ||||||
|  | 						description = S(treedesc.." Trunk"), | ||||||
|  | 						drop = treename.."_trunk", | ||||||
|  | 					} | ||||||
|  | 				) | ||||||
|  |  | ||||||
|  | 				stairsplus:register_all( | ||||||
|  | 					"moretrees", | ||||||
|  | 					treename.."_planks", | ||||||
|  | 					"moretrees:"..treename.."_planks", | ||||||
|  | 					{ | ||||||
|  | 						groups = { snappy=1, choppy=2, oddly_breakable_by_hand=2, flammable=3, not_in_creative_inventory=1 }, | ||||||
|  | 						tiles = { "moretrees_"..treename.."_wood.png" }, | ||||||
|  | 						description = S(treedesc.." Planks"), | ||||||
|  | 						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_top.png", | ||||||
| 						"moretrees_"..treename.."_trunk.png" | 						"moretrees_"..treename.."_trunk.png" | ||||||
| 					}, | 					}, | ||||||
| 					description = S(treedesc.." Trunk"), | 					S(treedesc.." Trunk Stair"), | ||||||
| 					drop = treename.."_trunk", | 					S(treedesc.." Trunk Slab"), | ||||||
| 				} | 					default.node_sound_wood_defaults() | ||||||
| 			) | 				) | ||||||
|  |  | ||||||
| 			stairsplus:register_all( | 				stairs.register_stair_and_slab( | ||||||
| 				"moretrees", | 					"moretrees_"..treename.."_planks", | ||||||
| 				treename.."_planks", | 					"moretrees:"..treename.."_planks", | ||||||
| 				"moretrees:"..treename.."_planks", | 					{ snappy=1, choppy=2, oddly_breakable_by_hand=2, flammable=3 }, | ||||||
| 				{ | 					{ "moretrees_"..treename.."_wood.png" }, | ||||||
|  					groups = { snappy=1, choppy=2, oddly_breakable_by_hand=2, flammable=3, not_in_creative_inventory=1 }, | 					S(treedesc.." Planks Stair"), | ||||||
| 					tiles = { "moretrees_"..treename.."_wood.png" }, | 					S(treedesc.." Planks Slab"), | ||||||
| 					description = S(treedesc.." Planks"), | 					default.node_sound_wood_defaults() | ||||||
| 					drop = treename.."_planks", | 				) | ||||||
| 				} |  | ||||||
| 			) | 			end | ||||||
| 		end | 		end | ||||||
| 	end | 	end | ||||||
|  |  | ||||||
|  | 	minetest.register_node("moretrees:"..treename.."_sapling_ongen", { | ||||||
|  | 		description = S(treedesc.." Sapling (on-generated)"), | ||||||
|  | 		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,not_in_creative_inventory=1,sapling=1}, | ||||||
|  | 		sounds = default.node_sound_defaults(), | ||||||
|  | 		drop = "moretrees:"..treename.."_sapling" | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	local fruitname = nil | ||||||
| 	if fruit then | 	if fruit then | ||||||
| 		minetest.register_node("moretrees:"..fruit, { | 		fruitname = "moretrees:"..fruit | ||||||
|  | 		minetest.register_node(fruitname, { | ||||||
| 			description = S(fruitdesc), | 			description = S(fruitdesc), | ||||||
| 			drawtype = "plantlike", | 			drawtype = "plantlike", | ||||||
| 			tiles = { "moretrees_"..fruit..".png" }, | 			tiles = { "moretrees_"..fruit..".png" }, | ||||||
| @@ -217,16 +241,27 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 			visual_scale = vscale, | 			visual_scale = vscale, | ||||||
| 			paramtype = "light", | 			paramtype = "light", | ||||||
| 			sunlight_propagates = true, | 			sunlight_propagates = true, | ||||||
|  | 			is_ground_content = false, | ||||||
| 			walkable = false, | 			walkable = false, | ||||||
| 			selection_box = { | 			selection_box = { | ||||||
| 				type = "fixed", | 				type = "fixed", | ||||||
| 					fixed = selbox | 					fixed = selbox | ||||||
| 				}, | 				}, | ||||||
| 			groups = {fleshy=3,dig_immediate=3,flammable=2, attached_node=1}, | 			groups = {fleshy=3,dig_immediate=3,flammable=2, attached_node=1, leafdecay = 1, leafdecay_drop = 1}, | ||||||
| 			sounds = default.node_sound_defaults(), | 			sounds = default.node_sound_defaults(), | ||||||
| 		}) | 		}) | ||||||
| 	end | 	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({ | 	minetest.register_abm({ | ||||||
| 		nodenames = { "moretrees:"..treename.."_trunk_sideways" }, | 		nodenames = { "moretrees:"..treename.."_trunk_sideways" }, | ||||||
| 		interval = 1, | 		interval = 1, | ||||||
| @@ -246,48 +281,50 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 	end | 	end | ||||||
| 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: | -- Extra nodes for jungle trees: | ||||||
|  |  | ||||||
| minetest.register_node("moretrees:jungletree_sapling", { | local jungleleaves = {"yellow","red"} | ||||||
|         description = S("Jungle Sapling"), | local jungleleavesnames = {"Yellow", "Red"} | ||||||
|         drawtype = "plantlike", | for color = 1, #jungleleaves do | ||||||
|         visual_scale = 1.0, |  | ||||||
|         tiles = {"default_junglesapling.png"}, |  | ||||||
|         inventory_image = "default_junglesapling.png", |  | ||||||
|         wield_image = "default_junglesapling.png", |  | ||||||
|         paramtype = "light", |  | ||||||
| 		paramtype2 = "waving", |  | ||||||
|         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", |  | ||||||
| 		paramtype2 = "waving", |  | ||||||
|         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 leave_name = "moretrees:jungletree_leaves_"..jungleleaves[color] | ||||||
|  |  | ||||||
| 	local moretrees_leaves_inventory_image = nil | 	local moretrees_leaves_inventory_image = nil | ||||||
| @@ -306,11 +343,12 @@ for color = 1, 3 do | |||||||
| 		tiles = {"moretrees_jungletree_leaves_"..jungleleaves[color]..".png"}, | 		tiles = {"moretrees_jungletree_leaves_"..jungleleaves[color]..".png"}, | ||||||
| 		inventory_image = moretrees_leaves_inventory_image, | 		inventory_image = moretrees_leaves_inventory_image, | ||||||
| 		paramtype = "light", | 		paramtype = "light", | ||||||
| 		groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1}, | 		is_ground_content = false, | ||||||
|  | 		groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 3 }, | ||||||
| 		drop = { | 		drop = { | ||||||
| 			max_items = 1, | 			max_items = 1, | ||||||
| 			items = { | 			items = { | ||||||
| 				{items = {'moretrees:jungletree_sapling'}, rarity = 100 }, | 				{items = {"default:junglesapling"}, rarity = 100 }, | ||||||
| 				{items = {"moretrees:jungletree_leaves_"..jungleleaves[color]} } | 				{items = {"moretrees:jungletree_leaves_"..jungleleaves[color]} } | ||||||
| 			} | 			} | ||||||
| 		}, | 		}, | ||||||
| @@ -318,6 +356,18 @@ for color = 1, 3 do | |||||||
| 	}) | 	}) | ||||||
| end | 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 | -- Extra needles for firs | ||||||
|  |  | ||||||
| local moretrees_leaves_inventory_image = nil | local moretrees_leaves_inventory_image = nil | ||||||
| @@ -334,7 +384,8 @@ minetest.register_node("moretrees:fir_leaves_bright", { | |||||||
| 	tiles = { "moretrees_fir_leaves_bright.png" }, | 	tiles = { "moretrees_fir_leaves_bright.png" }, | ||||||
| 	inventory_image = moretrees_leaves_inventory_image, | 	inventory_image = moretrees_leaves_inventory_image, | ||||||
| 	paramtype = "light", | 	paramtype = "light", | ||||||
| 	groups = {snappy=3, flammable=2, leaves=1, moretrees_leaves=1 }, | 	is_ground_content = false, | ||||||
|  | 	groups = {snappy = 3, flammable = 2, leaves = 1, moretrees_leaves = 1, leafdecay = 3 }, | ||||||
| 	drop = { | 	drop = { | ||||||
| 		max_items = 1, | 		max_items = 1, | ||||||
| 		items = { | 		items = { | ||||||
| @@ -345,22 +396,29 @@ minetest.register_node("moretrees:fir_leaves_bright", { | |||||||
| 	sounds = default.node_sound_leaves_defaults() | 	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 | if moretrees.enable_redefine_apple then | ||||||
| 	minetest.override_item("default:apple", | 	local appledef = moretrees.clone_node("default:apple") | ||||||
| 		{groups = { fleshy=3, dig_immediate=3, flammable=2, leafdecay=3, leafdecay_drop=1, attached_node = 1} | 	appledef.groups.attached_node = 1 | ||||||
| 	}) | 	minetest.register_node(":default:apple", appledef) | ||||||
| end | end | ||||||
|  |  | ||||||
| table.insert(moretrees.avoidnodes, "default:jungletree") | table.insert(moretrees.avoidnodes, "default:jungletree") | ||||||
| table.insert(moretrees.avoidnodes, "moretrees:jungletree_trunk") | table.insert(moretrees.avoidnodes, "default:pine_tree") | ||||||
|  | table.insert(moretrees.avoidnodes, "default:acacia_tree") | ||||||
| table.insert(moretrees.avoidnodes, "moretrees:fir_trunk") | table.insert(moretrees.avoidnodes, "moretrees:fir_trunk") | ||||||
| table.insert(moretrees.avoidnodes, "default:tree") | table.insert(moretrees.avoidnodes, "default:tree") | ||||||
|  |  | ||||||
| if moretrees.spawn_saplings then | if moretrees.spawn_saplings then | ||||||
| 		table.insert(moretrees.avoidnodes, "snow:sapling_pine") | 	table.insert(moretrees.avoidnodes, "snow:sapling_pine") | ||||||
| 		table.insert(moretrees.avoidnodes, "default:junglesapling") | 	table.insert(moretrees.avoidnodes, "default:junglesapling") | ||||||
| 		table.insert(moretrees.avoidnodes, "moretrees:jungle_tree_sapling") | 	table.insert(moretrees.avoidnodes, "default:pine_sapling") | ||||||
| 		table.insert(moretrees.avoidnodes, "moretrees:jungle_tree_sapling_ongen") |  | ||||||
| end | end | ||||||
|  |  | ||||||
| -- "empty" (tapped) rubber tree nodes | -- "empty" (tapped) rubber tree nodes | ||||||
| @@ -376,6 +434,7 @@ minetest.register_node("moretrees:rubber_tree_trunk_empty", { | |||||||
| 	groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | 	groups = {tree=1,snappy=1,choppy=2,oddly_breakable_by_hand=1,flammable=2}, | ||||||
| 	sounds = default.node_sound_wood_defaults(), | 	sounds = default.node_sound_wood_defaults(), | ||||||
| 	paramtype2 = "facedir", | 	paramtype2 = "facedir", | ||||||
|  | 	is_ground_content = false, | ||||||
| 	on_place = minetest.rotate_node, | 	on_place = minetest.rotate_node, | ||||||
| }) | }) | ||||||
|  |  | ||||||
| @@ -390,38 +449,48 @@ minetest.register_abm({ | |||||||
| 	end, | 	end, | ||||||
| }) | }) | ||||||
|  |  | ||||||
| -- For compatibility with old nodes and recently-changed nodes. | -- For compatibility with old nodes, recently-changed nodes, and default nodes | ||||||
|  |  | ||||||
| minetest.register_alias("technic:rubber_tree_full",      "moretrees:rubber_tree_trunk") | minetest.register_alias("technic:rubber_tree_full",				"moretrees:rubber_tree_trunk") | ||||||
| minetest.register_alias("farming_plus:rubber_tree_full", "moretrees:rubber_tree_trunk") | minetest.register_alias("farming_plus:rubber_tree_full",		"moretrees:rubber_tree_trunk") | ||||||
| minetest.register_alias("farming:rubber_tree_full", "moretrees:rubber_tree_trunk") | minetest.register_alias("farming:rubber_tree_full",				"moretrees:rubber_tree_trunk") | ||||||
|  |  | ||||||
| minetest.register_alias("technic:rubber_leaves",      "moretrees:rubber_tree_leaves") | minetest.register_alias("technic:rubber_leaves",				"moretrees:rubber_tree_leaves") | ||||||
| minetest.register_alias("farming_plus:rubber_leaves", "moretrees:rubber_tree_leaves") | minetest.register_alias("farming_plus:rubber_leaves",			"moretrees:rubber_tree_leaves") | ||||||
| minetest.register_alias("farming:rubber_leaves", "moretrees:rubber_tree_leaves") | minetest.register_alias("farming:rubber_leaves",				"moretrees:rubber_tree_leaves") | ||||||
|  |  | ||||||
| minetest.register_alias("technic:rubber_tree_sapling", "moretrees:rubber_tree_sapling") | minetest.register_alias("technic:rubber_tree_sapling",			"moretrees:rubber_tree_sapling") | ||||||
| minetest.register_alias("farming_plus:rubber_sapling", "moretrees:rubber_tree_sapling") | minetest.register_alias("farming_plus:rubber_sapling",			"moretrees:rubber_tree_sapling") | ||||||
| minetest.register_alias("farming: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:conifer_trunk",				"moretrees:fir_trunk") | ||||||
| minetest.register_alias("moretrees:jungletree_trunk_sideways", "moreblocks:horizontal_jungle_tree") | minetest.register_alias("moretrees:conifer_trunk_sideways",		"moretrees:fir_trunk_sideways") | ||||||
| minetest.register_alias("moretrees:jungletree_trunk", "default:jungletree") | minetest.register_alias("moretrees:conifer_leaves",				"moretrees:fir_leaves") | ||||||
| minetest.register_alias("moretrees:jungletree_planks", "default:junglewood") | minetest.register_alias("moretrees:conifer_leaves_bright",		"moretrees:fir_leaves_bright") | ||||||
|  | minetest.register_alias("moretrees:conifer_sapling",			"moretrees:fir_sapling") | ||||||
|  |  | ||||||
| minetest.register_alias("jungletree:leaves_green", "moretrees:jungletree_leaves_green") | minetest.register_alias("conifers:trunk",						"moretrees:fir_trunk") | ||||||
| minetest.register_alias("jungletree:leaves_red", "moretrees:jungletree_leaves_red") | minetest.register_alias("conifers:trunk_reversed",				"moretrees:fir_trunk_sideways") | ||||||
| minetest.register_alias("jungletree:leaves_yellow", "moretrees:jungletree_leaves_yellow") | 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:conifer_trunk", "moretrees:fir_trunk") | minetest.register_alias("moretrees:jungletree_sapling",			"default:junglesapling") | ||||||
| minetest.register_alias("moretrees:conifer_trunk_sideways", "moretrees:fir_trunk_sideways") | minetest.register_alias("moretrees:jungletree_trunk_sideways",	"moreblocks:horizontal_jungle_tree") | ||||||
| minetest.register_alias("moretrees:conifer_leaves", "moretrees:fir_leaves") | minetest.register_alias("moretrees:jungletree_planks",			"default:junglewood") | ||||||
| minetest.register_alias("moretrees:conifer_leaves_bright", "moretrees:fir_leaves_bright") | minetest.register_alias("moretrees:jungletree_leaves_green",	"default:jungleleaves") | ||||||
| minetest.register_alias("moretrees:conifer_sapling", "moretrees:fir_sapling") |  | ||||||
|  |  | ||||||
| minetest.register_alias("conifers:trunk", "moretrees:fir_trunk") | minetest.register_alias("moretrees:acacia_trunk",				"default:acacia_tree") | ||||||
| minetest.register_alias("conifers:trunk_reversed", "moretrees:fir_trunk_sideways") | minetest.register_alias("moretrees:acacia_planks",				"default:acacia_wood") | ||||||
| minetest.register_alias("conifers:leaves", "moretrees:fir_leaves") | minetest.register_alias("moretrees:acacia_sapling",				"default:acacia_sapling") | ||||||
| minetest.register_alias("conifers:leaves_special", "moretrees:fir_leaves_bright") | minetest.register_alias("moretrees:acacia_leaves",				"default:acacia_leaves") | ||||||
| minetest.register_alias("conifers:sapling", "moretrees:fir_sapling") |  | ||||||
|  |  | ||||||
|  | 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") | ||||||
|   | |||||||
							
								
								
									
										101
									
								
								saplings.lua
									
									
									
									
									
								
							
							
						
						| @@ -7,8 +7,8 @@ for i in ipairs(moretrees.treelist) do | |||||||
|  |  | ||||||
| 	if treename ~= "birch" and treename ~= "spruce" and treename ~= "fir" and treename ~= "jungletree" then | 	if treename ~= "birch" and treename ~= "spruce" and treename ~= "fir" and treename ~= "jungletree" then | ||||||
|  |  | ||||||
| 		plantslib:dbg(dump(moretrees[tree_biome].surface)) | 		biome_lib:dbg(dump(moretrees[tree_biome].surface)) | ||||||
| 		plantslib:grow_plants({ | 		biome_lib:grow_plants({ | ||||||
| 			grow_delay = moretrees.sapling_interval, | 			grow_delay = moretrees.sapling_interval, | ||||||
| 			grow_chance = moretrees.sapling_chance, | 			grow_chance = moretrees.sapling_chance, | ||||||
| 			grow_plant = "moretrees:"..treename.."_sapling", | 			grow_plant = "moretrees:"..treename.."_sapling", | ||||||
| @@ -16,78 +16,83 @@ for i in ipairs(moretrees.treelist) do | |||||||
| 			grow_function = moretrees[tree_model], | 			grow_function = moretrees[tree_model], | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
| 		plantslib:grow_plants({ | 		if moretrees.spawn_saplings then | ||||||
| 			grow_delay = 2, | 			biome_lib:grow_plants({ | ||||||
| 			grow_chance = 30, | 				grow_delay = 2, | ||||||
| 			grow_plant = "moretrees:"..treename.."_sapling_ongen", | 				grow_chance = 30, | ||||||
| 			grow_nodes = moretrees[tree_biome].surface, | 				grow_plant = "moretrees:"..treename.."_sapling_ongen", | ||||||
| 			grow_function = moretrees[tree_model], | 				grow_nodes = moretrees[tree_biome].surface, | ||||||
| 		}) | 				grow_function = moretrees[tree_model], | ||||||
|  | 			}) | ||||||
|  | 		end | ||||||
|  |  | ||||||
| 	end | 	end | ||||||
| end | end | ||||||
|  |  | ||||||
| plantslib:grow_plants({ | biome_lib:grow_plants({ | ||||||
| 	grow_delay = moretrees.sapling_interval, | 	grow_delay = moretrees.sapling_interval, | ||||||
| 	grow_chance = moretrees.sapling_chance, | 	grow_chance = moretrees.sapling_chance, | ||||||
| 	grow_plant = "moretrees:birch_sapling", | 	grow_plant = "moretrees:birch_sapling", | ||||||
| 	grow_nodes = moretrees.birch_biome.surface, | 	grow_nodes = moretrees.birch_biome.surface, | ||||||
| 	grow_function = "moretrees:grow_birch" | 	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({ | biome_lib:grow_plants({ | ||||||
| 	grow_delay = moretrees.sapling_interval, | 	grow_delay = moretrees.sapling_interval, | ||||||
| 	grow_chance = moretrees.sapling_chance, | 	grow_chance = moretrees.sapling_chance, | ||||||
| 	grow_plant = "moretrees:spruce_sapling", | 	grow_plant = "moretrees:spruce_sapling", | ||||||
| 	grow_nodes = moretrees.spruce_biome.surface, | 	grow_nodes = moretrees.spruce_biome.surface, | ||||||
| 	grow_function = "moretrees:grow_spruce" | 	grow_function = "moretrees.grow_spruce" | ||||||
| }) | }) | ||||||
|  |  | ||||||
| plantslib:grow_plants({ | biome_lib: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_delay = moretrees.sapling_interval, | ||||||
| 	grow_chance = moretrees.sapling_chance, | 	grow_chance = moretrees.sapling_chance, | ||||||
| 	grow_plant = "moretrees:fir_sapling", | 	grow_plant = "moretrees:fir_sapling", | ||||||
| 	grow_nodes = moretrees.fir_biome.surface, | 	grow_nodes = moretrees.fir_biome.surface, | ||||||
| 	grow_function = "moretrees:grow_fir" | 	grow_function = "moretrees.grow_fir" | ||||||
| }) | }) | ||||||
|  |  | ||||||
| plantslib:grow_plants({ | biome_lib: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_delay = moretrees.sapling_interval, | ||||||
| 	grow_chance = moretrees.sapling_chance, | 	grow_chance = moretrees.sapling_chance, | ||||||
| 	grow_plant = "moretrees:jungletree_sapling", | 	grow_plant = "default:junglesapling", | ||||||
| 	grow_nodes = moretrees.jungletree_biome.surface, | 	grow_nodes = moretrees.jungletree_biome.surface, | ||||||
| 	grow_function = "moretrees:grow_jungletree" | 	grow_function = "moretrees.grow_jungletree" | ||||||
| }) | }) | ||||||
|  |  | ||||||
| plantslib:grow_plants({ | if moretrees.spawn_saplings then | ||||||
| 	grow_delay = 2, | 	biome_lib:grow_plants({ | ||||||
| 	grow_chance = 30, | 		grow_delay = 2, | ||||||
| 	grow_plant = "moretrees:jungletree_sapling_ongen", | 		grow_chance = 30, | ||||||
| 	grow_nodes = moretrees.jungletree_biome.surface, | 		grow_plant = "moretrees:jungletree_sapling_ongen", | ||||||
| 	grow_function = "moretrees:grow_jungletree" | 		grow_nodes = moretrees.jungletree_biome.surface, | ||||||
| }) | 		grow_function = "moretrees.grow_jungletree" | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	biome_lib: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" | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	biome_lib: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" | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	biome_lib: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" | ||||||
|  | 	}) | ||||||
|  | end | ||||||
|   | |||||||
							
								
								
									
										168
									
								
								screenshot.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,168 @@ | |||||||
|  | -- 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
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 101 KiB | 
| Before Width: | Height: | Size: 150 B | 
| Before Width: | Height: | Size: 205 B | 
| Before Width: | Height: | Size: 599 B | 
| Before Width: | Height: | Size: 651 B | 
| Before Width: | Height: | Size: 175 B | 
| 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 | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_coconut_0.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 877 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_coconut_1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 429 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_coconut_2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1009 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_nut_bar.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 685 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_nut_batter.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 608 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_nut_cake.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 695 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_nut_snack.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 450 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_palm_leaves.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 757 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_palm_sapling.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 261 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_palm_trunk.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_palm_trunk_top.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 673 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_date_palm_wood.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 735 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_f0.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_f1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_f2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 8.9 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_f3.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 8.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_f4.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_fn.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_m0.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_dates_n.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_leaves.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 563 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_sapling.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 447 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_small_sapling.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 347 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_trunk-1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_trunk.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 980 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_trunk_top.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 751 B | 
							
								
								
									
										
											BIN
										
									
								
								textures/moretrees_poplar_wood.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 726 B | 
| @@ -1,23 +1,24 @@ | |||||||
|  |  | ||||||
| 		Elevation	Temperature	Nearness to	Nearby	What nodes		Perlin		Avoid | 				Elevation		Temperature		Nearness to		Nearby	What nodes			Perlin		Avoid | ||||||
| Tree type	   (m)		(approx., °C)	some node	water	to spawn on		seed diff	radius | 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 | 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 | 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 | 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 | palm			- 1 to + 1		+15 to +32		water, 15		 10		sand				330			 5 | ||||||
| apple		+ 1 to +10	+23 to +32	  n/a		n/a	dirt_with grass		331		15 | date palm		- 1 to +10		 above +39		water, 20h,20v	100		desert_sand			339			10 | ||||||
| oak		  0 to +10	+ 4 to +16	  n/a		n/a	dirt_with grass		332		15 | date palm		+11 to +30		 above +39		water, 1h,30v	  1		desert_sand			340			10 | ||||||
| sequoia		  0 to +10	-30 to +50	  n/a		n/a	dirt_with grass		333		10 | apple			+ 1 to +10		+23 to +32		  n/a			n/a		dirt_with grass		331			15 | ||||||
| birch		+10 to +15	-20 to +10	  n/a		n/a	dirt_with grass		334		 5 | oak				  0 to +10		+ 4 to +16		  n/a			n/a		dirt_with grass		332			15 | ||||||
| spruce		 above +20	-20 to +10	  n/a		n/a	dirt_with grass		335		10 | sequoia			  0 to +10		-30 to +50		  n/a			n/a		dirt_with grass		333			10 | ||||||
| pine		    n/a		    n/a		water, 15	   5	dirt_with grass		336		10 | birch			+10 to +15		-20 to +10		  n/a			n/a		dirt_with grass		334			 5 | ||||||
| willow		- 5 to + 5	    n/a		water, 15	   5	dirt_with grass		337		20 | spruce			 above +20		-20 to +10		  n/a			n/a		dirt_with grass		335			10 | ||||||
| acacia		    n/a		    n/a		  n/a		n/a	dirt_with_grass, | cedar			    n/a			    n/a			water, 15		  5		dirt_with grass		336			10 | ||||||
| 									desert_sand		n/a		15 | willow			- 5 to + 5		    n/a			water, 15		  5		dirt_with grass		337			20 | ||||||
| rubber		- 5 to + 5	 above +32	water, 15	  10	dirt_with_grass		338		20 | 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 | beech			    n/a			    n/a			  n/a			n/a		dirt_with_grass		2			10 | ||||||
|  |  | ||||||
|  |  | ||||||
| Notes: | Notes: | ||||||
| @@ -26,9 +27,8 @@ Notes: | |||||||
| Beech trees are meant to replace default trees, but are themselves disabled by default.  They grow in the same areas | 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. | 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, | 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. | 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. | ||||||
|   | |||||||
							
								
								
									
										108
									
								
								tree_models.lua
									
									
									
									
									
								
							
							
						
						| @@ -43,6 +43,37 @@ moretrees.oak_model={ | |||||||
| 	fruit_chance=3, | 	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, | ||||||
|  | } | ||||||
|  |  | ||||||
| moretrees.sequoia_model={ | moretrees.sequoia_model={ | ||||||
| 	axiom="FFFFFFFFFFddccA///cccFddcFA///ddFcFA/cFFddFcdBddd/A/ccdcddd/ccAddddcFBcccAccFdFcFBcccc/BFdFFcFFdcccc/B", | 	axiom="FFFFFFFFFFddccA///cccFddcFA///ddFcFA/cFFddFcdBddd/A/ccdcddd/ccAddddcFBcccAccFdFcFBcccc/BFdFFcFFdcccc/B", | ||||||
| 	rules_a="[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]////[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]////[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]", | 	rules_a="[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]////[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]////[&&&GGF[++^FFdd][--&Fddd]//Fdd[+^Fd][--&Fdd]]", | ||||||
| @@ -88,8 +119,13 @@ moretrees.birch_model2={ | |||||||
| 	thin_branches=true | 	thin_branches=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={ | moretrees.palm_model={ | ||||||
| 	axiom="FFcccccc&FFFFFddd[^&&&GR][^///&&&GR][^//////&&&GR][^***&&&GR]FA//A//A//A//A//A", | 	axiom="FFcccccc&FFFFFdddRA//A//A//A//A//A", | ||||||
| 	rules_a="[&fb&bbb[++f--&ffff&ff][--f++&ffff&ff]&ffff&bbbb&b]", | 	rules_a="[&fb&bbb[++f--&ffff&ff][--f++&ffff&ff]&ffff&bbbb&b]", | ||||||
| 	rules_b="f", | 	rules_b="f", | ||||||
| 	rules_c="/", | 	rules_c="/", | ||||||
| @@ -101,7 +137,32 @@ moretrees.palm_model={ | |||||||
| 	random_level=0, | 	random_level=0, | ||||||
| 	trunk_type="single", | 	trunk_type="single", | ||||||
| 	thin_branches=true, | 	thin_branches=true, | ||||||
| 	fruit="moretrees:coconut", | 	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_chance=0 | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -138,20 +199,20 @@ moretrees.spruce_model2={ | |||||||
| 	fruit_chance=8 | 	fruit_chance=8 | ||||||
| } | } | ||||||
|  |  | ||||||
| moretrees.pine_model={ | moretrees.cedar_model={ | ||||||
| 	axiom="FFFFFcccdddB///cFdFB////cFdFB///cFdFB///cFdFA///cFdFA///cFdFB[FF]f", | 	axiom="FFFFFcccdddB///cFdFB////cFdFB///cFdFB///cFdFA///cFdFA///cFdFB[FF]f", | ||||||
| 	rules_a="[&&&TTTT[++^TFdd][--&TFd]//Tdd[+^Fd][--&Fdd]]", | 	rules_a="[&&&TTTT[++^TFdd][--&TFd]//Tdd[+^Fd][--&Fdd]]", | ||||||
| 	rules_b="[&&&TTT[++^Fdd][--&Fdd]//dd[+^d][--&Fd]]", | 	rules_b="[&&&TTT[++^Fdd][--&Fdd]//dd[+^d][--&Fd]]", | ||||||
| 	rules_c="/", | 	rules_c="/", | ||||||
| 	rules_d="F", | 	rules_d="F", | ||||||
| 	trunk="moretrees:pine_trunk", | 	trunk="moretrees:cedar_trunk", | ||||||
| 	leaves="moretrees:pine_leaves", | 	leaves="moretrees:cedar_leaves", | ||||||
| 	angle=30, | 	angle=30, | ||||||
| 	iterations=2, | 	iterations=2, | ||||||
| 	random_level=0, | 	random_level=0, | ||||||
| 	trunk_type="single", | 	trunk_type="single", | ||||||
| 	thin_branches=true, | 	thin_branches=true, | ||||||
| 	fruit="moretrees:pine_cone", | 	fruit="moretrees:cedar_cone", | ||||||
| 	fruit_chance=8 | 	fruit_chance=8 | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -169,37 +230,6 @@ moretrees.willow_model={ | |||||||
| 	thin_branches=true | 	thin_branches=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, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| moretrees.rubber_tree_model={ | moretrees.rubber_tree_model={ | ||||||
| 	axiom="FFFFA", | 	axiom="FFFFA", | ||||||
| 	rules_a="[&FFBFA]////[&BFFFA]////[&FBFFA]", | 	rules_a="[&FFBFA]////[&BFFFA]////[&FBFFA]", | ||||||
| @@ -217,8 +247,8 @@ moretrees.jungletree_model={ | |||||||
| 	axiom=nil, | 	axiom=nil, | ||||||
| 	rules_a=nil, | 	rules_a=nil, | ||||||
| 	rules_b=nil, | 	rules_b=nil, | ||||||
| 	trunk="default:jungletree", | 	trunk="moretrees:jungletree_trunk", | ||||||
| 	leaves="moretrees:jungletree_leaves_green", | 	leaves="default:jungleleaves", | ||||||
| 	leaves2=nil, | 	leaves2=nil, | ||||||
| 	leaves2_chance=nil, | 	leaves2_chance=nil, | ||||||
| 	angle=45, | 	angle=45, | ||||||
|   | |||||||