Compare commits
	
		
			394 Commits
		
	
	
		
			b931abb735
			...
			developmen
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| cac69f5f35 | |||
| 1660915678 | |||
| 1d02b0eba2 | |||
| 2ef7fa6577 | |||
| 193b2c5af0 | |||
| a859c0cbff | |||
| fba64c3854 | |||
| a975cbb56b | |||
| 1664a9ccf7 | |||
| 988a6f67f2 | |||
| 2f32038f04 | |||
| 4b756fa232 | |||
| 0db2cae1bb | |||
| 896f7876c1 | |||
| 56d3112e35 | |||
| 32ec2325dc | |||
| b42f1f1881 | |||
| f8096377b2 | |||
| a9fc819268 | |||
| 1d63391975 | |||
| 61ff0887e2 | |||
| 16344dccc7 | |||
| dc4bea3eef | |||
| d1b2723a70 | |||
| 2f5c04e66b | |||
| f753da1f87 | |||
| 6901159106 | |||
| 7469c9ab0c | |||
| ede90adbdc | |||
| eeaca3a6c7 | |||
| 3b984a0a4b | |||
| c5afb70b18 | |||
| 9d2a192f04 | |||
| 598debc233 | |||
| ab05a89175 | |||
| dbd15cbbc2 | |||
| e051f5cfb4 | |||
| e70b7f112f | |||
| f55ba499b6 | |||
| b75f30f864 | |||
| 6f1f30bd53 | |||
| 92a5c276a4 | |||
| 69bc6573d1 | |||
| 28bc022587 | |||
| 25db60e436 | |||
| 7c62440bba | |||
| 4bec7bce6e | |||
| 8d31372c24 | |||
| a2e704916e | |||
| c7d170fad9 | |||
| 9ccf7b754d | |||
| e3d4899112 | |||
| 566c16d09c | |||
| ae9d4f02ef | |||
| e77772cbc2 | |||
| 4c542df401 | |||
| 28ca343b43 | |||
| 651b0614c4 | |||
| f47488c6f1 | |||
| 6d159330a1 | |||
| 8e314f3269 | |||
| f5a7077570 | |||
| 746d29fb7a | |||
| cf68f6ca6f | |||
| a4b83679b1 | |||
| a31b39fd1d | |||
| 0205354202 | |||
| 949dfeb3d9 | |||
| 620ef911fa | |||
| efed24de20 | |||
| 3912706d27 | |||
| d78c42a653 | |||
| b04e0f81cd | |||
| 65dcb0c564 | |||
| 3d183b21cd | |||
| 1644a751bb | |||
| 6631cae7b0 | |||
| 3452194941 | |||
| 11612ff0db | |||
| 63bc94c7a6 | |||
| e00319d7ff | |||
| 11719440dc | |||
| f246d68aa7 | |||
| 6e87c67096 | |||
| b8217f2106 | |||
| 9824980cbf | |||
| 93a79cd075 | |||
| f6e52abcc1 | |||
| 03232f72e8 | |||
| 37aca44e45 | |||
| 9f4d95a57b | |||
| 65eac57fce | |||
| 08311acc9a | |||
| f8fbae6130 | |||
| df06e8d134 | |||
| ad365dc722 | |||
| 200e8ae7da | |||
| 65cfaf1b4a | |||
| 83b155fc5e | |||
| 7db56e7f3e | |||
| 42064875a0 | |||
| 41245c0c1c | |||
| 0e5cc8f898 | |||
| c8bb991865 | |||
| bc1c76d746 | |||
| 8f03628bd6 | |||
| a1feb0bad3 | |||
| 978cba96c8 | |||
| 7212094a3d | |||
| 14843ddeba | |||
| 5315db0077 | |||
| 026f343d43 | |||
| da5f31f9d7 | |||
| fa1614f238 | |||
| 0c096d39db | |||
| dae6549bad | |||
| 767fc28488 | |||
| c3be8f60b7 | |||
| 33cb44bf36 | |||
| 4c1018ddec | |||
| cf7061fd58 | |||
| e6b7b9953f | |||
| 4a3775a0de | |||
| 4d353662a1 | |||
| ca0b2de917 | |||
| 2335c3ec62 | |||
| 30ccab1b93 | |||
| f56d6a7fc8 | |||
| 29a7f5880f | |||
| eee3056614 | |||
| 152b0e93db | |||
| 3f914fe46f | |||
| 62b54ee836 | |||
| 6a41407005 | |||
| adfa6c6ba0 | |||
| a53766f472 | |||
| 40735c713a | |||
| 2054ae3a35 | |||
| 9066e11c12 | |||
| f16a7e55c9 | |||
| e3b32b3c4a | |||
| a02584f3b6 | |||
| 45524e474e | |||
| fbdea47dc7 | |||
| f5fbd4e5ef | |||
| c7f63dc638 | |||
| beecefec1c | |||
| 24d1a1d764 | |||
| 9edf3b0aa6 | |||
| 8d49fb467c | |||
| 2caa042317 | |||
| fe8bde855d | |||
| ac620264b1 | |||
| f31b84f519 | |||
| efb7cc7452 | |||
| 7a3202a053 | |||
| 86c9ed2ba9 | |||
| 56321864fb | |||
| 6adc002f1a | |||
| 1acc8bdb8f | |||
| 61e2761580 | |||
| 996e61d0ad | |||
| b1b5af94d3 | |||
| b0f8b0dad6 | |||
| 67d7f401b8 | |||
| 9bf17cc191 | |||
| bf8fbebae3 | |||
| 1b0f25e854 | |||
| 61a7f685c1 | |||
| feb2a05aa3 | |||
| cd30047e4a | |||
| a3b03efd47 | |||
| 4213b3f8b5 | |||
| d3fb612904 | |||
| 8f8558a262 | |||
| 2df41e1881 | |||
| 114fa82b9d | |||
| bcce427376 | |||
| 6a750f8ce0 | |||
| 3e02ee7b6f | |||
| 6b9020bd24 | |||
| 832514ba7d | |||
| 877a004a13 | |||
| b1970d93f9 | |||
| e7bd924494 | |||
| 37b87f0f85 | |||
| 3b6a93d37a | |||
| 0bf38234c6 | |||
| ed6969c16a | |||
| b0b421151f | |||
| 41c5def097 | |||
| fbbdfb07fa | |||
| bf283d804c | |||
| 063ea08707 | |||
| fd11a94ddf | |||
| be2295b92d | |||
| a93e55619c | |||
| 48ae24af47 | |||
| 1366a417f1 | |||
| 4bfe98852c | |||
| 98edbe1af5 | |||
| 3725a3b0fd | |||
| f43ab36742 | |||
| c7aafd85bc | |||
| 5de08b8fe4 | |||
| 16e4077d40 | |||
| fc3c1ed1f9 | |||
| b100b5c2fe | |||
| 5e28ba8814 | |||
| 4c235e3230 | |||
| 131203d578 | |||
| bd5eb432b7 | |||
| d2ca85568f | |||
| 4c41870732 | |||
| f77afa3632 | |||
| eb61598489 | |||
| efe51b491d | |||
| fa3a4d1e0d | |||
| 6e7a0993f5 | |||
| d70bee2c6b | |||
| 5812f43117 | |||
| d102c5471d | |||
| fb363970fc | |||
| 791349686b | |||
| 3a0942ff46 | |||
| b002dd469a | |||
| f92f36442c | |||
| bb934b59f3 | |||
| c704173183 | |||
| c3876add1e | |||
| 35a75d993b | |||
| 2637f99456 | |||
| 9581f5aa54 | |||
| 82cc25a9ef | |||
| 336e7e16e7 | |||
| a3a8fb4e84 | |||
| 35f6c3850e | |||
| f51d5f342e | |||
| 9c129cefe2 | |||
| a254bb721b | |||
| 5fa7420c04 | |||
| 5bcc256777 | |||
| 680d718957 | |||
| 20bc6a1adb | |||
| eb454a471c | |||
| c205e710bc | |||
| cddb30c631 | |||
| 29f6c83bf0 | |||
| c20f210b29 | |||
| 1ea1844677 | |||
| 5b2c13f8bf | |||
| c39ee44442 | |||
| 4623b4861a | |||
| 0a868b82e5 | |||
| d92d16cfad | |||
| 0184d1758c | |||
| 6e5b805803 | |||
| 8293c58f9f | |||
| 94d01521d4 | |||
| 5c1c025fe3 | |||
| 1d292a104e | |||
| 70c884acfe | |||
| a9f5974568 | |||
| dae72b11c5 | |||
| 58eb373c79 | |||
| 00f7b1aaab | |||
| 9e4c74ed1d | |||
| 2e2306c5bb | |||
| 86b8cd9b55 | |||
| bfbcfdce4f | |||
| 80202d4a07 | |||
| 2be99d2142 | |||
| 4081693d32 | |||
| 193d23c877 | |||
| c135035d5b | |||
| fabc485689 | |||
| 48710b0a7f | |||
| bf34e52dc8 | |||
| e3845a2f5c | |||
| 26a80452bc | |||
| 2535a1d6ec | |||
| 3a385900fb | |||
| b94bbc8ed7 | |||
| 3f7a646bf0 | |||
| f119a23d2b | |||
| 61488aa0e5 | |||
| 66b46e3d36 | |||
| 1ee07b41f8 | |||
| 6f425776cc | |||
| 98c9dde98a | |||
| 04d325f38b | |||
| 901585d4bb | |||
| 33a452a62e | |||
| 906edf096e | |||
| d1129c95df | |||
| 40f483974d | |||
| 4b856420f9 | |||
| 7f93d95f6b | |||
| 5756b96002 | |||
| c71bf71fb3 | |||
| 417ddca972 | |||
| d4c6288b38 | |||
| 21600b856f | |||
| 803b670433 | |||
| 067bc51487 | |||
| 9f3e39e337 | |||
| cd2cd89eae | |||
| 90c1dd9348 | |||
| e8ef41af41 | |||
| d1a289885b | |||
| 6170de4a73 | |||
| e2fdf1f538 | |||
| 7a1dd7eb1a | |||
| b73c9ed0ae | |||
| 82705ba378 | |||
| f9785462b0 | |||
| 5c3e0f6581 | |||
| b9ee1ec232 | |||
| 4ec1a32db2 | |||
| d825bb55d3 | |||
| 5fc8c012b3 | |||
| 95ddba0230 | |||
| 30caa202dc | |||
| b2a286b5e5 | |||
| 5c542039ed | |||
| 25043bbcde | |||
| 183966d239 | |||
| 9ecf0b900f | |||
| 62b43025b9 | |||
| 9af44d48b3 | |||
| d71c135491 | |||
| e73c076243 | |||
| 7743ccadbf | |||
| d0ab442f7f | |||
| cdfe655ac4 | |||
| b4659def55 | |||
| 58a9ada345 | |||
| eb5345dc77 | |||
| 4416f64287 | |||
| 1b3f40be5f | |||
| e725a4e89c | |||
| 981db0190f | |||
| c0c48c7d51 | |||
| cc4068fa2e | |||
| 4d59dcb9ab | |||
| 1b3b999a95 | |||
| 1545291942 | |||
| 81625abd25 | |||
| ea94bed00d | |||
| 12cb144688 | |||
| 85f0555c59 | |||
| 55ed8b84f6 | |||
| 4856800f5f | |||
| cc44e1ea69 | |||
| cb60c71184 | |||
| eb445604e8 | |||
| 8eb34a0a6d | |||
| 9f522bdb66 | |||
| d7f0b76485 | |||
| 90370a2b43 | |||
| 0f3f1594d0 | |||
| 24b50eba12 | |||
| 6bc9043a86 | |||
| 43f1749b04 | |||
| 62e50aefc1 | |||
| 9c2b098821 | |||
| 91aa26e15a | |||
| cf8a5de580 | |||
| 1f8fa78b76 | |||
| fdc38fc800 | |||
| fc34a60f30 | |||
| eca23c5b89 | |||
| f08f721f52 | |||
| fb402acc30 | |||
| 923b25e26e | |||
| 2bcd1c5a89 | |||
| 40d1ce7c68 | |||
| e7c80871fe | |||
| c51eda49bf | |||
| 0f8a7db567 | |||
| ffa0128813 | |||
| 15984bcc06 | |||
| ef21cdf213 | |||
| 2cf6135063 | |||
| be06575f91 | |||
| ed6975bf24 | |||
| d9660c08b1 | |||
| 3902f1caca | |||
| 14e3337daa | |||
| f729cdc0a8 | |||
| c767e1e856 | |||
| f96c58cbd4 | |||
| fed288859f | |||
| 6e4c9b0ef8 | 
							
								
								
									
										225
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										225
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,225 @@ | |||||||
|  | # EditorConfig is awesome: https://EditorConfig.org | ||||||
|  |  | ||||||
|  | # top-most EditorConfig file | ||||||
|  | root = true | ||||||
|  |  | ||||||
|  | # Don't use tabs for indentation. | ||||||
|  | [*] | ||||||
|  | indent_style = space | ||||||
|  | # (Please don't specify an indent_size here; that has too many unintended consequences.) | ||||||
|  | spelling_exclusion_path = SpellingExclusions.dic | ||||||
|  |  | ||||||
|  | # Code files | ||||||
|  | [*.{cs,csx,vb,vbx}] | ||||||
|  | indent_size = 4 | ||||||
|  | insert_final_newline = true | ||||||
|  | charset = utf-8-bom | ||||||
|  |  | ||||||
|  | # XML project files | ||||||
|  | [*.{csproj,vbproj,vcxproj,vcxproj.filters,proj,projitems,shproj}] | ||||||
|  | indent_size = 2 | ||||||
|  |  | ||||||
|  | # XML config files | ||||||
|  | [*.{props,targets,ruleset,config,nuspec,resx,vsixmanifest,vsct}] | ||||||
|  | indent_size = 2 | ||||||
|  |  | ||||||
|  | # JSON files | ||||||
|  | [*.json] | ||||||
|  | indent_size = 2 | ||||||
|  |  | ||||||
|  | # Powershell files | ||||||
|  | [*.ps1] | ||||||
|  | indent_size = 2 | ||||||
|  |  | ||||||
|  | # Shell script files | ||||||
|  | [*.sh] | ||||||
|  | end_of_line = lf | ||||||
|  | indent_size = 2 | ||||||
|  |  | ||||||
|  | # Dotnet code style settings: | ||||||
|  | [*.{cs,vb}] | ||||||
|  |  | ||||||
|  | # Sort using and Import directives with System.* appearing first | ||||||
|  | dotnet_sort_system_directives_first = true | ||||||
|  | dotnet_separate_import_directive_groups = false | ||||||
|  | # Avoid "this." and "Me." if not necessary | ||||||
|  | dotnet_style_qualification_for_field = false:refactoring | ||||||
|  | dotnet_style_qualification_for_property = false:refactoring | ||||||
|  | dotnet_style_qualification_for_method = false:refactoring | ||||||
|  | dotnet_style_qualification_for_event = false:refactoring | ||||||
|  |  | ||||||
|  | # Use language keywords instead of framework type names for type references | ||||||
|  | dotnet_style_predefined_type_for_locals_parameters_members = true:suggestion | ||||||
|  | dotnet_style_predefined_type_for_member_access = true:suggestion | ||||||
|  |  | ||||||
|  | # Suggest more modern language features when available | ||||||
|  | dotnet_style_object_initializer = true:suggestion | ||||||
|  | dotnet_style_collection_initializer = true:suggestion | ||||||
|  | dotnet_style_coalesce_expression = true:none | ||||||
|  | dotnet_style_null_propagation = true:suggestion | ||||||
|  | dotnet_style_explicit_tuple_names = true:suggestion | ||||||
|  |  | ||||||
|  | # Whitespace options | ||||||
|  | dotnet_style_allow_multiple_blank_lines_experimental = false:error | ||||||
|  |  | ||||||
|  | # IDE0055: Fix formatting | ||||||
|  | # Workaround for https://github.com/dotnet/roslyn/issues/70570 | ||||||
|  | dotnet_diagnostic.IDE0055.severity = suggestion | ||||||
|  |  | ||||||
|  | # https://github.com/dotnet/roslyn-analyzers/issues/7436 - False positives from valid GetDeclaredSymbol calls | ||||||
|  | dotnet_diagnostic.RS1039.severity = none | ||||||
|  |  | ||||||
|  | # CSharp code style settings: | ||||||
|  |  | ||||||
|  | # IDE0029: Use coalesce expression | ||||||
|  | dotnet_diagnostic.IDE0029.severity = none | ||||||
|  |  | ||||||
|  | # IDE0040: Add accessibility modifiers | ||||||
|  | dotnet_diagnostic.IDE0040.severity = warning | ||||||
|  |  | ||||||
|  | [*.cs] | ||||||
|  | # Newline settings | ||||||
|  | csharp_new_line_before_open_brace = all | ||||||
|  | csharp_new_line_before_else = true | ||||||
|  | csharp_new_line_before_catch = true | ||||||
|  | csharp_new_line_before_finally = true | ||||||
|  | csharp_new_line_before_members_in_object_initializers = true | ||||||
|  | csharp_new_line_before_members_in_anonymous_types = true | ||||||
|  | csharp_new_line_between_query_expression_clauses = true | ||||||
|  |  | ||||||
|  | # Indentation preferences | ||||||
|  | csharp_indent_block_contents = true | ||||||
|  | csharp_indent_braces = false | ||||||
|  | csharp_indent_case_contents = true | ||||||
|  | csharp_indent_case_contents_when_block = true | ||||||
|  | csharp_indent_switch_labels = true | ||||||
|  | csharp_indent_labels = flush_left | ||||||
|  |  | ||||||
|  | # Whitespace options | ||||||
|  | csharp_style_allow_embedded_statements_on_same_line_experimental = true | ||||||
|  | csharp_style_allow_blank_lines_between_consecutive_braces_experimental = false:error | ||||||
|  | csharp_style_allow_blank_line_after_colon_in_constructor_initializer_experimental = true | ||||||
|  | csharp_style_allow_blank_line_after_token_in_conditional_expression_experimental = true | ||||||
|  | csharp_style_allow_blank_line_after_token_in_arrow_expression_clause_experimental = true | ||||||
|  |  | ||||||
|  | # Prefer "var" everywhere | ||||||
|  | csharp_style_var_for_built_in_types = false:warning | ||||||
|  | csharp_style_var_when_type_is_apparent = false:warning | ||||||
|  | csharp_style_var_elsewhere = false:warning | ||||||
|  |  | ||||||
|  | # Prefer method-like constructs to have a block body | ||||||
|  | csharp_style_expression_bodied_methods = false:none | ||||||
|  | csharp_style_expression_bodied_constructors = false:none | ||||||
|  | csharp_style_expression_bodied_operators = false:none | ||||||
|  |  | ||||||
|  | # Prefer property-like constructs to have an expression-body | ||||||
|  | csharp_style_expression_bodied_properties = true:none | ||||||
|  | csharp_style_expression_bodied_indexers = true:none | ||||||
|  | csharp_style_expression_bodied_accessors = true:none | ||||||
|  |  | ||||||
|  | # Suggest more modern language features when available | ||||||
|  | csharp_style_pattern_matching_over_is_with_cast_check = true:suggestion | ||||||
|  | csharp_style_pattern_matching_over_as_with_null_check = true:suggestion | ||||||
|  | csharp_style_inlined_variable_declaration = true:suggestion | ||||||
|  | csharp_style_throw_expression = true:suggestion | ||||||
|  | csharp_style_conditional_delegate_call = true:suggestion | ||||||
|  | csharp_style_prefer_extended_property_pattern = true:suggestion | ||||||
|  |  | ||||||
|  | # Space preferences | ||||||
|  | csharp_space_after_cast = false | ||||||
|  | csharp_space_after_colon_in_inheritance_clause = true | ||||||
|  | csharp_space_after_comma = true | ||||||
|  | csharp_space_after_dot = false | ||||||
|  | csharp_space_after_keywords_in_control_flow_statements = true | ||||||
|  | csharp_space_after_semicolon_in_for_statement = true | ||||||
|  | csharp_space_around_binary_operators = before_and_after | ||||||
|  | csharp_space_around_declaration_statements = do_not_ignore | ||||||
|  | csharp_space_before_colon_in_inheritance_clause = true | ||||||
|  | csharp_space_before_comma = false | ||||||
|  | csharp_space_before_dot = false | ||||||
|  | csharp_space_before_open_square_brackets = false | ||||||
|  | csharp_space_before_semicolon_in_for_statement = false | ||||||
|  | csharp_space_between_empty_square_brackets = false | ||||||
|  | csharp_space_between_method_call_empty_parameter_list_parentheses = false | ||||||
|  | csharp_space_between_method_call_name_and_opening_parenthesis = false | ||||||
|  | csharp_space_between_method_call_parameter_list_parentheses = false | ||||||
|  | csharp_space_between_method_declaration_empty_parameter_list_parentheses = false | ||||||
|  | csharp_space_between_method_declaration_name_and_open_parenthesis = false | ||||||
|  | csharp_space_between_method_declaration_parameter_list_parentheses = false | ||||||
|  | csharp_space_between_parentheses = false | ||||||
|  | csharp_space_between_square_brackets = false | ||||||
|  |  | ||||||
|  | # Blocks are allowed | ||||||
|  | csharp_prefer_braces = true:silent | ||||||
|  | csharp_preserve_single_line_blocks = true | ||||||
|  | csharp_preserve_single_line_statements = true | ||||||
|  |  | ||||||
|  | # IDE0060: Remove unused parameter | ||||||
|  | dotnet_diagnostic.IDE0060.severity = warning | ||||||
|  |  | ||||||
|  | [src/{Compilers,ExpressionEvaluator,Scripting}/**Test**/*.{cs,vb}] | ||||||
|  |  | ||||||
|  | # IDE0060: Remove unused parameter | ||||||
|  | dotnet_diagnostic.IDE0060.severity = none | ||||||
|  |  | ||||||
|  | [src/{Analyzers,CodeStyle,Features,Workspaces,EditorFeatures,VisualStudio}/**/*.{cs,vb}] | ||||||
|  |  | ||||||
|  | # IDE0011: Add braces | ||||||
|  | csharp_prefer_braces = false:warning | ||||||
|  | # NOTE: We need the below severity entry for Add Braces due to https://github.com/dotnet/roslyn/issues/44201 | ||||||
|  | dotnet_diagnostic.IDE0011.severity = warning | ||||||
|  |  | ||||||
|  | # IDE0040: Add accessibility modifiers | ||||||
|  | dotnet_diagnostic.IDE0040.severity = warning | ||||||
|  |  | ||||||
|  | # IDE0052: Remove unread private member | ||||||
|  | dotnet_diagnostic.IDE0052.severity = warning | ||||||
|  |  | ||||||
|  | # IDE0059: Unnecessary assignment to a value | ||||||
|  | dotnet_diagnostic.IDE0059.severity = warning | ||||||
|  |  | ||||||
|  | # CA1012: Abstract types should not have public constructors | ||||||
|  | dotnet_diagnostic.CA1012.severity = warning | ||||||
|  |  | ||||||
|  | # CA1822: Make member static | ||||||
|  | dotnet_diagnostic.CA1822.severity = warning | ||||||
|  |  | ||||||
|  | # Prefer "var" everywhere | ||||||
|  | dotnet_diagnostic.IDE0007.severity = warning | ||||||
|  | csharp_style_var_for_built_in_types = false:warning | ||||||
|  | csharp_style_var_when_type_is_apparent = false:warning | ||||||
|  | csharp_style_var_elsewhere = false:warning | ||||||
|  |  | ||||||
|  | # csharp_style_allow_embedded_statements_on_same_line_experimental | ||||||
|  | dotnet_diagnostic.IDE2001.severity = warning | ||||||
|  |  | ||||||
|  | # csharp_style_allow_blank_line_after_colon_in_constructor_initializer_experimental | ||||||
|  | dotnet_diagnostic.IDE2004.severity = warning | ||||||
|  |  | ||||||
|  | # csharp_style_allow_blank_line_after_token_in_conditional_expression_experimental | ||||||
|  | dotnet_diagnostic.IDE2005.severity = warning | ||||||
|  |  | ||||||
|  | # csharp_style_allow_blank_line_after_token_in_arrow_expression_clause_experimental | ||||||
|  | dotnet_diagnostic.IDE2006.severity = warning | ||||||
|  |  | ||||||
|  | [src/{VisualStudio}/**/*.{cs,vb}] | ||||||
|  | # CA1822: Make member static | ||||||
|  | # There is a risk of accidentally breaking an internal API that partners rely on though IVT. | ||||||
|  | dotnet_code_quality.CA1822.api_surface = private | ||||||
|  |  | ||||||
|  | [**/{ExternalAccess}/**/*.{cs,vb}] | ||||||
|  |  | ||||||
|  | # RS0016: Only enable if API files are present | ||||||
|  | dotnet_public_api_analyzer.require_api_files = true | ||||||
|  |  | ||||||
|  | dotnet_diagnostic.RS0051.severity = error | ||||||
|  | dotnet_diagnostic.RS0052.severity = error | ||||||
|  | dotnet_diagnostic.RS0053.severity = error | ||||||
|  | dotnet_diagnostic.RS0054.severity = error | ||||||
|  | dotnet_diagnostic.RS0055.severity = error | ||||||
|  | dotnet_diagnostic.RS0056.severity = error | ||||||
|  | dotnet_diagnostic.RS0057.severity = error | ||||||
|  | dotnet_diagnostic.RS0058.severity = error | ||||||
|  | dotnet_diagnostic.RS0059.severity = error | ||||||
|  | dotnet_diagnostic.RS0060.severity = error | ||||||
|  | dotnet_diagnostic.RS0061.severity = error | ||||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | [submodule "Engine.Integration/YamlDotNet"] | ||||||
|  | 	path = Engine.Integration/YamlDotNet | ||||||
|  | 	url = https://github.com/Syntriax/YamlDotNet.git | ||||||
							
								
								
									
										2
									
								
								Engine.Core/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								Engine.Core/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -482,3 +482,5 @@ $RECYCLE.BIN/ | |||||||
|  |  | ||||||
| # Vim temporary swap files | # Vim temporary swap files | ||||||
| *.swp | *.swp | ||||||
|  |  | ||||||
|  | !Debug | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Indicates the class implementing it has Assignable fields that are necessary for the engine to work properly. | /// Indicates the class implementing it has Assignable fields that are necessary for the engine to work properly. | ||||||
| @@ -10,7 +8,7 @@ public interface IAssignable | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="IAssignable"/>'s fields are unassigned and completely ready to recycle. |     /// Event triggered when the <see cref="IAssignable"/>'s fields are unassigned and completely ready to recycle. | ||||||
|     /// </summary>  |     /// </summary>  | ||||||
|     Action<IAssignable>? OnUnassigned { get; set; } |     Event<IAssignable>? OnUnassigned { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Unassign <see cref="IAssignable"/>'s all fields and make it ready to recycle. |     /// Unassign <see cref="IAssignable"/>'s all fields and make it ready to recycle. | ||||||
|   | |||||||
| @@ -1,26 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="ITransform"/> field. |  | ||||||
| /// </summary> |  | ||||||
| public interface IAssignableGameManager : IAssignable |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="IGameManager"/> value has has been assigned a new value. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IAssignableGameManager>? OnGameManagerAssigned { get; set; } |  | ||||||
|  |  | ||||||
|     /// <inheritdoc cref="IGameManager" /> |  | ||||||
|     IGameManager GameManager { get; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Assign a value to the <see cref="IGameManager"/> field of this object. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="gameManager">New <see cref="IGameManager"/> to assign.</param> |  | ||||||
|     /// <returns>  |  | ||||||
|     /// <see cref="true"/>, if the value given assigned successfully assigned, <see cref="false"/> if not. |  | ||||||
|     /// </returns> |  | ||||||
|     bool Assign(IGameManager gameManager); |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IGameObject"/> field. |  | ||||||
| /// </summary> |  | ||||||
| public interface IAssignableGameObject : IAssignable |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="IGameObject"/> value has has been assigned a new value. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IAssignableGameObject>? OnGameObjectAssigned { get; set; } |  | ||||||
|  |  | ||||||
|     /// <inheritdoc cref="IGameObject" /> |  | ||||||
|     IGameObject GameObject { get; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Assign a value to the <see cref="IGameObject"/> field of this object. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="gameObject">New <see cref="IGameObject"/> to assign.</param> |  | ||||||
|     /// <returns>  |  | ||||||
|     /// <see cref="true"/>, if the value given assigned successfully assigned, <see cref="false"/> if not. |  | ||||||
|     /// </returns> |  | ||||||
|     bool Assign(IGameObject gameObject); |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="ITransform"/> field. |  | ||||||
| /// </summary> |  | ||||||
| public interface IAssignableTransform : IAssignable |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="ITransform"/> value has has been assigned a new value. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IAssignableTransform>? OnTransformAssigned { get; set; } |  | ||||||
|  |  | ||||||
|     /// <inheritdoc cref="ITransform" /> |  | ||||||
|     ITransform Transform { get; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Assign a value to the <see cref="ITransform"/> field of this object. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="transform">New <see cref="ITransform"/> to assign.</param> |  | ||||||
|     /// <returns>  |  | ||||||
|     /// <see cref="true"/>, if the value given assigned successfully assigned, <see cref="false"/> if not. |  | ||||||
|     /// </returns> |  | ||||||
|     bool Assign(ITransform transform); |  | ||||||
| } |  | ||||||
| @@ -1,16 +1,14 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
| 
 |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
| 
 | 
 | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IBehaviourController"/> field. | /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IBehaviourController"/> field. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IAssignableBehaviourController : IAssignable | public interface IHasBehaviourController : IAssignable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="IBehaviourController"/> value has has been assigned a new value. |     /// Event triggered when the <see cref="IBehaviourController"/> value has has been assigned a new value. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IAssignableBehaviourController>? OnBehaviourControllerAssigned { get; set; } |     Event<IHasBehaviourController> OnBehaviourControllerAssigned { get; } | ||||||
| 
 | 
 | ||||||
|     /// <inheritdoc cref="IBehaviourController" /> |     /// <inheritdoc cref="IBehaviourController" /> | ||||||
|     IBehaviourController BehaviourController { get; } |     IBehaviourController BehaviourController { get; } | ||||||
| @@ -1,16 +1,14 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
| 
 |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
| 
 | 
 | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IEntity"/> field. | /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IEntity"/> field. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IAssignableEntity : IAssignable | public interface IHasEntity : IAssignable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="IEntity"/> value has has been assigned a new value. |     /// Event triggered when the <see cref="IEntity"/> value has has been assigned a new value. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IAssignableEntity>? OnEntityAssigned { get; set; } |     Event<IHasEntity> OnEntityAssigned { get; } | ||||||
| 
 | 
 | ||||||
|     /// <inheritdoc cref="IEntity" /> |     /// <inheritdoc cref="IEntity" /> | ||||||
|     IEntity Entity { get; } |     IEntity Entity { get; } | ||||||
| @@ -1,16 +1,14 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
| 
 |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
| 
 | 
 | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IStateEnable"/> field. | /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IStateEnable"/> field. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IAssignableStateEnable : IAssignable | public interface IHasStateEnable : IAssignable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="IStateEnable"/> value has has been assigned a new value. |     /// Event triggered when the <see cref="IStateEnable"/> value has has been assigned a new value. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IAssignableStateEnable>? OnStateEnableAssigned { get; set; } |     Event<IHasStateEnable> OnStateEnableAssigned { get; } | ||||||
| 
 | 
 | ||||||
|     /// <inheritdoc cref="IStateEnable" /> |     /// <inheritdoc cref="IStateEnable" /> | ||||||
|     IStateEnable StateEnable { get; } |     IStateEnable StateEnable { get; } | ||||||
							
								
								
									
										24
									
								
								Engine.Core/Abstract/Assignable/IHasUniverse.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								Engine.Core/Abstract/Assignable/IHasUniverse.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IUniverse"/> field. | ||||||
|  | /// </summary> | ||||||
|  | public interface IHasUniverse : IAssignable | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="IUniverse"/> value has has been assigned a new value. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IHasUniverse> OnUniverseAssigned { get; } | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="IUniverse" /> | ||||||
|  |     IUniverse Universe { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Assign a value to the <see cref="IUniverse"/> field of this object. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universe">New <see cref="IUniverse"/> to assign.</param> | ||||||
|  |     /// <returns>  | ||||||
|  |     /// <see cref="true"/>, if the value given assigned successfully assigned, <see cref="false"/> if not. | ||||||
|  |     /// </returns> | ||||||
|  |     bool Assign(IUniverse universe); | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								Engine.Core/Abstract/Assignable/IHasUniverseObject.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								Engine.Core/Abstract/Assignable/IHasUniverseObject.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Indicates the object is an <see cref="IAssignable"/> with an assignable <see cref="IUniverseObject"/> field. | ||||||
|  | /// </summary> | ||||||
|  | public interface IHasUniverseObject : IAssignable | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="IUniverseObject"/> value has has been assigned a new value. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IHasUniverseObject> OnUniverseObjectAssigned { get; } | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="IUniverseObject" /> | ||||||
|  |     IUniverseObject UniverseObject { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Assign a value to the <see cref="IUniverseObject"/> field of this object. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeObject">New <see cref="IUniverseObject"/> to assign.</param> | ||||||
|  |     /// <returns>  | ||||||
|  |     /// <see cref="true"/>, if the value given assigned successfully assigned, <see cref="false"/> if not. | ||||||
|  |     /// </returns> | ||||||
|  |     bool Assign(IUniverseObject universeObject); | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								Engine.Core/Abstract/IActive.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								Engine.Core/Abstract/IActive.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents an entity which can be active or not. | ||||||
|  | /// </summary> | ||||||
|  | public interface IActive | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="IsActive"/> state of the <see cref="IActive"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IActive, ActiveChangedArguments> OnActiveChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The value indicating whether the <see cref="IActive"/> is enabled. | ||||||
|  |     /// </summary> | ||||||
|  |     bool IsActive { get; } | ||||||
|  |  | ||||||
|  |     readonly record struct ActiveChangedArguments(bool PreviousState); | ||||||
|  | } | ||||||
| @@ -1,24 +1,19 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents a behaviour that any object in the game might use to interact with itself or other objects.  | /// Represents a behaviour that any object in the engine that might use to interact with itself or other objects.  | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IBehaviour : IEntity, IAssignableBehaviourController, IAssignableStateEnable, IInitialize | public interface IBehaviour : IEntity, IActive, IHasBehaviourController, IHasStateEnable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the priority of the <see cref="IBehaviour"/> changes. |     /// Event triggered when the priority of the <see cref="IBehaviour"/> changes. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IBehaviour>? OnPriorityChanged { get; set; } |     Event<IBehaviour, PriorityChangedArguments> OnPriorityChanged { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The priority of the <see cref="IBehaviour"/>. |     /// The priority of the <see cref="IBehaviour"/>. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     int Priority { get; set; } |     int Priority { get; set; } | ||||||
|  |  | ||||||
|     /// <summary> |     readonly record struct PriorityChangedArguments(int PreviousPriority); | ||||||
|     /// The value indicating whether the <see cref="IBehaviour"/> is active. |  | ||||||
|     /// </summary> |  | ||||||
|     bool IsActive { get; } |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								Engine.Core/Abstract/IBehaviour2D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								Engine.Core/Abstract/IBehaviour2D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface IBehaviour2D : IBehaviour | ||||||
|  | { | ||||||
|  |     ITransform2D Transform { get; } | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								Engine.Core/Abstract/IBehaviour3D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								Engine.Core/Abstract/IBehaviour3D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface IBehaviour3D : IBehaviour | ||||||
|  | { | ||||||
|  |     ITransform3D Transform { get; } | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								Engine.Core/Abstract/IBehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								Engine.Core/Abstract/IBehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents a collector for the class type of <typeparamref name="T"/>. | ||||||
|  | /// Provides mechanisms for tracking additions and removals, and notifies subscribers when such events occur on the assigned <see cref="IUniverse"/>. | ||||||
|  | /// </summary> | ||||||
|  | /// <typeparam name="T">The type of objects tracked by the collector.</typeparam> | ||||||
|  | public interface IBehaviourCollector<T> : IHasUniverse where T : class | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when an object of type <typeparamref name="T"/> is added to the collector. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IBehaviourCollector<T>, BehaviourCollectedArguments> OnCollected { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when an object of type <typeparamref name="T"/> is removed from the collector. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IBehaviourCollector<T>, BehaviourRemovedArguments> OnRemoved { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Amount of <typeparamref name="T"/> collected. | ||||||
|  |     /// </summary> | ||||||
|  |     int Count { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Get a <typeparamref name="T"/> collected by it's index. | ||||||
|  |     /// </summary> | ||||||
|  |     T this[System.Index index] { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Delegate for handling the <see cref="OnCollected"/> event. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The instance of the <see cref="IBehaviourCollector{T}"/> that triggered the event.</param> | ||||||
|  |     /// <param name="behaviourCollected">The object of type <typeparamref name="T"/> that was added to the collector.</param> | ||||||
|  |     readonly record struct BehaviourCollectedArguments(T BehaviourCollected); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Delegate for handling the <see cref="OnRemoved"/> event. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="BehaviourRemoved">The object of type <typeparamref name="T"/> that was removed from the collector.</param> | ||||||
|  |     readonly record struct BehaviourRemovedArguments(T BehaviourRemoved); | ||||||
|  | } | ||||||
| @@ -1,38 +1,31 @@ | |||||||
| using System; |  | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Diagnostics.CodeAnalysis; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; | namespace Engine.Core; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents a controller for managing <see cref="IBehaviour"/>s and notify them accordingly about the engine's updates. Connected to an <see cref="IGameObject"/>. | /// Represents a controller for managing <see cref="IBehaviour"/>s. Connected to an <see cref="IUniverseObject"/>. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IBehaviourController : IAssignableGameObject, IEnumerable<IBehaviour> | public interface IBehaviourController : IEntity, IHasUniverseObject | ||||||
| { | { | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered before the update of <see cref="IBehaviour"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IBehaviourController>? OnPreUpdate { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered during the update of <see cref="IBehaviour"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IBehaviourController>? OnUpdate { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered before the drawing phase. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IBehaviourController>? OnPreDraw { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when a <see cref="IBehaviour"/> is added to the <see cref="IBehaviourController"/>. |     /// Event triggered when a <see cref="IBehaviour"/> is added to the <see cref="IBehaviourController"/>. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IBehaviourController, IBehaviour>? OnBehaviourAdded { get; set; } |     Event<IBehaviourController, BehaviourAddedArguments> OnBehaviourAdded { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when a <see cref="IBehaviour"/> is removed from the <see cref="IBehaviourController"/>. |     /// Event triggered when a <see cref="IBehaviour"/> is removed from the <see cref="IBehaviourController"/>. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IBehaviourController, IBehaviour>? OnBehaviourRemoved { get; set; } |     Event<IBehaviourController, BehaviourRemovedArguments> OnBehaviourRemoved { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Amount of <see cref="IBehaviour"/> collected. | ||||||
|  |     /// </summary> | ||||||
|  |     int Count { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Get a <see cref="IBehaviour"/> collected by it's index. | ||||||
|  |     /// </summary> | ||||||
|  |     IBehaviour this[System.Index index] { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Adds a <see cref="IBehaviour"/> to the <see cref="IBehaviourController"/>. |     /// Adds a <see cref="IBehaviour"/> to the <see cref="IBehaviourController"/>. | ||||||
| @@ -57,27 +50,19 @@ public interface IBehaviourController : IAssignableGameObject, IEnumerable<IBeha | |||||||
|     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, <see cref="null"/>.</returns> |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, <see cref="null"/>.</returns> | ||||||
|     T? GetBehaviour<T>(); |     T? GetBehaviour<T>(); | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Tries to get a <see cref="IBehaviour"/> of the specified type. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> |  | ||||||
|     /// <param name="behaviour">When this method returns, contains the <see cref="IBehaviour"/> of the specified type, if found; otherwise, see.</param> |  | ||||||
|     /// <returns><see cref="true"/> if a <see cref="IBehaviour"/> of the specified type was found; otherwise, <see cref="false"/>.</returns> |  | ||||||
|     bool TryGetBehaviour<T>([NotNullWhen(returnValue: true)] out T? behaviour); |  | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Gets all <see cref="IBehaviour"/>s of the specified type. |     /// Gets all <see cref="IBehaviour"/>s of the specified type. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> |     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> | ||||||
|     /// <returns>A list of <see cref="IBehaviour"/>s of the specified type.</returns> |     /// <returns>A list of <see cref="IBehaviour"/>s of the specified type.</returns> | ||||||
|     IList<T> GetBehaviours<T>(); |     IReadOnlyList<T> GetBehaviours<T>(); | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Gets all <see cref="IBehaviour"/>s of the specified type and stores them in the provided list. |     /// Gets all <see cref="IBehaviour"/>s of the specified type and stores them in the provided list. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> |     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> | ||||||
|     /// <param name="behaviours">The list to store the <see cref="IBehaviour"/>s.</param> |     /// <param name="results">The list to store the <see cref="IBehaviour"/>s.</param> | ||||||
|     void GetBehaviours<T>(List<T> behaviours); |     void GetBehaviours<T>(IList<T> results); | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Removes <see cref="IBehaviour"/>s of the specified type from the <see cref="IBehaviourController"/>. |     /// Removes <see cref="IBehaviour"/>s of the specified type from the <see cref="IBehaviourController"/>. | ||||||
| @@ -93,13 +78,6 @@ public interface IBehaviourController : IAssignableGameObject, IEnumerable<IBeha | |||||||
|     /// <param name="behaviour">The <see cref="IBehaviour"/> to remove.</param> |     /// <param name="behaviour">The <see cref="IBehaviour"/> to remove.</param> | ||||||
|     void RemoveBehaviour<T>(T behaviour) where T : class, IBehaviour; |     void RemoveBehaviour<T>(T behaviour) where T : class, IBehaviour; | ||||||
|  |  | ||||||
|     /// <summary> |     readonly record struct BehaviourAddedArguments(IBehaviour BehaviourAdded); | ||||||
|     /// Updates all <see cref="IBehaviour"/>s in the <see cref="IBehaviourController"/>. |     readonly record struct BehaviourRemovedArguments(IBehaviour BehaviourRemoved); | ||||||
|     /// </summary> |  | ||||||
|     void Update(); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Performs pre-draw operations. |  | ||||||
|     /// </summary> |  | ||||||
|     void UpdatePreDraw(); |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| namespace Syntriax.Engine.Core.Abstract; | namespace Engine.Core; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents a 2D camera in the engine. | /// Represents a 2D camera in the engine. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface ICamera2D : IBehaviour, IAssignableTransform | public interface ICamera2D : IBehaviour2D | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The zoom level of the camera. |     /// The zoom level of the camera. | ||||||
|   | |||||||
							
								
								
									
										55
									
								
								Engine.Core/Abstract/ICamera3D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								Engine.Core/Abstract/ICamera3D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents a 3D camera in the engine. | ||||||
|  | /// </summary> | ||||||
|  | public interface ICamera3D : IBehaviour3D | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the near plane of the <see cref="ICamera3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ICamera3D, NearPlaneChangedArguments> OnNearPlaneChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the far plane of the <see cref="ICamera3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ICamera3D, FarPlaneChangedArguments> OnFarPlaneChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the field of view of the <see cref="ICamera3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ICamera3D, FieldOfViewChangedArguments> OnFieldOfViewChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Near plane distance of the camera. | ||||||
|  |     /// </summary> | ||||||
|  |     float NearPlane { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Far plane distance of the camera. | ||||||
|  |     /// </summary> | ||||||
|  |     float FarPlane { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Field of View (FOV) value of the camera in degrees. | ||||||
|  |     /// </summary> | ||||||
|  |     float FieldOfView { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Converts a position from screen coordinates to a <see cref="Ray3D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="screenPosition">The position in screen coordinates.</param> | ||||||
|  |     /// <returns>The <see cref="Ray3D"/> originating from the camera to the screen position in world coordinates.</returns> | ||||||
|  |     Ray3D ScreenToWorldRay(Vector2D screenPosition); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Converts a position from world coordinates to screen coordinates. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="worldPosition">The position in world coordinates.</param> | ||||||
|  |     /// <returns>The position in screen coordinates.</returns> | ||||||
|  |     Vector2D WorldToScreenPosition(Vector3D worldPosition); | ||||||
|  |  | ||||||
|  |     readonly record struct NearPlaneChangedArguments(float PreviousNearPlane); | ||||||
|  |     readonly record struct FarPlaneChangedArguments(float PreviousFarPlane); | ||||||
|  |     readonly record struct FieldOfViewChangedArguments(float PreviousFieldOfView); | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								Engine.Core/Abstract/ICoroutineYield.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								Engine.Core/Abstract/ICoroutineYield.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface ICoroutineYield | ||||||
|  | { | ||||||
|  |     bool Yield(); | ||||||
|  | } | ||||||
| @@ -1,20 +1,20 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents a basic entity in the engine. | /// Represents a basic entity in the engine. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IEntity : IInitialize, IAssignableStateEnable | public interface IEntity : IInitializable, IHasStateEnable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="Id"/> of the <see cref="IEntity"/> changes. |     /// Event triggered when the <see cref="Id"/> of the <see cref="IEntity"/> changes. | ||||||
|     /// The string action parameter is the previous <see cref="Id"/> of the <see cref="IEntity"/>.  |     /// The string action parameter is the previous <see cref="Id"/> of the <see cref="IEntity"/>.  | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IEntity, string>? OnIdChanged { get; set; } |     Event<IEntity, IdChangedArguments> OnIdChanged { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The ID of the <see cref="IEntity"/>. |     /// The ID of the <see cref="IEntity"/>. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     string Id { get; set; } |     string Id { get; set; } | ||||||
|  |  | ||||||
|  |     readonly record struct IdChangedArguments(string PreviousId); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,57 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Collections.Generic; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Represents a game world responsible for managing <see cref="IGameObject"/>s. |  | ||||||
| /// </summary> |  | ||||||
| public interface IGameManager : IEntity, IEnumerable<IGameObject> |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when a <see cref="IGameObject"/> is registered to the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IGameManager, IGameObject>? OnGameObjectRegistered { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when a <see cref="IGameObject"/> is unregistered from the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IGameManager, IGameObject>? OnGameObjectUnRegistered { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Gets a read-only list of <see cref="IGameObject"/>s managed by the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     IReadOnlyList<IGameObject> GameObjects { get; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Registers a <see cref="IGameObject"/> to the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="gameObject">The <see cref="IGameObject"/> to register.</param> |  | ||||||
|     void RegisterGameObject(IGameObject gameObject); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Instantiates a <see cref="IGameObject"/> of type T with the given arguments and registers it to the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <typeparam name="T">The type of <see cref="IGameObject"/> to instantiate.</typeparam> |  | ||||||
|     /// <param name="args">Constructor parameters for the given type of <see cref="IGameObject"/>.</param> |  | ||||||
|     /// <returns>The instantiated <see cref="IGameObject"/>.</returns> |  | ||||||
|     T InstantiateGameObject<T>(params object?[]? args) where T : class, IGameObject; |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Removes a <see cref="IGameObject"/> from the <see cref="IGameManager"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="gameObject">The <see cref="IGameObject"/> to remove.</param> |  | ||||||
|     /// <returns>The removed <see cref="IGameObject"/>.</returns> |  | ||||||
|     IGameObject RemoveGameObject(IGameObject gameObject); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Updates the <see cref="IGameManager"/> with the given engine time data. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="time">The engine time.</param> |  | ||||||
|     void Update(EngineTime time); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Performs operations that should be done before the draw calls. |  | ||||||
|     /// </summary> |  | ||||||
|     void PreDraw(); |  | ||||||
| } |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Represents a game object with various properties and functionalities. |  | ||||||
| /// </summary> |  | ||||||
| public interface IGameObject : IEntity, IAssignableGameManager, IAssignableTransform, IAssignableBehaviourController, INameable, IInitialize |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="Update"/> method is called. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<IGameObject>? OnUpdated { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Updates the game object. |  | ||||||
|     /// </summary> |  | ||||||
|     void Update(); |  | ||||||
| } |  | ||||||
| @@ -1,26 +1,24 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
| 
 |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
| 
 | 
 | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents an entity that can be initialized and finalized. This information is useful for objects we know that are not in use and can be either recycled or dropped for garbage collection. | /// Represents an entity that can be initialized and finalized. This information is useful for objects we know that are not in use and can be either recycled or dropped for garbage collection. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IInitialize | public interface IInitializable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="Initialize"/> method is called successfully. |     /// Event triggered when the <see cref="Initialize"/> method is called successfully. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IInitialize>? OnInitialized { get; set; } |     Event<IInitializable> OnInitialized { get; } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="Finalize"/> method is called successfully. |     /// Event triggered when the <see cref="IInitializable"/> method is called successfully. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IInitialize>? OnFinalized { get; set; } |     Event<IInitializable> OnFinalized { get; } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The value indicating whether the entity has been initialized. |     /// The value indicating whether the entity has been initialized. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     bool Initialized { get; } |     bool IsInitialized { get; } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Initializes the entity. |     /// Initializes the entity. | ||||||
| @@ -1,6 +1,4 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents an entity with a name. | /// Represents an entity with a name. | ||||||
| @@ -10,10 +8,12 @@ public interface INameable | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the name of the entity changes. |     /// Event triggered when the name of the entity changes. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IEntity>? OnNameChanged { get; set; } |     Event<INameable, NameChangedArguments> OnNameChanged { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The name of the entity. |     /// The name of the entity. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     string Name { get; set; } |     string Name { get; set; } | ||||||
|  |  | ||||||
|  |     readonly record struct NameChangedArguments(string PreviousName); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,19 +1,19 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// Represents an entity with an enable state that can be toggled. | /// Represents an entity with an enable state that can be toggled. | ||||||
| /// </summary> | /// </summary> | ||||||
| public interface IStateEnable : IAssignableEntity | public interface IStateEnable : IHasEntity | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Event triggered when the <see cref="Enabled"/> state of the <see cref="IStateEnable"/> changes. |     /// Event triggered when the <see cref="Enabled"/> state of the <see cref="IStateEnable"/> changes. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     Action<IStateEnable>? OnEnabledChanged { get; set; } |     Event<IStateEnable, EnabledChangedArguments> OnEnabledChanged { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The value indicating whether the <see cref="IStateEnable"/> is enabled. |     /// The value indicating whether the <see cref="IStateEnable"/> is enabled. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     bool Enabled { get; set; } |     bool Enabled { get; set; } | ||||||
|  |  | ||||||
|  |     readonly record struct EnabledChangedArguments(bool PreviousState); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,39 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Represents the transformation properties of an object such as position, scale, and rotation. |  | ||||||
| /// </summary> |  | ||||||
| public interface ITransform |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="Position"/> of the <see cref="ITransform"/> changes. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<ITransform>? OnPositionChanged { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="Scale"/> of the <see cref="ITransform"/> changes. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<ITransform>? OnScaleChanged { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Event triggered when the <see cref="Rotation"/> of the <see cref="ITransform"/> changes. |  | ||||||
|     /// </summary> |  | ||||||
|     Action<ITransform>? OnRotationChanged { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// The position of the <see cref="ITransform"/> in 2D space. |  | ||||||
|     /// </summary> |  | ||||||
|     Vector2D Position { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// The scale of the <see cref="ITransform"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     Vector2D Scale { get; set; } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// The rotation of the <see cref="ITransform"/> in degrees. |  | ||||||
|     /// </summary> |  | ||||||
|     float Rotation { get; set; } |  | ||||||
| } |  | ||||||
							
								
								
									
										95
									
								
								Engine.Core/Abstract/ITransform2D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								Engine.Core/Abstract/ITransform2D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents the transformation properties of an object such as position, scale, and rotation in 2D space. | ||||||
|  | /// </summary> | ||||||
|  | public interface ITransform2D : IBehaviour | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Position"/> of the <see cref="ITransform2D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform2D, PositionChangedArguments> OnPositionChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Scale"/> of the <see cref="ITransform2D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform2D, ScaleChangedArguments> OnScaleChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Rotation"/> of the <see cref="ITransform2D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform2D, RotationChangedArguments> OnRotationChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when any of the properties of the <see cref="ITransform2D"/> gets updated. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform2D> OnTransformUpdated { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector2D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Up { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector2D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Down { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector2D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Left { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector2D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Right { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world position of the <see cref="ITransform2D"/> in 2D space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Position { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world scale of the <see cref="ITransform2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D Scale { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world rotation of the <see cref="ITransform2D"/> in degrees. | ||||||
|  |     /// </summary> | ||||||
|  |     float Rotation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local position of the <see cref="ITransform2D"/> in 2D space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D LocalPosition { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local scale of the <see cref="ITransform2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector2D LocalScale { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local rotation of the <see cref="ITransform2D"/> in degrees. | ||||||
|  |     /// </summary> | ||||||
|  |     float LocalRotation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform2D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousPosition">The previous <see cref="Position"/> of the <see cref="ITransform2D"/>.</param> | ||||||
|  |     readonly record struct PositionChangedArguments(Vector2D PreviousPosition); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform2D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousScale">The previous <see cref="Scale"/> of the <see cref="ITransform2D"/>.</param> | ||||||
|  |     readonly record struct ScaleChangedArguments(Vector2D PreviousScale); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform2D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousRotation">The previous <see cref="Rotation"/> of the <see cref="ITransform2D"/>.</param> | ||||||
|  |     readonly record struct RotationChangedArguments(float PreviousRotation); | ||||||
|  | } | ||||||
							
								
								
									
										105
									
								
								Engine.Core/Abstract/ITransform3D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								Engine.Core/Abstract/ITransform3D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents the transformation properties of an object such as position, scale, and rotation in 3D space. | ||||||
|  | /// </summary> | ||||||
|  | public interface ITransform3D : IBehaviour | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Position"/> of the <see cref="ITransform3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform3D, PositionChangedArguments> OnPositionChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Scale"/> of the <see cref="ITransform3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform3D, ScaleChangedArguments> OnScaleChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Rotation"/> of the <see cref="ITransform3D"/> changes. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform3D, RotationChangedArguments> OnRotationChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when any of the properties of the <see cref="ITransform3D"/> gets updated. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<ITransform3D> OnTransformUpdated { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Up { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Down { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Left { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing upwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Right { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing forwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Forward { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="Vector3D"/> pointing backwards in world space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Backward { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world position of the <see cref="ITransform3D"/> in 3D space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Position { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world scale of the <see cref="ITransform3D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D Scale { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The world rotation of the <see cref="ITransform3D"/> in degrees. | ||||||
|  |     /// </summary> | ||||||
|  |     Quaternion Rotation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local position of the <see cref="ITransform3D"/> in 3D space. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D LocalPosition { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local scale of the <see cref="ITransform3D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Vector3D LocalScale { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The local rotation of the <see cref="ITransform3D"/> in degrees. | ||||||
|  |     /// </summary> | ||||||
|  |     Quaternion LocalRotation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform3D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousPosition">The previous <see cref="Position"/> of the <see cref="ITransform3D"/>.</param> | ||||||
|  |     readonly record struct PositionChangedArguments(Vector3D PreviousPosition); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform3D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousScale">The previous <see cref="Scale"/> of the <see cref="ITransform3D"/>.</param> | ||||||
|  |     readonly record struct ScaleChangedArguments(Vector3D PreviousScale); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="ITransform3D"/>'s rotation changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="PreviousRotation">The previous <see cref="Rotation"/> of the <see cref="ITransform3D"/>.</param> | ||||||
|  |     readonly record struct RotationChangedArguments(Quaternion PreviousRotation); | ||||||
|  | } | ||||||
							
								
								
									
										120
									
								
								Engine.Core/Abstract/IUniverse.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								Engine.Core/Abstract/IUniverse.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents a universe responsible for managing <see cref="IUniverseObject"/>s. | ||||||
|  | /// </summary> | ||||||
|  | public interface IUniverse : IEntity, IEnumerable<IUniverseObject> | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when <see cref="Update(UniverseTime)"/> is about to be called called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UpdateArguments> OnPreUpdate { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when <see cref="Update(UniverseTime)"/> is called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UpdateArguments> OnUpdate { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered after <see cref="Update(UniverseTime)"/> is called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UpdateArguments> OnPostUpdate { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when <see cref="Draw"/> is about to be called called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse> OnPreDraw { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when <see cref="Draw"/> is called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse> OnDraw { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered after <see cref="Draw"/> is called on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse> OnPostDraw { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when a <see cref="IUniverseObject"/> is about to be registered to the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UniverseObjectRegisteredArguments> OnPreUniverseObjectRegistered { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when a <see cref="IUniverseObject"/> is registered to the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UniverseObjectRegisteredArguments> OnUniverseObjectRegistered { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when a <see cref="IUniverseObject"/> is about to be unregistered from the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UniverseObjectUnRegisteredArguments> OnPreUniverseObjectUnRegistered { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when a <see cref="IUniverseObject"/> is unregistered from the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, UniverseObjectUnRegisteredArguments> OnUniverseObjectUnRegistered { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when <see cref="TimeScale"/> is changed on the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverse, TimeScaleChangedArguments> OnTimeScaleChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Current time scale the <see cref="IUniverse"/> operates on. | ||||||
|  |     /// </summary> | ||||||
|  |     float TimeScale { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Contains time data related to this <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     UniverseTime Time { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Contains unscaled time data related to this <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     UniverseTime UnscaledTime { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a read-only list of <see cref="IUniverseObject"/>s managed by the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     IReadOnlyList<IUniverseObject> UniverseObjects { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Registers an <see cref="IUniverseObject"/> to the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to register.</param> | ||||||
|  |     void Register(IUniverseObject universeObject); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Instantiates a <see cref="IUniverseObject"/> of type T with the given arguments and registers it to the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to instantiate.</typeparam> | ||||||
|  |     /// <param name="args">Constructor parameters for the given type of <see cref="IUniverseObject"/>.</param> | ||||||
|  |     /// <returns>The instantiated <see cref="IUniverseObject"/>.</returns> | ||||||
|  |     T InstantiateUniverseObject<T>(params object?[]? args) where T : class, IUniverseObject; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Removes an <see cref="IUniverseObject"/> from the <see cref="IUniverse"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to remove.</param> | ||||||
|  |     void Remove(IUniverseObject universeObject); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Updates the <see cref="IUniverse"/> with the given delta time. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeTime">Delta time.</param> | ||||||
|  |     void Update(UniverseTime universeTime); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Performs operations that should be done to the draw. | ||||||
|  |     /// </summary> | ||||||
|  |     void Draw(); | ||||||
|  |  | ||||||
|  |     readonly record struct TimeScaleChangedArguments(float PreviousTimeScale); | ||||||
|  |     readonly record struct UpdateArguments(UniverseTime EngineTime); | ||||||
|  |     readonly record struct UniverseObjectRegisteredArguments(IUniverseObject UniverseObjectRegistered); | ||||||
|  |     readonly record struct UniverseObjectUnRegisteredArguments(IUniverseObject UniverseObjectUnregistered); | ||||||
|  | } | ||||||
							
								
								
									
										125
									
								
								Engine.Core/Abstract/IUniverseObject.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								Engine.Core/Abstract/IUniverseObject.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents an <see cref="IEntity"/> that can enter and exit a universe within the <see cref="IUniverse"/> system. | ||||||
|  | /// This interface allows for tracking the object's presence in the universe and provides events  | ||||||
|  | /// for notifying when the see enters or exits the universe. | ||||||
|  | /// </summary> | ||||||
|  | public interface IUniverseObject : IEntity, IActive, INameable, IHasBehaviourController | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="IUniverseObject"/> enters the universe. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverseObject, EnteredUniverseArguments> OnEnteredUniverse { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="IUniverseObject"/> exits the universe. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverseObject, ExitedUniverseArguments> OnExitedUniverse { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when the <see cref="Parent"/> of the <see cref="IUniverseObject"/> changes. The second parameter is the old <see cref="IUniverseObject"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverseObject, ParentChangedArguments> OnParentChanged { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when a new <see cref="IUniverseObject"/> is added to the <see cref="Children"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverseObject, ChildrenAddedArguments> OnChildrenAdded { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Event triggered when an <see cref="IUniverseObject"/> is removed from the <see cref="Children"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     Event<IUniverseObject, ChildrenRemovedArguments> OnChildrenRemoved { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets the <see cref="IUniverse"/> this <see cref="IUniverseObject"/> is connected to, if any. | ||||||
|  |     /// </summary> | ||||||
|  |     IUniverse Universe { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Indicates whether the <see cref="IUniverseObject"/> is currently in the universe. | ||||||
|  |     /// </summary> | ||||||
|  |     bool IsInUniverse { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The parent <see cref="IUniverseObject"/> of the <see cref="IUniverseObject"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     IUniverseObject? Parent { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The <see cref="IUniverseObject"/>s that have this <see cref="IUniverseObject"/> as their <see cref="Parent"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     IReadOnlyList<IUniverseObject> Children { get; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Internal method to handle entering the universe. | ||||||
|  |     /// This should be called by the system to properly manage universe states. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universe">The <see cref="IUniverse"/> that is managing this universe.</param> | ||||||
|  |     /// <returns> | ||||||
|  |     /// <see cref="true"/> if the <see cref="IUniverseObject"/> successfully entered the universe;  | ||||||
|  |     /// <see cref="false"/> if it failed to do so. | ||||||
|  |     /// </returns> | ||||||
|  |     internal bool EnterUniverse(IUniverse universe); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Internal method to handle exiting the universe. | ||||||
|  |     /// This should be called by the system to properly manage universe states. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <returns> | ||||||
|  |     /// <see cref="true"/> if the <see cref="IUniverseObject"/> successfully exited the universe;  | ||||||
|  |     /// <see cref="false"/> if it failed to do so. | ||||||
|  |     /// </returns> | ||||||
|  |     internal bool ExitUniverse(); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Adds a child <see cref="IUniverseObject"/> to this <see cref="IUniverseObject"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeObject">The child <see cref="IUniverseObject"/> to add.</param> | ||||||
|  |     void AddChild(IUniverseObject universeObject); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Removes a child <see cref="IUniverseObject"/> from this <see cref="IUniverseObject"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="universeObject">The child <see cref="IUniverseObject"/> to remove.</param> | ||||||
|  |     void RemoveChild(IUniverseObject universeObject); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="IUniverseObject"/> enters the universe of a <see cref="IUniverse">. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The <see cref="IUniverseObject"/> that entered the universe.</param> | ||||||
|  |     /// <param name="universe">The <see cref="IUniverse"/> that the <see cref="IUniverseObject"/> has entered it's universe.</param> | ||||||
|  |     readonly record struct EnteredUniverseArguments(IUniverse Universe); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="IUniverseObject"/> exits the universe of a <see cref="IUniverse">. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The <see cref="IUniverseObject"/> that exited the universe.</param> | ||||||
|  |     /// <param name="universe">The <see cref="IUniverse"/> that the <see cref="IUniverseObject"/> has exited it's universe.</param> | ||||||
|  |     readonly record struct ExitedUniverseArguments(IUniverse Universe); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when the <see cref="IUniverseObject"/>'s parent changes. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The <see cref="IUniverseObject"/> that the parent has changed.</param> | ||||||
|  |     /// <param name="previousParent">The previous <see cref="IUniverseObject"/> the sender was a child of.</param> | ||||||
|  |     /// <param name="newParent">The new and current <see cref="IUniverseObject"/> the sender is a child of.</param> | ||||||
|  |     readonly record struct ParentChangedArguments(IUniverseObject? PreviousParent, IUniverseObject? CurrentParent); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Arguments for the event triggered when a new <see cref="IUniverseObject"/> added as a child. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The parent <see cref="IUniverseObject"/> this event is being called from.</param> | ||||||
|  |     /// <param name="childrenAdded">The <see cref="IUniverseObject"/> that got removed as a children of the sender <see cref="IUniverseObject"/>.</param> | ||||||
|  |     readonly record struct ChildrenAddedArguments(IUniverseObject ChildrenAdded); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Delegate for the event triggered when a new <see cref="IUniverseObject"/> removed from being a child. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The parent <see cref="IUniverseObject"/> this event is being called from.</param> | ||||||
|  |     /// <param name="childrenAdded">The <see cref="IUniverseObject"/> that got removed as a children of the sender <see cref="IUniverseObject"/>.</param> | ||||||
|  |     readonly record struct ChildrenRemovedArguments(IUniverseObject ChildrenRemoved); | ||||||
|  | } | ||||||
| @@ -1,17 +1,14 @@ | |||||||
| using System; | using System; | ||||||
| 
 | 
 | ||||||
| namespace Syntriax.Engine.Core.Abstract; | namespace Engine.Core; | ||||||
| 
 | 
 | ||||||
| public abstract class BaseEntity : IEntity | public abstract class BaseEntity : IEntity | ||||||
| { | { | ||||||
|     public Action<IEntity, string>? OnIdChanged { get; set; } = null; |     public Event<IEntity, IEntity.IdChangedArguments> OnIdChanged { get; } = new(); | ||||||
| 
 |     public Event<IInitializable> OnInitialized { get; } = new(); | ||||||
|     public Action<IAssignable>? OnUnassigned { get; set; } = null; |     public Event<IInitializable> OnFinalized { get; } = new(); | ||||||
|     public Action<IAssignableStateEnable>? OnStateEnableAssigned { get; set; } = null; |     public Event<IHasStateEnable> OnStateEnableAssigned { get; } = new(); | ||||||
| 
 |     public Event<IAssignable> OnUnassigned { get; } = new(); | ||||||
|     public Action<IInitialize>? OnInitialized { get; set; } = null; |  | ||||||
|     public Action<IInitialize>? OnFinalized { get; set; } = null; |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
|     private IStateEnable _stateEnable = null!; |     private IStateEnable _stateEnable = null!; | ||||||
| 
 | 
 | ||||||
| @@ -20,24 +17,25 @@ public abstract class BaseEntity : IEntity | |||||||
| 
 | 
 | ||||||
|     public virtual IStateEnable StateEnable => _stateEnable; |     public virtual IStateEnable StateEnable => _stateEnable; | ||||||
| 
 | 
 | ||||||
|     public virtual bool IsActive => StateEnable.Enabled; |  | ||||||
| 
 |  | ||||||
|     public string Id |     public string Id | ||||||
|     { |     { | ||||||
|         get => _id; |         get => _id; | ||||||
|         set |         set | ||||||
|         { |         { | ||||||
|  |             if (IsInitialized) | ||||||
|  |                 throw new($"Can't change {nameof(Id)} of {_id} because it's initialized"); | ||||||
|  | 
 | ||||||
|             if (value == _id) |             if (value == _id) | ||||||
|                 return; |                 return; | ||||||
| 
 | 
 | ||||||
|             string previousId = _id; |             string previousId = _id; | ||||||
| 
 | 
 | ||||||
|             _id = value; |             _id = value; | ||||||
|             OnIdChanged?.Invoke(this, previousId); |             OnIdChanged?.Invoke(this, new(previousId)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public bool Initialized |     public bool IsInitialized | ||||||
|     { |     { | ||||||
|         get => _initialized; |         get => _initialized; | ||||||
|         private set |         private set | ||||||
| @@ -53,13 +51,15 @@ public abstract class BaseEntity : IEntity | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     protected virtual void OnAssign(IStateEnable stateEnable) { } | ||||||
|     public bool Assign(IStateEnable stateEnable) |     public bool Assign(IStateEnable stateEnable) | ||||||
|     { |     { | ||||||
|         if (Initialized) |         if (IsInitialized) | ||||||
|             return false; |             return false; | ||||||
| 
 | 
 | ||||||
|         _stateEnable = stateEnable; |         _stateEnable = stateEnable; | ||||||
|         _stateEnable.Assign(this); |         _stateEnable.Assign(this); | ||||||
|  |         OnAssign(stateEnable); | ||||||
|         OnStateEnableAssigned?.Invoke(this); |         OnStateEnableAssigned?.Invoke(this); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| @@ -67,11 +67,13 @@ public abstract class BaseEntity : IEntity | |||||||
|     protected virtual void UnassignInternal() { } |     protected virtual void UnassignInternal() { } | ||||||
|     public bool Unassign() |     public bool Unassign() | ||||||
|     { |     { | ||||||
|         if (Initialized) |         if (IsInitialized) | ||||||
|             return false; |             return false; | ||||||
| 
 | 
 | ||||||
|         UnassignInternal(); |         UnassignInternal(); | ||||||
| 
 | 
 | ||||||
|  |         _stateEnable = null!; | ||||||
|  |         _stateEnable.Unassign(); | ||||||
|         OnUnassigned?.Invoke(this); |         OnUnassigned?.Invoke(this); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| @@ -79,24 +81,26 @@ public abstract class BaseEntity : IEntity | |||||||
|     protected virtual void InitializeInternal() { } |     protected virtual void InitializeInternal() { } | ||||||
|     public bool Initialize() |     public bool Initialize() | ||||||
|     { |     { | ||||||
|         if (Initialized) |         if (IsInitialized) | ||||||
|             return false; |             return false; | ||||||
| 
 | 
 | ||||||
|  |         _stateEnable ??= Factory.StateEnableFactory.Instantiate(this); | ||||||
|  | 
 | ||||||
|         InitializeInternal(); |         InitializeInternal(); | ||||||
| 
 | 
 | ||||||
|         Initialized = true; |         IsInitialized = true; | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     protected virtual void FinalizeInternal() { } |     protected virtual void FinalizeInternal() { } | ||||||
|     public bool Finalize() |     public bool Finalize() | ||||||
|     { |     { | ||||||
|         if (!Initialized) |         if (!IsInitialized) | ||||||
|             return false; |             return false; | ||||||
| 
 | 
 | ||||||
|         FinalizeInternal(); |         FinalizeInternal(); | ||||||
| 
 | 
 | ||||||
|         Initialized = false; |         IsInitialized = false; | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -1,26 +1,23 @@ | |||||||
| using System; | namespace Engine.Core; | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| [System.Diagnostics.DebuggerDisplay("{GetType().Name, nq}, Priority: {Priority}, Initialized: {Initialized}")] | [System.Diagnostics.DebuggerDisplay("{GetType().Name, nq}, Priority: {Priority}, Initialized: {Initialized}")] | ||||||
| public abstract class Behaviour : BaseEntity, IBehaviour | public abstract class Behaviour : BaseEntity, IBehaviour | ||||||
| { | { | ||||||
|     public Action<IAssignableBehaviourController>? OnBehaviourControllerAssigned { get; set; } = null; |     public Event<IBehaviour, IBehaviour.PriorityChangedArguments> OnPriorityChanged { get; } = new(); | ||||||
|  |     public Event<IActive, IActive.ActiveChangedArguments> OnActiveChanged { get; } = new(); | ||||||
|  |     public Event<IHasBehaviourController> OnBehaviourControllerAssigned { get; } = new(); | ||||||
|  |  | ||||||
|     public Action<IBehaviour>? OnPriorityChanged { get; set; } = null; |     private readonly Event<IHasUniverseObject>.EventHandler delegateOnUniverseObjectAssigned = null!; | ||||||
|  |     private readonly Event<IActive, IActive.ActiveChangedArguments>.EventHandler delegateOnUniverseObjectActiveChanged = null!; | ||||||
|  |     private readonly Event<IStateEnable, IStateEnable.EnabledChangedArguments>.EventHandler delegateOnStateEnabledChanged = null!; | ||||||
|  |  | ||||||
|  |     public IUniverse Universe => BehaviourController.UniverseObject.Universe; | ||||||
|  |     public IUniverseObject UniverseObject => BehaviourController.UniverseObject; | ||||||
|  |  | ||||||
|     private IBehaviourController _behaviourController = null!; |     private IBehaviourController _behaviourController = null!; | ||||||
|  |  | ||||||
|     private int _priority = 0; |  | ||||||
|  |  | ||||||
|     public IBehaviourController BehaviourController => _behaviourController; |     public IBehaviourController BehaviourController => _behaviourController; | ||||||
|  |  | ||||||
|     public override bool IsActive => base.IsActive && BehaviourController.GameObject.StateEnable.Enabled; |     private int _priority = 0; | ||||||
|  |  | ||||||
|     public int Priority |     public int Priority | ||||||
|     { |     { | ||||||
|         get => _priority; |         get => _priority; | ||||||
| @@ -29,31 +26,78 @@ public abstract class Behaviour : BaseEntity, IBehaviour | |||||||
|             if (value == _priority) |             if (value == _priority) | ||||||
|                 return; |                 return; | ||||||
|  |  | ||||||
|  |             int previousPriority = _priority; | ||||||
|             _priority = value; |             _priority = value; | ||||||
|             OnPriorityChanged?.Invoke(this); |             OnPriorityChanged?.Invoke(this, new(previousPriority)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private bool _isActive = false; | ||||||
|  |     public bool IsActive => _isActive; | ||||||
|  |  | ||||||
|  |     protected virtual void OnAssign(IBehaviourController behaviourController) { } | ||||||
|     public bool Assign(IBehaviourController behaviourController) |     public bool Assign(IBehaviourController behaviourController) | ||||||
|     { |     { | ||||||
|         if (Initialized) |         if (IsInitialized) | ||||||
|             return false; |             return false; | ||||||
|  |  | ||||||
|         _behaviourController = behaviourController; |         _behaviourController = behaviourController; | ||||||
|  |         OnAssign(behaviourController); | ||||||
|  |         behaviourController.OnUniverseObjectAssigned.AddListener(delegateOnUniverseObjectAssigned); | ||||||
|  |         behaviourController.StateEnable.OnEnabledChanged.AddListener(delegateOnStateEnabledChanged); | ||||||
|  |         if (behaviourController.UniverseObject is not null) | ||||||
|  |             OnUniverseObjectAssigned(behaviourController); | ||||||
|         OnBehaviourControllerAssigned?.Invoke(this); |         OnBehaviourControllerAssigned?.Invoke(this); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private void OnUniverseObjectAssigned(IHasUniverseObject sender) | ||||||
|  |     { | ||||||
|  |         sender.UniverseObject.OnActiveChanged.AddListener(delegateOnUniverseObjectActiveChanged); | ||||||
|  |         UpdateActive(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override void OnAssign(IStateEnable stateEnable) | ||||||
|  |     { | ||||||
|  |         base.OnAssign(stateEnable); | ||||||
|  |  | ||||||
|  |         stateEnable.OnEnabledChanged.AddListener(delegateOnStateEnabledChanged); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     protected override void UnassignInternal() |     protected override void UnassignInternal() | ||||||
|     { |     { | ||||||
|  |         BehaviourController.UniverseObject.OnActiveChanged.RemoveListener(delegateOnUniverseObjectActiveChanged); | ||||||
|  |         StateEnable.OnEnabledChanged.RemoveListener(delegateOnStateEnabledChanged); | ||||||
|  |         BehaviourController.OnUniverseObjectAssigned.RemoveListener(delegateOnUniverseObjectAssigned); | ||||||
|  |         BehaviourController.StateEnable.OnEnabledChanged.RemoveListener(delegateOnStateEnabledChanged); | ||||||
|         base.UnassignInternal(); |         base.UnassignInternal(); | ||||||
|         _behaviourController = null!; |         _behaviourController = null!; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected override void InitializeInternal() |     protected override void InitializeInternal() | ||||||
|     { |     { | ||||||
|         base.InitializeInternal(); |         Debug.Assert.AssertBehaviourControllerAssigned(this); | ||||||
|         NotAssignedException.Check(this, _behaviourController); |         Debug.Assert.AssertStateEnableAssigned(this); | ||||||
|         NotAssignedException.Check(this, StateEnable); |  | ||||||
|  |         UpdateActive(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void OnStateEnabledChanged(IStateEnable sender, IStateEnable.EnabledChangedArguments args) => UpdateActive(); | ||||||
|  |     private void OnUniverseObjectActiveChanged(IActive sender, IActive.ActiveChangedArguments args) => UpdateActive(); | ||||||
|  |  | ||||||
|  |     private void UpdateActive() | ||||||
|  |     { | ||||||
|  |         bool previousActive = IsActive; | ||||||
|  |         _isActive = StateEnable.Enabled && _behaviourController.StateEnable.Enabled && _behaviourController.UniverseObject.IsActive; | ||||||
|  |  | ||||||
|  |         if (previousActive != IsActive) | ||||||
|  |             OnActiveChanged?.Invoke(this, new(previousActive)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected Behaviour() | ||||||
|  |     { | ||||||
|  |         delegateOnUniverseObjectAssigned = OnUniverseObjectAssigned; | ||||||
|  |         delegateOnUniverseObjectActiveChanged = OnUniverseObjectActiveChanged; | ||||||
|  |         delegateOnStateEnabledChanged = OnStateEnabledChanged; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								Engine.Core/Behaviour2D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								Engine.Core/Behaviour2D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | // TODO this should not use independent behaviour, the OnInitialize usage for getting the transform can cause very unexpected issues | ||||||
|  | public abstract class Behaviour2D : Internal.BehaviourIndependent, IBehaviour2D | ||||||
|  | { | ||||||
|  |     public ITransform2D Transform { get; private set; } = null!; | ||||||
|  |  | ||||||
|  |     protected override void OnInitialize() => Transform = BehaviourController.GetRequiredBehaviour<ITransform2D>(); | ||||||
|  |     protected override void OnFinalize() => Transform = null!; | ||||||
|  | } | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; |  | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public class BehaviourCacher<T> : IAssignableGameManager, IEnumerable<T> |  | ||||||
| { |  | ||||||
|     public Action<IAssignable>? OnUnassigned { get; set; } = null; |  | ||||||
|     public Action<IAssignableGameManager>? OnGameManagerAssigned { get; set; } = null; |  | ||||||
|  |  | ||||||
|     public Action<BehaviourCacher<T>, T>? OnCached { get; set; } = null; |  | ||||||
|     public Action<BehaviourCacher<T>, T>? OnUncached { get; set; } = null; |  | ||||||
|  |  | ||||||
|     private readonly List<T> _behaviours = new(32); |  | ||||||
|  |  | ||||||
|     public IReadOnlyList<T> Behaviours => _behaviours; |  | ||||||
|     public IGameManager GameManager { get; private set; } = null!; |  | ||||||
|  |  | ||||||
|     public T this[Index index] => _behaviours[index]; |  | ||||||
|  |  | ||||||
|     public BehaviourCacher() { } |  | ||||||
|     public BehaviourCacher(IGameManager gameManager) => Assign(gameManager); |  | ||||||
|  |  | ||||||
|     private void OnGameObjectRegistered(IGameManager manager, IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         gameObject.BehaviourController.OnBehaviourAdded += OnBehaviourAdded; |  | ||||||
|         gameObject.BehaviourController.OnBehaviourRemoved += OnBehaviourRemoved; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private void OnGameObjectUnregistered(IGameManager manager, IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         gameObject.BehaviourController.OnBehaviourAdded -= OnBehaviourAdded; |  | ||||||
|         gameObject.BehaviourController.OnBehaviourRemoved -= OnBehaviourRemoved; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private void OnBehaviourAdded(IBehaviourController controller, IBehaviour behaviour) |  | ||||||
|     { |  | ||||||
|         if (behaviour is not T tBehaviour) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         _behaviours.Add(tBehaviour); |  | ||||||
|         OnCached?.Invoke(this, tBehaviour); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private void OnBehaviourRemoved(IBehaviourController controller, IBehaviour behaviour) |  | ||||||
|     { |  | ||||||
|         if (behaviour is not T tBehaviour) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         if (!_behaviours.Remove(tBehaviour)) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         OnUncached?.Invoke(this, tBehaviour); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Assign(IGameManager gameManager) |  | ||||||
|     { |  | ||||||
|         if (GameManager is not null) |  | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         foreach (IGameObject gameObject in gameManager) |  | ||||||
|         { |  | ||||||
|             OnGameObjectRegistered(gameManager, gameObject); |  | ||||||
|             foreach (IBehaviour behaviour in gameObject.BehaviourController) |  | ||||||
|                 OnBehaviourAdded(gameObject.BehaviourController, behaviour); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         gameManager.OnGameObjectRegistered += OnGameObjectRegistered; |  | ||||||
|         gameManager.OnGameObjectUnRegistered += OnGameObjectUnregistered; |  | ||||||
|  |  | ||||||
|         GameManager = gameManager; |  | ||||||
|         OnGameManagerAssigned?.Invoke(this); |  | ||||||
|  |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Unassign() |  | ||||||
|     { |  | ||||||
|         if (GameManager is null) |  | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         foreach (IGameObject gameObject in GameManager) |  | ||||||
|         { |  | ||||||
|             OnGameObjectUnregistered(GameManager, gameObject); |  | ||||||
|             foreach (IBehaviour behaviour in gameObject.BehaviourController) |  | ||||||
|                 OnBehaviourRemoved(gameObject.BehaviourController, behaviour); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         GameManager.OnGameObjectRegistered -= OnGameObjectRegistered; |  | ||||||
|         GameManager.OnGameObjectUnRegistered -= OnGameObjectUnregistered; |  | ||||||
|  |  | ||||||
|         GameManager = null!; |  | ||||||
|         OnUnassigned?.Invoke(this); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public IEnumerator<T> GetEnumerator() => _behaviours.GetEnumerator(); |  | ||||||
|     IEnumerator IEnumerable.GetEnumerator() => _behaviours.GetEnumerator(); |  | ||||||
| } |  | ||||||
| @@ -1,91 +1,78 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Diagnostics.CodeAnalysis; |  | ||||||
| using System.Linq; |  | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| [System.Diagnostics.DebuggerDisplay("Behaviour Count: {behaviours.Count}")] | [System.Diagnostics.DebuggerDisplay("Behaviour Count: {behaviours.Count}")] | ||||||
| public class BehaviourController : IBehaviourController | public class BehaviourController : BaseEntity, IBehaviourController | ||||||
| { | { | ||||||
|     public Action<IBehaviourController>? OnPreUpdate { get; set; } |     public Event<IBehaviourController, IBehaviourController.BehaviourAddedArguments> OnBehaviourAdded { get; } = new(); | ||||||
|     public Action<IBehaviourController>? OnUpdate { get; set; } = null; |     public Event<IBehaviourController, IBehaviourController.BehaviourRemovedArguments> OnBehaviourRemoved { get; } = new(); | ||||||
|     public Action<IBehaviourController>? OnPreDraw { get; set; } = null; |     public Event<IHasUniverseObject> OnUniverseObjectAssigned { get; } = new(); | ||||||
|  |  | ||||||
|     public Action<IBehaviourController, IBehaviour>? OnBehaviourAdded { get; set; } = null; |     private readonly FastList<IBehaviour> behaviours = new(Constants.BEHAVIOURS_SIZE_INITIAL); | ||||||
|     public Action<IBehaviourController, IBehaviour>? OnBehaviourRemoved { get; set; } = null; |  | ||||||
|     public Action<IAssignable>? OnUnassigned { get; set; } = null; |  | ||||||
|     public Action<IAssignableGameObject>? OnGameObjectAssigned { get; set; } = null; |  | ||||||
|  |  | ||||||
|  |     private IUniverseObject _universeObject = null!; | ||||||
|  |  | ||||||
|     private readonly IList<IBehaviour> behaviours = new List<IBehaviour>(Constants.BEHAVIOURS_SIZE_INITIAL); |     public IUniverseObject UniverseObject => _universeObject; | ||||||
|  |     public int Count => behaviours.Count; | ||||||
|     private IGameObject _gameObject = null!; |     public IBehaviour this[Index index] => behaviours[index]; | ||||||
|  |  | ||||||
|  |  | ||||||
|     public IGameObject GameObject => _gameObject; |  | ||||||
|  |  | ||||||
|     public T AddBehaviour<T>(T behaviour) where T : class, IBehaviour |     public T AddBehaviour<T>(T behaviour) where T : class, IBehaviour | ||||||
|     { |     { | ||||||
|         InsertBehaviourByPriority(behaviour); |         InsertBehaviourByPriority(behaviour); | ||||||
|  |  | ||||||
|         behaviour.Initialize(); |         behaviour.Assign(this); | ||||||
|         behaviour.OnPriorityChanged += OnPriorityChange; |  | ||||||
|         OnBehaviourAdded?.Invoke(this, behaviour); |         if (IsInitialized) | ||||||
|  |             behaviour.Initialize(); | ||||||
|  |  | ||||||
|  |         behaviour.OnPriorityChanged.AddListener(OnPriorityChange); | ||||||
|  |         OnBehaviourAdded?.Invoke(this, new(behaviour)); | ||||||
|         return behaviour; |         return behaviour; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public T AddBehaviour<T>(params object?[]? args) where T : class, IBehaviour |     public T AddBehaviour<T>(params object?[]? args) where T : class, IBehaviour | ||||||
|         => AddBehaviour(new Factory.BehaviourFactory().Instantiate<T>(_gameObject, args)); |     { | ||||||
|  |         T behaviour = Factory.BehaviourFactory.Instantiate<T>(args); | ||||||
|  |         return AddBehaviour(behaviour); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public T? GetBehaviour<T>() |     public T? GetBehaviour<T>() | ||||||
|     { |     { | ||||||
|         foreach (var behaviourItem in behaviours) |         foreach (IBehaviour behaviourItem in behaviours) | ||||||
|             if (behaviourItem is T result) |             if (behaviourItem is T result) | ||||||
|                 return result; |                 return result; | ||||||
|  |  | ||||||
|         return default; |         return default; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public bool TryGetBehaviour<T>([NotNullWhen(returnValue: true)] out T? behaviour) |     public IReadOnlyList<T> GetBehaviours<T>() | ||||||
|     { |     { | ||||||
|         behaviour = GetBehaviour<T>(); |         List<T> behaviours = []; | ||||||
|         return behaviour is not null; |  | ||||||
|  |         foreach (IBehaviour behaviourItem in this.behaviours) | ||||||
|  |             if (behaviourItem is T behaviour) | ||||||
|  |                 behaviours.Add(behaviour); | ||||||
|  |  | ||||||
|  |         return behaviours; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public IList<T> GetBehaviours<T>() |     public void GetBehaviours<T>(IList<T> results) | ||||||
|     { |     { | ||||||
|         List<T>? behaviours = null; |         results.Clear(); | ||||||
|         foreach (var behaviourItem in this.behaviours) |         foreach (IBehaviour behaviourItem in behaviours) | ||||||
|         { |         { | ||||||
|             if (behaviourItem is not T behaviour) |             if (behaviourItem is not T behaviour) | ||||||
|                 continue; |                 continue; | ||||||
|  |  | ||||||
|             behaviours ??= []; |             results.Add(behaviour); | ||||||
|             behaviours.Add(behaviour); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return behaviours ?? Enumerable.Empty<T>().ToList(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void GetBehaviours<T>(List<T> behaviors) |  | ||||||
|     { |  | ||||||
|         behaviors.Clear(); |  | ||||||
|         foreach (var behaviourItem in behaviours) |  | ||||||
|         { |  | ||||||
|             if (behaviourItem is not T _) |  | ||||||
|                 continue; |  | ||||||
|  |  | ||||||
|             behaviours.Add(behaviourItem); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void RemoveBehaviour<T>(bool removeAll = false) where T : class, IBehaviour |     public void RemoveBehaviour<T>(bool removeAll = false) where T : class, IBehaviour | ||||||
|     { |     { | ||||||
|         for (int i = behaviours.Count; i >= 0; i--) |         for (int i = behaviours.Count - 1; i >= 0; i--) | ||||||
|         { |         { | ||||||
|             if (behaviours[i] is not T behaviour) |             if (behaviours[i] is not T behaviour) | ||||||
|                 continue; |                 continue; | ||||||
| @@ -100,53 +87,42 @@ public class BehaviourController : IBehaviourController | |||||||
|     public void RemoveBehaviour<T>(T behaviour) where T : class, IBehaviour |     public void RemoveBehaviour<T>(T behaviour) where T : class, IBehaviour | ||||||
|     { |     { | ||||||
|         if (!behaviours.Contains(behaviour)) |         if (!behaviours.Contains(behaviour)) | ||||||
|             throw new Exception($"{behaviour.GetType().Name} does not exist in {GameObject.Name}'s {nameof(IBehaviourController)}."); |             throw new Exception($"{behaviour.GetType().Name} does not exist in {UniverseObject.Name}'s {nameof(IBehaviourController)}."); | ||||||
|  |  | ||||||
|         behaviour.OnPriorityChanged -= OnPriorityChange; |         behaviour.OnPriorityChanged.RemoveListener(OnPriorityChange); | ||||||
|         behaviour.Finalize(); |         behaviour.Finalize(); | ||||||
|         behaviours.Remove(behaviour); |         behaviours.Remove(behaviour); | ||||||
|         OnBehaviourRemoved?.Invoke(this, behaviour); |         OnBehaviourRemoved?.Invoke(this, new(behaviour)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public bool Assign(IGameObject gameObject) |     protected virtual void OnAssign(IUniverseObject universeObject) { } | ||||||
|  |     public bool Assign(IUniverseObject universeObject) | ||||||
|     { |     { | ||||||
|         if (GameObject is not null && GameObject.Initialized) |         if (UniverseObject is not null && UniverseObject.IsInitialized) | ||||||
|             return false; |             return false; | ||||||
|  |  | ||||||
|         _gameObject = gameObject; |         _universeObject = universeObject; | ||||||
|         OnGameObjectAssigned?.Invoke(this); |         OnAssign(universeObject); | ||||||
|  |         OnUniverseObjectAssigned?.Invoke(this); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public bool Unassign() |     protected override void InitializeInternal() | ||||||
|     { |     { | ||||||
|         if (GameObject is not null && GameObject.Initialized) |         Debug.Assert.AssertUniverseObjectAssigned(this); | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         _gameObject = null!; |         foreach (IBehaviour behaviour in behaviours) | ||||||
|         OnUnassigned?.Invoke(this); |             behaviour.Initialize(); | ||||||
|         return true; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void Update() |     protected override void FinalizeInternal() | ||||||
|     { |     { | ||||||
|         if (!GameObject.StateEnable.Enabled) |         foreach (IBehaviour behaviour in behaviours) | ||||||
|             return; |             behaviour.Finalize(); | ||||||
|  |  | ||||||
|         OnPreUpdate?.Invoke(this); |  | ||||||
|         OnUpdate?.Invoke(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void UpdatePreDraw() |  | ||||||
|     { |  | ||||||
|         if (!GameObject.StateEnable.Enabled) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         OnPreDraw?.Invoke(this); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public BehaviourController() { } |     public BehaviourController() { } | ||||||
|     public BehaviourController(IGameObject gameObject) => Assign(gameObject); |     public BehaviourController(IUniverseObject universeObject) => Assign(universeObject); | ||||||
|  |  | ||||||
|     private void InsertBehaviourByPriority<T>(T behaviour) where T : class, IBehaviour |     private void InsertBehaviourByPriority<T>(T behaviour) where T : class, IBehaviour | ||||||
|     { |     { | ||||||
| @@ -165,12 +141,9 @@ public class BehaviourController : IBehaviourController | |||||||
|             behaviours.Add(behaviour); |             behaviours.Add(behaviour); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void OnPriorityChange(IBehaviour behaviour) |     private void OnPriorityChange(IBehaviour sender, IBehaviour.PriorityChangedArguments args) | ||||||
|     { |     { | ||||||
|         behaviours.Remove(behaviour); |         behaviours.Remove(sender); | ||||||
|         InsertBehaviourByPriority(behaviour); |         InsertBehaviourByPriority(sender); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public IEnumerator<IBehaviour> GetEnumerator() => behaviours.GetEnumerator(); |  | ||||||
|     IEnumerator IEnumerable.GetEnumerator() => behaviours.GetEnumerator(); |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										56
									
								
								Engine.Core/BehaviourInternal.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								Engine.Core/BehaviourInternal.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | namespace Engine.Core.Internal; | ||||||
|  |  | ||||||
|  | [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] | ||||||
|  | /// <summary> | ||||||
|  | /// This behaviour can be used for core managers like <see cref="UpdateManager"/> etc. which need to be able to work even | ||||||
|  | /// in a very bare minimum setup without the presence of <see cref="UniverseEntranceManager"/> to set themselves up on universe entrance. | ||||||
|  | /// I recommend not using this unless you know what you are doing but it might come in handy for some use cases. | ||||||
|  | /// </summary> | ||||||
|  | public abstract class BehaviourIndependent : Behaviour | ||||||
|  | { | ||||||
|  |     private readonly Event<IUniverseObject, IUniverseObject.EnteredUniverseArguments>.EventHandler delegateEnteredUniverse = null!; | ||||||
|  |     private readonly Event<IUniverseObject, IUniverseObject.ExitedUniverseArguments>.EventHandler delegateExitedUniverse = null!; | ||||||
|  |  | ||||||
|  |     public BehaviourIndependent() | ||||||
|  |     { | ||||||
|  |         OnInitialized.AddListener(OnInitialize); | ||||||
|  |         OnFinalized.AddListener(OnFinalize); | ||||||
|  |         OnUnassigned.AddListener(OnUnassign); | ||||||
|  |  | ||||||
|  |         delegateEnteredUniverse = EnteredUniverse; | ||||||
|  |         delegateExitedUniverse = ExitedUniverse; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected virtual void OnUnassign() { } | ||||||
|  |     protected void OnUnassign(IAssignable assignable) => OnUnassign(); | ||||||
|  |  | ||||||
|  |     protected virtual void OnInitialize() { } | ||||||
|  |     protected void OnInitialize(IInitializable _) | ||||||
|  |     { | ||||||
|  |         BehaviourController.UniverseObject.OnEnteredUniverse.AddListener(delegateEnteredUniverse); | ||||||
|  |         BehaviourController.UniverseObject.OnExitedUniverse.AddListener(delegateExitedUniverse); | ||||||
|  |  | ||||||
|  |         OnInitialize(); | ||||||
|  |  | ||||||
|  |         if (UniverseObject.IsInUniverse) | ||||||
|  |             EnteredUniverse(UniverseObject, new(Universe)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected virtual void OnFinalize() { } | ||||||
|  |     protected void OnFinalize(IInitializable _) | ||||||
|  |     { | ||||||
|  |         BehaviourController.UniverseObject.OnEnteredUniverse.RemoveListener(delegateEnteredUniverse); | ||||||
|  |         BehaviourController.UniverseObject.OnExitedUniverse.RemoveListener(delegateExitedUniverse); | ||||||
|  |  | ||||||
|  |         OnFinalize(); | ||||||
|  |  | ||||||
|  |         if (UniverseObject.IsInUniverse) | ||||||
|  |             ExitedUniverse(UniverseObject, new(Universe)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected virtual void OnEnteredUniverse(IUniverse universe) { } | ||||||
|  |     protected void EnteredUniverse(IUniverseObject sender, IUniverseObject.EnteredUniverseArguments args) => OnEnteredUniverse(args.Universe); | ||||||
|  |  | ||||||
|  |     protected virtual void OnExitedUniverse(IUniverse universe) { } | ||||||
|  |     protected void ExitedUniverse(IUniverseObject sender, IUniverseObject.ExitedUniverseArguments args) => OnExitedUniverse(args.Universe); | ||||||
|  | } | ||||||
| @@ -1,87 +0,0 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public abstract class BehaviourOverride : Behaviour |  | ||||||
| { |  | ||||||
|     private bool isInitializedThisFrame = false; |  | ||||||
|  |  | ||||||
|     protected IGameObject GameObject => BehaviourController.GameObject; |  | ||||||
|     protected ITransform Transform => BehaviourController.GameObject.Transform; |  | ||||||
|  |  | ||||||
|     public BehaviourOverride() |  | ||||||
|     { |  | ||||||
|         OnInitialized += OnInitialize; |  | ||||||
|         OnFinalized += OnFinalize; |  | ||||||
|         OnUnassigned += OnUnassign; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnUnassign() { } |  | ||||||
|     private void OnUnassign(IAssignable assignable) => OnUnassign(); |  | ||||||
|  |  | ||||||
|     protected virtual void OnInitialize() { } |  | ||||||
|     private void OnInitialize(IInitialize _) |  | ||||||
|     { |  | ||||||
|         isInitializedThisFrame = true; |  | ||||||
|  |  | ||||||
|         BehaviourController.OnPreUpdate += PreUpdate; |  | ||||||
|         BehaviourController.OnPreDraw += PreDraw; |  | ||||||
|         BehaviourController.OnUpdate += Update; |  | ||||||
|         OnInitialize(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnFinalize() { } |  | ||||||
|     private void OnFinalize(IInitialize _) |  | ||||||
|     { |  | ||||||
|         BehaviourController.OnPreUpdate -= PreUpdate; |  | ||||||
|         BehaviourController.OnPreDraw -= PreDraw; |  | ||||||
|         BehaviourController.OnUpdate -= Update; |  | ||||||
|         OnFinalize(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnPreUpdatePreActiveCheck() { } |  | ||||||
|     protected virtual void OnPreUpdate() { } |  | ||||||
|     private void PreUpdate(IBehaviourController _) |  | ||||||
|     { |  | ||||||
|         OnPreUpdatePreActiveCheck(); |  | ||||||
|  |  | ||||||
|         if (!IsActive) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         if (isInitializedThisFrame) |  | ||||||
|             FirstActiveFrame(); |  | ||||||
|  |  | ||||||
|         OnPreUpdate(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnFirstActiveFrame() { } |  | ||||||
|     private void FirstActiveFrame() |  | ||||||
|     { |  | ||||||
|         OnFirstActiveFrame(); |  | ||||||
|         isInitializedThisFrame = false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnUpdatePreActiveCheck() { } |  | ||||||
|     protected virtual void OnUpdate() { } |  | ||||||
|     private void Update(IBehaviourController _) |  | ||||||
|     { |  | ||||||
|         OnUpdatePreActiveCheck(); |  | ||||||
|  |  | ||||||
|         if (!IsActive) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         OnUpdate(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected virtual void OnPreDrawPreActiveCheck() { } |  | ||||||
|     protected virtual void OnPreDraw() { } |  | ||||||
|     private void PreDraw(IBehaviourController _) |  | ||||||
|     { |  | ||||||
|         OnPreDrawPreActiveCheck(); |  | ||||||
|  |  | ||||||
|         if (!StateEnable.Enabled) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         OnPreDraw(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										16
									
								
								Engine.Core/Collectors/ActiveBehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								Engine.Core/Collectors/ActiveBehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class ActiveBehaviourCollector<T> : ActiveBehaviourCollectorBase<T> where T : class, IBehaviour | ||||||
|  | { | ||||||
|  |     protected readonly FastList<T> activeBehaviours = new(32); | ||||||
|  |     public override T this[Index index] => activeBehaviours[index]; | ||||||
|  |     public override int Count => activeBehaviours.Count; | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollector() { } | ||||||
|  |     public ActiveBehaviourCollector(IUniverse universe) : base(universe) { } | ||||||
|  |  | ||||||
|  |     protected override void AddBehaviour(T behaviour) => activeBehaviours.Add(behaviour); | ||||||
|  |     protected override bool RemoveBehaviour(T tBehaviour) => activeBehaviours.Remove(tBehaviour); | ||||||
|  | } | ||||||
							
								
								
									
										147
									
								
								Engine.Core/Collectors/ActiveBehaviourCollectorBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								Engine.Core/Collectors/ActiveBehaviourCollectorBase.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,147 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public abstract class ActiveBehaviourCollectorBase<T> : IBehaviourCollector<T> where T : class, IBehaviour | ||||||
|  | { | ||||||
|  |     protected readonly Dictionary<IActive, T> monitoringActiveToBehaviour = new(32); | ||||||
|  |     protected readonly FastList<T> monitoringBehaviours = new(32); | ||||||
|  |  | ||||||
|  |     private readonly Event<IBehaviourController, IBehaviourController.BehaviourAddedArguments>.EventHandler delegateOnBehaviourAdded = null!; | ||||||
|  |     private readonly Event<IBehaviourController, IBehaviourController.BehaviourRemovedArguments>.EventHandler delegateOnBehaviourRemoved = null!; | ||||||
|  |     private readonly Event<IActive, IActive.ActiveChangedArguments>.EventHandler delegateOnBehaviourStateChanged = null!; | ||||||
|  |     private readonly Event<IUniverse, IUniverse.UniverseObjectRegisteredArguments>.EventHandler delegateOnUniverseObjectRegistered = null!; | ||||||
|  |     private readonly Event<IUniverse, IUniverse.UniverseObjectUnRegisteredArguments>.EventHandler delegateOnUniverseObjectUnregistered = null!; | ||||||
|  |  | ||||||
|  |     public Event<IBehaviourCollector<T>, IBehaviourCollector<T>.BehaviourCollectedArguments> OnCollected { get; } = new(); | ||||||
|  |     public Event<IBehaviourCollector<T>, IBehaviourCollector<T>.BehaviourRemovedArguments> OnRemoved { get; } = new(); | ||||||
|  |     public Event<IHasUniverse> OnUniverseAssigned { get; } = new(); | ||||||
|  |     public Event<IAssignable>? OnUnassigned { get; } = new(); | ||||||
|  |  | ||||||
|  |     public abstract int Count { get; } | ||||||
|  |  | ||||||
|  |     public abstract T this[Index index] { get; } | ||||||
|  |     public IUniverse Universe { get; private set; } = null!; | ||||||
|  |  | ||||||
|  |     public bool Assign(IUniverse universe) | ||||||
|  |     { | ||||||
|  |         if (Universe is not null) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in universe.UniverseObjects) | ||||||
|  |             OnUniverseObjectRegistered(universe, new(universeObject)); | ||||||
|  |  | ||||||
|  |         universe.OnUniverseObjectRegistered.AddListener(delegateOnUniverseObjectRegistered); | ||||||
|  |         universe.OnUniverseObjectUnRegistered.AddListener(delegateOnUniverseObjectUnregistered); | ||||||
|  |  | ||||||
|  |         Universe = universe; | ||||||
|  |         OnUniverseAssigned?.Invoke(this); | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool Unassign() | ||||||
|  |     { | ||||||
|  |         if (Universe is null) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in Universe.UniverseObjects) | ||||||
|  |             OnUniverseObjectUnregistered(Universe, new(universeObject)); | ||||||
|  |  | ||||||
|  |         Universe.OnUniverseObjectRegistered.RemoveListener(delegateOnUniverseObjectRegistered); | ||||||
|  |         Universe.OnUniverseObjectUnRegistered.RemoveListener(delegateOnUniverseObjectUnregistered); | ||||||
|  |  | ||||||
|  |         Universe = null!; | ||||||
|  |         OnUnassigned?.Invoke(this); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected abstract void AddBehaviour(T behaviour); | ||||||
|  |     protected virtual void OnBehaviourAdd(IBehaviour behaviour) { } | ||||||
|  |     private void OnBehaviourAdded(IBehaviourController controller, IBehaviourController.BehaviourAddedArguments args) | ||||||
|  |     { | ||||||
|  |         if (args.BehaviourAdded is not T tBehaviour) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         monitoringBehaviours.Add(tBehaviour); | ||||||
|  |         monitoringActiveToBehaviour.Add(tBehaviour, tBehaviour); | ||||||
|  |         tBehaviour.OnActiveChanged.AddListener(delegateOnBehaviourStateChanged); | ||||||
|  |         OnBehaviourStateChanged(tBehaviour, new(!tBehaviour.IsActive)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected abstract bool RemoveBehaviour(T behaviour); | ||||||
|  |     protected virtual void OnBehaviourRemove(IBehaviour behaviour) { } | ||||||
|  |     private void OnBehaviourRemoved(IBehaviourController controller, IBehaviourController.BehaviourRemovedArguments args) | ||||||
|  |     { | ||||||
|  |         if (args.BehaviourRemoved is not T tBehaviour) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         if (!monitoringBehaviours.Remove(tBehaviour) || !monitoringActiveToBehaviour.Remove(tBehaviour)) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         tBehaviour.OnActiveChanged.RemoveListener(delegateOnBehaviourStateChanged); | ||||||
|  |         if (!RemoveBehaviour(tBehaviour)) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         OnBehaviourRemove(tBehaviour); | ||||||
|  |         OnRemoved?.Invoke(this, new(tBehaviour)); | ||||||
|  |     } | ||||||
|  |     private void OnBehaviourStateChanged(IActive sender, IActive.ActiveChangedArguments args) | ||||||
|  |     { | ||||||
|  |         T behaviour = monitoringActiveToBehaviour[sender]; | ||||||
|  |         if (sender.IsActive) | ||||||
|  |         { | ||||||
|  |             AddBehaviour(behaviour); | ||||||
|  |             OnBehaviourAdd(behaviour); | ||||||
|  |             OnCollected?.Invoke(this, new(behaviour)); | ||||||
|  |         } | ||||||
|  |         else if (RemoveBehaviour(behaviour)) | ||||||
|  |         { | ||||||
|  |             OnBehaviourRemove(behaviour); | ||||||
|  |             OnRemoved?.Invoke(this, new(behaviour)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void OnUniverseObjectRegistered(IUniverse manager, IUniverse.UniverseObjectRegisteredArguments args) | ||||||
|  |     { | ||||||
|  |         IUniverseObject universeObject = args.UniverseObjectRegistered; | ||||||
|  |  | ||||||
|  |         universeObject.BehaviourController.OnBehaviourAdded.AddListener(delegateOnBehaviourAdded); | ||||||
|  |         universeObject.BehaviourController.OnBehaviourRemoved.AddListener(delegateOnBehaviourRemoved); | ||||||
|  |  | ||||||
|  |         for (int i = 0; i < universeObject.BehaviourController.Count; i++) | ||||||
|  |             OnBehaviourAdded(universeObject.BehaviourController, new(universeObject.BehaviourController[i])); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void OnUniverseObjectUnregistered(IUniverse manager, IUniverse.UniverseObjectUnRegisteredArguments args) | ||||||
|  |     { | ||||||
|  |         IUniverseObject universeObject = args.UniverseObjectUnregistered; | ||||||
|  |  | ||||||
|  |         universeObject.BehaviourController.OnBehaviourAdded.RemoveListener(delegateOnBehaviourAdded); | ||||||
|  |         universeObject.BehaviourController.OnBehaviourRemoved.RemoveListener(delegateOnBehaviourRemoved); | ||||||
|  |  | ||||||
|  |         for (int i = 0; i < universeObject.BehaviourController.Count; i++) | ||||||
|  |             OnBehaviourRemoved(universeObject.BehaviourController, new(universeObject.BehaviourController[i])); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorBase() | ||||||
|  |     { | ||||||
|  |         delegateOnBehaviourAdded = OnBehaviourAdded; | ||||||
|  |         delegateOnBehaviourRemoved = OnBehaviourRemoved; | ||||||
|  |         delegateOnBehaviourStateChanged = OnBehaviourStateChanged; | ||||||
|  |         delegateOnUniverseObjectRegistered = OnUniverseObjectRegistered; | ||||||
|  |         delegateOnUniverseObjectUnregistered = OnUniverseObjectUnregistered; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorBase(IUniverse universe) | ||||||
|  |     { | ||||||
|  |         delegateOnBehaviourAdded = OnBehaviourAdded; | ||||||
|  |         delegateOnBehaviourRemoved = OnBehaviourRemoved; | ||||||
|  |         delegateOnBehaviourStateChanged = OnBehaviourStateChanged; | ||||||
|  |         delegateOnUniverseObjectRegistered = OnUniverseObjectRegistered; | ||||||
|  |         delegateOnUniverseObjectUnregistered = OnUniverseObjectUnregistered; | ||||||
|  |  | ||||||
|  |         Assign(universe); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										104
									
								
								Engine.Core/Collectors/ActiveBehaviourCollectorOrdered.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								Engine.Core/Collectors/ActiveBehaviourCollectorOrdered.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class ActiveBehaviourCollectorOrdered<TIndex, TItem> : ActiveBehaviourCollector<TItem> where TItem : class, IBehaviour where TIndex : IComparable | ||||||
|  | { | ||||||
|  |     private readonly Event<IBehaviour, IBehaviour.PriorityChangedArguments>.EventHandler delegateOnPriorityChanged = null!; | ||||||
|  |  | ||||||
|  |     private readonly SortedDictionary<TIndex, FastList<TItem>> behaviours = null!; | ||||||
|  |  | ||||||
|  |     private readonly Func<TItem, TIndex> getIndexFunc = null!; | ||||||
|  |     private readonly IComparer<TIndex> sortBy = null!; | ||||||
|  |  | ||||||
|  |     private int count = 0; | ||||||
|  |     public override int Count => count; | ||||||
|  |  | ||||||
|  |     public override TItem this[Index index] | ||||||
|  |     { | ||||||
|  |         get | ||||||
|  |         { | ||||||
|  |             int actualIndex = index.IsFromEnd | ||||||
|  |                 ? count - index.Value | ||||||
|  |                 : index.Value; | ||||||
|  |  | ||||||
|  |             if (actualIndex < 0 || actualIndex >= count) | ||||||
|  |                 throw new IndexOutOfRangeException(); | ||||||
|  |  | ||||||
|  |             int leftIndex = actualIndex; | ||||||
|  |             foreach ((TIndex i, FastList<TItem> list) in behaviours) | ||||||
|  |             { | ||||||
|  |                 if (leftIndex < list.Count) | ||||||
|  |                     return list[leftIndex]; | ||||||
|  |                 leftIndex -= list.Count; | ||||||
|  |             } | ||||||
|  |             throw new IndexOutOfRangeException(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override bool RemoveBehaviour(TItem tBehaviour) | ||||||
|  |     { | ||||||
|  |         TIndex index = getIndexFunc(tBehaviour); | ||||||
|  |         if (!behaviours.TryGetValue(index, out FastList<TItem>? list)) | ||||||
|  |             throw new Exceptions.NotFoundException($"Index of '{index}' is not found in the collector"); | ||||||
|  |  | ||||||
|  |         if (!list.Remove(tBehaviour)) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         count--; | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override void AddBehaviour(TItem behaviour) | ||||||
|  |     { | ||||||
|  |         TIndex key = getIndexFunc(behaviour); | ||||||
|  |         if (!behaviours.TryGetValue(key, out FastList<TItem>? list)) | ||||||
|  |             behaviours[key] = list = []; | ||||||
|  |  | ||||||
|  |         count++; | ||||||
|  |         list.Add(behaviour); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override void OnBehaviourAdd(IBehaviour behaviour) => behaviour.OnPriorityChanged.AddListener(delegateOnPriorityChanged); | ||||||
|  |     protected override void OnBehaviourRemove(IBehaviour behaviour) => behaviour.OnPriorityChanged.RemoveListener(delegateOnPriorityChanged); | ||||||
|  |  | ||||||
|  |     private void OnPriorityChanged(IBehaviour sender, IBehaviour.PriorityChangedArguments args) | ||||||
|  |     { | ||||||
|  |         TItem behaviour = (TItem)sender; | ||||||
|  |         RemoveBehaviour(behaviour); | ||||||
|  |         AddBehaviour(behaviour); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorOrdered(Func<TItem, TIndex> getIndexFunc, Comparison<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = Comparer<TIndex>.Create(sortBy); | ||||||
|  |         behaviours = new(this.sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorOrdered(IUniverse universe, Func<TItem, TIndex> getIndexFunc, Comparison<TIndex> sortBy) : base(universe) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = Comparer<TIndex>.Create(sortBy); | ||||||
|  |         behaviours = new(this.sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorOrdered(Func<TItem, TIndex> getIndexFunc, IComparer<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.sortBy = sortBy; | ||||||
|  |         behaviours = new(sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ActiveBehaviourCollectorOrdered(IUniverse universe, Func<TItem, TIndex> getIndexFunc, IComparer<TIndex> sortBy) : base(universe) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = sortBy; | ||||||
|  |         behaviours = new(sortBy); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								Engine.Core/Collectors/BehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								Engine.Core/Collectors/BehaviourCollector.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class BehaviourCollector<T> : BehaviourCollectorBase<T> where T : class | ||||||
|  | { | ||||||
|  |     protected readonly FastList<T> behaviours = new(32); | ||||||
|  |  | ||||||
|  |     public override T this[Index index] => behaviours[index]; | ||||||
|  |     public override int Count => behaviours.Count; | ||||||
|  |  | ||||||
|  |     protected override void AddBehaviour(T behaviour) => behaviours.Add(behaviour); | ||||||
|  |     protected override bool RemoveBehaviour(T tBehaviour) => behaviours.Remove(tBehaviour); | ||||||
|  |  | ||||||
|  |     public BehaviourCollector() { } | ||||||
|  |     public BehaviourCollector(IUniverse universe) : base(universe) { } | ||||||
|  | } | ||||||
							
								
								
									
										124
									
								
								Engine.Core/Collectors/BehaviourCollectorBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								Engine.Core/Collectors/BehaviourCollectorBase.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public abstract class BehaviourCollectorBase<T> : IBehaviourCollector<T> where T : class | ||||||
|  | { | ||||||
|  |     private readonly Event<IBehaviourController, IBehaviourController.BehaviourAddedArguments>.EventHandler delegateOnBehaviourAdded = null!; | ||||||
|  |     private readonly Event<IBehaviourController, IBehaviourController.BehaviourRemovedArguments>.EventHandler delegateOnBehaviourRemoved = null!; | ||||||
|  |     private readonly Event<IUniverse, IUniverse.UniverseObjectRegisteredArguments>.EventHandler delegateOnUniverseObjectRegistered = null!; | ||||||
|  |     private readonly Event<IUniverse, IUniverse.UniverseObjectUnRegisteredArguments>.EventHandler delegateOnUniverseObjectUnregistered = null!; | ||||||
|  |  | ||||||
|  |     public Event<IBehaviourCollector<T>, IBehaviourCollector<T>.BehaviourCollectedArguments> OnCollected { get; } = new(); | ||||||
|  |     public Event<IBehaviourCollector<T>, IBehaviourCollector<T>.BehaviourRemovedArguments> OnRemoved { get; } = new(); | ||||||
|  |     public Event<IHasUniverse> OnUniverseAssigned { get; } = new(); | ||||||
|  |     public Event<IAssignable>? OnUnassigned { get; } = new(); | ||||||
|  |  | ||||||
|  |     public IUniverse Universe { get; private set; } = null!; | ||||||
|  |  | ||||||
|  |     public abstract int Count { get; } | ||||||
|  |  | ||||||
|  |     public abstract T this[Index index] { get; } | ||||||
|  |  | ||||||
|  |     public bool Assign(IUniverse universe) | ||||||
|  |     { | ||||||
|  |         if (Universe is not null) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in universe.UniverseObjects) | ||||||
|  |             OnUniverseObjectRegistered(universe, new(universeObject)); | ||||||
|  |  | ||||||
|  |         universe.OnUniverseObjectRegistered.AddListener(delegateOnUniverseObjectRegistered); | ||||||
|  |         universe.OnPreUniverseObjectUnRegistered.AddListener(delegateOnUniverseObjectUnregistered); | ||||||
|  |  | ||||||
|  |         Universe = universe; | ||||||
|  |         OnAssign(universe); | ||||||
|  |         OnUniverseAssigned?.Invoke(this); | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool Unassign() | ||||||
|  |     { | ||||||
|  |         if (Universe is null) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in Universe.UniverseObjects) | ||||||
|  |             OnUniverseObjectUnregistered(Universe, new(universeObject)); | ||||||
|  |  | ||||||
|  |         Universe.OnUniverseObjectRegistered.RemoveListener(delegateOnUniverseObjectRegistered); | ||||||
|  |         Universe.OnPreUniverseObjectUnRegistered.RemoveListener(delegateOnUniverseObjectUnregistered); | ||||||
|  |  | ||||||
|  |         Universe = null!; | ||||||
|  |         OnUnassigned?.Invoke(this); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected virtual void OnAssign(IUniverse universe) { } | ||||||
|  |  | ||||||
|  |     protected abstract void AddBehaviour(T behaviour); | ||||||
|  |     protected virtual void OnBehaviourAdd(IBehaviour behaviour) { } | ||||||
|  |     private void OnBehaviourAdded(IBehaviourController controller, IBehaviourController.BehaviourAddedArguments args) | ||||||
|  |     { | ||||||
|  |         if (args.BehaviourAdded is not T tBehaviour) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         AddBehaviour(tBehaviour); | ||||||
|  |         OnBehaviourAdd(args.BehaviourAdded); | ||||||
|  |         OnCollected?.Invoke(this, new(tBehaviour)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected abstract bool RemoveBehaviour(T tBehaviour); | ||||||
|  |     protected virtual void OnBehaviourRemove(IBehaviour behaviour) { } | ||||||
|  |     private void OnBehaviourRemoved(IBehaviourController controller, IBehaviourController.BehaviourRemovedArguments args) | ||||||
|  |     { | ||||||
|  |         if (args.BehaviourRemoved is not T tBehaviour) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         if (!RemoveBehaviour(tBehaviour)) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         OnBehaviourRemove(args.BehaviourRemoved); | ||||||
|  |         OnRemoved?.Invoke(this, new(tBehaviour)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void OnUniverseObjectRegistered(IUniverse manager, IUniverse.UniverseObjectRegisteredArguments args) | ||||||
|  |     { | ||||||
|  |         IUniverseObject universeObject = args.UniverseObjectRegistered; | ||||||
|  |  | ||||||
|  |         universeObject.BehaviourController.OnBehaviourAdded.AddListener(delegateOnBehaviourAdded); | ||||||
|  |         universeObject.BehaviourController.OnBehaviourRemoved.AddListener(delegateOnBehaviourRemoved); | ||||||
|  |  | ||||||
|  |         for (int i = 0; i < universeObject.BehaviourController.Count; i++) | ||||||
|  |             OnBehaviourAdded(universeObject.BehaviourController, new(universeObject.BehaviourController[i])); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void OnUniverseObjectUnregistered(IUniverse manager, IUniverse.UniverseObjectUnRegisteredArguments args) | ||||||
|  |     { | ||||||
|  |         IUniverseObject universeObject = args.UniverseObjectUnregistered; | ||||||
|  |  | ||||||
|  |         universeObject.BehaviourController.OnBehaviourAdded.RemoveListener(delegateOnBehaviourAdded); | ||||||
|  |         universeObject.BehaviourController.OnBehaviourRemoved.RemoveListener(delegateOnBehaviourRemoved); | ||||||
|  |  | ||||||
|  |         for (int i = 0; i < universeObject.BehaviourController.Count; i++) | ||||||
|  |             OnBehaviourRemoved(universeObject.BehaviourController, new(universeObject.BehaviourController[i])); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorBase() | ||||||
|  |     { | ||||||
|  |         delegateOnBehaviourAdded = OnBehaviourAdded; | ||||||
|  |         delegateOnBehaviourRemoved = OnBehaviourRemoved; | ||||||
|  |         delegateOnUniverseObjectRegistered = OnUniverseObjectRegistered; | ||||||
|  |         delegateOnUniverseObjectUnregistered = OnUniverseObjectUnregistered; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorBase(IUniverse universe) | ||||||
|  |     { | ||||||
|  |         delegateOnBehaviourAdded = OnBehaviourAdded; | ||||||
|  |         delegateOnBehaviourRemoved = OnBehaviourRemoved; | ||||||
|  |         delegateOnUniverseObjectRegistered = OnUniverseObjectRegistered; | ||||||
|  |         delegateOnUniverseObjectUnregistered = OnUniverseObjectUnregistered; | ||||||
|  |  | ||||||
|  |         Assign(universe); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										104
									
								
								Engine.Core/Collectors/BehaviourCollectorOrdered.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								Engine.Core/Collectors/BehaviourCollectorOrdered.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class BehaviourCollectorOrdered<TIndex, TItem> : BehaviourCollectorBase<TItem> where TItem : class where TIndex : IComparable | ||||||
|  | { | ||||||
|  |     private readonly Event<IBehaviour, IBehaviour.PriorityChangedArguments>.EventHandler delegateOnPriorityChanged = null!; | ||||||
|  |  | ||||||
|  |     private readonly SortedDictionary<TIndex, FastList<TItem>> behaviours = null!; | ||||||
|  |  | ||||||
|  |     private readonly Func<TItem, TIndex> getIndexFunc = null!; | ||||||
|  |     private readonly IComparer<TIndex> sortBy = null!; | ||||||
|  |  | ||||||
|  |     private int count = 0; | ||||||
|  |     public override int Count => count; | ||||||
|  |  | ||||||
|  |     public override TItem this[Index index] | ||||||
|  |     { | ||||||
|  |         get | ||||||
|  |         { | ||||||
|  |             int actualIndex = index.IsFromEnd | ||||||
|  |                 ? count - index.Value | ||||||
|  |                 : index.Value; | ||||||
|  |  | ||||||
|  |             if (actualIndex < 0 || actualIndex >= count) | ||||||
|  |                 throw new IndexOutOfRangeException(); | ||||||
|  |  | ||||||
|  |             int leftIndex = actualIndex; | ||||||
|  |             foreach ((TIndex i, FastList<TItem> list) in behaviours) | ||||||
|  |             { | ||||||
|  |                 if (leftIndex < list.Count) | ||||||
|  |                     return list[leftIndex]; | ||||||
|  |                 leftIndex -= list.Count; | ||||||
|  |             } | ||||||
|  |             throw new IndexOutOfRangeException(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override bool RemoveBehaviour(TItem tBehaviour) | ||||||
|  |     { | ||||||
|  |         TIndex index = getIndexFunc(tBehaviour); | ||||||
|  |         if (!behaviours.TryGetValue(index, out FastList<TItem>? list)) | ||||||
|  |             throw new Exceptions.NotFoundException($"Index of '{index}' is not found in the collector"); | ||||||
|  |  | ||||||
|  |         if (!list.Remove(tBehaviour)) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         count--; | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override void AddBehaviour(TItem behaviour) | ||||||
|  |     { | ||||||
|  |         TIndex key = getIndexFunc(behaviour); | ||||||
|  |         if (!behaviours.TryGetValue(key, out FastList<TItem>? list)) | ||||||
|  |             behaviours[key] = list = []; | ||||||
|  |  | ||||||
|  |         count++; | ||||||
|  |         list.Add(behaviour); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected override void OnBehaviourAdd(IBehaviour behaviour) => behaviour.OnPriorityChanged.AddListener(delegateOnPriorityChanged); | ||||||
|  |     protected override void OnBehaviourRemove(IBehaviour behaviour) => behaviour.OnPriorityChanged.RemoveListener(delegateOnPriorityChanged); | ||||||
|  |  | ||||||
|  |     private void OnPriorityChanged(IBehaviour sender, IBehaviour.PriorityChangedArguments args) | ||||||
|  |     { | ||||||
|  |         TItem behaviour = (TItem)sender; | ||||||
|  |         RemoveBehaviour(behaviour); | ||||||
|  |         AddBehaviour(behaviour); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorOrdered(Func<TItem, TIndex> getIndexFunc, Comparison<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = Comparer<TIndex>.Create(sortBy); | ||||||
|  |         behaviours = new(this.sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorOrdered(IUniverse universe, Func<TItem, TIndex> getIndexFunc, Comparison<TIndex> sortBy) : base(universe) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = Comparer<TIndex>.Create(sortBy); | ||||||
|  |         behaviours = new(this.sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorOrdered(Func<TItem, TIndex> getIndexFunc, IComparer<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.sortBy = sortBy; | ||||||
|  |         behaviours = new(sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public BehaviourCollectorOrdered(IUniverse universe, Func<TItem, TIndex> getIndexFunc, IComparer<TIndex> sortBy) : base(universe) | ||||||
|  |     { | ||||||
|  |         delegateOnPriorityChanged = OnPriorityChanged; | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = sortBy; | ||||||
|  |         behaviours = new(sortBy); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										46
									
								
								Engine.Core/Debug/AssertHelpers.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								Engine.Core/Debug/AssertHelpers.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | using System.Runtime.CompilerServices; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public static class Assert | ||||||
|  | { | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertTrue(bool value, string errorMessage) | ||||||
|  |         => System.Diagnostics.Debug.Assert(value, errorMessage); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertFalse(bool value, string errorMessage) | ||||||
|  |         => System.Diagnostics.Debug.Assert(!value, errorMessage); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertNull(object? value, string errorMessage) | ||||||
|  |         => System.Diagnostics.Debug.Assert(value is null, errorMessage); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertNotNull(object? value, string errorMessage) | ||||||
|  |         => System.Diagnostics.Debug.Assert(value is not null, errorMessage); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertInitialized(IInitializable initializable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(initializable.IsInitialized, $"{initializable.GetType().Name} must be initialized"); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertBehaviourControllerAssigned(IHasBehaviourController assignable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(assignable.BehaviourController is not null, $"{assignable.GetType().Name} must be assigned an {nameof(IBehaviourController)}"); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertEntityAssigned(IHasEntity assignable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(assignable.Entity is not null, $"{assignable.GetType().Name} must be assigned an {nameof(IEntity)}"); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertUniverseAssigned(IHasUniverse assignable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(assignable.Universe is not null, $"{assignable.GetType().Name} must be assigned an {nameof(IUniverse)}"); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertUniverseObjectAssigned(IHasUniverseObject assignable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(assignable.UniverseObject is not null, $"{assignable.GetType().Name} must be assigned an {nameof(IUniverseObject)}"); | ||||||
|  |  | ||||||
|  |     [MethodImpl(MethodImplOptions.AggressiveInlining)] | ||||||
|  |     public static void AssertStateEnableAssigned(IHasStateEnable assignable) | ||||||
|  |         => System.Diagnostics.Debug.Assert(assignable.StateEnable is not null, $"{assignable.GetType().Name} must be assigned an {nameof(IStateEnable)}"); | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								Engine.Core/Debug/ConsoleLogger.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								Engine.Core/Debug/ConsoleLogger.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public class ConsoleLogger : LoggerBase | ||||||
|  | { | ||||||
|  |     protected override void Write(string message) => Console.WriteLine(message); | ||||||
|  | } | ||||||
							
								
								
									
										32
									
								
								Engine.Core/Debug/FileLogger.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								Engine.Core/Debug/FileLogger.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | using System; | ||||||
|  | using System.IO; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public class FileLogger : LoggerBase | ||||||
|  | { | ||||||
|  |     public readonly string FilePath; | ||||||
|  |  | ||||||
|  |     protected override void Write(string message) | ||||||
|  |     { | ||||||
|  |         File.AppendAllTextAsync(FilePath, $"{message}{Environment.NewLine}"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public FileLogger(string filePath) | ||||||
|  |     { | ||||||
|  |         if (!filePath.EndsWith(".log")) | ||||||
|  |             filePath += ".log"; | ||||||
|  |  | ||||||
|  |         FilePath = filePath; | ||||||
|  |  | ||||||
|  |         bool isRelativePath = Path.GetFullPath(filePath).CompareTo(filePath) != 0; | ||||||
|  |  | ||||||
|  |         if (isRelativePath) | ||||||
|  |             FilePath = Path.GetFullPath(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, filePath)); | ||||||
|  |  | ||||||
|  |         if (Path.GetDirectoryName(FilePath) is string directoryPath) | ||||||
|  |             Directory.CreateDirectory(directoryPath); | ||||||
|  |  | ||||||
|  |         File.Open(FilePath, FileMode.Create).Close(); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								Engine.Core/Debug/ILogger.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								Engine.Core/Debug/ILogger.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public interface ILogger | ||||||
|  | { | ||||||
|  |     static ILogger Shared { get; set; } = new ConsoleLogger(); | ||||||
|  |  | ||||||
|  |     Level FilterLevel { get; set; } | ||||||
|  |  | ||||||
|  |     void Log(string message, Level level = Level.Info, bool force = false); | ||||||
|  |  | ||||||
|  |     enum Level | ||||||
|  |     { | ||||||
|  |         Trace, | ||||||
|  |         Info, | ||||||
|  |         Warning, | ||||||
|  |         Error, | ||||||
|  |     }; | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								Engine.Core/Debug/LoggerBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								Engine.Core/Debug/LoggerBase.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public abstract class LoggerBase : ILogger | ||||||
|  | { | ||||||
|  |     public ILogger.Level FilterLevel { get; set; } = ILogger.Level.Trace; | ||||||
|  |  | ||||||
|  |     public void Log(string message, ILogger.Level level = ILogger.Level.Info, bool force = false) | ||||||
|  |     { | ||||||
|  |         if (!force && level < FilterLevel) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         string timestamp = DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss tt"); | ||||||
|  |  | ||||||
|  |         Write($"[{timestamp}] [{level}] \t{message}"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected abstract void Write(string message); | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								Engine.Core/Debug/LoggerContainer.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Engine.Core/Debug/LoggerContainer.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public class LoggerContainer : Behaviour, ILogger | ||||||
|  | { | ||||||
|  |     public ILogger Logger { get; set; } = ILogger.Shared; | ||||||
|  |  | ||||||
|  |     public ILogger.Level FilterLevel { get => Logger.FilterLevel; set => Logger.FilterLevel = value; } | ||||||
|  |     public void Log(string message, ILogger.Level level = ILogger.Level.Info, bool force = false) => Logger.Log(message, level, force); | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								Engine.Core/Debug/LoggerExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								Engine.Core/Debug/LoggerExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | using System; | ||||||
|  | using System.Diagnostics; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public static class LoggerExtensions | ||||||
|  | { | ||||||
|  |     public static void Log<T>(this ILogger logger, T caller, string message, ILogger.Level level = ILogger.Level.Info, bool force = false) | ||||||
|  |     { | ||||||
|  |         string body = $"{caller?.GetType().Name ?? typeof(T).Name}: {message}"; | ||||||
|  |         logger.Log(body, level, force); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void LogWarning<T>(this ILogger logger, T caller, string message, bool force = false) => Log(logger, caller, message, ILogger.Level.Info, force); | ||||||
|  |  | ||||||
|  |     public static void LogError<T>(this ILogger logger, T caller, string message, bool force = false) | ||||||
|  |     { | ||||||
|  |         Log(logger, caller, message, ILogger.Level.Error, force); | ||||||
|  |         LogTrace(logger, caller, new StackTrace(), force); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void LogException<T>(this ILogger logger, T caller, Exception exception, bool force = false) | ||||||
|  |     { | ||||||
|  |         Log(logger, caller, $"Exception of type {exception.GetType().Name} occured", ILogger.Level.Error, force); | ||||||
|  |         Log(logger, caller, $"Message: {exception.Message}", ILogger.Level.Error, force); | ||||||
|  |         Log(logger, caller, $"InnerException: {exception.InnerException}", ILogger.Level.Error, force); | ||||||
|  |  | ||||||
|  |         // Not using LogTrace because exception.StackTrace is a type of string | ||||||
|  |         Log(logger, caller, $"{nameof(StackTrace)}:{Environment.NewLine}{exception.StackTrace}", ILogger.Level.Trace); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void LogTrace<T>(this ILogger logger, T caller, StackTrace? stackTrace = null, bool force = false) | ||||||
|  |     { | ||||||
|  |         Log(logger, caller, $"{nameof(StackTrace)}:{Environment.NewLine}{stackTrace ?? new()}", ILogger.Level.Trace, force); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								Engine.Core/Debug/LoggerWrapper.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								Engine.Core/Debug/LoggerWrapper.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public class LoggerWrapper(ILogger firstLogger, ILogger secondLogger) : ILogger | ||||||
|  | { | ||||||
|  |     private readonly ILogger firstLogger = firstLogger; | ||||||
|  |     private readonly ILogger secondLogger = secondLogger; | ||||||
|  |  | ||||||
|  |     public ILogger.Level FilterLevel | ||||||
|  |     { | ||||||
|  |         get => firstLogger.FilterLevel; | ||||||
|  |         set | ||||||
|  |         { | ||||||
|  |             firstLogger.FilterLevel = value; | ||||||
|  |             secondLogger.FilterLevel = value; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Log(string message, ILogger.Level level = ILogger.Level.Info, bool force = false) | ||||||
|  |     { | ||||||
|  |         firstLogger.Log(message, level, force); | ||||||
|  |         secondLogger.Log(message, level, force); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								Engine.Core/Debug/LoggerWrapperExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								Engine.Core/Debug/LoggerWrapperExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public static class LoggerWrapperExtensions | ||||||
|  | { | ||||||
|  |     public static ILogger WrapWith(this ILogger thisLogger, ILogger logger) => new LoggerWrapper(thisLogger, logger); | ||||||
|  | } | ||||||
							
								
								
									
										71
									
								
								Engine.Core/Debug/RotatingFileLogger.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								Engine.Core/Debug/RotatingFileLogger.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | |||||||
|  | using System; | ||||||
|  | using System.IO; | ||||||
|  | using System.Linq; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | public class RotatingFileLogger : ILogger | ||||||
|  | { | ||||||
|  |     public readonly FileLogger FileLogger = null!; | ||||||
|  |     public readonly string Directory = string.Empty; | ||||||
|  |     public readonly int RotateLength = 3; | ||||||
|  |  | ||||||
|  |     public RotatingFileLogger(string directory, string namePrefix, string nameSuffix = "", int rotateLength = 3) | ||||||
|  |     { | ||||||
|  |         RotateLength = rotateLength; | ||||||
|  |  | ||||||
|  |         string fileName = Path.Combine(directory, namePrefix); | ||||||
|  |         if (!string.IsNullOrWhiteSpace(nameSuffix)) | ||||||
|  |             fileName += $"_{nameSuffix}"; | ||||||
|  |  | ||||||
|  |         bool isRelativePath = Path.GetFullPath(fileName).CompareTo(fileName) != 0; | ||||||
|  |  | ||||||
|  |         if (isRelativePath) | ||||||
|  |             fileName = Path.GetFullPath(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, fileName)); | ||||||
|  |  | ||||||
|  |         if (File.Exists($"{fileName}.log")) | ||||||
|  |             RenameExistingLogs(fileName, RotateLength); | ||||||
|  |  | ||||||
|  |         FileLogger = new(fileName); | ||||||
|  |  | ||||||
|  |         Directory = Path.GetDirectoryName(fileName) ?? throw new("Unexpected error on getting directory of logger path"); | ||||||
|  |         RotateLastLogs(Directory, namePrefix, RotateLength); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void RenameExistingLogs(string filePath, int rotateLength) | ||||||
|  |     { | ||||||
|  |         for (int i = rotateLength - 1; i >= 0; i--) | ||||||
|  |         { | ||||||
|  |             string source = i == 0 | ||||||
|  |                 ? $"{filePath}.log" | ||||||
|  |                 : $"{filePath}_{i}.log"; | ||||||
|  |  | ||||||
|  |             string dest = $"{filePath}_{i + 1}.log"; | ||||||
|  |  | ||||||
|  |             if (!File.Exists(source)) | ||||||
|  |                 continue; | ||||||
|  |  | ||||||
|  |             if (File.Exists(dest)) | ||||||
|  |                 File.Delete(dest); | ||||||
|  |  | ||||||
|  |             File.Move(source, dest); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void RotateLastLogs(string directory, string prefix, int rotateLength) | ||||||
|  |     { | ||||||
|  |         IOrderedEnumerable<string> logs = System.IO.Directory.GetFiles(directory, $"{prefix}*.log") | ||||||
|  |             .OrderBy(File.GetCreationTime); | ||||||
|  |  | ||||||
|  |         foreach (string file in logs.Skip(rotateLength)) | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 ILogger.Shared.Log($"Removing log file located at \"{file}\" during rotation."); | ||||||
|  |                 File.Delete(file); | ||||||
|  |             } | ||||||
|  |             catch (Exception e) { ILogger.Shared.LogException($"Failed to rotate log file at \"{file}\"", e); } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ILogger.Level FilterLevel { get => FileLogger.FilterLevel; set => FileLogger.FilterLevel = value; } | ||||||
|  |     public void Log(string message, ILogger.Level level = ILogger.Level.Info, bool force = false) => FileLogger.Log(message, level, force); | ||||||
|  | } | ||||||
| @@ -1,8 +1,11 @@ | |||||||
| <Project Sdk="Microsoft.NET.Sdk"> | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <TargetFramework>net8.0</TargetFramework> |     <TargetFramework>net9.0</TargetFramework> | ||||||
|     <ImplicitUsings>false</ImplicitUsings> |     <ImplicitUsings>false</ImplicitUsings> | ||||||
|     <Nullable>enable</Nullable> |     <Nullable>enable</Nullable> | ||||||
|  |     <RootNamespace>Engine.Core</RootNamespace> | ||||||
|  |     <AssemblyName>Engine.Core</AssemblyName> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								Engine.Core/Engine.Core.puml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								Engine.Core/Engine.Core.puml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | @startuml "Core Engine Relations" | ||||||
|  |  | ||||||
|  | top to bottom direction | ||||||
|  | skinparam linetype ortho | ||||||
|  | skinparam nodesep 100 | ||||||
|  |  | ||||||
|  | title Core Engine Relations | ||||||
|  |  | ||||||
|  | interface Engine.Core.IEntity extends Engine.Core.IInitializable {} | ||||||
|  | interface Engine.Core.IUniverseObject extends Engine.Core.IEntity, Engine.Core.INameable {} | ||||||
|  |  | ||||||
|  | interface Engine.Core.INameable {} | ||||||
|  |  | ||||||
|  | Engine.Core.IUniverseObject --> Engine.Core.IBehaviourController: has | ||||||
|  | Engine.Core.IBehaviourController "1" --> "0..*" Engine.Core.IBehaviour: has | ||||||
|  |  | ||||||
|  | interface Engine.Core.IBehaviourController {} | ||||||
|  | interface Engine.Core.IBehaviour {} | ||||||
|  | interface Engine.Core.IBehaviour2D extends Engine.Core.IBehaviour {} | ||||||
|  | interface Engine.Core.IBehaviour3D extends Engine.Core.IBehaviour {} | ||||||
|  |  | ||||||
|  | interface Engine.Core.IUniverse {} | ||||||
|  | Engine.Core.IUniverse "1" -r-> "0..*" Engine.Core.IUniverseObject: has | ||||||
|  |  | ||||||
|  | ' together { | ||||||
|  | '     interface Engine.Core.IAssignable {} | ||||||
|  | '     interface Engine.Core.IHasStateEnable extends Engine.Core.IAssignable {} | ||||||
|  | '     interface Engine.Core.IHasUniverse extends Engine.Core.IAssignable {} | ||||||
|  | '     interface Engine.Core.IHasUniverseObject extends Engine.Core.IAssignable {} | ||||||
|  | '     interface Engine.Core.IHasBehaviourController extends Engine.Core.IAssignable {} | ||||||
|  | '     ' Engine.Core.IHasStateEnable --> Engine.Core.IStateEnable: has | ||||||
|  | '     ' Engine.Core.IHasUniverse --> Engine.Core.IUniverse: has | ||||||
|  | '     ' Engine.Core.IHasUniverseObject --> Engine.Core.IUniverseObject: has | ||||||
|  | '     ' Engine.Core.IHasBehaviourController --> Engine.Core.IBehaviourController: has | ||||||
|  | ' } | ||||||
|  |  | ||||||
|  | together { | ||||||
|  |     interface Engine.Core.ITransform2D {} | ||||||
|  |     interface Engine.Core.ICamera2D {} | ||||||
|  |     interface Engine.Core.ICoroutineYield {} | ||||||
|  |     interface Engine.Core.IStateEnable {} | ||||||
|  |     interface Engine.Core.IInitializable {} | ||||||
|  |     interface Engine.Core.IBehaviourCollector {} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @enduml | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public readonly struct EngineTime(TimeSpan Total, TimeSpan Elapsed) |  | ||||||
| { |  | ||||||
|     public readonly TimeSpan Total = Total; |  | ||||||
|     public readonly TimeSpan Elapsed = Elapsed; |  | ||||||
|  |  | ||||||
|     public readonly float DeltaTimeFrame = (float)Elapsed.TotalMilliseconds; |  | ||||||
| } |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| using System; |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| public class AssignException : Exception |  | ||||||
| { |  | ||||||
|     public AssignException() : base("Assign operation has failed.") { } |  | ||||||
|     public AssignException(string? message) : base(message) { } |  | ||||||
|  |  | ||||||
|     // public static AssignException FromStateEnable(IStateEnable? stateEnable) |  | ||||||
|     //     => new AssignException($"{nameof(IGameObject.AssignStateEnable)} failed on type {stateEnable?.GetType().ToString() ?? "\"null\""}"); |  | ||||||
|     public static AssignException From<T, T2>(T to, T2? value) |  | ||||||
|         => new AssignException($"Assign operation has failed on T: {typeof(T).FullName}, value: {value?.GetType().ToString() ?? "\"null\""}"); |  | ||||||
|     // public static AssignException FromBehaviourController(IBehaviourController? behaviourController) |  | ||||||
|     //     => new AssignException($"{nameof(IGameObject.AssignBehaviourController)} failed on type {behaviourController?.GetType().ToString() ?? "\"null\""}"); |  | ||||||
| } |  | ||||||
| // throw new Exception($"{nameof(IGameObject.AssignTransform)} failed on type {transform?.GetType().ToString() ?? "null"} for type {typeof(T).FullName}"); |  | ||||||
| // throw new Exception($"{nameof(IGameObject.AssignBehaviourController)} failed on type {behaviourController?.GetType().ToString() ?? "null"} for type {typeof(T).FullName}"); |  | ||||||
| // throw new Exception($"{nameof(IGameObject.AssignStateEnable)} failed on type {stateEnable?.GetType().ToString() ?? "null"} for type {typeof(T).FullName}"); |  | ||||||
							
								
								
									
										9
									
								
								Engine.Core/Exceptions/AssignFailedException.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Engine.Core/Exceptions/AssignFailedException.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | public class AssignFailedException(string? message) : Exception(message) | ||||||
|  | { | ||||||
|  |     public static AssignFailedException From<T, T2>(T to, T2? value) | ||||||
|  |         => new($"Assign operation has failed on T: {to?.GetType().FullName ?? "\"null\""}, value: {value?.GetType().ToString() ?? "\"null\""}"); | ||||||
|  | } | ||||||
							
								
								
									
										3
									
								
								Engine.Core/Exceptions/BehaviourNotFoundException.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								Engine.Core/Exceptions/BehaviourNotFoundException.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | namespace Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | public class BehaviourNotFoundException(string? message) : NotFoundException(message); | ||||||
| @@ -1,21 +1,9 @@ | |||||||
| using System; | using System; | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Exceptions; | namespace Engine.Core.Exceptions; | ||||||
|  |  | ||||||
| public class NotAssignedException : Exception | public class NotAssignedException(string? message) : Exception(message) | ||||||
| { | { | ||||||
|     public NotAssignedException() : base("The object has not been assigned.") { } |     public static NotAssignedException From<T1, T2>(T1 to, T2? value) | ||||||
|     public NotAssignedException(string? message) : base(message) { } |         => new($"{value?.GetType().FullName ?? "\"null\""} has not been assigned to {to?.GetType().FullName ?? "\"null\""}"); | ||||||
|  |  | ||||||
|     public static NotAssignedException From<T1, T2>(T1 to, T2? value) where T1 : IAssignable |  | ||||||
|         => new NotAssignedException($"{typeof(T2).Name} has not been assigned to {typeof(T1).Name}"); |  | ||||||
|  |  | ||||||
|     public static void Check<T1, T2>(T1 to, T2? value) where T1 : IAssignable |  | ||||||
|     { |  | ||||||
|         if (value is not null) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         throw From(to, value); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								Engine.Core/Exceptions/NotFoundException.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Engine.Core/Exceptions/NotFoundException.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | public class NotFoundException(string? message) : Exception(message) | ||||||
|  | { | ||||||
|  |     public static NotAssignedException FromType<T>() | ||||||
|  |         => new($"{typeof(T).FullName} was not found"); | ||||||
|  | } | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | namespace Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | public class UniverseObjectNotFoundException(string? message) : NotFoundException(message); | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| namespace Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| public static class TransformExtensions |  | ||||||
| { |  | ||||||
|     public static Vector2D TransformVector2D(this ITransform transform, Vector2D vector) |  | ||||||
|         => vector.Scale(transform.Scale) |  | ||||||
|                  .Rotate(transform.Rotation * Math.DegreeToRadian) |  | ||||||
|                  .Add(transform.Position); |  | ||||||
| } |  | ||||||
							
								
								
									
										182
									
								
								Engine.Core/Extensions/BehaviourControllerExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										182
									
								
								Engine.Core/Extensions/BehaviourControllerExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,182 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Diagnostics.CodeAnalysis; | ||||||
|  |  | ||||||
|  | using Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public static class BehaviourControllerExtensions | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IBehaviour"/> of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to search in.</param> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IBehaviour"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IBehaviour"/> of the specified type was found; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetBehaviour<T>(this IBehaviourController behaviourController, [NotNullWhen(returnValue: true)] out T? behaviour) | ||||||
|  |     { | ||||||
|  |         behaviour = behaviourController.GetBehaviour<T>(); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IBehaviour"/> of the specified type in the provided <see cref="IBehaviourController"/>. Throws an error if not found. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, throws <see cref="BehaviourNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredBehaviour<T>(this IBehaviourController behaviourController) where T : class | ||||||
|  |         => behaviourController.GetBehaviour<T>() ?? throw new BehaviourNotFoundException($"{behaviourController.UniverseObject?.Name ?? "NULL"}'s {nameof(IBehaviourController)} does not contain any {typeof(T).FullName}"); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets an existing <see cref="IBehaviour"/> of the specified type, or adds and returns a new one if it doesn't exist. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get or add.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to search in.</param> | ||||||
|  |     /// <param name="args">Optional arguments to pass to the constructor of the <see cref="IBehaviour"/> if a new one is added.</param> | ||||||
|  |     /// <returns>The existing or newly added <see cref="IBehaviour"/> of the specified type.</returns> | ||||||
|  |     public static T GetOrAddBehaviour<T>(this IBehaviourController behaviourController, params object?[]? args) where T : class, IBehaviour | ||||||
|  |         => behaviourController.GetBehaviour<T>() ?? behaviourController.AddBehaviour<T>(args); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets an existing <see cref="IBehaviour"/> of the specified type, or adds and returns the fallback type if it doesn't exist. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="TOriginal">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <typeparam name="TFallback">The type of <see cref="IBehaviour"/> to add. It must be assignable from <typeparamref name="TOriginal"/></typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to search in.</param> | ||||||
|  |     /// <param name="args">Optional arguments to pass to the constructor of the <see cref="IBehaviour"/> if a new one is added.</param> | ||||||
|  |     /// <returns>The existing or newly added <see cref="IBehaviour"/> of the specified type.</returns> | ||||||
|  |     public static TOriginal GetOrAddBehaviour<TOriginal, TFallback>(this IBehaviourController behaviourController, params object?[]? args) | ||||||
|  |             where TOriginal : class | ||||||
|  |             where TFallback : class, IBehaviour, TOriginal | ||||||
|  |         => behaviourController.GetBehaviour<TOriginal>() ?? behaviourController.AddBehaviour<TFallback>(args); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IBehaviour"/> of the specified type in it's <see cref="IUniverseObject"/>'s parents recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IBehaviour"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IBehaviour"/> of the specified type was found in the parent universe; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetBehaviourInParent<T>(this IBehaviourController behaviourController, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = GetBehaviourInParent<T>(behaviourController); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IBehaviour"/> of the specified type in it's <see cref="IUniverseObject"/>'s parents recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, null.</returns> | ||||||
|  |     public static T? GetBehaviourInParent<T>(this IBehaviourController behaviourController) where T : class | ||||||
|  |     { | ||||||
|  |         IBehaviourController? controller = behaviourController; | ||||||
|  |  | ||||||
|  |         while (controller is not null) | ||||||
|  |         { | ||||||
|  |             if (controller.GetBehaviour<T>() is T behaviour) | ||||||
|  |                 return behaviour; | ||||||
|  |  | ||||||
|  |             controller = controller.UniverseObject.Parent?.BehaviourController; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IBehaviour"/> of the specified type in it's <see cref="IUniverseObject"/>'s parents recursively. Throws an error if not found. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, throws <see cref="BehaviourNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredBehaviourInParent<T>(this IBehaviourController behaviourController) where T : class | ||||||
|  |         => behaviourController.GetBehaviourInParent<T>() ?? throw new BehaviourNotFoundException($"{behaviourController.UniverseObject?.Name ?? "NULL"}'s {nameof(IBehaviourController)} does not contain any {typeof(T).FullName} on any parent"); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets all <see cref="IBehaviour"/>s of the specified type in it's <see cref="IUniverseObject"/>'s parents recursively and stores them in the provided list. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> | ||||||
|  |     /// <param name="behavioursInParent">The list to store the <see cref="IBehaviour"/>s.</param> | ||||||
|  |     public static void GetBehavioursInParent<T>(this IBehaviourController behaviourController, IList<T> behavioursInParent) where T : class | ||||||
|  |     { | ||||||
|  |         IBehaviourController? controller = behaviourController; | ||||||
|  |         List<T> cache = []; | ||||||
|  |         behavioursInParent.Clear(); | ||||||
|  |  | ||||||
|  |         while (controller is not null) | ||||||
|  |         { | ||||||
|  |             controller.GetBehaviours(cache); | ||||||
|  |             foreach (T behaviour in cache) | ||||||
|  |                 behavioursInParent.Add(behaviour); | ||||||
|  |  | ||||||
|  |             controller = controller.UniverseObject.Parent?.BehaviourController; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IBehaviour"/> of the specified type in it's <see cref="IUniverseObject"/>'s children recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IBehaviour"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IBehaviour"/> of the specified type was found in the child universe; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetBehaviourInChildren<T>(this IBehaviourController behaviourController, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = GetBehaviourInChildren<T>(behaviourController); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IBehaviour"/> of the specified type in it's <see cref="IUniverseObject"/>'s children recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, null.</returns> | ||||||
|  |     public static T? GetBehaviourInChildren<T>(this IBehaviourController behaviourController) where T : class | ||||||
|  |     { | ||||||
|  |         if (behaviourController.GetBehaviour<T>() is T localBehaviour) | ||||||
|  |             return localBehaviour; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject child in behaviourController.UniverseObject.Children) | ||||||
|  |             if (GetBehaviourInChildren<T>(child.BehaviourController) is T behaviour) | ||||||
|  |                 return behaviour; | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IBehaviour"/> of the specified type in the children recursively. Throws an error if not found. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviourController">The <see cref="IBehaviourController"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IBehaviour"/> of the specified type if found; otherwise, throws <see cref="BehaviourNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredBehaviourInChildren<T>(this IBehaviourController behaviourController) where T : class | ||||||
|  |         => behaviourController.GetBehaviourInChildren<T>() ?? throw new BehaviourNotFoundException($"{behaviourController.UniverseObject?.Name ?? "NULL"}'s {nameof(IBehaviourController)} does not contain any {typeof(T).FullName} on any children "); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets all <see cref="IBehaviour"/>s of the specified type in it's <see cref="IUniverseObject"/>'s children recursively and stores them in the provided list. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/>s to get.</typeparam> | ||||||
|  |     /// <param name="behavioursInChildren">The list to store the <see cref="IBehaviour"/>s.</param> | ||||||
|  |     public static void GetBehavioursInChildren<T>(this IBehaviourController behaviourController, IList<T> behavioursInChildren) where T : class | ||||||
|  |     { | ||||||
|  |         List<T> cache = []; | ||||||
|  |         behavioursInChildren.Clear(); | ||||||
|  |  | ||||||
|  |         TraverseChildrenForBehaviour(behaviourController.UniverseObject, behavioursInChildren, cache); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void TraverseChildrenForBehaviour<T>(IUniverseObject universeObject, IList<T> behaviours, IList<T> cache) where T : class | ||||||
|  |     { | ||||||
|  |         universeObject.BehaviourController.GetBehaviours(cache); | ||||||
|  |  | ||||||
|  |         foreach (T behaviour in cache) | ||||||
|  |             behaviours.Add(behaviour); | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject child in universeObject.Children) | ||||||
|  |             TraverseChildrenForBehaviour(child, behaviours, cache); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| using System.Collections.Generic; |  | ||||||
| using System.Diagnostics.CodeAnalysis; |  | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public static class BehaviourExtensions |  | ||||||
| { |  | ||||||
|     public static bool TryFindBehaviour<T>(this IEnumerable<IGameObject> gameObjects, [NotNullWhen(returnValue: true)] out T? behaviour) |  | ||||||
|     { |  | ||||||
|         behaviour = default; |  | ||||||
|  |  | ||||||
|         foreach (IGameObject gameObject in gameObjects) |  | ||||||
|             if (gameObject.BehaviourController.TryGetBehaviour(out behaviour)) |  | ||||||
|                 return true; |  | ||||||
|  |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static void FindBehaviours<T>(this IEnumerable<IGameObject> gameObjects, List<T> behaviours) |  | ||||||
|     { |  | ||||||
|         behaviours.Clear(); |  | ||||||
|         List<T> cache = []; |  | ||||||
|  |  | ||||||
|         foreach (IGameObject gameObject in gameObjects) |  | ||||||
|         { |  | ||||||
|             gameObject.BehaviourController.GetBehaviours(cache); |  | ||||||
|             behaviours.AddRange(cache); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										9
									
								
								Engine.Core/Extensions/EnumExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Engine.Core/Extensions/EnumExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public static class EnumExtensions | ||||||
|  | { | ||||||
|  |     public static bool CheckFlag(this Enum left, Enum right) | ||||||
|  |         => ((int)(object)left & (int)(object)right) != 0; | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| namespace Syntriax.Engine.Core; | namespace Engine.Core; | ||||||
|  |  | ||||||
| public static class FloatExtensions | public static class FloatExtensions | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,9 +0,0 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public static class GameManagerExtensions |  | ||||||
| { |  | ||||||
|     public static IGameObject InstantiateGameObject(this IGameManager gameManager, params object?[]? args) |  | ||||||
|         => gameManager.InstantiateGameObject<GameObject>(args); |  | ||||||
| } |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public static class GameObjectExtensions |  | ||||||
| { |  | ||||||
|     public static IGameObject SetGameObject(this IGameObject gameObject, string name) |  | ||||||
|     { |  | ||||||
|         gameObject.Name = name; |  | ||||||
|         return gameObject; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,14 +1,30 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; | namespace Engine.Core; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| public static class TransformExtensions | public static class TransformExtensions | ||||||
| { | { | ||||||
|     public static ITransform SetTransform(this ITransform transform, Vector2D? position = null, float? rotation = null, Vector2D? scale = null) |     public static ITransform2D SetTransform(this ITransform2D transform, | ||||||
|  |         Vector2D? position = null, float? rotation = null, Vector2D? scale = null, | ||||||
|  |         Vector2D? localPosition = null, float? localRotation = null, Vector2D? localScale = null) | ||||||
|     { |     { | ||||||
|         if (position.HasValue) transform.Position = position.Value; |         if (position.HasValue) transform.Position = position.Value; | ||||||
|         if (rotation.HasValue) transform.Rotation = rotation.Value; |         if (rotation.HasValue) transform.Rotation = rotation.Value; | ||||||
|         if (scale.HasValue) transform.Scale = scale.Value; |         if (scale.HasValue) transform.Scale = scale.Value; | ||||||
|  |         if (localPosition.HasValue) transform.LocalPosition = localPosition.Value; | ||||||
|  |         if (localRotation.HasValue) transform.LocalRotation = localRotation.Value; | ||||||
|  |         if (localScale.HasValue) transform.LocalScale = localScale.Value; | ||||||
|  |         return transform; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static ITransform3D SetTransform(this ITransform3D transform, | ||||||
|  |         Vector3D? position = null, Quaternion? rotation = null, Vector3D? scale = null, | ||||||
|  |         Vector3D? localPosition = null, Quaternion? localRotation = null, Vector3D? localScale = null) | ||||||
|  |     { | ||||||
|  |         if (position.HasValue) transform.Position = position.Value; | ||||||
|  |         if (rotation.HasValue) transform.Rotation = rotation.Value; | ||||||
|  |         if (scale.HasValue) transform.Scale = scale.Value; | ||||||
|  |         if (localPosition.HasValue) transform.LocalPosition = localPosition.Value; | ||||||
|  |         if (localRotation.HasValue) transform.LocalRotation = localRotation.Value; | ||||||
|  |         if (localScale.HasValue) transform.LocalScale = localScale.Value; | ||||||
|         return transform; |         return transform; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								Engine.Core/Extensions/UniverseExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								Engine.Core/Extensions/UniverseExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | using Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public static class UniverseExtensions | ||||||
|  | { | ||||||
|  |     public static IUniverseObject InstantiateUniverseObject(this IUniverse universe, params object?[]? args) | ||||||
|  |         => universe.InstantiateUniverseObject<UniverseObject>(args); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through all <see cref="IUniverseObject"/>s to find the specified instance of the type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">Type to be searched through the <see cref="IUniverse"/>.</typeparam> | ||||||
|  |     /// <returns>The specified type if found; otherwise, throws <see cref="UniverseObjectNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredUniverseObject<T>(this IUniverse universe) where T : class | ||||||
|  |         => universe.GetUniverseObject<T>() ?? throw new UniverseObjectNotFoundException($"{universe.GetType().FullName}({universe.Id}) does not contain any {nameof(IUniverseObject)} object of type {typeof(T).FullName}"); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through all <see cref="IBehaviours"/>s to find the specified instance of the type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">Type to be searched through the <see cref="IUniverse"/>.</typeparam> | ||||||
|  |     /// <returns>The specified type if found; otherwise, throws <see cref="BehaviourNotFoundException"/>.</returns> | ||||||
|  |     public static T FindRequiredBehaviour<T>(this IUniverse universe) where T : class | ||||||
|  |         => universe.FindBehaviour<T>() ?? throw new BehaviourNotFoundException($"{universe.GetType().FullName}({universe.Id}) does not contain any {nameof(IUniverseObject)} with {nameof(IBehaviour)} of type {typeof(T).FullName}"); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through all <see cref="IUniverseObject"/>s and <see cref="IBehaviours"/>s to find the specified instance of the type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <remarks> | ||||||
|  |     /// WARNING: This is more expensive compared to <see cref="GetRequiredUniverseObject{T}(IUniverse)"/> or <see cref="FindRequiredBehaviour{T}(IUniverse)"/> as it combines the two. If you know whether the type is either a type that gets implemented on an <see cref="IBehaviour"/> or <see cref="IUniverseObject"/> use the method appropriate for it for performance. | ||||||
|  |     /// </remarks> | ||||||
|  |     /// <typeparam name="T">Type to be searched through the <see cref="IUniverse"/>.</typeparam> | ||||||
|  |     /// <returns>The specified type if found; otherwise, throws <see cref="NotFoundException"/>.</returns> | ||||||
|  |     public static T FindRequired<T>(this IUniverse universe) where T : class | ||||||
|  |         => universe.Find<T>() ?? throw new NotFoundException($"{universe.GetType().FullName}({universe.Id}) does not contain any {nameof(IUniverseObject)} or {nameof(IBehaviour)} of type {typeof(T).FullName}"); | ||||||
|  | } | ||||||
							
								
								
									
										255
									
								
								Engine.Core/Extensions/UniverseObjectExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										255
									
								
								Engine.Core/Extensions/UniverseObjectExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,255 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Diagnostics.CodeAnalysis; | ||||||
|  |  | ||||||
|  | using Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public static class UniverseObjectExtensions | ||||||
|  | { | ||||||
|  |     public static T SetUniverseObject<T>(this T universeObject, string? name = "", IUniverseObject? parent = null) where T : IUniverseObject | ||||||
|  |     { | ||||||
|  |         if (!string.IsNullOrWhiteSpace(name)) | ||||||
|  |             universeObject.Name = name; | ||||||
|  |         if (parent is not null) | ||||||
|  |             universeObject.Parent = parent; | ||||||
|  |         return universeObject; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #region Universe Object Search | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IUniverseObject"/> of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObjects">The <see cref="IUniverseObject"/>s to search.</param> | ||||||
|  |     /// <returns>The first found <see cref="IUniverseObject"/> of the specified type; otherwise, null.</returns> | ||||||
|  |     public static T? GetUniverseObject<T>(this IEnumerable<IUniverseObject> universeObjects) where T : class | ||||||
|  |     { | ||||||
|  |         foreach (IUniverseObject universeObject in universeObjects) | ||||||
|  |             if (universeObject is T @object) | ||||||
|  |                 return @object; | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IUniverseObject"/> of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObjects">The <see cref="IUniverseObject"/>s to search.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IUniverseObject"/> of the specified type was found in the universe objects; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetUniverseObject<T>(this IEnumerable<IUniverseObject> universeObjects, [NotNullWhen(returnValue: true)] out T? universeObject) where T : class | ||||||
|  |     { | ||||||
|  |         universeObject = GetUniverseObject<T>(universeObjects); | ||||||
|  |         return universeObject is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through the provided <see cref="IUniverseObject"/>s to collect a list of <see cref="IUniverseObject"/>s of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to search.</param> | ||||||
|  |     /// <returns>The found <see cref="IUniverseObject"/>s of the specified types</returns> | ||||||
|  |     public static void GetUniverseObjects<T>(this IEnumerable<IUniverseObject> universeObjects, IList<T> foundUniverseObjects) where T : class | ||||||
|  |     { | ||||||
|  |         foundUniverseObjects.Clear(); | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in universeObjects) | ||||||
|  |             if (universeObject is T @object) | ||||||
|  |                 foundUniverseObjects.Add(@object); | ||||||
|  |     } | ||||||
|  |     #endregion | ||||||
|  |  | ||||||
|  |     #region Universe Object Search In Parent | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IUniverseObject"/> of the specified type in it's parents recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IUniverseObject"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IUniverseObject"/> of the specified type was found in the parent universe objects; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetUniverseObjectInParent<T>(this IUniverseObject universeObject, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = GetUniverseObjectInParent<T>(universeObject); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IUniverseObject"/> of the specified type in it's parents recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IUniverseObject"/> of the specified type if found; otherwise, null.</returns> | ||||||
|  |     public static T? GetUniverseObjectInParent<T>(this IUniverseObject universeObject) where T : class | ||||||
|  |     { | ||||||
|  |         if (universeObject.Children.GetUniverseObject<T>() is T localUniverseObject) | ||||||
|  |             return localUniverseObject; | ||||||
|  |  | ||||||
|  |         IUniverseObject? parent = universeObject; | ||||||
|  |  | ||||||
|  |         while (parent is not null) | ||||||
|  |         { | ||||||
|  |             if (parent is T behaviour) | ||||||
|  |                 return behaviour; | ||||||
|  |  | ||||||
|  |             parent = universeObject.Parent; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IUniverseObject"/> of the specified type in the parents recursively. Throws an error if not found. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IUniverseObject"/> of the specified type if found; otherwise, throws <see cref="UniverseObjectNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredUniverseObjectInParent<T>(this IUniverseObject universeObject) where T : class | ||||||
|  |         => universeObject.GetUniverseObjectInParent<T>() ?? throw new UniverseObjectNotFoundException($"{universeObject.Name}'s {nameof(IUniverseObject)} does not contain any {typeof(T).FullName} on any parent "); | ||||||
|  |  | ||||||
|  |     #endregion | ||||||
|  |  | ||||||
|  |     #region Universe Object Search In Children | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to get a <see cref="IUniverseObject"/> of the specified type in it's children recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IUniverseObject"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IUniverseObject"/> of the specified type was found in the child universe objects; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryGetUniverseObjectInChildren<T>(this IUniverseObject universeObject, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = GetUniverseObjectInChildren<T>(universeObject); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IUniverseObject"/> of the specified type in it's children recursively. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IUniverseObject"/> of the specified type if found; otherwise, null.</returns> | ||||||
|  |     public static T? GetUniverseObjectInChildren<T>(this IUniverseObject universeObject) where T : class | ||||||
|  |     { | ||||||
|  |         if (universeObject.Children.GetUniverseObject<T>() is T localUniverseObject) | ||||||
|  |             return localUniverseObject; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject child in universeObject.Children) | ||||||
|  |             if (GetUniverseObjectInChildren<T>(child) is T behaviour) | ||||||
|  |                 return behaviour; | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets a <see cref="IUniverseObject"/> of the specified type in the children recursively. Throws an error if not found. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IUniverseObject"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObject">The <see cref="IUniverseObject"/> to start searching from.</param> | ||||||
|  |     /// <returns>The <see cref="IUniverseObject"/> of the specified type if found; otherwise, throws <see cref="UniverseObjectNotFoundException"/>.</returns> | ||||||
|  |     public static T GetRequiredUniverseObjectInChildren<T>(this IUniverseObject universeObject) where T : class | ||||||
|  |         => universeObject.GetUniverseObjectInChildren<T>() ?? throw new UniverseObjectNotFoundException($"{universeObject.Name}'s {nameof(IUniverseObject)} does not contain any {typeof(T).FullName} on any children "); | ||||||
|  |     #endregion | ||||||
|  |  | ||||||
|  |     #region Behaviour Search | ||||||
|  |     /// <summary> | ||||||
|  |     /// Finds a <see cref="IBehaviour"/> of the specified type in the provided <see cref="IUniverseObject"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to find.</typeparam> | ||||||
|  |     /// <returns>The first found <see cref="IBehaviour"/> of the specified type; otherwise, null.</returns> | ||||||
|  |     public static T? FindBehaviour<T>(this IEnumerable<IUniverseObject> universeObjects) where T : class | ||||||
|  |     { | ||||||
|  |         foreach (IUniverseObject universeObject in universeObjects) | ||||||
|  |             if (universeObject.BehaviourController.GetBehaviour<T>() is T behaviour) | ||||||
|  |                 return behaviour; | ||||||
|  |  | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to find a <see cref="IBehaviour"/> of the specified type in the provided <see cref="IUniverseObject"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to find.</typeparam> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IUniverseObject"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if a <see cref="IBehaviour"/> of the specified type was found in the provided <see cref="IUniverseObject"/>s; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryFindBehaviour<T>(this IEnumerable<IUniverseObject> universeObjects, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = FindBehaviour<T>(universeObjects); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through the provided <see cref="IUniverseObject"/>s to collect a list of <see cref="IBehaviour"/>s of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="universeObjects">The <see cref="IUniverseObject"/>s to search.</param> | ||||||
|  |     public static void FindBehaviours<T>(this IEnumerable<IUniverseObject> universeObjects, IList<T> behaviours) where T : class | ||||||
|  |     { | ||||||
|  |         behaviours.Clear(); | ||||||
|  |         List<T> cache = []; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in universeObjects) | ||||||
|  |         { | ||||||
|  |             universeObject.BehaviourController.GetBehaviours(cache); | ||||||
|  |             foreach (T behaviour in cache) | ||||||
|  |                 behaviours.Add(behaviour); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     #endregion | ||||||
|  |  | ||||||
|  |     #region General Search | ||||||
|  |     /// <summary> | ||||||
|  |     /// Finds an object of the specified type in the provided <see cref="IUniverseObject"/>s and their <see cref="IBehaviour"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <remarks> | ||||||
|  |     /// WARNING: This is more expensive compared to <see cref="GetUniverseObject{T}(IEnumerable{IUniverseObject})"/> or <see cref="FindBehaviour{T}(IEnumerable{IUniverseObject})"/> as it combines the two. If you know whether the type is either a type that gets implemented on an <see cref="IBehaviour"/> or <see cref="IUniverseObject"/> use the method appropriate for it for performance. | ||||||
|  |     /// </remarks> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to find.</typeparam> | ||||||
|  |     /// <returns>The first found instance of the specified type; otherwise, null.</returns> | ||||||
|  |     public static T? Find<T>(this IEnumerable<IUniverseObject> universeObjects) where T : class | ||||||
|  |     { | ||||||
|  |         if (universeObjects.GetUniverseObject<T>() is T foundUniverseObject) | ||||||
|  |             return foundUniverseObject; | ||||||
|  |  | ||||||
|  |         if (universeObjects.FindBehaviour<T>() is T foundBehaviour) | ||||||
|  |             return foundBehaviour; | ||||||
|  |  | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Tries to find an object of the specified type in the provided <see cref="IUniverseObject"/>s and their <see cref="IBehaviour"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <remarks> | ||||||
|  |     /// WARNING: This is more expensive compared to <see cref="TryGetUniverseObject{T}(IEnumerable{IUniverseObject}, out T?)"/> or <see cref="TryFindBehaviour{T}(IEnumerable{IUniverseObject}, out T?)"/> as it combines the two. If you know whether the type is either a type that gets implemented on an <see cref="IBehaviour"/> or <see cref="IUniverseObject"/> use the method appropriate for it for performance. | ||||||
|  |     /// </remarks> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to find.</typeparam> | ||||||
|  |     /// <param name="behaviour">When this method returns, contains the <see cref="IUniverseObject"/> of the specified type, if found; otherwise, null.</param> | ||||||
|  |     /// <returns><see cref="true"/> if an object of the specified type was found in the provided <see cref="IUniverseObject"/>s; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool TryFind<T>(this IEnumerable<IUniverseObject> universeObjects, [NotNullWhen(returnValue: true)] out T? behaviour) where T : class | ||||||
|  |     { | ||||||
|  |         behaviour = Find<T>(universeObjects); | ||||||
|  |         return behaviour is not null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Searches through the provided <see cref="IUniverseObject"/>s and their <see cref="IBehaviour"/>s to collect a list of the specified type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <remarks> | ||||||
|  |     /// WARNING: This is more expensive compared to <see cref="GetUniverseObjects{T}(IEnumerable{IUniverseObject}, IList{T})"/> or <see cref="FindBehaviours{T}(IEnumerable{IUniverseObject}, IList{T})"/> as it combines the two. If you know whether the type is either a type that gets implemented on an <see cref="IBehaviour"/> or <see cref="IUniverseObject"/> use the method appropriate for it for performance. | ||||||
|  |     /// </remarks> | ||||||
|  |     /// <typeparam name="T">The type of <see cref="IBehaviour"/> to get.</typeparam> | ||||||
|  |     /// <param name="instances">List of objects found wit the specified type.</param> | ||||||
|  |     /// <param name="universeObjects">The <see cref="IUniverseObject"/>s to search.</param> | ||||||
|  |     public static void Find<T>(this IEnumerable<IUniverseObject> universeObjects, IList<T> instances) where T : class | ||||||
|  |     { | ||||||
|  |         instances.Clear(); | ||||||
|  |         List<T> cache = []; | ||||||
|  |  | ||||||
|  |         foreach (IUniverseObject universeObject in universeObjects) | ||||||
|  |         { | ||||||
|  |             universeObject.Children.Find(cache); | ||||||
|  |             foreach (T behaviour in cache) | ||||||
|  |                 instances.Add(behaviour); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     #endregion | ||||||
|  | } | ||||||
| @@ -1,188 +0,0 @@ | |||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| /// <summary> |  | ||||||
| /// Provides extension methods for <see cref="Vector2D"/> type. |  | ||||||
| /// </summary> |  | ||||||
| public static class Vector2DExtensions |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the length of the <see cref="Vector2D"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The length of the <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static float Length(this Vector2D vector) => Vector2D.Length(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the squared length of the <see cref="Vector2D"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The squared length of the <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static float LengthSquared(this Vector2D vector) => Vector2D.LengthSquared(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the distance between two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="from">The starting <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="to">The ending <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The distance between the two <see cref="Vector2D"/>s.</returns> |  | ||||||
|     public static float Distance(this Vector2D from, Vector2D to) => Vector2D.Distance(from, to); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Returns the <see cref="Vector2D"/> with its components inverted. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The inverted <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Invert(this Vector2D vector) => Vector2D.Invert(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Adds two <see cref="Vector2D"/>s component-wise. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="vectorToAdd">The vector <see cref="Vector2D"/> to be added.</param> |  | ||||||
|     /// <returns>The result of the addition.</returns> |  | ||||||
|     public static Vector2D Add(this Vector2D vector, Vector2D vectorToAdd) => Vector2D.Add(vector, vectorToAdd); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Subtracts one <see cref="Vector2D"/> from another component-wise. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="vectorToSubtract">The <see cref="Vector2D"/> to be subtracted.</param> |  | ||||||
|     /// <returns>The result of the subtraction.</returns> |  | ||||||
|     public static Vector2D Subtract(this Vector2D vector, Vector2D vectorToSubtract) => Vector2D.Subtract(vector, vectorToSubtract); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Multiplies a <see cref="Vector2D"/> by a scalar value. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to multiply.</param> |  | ||||||
|     /// <param name="value">The scalar value to multiply with.</param> |  | ||||||
|     /// <returns>The result of the multiplication.</returns> |  | ||||||
|     public static Vector2D Multiply(this Vector2D vector, float value) => Vector2D.Multiply(vector, value); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Divides a <see cref="Vector2D"/> by a scalar value. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to divide.</param> |  | ||||||
|     /// <param name="value">The scalar value to divide with.</param> |  | ||||||
|     /// <returns>The result of the division.</returns> |  | ||||||
|     public static Vector2D Divide(this Vector2D vector, float value) => Vector2D.Divide(vector, value); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Returns a <see cref="Vector2D"/> with the absolute values of each component. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The <see cref="Vector2D"/> with absolute values.</returns> |  | ||||||
|     public static Vector2D Abs(this Vector2D vector) => Vector2D.Abs(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Reflects a <see cref="Vector2D"/> off a surface with the specified normal. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to reflect.</param> |  | ||||||
|     /// <param name="normal">The normal <see cref="Vector2D"/> of the reflecting surface.</param> |  | ||||||
|     /// <returns>The reflected <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Reflect(this Vector2D vector, Vector2D normal) => Vector2D.Reflect(vector, normal); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Normalizes the <see cref="Vector2D"/> (creates a <see cref="Vector2D"/> with the same direction but with a length of 1). |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The normalized <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Normalize(this Vector2D vector) => Vector2D.Normalize(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Creates a <see cref="Vector2D"/> pointing from one point to another. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="from">The starting point.</param> |  | ||||||
|     /// <param name="to">The ending point.</param> |  | ||||||
|     /// <returns>The <see cref="Vector2D"/> pointing from <paramref name="from"/> to <paramref name="to"/>.</returns> |  | ||||||
|     public static Vector2D FromTo(this Vector2D from, Vector2D to) => Vector2D.FromTo(from, to); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Scales a <see cref="Vector2D"/> by another <see cref="Vector2D"/> component-wise. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to scale.</param> |  | ||||||
|     /// <param name="scale">The <see cref="Vector2D"/> containing the scaling factors for each component.</param> |  | ||||||
|     /// <returns>The scaled <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Scale(this Vector2D vector, Vector2D scale) => Vector2D.Scale(vector, scale); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the perpendicular <see cref="Vector2D"/> to the given <see cref="Vector2D"/>. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The input <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>A <see cref="Vector2D"/> perpendicular to the input <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Perpendicular(this Vector2D vector) => Vector2D.Perpendicular(vector); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Rotates a <see cref="Vector2D"/> by the specified angle (in radians). |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to rotate.</param> |  | ||||||
|     /// <param name="angleInRadian">The angle to rotate by, in radians.</param> |  | ||||||
|     /// <returns>The rotated <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Rotate(this Vector2D vector, float angleInRadian) => Vector2D.Rotate(vector, angleInRadian); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Returns the component-wise minimum of two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The <see cref="Vector2D"/> containing the minimum components from both input <see cref="Vector2D"/>s.</returns> |  | ||||||
|     public static Vector2D Min(this Vector2D left, Vector2D right) => Vector2D.Min(left, right); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Returns the component-wise maximum of two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The <see cref="Vector2D"/> containing the maximum components from both input <see cref="Vector2D"/>s.</returns> |  | ||||||
|     public static Vector2D Max(this Vector2D left, Vector2D right) => Vector2D.Max(left, right); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Clamps each component of a <see cref="Vector2D"/> between the corresponding component of two other <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="vector">The <see cref="Vector2D"/> to clamp.</param> |  | ||||||
|     /// <param name="min">The <see cref="Vector2D"/> representing the minimum values for each component.</param> |  | ||||||
|     /// <param name="max">The <see cref="Vector2D"/> representing the maximum values for each component.</param> |  | ||||||
|     /// <returns>The clamped <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Clamp(this Vector2D vector, Vector2D min, Vector2D max) => Vector2D.Clamp(vector, min, max); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Linearly interpolates between two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="from">The start <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="to">The end <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="t">The interpolation parameter (between 0 and 1).</param> |  | ||||||
|     /// <returns>The interpolated <see cref="Vector2D"/>.</returns> |  | ||||||
|     public static Vector2D Lerp(this Vector2D from, Vector2D to, float t) => Vector2D.Lerp(from, to, t); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the cross product of two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The cross product of the two <see cref="Vector2D"/>s.</returns> |  | ||||||
|     public static float Cross(this Vector2D left, Vector2D right) => Vector2D.Cross(left, right); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the angle in radians between two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The angle between the two <see cref="Vector2D"/>s in radians.</returns> |  | ||||||
|     public static float AngleBetween(this Vector2D left, Vector2D right) => Vector2D.Angle(left, right); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Calculates the dot product of two <see cref="Vector2D"/>s. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <returns>The dot product of the two <see cref="Vector2D"/>s.</returns> |  | ||||||
|     public static float Dot(this Vector2D left, Vector2D right) => Vector2D.Dot(left, right); |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Checks whether two <see cref="Vector2D"/>s are approximately equal within a certain epsilon range. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="left">The first <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="right">The second <see cref="Vector2D"/>.</param> |  | ||||||
|     /// <param name="epsilon">The maximum difference allowed between components.</param> |  | ||||||
|     /// <returns>True if the <see cref="Vector2D"/>s are approximately equal, false otherwise.</returns> |  | ||||||
|     public static bool ApproximatelyEquals(this Vector2D left, Vector2D right, float epsilon = float.Epsilon) => Vector2D.ApproximatelyEquals(left, right, epsilon); |  | ||||||
| } |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| namespace Syntriax.Engine.Core.Factory.Abstract; | namespace Engine.Core.Factory.Abstract; | ||||||
|  |  | ||||||
| public interface IFactory<TInterface> where TInterface : class | public interface IFactory<TInterface> where TInterface : class | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,20 +1,33 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; | using Engine.Core.Exceptions; | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| public class BehaviourControllerFactory | public class BehaviourControllerFactory | ||||||
| { | { | ||||||
|     public IBehaviourController Instantiate(IGameObject gameObject) |     public static IBehaviourController Instantiate(IUniverseObject universeObject, IStateEnable? stateEnable = null) | ||||||
|         => Instantiate<BehaviourController>(gameObject); |         => Instantiate<BehaviourController>(universeObject, stateEnable); | ||||||
|  |  | ||||||
|     public T Instantiate<T>(IGameObject gameObject, params object?[]? args) |     public static T Instantiate<T>(IUniverseObject universeObject, IStateEnable? stateEnable = null, params object?[]? args) | ||||||
|         where T : class, IBehaviourController |         where T : class, IBehaviourController | ||||||
|     { |     { | ||||||
|         T behaviourController = TypeFactory.Get<T>(args); |         T behaviourController = TypeFactory.Get<T>(args); | ||||||
|  |  | ||||||
|         if (!behaviourController.Assign(gameObject)) |         if (!universeObject.Assign(behaviourController)) | ||||||
|             throw AssignException.From(behaviourController, gameObject); |             throw AssignFailedException.From(universeObject, behaviourController); | ||||||
|  |  | ||||||
|  |         if (!behaviourController.Assign(universeObject)) | ||||||
|  |             throw AssignFailedException.From(behaviourController, universeObject); | ||||||
|  |  | ||||||
|  |         if (stateEnable is not null) | ||||||
|  |         { | ||||||
|  |             if (!stateEnable.Assign(behaviourController)) | ||||||
|  |                 throw AssignFailedException.From(stateEnable, behaviourController); | ||||||
|  |  | ||||||
|  |             if (!behaviourController.Assign(stateEnable)) | ||||||
|  |                 throw AssignFailedException.From(behaviourController, stateEnable); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             StateEnableFactory.Instantiate(behaviourController); | ||||||
|  |  | ||||||
|         return behaviourController; |         return behaviourController; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,26 +1,26 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; | using Engine.Core.Exceptions; | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| public class BehaviourFactory | public class BehaviourFactory | ||||||
| { | { | ||||||
|     public T Instantiate<T>(IGameObject gameObject, params object?[]? args) where T : class, IBehaviour |     public static T Instantiate<T>(params object?[]? args) where T : class, IBehaviour | ||||||
|         => Instantiate<T>(gameObject, stateEnable: null, args); |         => Instantiate<T>(stateEnable: null, args); | ||||||
|  |  | ||||||
|     public T Instantiate<T>(IGameObject gameObject, IStateEnable? stateEnable, params object?[]? args) |     public static T Instantiate<T>(IStateEnable? stateEnable, params object?[]? args) | ||||||
|         where T : class, IBehaviour |         where T : class, IBehaviour | ||||||
|     { |     { | ||||||
|         T behaviour = TypeFactory.Get<T>(args); |         T behaviour = TypeFactory.Get<T>(args); | ||||||
|  |  | ||||||
|         stateEnable ??= TypeFactory.Get<StateEnable>(); |         if (stateEnable is not null) | ||||||
|         if (!stateEnable.Assign(behaviour)) |         { | ||||||
|             throw AssignException.From(stateEnable, behaviour); |             if (!stateEnable.Assign(behaviour)) | ||||||
|  |                 throw AssignFailedException.From(stateEnable, behaviour); | ||||||
|         if (!behaviour.Assign(gameObject.BehaviourController)) |             if (!behaviour.Assign(stateEnable)) | ||||||
|             throw AssignException.From(behaviour, gameObject.BehaviourController); |                 throw AssignFailedException.From(behaviour, stateEnable); | ||||||
|         if (!behaviour.Assign(stateEnable)) |         } | ||||||
|             throw AssignException.From(behaviour, stateEnable); |         else | ||||||
|  |             StateEnableFactory.Instantiate(behaviour); | ||||||
|  |  | ||||||
|         return behaviour; |         return behaviour; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| using System; | using System; | ||||||
| using Syntriax.Engine.Core.Factory.Abstract; | using Engine.Core.Factory.Abstract; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| public abstract class FactoryBase<TInterface> : IFactory<TInterface> | public abstract class FactoryBase<TInterface> : IFactory<TInterface> | ||||||
|     where TInterface : class |     where TInterface : class | ||||||
|   | |||||||
| @@ -1,39 +0,0 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; |  | ||||||
|  |  | ||||||
| public class GameObjectFactory |  | ||||||
| { |  | ||||||
|     public T Instantiate<T>(params object?[]? args) where T : class, IGameObject |  | ||||||
|         => Instantiate<T>(transform: null, behaviourController: null, stateEnable: null, args); |  | ||||||
|  |  | ||||||
|     public T Instantiate<T>( |  | ||||||
|         ITransform? transform = null, |  | ||||||
|         IBehaviourController? behaviourController = null, |  | ||||||
|         IStateEnable? stateEnable = null, |  | ||||||
|         params object?[]? args |  | ||||||
|     ) |  | ||||||
|         where T : class, IGameObject |  | ||||||
|     { |  | ||||||
|         T gameObject = TypeFactory.Get<T>(args); |  | ||||||
|  |  | ||||||
|         transform ??= TypeFactory.Get<Transform>(); |  | ||||||
|         behaviourController ??= TypeFactory.Get<BehaviourController>(); |  | ||||||
|         stateEnable ??= TypeFactory.Get<StateEnable>(); |  | ||||||
|  |  | ||||||
|         if (!behaviourController.Assign(gameObject)) |  | ||||||
|             throw AssignException.From(behaviourController, gameObject); |  | ||||||
|         if (!stateEnable.Assign(gameObject)) |  | ||||||
|             throw AssignException.From(stateEnable, gameObject); |  | ||||||
|  |  | ||||||
|         if (!gameObject.Assign(transform)) |  | ||||||
|             throw AssignException.From(gameObject, transform); |  | ||||||
|         if (!gameObject.Assign(behaviourController)) |  | ||||||
|             throw AssignException.From(gameObject, behaviourController); |  | ||||||
|         if (!gameObject.Assign(stateEnable)) |  | ||||||
|             throw AssignException.From(gameObject, stateEnable); |  | ||||||
|  |  | ||||||
|         return gameObject; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,18 +1,20 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; | using Engine.Core.Exceptions; | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| public class StateEnableFactory | public class StateEnableFactory | ||||||
| { | { | ||||||
|     public IStateEnable Instantiate(IEntity entity) => Instantiate<StateEnable>(entity); |     public static IStateEnable Instantiate(IEntity entity) => Instantiate<StateEnable>(entity); | ||||||
|  |  | ||||||
|     public T Instantiate<T>(IEntity entity, params object?[]? args) where T : class, IStateEnable |     public static T Instantiate<T>(IEntity entity, params object?[]? args) where T : class, IStateEnable | ||||||
|     { |     { | ||||||
|         T stateEnable = TypeFactory.Get<T>(args); |         T stateEnable = TypeFactory.Get<T>(args); | ||||||
|  |  | ||||||
|  |         if (!entity.Assign(stateEnable)) | ||||||
|  |             throw AssignFailedException.From(entity, stateEnable); | ||||||
|  |  | ||||||
|         if (!stateEnable.Assign(entity)) |         if (!stateEnable.Assign(entity)) | ||||||
|             throw AssignException.From(stateEnable, entity); |             throw AssignFailedException.From(stateEnable, entity); | ||||||
|  |  | ||||||
|         return stateEnable; |         return stateEnable; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,10 +1,8 @@ | |||||||
| using Syntriax.Engine.Core.Abstract; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; |  | ||||||
|  |  | ||||||
| public class TransformFactory | public class TransformFactory | ||||||
| { | { | ||||||
|     public ITransform Instantiate() => TypeFactory.Get<Transform>(); |     public static ITransform2D Instantiate() => TypeFactory.Get<Transform2D>(); | ||||||
|     public T Instantiate<T>(params object?[]? args) where T : class, ITransform |     public static T Instantiate<T>(params object?[]? args) where T : class, ITransform2D | ||||||
|         => TypeFactory.Get<T>(args); |         => TypeFactory.Get<T>(args); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,21 +1,59 @@ | |||||||
| using System; | using System; | ||||||
|  | using System.Collections.Concurrent; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core.Factory; | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
| public static class TypeFactory | public static class TypeFactory | ||||||
| { | { | ||||||
|     public static T Get<T>(params object?[]? args) where T : class |     private static readonly ConcurrentDictionary<string, Type> registeredTypes = []; | ||||||
|  |  | ||||||
|  |     public static string GetTypeName(Type type) => type.FullName ?? throw new ArgumentException($"{type.Name} must be a resolvable type"); | ||||||
|  |  | ||||||
|  |     public static T Get<T>(params object?[]? args) where T : class => (T)Get(typeof(T), args); | ||||||
|  |     public static object Get(string fullName, params object?[]? args) => Get(GetType(fullName), args); | ||||||
|  |     public static object Get(Type type, params object?[]? args) | ||||||
|     { |     { | ||||||
|         T? result; |         object? result; | ||||||
|  |  | ||||||
|         if (args is not null && args.Length != 0) |         if (args is not null && args.Length != 0) | ||||||
|             result = Activator.CreateInstance(typeof(T), args) as T; |             result = Activator.CreateInstance(type, args); | ||||||
|         else |         else | ||||||
|             result = Activator.CreateInstance(typeof(T)) as T; |             result = Activator.CreateInstance(type); | ||||||
|  |  | ||||||
|         if (result is null) |         if (result is null) | ||||||
|             throw new Exception($"{typeof(T).Name} of type {typeof(T).Name} could not be created."); |             throw new Exception($"Type {type.Name} could not be created."); | ||||||
|  |  | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static Type GetType(string fullName) | ||||||
|  |     { | ||||||
|  |         if (registeredTypes.TryGetValue(fullName, out Type? result)) | ||||||
|  |             return result; | ||||||
|  |  | ||||||
|  |         ReloadTypes(); | ||||||
|  |  | ||||||
|  |         if (registeredTypes.TryGetValue(fullName, out Type? reloadedType)) | ||||||
|  |             return reloadedType; | ||||||
|  |  | ||||||
|  |         throw new Exception($"Type {fullName} could not be found in the current domain."); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void ReloadTypes() | ||||||
|  |     { | ||||||
|  |         registeredTypes.Clear(); | ||||||
|  |  | ||||||
|  |         IEnumerable<Type> domainTypes = AppDomain.CurrentDomain | ||||||
|  |             .GetAssemblies() | ||||||
|  |             .SelectMany(a => a.GetTypes()); | ||||||
|  |  | ||||||
|  |         // TODO: Replace this | ||||||
|  |         // There are some system & compiler generated types with duplicated names,  | ||||||
|  |         // it is ugly it will cause headaches in the future because it will not  | ||||||
|  |         // throw an error if there's a type with an unintended duplicate name | ||||||
|  |         foreach (Type type in domainTypes) | ||||||
|  |             registeredTypes.TryAdd(GetTypeName(type), type); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								Engine.Core/Factory/UniverseObjectFactory.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								Engine.Core/Factory/UniverseObjectFactory.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | using Engine.Core.Exceptions; | ||||||
|  |  | ||||||
|  | namespace Engine.Core.Factory; | ||||||
|  |  | ||||||
|  | public class UniverseObjectFactory | ||||||
|  | { | ||||||
|  |     public static IUniverseObject Instantiate() => Instantiate<UniverseObject>(); | ||||||
|  |     public static T Instantiate<T>(params object?[]? args) where T : class, IUniverseObject | ||||||
|  |         => Instantiate<T>(behaviourController: null, stateEnable: null, args); | ||||||
|  |  | ||||||
|  |     public static IUniverseObject Instantiate(IBehaviourController? behaviourController = null, IStateEnable? stateEnable = null) => Instantiate<UniverseObject>(behaviourController, stateEnable); | ||||||
|  |     public static T Instantiate<T>( | ||||||
|  |         IBehaviourController? behaviourController = null, | ||||||
|  |         IStateEnable? stateEnable = null, | ||||||
|  |         params object?[]? args | ||||||
|  |     ) | ||||||
|  |         where T : class, IUniverseObject | ||||||
|  |     { | ||||||
|  |         T universeObject = TypeFactory.Get<T>(args); | ||||||
|  |  | ||||||
|  |         if (behaviourController is not null) | ||||||
|  |         { | ||||||
|  |             if (!behaviourController.Assign(universeObject)) | ||||||
|  |                 throw AssignFailedException.From(behaviourController, universeObject); | ||||||
|  |             if (!universeObject.Assign(behaviourController)) | ||||||
|  |                 throw AssignFailedException.From(universeObject, behaviourController); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             BehaviourControllerFactory.Instantiate(universeObject); | ||||||
|  |  | ||||||
|  |         if (stateEnable is not null) | ||||||
|  |         { | ||||||
|  |             if (!stateEnable.Assign(universeObject)) | ||||||
|  |                 throw AssignFailedException.From(stateEnable, universeObject); | ||||||
|  |             if (!universeObject.Assign(stateEnable)) | ||||||
|  |                 throw AssignFailedException.From(universeObject, stateEnable); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             StateEnableFactory.Instantiate(universeObject); | ||||||
|  |  | ||||||
|  |         return universeObject; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,132 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; |  | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
| using Syntriax.Engine.Core.Factory; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| [System.Diagnostics.DebuggerDisplay("GameObject Count: {_gameObjects.Count}")] |  | ||||||
| public class GameManager : BaseEntity, IGameManager |  | ||||||
| { |  | ||||||
|     public Action<IGameManager, IGameObject>? OnGameObjectRegistered { get; set; } = null; |  | ||||||
|     public Action<IGameManager, IGameObject>? OnGameObjectUnRegistered { get; set; } = null; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     private readonly List<IGameObject> _gameObjects = new(Constants.GAME_OBJECTS_SIZE_INITIAL); |  | ||||||
|  |  | ||||||
|     private GameObjectFactory _gameObjectFactory = null!; |  | ||||||
|  |  | ||||||
|     private GameObjectFactory GameObjectFactory |  | ||||||
|     { |  | ||||||
|         get |  | ||||||
|         { |  | ||||||
|             if (_gameObjectFactory is null) |  | ||||||
|                 _gameObjectFactory = new GameObjectFactory(); |  | ||||||
|             return _gameObjectFactory; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public IReadOnlyList<IGameObject> GameObjects => _gameObjects; |  | ||||||
|  |  | ||||||
|     public override IStateEnable StateEnable |  | ||||||
|     { |  | ||||||
|         get |  | ||||||
|         { |  | ||||||
|             if (base.StateEnable is null) |  | ||||||
|             { |  | ||||||
|                 Assign(new StateEnableFactory().Instantiate(this)); |  | ||||||
|                 if (base.StateEnable is null) |  | ||||||
|                     throw NotAssignedException.From(this, base.StateEnable); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return base.StateEnable; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void RegisterGameObject(IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         if (_gameObjects.Contains(gameObject)) |  | ||||||
|             throw new Exception($"{nameof(IGameObject)} named {gameObject.Name} is already registered to the {nameof(GameManager)}."); |  | ||||||
|  |  | ||||||
|         Register(gameObject); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public T InstantiateGameObject<T>(params object?[]? args) where T : class, IGameObject |  | ||||||
|     { |  | ||||||
|         T gameObject = GameObjectFactory.Instantiate<T>(args); |  | ||||||
|         Register(gameObject); |  | ||||||
|         return gameObject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public IGameObject RemoveGameObject(IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         if (!_gameObjects.Contains(gameObject)) |  | ||||||
|             throw new Exception($"{nameof(IGameObject)} named {gameObject.Name} is not registered to the {nameof(GameManager)}."); |  | ||||||
|  |  | ||||||
|         Unregister(gameObject); |  | ||||||
|         return gameObject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected override void InitializeInternal() |  | ||||||
|     { |  | ||||||
|         base.InitializeInternal(); |  | ||||||
|         NotAssignedException.Check(this, StateEnable); |  | ||||||
|  |  | ||||||
|         foreach (var gameObject in GameObjects) |  | ||||||
|             gameObject.Initialize(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected override void FinalizeInternal() |  | ||||||
|     { |  | ||||||
|         base.FinalizeInternal(); |  | ||||||
|         for (int i = GameObjects.Count; i >= 0; i--) |  | ||||||
|             GameObjects[i].Finalize(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void Update(EngineTime time) |  | ||||||
|     { |  | ||||||
|         Time.SetTime(time); |  | ||||||
|         foreach (var gameObject in GameObjects) |  | ||||||
|             gameObject.BehaviourController.Update(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void PreDraw() |  | ||||||
|     { |  | ||||||
|         foreach (var gameObject in GameObjects) |  | ||||||
|             gameObject.BehaviourController.UpdatePreDraw(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     ///////////////////////////////////////////////////////////////// |  | ||||||
|  |  | ||||||
|     private void Register(IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         gameObject.Assign(this); |  | ||||||
|  |  | ||||||
|         gameObject.OnFinalized += OnGameObjectFinalize; |  | ||||||
|  |  | ||||||
|         _gameObjects.Add(gameObject); |  | ||||||
|         OnGameObjectRegistered?.Invoke(this, gameObject); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private void Unregister(IGameObject gameObject) |  | ||||||
|     { |  | ||||||
|         gameObject.OnFinalized -= OnGameObjectFinalize; |  | ||||||
|  |  | ||||||
|         _gameObjects.Remove(gameObject); |  | ||||||
|         OnGameObjectUnRegistered?.Invoke(this, gameObject); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private void OnGameObjectFinalize(IInitialize initialize) |  | ||||||
|     { |  | ||||||
|         if (initialize is IGameObject gameObject) |  | ||||||
|             Unregister(gameObject); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     ///////////////////////////////////////////////////////////////// |  | ||||||
|  |  | ||||||
|     public IEnumerator<IGameObject> GetEnumerator() => _gameObjects.GetEnumerator(); |  | ||||||
|     IEnumerator IEnumerable.GetEnumerator() => _gameObjects.GetEnumerator(); |  | ||||||
| } |  | ||||||
| @@ -1,117 +0,0 @@ | |||||||
| using System; |  | ||||||
|  |  | ||||||
| using Syntriax.Engine.Core.Abstract; |  | ||||||
| using Syntriax.Engine.Core.Exceptions; |  | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; |  | ||||||
|  |  | ||||||
| [System.Diagnostics.DebuggerDisplay("Name: {Name}, Initialized: {Initialized}")] |  | ||||||
| public class GameObject : BaseEntity, IGameObject |  | ||||||
| { |  | ||||||
|     public Action<IAssignableTransform>? OnTransformAssigned { get; set; } = null; |  | ||||||
|     public Action<IAssignableBehaviourController>? OnBehaviourControllerAssigned { get; set; } = null; |  | ||||||
|     public Action<IAssignableGameManager>? OnGameManagerAssigned { get; set; } = null; |  | ||||||
|  |  | ||||||
|     public Action<IEntity>? OnNameChanged { get; set; } = null; |  | ||||||
|  |  | ||||||
|     public Action<IGameObject>? OnUpdated { get; set; } = null; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     private ITransform _transform = null!; |  | ||||||
|     private IBehaviourController _behaviourController = null!; |  | ||||||
|     private IStateEnable _stateEnable = null!; |  | ||||||
|     private IGameManager _gameManager = null!; |  | ||||||
|  |  | ||||||
|     private string _name = nameof(GameObject); |  | ||||||
|  |  | ||||||
|     public ITransform Transform => _transform; |  | ||||||
|     public IBehaviourController BehaviourController => _behaviourController; |  | ||||||
|     public IGameManager GameManager => _gameManager; |  | ||||||
|  |  | ||||||
|     public string Name |  | ||||||
|     { |  | ||||||
|         get => _name; |  | ||||||
|         set |  | ||||||
|         { |  | ||||||
|             if (value == _name) return; |  | ||||||
|  |  | ||||||
|             _name = value; |  | ||||||
|             OnNameChanged?.Invoke(this); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected override void InitializeInternal() |  | ||||||
|     { |  | ||||||
|         base.InitializeInternal(); |  | ||||||
|  |  | ||||||
|         NotAssignedException.Check(this, _transform); |  | ||||||
|         NotAssignedException.Check(this, _behaviourController); |  | ||||||
|         NotAssignedException.Check(this, _stateEnable); |  | ||||||
|         NotAssignedException.Check(this, _gameManager); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void Update() |  | ||||||
|     { |  | ||||||
|         if (!_stateEnable.Enabled) |  | ||||||
|             return; |  | ||||||
|  |  | ||||||
|         OnUpdated?.Invoke(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected override void FinalizeInternal() |  | ||||||
|     { |  | ||||||
|         base.FinalizeInternal(); |  | ||||||
|  |  | ||||||
|         foreach (IBehaviour behaviour in _behaviourController.GetBehaviours<IBehaviour>()) |  | ||||||
|             behaviour.Finalize(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Assign(ITransform transform) |  | ||||||
|     { |  | ||||||
|         if (Initialized) |  | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         _transform = transform; |  | ||||||
|         OnTransformAssigned?.Invoke(this); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Assign(IBehaviourController behaviourController) |  | ||||||
|     { |  | ||||||
|         if (Initialized) |  | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         _behaviourController = behaviourController; |  | ||||||
|         OnBehaviourControllerAssigned?.Invoke(this); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Assign(IGameManager gameManager) |  | ||||||
|     { |  | ||||||
|         if (Initialized) |  | ||||||
|             return false; |  | ||||||
|  |  | ||||||
|         _gameManager = gameManager; |  | ||||||
|         OnGameManagerAssigned?.Invoke(this); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected override void UnassignInternal() |  | ||||||
|     { |  | ||||||
|         base.UnassignInternal(); |  | ||||||
|  |  | ||||||
|         _stateEnable = null!; |  | ||||||
|         _transform = null!; |  | ||||||
|         _behaviourController = null!; |  | ||||||
|         _gameManager = null!; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public GameObject() { OnBehaviourControllerAssigned += ConnectBehaviourController; } |  | ||||||
|     private void ConnectBehaviourController(IAssignableBehaviourController controller) |  | ||||||
|     { |  | ||||||
|         controller.BehaviourController.OnBehaviourAdded += OnBehaviourAdded; |  | ||||||
|         controller.BehaviourController.OnBehaviourRemoved += OnBehaviourRemoved; |  | ||||||
|     } |  | ||||||
|     private void OnBehaviourRemoved(IBehaviourController _, IBehaviour behaviour) { if (Initialized) behaviour.Initialize(); } |  | ||||||
|     private void OnBehaviourAdded(IBehaviourController _, IBehaviour behaviour) { if (Initialized) behaviour.Finalize(); } |  | ||||||
| } |  | ||||||
							
								
								
									
										506
									
								
								Engine.Core/Helpers/Event.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										506
									
								
								Engine.Core/Helpers/Event.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,506 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | using Engine.Core.Debug; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | // TODO!: every reverse loop has a chance to have more than 1 unsubscription, | ||||||
|  | // for (int i = listeners.Count - 1; i >= 0; i--)  | ||||||
|  | // can be replaced with  | ||||||
|  | // for (int i = listeners.Count - 1; i >= 0; i = Math.Min(i - 1, listeners.Count - 1)) | ||||||
|  | // but this would causes possible double calls on already called callbacks, find a better method. | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents a simple event with no parameters. | ||||||
|  | /// <para>Example usage:</para>  | ||||||
|  | /// <code> | ||||||
|  | /// public class MyBehaviour : Behaviour, IUpdate | ||||||
|  | /// { | ||||||
|  | ///     public readonly Event MyEvent = new(); | ||||||
|  | ///  | ||||||
|  | ///     public MyBehaviour() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.AddListener(OnEventTriggered); | ||||||
|  | ///         MyEvent.AddOneTimeListener(OnEventTriggeredOneTime); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     public void Update() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.Invoke(); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private void OnEventTriggered() | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"Event occurred!"); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private static void OnEventTriggeredOneTime() | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"Event called once!"); | ||||||
|  | ///     } | ||||||
|  | /// } | ||||||
|  | /// </code> | ||||||
|  | /// The output of the example code above would be:  | ||||||
|  | /// <code> | ||||||
|  | /// Event occurred! | ||||||
|  | /// Event called once! | ||||||
|  | /// Event occurred! | ||||||
|  | /// Event occurred! | ||||||
|  | /// Event occurred! | ||||||
|  | /// ... | ||||||
|  | /// </code> | ||||||
|  | /// </summary> | ||||||
|  | public class Event | ||||||
|  | { | ||||||
|  |     // We use Ascending order because draw calls are running from last to first | ||||||
|  |     private static readonly Comparer<ListenerData> SortByAscendingPriority = Comparer<ListenerData>.Create((x, y) => x.Priority.CompareTo(y.Priority)); | ||||||
|  |  | ||||||
|  |     private ILogger _logger = ILogger.Shared; | ||||||
|  |     public ILogger Logger { get => _logger; set => _logger = value ?? ILogger.Shared; } | ||||||
|  |  | ||||||
|  |     private readonly List<ListenerData> listeners = null!; | ||||||
|  |     private readonly List<ListenerData> onceListeners = null!; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked whenever the event is triggered. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called when the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = listeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         listeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked the next time the event is triggered. The callback will be called only once. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called the next time the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddOneTimeListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = onceListeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         onceListeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             if (listeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 listeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveOneTimeListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = 0; i < onceListeners.Count; i++) | ||||||
|  |             if (onceListeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 onceListeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes all listeners that was previously registered by either <see cref="AddListener(EventHandler)"/> or <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public void Clear() { listeners.Clear(); onceListeners.Clear(); } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Triggers the event. | ||||||
|  |     /// </summary> | ||||||
|  |     public void Invoke() | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             try { listeners[i].Callback.Invoke(); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{listeners[i].Callback.Method.DeclaringType?.FullName}.{listeners[i].Callback.Method.Name}()"; | ||||||
|  |                 EventHelpers.LogInvocationException(listeners[i].Callback.Target ?? this, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         for (int i = onceListeners.Count - 1; i >= 0; i--) | ||||||
|  |         { | ||||||
|  |             try { onceListeners[i].Callback.Invoke(); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{onceListeners[i].Callback.Method.DeclaringType?.FullName}.{onceListeners[i].Callback.Method.Name}()"; | ||||||
|  |                 EventHelpers.LogInvocationException(onceListeners[i].Callback.Target ?? this, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |             onceListeners.RemoveAt(i); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event(int initialListenerCount = 4, int initialOnceListenerCount = 2) | ||||||
|  |     { | ||||||
|  |         listeners = new(initialListenerCount); | ||||||
|  |         onceListeners = new(initialOnceListenerCount); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event() | ||||||
|  |     { | ||||||
|  |         listeners = new(4); | ||||||
|  |         onceListeners = new(2); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public delegate void EventHandler(); | ||||||
|  |     private record struct ListenerData(EventHandler Callback, int Priority); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents an event with only sender parameters. | ||||||
|  | /// <para>Example usage:</para>  | ||||||
|  | /// <code> | ||||||
|  | /// public class MyBehaviour : Behaviour, IUpdate | ||||||
|  | /// { | ||||||
|  | ///     public readonly Event<MyBehaviour> MyEvent = new(); | ||||||
|  | ///  | ||||||
|  | ///     public MyBehaviour() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.AddListener(OnEventTriggered); | ||||||
|  | ///         MyEvent.AddOneTimeListener(OnEventTriggeredOneTime); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     public void Update() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.Invoke(this); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private void OnEventTriggered(MyBehaviour sender) | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"{sender.Id}'s event occurred!"); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private static void OnEventTriggeredOneTime(MyBehaviour sender) | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"{sender.Id}'s event called once!"); | ||||||
|  | ///     } | ||||||
|  | /// } | ||||||
|  | /// </code> | ||||||
|  | /// The output of the example code above would be:  | ||||||
|  | /// <code> | ||||||
|  | /// [Id]'s event occurred! | ||||||
|  | /// [Id]'s event called once! | ||||||
|  | /// [Id]'s event occurred! | ||||||
|  | /// [Id]'s event occurred! | ||||||
|  | /// [Id]'s event occurred! | ||||||
|  | /// ... | ||||||
|  | /// </code> | ||||||
|  | ///  | ||||||
|  | /// </summary> | ||||||
|  | /// <typeparam name="TSender">Sender type</typeparam> | ||||||
|  | public class Event<TSender> where TSender : class | ||||||
|  | { | ||||||
|  |     // We use Ascending order because draw calls are running from last to first | ||||||
|  |     private static readonly Comparer<ListenerData> SortByAscendingPriority = Comparer<ListenerData>.Create((x, y) => x.Priority.CompareTo(y.Priority)); | ||||||
|  |  | ||||||
|  |     private ILogger _logger = ILogger.Shared; | ||||||
|  |     public ILogger Logger { get => _logger; set => _logger = value ?? ILogger.Shared; } | ||||||
|  |  | ||||||
|  |     private readonly List<ListenerData> listeners = null!; | ||||||
|  |     private readonly List<ListenerData> onceListeners = null!; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked whenever the event is triggered. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called when the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = listeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         listeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked the next time the event is triggered. The callback will be called only once. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called the next time the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddOneTimeListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = onceListeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         onceListeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             if (listeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 listeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveOneTimeListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = 0; i < onceListeners.Count; i++) | ||||||
|  |             if (onceListeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 onceListeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes all listeners that was previously registered by either <see cref="AddListener(EventHandler)"/> or <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public void Clear() { listeners.Clear(); onceListeners.Clear(); } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Triggers the event. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The caller that's triggering this event.</param> | ||||||
|  |     public void Invoke(TSender sender) | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             try { listeners[i].Callback.Invoke(sender); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{listeners[i].Callback.Method.DeclaringType?.FullName}.{listeners[i].Callback.Method.Name}({sender})"; | ||||||
|  |                 EventHelpers.LogInvocationException(listeners[i].Callback.Target ?? sender, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         for (int i = onceListeners.Count - 1; i >= 0; i--) | ||||||
|  |         { | ||||||
|  |             try { onceListeners[i].Callback.Invoke(sender); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{onceListeners[i].Callback.Method.DeclaringType?.FullName}.{onceListeners[i].Callback.Method.Name}({sender})"; | ||||||
|  |                 EventHelpers.LogInvocationException(onceListeners[i].Callback.Target ?? sender, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |             onceListeners.RemoveAt(i); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event(int initialListenerCount = 4, int initialOnceListenerCount = 2) | ||||||
|  |     { | ||||||
|  |         listeners = new(initialListenerCount); | ||||||
|  |         onceListeners = new(initialOnceListenerCount); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event() | ||||||
|  |     { | ||||||
|  |         listeners = new(4); | ||||||
|  |         onceListeners = new(2); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public delegate void EventHandler(TSender sender); | ||||||
|  |     private record struct ListenerData(EventHandler Callback, int Priority); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents an event with sender and argument parameters. | ||||||
|  | /// <para>Example usage:</para>  | ||||||
|  | /// <code> | ||||||
|  | /// public class MyBehaviour : Behaviour, IUpdate | ||||||
|  | /// { | ||||||
|  | ///     public readonly Event<MyBehaviour, MyArguments> MyEvent = new(); | ||||||
|  | ///  | ||||||
|  | ///     private int myInt = 0; | ||||||
|  | ///     private bool myBool = false; | ||||||
|  | ///  | ||||||
|  | ///     public MyBehaviour() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.AddOneTimeListener(OnEventTriggeredOneTime); | ||||||
|  | ///         MyEvent.AddListener(OnEventTriggered); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     public void Update() | ||||||
|  | ///     { | ||||||
|  | ///         MyEvent.Invoke(this, new MyArguments(myInt, myBool)); | ||||||
|  | ///         myInt++; | ||||||
|  | ///         myBool = !myBool; | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private void OnEventTriggered(MyBehaviour sender, MyArguments args) | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"{sender.Id}'s event occurred with MyInt: {args.MyInt} and MyBool {args.MyBool}!"); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     private static void OnEventTriggeredOneTime(MyBehaviour sender, MyArguments args) | ||||||
|  | ///     { | ||||||
|  | ///         Console.WriteLine($"{sender.Id}'s event called once with MyInt: {args.MyInt} and MyBool {args.MyBool}!"); | ||||||
|  | ///     } | ||||||
|  | ///  | ||||||
|  | ///     public readonly record struct MyArguments(int MyInt, bool MyBool); | ||||||
|  | /// } | ||||||
|  | /// </code> | ||||||
|  | /// The output of the example code above would be:  | ||||||
|  | /// <code> | ||||||
|  | /// [Id]'s event occurred with MyInt: 0 and MyBool False! | ||||||
|  | /// [Id]'s event called once with MyInt: 0 and MyBool False! | ||||||
|  | /// [Id]'s event occurred with MyInt: 1 and MyBool True! | ||||||
|  | /// [Id]'s event occurred with MyInt: 2 and MyBool False! | ||||||
|  | /// [Id]'s event occurred with MyInt: 3 and MyBool True! | ||||||
|  | /// ... | ||||||
|  | /// </code> | ||||||
|  | ///  | ||||||
|  | /// </summary> | ||||||
|  | /// <typeparam name="TSender">Sender type</typeparam> | ||||||
|  | public class Event<TSender, TArguments> where TSender : class | ||||||
|  | { | ||||||
|  |     // We use Ascending order because draw calls are running from last to first | ||||||
|  |     private static readonly Comparer<ListenerData> SortByAscendingPriority = Comparer<ListenerData>.Create((x, y) => x.Priority.CompareTo(y.Priority)); | ||||||
|  |  | ||||||
|  |     private ILogger _logger = ILogger.Shared; | ||||||
|  |     public ILogger Logger { get => _logger; set => _logger = value ?? ILogger.Shared; } | ||||||
|  |  | ||||||
|  |     private readonly List<ListenerData> listeners = null!; | ||||||
|  |     private readonly List<ListenerData> onceListeners = null!; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked whenever the event is triggered. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called when the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = listeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         listeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subscribes the callback to be invoked the next time the event is triggered. The callback will be called only once. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback to be called the next time the event is triggered.</param> | ||||||
|  |     /// <param name="priority">Priority of the callback.</param> | ||||||
|  |     public void AddOneTimeListener(EventHandler listener, int priority = 0) | ||||||
|  |     { | ||||||
|  |         ListenerData listenerData = new(listener, priority); | ||||||
|  |  | ||||||
|  |         int insertIndex = onceListeners.BinarySearch(listenerData, SortByAscendingPriority); | ||||||
|  |         if (insertIndex < 0) | ||||||
|  |             insertIndex = ~insertIndex; | ||||||
|  |  | ||||||
|  |         onceListeners.Insert(insertIndex, listenerData); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             if (listeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 listeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes the callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="listener">The callback that was previously registered by <see cref="AddOneTimeListener(EventHandler)"/></param> | ||||||
|  |     public void RemoveOneTimeListener(EventHandler listener) | ||||||
|  |     { | ||||||
|  |         for (int i = 0; i < onceListeners.Count; i++) | ||||||
|  |             if (onceListeners[i].Callback == listener) | ||||||
|  |             { | ||||||
|  |                 onceListeners.RemoveAt(i); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Unsubscribes all listeners that was previously registered by either <see cref="AddListener(EventHandler)"/> or <see cref="AddOneTimeListener(EventHandler)"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public void Clear() { listeners.Clear(); onceListeners.Clear(); } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Triggers the event. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="sender">The caller that's triggering this event.</param> | ||||||
|  |     /// <param name="args">The arguments provided for this event.</param> | ||||||
|  |     public void Invoke(TSender sender, TArguments args) | ||||||
|  |     { | ||||||
|  |         for (int i = listeners.Count - 1; i >= 0; i--) | ||||||
|  |             try { listeners[i].Callback.Invoke(sender, args); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{listeners[i].Callback.Method.DeclaringType?.FullName}.{listeners[i].Callback.Method.Name}({sender}, {args})"; | ||||||
|  |                 EventHelpers.LogInvocationException(listeners[i].Callback.Target ?? sender, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         for (int i = onceListeners.Count - 1; i >= 0; i--) | ||||||
|  |         { | ||||||
|  |             try { onceListeners[i].Callback.Invoke(sender, args); } | ||||||
|  |             catch (Exception exception) | ||||||
|  |             { | ||||||
|  |                 string methodCallRepresentation = $"{onceListeners[i].Callback.Method.DeclaringType?.FullName}.{onceListeners[i].Callback.Method.Name}({sender}, {args})"; | ||||||
|  |                 EventHelpers.LogInvocationException(onceListeners[i].Callback.Target ?? sender, Logger, exception, methodCallRepresentation); | ||||||
|  |             } | ||||||
|  |             onceListeners.RemoveAt(i); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event(int initialListenerCount = 4, int initialOnceListenerCount = 2) | ||||||
|  |     { | ||||||
|  |         listeners = new(initialListenerCount); | ||||||
|  |         onceListeners = new(initialOnceListenerCount); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Event() | ||||||
|  |     { | ||||||
|  |         listeners = new(4); | ||||||
|  |         onceListeners = new(2); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public delegate void EventHandler(TSender sender, TArguments args); | ||||||
|  |     private record struct ListenerData(EventHandler Callback, int Priority); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | internal static class EventHelpers | ||||||
|  | { | ||||||
|  |     public static void LogInvocationException(object sender, ILogger logger, Exception exception, string methodCallRepresentation) | ||||||
|  |     { | ||||||
|  |         logger.LogException(sender, exception); | ||||||
|  |         logger.LogError(sender, $"Unexpected exception on invocation of method {methodCallRepresentation}"); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										102
									
								
								Engine.Core/Helpers/FastList.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								Engine.Core/Helpers/FastList.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | using System.Collections; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class FastList<T> : IList<T>, IReadOnlyList<T>, IEnumerable<T> where T : notnull | ||||||
|  | { | ||||||
|  |     private readonly List<T> items = []; | ||||||
|  |     private readonly Dictionary<T, int> indexMap = []; | ||||||
|  |  | ||||||
|  |     public bool IsReadOnly { get; set; } = false; | ||||||
|  |     public int Count => items.Count; | ||||||
|  |     public T this[int index] | ||||||
|  |     { | ||||||
|  |         get => items[index]; | ||||||
|  |         set | ||||||
|  |         { | ||||||
|  |             if (IsReadOnly) | ||||||
|  |                 throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |             items[index] = value; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Add(T item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         indexMap[item] = items.Count; | ||||||
|  |         items.Add(item); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void RemoveAt(int i) => Remove(items[i], i); | ||||||
|  |     public bool Remove(T item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         if (!indexMap.TryGetValue(item, out int index)) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         Remove(item, index); | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void Remove(T item, int index) | ||||||
|  |     { | ||||||
|  |         int lastIndex = items.Count - 1; | ||||||
|  |         T lastItem = items[lastIndex]; | ||||||
|  |  | ||||||
|  |         items[index] = lastItem; | ||||||
|  |         indexMap[lastItem] = index; | ||||||
|  |  | ||||||
|  |         items.RemoveAt(lastIndex); | ||||||
|  |         indexMap.Remove(item); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Insert(int index, T item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         items.Insert(index, item); | ||||||
|  |  | ||||||
|  |         for (int i = index; i < items.Count; i++) | ||||||
|  |             indexMap[items[i]] = i; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Clear() | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         items.Clear(); | ||||||
|  |         indexMap.Clear(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool Contains(T item) => indexMap.ContainsKey(item); | ||||||
|  |     public int IndexOf(T item) => items.IndexOf(item); | ||||||
|  |     public int BinarySearch(T item, IComparer<T>? comparer = null) => items.BinarySearch(item, comparer); | ||||||
|  |  | ||||||
|  |     public void Sort(IComparer<T> comparer) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         items.Sort(comparer); | ||||||
|  |  | ||||||
|  |         for (int i = 0; i < items.Count; i++) | ||||||
|  |             indexMap[items[i]] = i; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void CopyTo(T[] array, int arrayIndex) => items.CopyTo(array, arrayIndex); | ||||||
|  |  | ||||||
|  |     public IEnumerator<T> GetEnumerator() => items.GetEnumerator(); | ||||||
|  |     IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); | ||||||
|  |  | ||||||
|  |     public FastList() { } | ||||||
|  |     public FastList(int count) { items.Capacity = count; } | ||||||
|  | } | ||||||
							
								
								
									
										172
									
								
								Engine.Core/Helpers/FastListOrdered.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								Engine.Core/Helpers/FastListOrdered.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// TODO This is VEERY experimental, and doesn't work well with the indices access. Use with caution | ||||||
|  | /// </summary> | ||||||
|  | /// <typeparam name="TIndex"></typeparam> | ||||||
|  | /// <typeparam name="TItem"></typeparam> | ||||||
|  | public class FastListOrdered<TIndex, TItem> : IList<TItem>, IReadOnlyList<TItem>, IEnumerable<TItem> where TItem : notnull where TIndex : IComparable | ||||||
|  | { | ||||||
|  |     private readonly SortedDictionary<TIndex, FastList<TItem>> items = null!; | ||||||
|  |  | ||||||
|  |     private readonly Func<TItem, TIndex> getIndexFunc = null!; | ||||||
|  |     private readonly IComparer<TIndex> sortBy = null!; | ||||||
|  |  | ||||||
|  |     private int count = 0; | ||||||
|  |     public int Count => count; | ||||||
|  |  | ||||||
|  |     public bool IsReadOnly { get; set; } = false; | ||||||
|  |  | ||||||
|  |     public TItem this[int index] | ||||||
|  |     { | ||||||
|  |         get { (TIndex tIndex, int i) = GetAt(index); return items[tIndex][i]; } | ||||||
|  |         set | ||||||
|  |         { | ||||||
|  |             if (IsReadOnly) | ||||||
|  |                 throw new System.Data.ReadOnlyException(); | ||||||
|  |             (TIndex tIndex, int i) = GetAt(index); items[tIndex][i] = value; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private (TIndex TIndex, int i) GetAt(Index index) | ||||||
|  |     { | ||||||
|  |         int actualIndex = index.IsFromEnd | ||||||
|  |                 ? count - index.Value | ||||||
|  |                 : index.Value; | ||||||
|  |  | ||||||
|  |         if (actualIndex < 0 || actualIndex >= count) | ||||||
|  |             throw new IndexOutOfRangeException(); | ||||||
|  |  | ||||||
|  |         int leftIndex = actualIndex; | ||||||
|  |         foreach ((TIndex i, FastList<TItem> list) in items) | ||||||
|  |         { | ||||||
|  |             if (leftIndex < list.Count) | ||||||
|  |                 return (i, leftIndex); | ||||||
|  |             leftIndex -= list.Count; | ||||||
|  |         } | ||||||
|  |         throw new IndexOutOfRangeException(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public int IndexOf(TItem item) | ||||||
|  |     { | ||||||
|  |         int indexCounter = 0; | ||||||
|  |         foreach ((TIndex index, FastList<TItem> list) in items) | ||||||
|  |         { | ||||||
|  |             int i = list.IndexOf(item); | ||||||
|  |             if (i != -1) | ||||||
|  |                 return indexCounter + i; | ||||||
|  |             indexCounter += list.Count; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Add(TItem item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         TIndex key = getIndexFunc(item); | ||||||
|  |         if (!items.TryGetValue(key, out FastList<TItem>? list)) | ||||||
|  |             items[key] = list = []; | ||||||
|  |  | ||||||
|  |         list.Add(item); | ||||||
|  |         count++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Insert(int index, TItem item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         TIndex tIndex = getIndexFunc(item); | ||||||
|  |         if (!items.TryGetValue(tIndex, out FastList<TItem>? list)) | ||||||
|  |             items[tIndex] = list = []; | ||||||
|  |  | ||||||
|  |         list.Insert(index, item); | ||||||
|  |         count++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool Remove(TItem item) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         TIndex index = getIndexFunc(item); | ||||||
|  |         if (!items.TryGetValue(index, out FastList<TItem>? list)) | ||||||
|  |             throw new Exceptions.NotFoundException($"Index of '{index}' is not found in the collector"); | ||||||
|  |  | ||||||
|  |         if (!list.Remove(item)) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         count--; | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void RemoveAt(int index) | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         (TIndex tIndex, int i) = GetAt(index); | ||||||
|  |         items[tIndex].RemoveAt(i); | ||||||
|  |         count--; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Clear() | ||||||
|  |     { | ||||||
|  |         if (IsReadOnly) | ||||||
|  |             throw new System.Data.ReadOnlyException(); | ||||||
|  |  | ||||||
|  |         foreach ((TIndex index, FastList<TItem> list) in items) | ||||||
|  |             list.Clear(); | ||||||
|  |  | ||||||
|  |         count = 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool Contains(TItem item) | ||||||
|  |     { | ||||||
|  |         foreach ((TIndex index, FastList<TItem> list) in items) | ||||||
|  |             if (list.Contains(item)) | ||||||
|  |                 return true; | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void CopyTo(TItem[] array, int arrayIndex) | ||||||
|  |     { | ||||||
|  |         int indexCounter = 0; | ||||||
|  |  | ||||||
|  |         foreach ((TIndex index, FastList<TItem> list) in items) | ||||||
|  |         { | ||||||
|  |             list.CopyTo(array, indexCounter); | ||||||
|  |             indexCounter += list.Count; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public IEnumerator<TItem> GetEnumerator() | ||||||
|  |     { | ||||||
|  |         foreach ((TIndex index, FastList<TItem> list) in items) | ||||||
|  |             foreach (TItem item in list) | ||||||
|  |                 yield return item; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); | ||||||
|  |  | ||||||
|  |     public FastListOrdered(Func<TItem, TIndex> getIndexFunc, Comparison<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = Comparer<TIndex>.Create(sortBy); | ||||||
|  |         items = new(this.sortBy); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public FastListOrdered(Func<TItem, TIndex> getIndexFunc, IComparer<TIndex> sortBy) | ||||||
|  |     { | ||||||
|  |         this.getIndexFunc = getIndexFunc; | ||||||
|  |         this.sortBy = sortBy; | ||||||
|  |         items = new(sortBy); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								Engine.Core/Helpers/IPool.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								Engine.Core/Helpers/IPool.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface IPool<T> | ||||||
|  | { | ||||||
|  |     Event<IPool<T>, T> OnRemoved { get; } | ||||||
|  |     Event<IPool<T>, T> OnReturned { get; } | ||||||
|  |  | ||||||
|  |     T Get(); | ||||||
|  |     void Return(T item); | ||||||
|  | } | ||||||
							
								
								
									
										40
									
								
								Engine.Core/Helpers/ListPool.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								Engine.Core/Helpers/ListPool.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class ListPool<T> : IPool<List<T>> | ||||||
|  | { | ||||||
|  |     public Event<IPool<List<T>>, List<T>> OnReturned { get; } = new(); | ||||||
|  |     public Event<IPool<List<T>>, List<T>> OnRemoved { get; } = new(); | ||||||
|  |  | ||||||
|  |     private readonly Func<List<T>> generator = null!; | ||||||
|  |     private readonly Queue<List<T>> queue = new(); | ||||||
|  |  | ||||||
|  |     public List<T> Get() | ||||||
|  |     { | ||||||
|  |         if (!queue.TryDequeue(out List<T>? result)) | ||||||
|  |             result = generator(); | ||||||
|  |  | ||||||
|  |         result.Clear(); | ||||||
|  |         OnRemoved?.Invoke(this, result); | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Return(List<T> list) | ||||||
|  |     { | ||||||
|  |         if (queue.Contains(list)) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         list.Clear(); | ||||||
|  |         queue.Enqueue(list); | ||||||
|  |         OnReturned?.Invoke(this, list); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public ListPool(int initialListCount = 1, int initialListCapacity = 32) | ||||||
|  |     { | ||||||
|  |         generator = () => new(initialListCapacity); | ||||||
|  |         for (int i = 0; i < initialListCount; i++) | ||||||
|  |             queue.Enqueue(generator()); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										41
									
								
								Engine.Core/Helpers/Pool.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								Engine.Core/Helpers/Pool.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class Pool<T> : IPool<T> | ||||||
|  | { | ||||||
|  |     public Event<IPool<T>, T> OnRemoved { get; } = new(); | ||||||
|  |     public Event<IPool<T>, T> OnReturned { get; } = new(); | ||||||
|  |  | ||||||
|  |     private readonly Func<T> generator = null!; | ||||||
|  |     private readonly Queue<T> queue = new(); | ||||||
|  |     private readonly HashSet<T> queuedHashes = []; | ||||||
|  |  | ||||||
|  |     public T Get() | ||||||
|  |     { | ||||||
|  |         if (!queue.TryDequeue(out T? result)) | ||||||
|  |             result = generator(); | ||||||
|  |  | ||||||
|  |         queuedHashes.Remove(result); | ||||||
|  |         OnRemoved?.Invoke(this, result); | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Return(T item) | ||||||
|  |     { | ||||||
|  |         if (queuedHashes.Contains(item)) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         queue.Enqueue(item); | ||||||
|  |         queuedHashes.Add(item); | ||||||
|  |         OnReturned?.Invoke(this, item); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Pool(Func<T> generator, int initialCapacity = 1) | ||||||
|  |     { | ||||||
|  |         this.generator = generator; | ||||||
|  |         for (int i = 0; i < initialCapacity; i++) | ||||||
|  |             queue.Enqueue(generator()); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										7
									
								
								Engine.Core/Helpers/Progression/IProgressionTracker.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								Engine.Core/Helpers/Progression/IProgressionTracker.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface IProgressionTracker : IReadOnlyProgressionTracker | ||||||
|  | { | ||||||
|  |     void Set(float progression, string status); | ||||||
|  |     void Reset(); | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public interface IReadOnlyProgressionTracker | ||||||
|  | { | ||||||
|  |     Event<IReadOnlyProgressionTracker, ProgressionUpdatedArguments> OnUpdated { get; } | ||||||
|  |     Event<IReadOnlyProgressionTracker> OnEnded { get; } | ||||||
|  |  | ||||||
|  |     float Progression { get; } | ||||||
|  |     string Status { get; } | ||||||
|  |  | ||||||
|  |     readonly record struct ProgressionUpdatedArguments(float PreviousProgression, string PreviousStatus); | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								Engine.Core/Helpers/Progression/ProgressionTracker.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								Engine.Core/Helpers/Progression/ProgressionTracker.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public class ProgressionTracker : IProgressionTracker | ||||||
|  | { | ||||||
|  |     public Event<IReadOnlyProgressionTracker, IReadOnlyProgressionTracker.ProgressionUpdatedArguments> OnUpdated { get; } = new(); | ||||||
|  |     public Event<IReadOnlyProgressionTracker> OnEnded { get; } = new(); | ||||||
|  |  | ||||||
|  |     public float Progression { get; private set; } = 0f; | ||||||
|  |     public string Status { get; private set; } = "Default"; | ||||||
|  |  | ||||||
|  |     void IProgressionTracker.Set(float progression, string status) | ||||||
|  |     { | ||||||
|  |         if (Progression >= 1f) | ||||||
|  |             return; | ||||||
|  |  | ||||||
|  |         float previousProgression = Progression; | ||||||
|  |         string previousStatus = Status; | ||||||
|  |  | ||||||
|  |         Progression = progression.Clamp(Progression, 1f); | ||||||
|  |         Status = status; | ||||||
|  |  | ||||||
|  |         OnUpdated?.Invoke(this, new(previousProgression, previousStatus)); | ||||||
|  |  | ||||||
|  |         if (progression >= 1f) | ||||||
|  |             OnEnded?.Invoke(this); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void IProgressionTracker.Reset() | ||||||
|  |     { | ||||||
|  |         Progression = 0f; | ||||||
|  |         Status = "Default"; | ||||||
|  |  | ||||||
|  |         OnUpdated.Clear(); | ||||||
|  |         OnEnded.Clear(); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								Engine.Core/Helpers/Progression/ProgressiveTask.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Engine.Core/Helpers/Progression/ProgressiveTask.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public record struct ProgressiveTask<T>(IReadOnlyProgressionTracker ProgressionTracker, Task<T> Task) | ||||||
|  | { | ||||||
|  |     public static implicit operator (IReadOnlyProgressionTracker progressionTracker, Task<T> task)(ProgressiveTask<T> value) => (value.ProgressionTracker, value.Task); | ||||||
|  |     public static implicit operator ProgressiveTask<T>((IReadOnlyProgressionTracker progressionTracker, Task<T> task) value) => new(value.progressionTracker, value.task); | ||||||
|  | } | ||||||
| @@ -1,33 +1,92 @@ | |||||||
| using System; | using System; | ||||||
| using System.Numerics; | using System.Numerics; | ||||||
|  |  | ||||||
| namespace Syntriax.Engine.Core; | namespace Engine.Core; | ||||||
|  |  | ||||||
| public static class Math | public static class Math | ||||||
| {/// <summary> | { | ||||||
|  /// The value of Pi (π), a mathematical constant approximately equal to 3.14159. |  | ||||||
|  /// </summary> |  | ||||||
|     public const float PI = 3.1415926535897932f; |  | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The value of Tau (τ), a mathematical constant equal to 2π, approximately equal to 6.28319. |     /// The value of Pi (π). | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public const float Tau = 2f * PI; |     public const float Pi = 3.1415926535897932f; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The base of the natural logarithm, approximately equal to 2.71828. |     /// The value of Tau (τ), mathematical constant equal to 2π. | ||||||
|  |     /// </summary> | ||||||
|  |     public const float Tau = 2f * Pi; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The base of the natural logarithm. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public const float E = 2.718281828459045f; |     public const float E = 2.718281828459045f; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The conversion factor from radians to degrees. |     /// The conversion factor from radians to degrees. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public const float RadianToDegree = 180f / PI; |     public const float RadianToDegree = 180f / Pi; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// The conversion factor from degrees to radians. |     /// The conversion factor from degrees to radians. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public const float DegreeToRadian = PI / 180f; |     public const float DegreeToRadian = Pi / 180f; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets one minus of given <see cref="T"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="value">The value <see cref="T"/>.</param> | ||||||
|  |     /// <returns>One minus of given <see cref="T"/>.</returns> | ||||||
|  |     public static T OneMinus<T>(T value) where T : INumber<T> => T.One - value; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Adds two <see cref="T"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="left">The first <see cref="T"/>.</param> | ||||||
|  |     /// <param name="value">The second <see cref="T"/>.</param> | ||||||
|  |     /// <returns>The sum of the two <see cref="T"/>s.</returns> | ||||||
|  |     public static T Add<T>(T left, T value) where T : INumber<T> => left + value; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Subtracts one <see cref="T"/> from another. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="left">The <see cref="T"/> to subtract from.</param> | ||||||
|  |     /// <param name="value">The <see cref="T"/> to subtract.</param> | ||||||
|  |     /// <returns>The result of subtracting the second <see cref="T"/> from the first.</returns> | ||||||
|  |     public static T Subtract<T>(T left, T value) where T : INumber<T> => left - value; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Multiplies a <see cref="T"/> by a scalar value. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="left">The <see cref="T"/>.</param> | ||||||
|  |     /// <param name="multiplier">The scalar value.</param> | ||||||
|  |     /// <returns>The result of multiplying the <see cref="T"/> by the scalar value.</returns> | ||||||
|  |     public static T Multiply<T>(T left, T multiplier) where T : INumber<T> => left * multiplier; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Divides a <see cref="T"/> by a scalar value. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="left">The <see cref="T"/>.</param> | ||||||
|  |     /// <param name="divider">The scalar value.</param> | ||||||
|  |     /// <returns>The result of dividing the <see cref="T"/> by the scalar value.</returns> | ||||||
|  |     public static T Divide<T>(T left, T divider) where T : INumber<T> => left / divider; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Returns the true mathematical modulus of a <see cref="T"/> value.  | ||||||
|  |     /// Unlike the remainder operator (%), this result is always non-negative, | ||||||
|  |     /// even when the <paramref name="value"/> operand is negative. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">A numeric type that implements <see cref="INumber{T}"/>.</typeparam> | ||||||
|  |     /// <param name="value">The dividend <see cref="T"/> value.</param> | ||||||
|  |     /// <param name="modulus">The modulus <see cref="T"/> value (must be non-zero).</param> | ||||||
|  |     /// <returns> | ||||||
|  |     /// The non-negative remainder of <paramref name="value"/> divided by <paramref name="modulus"/>. | ||||||
|  |     /// </returns> | ||||||
|  |     public static T Mod<T>(T value, T modulus) where T : INumber<T> | ||||||
|  |     { | ||||||
|  |         T result = value % modulus; | ||||||
|  |         if (result < T.Zero) | ||||||
|  |             result += modulus; | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Returns the absolute value of a number. |     /// Returns the absolute value of a number. | ||||||
| @@ -37,6 +96,27 @@ public static class Math | |||||||
|     /// <returns>The absolute value of <paramref name="x"/>.</returns> |     /// <returns>The absolute value of <paramref name="x"/>.</returns> | ||||||
|     public static T Abs<T>(T x) where T : INumber<T> => x > T.Zero ? x : -x; |     public static T Abs<T>(T x) where T : INumber<T> => x > T.Zero ? x : -x; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Returns the cosine of a number. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="x">The number.</param> | ||||||
|  |     /// <returns>The cosine of <paramref name="x"/>.</returns> | ||||||
|  |     public static float Cos(float x) => MathF.Cos(x); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Returns the sine of a number. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="x">The number.</param> | ||||||
|  |     /// <returns>The sine of <paramref name="x"/>.</returns> | ||||||
|  |     public static float Sin(float x) => MathF.Sin(x); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Returns the tangent of a number. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="x">The angle, in radians.</param> | ||||||
|  |     /// <returns>The tangent of <paramref name="x"/>.</returns> | ||||||
|  |     public static float Tan(float x) => MathF.Tan(x); | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Returns the arccosine of a number. |     /// Returns the arccosine of a number. | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @@ -51,6 +131,13 @@ public static class Math | |||||||
|     /// <returns>The arcsine of <paramref name="x"/>.</returns> |     /// <returns>The arcsine of <paramref name="x"/>.</returns> | ||||||
|     public static float Asin(float x) => MathF.Asin(x); |     public static float Asin(float x) => MathF.Asin(x); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Returns the angle whose tangent is the specified number. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="x">The tangent value.</param> | ||||||
|  |     /// <returns>The angle, in radians.</returns> | ||||||
|  |     public static float Atan(float x) => MathF.Atan(x); | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Returns the angle whose tangent is the quotient of two specified numbers. |     /// Returns the angle whose tangent is the quotient of two specified numbers. | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @@ -74,7 +161,7 @@ public static class Math | |||||||
|     /// <param name="min">The minimum value.</param> |     /// <param name="min">The minimum value.</param> | ||||||
|     /// <param name="max">The maximum value.</param> |     /// <param name="max">The maximum value.</param> | ||||||
|     /// <returns>The clamped value.</returns> |     /// <returns>The clamped value.</returns> | ||||||
|     public static T Clamp<T>(this T x, T min, T max) where T : INumber<T> => (x < min) ? min : (x > max) ? max : x; |     public static T Clamp<T>(T x, T min, T max) where T : INumber<T> => (x < min) ? min : (x > max) ? max : x; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Returns the smallest integral value that is greater than or equal to the specified number. |     /// Returns the smallest integral value that is greater than or equal to the specified number. | ||||||
| @@ -157,13 +244,44 @@ public static class Math | |||||||
|     public static float Pow(float x, float y) => MathF.Pow(x, y); |     public static float Pow(float x, float y) => MathF.Pow(x, y); | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Rounds a number to a specified number of fractional digits. |     /// Performs linear interpolation between two specified values. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <typeparam name="T">The type of the values, which must implement <see cref="IFloatingPoint{T}"/>.</typeparam> | ||||||
|  |     /// <param name="x">The starting value of the interpolation.</param> | ||||||
|  |     /// <param name="y">The ending value of the interpolation.</param> | ||||||
|  |     /// <param name="t">The interpolation factor, typically in the range [0, 1].</param> | ||||||
|  |     /// <returns>A value that represents the linear interpolation between <paramref name="x"/> and <paramref name="y"/>.</returns> | ||||||
|  |     public static T Lerp<T>(T x, T y, T t) where T : IFloatingPoint<T> => x + (y - x) * t; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Rounds a number to the closest integer. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="x">The number to round.</param> |     /// <param name="x">The number to round.</param> | ||||||
|     /// <param name="digits">The number of fractional digits in the return value.</param> |     /// <param name="roundMode">Specification for how to round <paramref name="x"/> if it is midway between two other numbers.</param> | ||||||
|     /// <param name="mode">Specification for how to round <paramref name="x"/> if it is midway between two other numbers.</param> |     /// <returns>The number <paramref name="x"/> rounded to the closest integer.</returns> | ||||||
|     /// <returns>The number <paramref name="x"/> rounded to <paramref name="digits"/> fractional digits.</returns> |     public static float Round(float x, RoundMode roundMode) => RoundToInt(x, roundMode); | ||||||
|     public static float Round(float x, int digits, MidpointRounding mode) => MathF.Round(x, digits, mode); |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Rounds a number to the closest integer. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="x">The number to round.</param> | ||||||
|  |     /// <param name="roundMode">Specification for how to round <paramref name="x"/> if it's midway between two numbers</param> | ||||||
|  |     /// <returns>The number <paramref name="x"/> rounded to the closest integer.</returns> | ||||||
|  |     public static int RoundToInt(float x, RoundMode roundMode = RoundMode.Ceil) | ||||||
|  |     { | ||||||
|  |         float remainder = x.Mod(1f); | ||||||
|  |  | ||||||
|  |         if (remainder == .5f) | ||||||
|  |             if (roundMode == RoundMode.Floor) | ||||||
|  |                 return (int)x; | ||||||
|  |             else | ||||||
|  |                 return (int)(x + .5f); | ||||||
|  |  | ||||||
|  |         if (x < 0f) | ||||||
|  |             return (int)(x - .5f); | ||||||
|  |         return (int)(x + .5f); | ||||||
|  |     } | ||||||
|  |     public enum RoundMode { Ceil, Floor }; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Returns the square of a number. |     /// Returns the square of a number. | ||||||
| @@ -186,5 +304,4 @@ public static class Math | |||||||
|     /// <param name="x">The number.</param> |     /// <param name="x">The number.</param> | ||||||
|     /// <returns>The integral part of <paramref name="x"/>.</returns> |     /// <returns>The integral part of <paramref name="x"/>.</returns> | ||||||
|     public static float Truncate(float x) => MathF.Truncate(x); |     public static float Truncate(float x) => MathF.Truncate(x); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										97
									
								
								Engine.Core/MathExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								Engine.Core/MathExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | |||||||
|  | using System; | ||||||
|  | using System.Numerics; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | public static class MathExtensions | ||||||
|  | { | ||||||
|  |     /// <inheritdoc cref="Math.OneMinus{T}(T)" /> | ||||||
|  |     public static T OneMinus<T>(this T value) where T : INumber<T> => Math.OneMinus(value); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Add{T}(T, T)" /> | ||||||
|  |     public static T Add<T>(this T left, T value) where T : INumber<T> => Math.Add(left, value); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Subtract{T}(T, T)" /> | ||||||
|  |     public static T Subtract<T>(this T left, T value) where T : INumber<T> => Math.Subtract(left, value); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Multiply{T}(T, T)" /> | ||||||
|  |     public static T Multiply<T>(this T left, T multiplier) where T : INumber<T> => Math.Multiply(left, multiplier); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Divide{T}(T, T)" /> | ||||||
|  |     public static T Divide<T>(this T left, T divider) where T : INumber<T> => Math.Divide(left, divider); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Mod{T}(T, T)" /> | ||||||
|  |     public static T Mod<T>(this T value, T modulus) where T : INumber<T> => Math.Mod(value, modulus); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Abs{T}(T)" /> | ||||||
|  |     public static T Abs<T>(this T x) where T : INumber<T> => Math.Abs(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Cos(float)" /> | ||||||
|  |     public static float Cos(this float x) => Math.Cos(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Sin(float)" /> | ||||||
|  |     public static float Sin(this float x) => Math.Sin(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Acos(float)" /> | ||||||
|  |     public static float Acos(this float x) => Math.Acos(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Asin(float)" /> | ||||||
|  |     public static float Asin(this float x) => Math.Asin(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Atan2(float, float)" /> | ||||||
|  |     public static float Atan2(this float y, float x) => Math.Atan2(y, x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Atanh(float)" /> | ||||||
|  |     public static float Atanh(this float x) => Math.Atanh(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Clamp{T}(T, T, T)" /> | ||||||
|  |     public static T Clamp<T>(this T x, T min, T max) where T : INumber<T> => Math.Clamp(x, min, max); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Ceiling(float)" /> | ||||||
|  |     public static float Ceiling(this float x) => Math.Ceiling(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.CopySign(float, float)" /> | ||||||
|  |     public static float CopySign(this float x, float y) => Math.CopySign(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Floor(float)" /> | ||||||
|  |     public static float Floor(this float x) => Math.Floor(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.IEEERemainder(float, float)" /> | ||||||
|  |     public static float IEEERemainder(this float x, float y) => Math.IEEERemainder(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Log(float, float)" /> | ||||||
|  |     public static float Log(this float x, float y) => Math.Log(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Max{T}(T, T)" /> | ||||||
|  |     public static T Max<T>(this T x, T y) where T : INumber<T> => Math.Max(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.AbsMax{T}(T, T)" /> | ||||||
|  |     public static T AbsMax<T>(this T x, T y) where T : INumber<T> => Math.AbsMax(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Min{T}(T, T)" /> | ||||||
|  |     public static T Min<T>(this T x, T y) where T : INumber<T> => Math.Min(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.AbsMin{T}(T, T)" /> | ||||||
|  |     public static T AbsMin<T>(this T x, T y) where T : INumber<T> => Math.AbsMin(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Pow(float, float)" /> | ||||||
|  |     public static float Pow(this float x, float y) => Math.Pow(x, y); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Lerp{T}(T, T, T)" /> | ||||||
|  |     public static T Lerp<T>(this T x, T y, T t) where T : IFloatingPoint<T> => Math.Lerp(x, y, t); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Round(float, int, MidpointRounding)" /> | ||||||
|  |     public static float Round(this float x, Math.RoundMode mode) => Math.Round(x, mode); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.RoundToInt(float, Math.RoundMode)" /> | ||||||
|  |     public static int RoundToInt(this float x, Math.RoundMode roundMode = Math.RoundMode.Ceil) => Math.RoundToInt(x, roundMode); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Sqr{T}(T)" /> | ||||||
|  |     public static T Sqr<T>(this T x) where T : INumber<T> => Math.Sqr(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Sqrt(float)" /> | ||||||
|  |     public static float Sqrt(this float x) => Math.Sqrt(x); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="Math.Truncate(float)" /> | ||||||
|  |     public static float Truncate(this float x) => Math.Truncate(x); | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								Engine.Core/Preserver.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								Engine.Core/Preserver.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | namespace Engine.Core | ||||||
|  | { | ||||||
|  |     // This is pretty much so the assembly gets loaded automatically because  | ||||||
|  |     // the builds include the assembly but sometimes doesn't link load it at startup. | ||||||
|  |     // I will hopefully one day fix it and remove this.  | ||||||
|  |     public static class Preserver | ||||||
|  |     { | ||||||
|  |         public static void Preserve() { } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										114
									
								
								Engine.Core/Primitives/AABB2D.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								Engine.Core/Primitives/AABB2D.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  |  | ||||||
|  | namespace Engine.Core; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Represents an Axis-Aligned Bounding Box (AABB) in 2D space. | ||||||
|  | /// </summary> | ||||||
|  | /// <param name="lowerBoundary">The lower boundary of the <see cref="AABB2D"/>.</param> | ||||||
|  | /// <param name="upperBoundary">The upper boundary of the <see cref="AABB2D"/>.</param> | ||||||
|  | /// <remarks> | ||||||
|  | /// Initializes a new instance of the <see cref="AABB2D"/> struct with the specified lower and upper boundaries. | ||||||
|  | /// </remarks> | ||||||
|  | [System.Diagnostics.DebuggerDisplay("LowerBoundary: {LowerBoundary.ToString(), nq}, UpperBoundary: {UpperBoundary.ToString(), nq}")] | ||||||
|  | public readonly struct AABB2D(Vector2D lowerBoundary, Vector2D upperBoundary) : IEquatable<AABB2D> | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// The lower boundary of the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public readonly Vector2D LowerBoundary = lowerBoundary; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// The upper boundary of the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public readonly Vector2D UpperBoundary = upperBoundary; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets the center point of the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public readonly Vector2D Center => (LowerBoundary + UpperBoundary) * .5f; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets the size of the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public readonly Vector2D Size => LowerBoundary.FromTo(UpperBoundary).Abs(); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets half the size of the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     public readonly Vector2D SizeHalf => Size * .5f; | ||||||
|  |  | ||||||
|  |     public static bool operator ==(AABB2D left, AABB2D right) => left.UpperBoundary == right.UpperBoundary && left.LowerBoundary == right.LowerBoundary; | ||||||
|  |     public static bool operator !=(AABB2D left, AABB2D right) => left.UpperBoundary != right.UpperBoundary || left.LowerBoundary != right.LowerBoundary; | ||||||
|  |  | ||||||
|  |     public static implicit operator AABB2D(Circle circle) => new(circle.Center - new Vector2D(circle.Radius, circle.Radius), circle.Center + new Vector2D(circle.Radius, circle.Radius)); | ||||||
|  |     public static implicit operator AABB2D(Shape2D shape) => FromVectors(shape.Vertices); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Creates an <see cref="AABB2D"/> from a collection of <see cref="Vector2D"/>s. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="vectors">The collection of <see cref="Vector2D"/>s.</param> | ||||||
|  |     /// <returns>An <see cref="AABB2D"/> that bounds all the <see cref="Vector2D"/>s.</returns> | ||||||
|  |     public static AABB2D FromVectors(IEnumerable<Vector2D> vectors) | ||||||
|  |     { | ||||||
|  |         int counter = 0; | ||||||
|  |  | ||||||
|  |         Vector2D lowerBoundary = new(float.MaxValue, float.MaxValue); | ||||||
|  |         Vector2D upperBoundary = new(float.MinValue, float.MinValue); | ||||||
|  |  | ||||||
|  |         foreach (Vector2D vector in vectors) | ||||||
|  |         { | ||||||
|  |             lowerBoundary = Vector2D.Min(lowerBoundary, vector); | ||||||
|  |             upperBoundary = Vector2D.Max(upperBoundary, vector); | ||||||
|  |             counter++; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (counter < 2) | ||||||
|  |             throw new ArgumentException($"Parameter {nameof(vectors)} must have at least 2 items."); | ||||||
|  |  | ||||||
|  |         return new(lowerBoundary, upperBoundary); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Checks if two <see cref="AABB2D"/>s are approximately equal. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="left">The first <see cref="AABB2D"/>.</param> | ||||||
|  |     /// <param name="right">The second <see cref="AABB2D"/>.</param> | ||||||
|  |     /// <param name="epsilon">The epsilon range.</param> | ||||||
|  |     /// <returns><see cref="true"/> if the <see cref="AABB2D"/>s are approximately equal; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public static bool ApproximatelyEquals(AABB2D left, AABB2D right, float epsilon = float.Epsilon) | ||||||
|  |         => left.LowerBoundary.ApproximatelyEquals(right.LowerBoundary, epsilon) && left.UpperBoundary.ApproximatelyEquals(right.UpperBoundary, epsilon); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Determines whether the specified object is equal to the current <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="obj">The object to compare with the current <see cref="AABB2D"/>.</param> | ||||||
|  |     /// <returns><see cref="true"/> if the specified object is equal to the current <see cref="AABB2D"/>; otherwise, <see cref="false"/>.</returns> | ||||||
|  |     public override bool Equals(object? obj) => obj is AABB2D aabb && this == aabb; | ||||||
|  |     public bool Equals(AABB2D other) => this == other; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Generates a hash code for the <see cref="AABB2D"/>. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <returns>A hash code for the <see cref="AABB2D"/>.</returns> | ||||||
|  |     public override int GetHashCode() => System.HashCode.Combine(LowerBoundary, UpperBoundary); | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// Converts the <see cref="AABB2D"/> to its string representation. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <returns>A string representation of the <see cref="AABB2D"/>.</returns> | ||||||
|  |     public override string ToString() => $"{nameof(AABB2D)}({LowerBoundary}, {UpperBoundary})"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Provides extension methods for the <see cref="AABB2D"/> struct. | ||||||
|  | /// </summary> | ||||||
|  | public static class AABBExtensions | ||||||
|  | { | ||||||
|  |     /// <inheritdoc cref="AABB2D.FromVectors" /> | ||||||
|  |     public static AABB2D ToAABB(this IEnumerable<Vector2D> vectors) => AABB2D.FromVectors(vectors); | ||||||
|  |  | ||||||
|  |     /// <inheritdoc cref="AABB2D.ApproximatelyEquals" /> | ||||||
|  |     public static bool ApproximatelyEquals(this AABB2D left, AABB2D right, float epsilon = float.Epsilon) => AABB2D.ApproximatelyEquals(left, right, epsilon); | ||||||
|  | } | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user