| 
									
										
										
										
											2023-01-05 13:25:55 +01:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*  parallax_layer.cpp                                                    */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*                         This file is part of:                          */ | 
					
						
							|  |  |  | /*                             GODOT ENGINE                               */ | 
					
						
							|  |  |  | /*                        https://godotengine.org                         */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
					
						
							|  |  |  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining  */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the        */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including    */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,    */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to     */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to  */ | 
					
						
							|  |  |  | /* the following conditions:                                              */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be         */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.        */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,        */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF     */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY   */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,   */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE      */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-05 00:50:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include "parallax_layer.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-19 01:02:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include "parallax_background.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | void ParallaxLayer::set_motion_scale(const Size2 &p_scale) { | 
					
						
							| 
									
										
										
										
											2017-08-22 15:13:31 +02:00
										 |  |  | 	motion_scale = p_scale; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 22:58:51 +02:00
										 |  |  | 	ParallaxBackground *pb = Object::cast_to<ParallaxBackground>(get_parent()); | 
					
						
							|  |  |  | 	if (pb && is_inside_tree()) { | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 		Vector2 final_ofs = pb->get_final_offset(); | 
					
						
							|  |  |  | 		real_t scroll_scale = pb->get_scroll_scale(); | 
					
						
							|  |  |  | 		set_base_offset_and_scale(final_ofs, scroll_scale); | 
					
						
							| 
									
										
										
										
											2016-06-26 16:22:20 -03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2 ParallaxLayer::get_motion_scale() const { | 
					
						
							|  |  |  | 	return motion_scale; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | void ParallaxLayer::set_motion_offset(const Size2 &p_offset) { | 
					
						
							| 
									
										
										
										
											2017-08-22 15:13:31 +02:00
										 |  |  | 	motion_offset = p_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 22:58:51 +02:00
										 |  |  | 	ParallaxBackground *pb = Object::cast_to<ParallaxBackground>(get_parent()); | 
					
						
							|  |  |  | 	if (pb && is_inside_tree()) { | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 		Vector2 final_ofs = pb->get_final_offset(); | 
					
						
							|  |  |  | 		real_t scroll_scale = pb->get_scroll_scale(); | 
					
						
							|  |  |  | 		set_base_offset_and_scale(final_ofs, scroll_scale); | 
					
						
							| 
									
										
										
										
											2016-06-26 16:22:20 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Size2 ParallaxLayer::get_motion_offset() const { | 
					
						
							|  |  |  | 	return motion_offset; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void ParallaxLayer::_update_mirroring() { | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!is_inside_tree()) { | 
					
						
							| 
									
										
										
										
											2019-08-13 20:59:41 +02:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-08-13 20:59:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 22:58:51 +02:00
										 |  |  | 	ParallaxBackground *pb = Object::cast_to<ParallaxBackground>(get_parent()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (pb) { | 
					
						
							| 
									
										
										
										
											2018-04-07 19:42:11 -03:00
										 |  |  | 		RID c = pb->get_canvas(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		RID ci = get_canvas_item(); | 
					
						
							| 
									
										
										
										
											2024-03-16 10:55:26 -04:00
										 |  |  | 		Point2 mirror_scale = mirroring * orig_scale; | 
					
						
							|  |  |  | 		RenderingServer::get_singleton()->canvas_set_item_mirroring(c, ci, mirror_scale); | 
					
						
							| 
									
										
										
										
											2024-02-17 00:57:32 +01:00
										 |  |  | 		RenderingServer::get_singleton()->canvas_item_set_interpolated(ci, false); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | void ParallaxLayer::set_mirroring(const Size2 &p_mirroring) { | 
					
						
							| 
									
										
										
										
											2024-08-18 19:36:08 +02:00
										 |  |  | 	mirroring = p_mirroring.maxf(0); | 
					
						
							| 
									
										
										
										
											2016-06-26 16:51:13 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_update_mirroring(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | Size2 ParallaxLayer::get_mirroring() const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return mirroring; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParallaxLayer::_notification(int p_what) { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 			orig_offset = get_position(); | 
					
						
							|  |  |  | 			orig_scale = get_scale(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			_update_mirroring(); | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2022-02-15 18:06:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 10:08:49 -07:00
										 |  |  | 		case NOTIFICATION_EXIT_TREE: { | 
					
						
							| 
									
										
										
										
											2021-09-01 14:57:14 +02:00
										 |  |  | 			if (Engine::get_singleton()->is_editor_hint()) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 10:08:49 -07:00
										 |  |  | 			set_position(orig_offset); | 
					
						
							|  |  |  | 			set_scale(orig_scale); | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 18:47:22 +00:00
										 |  |  | void ParallaxLayer::set_base_offset_and_scale(const Point2 &p_offset, real_t p_scale) { | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!is_inside_tree()) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (Engine::get_singleton()->is_editor_hint()) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-10-26 20:08:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 18:47:22 +00:00
										 |  |  | 	Point2 new_ofs = p_offset * motion_scale + motion_offset * p_scale + orig_offset * p_scale; | 
					
						
							| 
									
										
										
										
											2017-10-26 20:08:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (mirroring.x) { | 
					
						
							| 
									
										
										
										
											2021-09-07 23:25:35 -05:00
										 |  |  | 		real_t den = mirroring.x * p_scale; | 
					
						
							| 
									
										
										
										
											2017-11-23 01:52:34 +00:00
										 |  |  | 		new_ofs.x -= den * ceil(new_ofs.x / den); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mirroring.y) { | 
					
						
							| 
									
										
										
										
											2021-09-07 23:25:35 -05:00
										 |  |  | 		real_t den = mirroring.y * p_scale; | 
					
						
							| 
									
										
										
										
											2017-11-23 01:52:34 +00:00
										 |  |  | 		new_ofs.y -= den * ceil(new_ofs.y / den); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-04 01:16:14 -03:00
										 |  |  | 	set_position(new_ofs); | 
					
						
							| 
									
										
										
										
											2017-11-23 01:52:34 +00:00
										 |  |  | 	set_scale(Vector2(1, 1) * p_scale * orig_scale); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_update_mirroring(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-17 19:03:21 +01:00
										 |  |  | PackedStringArray ParallaxLayer::get_configuration_warnings() const { | 
					
						
							| 
									
										
										
										
											2024-06-19 16:40:50 +02:00
										 |  |  | 	PackedStringArray warnings = Node2D::get_configuration_warnings(); | 
					
						
							| 
									
										
										
										
											2020-05-14 23:59:27 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 22:58:51 +02:00
										 |  |  | 	if (!Object::cast_to<ParallaxBackground>(get_parent())) { | 
					
						
							| 
									
										
										
										
											2022-03-28 15:24:14 +02:00
										 |  |  | 		warnings.push_back(RTR("ParallaxLayer node only works when set as child of a ParallaxBackground node.")); | 
					
						
							| 
									
										
										
										
											2016-05-17 18:27:15 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 05:01:28 -05:00
										 |  |  | 	return warnings; | 
					
						
							| 
									
										
										
										
											2016-05-17 18:27:15 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void ParallaxLayer::_bind_methods() { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("set_motion_scale", "scale"), &ParallaxLayer::set_motion_scale); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_motion_scale"), &ParallaxLayer::get_motion_scale); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("set_motion_offset", "offset"), &ParallaxLayer::set_motion_offset); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_motion_offset"), &ParallaxLayer::get_motion_offset); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("set_mirroring", "mirror"), &ParallaxLayer::set_mirroring); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_mirroring"), &ParallaxLayer::get_mirroring); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ADD_GROUP("Motion", "motion_"); | 
					
						
							| 
									
										
										
										
											2022-06-14 09:33:44 -05:00
										 |  |  | 	ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "motion_scale", PROPERTY_HINT_LINK), "set_motion_scale", "get_motion_scale"); | 
					
						
							| 
									
										
										
										
											2021-12-02 18:09:19 -06:00
										 |  |  | 	ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "motion_offset", PROPERTY_HINT_NONE, "suffix:px"), "set_motion_offset", "get_motion_offset"); | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "motion_mirroring"), "set_mirroring", "get_mirroring"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | ParallaxLayer::ParallaxLayer() { | 
					
						
							| 
									
										
										
										
											2024-02-17 00:57:32 +01:00
										 |  |  | 	// ParallaxLayer is always updated every frame so there is no need to interpolate.
 | 
					
						
							|  |  |  | 	set_physics_interpolation_mode(Node::PHYSICS_INTERPOLATION_MODE_OFF); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } |