Compare commits
	
		
			495 Commits
		
	
	
		
			patch-tabc
			...
			16646efb3c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 16646efb3c | |||
| ec1e8483b9 | |||
| d39b527c3e | |||
| af9e6d9118 | |||
| da736d8ad9 | |||
| 57da3ee6ca | |||
|   | 296b31bd56 | ||
|   | 8f6768ae00 | ||
|   | 70603d5413 | ||
|   | 2e12caad03 | ||
|   | f9ce9fad28 | ||
|   | 69b476fcbc | ||
|   | d37a430f5a | ||
|   | 4c02676b7a | ||
|   | a2558484f5 | ||
|   | 97c6167272 | ||
|   | c2162eeddb | ||
|   | 0124b66918 | ||
|   | 7be06f141d | ||
|   | 01048b4fca | ||
|   | d2a317eee2 | ||
|   | 0be632a4d6 | ||
|   | f27f4fbca9 | ||
|   | e62fc6c291 | ||
|   | 8e99a4c5bf | ||
|   | df53053677 | ||
|   | 4cd9a17a40 | ||
|   | e9558ab370 | ||
|   | 3a5c731826 | ||
|   | c13e6df67e | ||
|   | 704e866413 | ||
|   | aea5870ac8 | ||
|   | f1a4a42d51 | ||
|   | a485d9f314 | ||
|   | 131125c7d2 | ||
|   | 7fcc62067b | ||
|   | 7c7cb3de0f | ||
|   | 88436c44a6 | ||
|   | bdd32d5a58 | ||
|   | 5e59b6dc85 | ||
|   | bccce74c3c | ||
|   | 0e9e0b58d2 | ||
|   | 8cb49bc10a | ||
|   | 5b05934fe8 | ||
|   | 97f65726d2 | ||
|   | e3ab8ef15f | ||
|   | d5bcabdc60 | ||
|   | 3cd530f007 | ||
|   | 2b9808cd13 | ||
|   | 70fa02f3a4 | ||
|   | 4ebc3c96b2 | ||
|   | dd1531e28d | ||
|   | 5709a65785 | ||
|   | 5348aad094 | ||
|   | b60c1bdb37 | ||
|   | 93508d5083 | ||
|   | 1f159f8eaa | ||
|   | 31f7ef8c54 | ||
|   | 244412405a | ||
|   | 9cd0d3289f | ||
|   | 7cde213e63 | ||
|   | aa44ebe770 | ||
|   | 5dad41034b | ||
|   | cd1ceb4c31 | ||
|   | 23ba5141f1 | ||
|   | 415ac8c81e | ||
|   | 41e49dad6b | ||
|   | bec329352d | ||
|   | 89e66ed648 | ||
|   | d8f9d81b30 | ||
|   | 363003d8c7 | ||
|   | a696bb0e9f | ||
|   | 68f4f6bd40 | ||
|   | 442ff808f3 | ||
|   | fbbcc454d5 | ||
|   | f8de305477 | ||
|   | f1f5be18f9 | ||
|   | e05560976b | ||
|   | 8bff00f15b | ||
|   | 4d37c2488e | ||
|   | 75456b2c0a | ||
|   | 53365e4b18 | ||
|   | 69e4872f40 | ||
|   | a336efb8fa | ||
|   | ae2fc30b7b | ||
|   | 2bafb70581 | ||
|   | 617c2728a2 | ||
|   | 89069a362d | ||
|   | 1279cca971 | ||
|   | 26433bf021 | ||
|   | d81040cd6f | ||
|   | d7538df91b | ||
|   | 2516de6586 | ||
|   | 1da3a8c240 | ||
|   | f6151dce56 | ||
|   | 9667743735 | ||
|   | 7587f03306 | ||
|   | 77b81f2612 | ||
|   | fa6d47732d | ||
|   | 252e7b0027 | ||
|   | c820b3a062 | ||
|   | 687c302610 | ||
|   | 4b0262312e | ||
|   | 6f13c2d6b6 | ||
|   | cd186999e5 | ||
|   | e3c7fd8cc5 | ||
|   | 9476ffccdb | ||
|   | 47f8c29a7c | ||
|   | 458246505f | ||
|   | 362bd0f4c4 | ||
|   | 598d73e6f0 | ||
|   | f797bd488f | ||
|   | 0d153feee7 | ||
|   | 2a78233cc2 | ||
|   | 591e18753d | ||
|   | 556a15a6f8 | ||
|   | 774a6fd68c | ||
|   | 0070421549 | ||
|   | 4dad940a2f | ||
|   | ed4a80eb0b | ||
|   | dd2033bf1a | ||
|   | cceebdad2a | ||
|   | 05bdf5d3c1 | ||
|   | c3e4a6ef5b | ||
|   | 2337acfcc1 | ||
|   | 69861e5334 | ||
|   | 22aa6f5faf | ||
|   | 508c2f7ac3 | ||
|   | 9dd5fb626d | ||
|   | b5ae0196fc | ||
|   | 80bb237289 | ||
|   | 4fded9828f | ||
|   | 6b22690971 | ||
|   | 60a3bf082f | ||
|   | 0aa2871b26 | ||
|   | 1265a9927b | ||
|   | d99570214a | ||
|   | 15bd33b25b | ||
|   | 7340f1a035 | ||
|   | 8a80435e64 | ||
|   | 20a71b06a9 | ||
|   | b376f61578 | ||
|   | 373dab05ad | ||
|   | f6b40b1186 | ||
|   | 81b118a8ba | ||
|   | 7a42f12716 | ||
|   | 4886c4be01 | ||
|   | 7338d0f444 | ||
|   | 8212e10c7c | ||
|   | 2593130b3e | ||
|   | 6ea49962c5 | ||
|   | 672db9fe47 | ||
|   | 2bacf6572b | ||
|   | 9813e46e66 | ||
|   | 01a5f36012 | ||
|   | f0a30c43cd | ||
|   | acb85e30fa | ||
|   | 9437cedc48 | ||
|   | a89cf5f36d | ||
|   | b309e4ac50 | ||
|   | 477ea5983c | ||
|   | eca6090f1e | ||
|   | 8f8c270f3b | ||
|   | 84ac7ab944 | ||
|   | 5fbcc6b119 | ||
|   | 79f85a2ce2 | ||
|   | d32eedd333 | ||
|   | e1d4b6adc7 | ||
|   | 534148763f | ||
|   | cd56fb32c2 | ||
|   | 6b612302e1 | ||
|   | e49759025f | ||
|   | c310e3339f | ||
|   | b64615e298 | ||
|   | 45d2f44003 | ||
|   | a57adcce00 | ||
|   | 8b195d1d21 | ||
|   | cda4537fba | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | df413f62db | ||
|   | 8a88ce464e | ||
|   | 006a14a75c | ||
|   | 07df657f3c | ||
|   | b8b373a53e | ||
|   | e7e0b97cff | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52ab21b1ff | ||
|   | 8e8a635361 | ||
|   | 18eae8a1a6 | ||
|   | 6e1751733f | ||
|   | 6335af840b | ||
|   | 336333acb1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d110f6629b | ||
|   | 6f70b15e2e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b30499e2b6 | ||
|   | 5079181c28 | ||
|   | ee02d98cb2 | ||
|   | c7ff3b8a14 | ||
|   | de60af0d7b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9218a7aa7 | ||
|   | 67c65e0464 | ||
|   | 1be25b6c74 | ||
|   | 8525b44961 | ||
|   | 1fca510a08 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3384185285 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3075d2c19d | ||
|   | bc528d5d98 | ||
|   | 25cf8d682b | ||
|   | 17e23d5c3f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d6c5197cb9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dd96f0f878 | ||
|   | 8a9501ffe4 | ||
|   | 5e25c63c5a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd963501ec | ||
|   | da795a7094 | ||
|   | 84d0ea73fa | ||
|   | 0851e39197 | ||
|   | 86e6fdf8a2 | ||
|   | 6ab0f5eba7 | ||
|   | f224787222 | ||
|   | 82684c7b6b | ||
|   | c2f73d32b8 | ||
|   | e642b9dde1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | db623d10c5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 61bb9f5b93 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9551b45328 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dc680b87eb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 156eda78c6 | ||
|   | 31be68af51 | ||
|   | ffa011c7b1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22536c11bd | ||
|   | 2394e204fa | ||
|   | 1b88a84710 | ||
|   | 7606d4437b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e9a7e45c4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f6c5332c1a | ||
|   | d0fa62d424 | ||
|   | 464ed0184c | ||
|   | eda268b481 | ||
|   | 3e1007527c | ||
|   | b52b14696c | ||
|   | 94d5b0d03c | ||
|   | c3f228f626 | ||
|   | 02c5c1ee76 | ||
|   | c69acf728c | ||
|   | a1cd694363 | ||
|   | 3e2bc8e2d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad7163d2d6 | ||
|   | 19918c694f | ||
|   | 21c8f2815a | ||
|   | 737d545fb6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b23a51825e | ||
|   | 708c5b6254 | ||
|   | f5af11193c | ||
|   | b711e4033f | ||
|   | 3deaaadc3a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 51b9a6b0b8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a807731a5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 772ad9951f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2431c40a5c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8144ae8d7b | ||
|   | 0757c39a6f | ||
|   | 231024ba42 | ||
|   | 8ce7a7f8b6 | ||
|   | e1462ccdd1 | ||
|   | 70f346c1dc | ||
|   | 197bf13a28 | ||
|   | 0925c06f9b | ||
|   | 16298a75f2 | ||
|   | 39b10c0b16 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd8d114992 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 30e12c6fe0 | ||
|   | bd009ca52d | ||
|   | 65d8edf62d | ||
|   | f5157f12a4 | ||
|   | df20effacc | ||
|   | c92581d0dc | ||
|   | e442c3da5c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f903c54d55 | ||
|   | 0d45378986 | ||
|   | 0f5f09b6c5 | ||
|   | e5c80d0044 | ||
|   | 9cdb2ba3ea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d0e5cf7ce5 | ||
|   | c8568764f6 | ||
|   | a7dbbc2f0a | ||
|   | 68b2df2b1e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1ef4d27dbe | ||
|   | 94a1fb5117 | ||
|   | 78aef86a8f | ||
|   | b34cfcde5a | ||
|   | 86e079a4b1 | ||
|   | 1c42c34081 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fed646d18b | ||
|   | 653f1691d7 | ||
|   | 3cb7a12738 | ||
|   | f3397b3003 | ||
|   | 497c6879e0 | ||
|   | 7b27dfaf5e | ||
|   | f9b75c4a3a | ||
|   | 0509303fd3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f486a251f3 | ||
|   | 5a1e342e0d | ||
|   | d9bbdc3281 | ||
|   | cfe00fa47c | ||
|   | d68ebd1eaf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a7cd79eb41 | ||
|   | 9e83ee6f0c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7c81d91740 | ||
|   | 5b126b7f4d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9fe7d21f4b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 94ea0271ba | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3af672d2f2 | ||
|   | 0dd7b98428 | ||
|   | a793692a2c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 23fb838227 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2d6d89d668 | ||
|   | 0199cb90ff | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 958cef5084 | ||
|   | 9f5ace9025 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a6e2631bf | ||
|   | c7adcf9fdf | ||
|   | da3616e636 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b371fe67a5 | ||
|   | 6324c7d527 | ||
|   | 6263fe283b | ||
|   | 9a7617f9b8 | ||
|   | 9a71358dfa | ||
|   | a96a2e80a1 | ||
|   | 68200133b6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 188d502c59 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 84ac683c1d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b418c94215 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 38e593a698 | ||
|   | 38028e8e90 | ||
|   | 3db27052a1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e24f9223df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9e5ed82c99 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 606fa278c4 | ||
|   | 7dd549ff1e | ||
|   | 3c12b04c98 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5545850f9d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f909b44d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ff155ebbb4 | ||
|   | a0a4fa0e56 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1b76a26691 | ||
|   | bd7bd2739a | ||
|   | a7ad407f4b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 931ff0fde6 | ||
|   | dfd847f705 | ||
|   | a1fee720b9 | ||
|   | 963854f8d5 | ||
|   | 2ef5e7004b | ||
|   | 2e6f0dd442 | ||
|   | 7790783949 | ||
|   | f4534c8273 | ||
|   | 76673f02a4 | ||
|   | b47ae0944c | ||
|   | f9712cbc7c | ||
|   | 1b6d845530 | ||
|   | 19424aba9d | ||
|   | 71ac9b34fa | ||
|   | 7651d4a249 | ||
|   | f8e0bccdf0 | ||
|   | a5b6eb6afa | ||
|   | 41471da9db | ||
|   | e71767688d | ||
|   | 5467e3a842 | ||
|   | 511017ab35 | ||
|   | c3e8cfac79 | ||
|   | bf2b3c68f8 | ||
|   | 68e74a8c03 | ||
|   | 5b4a540440 | ||
|   | 88da5c05c7 | ||
|   | 2d369e8945 | ||
|   | 02548c4b9b | ||
|   | 71990e3ccc | ||
|   | 5e7dcc48b9 | ||
|   | 5cdba87b87 | ||
|   | 696315615d | ||
|   | dd3f820040 | ||
|   | 78ca16dfe3 | ||
|   | adc32d5a5c | ||
|   | 12e4514813 | ||
|   | 587fb37bdf | ||
|   | d221e52929 | ||
|   | e151a6cf92 | ||
|   | 9ced5ce131 | ||
|   | c8e876bfe2 | ||
|   | 2a716bbc7f | ||
|   | 00590b6c0d | ||
|   | 2ff4be7846 | ||
|   | ff5727c5ef | ||
|   | e46bc343e4 | ||
|   | 5972fd2353 | ||
|   | 8c0e4b1d33 | ||
|   | a737a754d1 | ||
|   | fc8685a042 | ||
|   | cc4765b4fe | ||
|   | eccdf87f22 | ||
|   | 862bb2ac72 | ||
|   | 34d416a4e8 | ||
|   | 410f64bc9f | ||
|   | 978e68fc74 | ||
|   | a17d8f8a66 | ||
|   | 7e47490e70 | ||
|   | f4f94d3b56 | ||
|   | eae9d45c8a | ||
|   | d2d157c1fe | ||
|   | 9c95d4ba43 | ||
|   | 6cbd7404f4 | ||
|   | ad8a8ef5a9 | ||
|   | e6766a1ee2 | ||
|   | b4ccdaa51c | ||
|   | 3a11656909 | ||
|   | 2479fab632 | ||
|   | 51eb1ac623 | ||
|   | 879f37f046 | ||
|   | f2aadd6014 | ||
|   | 1ad81504ca | ||
|   | 425ee4e142 | ||
|   | 42d8300bb7 | ||
|   | a9d75c5255 | ||
|   | 98afd548d1 | ||
|   | 7fc256dba7 | ||
|   | 21b23624ad | ||
|   | 1ace5c0c8b | ||
|   | bee99beab1 | ||
|   | 8b363d3d1f | ||
|   | c7b0c3cd48 | ||
|   | c0c9b28582 | ||
|   | c3fffbc919 | ||
|   | 6613aaea95 | ||
|   | 53ce6b93a2 | ||
|   | d8e293842f | ||
|   | 5cf869df1a | ||
|   | f26f7d8809 | ||
|   | c5a90475af | ||
|   | 3008d7ef2f | ||
|   | 1823f86dbb | ||
|   | 06bf088d27 | ||
|   | 9953698a7c | ||
|   | bda1605627 | ||
|   | 2e0e88db0d | ||
|   | 96482cc0cf | ||
|   | a283aaf724 | ||
|   | 5db276eb52 | ||
|   | c866619f56 | ||
|   | b9da505efe | ||
|   | 061a7c67bd | ||
|   | 6f7331e852 | ||
|   | 1b489bcc11 | ||
|   | da27924a49 | ||
|   | 15b39887c5 | ||
|   | f9583a7652 | ||
|   | cb738188de | ||
|   | a8b2f5268d | ||
|   | ad50fc9ad3 | ||
|   | a25c2b325b | ||
|   | c57bf61114 | ||
|   | b7935d4b14 | ||
|   | 00982f3620 | ||
|   | 088b2045d0 | ||
|   | 633ff1cfc8 | ||
|   | 6cda6b6c10 | ||
|   | 90573625f1 | ||
|   | d49e97c423 | ||
|   | 39a80e414e | ||
|   | ab9153ddc3 | ||
|   | 7ec1f487c1 | ||
|   | c96628b72e | ||
|   | e5ded9a2fb | ||
|   | 5823f47467 | ||
|   | a0b7f09252 | ||
|   | b60a30c705 | ||
|   | 4fc1a9e770 | ||
|   | f0908b663f | ||
|   | 5fa596fee9 | ||
|   | ada1b95ffc | ||
|   | 72b3bdf676 | ||
|   | 71d1246374 | ||
|   | ac371bb596 | ||
|   | 830ee8f27d | ||
|   | 425dd45109 | ||
|   | 6a039de8db | ||
|   | 8d783aa172 | ||
|   | a4e5f5005b | ||
|   | a7c6edeb63 | ||
|   | 4f23b49fef | ||
|   | cfcc8b1a6f | ||
|   | ebec582ce2 | ||
|   | 3d701fbe0e | ||
|   | e95da11115 | ||
|   | 9f6a798ea6 | ||
|   | 36c8df4d2f | ||
|   | baf2f60850 | ||
|   | 9ac39005f8 | 
							
								
								
									
										63
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| name: Bug inside BungeeCord | ||||
| description: Create a bug report about a problem inside BungeeCord. | ||||
| body: | ||||
|   - type: markdown | ||||
|     attributes: | ||||
|       value: | | ||||
|         #### Report a bug inside bungeecord | ||||
|         Issues happening with forks of BungeeCord should **not** be reported here. | ||||
|   - type: input | ||||
|     id: bungee-version | ||||
|     attributes: | ||||
|       label: Bungeecord version | ||||
|       description: The output of the /bungee command (or just the bungee build number) (execute in bungeecord console for easy text copy) | ||||
|       placeholder: e.g. git:BungeeCord-Bootstrap:1.xx-SNAPSHOT:xxxxxxx:xxxx | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: input | ||||
|     id: server-version | ||||
|     attributes: | ||||
|       label: Server version | ||||
|       description: The output of the /version command (execute in server console for easy text copy) | ||||
|       placeholder: "e.g. git-Spigot-xxxxxxx-xxxxxxx (MC: 1.x.x)" | ||||
|   - type: input | ||||
|     id: client-version | ||||
|     attributes: | ||||
|       label: Client version | ||||
|       description: Minecraft Client Version | ||||
|       placeholder: e.g. 1.18.2 | ||||
|   - type: textarea | ||||
|     id: bungee-plugins | ||||
|     attributes: | ||||
|       label: Bungeecord plugins | ||||
|       description: Please list all BungeeCord plugins you are using. | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: the-bug | ||||
|     attributes: | ||||
|       label: The bug | ||||
|       description: Please describe the bug. Include **details** you find neccessary. If you just have a question, please ask it in [SpigotMC Forums](https://www.spigotmc.org) and not here. | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: logs | ||||
|     attributes: | ||||
|       label: Log output (links) | ||||
|       description: Please put your log output inbetween three backticks (```` ``` ````). Upload your log files to [gist.github.com](https://gist.github.com) and put them in here. | ||||
|       placeholder: | | ||||
|         ``` | ||||
|         log output | ||||
|         ``` | ||||
|   - type: checkboxes | ||||
|     id: checkboxes | ||||
|     attributes: | ||||
|       label: Checking | ||||
|       options: | ||||
|         - label: I am using BungeeCord and **not a fork**. Issues with forks should not be reported here. | ||||
|           required: true | ||||
|         - label: I think this is **not** an issue with a bungeecord plugin. | ||||
|           required: true | ||||
|         - label: I have not read these checkboxes and therefore I just ticked them all. | ||||
|         - label: This is not a question or plugin creation help request. | ||||
|           required: true | ||||
							
								
								
									
										14
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: Configuration help | ||||
|     url: https://www.spigotmc.org/forums/bungeecord-help.70/create-thread | ||||
|     about: Help for configuring bungeecord will only be answered in spigotmc.org forums. | ||||
|   - name: I have a problem with a bungee plugin | ||||
|     url: https://www.spigotmc.org/forums/bungeecord-plugin-help.71/create-thread | ||||
|     about: Help about plugins can be recieved in spigotmc.org forums. | ||||
|   - name: Questions and discussions | ||||
|     url: https://www.spigotmc.org/forums/bungeecord-discussion.21/create-thread | ||||
|     about: spigotmc.org forums are the best place to ask your questions regarding bungeecord. | ||||
|   - name: Plugin creation help | ||||
|     url: https://www.spigotmc.org/forums/bungeecord-plugin-development.23/create-thread | ||||
|     about: Plugin creation help for bungee plugins can be recieved in spigotmc.org forums. | ||||
							
								
								
									
										36
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| name: Feature request | ||||
| description: Suggest a feature which bungeecord should include. | ||||
| body: | ||||
|   - type: textarea | ||||
|     id: the-feature | ||||
|     attributes: | ||||
|       label: Feature description | ||||
|       description: Please describe your feature or improvement. Please include **details**. | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: goal | ||||
|     attributes: | ||||
|       label: Goal of the feature | ||||
|       description: What is the goal of your feature? | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: alternatives | ||||
|     attributes: | ||||
|       label: Unfitting alternatives | ||||
|       description: What alternatives have you considered and why are they not sufficient for your use case? | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: checkboxes | ||||
|     id: checkboxes | ||||
|     attributes: | ||||
|       label: Checking | ||||
|       options: | ||||
|         - label: This is not a question or plugin creation help request. | ||||
|           required: true | ||||
|         - label: This is a **feature or improvement request**. | ||||
|           required: true | ||||
|         - label: I have not read these checkboxes and therefore I just ticked them all. | ||||
|         - label: I did not use this form to report a bug. | ||||
|           required: true | ||||
							
								
								
									
										27
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| version: 2 | ||||
|  | ||||
| updates: | ||||
|   - package-ecosystem: "maven" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "daily" | ||||
|     open-pull-requests-limit: 50 | ||||
|     ignore: | ||||
|         # Synchronised with Minecraft | ||||
|       - dependency-name: "com.google.code.gson:gson" | ||||
|         # 9.x has performance issues (see, eg, checkstyle/checkstyle#10934) and 10.x is incompatible | ||||
|       - dependency-name: "com.puppycrawl.tools:checkstyle" | ||||
|         # Newer versions have issues, see #1909 and #2050 | ||||
|       - dependency-name: "jline:jline" | ||||
|         # Needs to be synchronised with maven-resolver-provider dependencies | ||||
|       - dependency-name: "org.apache.maven.resolver:maven-resolver-connector-basic" | ||||
|       - dependency-name: "org.apache.maven.resolver:maven-resolver-transport-http" | ||||
|         # Used with maven-resolver dependencies; 2.0 update breaks other providers | ||||
|       - dependency-name: "org.slf4j:slf4j-api" | ||||
|         update-types: ["version-update:semver-major"] | ||||
|  | ||||
|   - package-ecosystem: "github-actions" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "daily" | ||||
|     open-pull-requests-limit: 50 | ||||
							
								
								
									
										10
									
								
								.github/workflows/maven.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/maven.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,18 +4,20 @@ on: [push, pull_request] | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|     runs-on: ubuntu-24.04 | ||||
|  | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         java: [8, 11] | ||||
|         java: [8, 11, 17, 21, 25] | ||||
|  | ||||
|     name: Java ${{ matrix.java }} | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/setup-java@v1 | ||||
|     - uses: actions/checkout@v4 | ||||
|     - uses: actions/setup-java@v4 | ||||
|       with: | ||||
|         distribution: zulu | ||||
|         java-version: ${{ matrix.java }} | ||||
|     - run: java -version && mvn --version | ||||
|     - run: mvn --activate-profiles dist --no-transfer-progress package | ||||
|   | ||||
							
								
								
									
										6
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| [submodule "native/mbedtls"] | ||||
| 	path = native/mbedtls | ||||
| 	url = https://github.com/ARMmbed/mbedtls.git | ||||
| [submodule "native/zlib"] | ||||
| 	path = native/zlib | ||||
| 	url = https://github.com/cloudflare/zlib.git | ||||
| @@ -23,4 +23,4 @@ Binaries | ||||
| -------- | ||||
| Precompiled binaries are available for end users on [Jenkins](https://www.spigotmc.org/go/bungeecord-dl). | ||||
|  | ||||
| (c) 2012-2020 SpigotMC Pty. Ltd. | ||||
| (c) 2012-2025 SpigotMC Pty. Ltd. | ||||
|   | ||||
							
								
								
									
										51
									
								
								api/pom.xml
									
									
									
									
									
								
							
							
						
						
									
										51
									
								
								api/pom.xml
									
									
									
									
									
								
							| @@ -4,15 +4,14 @@ | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <parent> | ||||
|         <groupId>net.md-5</groupId> | ||||
|         <groupId>fr.pandacube.bungeecord</groupId> | ||||
|         <artifactId>bungeecord-parent</artifactId> | ||||
|         <version>1.16-R0.4-SNAPSHOT</version> | ||||
|         <version>1.21-R0.5-SNAPSHOT</version> | ||||
|         <relativePath>../pom.xml</relativePath> | ||||
|     </parent> | ||||
|  | ||||
|     <groupId>net.md-5</groupId> | ||||
|     <artifactId>bungeecord-api</artifactId> | ||||
|     <version>1.16-R0.4-SNAPSHOT</version> | ||||
|     <version>1.21-R0.5-SNAPSHOT</version> | ||||
|     <packaging>jar</packaging> | ||||
|  | ||||
|     <name>BungeeCord-API</name> | ||||
| @@ -20,25 +19,31 @@ | ||||
|  | ||||
|     <dependencies> | ||||
|         <dependency> | ||||
|             <groupId>net.md-5</groupId> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-chat</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>net.md-5</groupId> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-config</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>net.md-5</groupId> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-dialog</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-event</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>net.md-5</groupId> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-protocol</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
| @@ -46,13 +51,39 @@ | ||||
|         <dependency> | ||||
|             <groupId>io.netty</groupId> | ||||
|             <artifactId>netty-transport-native-unix-common</artifactId> | ||||
|             <version>${netty.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.maven</groupId> | ||||
|             <artifactId>maven-resolver-provider</artifactId> | ||||
|             <version>3.9.6</version> | ||||
|             <!-- not part of the API proper --> | ||||
|             <scope>provided</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.maven.resolver</groupId> | ||||
|             <artifactId>maven-resolver-connector-basic</artifactId> | ||||
|             <version>1.9.18</version> | ||||
|             <!-- not part of the API proper --> | ||||
|             <scope>provided</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.maven.resolver</groupId> | ||||
|             <artifactId>maven-resolver-transport-http</artifactId> | ||||
|             <version>1.9.18</version> | ||||
|             <!-- not part of the API proper --> | ||||
|             <scope>provided</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.ow2.asm</groupId> | ||||
|             <artifactId>asm-commons</artifactId> | ||||
|             <version>9.8</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.yaml</groupId> | ||||
|             <artifactId>snakeyaml</artifactId> | ||||
|             <version>1.26</version> | ||||
|             <version>2.2</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import java.net.InetSocketAddress; | ||||
| import java.net.SocketAddress; | ||||
| import java.net.URI; | ||||
| import java.net.URISyntaxException; | ||||
| import java.util.Locale; | ||||
| import java.util.UUID; | ||||
|  | ||||
| /** | ||||
| @@ -68,6 +69,17 @@ public class Util | ||||
|         return String.format( "0x%02X", i ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Formats an char as a unicode value. | ||||
|      * | ||||
|      * @param c the character to format | ||||
|      * @return the unicode representation of the character | ||||
|      */ | ||||
|     public static String unicode(char c) | ||||
|     { | ||||
|         return "\\u" + String.format( "%04x", (int) c ).toUpperCase( Locale.ROOT ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Constructs a pretty one line version of a {@link Throwable}. Useful for | ||||
|      * debugging. | ||||
| @@ -76,11 +88,24 @@ public class Util | ||||
|      * @return a string representing information about the {@link Throwable} | ||||
|      */ | ||||
|     public static String exception(Throwable t) | ||||
|     { | ||||
|         return exception( t, true ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Constructs a pretty one line version of a {@link Throwable}. Useful for | ||||
|      * debugging. | ||||
|      * | ||||
|      * @param t the {@link Throwable} to format. | ||||
|      * @param includeLineNumbers whether to include line numbers | ||||
|      * @return a string representing information about the {@link Throwable} | ||||
|      */ | ||||
|     public static String exception(Throwable t, boolean includeLineNumbers) | ||||
|     { | ||||
|         // TODO: We should use clear manually written exceptions | ||||
|         StackTraceElement[] trace = t.getStackTrace(); | ||||
|         return t.getClass().getSimpleName() + " : " + t.getMessage() | ||||
|                 + ( ( trace.length > 0 ) ? " @ " + t.getStackTrace()[0].getClassName() + ":" + t.getStackTrace()[0].getLineNumber() : "" ); | ||||
|                 + ( ( includeLineNumbers && trace.length > 0 ) ? " @ " + t.getStackTrace()[0].getClassName() + ":" + t.getStackTrace()[0].getLineNumber() : "" ); | ||||
|     } | ||||
|  | ||||
|     public static String csv(Iterable<?> objects) | ||||
| @@ -88,6 +113,16 @@ public class Util | ||||
|         return format( objects, ", " ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns a string of objects, each separated by a separator. | ||||
|      * | ||||
|      * @param objects the objects to join | ||||
|      * @param separators the separator | ||||
|      * @return joined string | ||||
|      * @see String#join(java.lang.CharSequence, java.lang.Iterable) | ||||
|      * @deprecated use {@link String} join methods | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public static String format(Iterable<?> objects, String separators) | ||||
|     { | ||||
|         return Joiner.on( separators ).join( objects ); | ||||
|   | ||||
| @@ -28,18 +28,13 @@ public abstract class AbstractReconnectHandler implements ReconnectHandler | ||||
|  | ||||
|     public static ServerInfo getForcedHost(PendingConnection con) | ||||
|     { | ||||
|         if ( con.getVirtualHost() == null ) | ||||
|         { | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         String forced = con.getListener().getForcedHosts().get( con.getVirtualHost().getHostString() ); | ||||
|         String forced = ( con.getVirtualHost() == null ) ? null : con.getListener().getForcedHosts().get( con.getVirtualHost().getHostString() ); | ||||
|  | ||||
|         if ( forced == null && con.getListener().isForceDefault() ) | ||||
|         { | ||||
|             forced = con.getListener().getDefaultServer(); | ||||
|         } | ||||
|         return ProxyServer.getInstance().getServerInfo( forced ); | ||||
|         return ( forced == null ) ? null : ProxyServer.getInstance().getServerInfo( forced ); | ||||
|     } | ||||
|  | ||||
|     protected abstract ServerInfo getStoredServer(ProxiedPlayer player); | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| package net.md_5.bungee.api; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.common.io.BaseEncoding; | ||||
| import com.google.gson.TypeAdapter; | ||||
| import com.google.gson.internal.bind.TypeAdapters; | ||||
| import com.google.gson.stream.JsonReader; | ||||
| import com.google.gson.stream.JsonToken; | ||||
| import com.google.gson.stream.JsonWriter; | ||||
| import java.awt.image.BufferedImage; | ||||
| import java.io.ByteArrayOutputStream; | ||||
| @@ -26,13 +27,26 @@ public class Favicon | ||||
|         @Override | ||||
|         public void write(JsonWriter out, Favicon value) throws IOException | ||||
|         { | ||||
|             TypeAdapters.STRING.write( out, value == null ? null : value.getEncoded() ); | ||||
|             if ( value == null ) | ||||
|             { | ||||
|                 out.nullValue(); | ||||
|             } else | ||||
|             { | ||||
|                 out.value( value.getEncoded() ); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public Favicon read(JsonReader in) throws IOException | ||||
|         { | ||||
|             String enc = TypeAdapters.STRING.read( in ); | ||||
|             JsonToken peek = in.peek(); | ||||
|             if ( peek == JsonToken.NULL ) | ||||
|             { | ||||
|                 in.nextNull(); | ||||
|                 return null; | ||||
|             } | ||||
|  | ||||
|             String enc = in.nextString(); | ||||
|             return enc == null ? null : create( enc ); | ||||
|         } | ||||
|     }; | ||||
| @@ -59,6 +73,7 @@ public class Favicon | ||||
|      */ | ||||
|     public static Favicon create(BufferedImage image) | ||||
|     { | ||||
|         Preconditions.checkArgument( image != null, "image is null" ); | ||||
|         // check size | ||||
|         if ( image.getWidth() != 64 || image.getHeight() != 64 ) | ||||
|         { | ||||
|   | ||||
| @@ -16,6 +16,7 @@ import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.plugin.Plugin; | ||||
| import net.md_5.bungee.api.plugin.PluginManager; | ||||
| import net.md_5.bungee.api.scheduler.TaskScheduler; | ||||
| import net.md_5.bungee.protocol.channel.BungeeChannelInitializer; | ||||
|  | ||||
| public abstract class ProxyServer | ||||
| { | ||||
| @@ -311,4 +312,56 @@ public abstract class ProxyServer | ||||
|      */ | ||||
|     public abstract Title createTitle(); | ||||
|  | ||||
|     /** | ||||
|      * Get the unsafe methods of this class. | ||||
|      * | ||||
|      * @return the unsafe method interface | ||||
|      */ | ||||
|     public abstract Unsafe unsafe(); | ||||
|  | ||||
|     public interface Unsafe | ||||
|     { | ||||
|  | ||||
|         /** | ||||
|          * Gets the frontend channel initializer | ||||
|          * | ||||
|          * @return the frontend channel initializer | ||||
|          */ | ||||
|         BungeeChannelInitializer getFrontendChannelInitializer(); | ||||
|  | ||||
|         /** | ||||
|          * Set the frontend channel initializer of this proxy | ||||
|          * | ||||
|          * @param channelInitializer the frontend channelInitializer to set | ||||
|          */ | ||||
|         void setFrontendChannelInitializer(BungeeChannelInitializer channelInitializer); | ||||
|  | ||||
|         /** | ||||
|          * Gets the backend channel initializer | ||||
|          * | ||||
|          * @return the backend channel initializer | ||||
|          */ | ||||
|         BungeeChannelInitializer getBackendChannelInitializer(); | ||||
|  | ||||
|         /** | ||||
|          * Set the backend channel initializer of this proxy | ||||
|          * | ||||
|          * @param channelInitializer the backend channelInitializer to set | ||||
|          */ | ||||
|         void setBackendChannelInitializer(BungeeChannelInitializer channelInitializer); | ||||
|  | ||||
|         /** | ||||
|          * Gets the server info channel initializer | ||||
|          * | ||||
|          * @return the server info channel initializer | ||||
|          */ | ||||
|         BungeeChannelInitializer getServerInfoChannelInitializer(); | ||||
|  | ||||
|         /** | ||||
|          * Set the server info channel initializer of this proxy | ||||
|          * | ||||
|          * @param channelInitializer the server info channelInitializer to set | ||||
|          */ | ||||
|         void setServerInfoChannelInitializer(BungeeChannelInitializer channelInitializer); | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										77
									
								
								api/src/main/java/net/md_5/bungee/api/ServerLink.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								api/src/main/java/net/md_5/bungee/api/ServerLink.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| package net.md_5.bungee.api; | ||||
|  | ||||
| import lombok.AccessLevel; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| /** | ||||
|  * Represents a server link which may be sent to the client. | ||||
|  */ | ||||
| @Data | ||||
| @RequiredArgsConstructor(access = AccessLevel.PRIVATE) | ||||
| public final class ServerLink | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The links type. | ||||
|      * | ||||
|      * Note: This value is nullable, if null, label is non-null. | ||||
|      */ | ||||
|     private final LinkType type; | ||||
|  | ||||
|     /** | ||||
|      * The label for the link. | ||||
|      * | ||||
|      * Note: This value is nullable, if null, type is non-null. | ||||
|      */ | ||||
|     private final BaseComponent label; | ||||
|  | ||||
|     /** | ||||
|      * The URL that is displayed. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private final String url; | ||||
|  | ||||
|     /** | ||||
|      * Creates a link with a specified type and URL. | ||||
|      * | ||||
|      * @param type the type of the link | ||||
|      * @param url the URL to be displayed | ||||
|      */ | ||||
|     public ServerLink(@NonNull LinkType type, @NonNull String url) | ||||
|     { | ||||
|         this.type = type; | ||||
|         this.label = null; | ||||
|         this.url = url; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates a link with a label and URL. | ||||
|      * | ||||
|      * @param label the label to be displayed | ||||
|      * @param url the URL to be displayed | ||||
|      */ | ||||
|     public ServerLink(@NonNull BaseComponent label, @NonNull String url) | ||||
|     { | ||||
|         this.type = null; | ||||
|         this.label = label; | ||||
|         this.url = url; | ||||
|     } | ||||
|  | ||||
|     public enum LinkType | ||||
|     { | ||||
|  | ||||
|         REPORT_BUG, | ||||
|         COMMUNITY_GUIDELINES, | ||||
|         SUPPORT, | ||||
|         STATUS, | ||||
|         FEEDBACK, | ||||
|         COMMUNITY, | ||||
|         WEBSITE, | ||||
|         FORUMS, | ||||
|         NEWS, | ||||
|         ANNOUNCEMENTS; | ||||
|     } | ||||
| } | ||||
| @@ -105,13 +105,13 @@ public class ServerPing | ||||
|     @Deprecated | ||||
|     public ServerPing(Protocol version, Players players, String description, String favicon) | ||||
|     { | ||||
|         this( version, players, new TextComponent( TextComponent.fromLegacyText( description ) ), favicon == null ? null : Favicon.create( favicon ) ); | ||||
|         this( version, players, TextComponent.fromLegacy( description ), favicon == null ? null : Favicon.create( favicon ) ); | ||||
|     } | ||||
|  | ||||
|     @Deprecated | ||||
|     public ServerPing(Protocol version, Players players, String description, Favicon favicon) | ||||
|     { | ||||
|         this( version, players, new TextComponent( TextComponent.fromLegacyText( description ) ), favicon ); | ||||
|         this( version, players, TextComponent.fromLegacy( description ), favicon ); | ||||
|     } | ||||
|  | ||||
|     @Deprecated | ||||
| @@ -139,7 +139,7 @@ public class ServerPing | ||||
|     @Deprecated | ||||
|     public void setDescription(String description) | ||||
|     { | ||||
|         this.description = new TextComponent( TextComponent.fromLegacyText( description ) ); | ||||
|         this.description = TextComponent.fromLegacy( description ); | ||||
|     } | ||||
|  | ||||
|     @Deprecated | ||||
|   | ||||
| @@ -84,5 +84,17 @@ public interface Connection | ||||
|          * @param packet the packet to send | ||||
|          */ | ||||
|         void sendPacket(DefinedPacket packet); | ||||
|  | ||||
|         /** | ||||
|          * Queue a packet to this connection. | ||||
|          * | ||||
|          * If the packet is not registered for the connections current encoder | ||||
|          * protocol, it will be queued until it is, otherwise it will be sent | ||||
|          * immediately. | ||||
|          * | ||||
|          * @param packet the packet to be queued | ||||
|          * @throws UnsupportedOperationException if used for a PendingConnection | ||||
|          */ | ||||
|         void sendPacketQueued(DefinedPacket packet); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,9 @@ package net.md_5.bungee.api.connection; | ||||
|  | ||||
| import java.net.InetSocketAddress; | ||||
| import java.util.UUID; | ||||
| import java.util.concurrent.CompletableFuture; | ||||
| import net.md_5.bungee.api.config.ListenerInfo; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents a user attempting to log into the proxy. | ||||
| @@ -89,4 +91,40 @@ public interface PendingConnection extends Connection | ||||
|      * @return Whether the client is using a legacy client. | ||||
|      */ | ||||
|     boolean isLegacy(); | ||||
|  | ||||
|     /** | ||||
|      * Gets if this connection has been transferred from another server. | ||||
|      * | ||||
|      * @return true if the connection has been transferred | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     boolean isTransferred(); | ||||
|  | ||||
|     /** | ||||
|      * Retrieves a cookie from this pending connection. | ||||
|      * | ||||
|      * @param cookie the resource location of the cookie, for example | ||||
|      * "bungeecord:my_cookie" | ||||
|      * @return a {@link CompletableFuture} that will be completed when the | ||||
|      * Cookie response is received. If the cookie is not set in the client, the | ||||
|      * {@link CompletableFuture} will complete with a null value | ||||
|      * @throws IllegalStateException if the player's version is not at least | ||||
|      * 1.20.5 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     CompletableFuture<byte[]> retrieveCookie(String cookie); | ||||
|  | ||||
|     /** | ||||
|      * Sends a login payload request to the client. | ||||
|      * | ||||
|      * @param channel the channel to send this data via | ||||
|      * @param data the data to send | ||||
|      * @return a {@link CompletableFuture} that will be completed when the Login | ||||
|      * Payload response is received. If the Vanilla client doesn't know the | ||||
|      * channel, the {@link CompletableFuture} will complete with a null value | ||||
|      * @throws IllegalStateException if the player's version is not at least | ||||
|      * 1.13 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     CompletableFuture<byte[]> sendData(String channel, byte[] data); | ||||
| } | ||||
|   | ||||
| @@ -1,21 +1,26 @@ | ||||
| package net.md_5.bungee.api.connection; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Locale; | ||||
| import java.util.Map; | ||||
| import java.util.UUID; | ||||
| import java.util.concurrent.CompletableFuture; | ||||
| import net.md_5.bungee.api.Callback; | ||||
| import net.md_5.bungee.api.ChatMessageType; | ||||
| import net.md_5.bungee.api.CommandSender; | ||||
| import net.md_5.bungee.api.ServerConnectRequest; | ||||
| import net.md_5.bungee.api.ServerLink; | ||||
| import net.md_5.bungee.api.SkinConfiguration; | ||||
| import net.md_5.bungee.api.Title; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| import net.md_5.bungee.api.config.ServerInfo; | ||||
| import net.md_5.bungee.api.dialog.Dialog; | ||||
| import net.md_5.bungee.api.event.ServerConnectEvent; | ||||
| import net.md_5.bungee.api.score.Scoreboard; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents a player who's connection is being connected to somewhere else, | ||||
|  * Represents a player whose connection is being connected to somewhere else, | ||||
|  * whether it be a remote or embedded server. | ||||
|  */ | ||||
| public interface ProxiedPlayer extends Connection, CommandSender | ||||
| @@ -57,8 +62,7 @@ public interface ProxiedPlayer extends Connection, CommandSender | ||||
|     String getDisplayName(); | ||||
|  | ||||
|     /** | ||||
|      * Sets this players display name to be used as their nametag and tab list | ||||
|      * name. | ||||
|      * Sets this player's display name to be used by proxy commands and plugins. | ||||
|      * | ||||
|      * @param name the name to set | ||||
|      */ | ||||
| @@ -335,6 +339,90 @@ public interface ProxiedPlayer extends Connection, CommandSender | ||||
|      * Get the {@link Scoreboard} that belongs to this player. | ||||
|      * | ||||
|      * @return this player's {@link Scoreboard} | ||||
|      * @deprecated for internal use only, setters will not have the expected | ||||
|      * effect, will not update client state, and may corrupt proxy state | ||||
|      */ | ||||
|     @Deprecated | ||||
|     Scoreboard getScoreboard(); | ||||
|  | ||||
|     /** | ||||
|      * Retrieves a cookie from this player. | ||||
|      * | ||||
|      * @param cookie the resource location of the cookie, for example | ||||
|      * "bungeecord:my_cookie" | ||||
|      * @return a {@link CompletableFuture} that will be completed when the | ||||
|      * Cookie response is received. If the cookie is not set in the client, the | ||||
|      * {@link CompletableFuture} will complete with a null value | ||||
|      * @throws IllegalStateException if the player's version is not at least | ||||
|      * 1.20.5 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     CompletableFuture<byte[]> retrieveCookie(String cookie); | ||||
|  | ||||
|     /** | ||||
|      * Stores a cookie in this player's client. | ||||
|      * | ||||
|      * @param cookie the resource location of the cookie, for example | ||||
|      * "bungeecord:my_cookie" | ||||
|      * @param data the data to store in the cookie | ||||
|      * @throws IllegalStateException if the player's version is not at least | ||||
|      * 1.20.5 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     void storeCookie(String cookie, byte[] data); | ||||
|  | ||||
|     /** | ||||
|      * Requests this player to connect to a different server specified by host | ||||
|      * and port. | ||||
|      * | ||||
|      * This is a client-side transfer - host and port should not specify a | ||||
|      * BungeeCord backend server. | ||||
|      * | ||||
|      * @param host the host of the server to transfer to | ||||
|      * @param port the port of the server to transfer to | ||||
|      * @throws IllegalStateException if the players version is not at least | ||||
|      * 1.20.5 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     void transfer(String host, int port); | ||||
|  | ||||
|     /** | ||||
|      * Gets the client brand of this player. | ||||
|      * | ||||
|      * If the player has not sent a brand packet yet, it will return null. | ||||
|      * | ||||
|      * @return the brand of the client, or null if not received yet | ||||
|      */ | ||||
|     String getClientBrand(); | ||||
|  | ||||
|     /** | ||||
|      * Clear the player's open dialog. | ||||
|      * | ||||
|      * @throws IllegalStateException if the players version is not at least | ||||
|      * 1.21.6 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     void clearDialog(); | ||||
|  | ||||
|     /** | ||||
|      * Show a dialog to the player. | ||||
|      * | ||||
|      * @param dialog the dialog to show | ||||
|      * @throws IllegalStateException if the players version is not at least | ||||
|      * 1.21.6 | ||||
|      */ | ||||
|     @ApiStatus.Experimental | ||||
|     void showDialog(Dialog dialog); | ||||
|  | ||||
|     /** | ||||
|      * Sends server links to the player. | ||||
|      * | ||||
|      * Note: The links already sent to the player will be overwritten. Also, the | ||||
|      * backend server is able to override links sent by the proxy. | ||||
|      * | ||||
|      * @param serverLinks the server links to send | ||||
|      * @throws IllegalStateException if the player's version is not at least | ||||
|      * 1.21 | ||||
|      */ | ||||
|     void sendServerLinks(List<ServerLink> serverLinks); | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,157 @@ | ||||
| package net.md_5.bungee.api.event; | ||||
|  | ||||
| import com.mojang.brigadier.arguments.ArgumentType; | ||||
| import com.mojang.brigadier.arguments.IntegerArgumentType; | ||||
| import com.mojang.brigadier.arguments.StringArgumentType; | ||||
| import com.mojang.brigadier.builder.ArgumentBuilder; | ||||
| import com.mojang.brigadier.builder.RequiredArgumentBuilder; | ||||
| import com.mojang.brigadier.suggestion.SuggestionProvider; | ||||
| import com.mojang.brigadier.tree.CommandNode; | ||||
| import com.mojang.brigadier.tree.RootCommandNode; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Setter; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.CommandSender; | ||||
| import net.md_5.bungee.api.connection.Connection; | ||||
| import net.md_5.bungee.api.plugin.Command; | ||||
| import net.md_5.bungee.api.plugin.Plugin; | ||||
| import net.md_5.bungee.api.plugin.PluginManager; | ||||
| import net.md_5.bungee.api.plugin.TabExecutor; | ||||
|  | ||||
| /** | ||||
|  * Event called when a downstream server (on 1.13+) sends the command structure | ||||
|  * to a player, but before BungeeCord adds the dummy command nodes of | ||||
|  * registered commands. | ||||
|  * <p> | ||||
|  * BungeeCord will not overwrite the modifications made by the listeners. | ||||
|  * | ||||
|  * <h2>Usage example</h2> | ||||
|  * Here is a usage example of this event, to declare a command structure. | ||||
|  * This illustrates the commands /server and /send of Bungee. | ||||
|  * <pre> | ||||
|  * event.getRoot().addChild( LiteralArgumentBuilder.<CommandSender>literal( "server" ) | ||||
|  *         .requires( sender -> sender.hasPermission( "bungeecord.command.server" ) ) | ||||
|  *         .executes( a -> 0 ) | ||||
|  *         .then( RequiredArgumentBuilder.argument( "serverName", StringArgumentType.greedyString() ) | ||||
|  *                 .suggests( SuggestionRegistry.ASK_SERVER ) | ||||
|  *         ) | ||||
|  *         .build() | ||||
|  * ); | ||||
|  * event.getRoot().addChild( LiteralArgumentBuilder.<CommandSender>literal( "send" ) | ||||
|  *         .requires( sender -> sender.hasPermission( "bungeecord.command.send" ) ) | ||||
|  *         .then( RequiredArgumentBuilder.argument( "playerName", StringArgumentType.word() ) | ||||
|  *                 .suggests( SuggestionRegistry.ASK_SERVER ) | ||||
|  *                 .then( RequiredArgumentBuilder.argument( "serverName", StringArgumentType.greedyString() ) | ||||
|  *                         .suggests( SuggestionRegistry.ASK_SERVER ) | ||||
|  *                 ) | ||||
|  *         ) | ||||
|  *         .build() | ||||
|  * ); | ||||
|  * </pre> | ||||
|  * | ||||
|  * <h2>Flag a {@link CommandNode} as executable or not</h2> | ||||
|  * The implementation of a {@link com.mojang.brigadier.Command Command} used in | ||||
|  * {@link ArgumentBuilder#executes(com.mojang.brigadier.Command)} will never be | ||||
|  * executed. This will only tell to the client if the current node is | ||||
|  * executable or not. | ||||
|  * <ul> | ||||
|  *     <li> | ||||
|  *         {@code builder.executes(null)} (default) to mark the node as not | ||||
|  *         executable. | ||||
|  *     </li> | ||||
|  *     <li> | ||||
|  *         {@code builder.executes(a -> 0)}, or any non null argument, to mark | ||||
|  *         the node as executable (the child arguments are displayed as | ||||
|  *         optional). | ||||
|  *     </li> | ||||
|  * </ul> | ||||
|  * | ||||
|  * <h2>{@link CommandNode}’s suggestions management</h2> | ||||
|  * The implementation of a SuggestionProvider used in | ||||
|  * {@link RequiredArgumentBuilder#suggests(SuggestionProvider)} will never be | ||||
|  * executed. This will only tell to the client how to deal with the | ||||
|  * auto-completion of the argument. | ||||
|  * <ul> | ||||
|  *     <li> | ||||
|  *         {@code builder.suggests(null)} (default) to disable auto-completion | ||||
|  *         for this argument. | ||||
|  *     </li> | ||||
|  *     <li> | ||||
|  *         {@code builder.suggests(SuggestionRegistry.ALL_RECIPES)} to suggest | ||||
|  *         Minecraft’s recipes. | ||||
|  *     </li> | ||||
|  *     <li> | ||||
|  *         {@code builder.suggests(SuggestionRegistry.AVAILABLE_SOUNDS)} to | ||||
|  *         suggest Minecraft’s default sound identifiers. | ||||
|  *     </li> | ||||
|  *     <li> | ||||
|  *         {@code builder.suggests(SuggestionRegistry.SUMMONABLE_ENTITIES)} to | ||||
|  *         suggest Minecraft’s default summonable entities identifiers. | ||||
|  *     </li> | ||||
|  *     <li> | ||||
|  *         {@code builder.suggests(SuggestionRegistry.ASK_SERVER)}, or any | ||||
|  *         other non null argument, to make the Minecraft client ask | ||||
|  *         auto-completion to the server. Any specified implementation of | ||||
|  *         {@link SuggestionProvider} will never be executed. | ||||
|  *     </li> | ||||
|  * </ul> | ||||
|  * | ||||
|  * <h2>Argument types</h2> | ||||
|  * When building a new argument command node using | ||||
|  * {@link RequiredArgumentBuilder#argument(String, ArgumentType)}, you have to | ||||
|  * specify an {@link ArgumentType}. You can use all subclasses of | ||||
|  * {@link ArgumentType} provided with brigadier (for instance, | ||||
|  * {@link StringArgumentType} or {@link IntegerArgumentType}), or call any | ||||
|  * {@code ArgumentRegistry.minecraft*()} methods to use a {@code minecraft:*} | ||||
|  * argument type. | ||||
|  * | ||||
|  * <h2>Limitations with brigadier API</h2> | ||||
|  * This event is only used for the client to show command syntax, suggest | ||||
|  * sub-commands and color the arguments in the chat box. The command execution | ||||
|  * needs to be implemented using {@link PluginManager#registerCommand(Plugin, | ||||
|  * Command)} and the server-side tab-completion using {@link TabCompleteEvent} | ||||
|  * or {@link TabExecutor}. | ||||
|  */ | ||||
| @Data | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public class CommandsDeclareEvent extends TargetedEvent | ||||
| { | ||||
|     /** | ||||
|      * Wether or not the command tree is modified by this event. | ||||
|      * | ||||
|      * If this value is set to true, BungeeCord will ensure that the | ||||
|      * modifications made in the command tree, will be sent to the player. | ||||
|      * If this is false, the modifications may not be taken into account. | ||||
|      * | ||||
|      * When calling {@link #getRoot()}, this value is automatically set | ||||
|      * to true. | ||||
|      */ | ||||
|     @Setter(value = AccessLevel.NONE) | ||||
|     private boolean modified = false; | ||||
|  | ||||
|     /** | ||||
|      * The root command node of the command structure that will be send to the | ||||
|      * player. | ||||
|      */ | ||||
|     private final RootCommandNode<CommandSender> root; | ||||
|  | ||||
|     public CommandsDeclareEvent(Connection sender, Connection receiver, RootCommandNode<CommandSender> root) | ||||
|     { | ||||
|         super( sender, receiver ); | ||||
|         this.root = root; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * The root command node of the command structure that will be send to the | ||||
|      * player. | ||||
|      * @return The root command node | ||||
|      */ | ||||
|     public RootCommandNode<CommandSender> getRoot() | ||||
|     { | ||||
|         modified = true; | ||||
|         return root; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,39 @@ | ||||
| package net.md_5.bungee.api.event; | ||||
|  | ||||
| import com.google.gson.JsonElement; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.plugin.Cancellable; | ||||
| import net.md_5.bungee.api.plugin.Event; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Called after a {@link ProxiedPlayer} runs a custom action from a chat event | ||||
|  * or form submission. | ||||
|  */ | ||||
| @Data | ||||
| @ToString(callSuper = false) | ||||
| @EqualsAndHashCode(callSuper = false) | ||||
| @ApiStatus.Experimental | ||||
| public class CustomClickEvent extends Event implements Cancellable | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * Player who clicked. | ||||
|      */ | ||||
|     private final ProxiedPlayer player; | ||||
|     /** | ||||
|      * Custom action ID. | ||||
|      */ | ||||
|     private final String id; | ||||
|     /** | ||||
|      * The data as submitted. | ||||
|      */ | ||||
|     private final JsonElement data; | ||||
|     /** | ||||
|      * Cancelled state. | ||||
|      */ | ||||
|     private boolean cancelled; | ||||
| } | ||||
| @@ -1,9 +1,7 @@ | ||||
| package net.md_5.bungee.api.event; | ||||
|  | ||||
| import lombok.AccessLevel; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Setter; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.Callback; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| @@ -27,8 +25,7 @@ public class LoginEvent extends AsyncEvent<LoginEvent> implements Cancellable | ||||
|     /** | ||||
|      * Message to use when kicking if this event is canceled. | ||||
|      */ | ||||
|     @Setter(AccessLevel.NONE) | ||||
|     private BaseComponent[] cancelReasonComponents; | ||||
|     private BaseComponent reason; | ||||
|     /** | ||||
|      * Connection attempting to login. | ||||
|      */ | ||||
| @@ -42,28 +39,44 @@ public class LoginEvent extends AsyncEvent<LoginEvent> implements Cancellable | ||||
|  | ||||
|     /** | ||||
|      * @return reason to be displayed | ||||
|      * @deprecated Use component methods instead. | ||||
|      * @deprecated use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public String getCancelReason() | ||||
|     { | ||||
|         return BaseComponent.toLegacyText( getCancelReasonComponents() ); | ||||
|         return TextComponent.toLegacyText( getReason() ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param cancelReason reason to be displayed | ||||
|      * @deprecated Use | ||||
|      * {@link #setCancelReason(net.md_5.bungee.api.chat.BaseComponent...)} | ||||
|      * instead. | ||||
|      * @deprecated use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setCancelReason(String cancelReason) | ||||
|     { | ||||
|         setCancelReason( TextComponent.fromLegacyText( cancelReason ) ); | ||||
|         setReason( TextComponent.fromLegacy( cancelReason ) ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return reason to be displayed | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public BaseComponent[] getCancelReasonComponents() | ||||
|     { | ||||
|         return new BaseComponent[] | ||||
|         { | ||||
|             getReason() | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param cancelReason reason to be displayed | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setCancelReason(BaseComponent... cancelReason) | ||||
|     { | ||||
|         this.cancelReasonComponents = cancelReason; | ||||
|         setReason( TextComponent.fromArray( cancelReason ) ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -3,8 +3,9 @@ package net.md_5.bungee.api.event; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.Callback; | ||||
| import net.md_5.bungee.api.config.ServerInfo; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.plugin.Event; | ||||
|  | ||||
| /** | ||||
|  * Event called as soon as a connection has a {@link ProxiedPlayer} and is ready | ||||
| @@ -13,11 +14,22 @@ import net.md_5.bungee.api.plugin.Event; | ||||
| @Data | ||||
| @ToString(callSuper = false) | ||||
| @EqualsAndHashCode(callSuper = false) | ||||
| public class PostLoginEvent extends Event | ||||
| public class PostLoginEvent extends AsyncEvent<PostLoginEvent> | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The player involved with this event. | ||||
|      */ | ||||
|     private final ProxiedPlayer player; | ||||
|     /** | ||||
|      * The server to which the player will initially be connected. | ||||
|      */ | ||||
|     private ServerInfo target; | ||||
|  | ||||
|     public PostLoginEvent(ProxiedPlayer player, ServerInfo target, Callback<PostLoginEvent> done) | ||||
|     { | ||||
|         super( done ); | ||||
|         this.player = player; | ||||
|         this.target = target; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,7 @@ | ||||
| package net.md_5.bungee.api.event; | ||||
|  | ||||
| import lombok.AccessLevel; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Setter; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.Callback; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| @@ -32,8 +30,7 @@ public class PreLoginEvent extends AsyncEvent<PreLoginEvent> implements Cancella | ||||
|     /** | ||||
|      * Message to use when kicking if this event is canceled. | ||||
|      */ | ||||
|     @Setter(AccessLevel.NONE) | ||||
|     private BaseComponent[] cancelReasonComponents; | ||||
|     private BaseComponent reason; | ||||
|     /** | ||||
|      * Connection attempting to login. | ||||
|      */ | ||||
| @@ -47,28 +44,44 @@ public class PreLoginEvent extends AsyncEvent<PreLoginEvent> implements Cancella | ||||
|  | ||||
|     /** | ||||
|      * @return reason to be displayed | ||||
|      * @deprecated Use component methods instead. | ||||
|      * @deprecated use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public String getCancelReason() | ||||
|     { | ||||
|         return BaseComponent.toLegacyText( getCancelReasonComponents() ); | ||||
|         return BaseComponent.toLegacyText( getReason() ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param cancelReason reason to be displayed | ||||
|      * @deprecated Use | ||||
|      * {@link #setCancelReason(net.md_5.bungee.api.chat.BaseComponent...)} | ||||
|      * instead. | ||||
|      * @deprecated Use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setCancelReason(String cancelReason) | ||||
|     { | ||||
|         setCancelReason( TextComponent.fromLegacyText( cancelReason ) ); | ||||
|         setReason( TextComponent.fromLegacy( cancelReason ) ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return reason to be displayed | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public BaseComponent[] getCancelReasonComponents() | ||||
|     { | ||||
|         return new BaseComponent[] | ||||
|         { | ||||
|             getReason() | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param cancelReason reason to be displayed | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setCancelReason(BaseComponent... cancelReason) | ||||
|     { | ||||
|         this.cancelReasonComponents = cancelReason; | ||||
|         setReason( TextComponent.fromArray( cancelReason ) ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import net.md_5.bungee.api.ServerPing; | ||||
| import net.md_5.bungee.api.connection.PendingConnection; | ||||
|  | ||||
| /** | ||||
|  * Called when the proxy is pinged with packet 0xFE from the server list. | ||||
|  * Called when the proxy is queried for status from the server list. | ||||
|  */ | ||||
| @Data | ||||
| @ToString(callSuper = false) | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package net.md_5.bungee.api.event; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Getter; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.CommandSender; | ||||
| import net.md_5.bungee.api.plugin.Event; | ||||
|  | ||||
| @@ -10,6 +11,7 @@ import net.md_5.bungee.api.plugin.Event; | ||||
|  * Called when somebody reloads BungeeCord | ||||
|  */ | ||||
| @Getter | ||||
| @ToString(callSuper = false) | ||||
| @AllArgsConstructor | ||||
| @EqualsAndHashCode(callSuper = false) | ||||
| public class ProxyReloadEvent extends Event | ||||
|   | ||||
| @@ -9,6 +9,13 @@ import net.md_5.bungee.api.config.ServerInfo; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.plugin.Event; | ||||
|  | ||||
| /** | ||||
|  * Called when the player is disconnected from a server, for example during | ||||
|  * server switching. | ||||
|  * | ||||
|  * If the player is kicked from a server, {@link ServerKickEvent} will be called | ||||
|  * instead. | ||||
|  */ | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| @ToString(callSuper = false) | ||||
|   | ||||
| @@ -35,7 +35,7 @@ public class ServerKickEvent extends Event implements Cancellable | ||||
|     /** | ||||
|      * Kick reason. | ||||
|      */ | ||||
|     private BaseComponent[] kickReasonComponent; | ||||
|     private BaseComponent reason; | ||||
|     /** | ||||
|      * Server to send player to if this event is cancelled. | ||||
|      */ | ||||
| @@ -63,24 +63,61 @@ public class ServerKickEvent extends Event implements Cancellable | ||||
|         this( player, player.getServer().getInfo(), kickReasonComponent, cancelServer, state ); | ||||
|     } | ||||
|  | ||||
|     @Deprecated | ||||
|     public ServerKickEvent(ProxiedPlayer player, ServerInfo kickedFrom, BaseComponent[] kickReasonComponent, ServerInfo cancelServer, State state) | ||||
|     { | ||||
|         this( player, kickedFrom, TextComponent.fromArray( kickReasonComponent ), cancelServer, state ); | ||||
|     } | ||||
|  | ||||
|     public ServerKickEvent(ProxiedPlayer player, ServerInfo kickedFrom, BaseComponent reason, ServerInfo cancelServer, State state) | ||||
|     { | ||||
|         this.player = player; | ||||
|         this.kickedFrom = kickedFrom; | ||||
|         this.kickReasonComponent = kickReasonComponent; | ||||
|         this.reason = reason; | ||||
|         this.cancelServer = cancelServer; | ||||
|         this.state = state; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return the kick reason | ||||
|      * @deprecated use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public String getKickReason() | ||||
|     { | ||||
|         return BaseComponent.toLegacyText( kickReasonComponent ); | ||||
|         return BaseComponent.toLegacyText( getReason() ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param reason the kick reason | ||||
|      * @deprecated use component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setKickReason(String reason) | ||||
|     { | ||||
|         kickReasonComponent = TextComponent.fromLegacyText( reason ); | ||||
|         this.setReason( TextComponent.fromLegacy( reason ) ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return the kick reason | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public BaseComponent[] getKickReasonComponent() | ||||
|     { | ||||
|         return new BaseComponent[] | ||||
|         { | ||||
|             getReason() | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param kickReasonComponent the kick reason | ||||
|      * @deprecated use single component methods instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public void setKickReasonComponent(BaseComponent[] kickReasonComponent) | ||||
|     { | ||||
|         this.setReason( TextComponent.fromArray( kickReasonComponent ) ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -9,7 +9,9 @@ import net.md_5.bungee.api.plugin.Cancellable; | ||||
|  | ||||
| /** | ||||
|  * Event called when a player uses tab completion. | ||||
|  * @deprecated please use {@link TabCompleteRequestEvent} to support 1.13+ suggestions. | ||||
|  */ | ||||
| @Deprecated | ||||
| @Data | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
|   | ||||
| @@ -0,0 +1,85 @@ | ||||
| package net.md_5.bungee.api.event; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.mojang.brigadier.context.StringRange; | ||||
| import com.mojang.brigadier.suggestion.Suggestions; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.connection.Connection; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.plugin.Cancellable; | ||||
| import net.md_5.bungee.protocol.ProtocolConstants; | ||||
|  | ||||
| /** | ||||
|  * Event called when a player uses tab completion. | ||||
|  */ | ||||
| @Data | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public class TabCompleteRequestEvent extends TargetedEvent implements Cancellable | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * Cancelled state. | ||||
|      */ | ||||
|     private boolean cancelled; | ||||
|     /** | ||||
|      * The message the player has already entered. | ||||
|      */ | ||||
|     private final String cursor; | ||||
|     /** | ||||
|      * Range corresponding to the last word of {@link #getCursor()}. | ||||
|      * If you want your suggestions to be compatible with 1.12 and older | ||||
|      * clients, you need to {@link #setSuggestions(Suggestions)} with | ||||
|      * a range equals to this one. | ||||
|      * For 1.13 and newer clients, any other range that cover any part of | ||||
|      * {@link #getCursor()} is fine.<br> | ||||
|      * To check if the client supports custom ranges, use | ||||
|      * {@link #supportsCustomRange()}. | ||||
|      */ | ||||
|     private final StringRange legacyCompatibleRange; | ||||
|     /** | ||||
|      * The suggestions that will be sent to the client. If this list is empty, | ||||
|      * the request will be forwarded to the server. | ||||
|      */ | ||||
|     private Suggestions suggestions; | ||||
|  | ||||
|     public TabCompleteRequestEvent(Connection sender, Connection receiver, String cursor, StringRange legacyCompatibleRange, Suggestions suggestions) | ||||
|     { | ||||
|         super( sender, receiver ); | ||||
|         this.cursor = cursor; | ||||
|         this.legacyCompatibleRange = legacyCompatibleRange; | ||||
|         this.suggestions = suggestions; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the suggestions that will be sent to the client. | ||||
|      * If this list is empty, the request will be forwarded to the server. | ||||
|      * @param suggestions the new Suggestions. Cannot be null. | ||||
|      * @throws IllegalArgumentException if the client is on 1.12 or lower and | ||||
|      * {@code suggestions.getRange()} is not equals to {@link #legacyCompatibleRange}. | ||||
|      */ | ||||
|     public void setSuggestions(Suggestions suggestions) | ||||
|     { | ||||
|         Preconditions.checkNotNull( suggestions ); | ||||
|         Preconditions.checkArgument( supportsCustomRange() || legacyCompatibleRange.equals( suggestions.getRange() ), | ||||
|                 "Clients on 1.12 or lower versions don't support the provided range for tab-completion: " + suggestions.getRange() | ||||
|                 + ". Please use TabCompleteRequestEvent.getLegacyCompatibleRange() for legacy clients." ); | ||||
|         this.suggestions = suggestions; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Convenient method to tell if the client supports custom range for | ||||
|      * suggestions. | ||||
|      * If the client is on 1.13 or above, this methods returns true, and any | ||||
|      * range can be used for {@link #setSuggestions(Suggestions)}. Otherwise, | ||||
|      * it returns false and the defined range must be equals to | ||||
|      * {@link #legacyCompatibleRange}. | ||||
|      * @return true if the client is on 1.13 or newer version, false otherwise. | ||||
|      */ | ||||
|     public boolean supportsCustomRange() | ||||
|     { | ||||
|         return ( (ProxiedPlayer) getSender() ).getPendingConnection().getVersion() >= ProtocolConstants.MINECRAFT_1_13; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										129
									
								
								api/src/main/java/net/md_5/bungee/api/plugin/LibraryLoader.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								api/src/main/java/net/md_5/bungee/api/plugin/LibraryLoader.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | ||||
| package net.md_5.bungee.api.plugin; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.net.MalformedURLException; | ||||
| import java.net.URL; | ||||
| import java.net.URLClassLoader; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.logging.Level; | ||||
| import java.util.logging.Logger; | ||||
| import org.apache.maven.repository.internal.MavenRepositorySystemUtils; | ||||
| import org.eclipse.aether.DefaultRepositorySystemSession; | ||||
| import org.eclipse.aether.RepositorySystem; | ||||
| import org.eclipse.aether.artifact.Artifact; | ||||
| import org.eclipse.aether.artifact.DefaultArtifact; | ||||
| import org.eclipse.aether.collection.CollectRequest; | ||||
| import org.eclipse.aether.connector.basic.BasicRepositoryConnectorFactory; | ||||
| import org.eclipse.aether.graph.Dependency; | ||||
| import org.eclipse.aether.impl.DefaultServiceLocator; | ||||
| import org.eclipse.aether.repository.LocalRepository; | ||||
| import org.eclipse.aether.repository.RemoteRepository; | ||||
| import org.eclipse.aether.repository.RepositoryPolicy; | ||||
| import org.eclipse.aether.resolution.ArtifactResult; | ||||
| import org.eclipse.aether.resolution.DependencyRequest; | ||||
| import org.eclipse.aether.resolution.DependencyResolutionException; | ||||
| import org.eclipse.aether.resolution.DependencyResult; | ||||
| import org.eclipse.aether.spi.connector.RepositoryConnectorFactory; | ||||
| import org.eclipse.aether.spi.connector.transport.TransporterFactory; | ||||
| import org.eclipse.aether.transfer.AbstractTransferListener; | ||||
| import org.eclipse.aether.transfer.TransferCancelledException; | ||||
| import org.eclipse.aether.transfer.TransferEvent; | ||||
| import org.eclipse.aether.transport.http.HttpTransporterFactory; | ||||
|  | ||||
| class LibraryLoader | ||||
| { | ||||
|  | ||||
|     private static final String REPOSITORY_PROPERTY = "net.md_5.bungee.api.plugin.centralURL"; | ||||
|     private final Logger logger; | ||||
|     private final RepositorySystem repository; | ||||
|     private final DefaultRepositorySystemSession session; | ||||
|     private final List<RemoteRepository> repositories; | ||||
|  | ||||
|     public LibraryLoader(Logger logger) | ||||
|     { | ||||
|         this.logger = logger; | ||||
|  | ||||
|         DefaultServiceLocator locator = MavenRepositorySystemUtils.newServiceLocator(); | ||||
|         locator.addService( RepositoryConnectorFactory.class, BasicRepositoryConnectorFactory.class ); | ||||
|         locator.addService( TransporterFactory.class, HttpTransporterFactory.class ); | ||||
|  | ||||
|         this.repository = locator.getService( RepositorySystem.class ); | ||||
|         this.session = MavenRepositorySystemUtils.newSession(); | ||||
|  | ||||
|         session.setChecksumPolicy( RepositoryPolicy.CHECKSUM_POLICY_FAIL ); | ||||
|         session.setLocalRepositoryManager( repository.newLocalRepositoryManager( session, new LocalRepository( "libraries" ) ) ); | ||||
|         session.setTransferListener( new AbstractTransferListener() | ||||
|         { | ||||
|             @Override | ||||
|             public void transferStarted(TransferEvent event) throws TransferCancelledException | ||||
|             { | ||||
|                 logger.log( Level.INFO, "Downloading {0}", event.getResource().getRepositoryUrl() + event.getResource().getResourceName() ); | ||||
|             } | ||||
|         } ); | ||||
|  | ||||
|         // SPIGOT-7638: Add system properties, | ||||
|         // since JdkVersionProfileActivator needs 'java.version' when a profile has the 'jdk' element | ||||
|         // otherwise it will silently fail and not resolves the dependencies in the affected pom. | ||||
|         session.setSystemProperties( System.getProperties() ); | ||||
|         session.setReadOnly(); | ||||
|  | ||||
|         this.repositories = repository.newResolutionRepositories( session, Arrays.asList( new RemoteRepository.Builder( "central", "default", System.getProperty( REPOSITORY_PROPERTY, "https://repo.maven.apache.org/maven2" ) ).build() ) ); | ||||
|     } | ||||
|  | ||||
|     public ClassLoader createLoader(PluginDescription desc) | ||||
|     { | ||||
|         if ( desc.getLibraries().isEmpty() ) | ||||
|         { | ||||
|             return null; | ||||
|         } | ||||
|         logger.log( Level.INFO, "[{0}] Loading {1} libraries... please wait", new Object[] | ||||
|         { | ||||
|             desc.getName(), desc.getLibraries().size() | ||||
|         } ); | ||||
|  | ||||
|         List<Dependency> dependencies = new ArrayList<>(); | ||||
|         for ( String library : desc.getLibraries() ) | ||||
|         { | ||||
|             Artifact artifact = new DefaultArtifact( library ); | ||||
|             Dependency dependency = new Dependency( artifact, null ); | ||||
|  | ||||
|             dependencies.add( dependency ); | ||||
|         } | ||||
|  | ||||
|         DependencyResult result; | ||||
|         try | ||||
|         { | ||||
|             result = repository.resolveDependencies( session, new DependencyRequest( new CollectRequest( (Dependency) null, dependencies, repositories ), null ) ); | ||||
|         } catch ( DependencyResolutionException ex ) | ||||
|         { | ||||
|             throw new RuntimeException( "Error resolving libraries", ex ); | ||||
|         } | ||||
|  | ||||
|         List<URL> jarFiles = new ArrayList<>(); | ||||
|         for ( ArtifactResult artifact : result.getArtifactResults() ) | ||||
|         { | ||||
|             File file = artifact.getArtifact().getFile(); | ||||
|  | ||||
|             URL url; | ||||
|             try | ||||
|             { | ||||
|                 url = file.toURI().toURL(); | ||||
|             } catch ( MalformedURLException ex ) | ||||
|             { | ||||
|                 throw new AssertionError( ex ); | ||||
|             } | ||||
|  | ||||
|             jarFiles.add( url ); | ||||
|             logger.log( Level.INFO, "[{0}] Loaded library {1}", new Object[] | ||||
|             { | ||||
|                 desc.getName(), file | ||||
|             } ); | ||||
|         } | ||||
|  | ||||
|         URLClassLoader loader = new URLClassLoader( jarFiles.toArray( new URL[ 0 ] ) ); | ||||
|  | ||||
|         return loader; | ||||
|     } | ||||
| } | ||||
| @@ -11,6 +11,7 @@ import lombok.Getter; | ||||
| import net.md_5.bungee.api.ProxyServer; | ||||
| import net.md_5.bungee.api.config.ConfigurationAdapter; | ||||
| import net.md_5.bungee.api.scheduler.GroupedThreadFactory; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents any Plugin that may be loaded at runtime to enhance existing | ||||
| @@ -108,7 +109,14 @@ public class Plugin | ||||
|     // | ||||
|     private ExecutorService service; | ||||
|  | ||||
|     /** | ||||
|      * Returns the executor service associated with this plugin. | ||||
|      * | ||||
|      * @return the executor service for this plugin | ||||
|      * @deprecated internal API. Use {@link ProxyServer#getScheduler()} instead | ||||
|      */ | ||||
|     @Deprecated | ||||
|     @ApiStatus.Internal | ||||
|     public ExecutorService getExecutorService() | ||||
|     { | ||||
|         if ( service == null ) | ||||
|   | ||||
| @@ -1,12 +1,31 @@ | ||||
| package net.md_5.bungee.api.plugin; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.common.collect.ImmutableMap; | ||||
| import com.google.common.io.ByteStreams; | ||||
| import java.io.File; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import java.net.URL; | ||||
| import java.net.URLClassLoader; | ||||
| import java.security.CodeSigner; | ||||
| import java.security.CodeSource; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
| import java.util.concurrent.CopyOnWriteArraySet; | ||||
| import java.util.jar.JarEntry; | ||||
| import java.util.jar.JarFile; | ||||
| import java.util.jar.Manifest; | ||||
| import java.util.logging.Level; | ||||
| import java.util.logging.Logger; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.ProxyServer; | ||||
| import org.objectweb.asm.ClassReader; | ||||
| import org.objectweb.asm.ClassWriter; | ||||
| import org.objectweb.asm.commons.ClassRemapper; | ||||
| import org.objectweb.asm.commons.SimpleRemapper; | ||||
|  | ||||
| @ToString(of = "desc") | ||||
| final class PluginClassloader extends URLClassLoader | ||||
| { | ||||
|  | ||||
| @@ -14,6 +33,10 @@ final class PluginClassloader extends URLClassLoader | ||||
|     // | ||||
|     private final ProxyServer proxy; | ||||
|     private final PluginDescription desc; | ||||
|     private final JarFile jar; | ||||
|     private final Manifest manifest; | ||||
|     private final URL url; | ||||
|     private final ClassLoader libraryLoader; | ||||
|     // | ||||
|     private Plugin plugin; | ||||
|  | ||||
| @@ -22,11 +45,18 @@ final class PluginClassloader extends URLClassLoader | ||||
|         ClassLoader.registerAsParallelCapable(); | ||||
|     } | ||||
|  | ||||
|     public PluginClassloader(ProxyServer proxy, PluginDescription desc, URL[] urls) | ||||
|     public PluginClassloader(ProxyServer proxy, PluginDescription desc, File file, ClassLoader libraryLoader) throws IOException | ||||
|     { | ||||
|         super( urls ); | ||||
|         super( new URL[] | ||||
|         { | ||||
|             file.toURI().toURL() | ||||
|         } ); | ||||
|         this.proxy = proxy; | ||||
|         this.desc = desc; | ||||
|         this.jar = new JarFile( file ); | ||||
|         this.manifest = jar.getManifest(); | ||||
|         this.url = file.toURI().toURL(); | ||||
|         this.libraryLoader = libraryLoader; | ||||
|  | ||||
|         allLoaders.add( this ); | ||||
|     } | ||||
| @@ -34,17 +64,34 @@ final class PluginClassloader extends URLClassLoader | ||||
|     @Override | ||||
|     protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException | ||||
|     { | ||||
|         return loadClass0( name, resolve, true ); | ||||
|         return loadClass0( name, resolve, true, true ); | ||||
|     } | ||||
|  | ||||
|     private Class<?> loadClass0(String name, boolean resolve, boolean checkOther) throws ClassNotFoundException | ||||
|     private Class<?> loadClass0(String name, boolean resolve, boolean checkOther, boolean checkLibraries) throws ClassNotFoundException | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             return super.loadClass( name, resolve ); | ||||
|             Class<?> result = super.loadClass( name, resolve ); | ||||
|  | ||||
|             // SPIGOT-6749: Library classes will appear in the above, but we don't want to return them to other plugins | ||||
|             if ( checkOther || result.getClassLoader() == this ) | ||||
|             { | ||||
|                 return result; | ||||
|             } | ||||
|         } catch ( ClassNotFoundException ex ) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|         if ( checkLibraries && libraryLoader != null ) | ||||
|         { | ||||
|             try | ||||
|             { | ||||
|                 return libraryLoader.loadClass( name ); | ||||
|             } catch ( ClassNotFoundException ex ) | ||||
|             { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if ( checkOther ) | ||||
|         { | ||||
|             for ( PluginClassloader loader : allLoaders ) | ||||
| @@ -53,16 +100,111 @@ final class PluginClassloader extends URLClassLoader | ||||
|                 { | ||||
|                     try | ||||
|                     { | ||||
|                         return loader.loadClass0( name, resolve, false ); | ||||
|                         return loader.loadClass0( name, resolve, false, proxy.getPluginManager().isTransitiveDepend( desc, loader.desc ) ); | ||||
|                     } catch ( ClassNotFoundException ex ) | ||||
|                     { | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         throw new ClassNotFoundException( name ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Class<?> findClass(String name) throws ClassNotFoundException | ||||
|     { | ||||
|         String path = name.replace( '.', '/' ).concat( ".class" ); | ||||
|         JarEntry entry = jar.getJarEntry( path ); | ||||
|  | ||||
|         if ( entry != null ) | ||||
|         { | ||||
|             byte[] classBytes; | ||||
|  | ||||
|             try ( InputStream is = jar.getInputStream( entry ) ) | ||||
|             { | ||||
|                 classBytes = ByteStreams.toByteArray( is ); | ||||
|             } catch ( IOException ex ) | ||||
|             { | ||||
|                 throw new ClassNotFoundException( name, ex ); | ||||
|             } | ||||
|  | ||||
|             try | ||||
|             { | ||||
|                 classBytes = remap( classBytes ); | ||||
|             } catch ( Exception ex ) | ||||
|             { | ||||
|                 Logger logger = ( plugin != null ) ? plugin.getLogger() : proxy.getLogger(); | ||||
|                 logger.log( Level.SEVERE, "Error trying to remap class " + path, ex ); | ||||
|             } | ||||
|  | ||||
|             int dot = name.lastIndexOf( '.' ); | ||||
|             if ( dot != -1 ) | ||||
|             { | ||||
|                 String pkgName = name.substring( 0, dot ); | ||||
|                 if ( getPackage( pkgName ) == null ) | ||||
|                 { | ||||
|                     try | ||||
|                     { | ||||
|                         if ( manifest != null ) | ||||
|                         { | ||||
|                             definePackage( pkgName, manifest, url ); | ||||
|                         } else | ||||
|                         { | ||||
|                             definePackage( pkgName, null, null, null, null, null, null, null ); | ||||
|                         } | ||||
|                     } catch ( IllegalArgumentException ex ) | ||||
|                     { | ||||
|                         if ( getPackage( pkgName ) == null ) | ||||
|                         { | ||||
|                             throw new IllegalStateException( "Cannot find package " + pkgName ); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             CodeSigner[] signers = entry.getCodeSigners(); | ||||
|             CodeSource source = new CodeSource( url, signers ); | ||||
|  | ||||
|             return defineClass( name, classBytes, 0, classBytes.length, source ); | ||||
|         } | ||||
|  | ||||
|         return super.findClass( name ); | ||||
|     } | ||||
|  | ||||
|     private static final Map<String, String> MAPPINGS = ImmutableMap.of( | ||||
|             "net/md_5/bungee/protocol/ChatChain", "net/md_5/bungee/protocol/data/ChatChain", | ||||
|             "net/md_5/bungee/protocol/Location", "net/md_5/bungee/protocol/data/Location", | ||||
|             "net/md_5/bungee/protocol/NumberFormat", "net/md_5/bungee/protocol/data/NumberFormat", | ||||
|             "net/md_5/bungee/protocol/PlayerPublicKey", "net/md_5/bungee/protocol/data/PlayerPublicKey", | ||||
|             "net/md_5/bungee/protocol/Property", "net/md_5/bungee/protocol/data/Property", | ||||
|             "net/md_5/bungee/protocol/SeenMessages", "net/md_5/bungee/protocol/data/SeenMessages", | ||||
|             "net/md_5/bungee/protocol/Either", "net/md_5/bungee/protocol/util/Either", | ||||
|             "net/md_5/bungee/protocol/TagUtil", "net/md_5/bungee/protocol/util/TagUtil" | ||||
|     ); | ||||
|  | ||||
|     private static byte[] remap(byte[] b) | ||||
|     { | ||||
|         ClassReader cr = new ClassReader( b ); | ||||
|         ClassWriter cw = new ClassWriter( cr, 0 ); | ||||
|  | ||||
|         cr.accept( new ClassRemapper( cw, new SimpleRemapper( MAPPINGS ) ), 0 ); | ||||
|  | ||||
|         return cw.toByteArray(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void close() throws IOException | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             super.close(); | ||||
|         } finally | ||||
|         { | ||||
|             jar.close(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void init(Plugin plugin) | ||||
|     { | ||||
|         Preconditions.checkArgument( plugin != null, "plugin" ); | ||||
|   | ||||
| @@ -2,6 +2,8 @@ package net.md_5.bungee.api.plugin; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.util.HashSet; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Set; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| @@ -48,4 +50,8 @@ public class PluginDescription | ||||
|      * Optional description. | ||||
|      */ | ||||
|     private String description = null; | ||||
|     /** | ||||
|      * Optional libraries. | ||||
|      */ | ||||
|     private List<String> libraries = new LinkedList<>(); | ||||
| } | ||||
|   | ||||
| @@ -4,10 +4,12 @@ import com.google.common.base.Preconditions; | ||||
| import com.google.common.collect.ArrayListMultimap; | ||||
| import com.google.common.collect.Multimap; | ||||
| import com.google.common.eventbus.Subscribe; | ||||
| import com.google.common.graph.GraphBuilder; | ||||
| import com.google.common.graph.Graphs; | ||||
| import com.google.common.graph.MutableGraph; | ||||
| import java.io.File; | ||||
| import java.io.InputStream; | ||||
| import java.lang.reflect.Method; | ||||
| import java.net.URL; | ||||
| import java.net.URLClassLoader; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collection; | ||||
| @@ -21,9 +23,14 @@ import java.util.Locale; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
| import java.util.Stack; | ||||
| import java.util.concurrent.locks.Lock; | ||||
| import java.util.concurrent.locks.ReadWriteLock; | ||||
| import java.util.concurrent.locks.ReentrantLock; | ||||
| import java.util.concurrent.locks.ReentrantReadWriteLock; | ||||
| import java.util.jar.JarEntry; | ||||
| import java.util.jar.JarFile; | ||||
| import java.util.logging.Level; | ||||
| import lombok.Locked; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import net.md_5.bungee.api.ChatColor; | ||||
| import net.md_5.bungee.api.CommandSender; | ||||
| @@ -31,6 +38,7 @@ import net.md_5.bungee.api.ProxyServer; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.event.EventBus; | ||||
| import net.md_5.bungee.event.EventHandler; | ||||
| import org.yaml.snakeyaml.LoaderOptions; | ||||
| import org.yaml.snakeyaml.Yaml; | ||||
| import org.yaml.snakeyaml.constructor.Constructor; | ||||
| import org.yaml.snakeyaml.introspector.PropertyUtils; | ||||
| @@ -49,24 +57,40 @@ public final class PluginManager | ||||
|     private final Yaml yaml; | ||||
|     private final EventBus eventBus; | ||||
|     private final Map<String, Plugin> plugins = new LinkedHashMap<>(); | ||||
|     private final MutableGraph<String> dependencyGraph = GraphBuilder.directed().build(); | ||||
|     private final LibraryLoader libraryLoader; | ||||
|     private final Map<String, Command> commandMap = new HashMap<>(); | ||||
|     private Map<String, PluginDescription> toLoad = new HashMap<>(); | ||||
|     private final Multimap<Plugin, Command> commandsByPlugin = ArrayListMultimap.create(); | ||||
|     private final Multimap<Plugin, Listener> listenersByPlugin = ArrayListMultimap.create(); | ||||
|  | ||||
|     private final ReadWriteLock commandsLock = new ReentrantReadWriteLock(); | ||||
|     private final Lock listenersLock = new ReentrantLock(); | ||||
|  | ||||
|     @SuppressWarnings("unchecked") | ||||
|     public PluginManager(ProxyServer proxy) | ||||
|     { | ||||
|         this.proxy = proxy; | ||||
|  | ||||
|         // Ignore unknown entries in the plugin descriptions | ||||
|         Constructor yamlConstructor = new Constructor(); | ||||
|         Constructor yamlConstructor = new Constructor( new LoaderOptions() ); | ||||
|         PropertyUtils propertyUtils = yamlConstructor.getPropertyUtils(); | ||||
|         propertyUtils.setSkipMissingProperties( true ); | ||||
|         yamlConstructor.setPropertyUtils( propertyUtils ); | ||||
|         yaml = new Yaml( yamlConstructor ); | ||||
|  | ||||
|         eventBus = new EventBus( proxy.getLogger() ); | ||||
|  | ||||
|         LibraryLoader libraryLoader = null; | ||||
|         try | ||||
|         { | ||||
|             libraryLoader = new LibraryLoader( proxy.getLogger() ); | ||||
|         } catch ( NoClassDefFoundError ex ) | ||||
|         { | ||||
|             // Provided depends were not added back | ||||
|             proxy.getLogger().warning( "Could not initialize LibraryLoader (missing dependencies?)" ); | ||||
|         } | ||||
|         this.libraryLoader = libraryLoader; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -75,6 +99,7 @@ public final class PluginManager | ||||
|      * @param plugin the plugin owning this command | ||||
|      * @param command the command to register | ||||
|      */ | ||||
|     @Locked.Write("commandsLock") | ||||
|     public void registerCommand(Plugin plugin, Command command) | ||||
|     { | ||||
|         commandMap.put( command.getName().toLowerCase( Locale.ROOT ), command ); | ||||
| @@ -90,6 +115,7 @@ public final class PluginManager | ||||
|      * | ||||
|      * @param command the command to unregister | ||||
|      */ | ||||
|     @Locked.Write("commandsLock") | ||||
|     public void unregisterCommand(Command command) | ||||
|     { | ||||
|         while ( commandMap.values().remove( command ) ); | ||||
| @@ -101,6 +127,7 @@ public final class PluginManager | ||||
|      * | ||||
|      * @param plugin the plugin to register the commands of | ||||
|      */ | ||||
|     @Locked.Write("commandsLock") | ||||
|     public void unregisterCommands(Plugin plugin) | ||||
|     { | ||||
|         for ( Iterator<Command> it = commandsByPlugin.get( plugin ).iterator(); it.hasNext(); ) | ||||
| @@ -121,7 +148,14 @@ public final class PluginManager | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         commandsLock.readLock().lock(); | ||||
|         try | ||||
|         { | ||||
|             return commandMap.get( commandLower ); | ||||
|         } finally | ||||
|         { | ||||
|             commandsLock.readLock().unlock(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -309,6 +343,7 @@ public final class PluginManager | ||||
|                 status = false; | ||||
|             } | ||||
|  | ||||
|             dependencyGraph.putEdge( plugin.getName(), dependName ); | ||||
|             if ( !status ) | ||||
|             { | ||||
|                 break; | ||||
| @@ -320,10 +355,7 @@ public final class PluginManager | ||||
|         { | ||||
|             try | ||||
|             { | ||||
|                 URLClassLoader loader = new PluginClassloader( proxy, plugin, new URL[] | ||||
|                 { | ||||
|                     plugin.getFile().toURI().toURL() | ||||
|                 } ); | ||||
|                 URLClassLoader loader = new PluginClassloader( proxy, plugin, plugin.getFile(), ( libraryLoader != null ) ? libraryLoader.createLoader( plugin ) : null ); | ||||
|                 Class<?> main = loader.loadClass( plugin.getMain() ); | ||||
|                 Plugin clazz = (Plugin) main.getDeclaredConstructor().newInstance(); | ||||
|  | ||||
| @@ -335,7 +367,7 @@ public final class PluginManager | ||||
|                 } ); | ||||
|             } catch ( Throwable t ) | ||||
|             { | ||||
|                 proxy.getLogger().log( Level.WARNING, "Error enabling plugin " + plugin.getName(), t ); | ||||
|                 proxy.getLogger().log( Level.WARNING, "Error loading plugin " + plugin.getName(), t ); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -418,12 +450,12 @@ public final class PluginManager | ||||
|      * @param plugin the owning plugin | ||||
|      * @param listener the listener to register events for | ||||
|      */ | ||||
|     @Locked("listenersLock") | ||||
|     public void registerListener(Plugin plugin, Listener listener) | ||||
|     { | ||||
|         for ( Method method : listener.getClass().getDeclaredMethods() ) | ||||
|         { | ||||
|             Preconditions.checkArgument( !method.isAnnotationPresent( Subscribe.class ), | ||||
|                     "Listener %s has registered using deprecated subscribe annotation! Please update to @EventHandler.", listener ); | ||||
|             Preconditions.checkArgument( !method.isAnnotationPresent( Subscribe.class ), "Listener %s has registered using deprecated subscribe annotation! Please update to @EventHandler.", listener ); | ||||
|         } | ||||
|         eventBus.register( listener ); | ||||
|         listenersByPlugin.put( plugin, listener ); | ||||
| @@ -434,6 +466,7 @@ public final class PluginManager | ||||
|      * | ||||
|      * @param listener the listener to unregister | ||||
|      */ | ||||
|     @Locked("listenersLock") | ||||
|     public void unregisterListener(Listener listener) | ||||
|     { | ||||
|         eventBus.unregister( listener ); | ||||
| @@ -445,6 +478,7 @@ public final class PluginManager | ||||
|      * | ||||
|      * @param plugin target plugin | ||||
|      */ | ||||
|     @Locked("listenersLock") | ||||
|     public void unregisterListeners(Plugin plugin) | ||||
|     { | ||||
|         for ( Iterator<Listener> it = listenersByPlugin.get( plugin ).iterator(); it.hasNext(); ) | ||||
| @@ -459,8 +493,24 @@ public final class PluginManager | ||||
|      * | ||||
|      * @return commands | ||||
|      */ | ||||
|     @Locked.Read("commandsLock") | ||||
|     public Collection<Map.Entry<String, Command>> getCommands() | ||||
|     { | ||||
|         return Collections.unmodifiableCollection( commandMap.entrySet() ); | ||||
|     } | ||||
|  | ||||
|     boolean isTransitiveDepend(PluginDescription plugin, PluginDescription depend) | ||||
|     { | ||||
|         Preconditions.checkArgument( plugin != null, "plugin" ); | ||||
|         Preconditions.checkArgument( depend != null, "depend" ); | ||||
|  | ||||
|         if ( dependencyGraph.nodes().contains( plugin.getName() ) ) | ||||
|         { | ||||
|             if ( Graphs.reachableNodes( dependencyGraph, plugin.getName() ).contains( depend.getName() ) ) | ||||
|             { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,22 +1,37 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import gnu.trove.strategy.HashingStrategy; | ||||
| import it.unimi.dsi.fastutil.Hash; | ||||
| import java.util.Locale; | ||||
|  | ||||
| class CaseInsensitiveHashingStrategy implements HashingStrategy | ||||
| class CaseInsensitiveHashingStrategy implements Hash.Strategy<String> | ||||
| { | ||||
|  | ||||
|     static final CaseInsensitiveHashingStrategy INSTANCE = new CaseInsensitiveHashingStrategy(); | ||||
|  | ||||
|     @Override | ||||
|     public int computeHashCode(Object object) | ||||
|     public int hashCode(String object) | ||||
|     { | ||||
|         return ( (String) object ).toLowerCase( Locale.ROOT ).hashCode(); | ||||
|         if ( object == null ) | ||||
|         { | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         return object.toLowerCase( Locale.ROOT ).hashCode(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean equals(Object o1, Object o2) | ||||
|     public boolean equals(String o1, String o2) | ||||
|     { | ||||
|         return o1.equals( o2 ) || ( o1 instanceof String && o2 instanceof String && ( (String) o1 ).toLowerCase( Locale.ROOT ).equals( ( (String) o2 ).toLowerCase( Locale.ROOT ) ) ); | ||||
|         if ( o1 == o2 ) | ||||
|         { | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         if ( o1 == null || o2 == null ) | ||||
|         { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         return o1.equals( o2 ) || o1.toLowerCase( Locale.ROOT ).equals( o2.toLowerCase( Locale.ROOT ) ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import gnu.trove.map.hash.TCustomHashMap; | ||||
| import it.unimi.dsi.fastutil.objects.Object2ObjectOpenCustomHashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| public class CaseInsensitiveMap<V> extends TCustomHashMap<String, V> | ||||
| public class CaseInsensitiveMap<V> extends Object2ObjectOpenCustomHashMap<String, V> | ||||
| { | ||||
|  | ||||
|     public CaseInsensitiveMap() | ||||
| @@ -13,6 +13,6 @@ public class CaseInsensitiveMap<V> extends TCustomHashMap<String, V> | ||||
|  | ||||
|     public CaseInsensitiveMap(Map<? extends String, ? extends V> map) | ||||
|     { | ||||
|         super( CaseInsensitiveHashingStrategy.INSTANCE, map ); | ||||
|         super( map, CaseInsensitiveHashingStrategy.INSTANCE ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import gnu.trove.set.hash.TCustomHashSet; | ||||
| import it.unimi.dsi.fastutil.objects.ObjectOpenCustomHashSet; | ||||
| import java.util.Collection; | ||||
|  | ||||
| public class CaseInsensitiveSet extends TCustomHashSet<String> | ||||
| public class CaseInsensitiveSet extends ObjectOpenCustomHashSet<String> | ||||
| { | ||||
|  | ||||
|     public CaseInsensitiveSet() | ||||
| @@ -13,6 +13,6 @@ public class CaseInsensitiveSet extends TCustomHashSet<String> | ||||
|  | ||||
|     public CaseInsensitiveSet(Collection<? extends String> collection) | ||||
|     { | ||||
|         super( CaseInsensitiveHashingStrategy.INSTANCE, collection ); | ||||
|         super( collection, CaseInsensitiveHashingStrategy.INSTANCE ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| package net.md_5.bungee.api; | ||||
|  | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import java.net.InetSocketAddress; | ||||
| import java.net.SocketAddress; | ||||
| import java.util.Collection; | ||||
| import net.md_5.bungee.api.config.ServerInfo; | ||||
| import net.md_5.bungee.api.connection.ProxiedPlayer; | ||||
| import net.md_5.bungee.api.event.ServerConnectEvent; | ||||
| import org.junit.Test; | ||||
| import org.junit.jupiter.api.Test; | ||||
|  | ||||
| public class ServerConnectRequestTest | ||||
| { | ||||
| @@ -78,15 +79,15 @@ public class ServerConnectRequestTest | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     @Test(expected = NullPointerException.class) | ||||
|     @Test | ||||
|     public void testNullTarget() | ||||
|     { | ||||
|         ServerConnectRequest.builder().target( null ).reason( ServerConnectEvent.Reason.JOIN_PROXY ).build(); | ||||
|         assertThrows( NullPointerException.class, () -> ServerConnectRequest.builder().target( null ).reason( ServerConnectEvent.Reason.JOIN_PROXY ).build() ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = NullPointerException.class) | ||||
|     @Test | ||||
|     public void testNullReason() | ||||
|     { | ||||
|         ServerConnectRequest.builder().target( DUMMY_INFO ).reason( null ).build(); | ||||
|         assertThrows( NullPointerException.class, () -> ServerConnectRequest.builder().target( DUMMY_INFO ).reason( null ).build() ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,63 +1,38 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import io.netty.channel.unix.DomainSocketAddress; | ||||
| import java.net.InetSocketAddress; | ||||
| import java.net.SocketAddress; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collection; | ||||
| import java.util.stream.Stream; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import net.md_5.bungee.Util; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
| import org.junit.runner.RunWith; | ||||
| import org.junit.runners.Parameterized; | ||||
| import org.junit.runners.Parameterized.Parameters; | ||||
| import org.junit.jupiter.params.ParameterizedTest; | ||||
| import org.junit.jupiter.params.provider.Arguments; | ||||
| import org.junit.jupiter.params.provider.MethodSource; | ||||
|  | ||||
| @RequiredArgsConstructor | ||||
| @RunWith(Parameterized.class) | ||||
| public class AddressParseTest | ||||
| { | ||||
|  | ||||
|     @Parameters | ||||
|     public static Collection<Object[]> data() | ||||
|     public static Stream<Arguments> data() | ||||
|     { | ||||
|         return Arrays.asList( new Object[][] | ||||
|         { | ||||
|             { | ||||
|                 "127.0.0.1", "127.0.0.1", Util.DEFAULT_PORT | ||||
|             }, | ||||
|             { | ||||
|                 "127.0.0.1:1337", "127.0.0.1", 1337 | ||||
|             }, | ||||
|             { | ||||
|                 "[::1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT | ||||
|             }, | ||||
|             { | ||||
|                 "[0:0:0:0::1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT | ||||
|             }, | ||||
|             { | ||||
|                 "[0:0:0:0:0:0:0:1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT | ||||
|             }, | ||||
|             { | ||||
|                 "[::1]:1337", "0:0:0:0:0:0:0:1", 1337 | ||||
|             }, | ||||
|             { | ||||
|                 "[0:0:0:0::1]:1337", "0:0:0:0:0:0:0:1", 1337 | ||||
|             }, | ||||
|             { | ||||
|                 "[0:0:0:0:0:0:0:1]:1337", "0:0:0:0:0:0:0:1", 1337 | ||||
|             }, | ||||
|             { | ||||
|                 "unix:///var/run/bungee.sock", "/var/run/bungee.sock", -1 | ||||
|         return Stream.of( | ||||
|                 Arguments.of( "127.0.0.1", "127.0.0.1", Util.DEFAULT_PORT ), | ||||
|                 Arguments.of( "127.0.0.1:1337", "127.0.0.1", 1337 ), | ||||
|                 Arguments.of( "[::1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT ), | ||||
|                 Arguments.of( "[0:0:0:0::1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT ), | ||||
|                 Arguments.of( "[0:0:0:0:0:0:0:1]", "0:0:0:0:0:0:0:1", Util.DEFAULT_PORT ), | ||||
|                 Arguments.of( "[::1]:1337", "0:0:0:0:0:0:0:1", 1337 ), | ||||
|                 Arguments.of( "[0:0:0:0::1]:1337", "0:0:0:0:0:0:0:1", 1337 ), | ||||
|                 Arguments.of( "[0:0:0:0:0:0:0:1]:1337", "0:0:0:0:0:0:0:1", 1337 ), | ||||
|                 Arguments.of( "unix:///var/run/bungee.sock", "/var/run/bungee.sock", -1 ) | ||||
|         ); | ||||
|     } | ||||
|         } ); | ||||
|     } | ||||
|     private final String line; | ||||
|     private final String host; | ||||
|     private final int port; | ||||
|  | ||||
|     @Test | ||||
|     public void test() | ||||
|     @ParameterizedTest | ||||
|     @MethodSource("data") | ||||
|     public void test(String line, String host, int port) | ||||
|     { | ||||
|         SocketAddress parsed = Util.getAddr( line ); | ||||
|  | ||||
| @@ -65,14 +40,14 @@ public class AddressParseTest | ||||
|         { | ||||
|             InetSocketAddress tcp = (InetSocketAddress) parsed; | ||||
|  | ||||
|             Assert.assertEquals( host, tcp.getHostString() ); | ||||
|             Assert.assertEquals( port, tcp.getPort() ); | ||||
|             assertEquals( host, tcp.getHostString() ); | ||||
|             assertEquals( port, tcp.getPort() ); | ||||
|         } else if ( parsed instanceof DomainSocketAddress ) | ||||
|         { | ||||
|             DomainSocketAddress unix = (DomainSocketAddress) parsed; | ||||
|  | ||||
|             Assert.assertEquals( host, unix.path() ); | ||||
|             Assert.assertEquals( -1, port ); | ||||
|             assertEquals( host, unix.path() ); | ||||
|             assertEquals( -1, port ); | ||||
|         } else | ||||
|         { | ||||
|             throw new AssertionError( "Unknown socket " + parsed ); | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import org.junit.jupiter.api.Test; | ||||
|  | ||||
| public class CaseInsensitiveTest | ||||
| { | ||||
| @@ -13,12 +13,12 @@ public class CaseInsensitiveTest | ||||
|         CaseInsensitiveMap<Object> map = new CaseInsensitiveMap<>(); | ||||
|  | ||||
|         map.put( "FOO", obj ); | ||||
|         Assert.assertTrue( map.contains( "foo" ) ); // Assert that contains is case insensitive | ||||
|         Assert.assertTrue( map.entrySet().iterator().next().getKey().equals( "FOO" ) ); // Assert that case is preserved | ||||
|         assertTrue( map.containsKey( "foo" ) ); // Assert that contains is case insensitive | ||||
|         assertTrue( map.entrySet().iterator().next().getKey().equals( "FOO" ) ); // Assert that case is preserved | ||||
|  | ||||
|         // Assert that remove is case insensitive | ||||
|         map.remove( "FoO" ); | ||||
|         Assert.assertFalse( map.contains( "foo" ) ); | ||||
|         assertFalse( map.containsKey( "foo" ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
| @@ -27,8 +27,8 @@ public class CaseInsensitiveTest | ||||
|         CaseInsensitiveSet set = new CaseInsensitiveSet(); | ||||
|  | ||||
|         set.add( "FOO" ); | ||||
|         Assert.assertTrue( set.contains( "foo" ) ); // Assert that contains is case insensitive | ||||
|         assertTrue( set.contains( "foo" ) ); // Assert that contains is case insensitive | ||||
|         set.remove( "FoO" ); | ||||
|         Assert.assertFalse( set.contains( "foo" ) ); // Assert that remove is case insensitive | ||||
|         assertFalse( set.contains( "foo" ) ); // Assert that remove is case insensitive | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| package net.md_5.bungee.util; | ||||
|  | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import java.util.UUID; | ||||
| import net.md_5.bungee.Util; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
| import org.junit.jupiter.api.Test; | ||||
|  | ||||
| public class UUIDTest | ||||
| { | ||||
| @@ -13,7 +13,7 @@ public class UUIDTest | ||||
|     { | ||||
|         UUID uuid = UUID.fromString( "af74a02d-19cb-445b-b07f-6866a861f783" ); | ||||
|         UUID uuid1 = Util.getUUID( "af74a02d19cb445bb07f6866a861f783" ); | ||||
|         Assert.assertEquals( uuid, uuid1 ); | ||||
|         assertEquals( uuid, uuid1 ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
| @@ -23,7 +23,7 @@ public class UUIDTest | ||||
|         { | ||||
|             UUID expected = UUID.randomUUID(); | ||||
|             UUID actual = Util.getUUID( expected.toString().replace( "-", "" ) ); | ||||
|             Assert.assertEquals( "Could not parse UUID " + expected, expected, actual ); | ||||
|             assertEquals( expected, actual, "Could not parse UUID " + expected ); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -4,31 +4,29 @@ | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <parent> | ||||
|         <groupId>net.md-5</groupId> | ||||
|         <groupId>fr.pandacube.bungeecord</groupId> | ||||
|         <artifactId>bungeecord-parent</artifactId> | ||||
|         <version>1.16-R0.4-SNAPSHOT</version> | ||||
|         <version>1.21-R0.5-SNAPSHOT</version> | ||||
|         <relativePath>../pom.xml</relativePath> | ||||
|     </parent> | ||||
|  | ||||
|     <groupId>net.md-5</groupId> | ||||
|     <artifactId>bungeecord-bootstrap</artifactId> | ||||
|     <version>1.16-R0.4-SNAPSHOT</version> | ||||
|     <version>1.21-R0.5-SNAPSHOT</version> | ||||
|     <packaging>jar</packaging> | ||||
|  | ||||
|     <name>BungeeCord-Bootstrap</name> | ||||
|     <description>Java 1.6 loader for BungeeCord</description> | ||||
|  | ||||
|     <properties> | ||||
|         <skipPublishing>true</skipPublishing> | ||||
|         <maven.deploy.skip>true</maven.deploy.skip> | ||||
|         <maven.javadoc.skip>true</maven.javadoc.skip> | ||||
|         <maven.compiler.source>1.6</maven.compiler.source> | ||||
|         <maven.compiler.target>1.6</maven.compiler.target> | ||||
|         <maven.build.timestamp.format>yyyyMMdd</maven.build.timestamp.format> | ||||
|     </properties> | ||||
|  | ||||
|     <dependencies> | ||||
|         <dependency> | ||||
|             <groupId>net.md-5</groupId> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-proxy</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
| @@ -36,18 +34,19 @@ | ||||
|     </dependencies> | ||||
|  | ||||
|     <build> | ||||
|         <finalName>BungeeCord</finalName> | ||||
|         <finalName>BungeeCord-${project.version}-${build.number}</finalName> | ||||
|         <plugins> | ||||
|             <plugin> | ||||
|                 <groupId>org.apache.maven.plugins</groupId> | ||||
|                 <artifactId>maven-jar-plugin</artifactId> | ||||
|                 <version>3.2.0</version> | ||||
|                 <version>3.4.1</version> | ||||
|                 <configuration> | ||||
|                     <archive> | ||||
|                         <manifestEntries> | ||||
|                             <Main-Class>net.md_5.bungee.Bootstrap</Main-Class>  | ||||
|                             <Implementation-Version>${describe}</Implementation-Version> | ||||
|                             <Specification-Version>${maven.build.timestamp}</Specification-Version> | ||||
|                             <Enable-Native-Access>ALL-UNNAMED</Enable-Native-Access> | ||||
|                         </manifestEntries> | ||||
|                     </archive> | ||||
|                 </configuration> | ||||
| @@ -55,7 +54,7 @@ | ||||
|             <plugin> | ||||
|                 <groupId>org.apache.maven.plugins</groupId> | ||||
|                 <artifactId>maven-shade-plugin</artifactId> | ||||
|                 <version>3.2.3</version> | ||||
|                 <version>3.5.3</version> | ||||
|                 <executions> | ||||
|                     <execution> | ||||
|                         <phase>package</phase> | ||||
| @@ -79,4 +78,34 @@ | ||||
|             </plugin> | ||||
|         </plugins> | ||||
|     </build> | ||||
|  | ||||
|     <profiles> | ||||
|         <profile> | ||||
|             <id>jdk-9-release</id> | ||||
|             <activation> | ||||
|                 <jdk>[9,)</jdk> | ||||
|             </activation> | ||||
|             <properties> | ||||
|                 <maven.compiler.release>6</maven.compiler.release> | ||||
|             </properties> | ||||
|         </profile> | ||||
|         <profile> | ||||
|             <id>jdk-12-release</id> | ||||
|             <activation> | ||||
|                 <jdk>[12,)</jdk> | ||||
|             </activation> | ||||
|             <properties> | ||||
|                 <maven.compiler.release>7</maven.compiler.release> | ||||
|             </properties> | ||||
|         </profile> | ||||
|         <profile> | ||||
|             <id>jdk-20-release</id> | ||||
|             <activation> | ||||
|                 <jdk>[20,)</jdk> | ||||
|             </activation> | ||||
|             <properties> | ||||
|                 <maven.compiler.release>8</maven.compiler.release> | ||||
|             </properties> | ||||
|         </profile> | ||||
|     </profiles> | ||||
| </project> | ||||
|   | ||||
| @@ -4,15 +4,14 @@ | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <parent> | ||||
|         <groupId>net.md-5</groupId> | ||||
|         <groupId>fr.pandacube.bungeecord</groupId> | ||||
|         <artifactId>bungeecord-parent</artifactId> | ||||
|         <version>1.16-R0.4-SNAPSHOT</version> | ||||
|         <version>1.21-R0.5-SNAPSHOT</version> | ||||
|         <relativePath>../pom.xml</relativePath> | ||||
|     </parent> | ||||
|  | ||||
|     <groupId>net.md-5</groupId> | ||||
|     <artifactId>bungeecord-chat</artifactId> | ||||
|     <version>1.16-R0.4-SNAPSHOT</version> | ||||
|     <version>1.21-R0.5-SNAPSHOT</version> | ||||
|     <packaging>jar</packaging> | ||||
|  | ||||
|     <name>BungeeCord-Chat</name> | ||||
| @@ -22,7 +21,7 @@ | ||||
|         <dependency> | ||||
|             <groupId>com.google.code.gson</groupId> | ||||
|             <artifactId>gson</artifactId> | ||||
|             <version>2.8.0</version> | ||||
|             <version>2.11.0</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
|   | ||||
| @@ -244,7 +244,7 @@ public final class ChatColor | ||||
|     public static ChatColor of(String string) | ||||
|     { | ||||
|         Preconditions.checkArgument( string != null, "string cannot be null" ); | ||||
|         if ( string.startsWith( "#" ) && string.length() == 7 ) | ||||
|         if ( string.length() == 7 && string.charAt( 0 ) == '#' ) | ||||
|         { | ||||
|             int rgb; | ||||
|             try | ||||
| @@ -300,7 +300,7 @@ public final class ChatColor | ||||
|     @Deprecated | ||||
|     public static ChatColor[] values() | ||||
|     { | ||||
|         return BY_CHAR.values().toArray( new ChatColor[ BY_CHAR.values().size() ] ); | ||||
|         return BY_CHAR.values().toArray( new ChatColor[ 0 ] ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -1,8 +1,10 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import java.awt.Color; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Getter; | ||||
| import lombok.Setter; | ||||
| @@ -20,38 +22,10 @@ public abstract class BaseComponent | ||||
|     BaseComponent parent; | ||||
|  | ||||
|     /** | ||||
|      * The color of this component and any child components (unless overridden) | ||||
|      * The component's style. | ||||
|      */ | ||||
|     private ChatColor color; | ||||
|     /** | ||||
|      * The font of this component and any child components (unless overridden) | ||||
|      */ | ||||
|     private String font; | ||||
|     /** | ||||
|      * Whether this component and any child components (unless overridden) is | ||||
|      * bold | ||||
|      */ | ||||
|     private Boolean bold; | ||||
|     /** | ||||
|      * Whether this component and any child components (unless overridden) is | ||||
|      * italic | ||||
|      */ | ||||
|     private Boolean italic; | ||||
|     /** | ||||
|      * Whether this component and any child components (unless overridden) is | ||||
|      * underlined | ||||
|      */ | ||||
|     private Boolean underlined; | ||||
|     /** | ||||
|      * Whether this component and any child components (unless overridden) is | ||||
|      * strikethrough | ||||
|      */ | ||||
|     private Boolean strikethrough; | ||||
|     /** | ||||
|      * Whether this component and any child components (unless overridden) is | ||||
|      * obfuscated | ||||
|      */ | ||||
|     private Boolean obfuscated; | ||||
|     @Getter | ||||
|     private ComponentStyle style = new ComponentStyle(); | ||||
|     /** | ||||
|      * The text to insert into the chat when this component (and child | ||||
|      * components) are clicked while pressing the shift key | ||||
| @@ -78,6 +52,12 @@ public abstract class BaseComponent | ||||
|     @Getter | ||||
|     private HoverEvent hoverEvent; | ||||
|  | ||||
|     /** | ||||
|      * Whether this component rejects previous formatting | ||||
|      */ | ||||
|     @Getter | ||||
|     private transient boolean reset; | ||||
|  | ||||
|     /** | ||||
|      * Default constructor. | ||||
|      * | ||||
| @@ -147,31 +127,35 @@ public abstract class BaseComponent | ||||
|         } | ||||
|         if ( retention == FormatRetention.FORMATTING || retention == FormatRetention.ALL ) | ||||
|         { | ||||
|             if ( replace || color == null ) | ||||
|             if ( replace || !style.hasColor() ) | ||||
|             { | ||||
|                 setColor( component.getColorRaw() ); | ||||
|             } | ||||
|             if ( replace || font == null ) | ||||
|             if ( replace || !style.hasShadowColor() ) | ||||
|             { | ||||
|                 setShadowColor( component.getShadowColorRaw() ); | ||||
|             } | ||||
|             if ( replace || !style.hasFont() ) | ||||
|             { | ||||
|                 setFont( component.getFontRaw() ); | ||||
|             } | ||||
|             if ( replace || bold == null ) | ||||
|             if ( replace || style.isBoldRaw() == null ) | ||||
|             { | ||||
|                 setBold( component.isBoldRaw() ); | ||||
|             } | ||||
|             if ( replace || italic == null ) | ||||
|             if ( replace || style.isItalicRaw() == null ) | ||||
|             { | ||||
|                 setItalic( component.isItalicRaw() ); | ||||
|             } | ||||
|             if ( replace || underlined == null ) | ||||
|             if ( replace || style.isUnderlinedRaw() == null ) | ||||
|             { | ||||
|                 setUnderlined( component.isUnderlinedRaw() ); | ||||
|             } | ||||
|             if ( replace || strikethrough == null ) | ||||
|             if ( replace || style.isStrikethroughRaw() == null ) | ||||
|             { | ||||
|                 setStrikethrough( component.isStrikethroughRaw() ); | ||||
|             } | ||||
|             if ( replace || obfuscated == null ) | ||||
|             if ( replace || style.isObfuscatedRaw() == null ) | ||||
|             { | ||||
|                 setObfuscated( component.isObfuscatedRaw() ); | ||||
|             } | ||||
| @@ -197,6 +181,7 @@ public abstract class BaseComponent | ||||
|         if ( retention == FormatRetention.EVENTS || retention == FormatRetention.NONE ) | ||||
|         { | ||||
|             setColor( null ); | ||||
|             setShadowColor( null ); | ||||
|             setBold( null ); | ||||
|             setItalic( null ); | ||||
|             setUnderlined( null ); | ||||
| @@ -260,6 +245,32 @@ public abstract class BaseComponent | ||||
|         return builder.toString(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the {@link ComponentStyle} for this component. | ||||
|      * <p> | ||||
|      * Unlike {@link #applyStyle(ComponentStyle)}, this method will overwrite | ||||
|      * all style values on this component. | ||||
|      * | ||||
|      * @param style the style to set, or null to set all style values to default | ||||
|      */ | ||||
|     public void setStyle(ComponentStyle style) | ||||
|     { | ||||
|         this.style = ( style != null ) ? style.clone() : new ComponentStyle(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set this component's color. | ||||
|      * <p> | ||||
|      * <b>Warning: This should be a color, not formatting code (ie, | ||||
|      * {@link ChatColor#color} should not be null).</b> | ||||
|      * | ||||
|      * @param color the component color, or null to use the default | ||||
|      */ | ||||
|     public void setColor(ChatColor color) | ||||
|     { | ||||
|         this.style.setColor( color ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the color of this component. This uses the parent's color if this | ||||
|      * component doesn't have one. {@link net.md_5.bungee.api.ChatColor#WHITE} | ||||
| @@ -269,7 +280,7 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public ChatColor getColor() | ||||
|     { | ||||
|         if ( color == null ) | ||||
|         if ( !style.hasColor() ) | ||||
|         { | ||||
|             if ( parent == null ) | ||||
|             { | ||||
| @@ -277,7 +288,7 @@ public abstract class BaseComponent | ||||
|             } | ||||
|             return parent.getColor(); | ||||
|         } | ||||
|         return color; | ||||
|         return style.getColor(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -288,7 +299,58 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public ChatColor getColorRaw() | ||||
|     { | ||||
|         return color; | ||||
|         return style.getColor(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set this component's shadow color. | ||||
|      * | ||||
|      * @param color the component shadow color, or null to use the default | ||||
|      */ | ||||
|     public void setShadowColor(Color color) | ||||
|     { | ||||
|         this.style.setShadowColor( color ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the shadow color of this component. This uses the parent's shadow | ||||
|      * color if this component doesn't have one. null is returned if no shadow | ||||
|      * color is found. | ||||
|      * | ||||
|      * @return the shadow color of this component | ||||
|      */ | ||||
|     public Color getShadowColor() | ||||
|     { | ||||
|         if ( !style.hasShadowColor() ) | ||||
|         { | ||||
|             if ( parent == null ) | ||||
|             { | ||||
|                 return null; | ||||
|             } | ||||
|             return parent.getShadowColor(); | ||||
|         } | ||||
|         return style.getShadowColor(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the shadow color of this component without checking the parents | ||||
|      * shadow color. May return null | ||||
|      * | ||||
|      * @return the shadow color of this component | ||||
|      */ | ||||
|     public Color getShadowColorRaw() | ||||
|     { | ||||
|         return style.getShadowColor(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set this component's font. | ||||
|      * | ||||
|      * @param font the font to set, or null to use the default | ||||
|      */ | ||||
|     public void setFont(String font) | ||||
|     { | ||||
|         this.style.setFont( font ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -299,7 +361,7 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public String getFont() | ||||
|     { | ||||
|         if ( font == null ) | ||||
|         if ( !style.hasFont() ) | ||||
|         { | ||||
|             if ( parent == null ) | ||||
|             { | ||||
| @@ -307,7 +369,7 @@ public abstract class BaseComponent | ||||
|             } | ||||
|             return parent.getFont(); | ||||
|         } | ||||
|         return font; | ||||
|         return style.getFont(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -318,7 +380,17 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public String getFontRaw() | ||||
|     { | ||||
|         return font; | ||||
|         return style.getFont(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set whether or not this component is bold. | ||||
|      * | ||||
|      * @param bold the new bold state, or null to use the default | ||||
|      */ | ||||
|     public void setBold(Boolean bold) | ||||
|     { | ||||
|         this.style.setBold( bold ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -330,11 +402,11 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean isBold() | ||||
|     { | ||||
|         if ( bold == null ) | ||||
|         if ( style.isBoldRaw() == null ) | ||||
|         { | ||||
|             return parent != null && parent.isBold(); | ||||
|         } | ||||
|         return bold; | ||||
|         return style.isBold(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -345,7 +417,17 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public Boolean isBoldRaw() | ||||
|     { | ||||
|         return bold; | ||||
|         return style.isBoldRaw(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set whether or not this component is italic. | ||||
|      * | ||||
|      * @param italic the new italic state, or null to use the default | ||||
|      */ | ||||
|     public void setItalic(Boolean italic) | ||||
|     { | ||||
|         this.style.setItalic( italic ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -357,11 +439,11 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean isItalic() | ||||
|     { | ||||
|         if ( italic == null ) | ||||
|         if ( style.isItalicRaw() == null ) | ||||
|         { | ||||
|             return parent != null && parent.isItalic(); | ||||
|         } | ||||
|         return italic; | ||||
|         return style.isItalic(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -372,7 +454,17 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public Boolean isItalicRaw() | ||||
|     { | ||||
|         return italic; | ||||
|         return style.isItalicRaw(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set whether or not this component is underlined. | ||||
|      * | ||||
|      * @param underlined the new underlined state, or null to use the default | ||||
|      */ | ||||
|     public void setUnderlined(Boolean underlined) | ||||
|     { | ||||
|         this.style.setUnderlined( underlined ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -384,11 +476,11 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean isUnderlined() | ||||
|     { | ||||
|         if ( underlined == null ) | ||||
|         if ( style.isUnderlinedRaw() == null ) | ||||
|         { | ||||
|             return parent != null && parent.isUnderlined(); | ||||
|         } | ||||
|         return underlined; | ||||
|         return style.isUnderlined(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -399,7 +491,18 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public Boolean isUnderlinedRaw() | ||||
|     { | ||||
|         return underlined; | ||||
|         return style.isUnderlinedRaw(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set whether or not this component is strikethrough. | ||||
|      * | ||||
|      * @param strikethrough the new strikethrough state, or null to use the | ||||
|      * default | ||||
|      */ | ||||
|     public void setStrikethrough(Boolean strikethrough) | ||||
|     { | ||||
|         this.style.setStrikethrough( strikethrough ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -411,11 +514,11 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean isStrikethrough() | ||||
|     { | ||||
|         if ( strikethrough == null ) | ||||
|         if ( style.isStrikethroughRaw() == null ) | ||||
|         { | ||||
|             return parent != null && parent.isStrikethrough(); | ||||
|         } | ||||
|         return strikethrough; | ||||
|         return style.isStrikethrough(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -426,7 +529,17 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public Boolean isStrikethroughRaw() | ||||
|     { | ||||
|         return strikethrough; | ||||
|         return style.isStrikethroughRaw(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set whether or not this component is obfuscated. | ||||
|      * | ||||
|      * @param obfuscated the new obfuscated state, or null to use the default | ||||
|      */ | ||||
|     public void setObfuscated(Boolean obfuscated) | ||||
|     { | ||||
|         this.style.setObfuscated( obfuscated ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -438,11 +551,11 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean isObfuscated() | ||||
|     { | ||||
|         if ( obfuscated == null ) | ||||
|         if ( style.isObfuscatedRaw() == null ) | ||||
|         { | ||||
|             return parent != null && parent.isObfuscated(); | ||||
|         } | ||||
|         return obfuscated; | ||||
|         return style.isObfuscated(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -453,7 +566,52 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public Boolean isObfuscatedRaw() | ||||
|     { | ||||
|         return obfuscated; | ||||
|         return style.isObfuscatedRaw(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Apply the style from the given {@link ComponentStyle} to this component. | ||||
|      * <p> | ||||
|      * Any style values that have been explicitly set in the style will be | ||||
|      * applied to this component. If a value is not set in the style, it will | ||||
|      * not override the style set in this component. | ||||
|      * | ||||
|      * @param style the style to apply | ||||
|      */ | ||||
|     public void applyStyle(ComponentStyle style) | ||||
|     { | ||||
|         if ( style.hasColor() ) | ||||
|         { | ||||
|             setColor( style.getColor() ); | ||||
|         } | ||||
|         if ( style.hasShadowColor() ) | ||||
|         { | ||||
|             setShadowColor( style.getShadowColor() ); | ||||
|         } | ||||
|         if ( style.hasFont() ) | ||||
|         { | ||||
|             setFont( style.getFont() ); | ||||
|         } | ||||
|         if ( style.isBoldRaw() != null ) | ||||
|         { | ||||
|             setBold( style.isBoldRaw() ); | ||||
|         } | ||||
|         if ( style.isItalicRaw() != null ) | ||||
|         { | ||||
|             setItalic( style.isItalicRaw() ); | ||||
|         } | ||||
|         if ( style.isUnderlinedRaw() != null ) | ||||
|         { | ||||
|             setUnderlined( style.isUnderlinedRaw() ); | ||||
|         } | ||||
|         if ( style.isStrikethroughRaw() != null ) | ||||
|         { | ||||
|             setStrikethrough( style.isStrikethroughRaw() ); | ||||
|         } | ||||
|         if ( style.isObfuscatedRaw() != null ) | ||||
|         { | ||||
|             setObfuscated( style.isObfuscatedRaw() ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void setExtra(List<BaseComponent> components) | ||||
| @@ -492,6 +650,16 @@ public abstract class BaseComponent | ||||
|         extra.add( component ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether the component has any styling applied to it. | ||||
|      * | ||||
|      * @return Whether any styling is applied | ||||
|      */ | ||||
|     public boolean hasStyle() | ||||
|     { | ||||
|         return !style.isEmpty(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether the component has any formatting or events applied to it | ||||
|      * | ||||
| @@ -499,10 +667,8 @@ public abstract class BaseComponent | ||||
|      */ | ||||
|     public boolean hasFormatting() | ||||
|     { | ||||
|         return color != null || font != null || bold != null | ||||
|                 || italic != null || underlined != null | ||||
|                 || strikethrough != null || obfuscated != null | ||||
|                 || insertion != null || hoverEvent != null || clickEvent != null; | ||||
|         return hasStyle() || insertion != null | ||||
|                 || hoverEvent != null || clickEvent != null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -513,11 +679,11 @@ public abstract class BaseComponent | ||||
|     public String toPlainText() | ||||
|     { | ||||
|         StringBuilder builder = new StringBuilder(); | ||||
|         toPlainText( builder ); | ||||
|         toPlainText( new LimitedStringVisitor( builder, Short.MAX_VALUE ) ); | ||||
|         return builder.toString(); | ||||
|     } | ||||
|  | ||||
|     void toPlainText(StringBuilder builder) | ||||
|     void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         if ( extra != null ) | ||||
|         { | ||||
| @@ -537,11 +703,11 @@ public abstract class BaseComponent | ||||
|     public String toLegacyText() | ||||
|     { | ||||
|         StringBuilder builder = new StringBuilder(); | ||||
|         toLegacyText( builder ); | ||||
|         toLegacyText( new LimitedStringVisitor( builder, Short.MAX_VALUE ) ); | ||||
|         return builder.toString(); | ||||
|     } | ||||
|  | ||||
|     void toLegacyText(StringBuilder builder) | ||||
|     void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         if ( extra != null ) | ||||
|         { | ||||
| @@ -552,7 +718,7 @@ public abstract class BaseComponent | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void addFormat(StringBuilder builder) | ||||
|     void addFormat(StringVisitor builder) | ||||
|     { | ||||
|         builder.append( getColor() ); | ||||
|         if ( isBold() ) | ||||
| @@ -576,4 +742,35 @@ public abstract class BaseComponent | ||||
|             builder.append( ChatColor.MAGIC ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @FunctionalInterface | ||||
|     protected static interface StringVisitor | ||||
|     { | ||||
|  | ||||
|         void append(String s); | ||||
|  | ||||
|         default void append(Object obj) | ||||
|         { | ||||
|             append( String.valueOf( obj ) ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Data | ||||
|     protected static class LimitedStringVisitor implements StringVisitor | ||||
|     { | ||||
|  | ||||
|         private final StringBuilder builder; | ||||
|         private final int maxLength; | ||||
|  | ||||
|         @Override | ||||
|         public void append(String s) | ||||
|         { | ||||
|             if ( builder.length() >= maxLength ) | ||||
|             { | ||||
|                 throw new IllegalArgumentException( "String exceeded maximum length " + maxLength ); | ||||
|             } | ||||
|  | ||||
|             builder.append( s ); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -4,12 +4,14 @@ import lombok.EqualsAndHashCode; | ||||
| import lombok.Getter; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import lombok.ToString; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| @Getter | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @RequiredArgsConstructor | ||||
| public final class ClickEvent | ||||
| @ApiStatus.NonExtendable | ||||
| public class ClickEvent | ||||
| { | ||||
|  | ||||
|     /** | ||||
| @@ -52,11 +54,19 @@ public final class ClickEvent | ||||
|          * {@link net.md_5.bungee.api.chat.ClickEvent#value} in a book. | ||||
|          */ | ||||
|         CHANGE_PAGE, | ||||
|         /** | ||||
|          * Must use subclass ShowDialogClickEvent. | ||||
|          */ | ||||
|         SHOW_DIALOG, | ||||
|         /** | ||||
|          * Copy the string given by | ||||
|          * {@link net.md_5.bungee.api.chat.ClickEvent#value} into the player's | ||||
|          * clipboard. | ||||
|          */ | ||||
|         COPY_TO_CLIPBOARD | ||||
|         COPY_TO_CLIPBOARD, | ||||
|         /** | ||||
|          * Must use subclass {@link ClickEventCustom}. | ||||
|          */ | ||||
|         CUSTOM, | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,30 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.ToString; | ||||
|  | ||||
| /** | ||||
|  * Click event which sends a custom payload to the server. | ||||
|  */ | ||||
| @Data | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public class ClickEventCustom extends ClickEvent | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The custom payload. | ||||
|      */ | ||||
|     private final String payload; | ||||
|  | ||||
|     /** | ||||
|      * @param id identifier for the event (lower case, no special characters) | ||||
|      * @param payload custom payload | ||||
|      */ | ||||
|     public ClickEventCustom(String id, String payload) | ||||
|     { | ||||
|         super( ClickEvent.Action.CUSTOM, id ); | ||||
|         this.payload = payload; | ||||
|     } | ||||
| } | ||||
| @@ -1,6 +1,7 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import java.awt.Color; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import lombok.Getter; | ||||
| @@ -57,7 +58,7 @@ public final class ComponentBuilder | ||||
|      */ | ||||
|     public ComponentBuilder(ComponentBuilder original) | ||||
|     { | ||||
|         this( original.parts.toArray( new BaseComponent[ original.parts.size() ] ) ); | ||||
|         this( original.parts.toArray( new BaseComponent[ 0 ] ) ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -161,7 +162,7 @@ public final class ComponentBuilder | ||||
|             previous = dummy; | ||||
|             dummy = null; | ||||
|         } | ||||
|         if ( previous != null ) | ||||
|         if ( previous != null && !component.isReset() ) | ||||
|         { | ||||
|             component.copyFormatting( previous, retention, false ); | ||||
|         } | ||||
| @@ -204,6 +205,33 @@ public final class ComponentBuilder | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Appends the {@link TranslationProvider} object to the builder and makes | ||||
|      * the last element the current target for formatting. The components will | ||||
|      * have all the formatting from previous part. | ||||
|      * | ||||
|      * @param translatable the translatable object to append | ||||
|      * @return this ComponentBuilder for chaining | ||||
|      */ | ||||
|     public ComponentBuilder append(TranslationProvider translatable) | ||||
|     { | ||||
|         return append( translatable, FormatRetention.ALL ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Appends the {@link TranslationProvider} object to the builder and makes | ||||
|      * the last element the current target for formatting. You can specify the | ||||
|      * amount of formatting retained from previous part. | ||||
|      * | ||||
|      * @param translatable the translatable object to append | ||||
|      * @param retention the formatting to retain | ||||
|      * @return this ComponentBuilder for chaining | ||||
|      */ | ||||
|     public ComponentBuilder append(TranslationProvider translatable, FormatRetention retention) | ||||
|     { | ||||
|         return append( translatable.asTranslatableComponent(), retention ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Appends the text to the builder and makes it the current target for | ||||
|      * formatting. The text will have all the formatting from previous part. | ||||
| @@ -324,6 +352,19 @@ public final class ComponentBuilder | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the shadow color of the current part. | ||||
|      * | ||||
|      * @param color the new shadow color | ||||
|      * @return this ComponentBuilder for chaining | ||||
|      * @since Minecraft 1.21.4-pre1 | ||||
|      */ | ||||
|     public ComponentBuilder shadowColor(Color color) | ||||
|     { | ||||
|         getCurrentComponent().setShadowColor( color ); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the font of the current part. | ||||
|      * | ||||
| @@ -396,6 +437,18 @@ public final class ComponentBuilder | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Applies the provided {@link ComponentStyle} to the current part. | ||||
|      * | ||||
|      * @param style the style to apply | ||||
|      * @return this ComponentBuilder for chaining | ||||
|      */ | ||||
|     public ComponentBuilder style(ComponentStyle style) | ||||
|     { | ||||
|         getCurrentComponent().applyStyle( style ); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the insertion text for the current part. | ||||
|      * | ||||
| @@ -454,9 +507,32 @@ public final class ComponentBuilder | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the component built by this builder. If this builder is empty, an | ||||
|      * empty text component will be returned. | ||||
|      * | ||||
|      * @return the component | ||||
|      */ | ||||
|     public BaseComponent build() | ||||
|     { | ||||
|         TextComponent base = new TextComponent(); | ||||
|         if ( !parts.isEmpty() ) | ||||
|         { | ||||
|             List<BaseComponent> cloned = new ArrayList<>( parts ); | ||||
|             cloned.replaceAll( BaseComponent::duplicate ); | ||||
|             base.setExtra( cloned ); | ||||
|         } | ||||
|         return base; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the components needed to display the message created by this | ||||
|      * builder.git | ||||
|      * <p> | ||||
|      * <strong>NOTE:</strong> {@link #build()} is preferred as it will | ||||
|      * consolidate all components into a single BaseComponent with extra | ||||
|      * contents as opposed to an array of components which is non-standard and | ||||
|      * may result in unexpected behavior. | ||||
|      * | ||||
|      * @return the created components | ||||
|      */ | ||||
|   | ||||
							
								
								
									
										263
									
								
								chat/src/main/java/net/md_5/bungee/api/chat/ComponentStyle.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										263
									
								
								chat/src/main/java/net/md_5/bungee/api/chat/ComponentStyle.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,263 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import java.awt.Color; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.Setter; | ||||
| import net.md_5.bungee.api.ChatColor; | ||||
|  | ||||
| /** | ||||
|  * Represents a style that may be applied to a {@link BaseComponent}. | ||||
|  */ | ||||
| @Setter | ||||
| @AllArgsConstructor | ||||
| @NoArgsConstructor | ||||
| @EqualsAndHashCode | ||||
| public final class ComponentStyle implements Cloneable | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The color of this style. | ||||
|      * <p> | ||||
|      * <b>Warning: This should be a color, not formatting code (ie, | ||||
|      * {@link ChatColor#color} should not be null).</b> | ||||
|      */ | ||||
|     private ChatColor color; | ||||
|     /** | ||||
|      * The shadow color of this style. | ||||
|      */ | ||||
|     private Color shadowColor; | ||||
|     /** | ||||
|      * The font of this style. | ||||
|      */ | ||||
|     private String font; | ||||
|     /** | ||||
|      * Whether this style is bold. | ||||
|      */ | ||||
|     private Boolean bold; | ||||
|     /** | ||||
|      * Whether this style is italic. | ||||
|      */ | ||||
|     private Boolean italic; | ||||
|     /** | ||||
|      * Whether this style is underlined. | ||||
|      */ | ||||
|     private Boolean underlined; | ||||
|     /** | ||||
|      * Whether this style is strikethrough. | ||||
|      */ | ||||
|     private Boolean strikethrough; | ||||
|     /** | ||||
|      * Whether this style is obfuscated. | ||||
|      */ | ||||
|     private Boolean obfuscated; | ||||
|  | ||||
|     /** | ||||
|      * Returns the color of this style. May return null. | ||||
|      * | ||||
|      * @return the color of this style, or null if default color | ||||
|      */ | ||||
|     public ChatColor getColor() | ||||
|     { | ||||
|         return color; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether or not this style has a color set. | ||||
|      * | ||||
|      * @return whether a color is set | ||||
|      */ | ||||
|     public boolean hasColor() | ||||
|     { | ||||
|         return ( color != null ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the shadow color of this style. May return null. | ||||
|      * | ||||
|      * @return the shadow color of this style, or null if default color | ||||
|      */ | ||||
|     public Color getShadowColor() | ||||
|     { | ||||
|         return shadowColor; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether or not this style has a shadow color set. | ||||
|      * | ||||
|      * @return whether a shadow color is set | ||||
|      */ | ||||
|     public boolean hasShadowColor() | ||||
|     { | ||||
|         return ( shadowColor != null ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the font of this style. May return null. | ||||
|      * | ||||
|      * @return the font of this style, or null if default font | ||||
|      */ | ||||
|     public String getFont() | ||||
|     { | ||||
|         return font; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether or not this style has a font set. | ||||
|      * | ||||
|      * @return whether a font is set | ||||
|      */ | ||||
|     public boolean hasFont() | ||||
|     { | ||||
|         return ( font != null ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is bold. | ||||
|      * | ||||
|      * @return whether the style is bold | ||||
|      */ | ||||
|     public boolean isBold() | ||||
|     { | ||||
|         return ( bold != null ) && bold.booleanValue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is bold. May return null. | ||||
|      * | ||||
|      * @return whether the style is bold, or null if not set | ||||
|      */ | ||||
|     public Boolean isBoldRaw() | ||||
|     { | ||||
|         return bold; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is italic. May return null. | ||||
|      * | ||||
|      * @return whether the style is italic | ||||
|      */ | ||||
|     public boolean isItalic() | ||||
|     { | ||||
|         return ( italic != null ) && italic.booleanValue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is italic. May return null. | ||||
|      * | ||||
|      * @return whether the style is italic, or null if not set | ||||
|      */ | ||||
|     public Boolean isItalicRaw() | ||||
|     { | ||||
|         return italic; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is underlined. | ||||
|      * | ||||
|      * @return whether the style is underlined | ||||
|      */ | ||||
|     public boolean isUnderlined() | ||||
|     { | ||||
|         return ( underlined != null ) && underlined.booleanValue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is underlined. May return null. | ||||
|      * | ||||
|      * @return whether the style is underlined, or null if not set | ||||
|      */ | ||||
|     public Boolean isUnderlinedRaw() | ||||
|     { | ||||
|         return underlined; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is strikethrough | ||||
|      * | ||||
|      * @return whether the style is strikethrough | ||||
|      */ | ||||
|     public boolean isStrikethrough() | ||||
|     { | ||||
|         return ( strikethrough != null ) && strikethrough.booleanValue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is strikethrough. May return null. | ||||
|      * | ||||
|      * @return whether the style is strikethrough, or null if not set | ||||
|      */ | ||||
|     public Boolean isStrikethroughRaw() | ||||
|     { | ||||
|         return strikethrough; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is obfuscated. | ||||
|      * | ||||
|      * @return whether the style is obfuscated | ||||
|      */ | ||||
|     public boolean isObfuscated() | ||||
|     { | ||||
|         return ( obfuscated != null ) && obfuscated.booleanValue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style is obfuscated. May return null. | ||||
|      * | ||||
|      * @return whether the style is obfuscated, or null if not set | ||||
|      */ | ||||
|     public Boolean isObfuscatedRaw() | ||||
|     { | ||||
|         return obfuscated; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this style has no formatting explicitly set. | ||||
|      * | ||||
|      * @return true if no value is set, false if at least one is set | ||||
|      */ | ||||
|     public boolean isEmpty() | ||||
|     { | ||||
|         return color == null && shadowColor == null && font == null && bold == null | ||||
|                 && italic == null && underlined == null | ||||
|                 && strikethrough == null && obfuscated == null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public ComponentStyle clone() | ||||
|     { | ||||
|         return new ComponentStyle( color, shadowColor, font, bold, italic, underlined, strikethrough, obfuscated ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a new {@link ComponentStyleBuilder}. | ||||
|      * | ||||
|      * @return the builder | ||||
|      */ | ||||
|     public static ComponentStyleBuilder builder() | ||||
|     { | ||||
|         return new ComponentStyleBuilder(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a new {@link ComponentStyleBuilder} with values initialized to the | ||||
|      * style values of the supplied {@link ComponentStyle}. | ||||
|      * | ||||
|      * @param other the component style whose values to copy into the builder | ||||
|      * @return the builder | ||||
|      */ | ||||
|     public static ComponentStyleBuilder builder(ComponentStyle other) | ||||
|     { | ||||
|         return new ComponentStyleBuilder() | ||||
|                 .color( other.color ) | ||||
|                 .shadowColor( other.shadowColor ) | ||||
|                 .font( other.font ) | ||||
|                 .bold( other.bold ) | ||||
|                 .italic( other.italic ) | ||||
|                 .underlined( other.underlined ) | ||||
|                 .strikethrough( other.strikethrough ) | ||||
|                 .obfuscated( other.obfuscated ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,140 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import java.awt.Color; | ||||
| import net.md_5.bungee.api.ChatColor; | ||||
|  | ||||
| /** | ||||
|  * <p> | ||||
|  * ComponentStyleBuilder simplifies creating component styles by allowing the | ||||
|  * use of a chainable builder. | ||||
|  * </p> | ||||
|  * <pre> | ||||
|  * ComponentStyle style = ComponentStyle.builder() | ||||
|  *     .color(ChatColor.RED) | ||||
|  *     .font("custom:font") | ||||
|  *     .bold(true).italic(true).create(); | ||||
|  * | ||||
|  * BaseComponent component = new ComponentBuilder("Hello world").style(style).create(); | ||||
|  * // Or it can be used directly on a component | ||||
|  * TextComponent text = new TextComponent("Hello world"); | ||||
|  * text.applyStyle(style); | ||||
|  * </pre> | ||||
|  * | ||||
|  * @see ComponentStyle#builder() | ||||
|  * @see ComponentStyle#builder(ComponentStyle) | ||||
|  */ | ||||
| public final class ComponentStyleBuilder | ||||
| { | ||||
|  | ||||
|     private ChatColor color; | ||||
|     private Color shadowColor; | ||||
|     private String font; | ||||
|     private Boolean bold, italic, underlined, strikethrough, obfuscated; | ||||
|  | ||||
|     /** | ||||
|      * Set the style color. | ||||
|      * | ||||
|      * @param color the color to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder color(ChatColor color) | ||||
|     { | ||||
|         this.color = color; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style shadow color. | ||||
|      * | ||||
|      * @param shadowColor the shadow color to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder shadowColor(Color shadowColor) | ||||
|     { | ||||
|         this.shadowColor = shadowColor; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style font. | ||||
|      * | ||||
|      * @param font the font key to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder font(String font) | ||||
|     { | ||||
|         this.font = font; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style's bold property. | ||||
|      * | ||||
|      * @param bold the bold value to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder bold(Boolean bold) | ||||
|     { | ||||
|         this.bold = bold; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style's italic property. | ||||
|      * | ||||
|      * @param italic the italic value to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder italic(Boolean italic) | ||||
|     { | ||||
|         this.italic = italic; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style's underlined property. | ||||
|      * | ||||
|      * @param underlined the underlined value to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder underlined(Boolean underlined) | ||||
|     { | ||||
|         this.underlined = underlined; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style's strikethrough property. | ||||
|      * | ||||
|      * @param strikethrough the strikethrough value to set, or null to use the | ||||
|      * default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder strikethrough(Boolean strikethrough) | ||||
|     { | ||||
|         this.strikethrough = strikethrough; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the style's obfuscated property. | ||||
|      * | ||||
|      * @param obfuscated the obfuscated value to set, or null to use the default | ||||
|      * @return this ComponentStyleBuilder for chaining | ||||
|      */ | ||||
|     public ComponentStyleBuilder obfuscated(Boolean obfuscated) | ||||
|     { | ||||
|         this.obfuscated = obfuscated; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Build the {@link ComponentStyle} using the values set in this builder. | ||||
|      * | ||||
|      * @return the created ComponentStyle | ||||
|      */ | ||||
|     public ComponentStyle build() | ||||
|     { | ||||
|         return new ComponentStyle( color, shadowColor, font, bold, italic, underlined, strikethrough, obfuscated ); | ||||
|     } | ||||
| } | ||||
| @@ -13,7 +13,7 @@ import net.md_5.bungee.api.chat.hover.content.Content; | ||||
| import net.md_5.bungee.api.chat.hover.content.Entity; | ||||
| import net.md_5.bungee.api.chat.hover.content.Item; | ||||
| import net.md_5.bungee.api.chat.hover.content.Text; | ||||
| import net.md_5.bungee.chat.ComponentSerializer; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| @Getter | ||||
| @ToString | ||||
| @@ -34,6 +34,7 @@ public final class HoverEvent | ||||
|      * Returns whether this hover event is prior to 1.16 | ||||
|      */ | ||||
|     @Setter | ||||
|     @ApiStatus.Internal | ||||
|     private boolean legacy = false; | ||||
|  | ||||
|     /** | ||||
| @@ -71,22 +72,6 @@ public final class HoverEvent | ||||
|         this.legacy = true; | ||||
|     } | ||||
|  | ||||
|     @Deprecated | ||||
|     public BaseComponent[] getValue() | ||||
|     { | ||||
|         Content content = contents.get( 0 ); | ||||
|         if ( content instanceof Text && ( (Text) content ).getValue() instanceof BaseComponent[] ) | ||||
|         { | ||||
|             return (BaseComponent[]) ( (Text) content ).getValue(); | ||||
|         } | ||||
|  | ||||
|         TextComponent component = new TextComponent( ComponentSerializer.toString( content ) ); | ||||
|         return new BaseComponent[] | ||||
|         { | ||||
|             component | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a content to this hover event. | ||||
|      * | ||||
|   | ||||
| @@ -50,14 +50,14 @@ public final class KeybindComponent extends BaseComponent | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringBuilder builder) | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         builder.append( getKeybind() ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringBuilder builder) | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         addFormat( builder ); | ||||
|         builder.append( getKeybind() ); | ||||
|   | ||||
| @@ -0,0 +1,73 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.Getter; | ||||
| import lombok.NonNull; | ||||
| import lombok.Setter; | ||||
| import lombok.ToString; | ||||
| import net.md_5.bungee.api.chat.objects.ChatObject; | ||||
|  | ||||
| /** | ||||
|  * An object component that can be used to display objects. | ||||
|  * <p> | ||||
|  * It can either display a player's head or an object by a specific sprite and | ||||
|  * an atlas. | ||||
|  * <p> | ||||
|  * Note: this was added in Minecraft 1.21.9. | ||||
|  */ | ||||
| @Getter | ||||
| @Setter | ||||
| @ToString | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public final class ObjectComponent extends BaseComponent | ||||
| { | ||||
|  | ||||
|     private ChatObject object; | ||||
|  | ||||
|     /** | ||||
|      * Creates a ObjectComponent from a given ChatObject. | ||||
|      * | ||||
|      * See {@link net.md_5.bungee.api.chat.objects.PlayerObject} and | ||||
|      * {@link net.md_5.bungee.api.chat.objects.SpriteObject}. | ||||
|      * | ||||
|      * @param object the ChatObject | ||||
|      */ | ||||
|     public ObjectComponent(@NonNull ChatObject object) | ||||
|     { | ||||
|         this.object = object; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates an object component from the original to clone it. | ||||
|      * | ||||
|      * @param original the original for the new score component | ||||
|      */ | ||||
|     public ObjectComponent(ObjectComponent original) | ||||
|     { | ||||
|         super( original ); | ||||
|         setObject( original.object ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public ObjectComponent duplicate() | ||||
|     { | ||||
|         return new ObjectComponent( this ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         // I guess we cannot convert this to plain text | ||||
|         // builder.append( this.value ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         addFormat( builder ); | ||||
|         // Same here... | ||||
|         // builder.append( this.value ); | ||||
|         super.toLegacyText( builder ); | ||||
|     } | ||||
| } | ||||
| @@ -85,14 +85,14 @@ public final class ScoreComponent extends BaseComponent | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringBuilder builder) | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         builder.append( this.value ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringBuilder builder) | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         addFormat( builder ); | ||||
|         builder.append( this.value ); | ||||
|   | ||||
| @@ -33,6 +33,13 @@ public final class SelectorComponent extends BaseComponent | ||||
|      */ | ||||
|     private String selector; | ||||
|  | ||||
|     /** | ||||
|      * The separator of multiple selected entities. | ||||
|      * <br> | ||||
|      * The default is {@code {"color": "gray", "text": ", "}}. | ||||
|      */ | ||||
|     private BaseComponent separator; | ||||
|  | ||||
|     /** | ||||
|      * Creates a selector component from the original to clone it. | ||||
|      * | ||||
| @@ -42,6 +49,17 @@ public final class SelectorComponent extends BaseComponent | ||||
|     { | ||||
|         super( original ); | ||||
|         setSelector( original.getSelector() ); | ||||
|         setSeparator( original.getSeparator() ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates a selector component from the selector | ||||
|      * | ||||
|      * @param selector the selector as a String | ||||
|      */ | ||||
|     public SelectorComponent(String selector) | ||||
|     { | ||||
|         setSelector( selector ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -51,14 +69,14 @@ public final class SelectorComponent extends BaseComponent | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringBuilder builder) | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         builder.append( this.selector ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringBuilder builder) | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         addFormat( builder ); | ||||
|         builder.append( this.selector ); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.function.Consumer; | ||||
| import java.util.regex.Matcher; | ||||
| import java.util.regex.Pattern; | ||||
| import lombok.AllArgsConstructor; | ||||
| @@ -27,6 +28,41 @@ public final class TextComponent extends BaseComponent | ||||
|      * @param message the text to convert | ||||
|      * @return the components needed to print the message to the client | ||||
|      */ | ||||
|     public static BaseComponent fromLegacy(String message) | ||||
|     { | ||||
|         return fromLegacy( message, ChatColor.WHITE ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Converts the old formatting system that used | ||||
|      * {@link net.md_5.bungee.api.ChatColor#COLOR_CHAR} into the new json based | ||||
|      * system. | ||||
|      * | ||||
|      * @param message the text to convert | ||||
|      * @param defaultColor color to use when no formatting is to be applied | ||||
|      * (i.e. after ChatColor.RESET). | ||||
|      * @return the components needed to print the message to the client | ||||
|      */ | ||||
|     public static BaseComponent fromLegacy(String message, ChatColor defaultColor) | ||||
|     { | ||||
|         ComponentBuilder componentBuilder = new ComponentBuilder(); | ||||
|         populateComponentStructure( message, defaultColor, componentBuilder::append ); | ||||
|         return componentBuilder.build(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Converts the old formatting system that used | ||||
|      * {@link net.md_5.bungee.api.ChatColor#COLOR_CHAR} into the new json based | ||||
|      * system. | ||||
|      * | ||||
|      * @param message the text to convert | ||||
|      * @return the components needed to print the message to the client | ||||
|      * @deprecated {@link #fromLegacy(String)} is preferred as it will | ||||
|      * consolidate all components into a single BaseComponent with extra | ||||
|      * contents as opposed to an array of components which is non-standard and | ||||
|      * may result in unexpected behavior. | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public static BaseComponent[] fromLegacyText(String message) | ||||
|     { | ||||
|         return fromLegacyText( message, ChatColor.WHITE ); | ||||
| @@ -41,10 +77,21 @@ public final class TextComponent extends BaseComponent | ||||
|      * @param defaultColor color to use when no formatting is to be applied | ||||
|      * (i.e. after ChatColor.RESET). | ||||
|      * @return the components needed to print the message to the client | ||||
|      * @deprecated {@link #fromLegacy(String, ChatColor)} is preferred as it | ||||
|      * will consolidate all components into a single BaseComponent with extra | ||||
|      * contents as opposed to an array of components which is non-standard and | ||||
|      * may result in unexpected behavior. | ||||
|      */ | ||||
|     @Deprecated | ||||
|     public static BaseComponent[] fromLegacyText(String message, ChatColor defaultColor) | ||||
|     { | ||||
|         ArrayList<BaseComponent> components = new ArrayList<BaseComponent>(); | ||||
|         ArrayList<BaseComponent> components = new ArrayList<>(); | ||||
|         populateComponentStructure( message, defaultColor, components::add ); | ||||
|         return components.toArray( new BaseComponent[ 0 ] ); | ||||
|     } | ||||
|  | ||||
|     private static void populateComponentStructure(String message, ChatColor defaultColor, Consumer<BaseComponent> appender) | ||||
|     { | ||||
|         StringBuilder builder = new StringBuilder(); | ||||
|         TextComponent component = new TextComponent(); | ||||
|         Matcher matcher = url.matcher( message ); | ||||
| @@ -94,7 +141,7 @@ public final class TextComponent extends BaseComponent | ||||
|                     component = new TextComponent( old ); | ||||
|                     old.setText( builder.toString() ); | ||||
|                     builder = new StringBuilder(); | ||||
|                     components.add( old ); | ||||
|                     appender.accept( old ); | ||||
|                 } | ||||
|                 if ( format == ChatColor.BOLD ) | ||||
|                 { | ||||
| @@ -111,19 +158,19 @@ public final class TextComponent extends BaseComponent | ||||
|                 } else if ( format == ChatColor.MAGIC ) | ||||
|                 { | ||||
|                     component.setObfuscated( true ); | ||||
|                 } else if ( format == ChatColor.RESET ) | ||||
|                 { | ||||
|                     format = defaultColor; | ||||
|                     component = new TextComponent(); | ||||
|                     component.setColor( format ); | ||||
|                 } else | ||||
|                 { | ||||
|                     if ( format == ChatColor.RESET ) | ||||
|                     { | ||||
|                         format = defaultColor; | ||||
|                     } | ||||
|                     component = new TextComponent(); | ||||
|                     component.setColor( format ); | ||||
|                     component.setReset( true ); | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|             int pos = message.indexOf( ' ', i ); | ||||
|             int pos = indexOfSpecial( message, i ); | ||||
|             if ( pos == -1 ) | ||||
|             { | ||||
|                 pos = message.length(); | ||||
| @@ -137,7 +184,7 @@ public final class TextComponent extends BaseComponent | ||||
|                     component = new TextComponent( old ); | ||||
|                     old.setText( builder.toString() ); | ||||
|                     builder = new StringBuilder(); | ||||
|                     components.add( old ); | ||||
|                     appender.accept( old ); | ||||
|                 } | ||||
|  | ||||
|                 TextComponent old = component; | ||||
| @@ -146,7 +193,7 @@ public final class TextComponent extends BaseComponent | ||||
|                 component.setText( urlString ); | ||||
|                 component.setClickEvent( new ClickEvent( ClickEvent.Action.OPEN_URL, | ||||
|                         urlString.startsWith( "http" ) ? urlString : "http://" + urlString ) ); | ||||
|                 components.add( component ); | ||||
|                 appender.accept( component ); | ||||
|                 i += pos - i - 1; | ||||
|                 component = old; | ||||
|                 continue; | ||||
| @@ -155,9 +202,43 @@ public final class TextComponent extends BaseComponent | ||||
|         } | ||||
|  | ||||
|         component.setText( builder.toString() ); | ||||
|         components.add( component ); | ||||
|         appender.accept( component ); | ||||
|     } | ||||
|  | ||||
|         return components.toArray( new BaseComponent[ components.size() ] ); | ||||
|     private static int indexOfSpecial(String message, int pos) | ||||
|     { | ||||
|         for ( int i = pos; i < message.length(); i++ ) | ||||
|         { | ||||
|             char c = message.charAt( i ); | ||||
|  | ||||
|             if ( c == ' ' || Character.isISOControl( c ) ) | ||||
|             { | ||||
|                 return i; | ||||
|             } | ||||
|         } | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Internal compatibility method to transform an array of components to a | ||||
|      * single component. | ||||
|      * | ||||
|      * @param components array | ||||
|      * @return single component | ||||
|      */ | ||||
|     public static BaseComponent fromArray(BaseComponent... components) | ||||
|     { | ||||
|         if ( components == null ) | ||||
|         { | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         if ( components.length == 1 ) | ||||
|         { | ||||
|             return components[0]; | ||||
|         } | ||||
|  | ||||
|         return new TextComponent( components ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -213,14 +294,14 @@ public final class TextComponent extends BaseComponent | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringBuilder builder) | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         builder.append( text ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringBuilder builder) | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         addFormat( builder ); | ||||
|         builder.append( text ); | ||||
| @@ -230,6 +311,6 @@ public final class TextComponent extends BaseComponent | ||||
|     @Override | ||||
|     public String toString() | ||||
|     { | ||||
|         return String.format( "TextComponent{text=%s, %s}", text, super.toString() ); | ||||
|         return "TextComponent{text=" + text + ", " + super.toString() + '}'; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -19,7 +19,7 @@ import net.md_5.bungee.chat.TranslationRegistry; | ||||
| public final class TranslatableComponent extends BaseComponent | ||||
| { | ||||
|  | ||||
|     private final Pattern format = Pattern.compile( "%(?:(\\d+)\\$)?([A-Za-z%]|$)" ); | ||||
|     private static final Pattern FORMAT = Pattern.compile( "%(?:(\\d+)\\$)?([A-Za-z%]|$)" ); | ||||
|  | ||||
|     /** | ||||
|      * The key into the Minecraft locale files to use for the translation. The | ||||
| @@ -30,6 +30,10 @@ public final class TranslatableComponent extends BaseComponent | ||||
|      * The components to substitute into the translation | ||||
|      */ | ||||
|     private List<BaseComponent> with; | ||||
|     /** | ||||
|      * The fallback, if the translation is not found | ||||
|      */ | ||||
|     private String fallback; | ||||
|  | ||||
|     /** | ||||
|      * Creates a translatable component from the original to clone it. | ||||
| @@ -40,10 +44,11 @@ public final class TranslatableComponent extends BaseComponent | ||||
|     { | ||||
|         super( original ); | ||||
|         setTranslate( original.getTranslate() ); | ||||
|         setFallback( original.getFallback() ); | ||||
|  | ||||
|         if ( original.getWith() != null ) | ||||
|         { | ||||
|             List<BaseComponent> temp = new ArrayList<BaseComponent>(); | ||||
|             List<BaseComponent> temp = new ArrayList<>(); | ||||
|             for ( BaseComponent baseComponent : original.getWith() ) | ||||
|             { | ||||
|                 temp.add( baseComponent.duplicate() ); | ||||
| @@ -82,6 +87,21 @@ public final class TranslatableComponent extends BaseComponent | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates a translatable component with the passed substitutions | ||||
|      * | ||||
|      * @param translatable the translatable object | ||||
|      * @param with the {@link java.lang.String}s and | ||||
|      * {@link net.md_5.bungee.api.chat.BaseComponent}s to use into the | ||||
|      * translation | ||||
|      * @see #translate | ||||
|      * @see #setWith(java.util.List) | ||||
|      */ | ||||
|     public TranslatableComponent(TranslationProvider translatable, Object... with) | ||||
|     { | ||||
|         this( translatable.getTranslationKey(), with ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates a duplicate of this TranslatableComponent. | ||||
|      * | ||||
| @@ -136,24 +156,29 @@ public final class TranslatableComponent extends BaseComponent | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toPlainText(StringBuilder builder) | ||||
|     protected void toPlainText(StringVisitor builder) | ||||
|     { | ||||
|         convert( builder, false ); | ||||
|         super.toPlainText( builder ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toLegacyText(StringBuilder builder) | ||||
|     protected void toLegacyText(StringVisitor builder) | ||||
|     { | ||||
|         convert( builder, true ); | ||||
|         super.toLegacyText( builder ); | ||||
|     } | ||||
|  | ||||
|     private void convert(StringBuilder builder, boolean applyFormat) | ||||
|     private void convert(StringVisitor builder, boolean applyFormat) | ||||
|     { | ||||
|         String trans = TranslationRegistry.INSTANCE.translate( translate ); | ||||
|  | ||||
|         Matcher matcher = format.matcher( trans ); | ||||
|         if ( trans.equals( translate ) && fallback != null ) | ||||
|         { | ||||
|             trans = fallback; | ||||
|         } | ||||
|  | ||||
|         Matcher matcher = FORMAT.matcher( trans ); | ||||
|         int position = 0; | ||||
|         int i = 0; | ||||
|         while ( matcher.find( position ) ) | ||||
|   | ||||
| @@ -0,0 +1,38 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| /** | ||||
|  * An object capable of being translated by the client in a | ||||
|  * {@link TranslatableComponent}. | ||||
|  */ | ||||
| public interface TranslationProvider | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * Get the translation key. | ||||
|      * | ||||
|      * @return the translation key | ||||
|      */ | ||||
|     String getTranslationKey(); | ||||
|  | ||||
|     /** | ||||
|      * Get this translatable object as a {@link TranslatableComponent}. | ||||
|      * | ||||
|      * @return the translatable component | ||||
|      */ | ||||
|     default TranslatableComponent asTranslatableComponent() | ||||
|     { | ||||
|         return asTranslatableComponent( (Object[]) null ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get this translatable object as a {@link TranslatableComponent}. | ||||
|      * | ||||
|      * @param with the {@link String Strings} and | ||||
|      * {@link BaseComponent BaseComponents} to use in the translation | ||||
|      * @return the translatable component | ||||
|      */ | ||||
|     default TranslatableComponent asTranslatableComponent(Object... with) | ||||
|     { | ||||
|         return new TranslatableComponent( this, with ); | ||||
|     } | ||||
| } | ||||
| @@ -1,40 +0,0 @@ | ||||
| package net.md_5.bungee.api.chat.hover.content; | ||||
|  | ||||
| import com.google.gson.JsonDeserializationContext; | ||||
| import com.google.gson.JsonDeserializer; | ||||
| import com.google.gson.JsonElement; | ||||
| import com.google.gson.JsonObject; | ||||
| import com.google.gson.JsonParseException; | ||||
| import com.google.gson.JsonSerializationContext; | ||||
| import com.google.gson.JsonSerializer; | ||||
| import java.lang.reflect.Type; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| public class EntitySerializer implements JsonSerializer<Entity>, JsonDeserializer<Entity> | ||||
| { | ||||
|  | ||||
|     @Override | ||||
|     public Entity deserialize(JsonElement element, Type type, JsonDeserializationContext context) throws JsonParseException | ||||
|     { | ||||
|         JsonObject value = element.getAsJsonObject(); | ||||
|  | ||||
|         return new Entity( | ||||
|                 ( value.has( "type" ) ) ? value.get( "type" ).getAsString() : null, | ||||
|                 value.get( "id" ).getAsString(), | ||||
|                 ( value.has( "name" ) ) ? context.deserialize( value.get( "name" ), BaseComponent.class ) : null | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public JsonElement serialize(Entity content, Type type, JsonSerializationContext context) | ||||
|     { | ||||
|         JsonObject object = new JsonObject(); | ||||
|         object.addProperty( "type", ( content.getType() != null ) ? content.getType() : "minecraft:pig" ); | ||||
|         object.addProperty( "id", content.getId() ); | ||||
|         if ( content.getName() != null ) | ||||
|         { | ||||
|             object.add( "name", context.serialize( content.getName() ) ); | ||||
|         } | ||||
|         return object; | ||||
|     } | ||||
| } | ||||
| @@ -23,6 +23,15 @@ public class Text extends Content | ||||
|         this.value = value; | ||||
|     } | ||||
|  | ||||
|     public Text(BaseComponent value) | ||||
|     { | ||||
|         // For legacy serialization reasons, this has to be an array of components | ||||
|         this( new BaseComponent[] | ||||
|         { | ||||
|             value | ||||
|         } ); | ||||
|     } | ||||
|  | ||||
|     public Text(String value) | ||||
|     { | ||||
|         this.value = value; | ||||
|   | ||||
| @@ -0,0 +1,5 @@ | ||||
| package net.md_5.bungee.api.chat.objects; | ||||
|  | ||||
| public interface ChatObject | ||||
| { | ||||
| } | ||||
| @@ -0,0 +1,39 @@ | ||||
| package net.md_5.bungee.api.chat.objects; | ||||
|  | ||||
| import java.util.UUID; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import net.md_5.bungee.api.chat.player.Profile; | ||||
| import net.md_5.bungee.api.chat.player.Property; | ||||
|  | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| public final class PlayerObject implements ChatObject | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The profile of the player. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private Profile profile; | ||||
|     /** | ||||
|      * If true, a hat layer will be rendered on the head. (default: true) | ||||
|      */ | ||||
|     private Boolean hat; | ||||
|  | ||||
|     public PlayerObject(@NonNull String name) | ||||
|     { | ||||
|         this.profile = new Profile( name ); | ||||
|     } | ||||
|  | ||||
|     public PlayerObject(@NonNull UUID uuid) | ||||
|     { | ||||
|         this.profile = new Profile( uuid ); | ||||
|     } | ||||
|  | ||||
|     public PlayerObject(@NonNull Property[] properties) | ||||
|     { | ||||
|         this.profile = new Profile( properties ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,21 @@ | ||||
| package net.md_5.bungee.api.chat.objects; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
|  | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| public final class SpriteObject implements ChatObject | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The namespaced ID of a sprite atlas, default value: minecraft:blocks. | ||||
|      */ | ||||
|     private String atlas; | ||||
|     /** | ||||
|      * The namespaced ID of a sprite in atlas, for example item/porkchop. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private String sprite; | ||||
| } | ||||
| @@ -0,0 +1,40 @@ | ||||
| package net.md_5.bungee.api.chat.player; | ||||
|  | ||||
| import java.util.UUID; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
|  | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| public class Profile | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The name of the profile. Can be null. | ||||
|      */ | ||||
|     private String name; | ||||
|     /** | ||||
|      * The UUID of the profile. Can be null. | ||||
|      */ | ||||
|     private UUID uuid; | ||||
|     /** | ||||
|      * The properties of the profile. Can be null. | ||||
|      */ | ||||
|     private Property[] properties; | ||||
|  | ||||
|     public Profile(@NonNull String name) | ||||
|     { | ||||
|         this( name, null, null ); | ||||
|     } | ||||
|  | ||||
|     public Profile(@NonNull UUID uuid) | ||||
|     { | ||||
|         this( null, uuid, null ); | ||||
|     } | ||||
|  | ||||
|     public Profile(@NonNull Property[] properties) | ||||
|     { | ||||
|         this( null, null, properties ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,22 @@ | ||||
| package net.md_5.bungee.api.chat.player; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
|  | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| public class Property | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     private String name; | ||||
|     @NonNull | ||||
|     private String value; | ||||
|     private String signature; | ||||
|  | ||||
|     public Property(@NonNull String name, @NonNull String value) | ||||
|     { | ||||
|         this( name, value, null ); | ||||
|     } | ||||
| } | ||||
| @@ -1,207 +0,0 @@ | ||||
| package net.md_5.bungee.chat; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.gson.JsonDeserializationContext; | ||||
| import com.google.gson.JsonElement; | ||||
| import com.google.gson.JsonObject; | ||||
| import com.google.gson.JsonParseException; | ||||
| import com.google.gson.JsonSerializationContext; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.IdentityHashMap; | ||||
| import java.util.Locale; | ||||
| import net.md_5.bungee.api.ChatColor; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| import net.md_5.bungee.api.chat.ClickEvent; | ||||
| import net.md_5.bungee.api.chat.HoverEvent; | ||||
| import net.md_5.bungee.api.chat.hover.content.Content; | ||||
|  | ||||
| public class BaseComponentSerializer | ||||
| { | ||||
|  | ||||
|     protected void deserialize(JsonObject object, BaseComponent component, JsonDeserializationContext context) | ||||
|     { | ||||
|         if ( object.has( "color" ) ) | ||||
|         { | ||||
|             component.setColor( ChatColor.of( object.get( "color" ).getAsString() ) ); | ||||
|         } | ||||
|         if ( object.has( "font" ) ) | ||||
|         { | ||||
|             component.setFont( object.get( "font" ).getAsString() ); | ||||
|         } | ||||
|         if ( object.has( "bold" ) ) | ||||
|         { | ||||
|             component.setBold( object.get( "bold" ).getAsBoolean() ); | ||||
|         } | ||||
|         if ( object.has( "italic" ) ) | ||||
|         { | ||||
|             component.setItalic( object.get( "italic" ).getAsBoolean() ); | ||||
|         } | ||||
|         if ( object.has( "underlined" ) ) | ||||
|         { | ||||
|             component.setUnderlined( object.get( "underlined" ).getAsBoolean() ); | ||||
|         } | ||||
|         if ( object.has( "strikethrough" ) ) | ||||
|         { | ||||
|             component.setStrikethrough( object.get( "strikethrough" ).getAsBoolean() ); | ||||
|         } | ||||
|         if ( object.has( "obfuscated" ) ) | ||||
|         { | ||||
|             component.setObfuscated( object.get( "obfuscated" ).getAsBoolean() ); | ||||
|         } | ||||
|         if ( object.has( "insertion" ) ) | ||||
|         { | ||||
|             component.setInsertion( object.get( "insertion" ).getAsString() ); | ||||
|         } | ||||
|         if ( object.has( "extra" ) ) | ||||
|         { | ||||
|             component.setExtra( Arrays.asList( context.<BaseComponent[]>deserialize( object.get( "extra" ), BaseComponent[].class ) ) ); | ||||
|         } | ||||
|  | ||||
|         //Events | ||||
|         if ( object.has( "clickEvent" ) ) | ||||
|         { | ||||
|             JsonObject event = object.getAsJsonObject( "clickEvent" ); | ||||
|             component.setClickEvent( new ClickEvent( | ||||
|                     ClickEvent.Action.valueOf( event.get( "action" ).getAsString().toUpperCase( Locale.ROOT ) ), | ||||
|                     ( event.has( "value" ) ) ? event.get( "value" ).getAsString() : "" ) ); | ||||
|         } | ||||
|         if ( object.has( "hoverEvent" ) ) | ||||
|         { | ||||
|             JsonObject event = object.getAsJsonObject( "hoverEvent" ); | ||||
|             HoverEvent hoverEvent = null; | ||||
|             HoverEvent.Action action = HoverEvent.Action.valueOf( event.get( "action" ).getAsString().toUpperCase( Locale.ROOT ) ); | ||||
|  | ||||
|             for ( String type : Arrays.asList( "value", "contents" ) ) | ||||
|             { | ||||
|                 if ( !event.has( type ) ) | ||||
|                 { | ||||
|                     continue; | ||||
|                 } | ||||
|                 JsonElement contents = event.get( type ); | ||||
|                 try | ||||
|                 { | ||||
|  | ||||
|                     // Plugins previously had support to pass BaseComponent[] into any action. | ||||
|                     // If the GSON is possible to be parsed as BaseComponent, attempt to parse as so. | ||||
|                     BaseComponent[] components; | ||||
|                     if ( contents.isJsonArray() ) | ||||
|                     { | ||||
|                         components = context.deserialize( contents, BaseComponent[].class ); | ||||
|                     } else | ||||
|                     { | ||||
|                         components = new BaseComponent[] | ||||
|                         { | ||||
|                                 context.deserialize( contents, BaseComponent.class ) | ||||
|                         }; | ||||
|                     } | ||||
|                     hoverEvent = new HoverEvent( action, components ); | ||||
|                 } catch ( JsonParseException ex ) | ||||
|                 { | ||||
|                     Content[] list; | ||||
|                     if ( contents.isJsonArray() ) | ||||
|                     { | ||||
|                         list = context.deserialize( contents, HoverEvent.getClass( action, true ) ); | ||||
|                     } else | ||||
|                     { | ||||
|                         list = new Content[] | ||||
|                         { | ||||
|                             context.deserialize( contents, HoverEvent.getClass( action, false ) ) | ||||
|                         }; | ||||
|                     } | ||||
|                     hoverEvent = new HoverEvent( action, new ArrayList<>( Arrays.asList( list ) ) ); | ||||
|                 } | ||||
|  | ||||
|                 // stop the loop as soon as either one is found | ||||
|                 break; | ||||
|             } | ||||
|             if ( hoverEvent != null ) | ||||
|             { | ||||
|                 component.setHoverEvent( hoverEvent ); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     protected void serialize(JsonObject object, BaseComponent component, JsonSerializationContext context) | ||||
|     { | ||||
|         boolean first = false; | ||||
|         if ( ComponentSerializer.serializedComponents.get() == null ) | ||||
|         { | ||||
|             first = true; | ||||
|             ComponentSerializer.serializedComponents.set( Collections.newSetFromMap( new IdentityHashMap<BaseComponent, Boolean>() ) ); | ||||
|         } | ||||
|         try | ||||
|         { | ||||
|             Preconditions.checkArgument( !ComponentSerializer.serializedComponents.get().contains( component ), "Component loop" ); | ||||
|             ComponentSerializer.serializedComponents.get().add( component ); | ||||
|             if ( component.getColorRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "color", component.getColorRaw().getName() ); | ||||
|             } | ||||
|             if ( component.getFontRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "font", component.getFontRaw() ); | ||||
|             } | ||||
|             if ( component.isBoldRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "bold", component.isBoldRaw() ); | ||||
|             } | ||||
|             if ( component.isItalicRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "italic", component.isItalicRaw() ); | ||||
|             } | ||||
|             if ( component.isUnderlinedRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "underlined", component.isUnderlinedRaw() ); | ||||
|             } | ||||
|             if ( component.isStrikethroughRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "strikethrough", component.isStrikethroughRaw() ); | ||||
|             } | ||||
|             if ( component.isObfuscatedRaw() != null ) | ||||
|             { | ||||
|                 object.addProperty( "obfuscated", component.isObfuscatedRaw() ); | ||||
|             } | ||||
|             if ( component.getInsertion() != null ) | ||||
|             { | ||||
|                 object.addProperty( "insertion", component.getInsertion() ); | ||||
|             } | ||||
|  | ||||
|             if ( component.getExtra() != null ) | ||||
|             { | ||||
|                 object.add( "extra", context.serialize( component.getExtra() ) ); | ||||
|             } | ||||
|  | ||||
|             //Events | ||||
|             if ( component.getClickEvent() != null ) | ||||
|             { | ||||
|                 JsonObject clickEvent = new JsonObject(); | ||||
|                 clickEvent.addProperty( "action", component.getClickEvent().getAction().toString().toLowerCase( Locale.ROOT ) ); | ||||
|                 clickEvent.addProperty( "value", component.getClickEvent().getValue() ); | ||||
|                 object.add( "clickEvent", clickEvent ); | ||||
|             } | ||||
|             if ( component.getHoverEvent() != null ) | ||||
|             { | ||||
|                 JsonObject hoverEvent = new JsonObject(); | ||||
|                 hoverEvent.addProperty( "action", component.getHoverEvent().getAction().toString().toLowerCase( Locale.ROOT ) ); | ||||
|                 if ( component.getHoverEvent().isLegacy() ) | ||||
|                 { | ||||
|                     hoverEvent.add( "value", context.serialize( component.getHoverEvent().getContents().get( 0 ) ) ); | ||||
|                 } else | ||||
|                 { | ||||
|                     hoverEvent.add( "contents", context.serialize( ( component.getHoverEvent().getContents().size() == 1 ) | ||||
|                             ? component.getHoverEvent().getContents().get( 0 ) : component.getHoverEvent().getContents() ) ); | ||||
|                 } | ||||
|                 object.add( "hoverEvent", hoverEvent ); | ||||
|             } | ||||
|         } finally | ||||
|         { | ||||
|             ComponentSerializer.serializedComponents.get().remove( component ); | ||||
|             if ( first ) | ||||
|             { | ||||
|                 ComponentSerializer.serializedComponents.set( null ); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,109 +0,0 @@ | ||||
| package net.md_5.bungee.chat; | ||||
|  | ||||
| import com.google.gson.Gson; | ||||
| import com.google.gson.GsonBuilder; | ||||
| import com.google.gson.JsonDeserializationContext; | ||||
| import com.google.gson.JsonDeserializer; | ||||
| import com.google.gson.JsonElement; | ||||
| import com.google.gson.JsonObject; | ||||
| import com.google.gson.JsonParseException; | ||||
| import com.google.gson.JsonParser; | ||||
| import java.lang.reflect.Type; | ||||
| import java.util.Set; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| import net.md_5.bungee.api.chat.ItemTag; | ||||
| import net.md_5.bungee.api.chat.KeybindComponent; | ||||
| import net.md_5.bungee.api.chat.ScoreComponent; | ||||
| import net.md_5.bungee.api.chat.SelectorComponent; | ||||
| import net.md_5.bungee.api.chat.TextComponent; | ||||
| import net.md_5.bungee.api.chat.TranslatableComponent; | ||||
| import net.md_5.bungee.api.chat.hover.content.Entity; | ||||
| import net.md_5.bungee.api.chat.hover.content.EntitySerializer; | ||||
| import net.md_5.bungee.api.chat.hover.content.Item; | ||||
| import net.md_5.bungee.api.chat.hover.content.ItemSerializer; | ||||
| import net.md_5.bungee.api.chat.hover.content.Text; | ||||
| import net.md_5.bungee.api.chat.hover.content.TextSerializer; | ||||
|  | ||||
| public class ComponentSerializer implements JsonDeserializer<BaseComponent> | ||||
| { | ||||
|  | ||||
|     private static final JsonParser JSON_PARSER = new JsonParser(); | ||||
|     private static final Gson gson = new GsonBuilder(). | ||||
|             registerTypeAdapter( BaseComponent.class, new ComponentSerializer() ). | ||||
|             registerTypeAdapter( TextComponent.class, new TextComponentSerializer() ). | ||||
|             registerTypeAdapter( TranslatableComponent.class, new TranslatableComponentSerializer() ). | ||||
|             registerTypeAdapter( KeybindComponent.class, new KeybindComponentSerializer() ). | ||||
|             registerTypeAdapter( ScoreComponent.class, new ScoreComponentSerializer() ). | ||||
|             registerTypeAdapter( SelectorComponent.class, new SelectorComponentSerializer() ). | ||||
|             registerTypeAdapter( Entity.class, new EntitySerializer() ). | ||||
|             registerTypeAdapter( Text.class, new TextSerializer() ). | ||||
|             registerTypeAdapter( Item.class, new ItemSerializer() ). | ||||
|             registerTypeAdapter( ItemTag.class, new ItemTag.Serializer() ). | ||||
|             create(); | ||||
|  | ||||
|     public static final ThreadLocal<Set<BaseComponent>> serializedComponents = new ThreadLocal<Set<BaseComponent>>(); | ||||
|  | ||||
|     public static BaseComponent[] parse(String json) | ||||
|     { | ||||
|         JsonElement jsonElement = JSON_PARSER.parse( json ); | ||||
|  | ||||
|         if ( jsonElement.isJsonArray() ) | ||||
|         { | ||||
|             return gson.fromJson( jsonElement, BaseComponent[].class ); | ||||
|         } else | ||||
|         { | ||||
|             return new BaseComponent[] | ||||
|             { | ||||
|                 gson.fromJson( jsonElement, BaseComponent.class ) | ||||
|             }; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static String toString(Object object) | ||||
|     { | ||||
|         return gson.toJson( object ); | ||||
|     } | ||||
|  | ||||
|     public static String toString(BaseComponent component) | ||||
|     { | ||||
|         return gson.toJson( component ); | ||||
|     } | ||||
|  | ||||
|     public static String toString(BaseComponent... components) | ||||
|     { | ||||
|         if ( components.length == 1 ) | ||||
|         { | ||||
|             return gson.toJson( components[0] ); | ||||
|         } else | ||||
|         { | ||||
|             return gson.toJson( new TextComponent( components ) ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public BaseComponent deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException | ||||
|     { | ||||
|         if ( json.isJsonPrimitive() ) | ||||
|         { | ||||
|             return new TextComponent( json.getAsString() ); | ||||
|         } | ||||
|         JsonObject object = json.getAsJsonObject(); | ||||
|         if ( object.has( "translate" ) ) | ||||
|         { | ||||
|             return context.deserialize( json, TranslatableComponent.class ); | ||||
|         } | ||||
|         if ( object.has( "keybind" ) ) | ||||
|         { | ||||
|             return context.deserialize( json, KeybindComponent.class ); | ||||
|         } | ||||
|         if ( object.has( "score" ) ) | ||||
|         { | ||||
|             return context.deserialize( json, ScoreComponent.class ); | ||||
|         } | ||||
|         if ( object.has( "selector" ) ) | ||||
|         { | ||||
|             return context.deserialize( json, SelectorComponent.class ); | ||||
|         } | ||||
|         return context.deserialize( json, TextComponent.class ); | ||||
|     } | ||||
| } | ||||
| @@ -1,11 +1,11 @@ | ||||
| package net.md_5.bungee.chat; | ||||
|  | ||||
| import com.google.common.base.Charsets; | ||||
| import com.google.gson.Gson; | ||||
| import com.google.gson.JsonElement; | ||||
| import com.google.gson.JsonObject; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStreamReader; | ||||
| import java.nio.charset.StandardCharsets; | ||||
| import java.util.HashMap; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| @@ -102,7 +102,7 @@ public final class TranslationRegistry | ||||
|  | ||||
|         public JsonProvider(String resourcePath) throws IOException | ||||
|         { | ||||
|             try ( InputStreamReader rd = new InputStreamReader( JsonProvider.class.getResourceAsStream( resourcePath ), Charsets.UTF_8 ) ) | ||||
|             try ( InputStreamReader rd = new InputStreamReader( JsonProvider.class.getResourceAsStream( resourcePath ), StandardCharsets.UTF_8 ) ) | ||||
|             { | ||||
|                 JsonObject obj = new Gson().fromJson( rd, JsonObject.class ); | ||||
|                 for ( Map.Entry<String, JsonElement> entries : obj.entrySet() ) | ||||
|   | ||||
| @@ -1,579 +0,0 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import java.awt.Color; | ||||
| import net.md_5.bungee.api.ChatColor; | ||||
| import net.md_5.bungee.api.chat.hover.content.Item; | ||||
| import net.md_5.bungee.api.chat.hover.content.Text; | ||||
| import net.md_5.bungee.chat.ComponentSerializer; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
|  | ||||
| public class ComponentsTest | ||||
| { | ||||
|  | ||||
|     public static void testDissembleReassemble(BaseComponent[] components) | ||||
|     { | ||||
|         String json = ComponentSerializer.toString( components ); | ||||
|         BaseComponent[] parsed = ComponentSerializer.parse( json ); | ||||
|         Assert.assertEquals( TextComponent.toLegacyText( parsed ), TextComponent.toLegacyText( components ) ); | ||||
|     } | ||||
|  | ||||
|     public static void testDissembleReassemble(String json) | ||||
|     { | ||||
|         BaseComponent[] parsed = ComponentSerializer.parse( json ); | ||||
|         Assert.assertEquals( json, ComponentSerializer.toString( parsed ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testItemParse() | ||||
|     { | ||||
|         // Declare all commonly used variables for reuse. | ||||
|         BaseComponent[] components; | ||||
|         TextComponent textComponent; | ||||
|         String json; | ||||
|  | ||||
|         textComponent = new TextComponent( "Test" ); | ||||
|         textComponent.setHoverEvent( new HoverEvent( HoverEvent.Action.SHOW_ITEM, new BaseComponent[] | ||||
|         { | ||||
|             new TextComponent( "{id:\"minecraft:netherrack\",Count:47b}" ) | ||||
|         } ) ); | ||||
|         testDissembleReassemble( new BaseComponent[] | ||||
|         { | ||||
|             textComponent | ||||
|         } ); | ||||
|         json = "{\"text\":\"Test\",\"hoverEvent\":{\"action\":\"show_item\",\"value\":[{\"text\":\"{id:\\\"minecraft:netherrack\\\",Count:47b}\"}]}}"; | ||||
|         testDissembleReassemble( json ); | ||||
|         ////////// | ||||
|         String hoverVal = "{\"text\":\"{id:\\\"minecraft:dirt\\\",Count:1b}\"}"; | ||||
|         json = "{\"extra\":[{\"text\":\"[\"},{\"extra\":[{\"translate\":\"block.minecraft.dirt\"}],\"text\":\"\"},{\"text\":\"]\"}],\"hoverEvent\":{\"action\":\"show_item\",\"value\":[" + hoverVal + "]},\"text\":\"\"}"; | ||||
|         components = ComponentSerializer.parse( json ); | ||||
|         Text contentText = ( (Text) components[0].getHoverEvent().getContents().get( 0 ) ); | ||||
|         Assert.assertEquals( hoverVal, ComponentSerializer.toString( (BaseComponent[]) contentText.getValue() ) ); | ||||
|         testDissembleReassemble( components ); | ||||
|         ////////// | ||||
|         TextComponent component1 = new TextComponent( "HoverableText" ); | ||||
|         String nbt = "{display:{Name:{text:Hello},Lore:[{text:Line_1},{text:Line_2}]},ench:[{id:49,lvl:5}],Unbreakable:1}}"; | ||||
|         Item contentItem = new Item( "minecraft:wood", 1, ItemTag.ofNbt( nbt ) ); | ||||
|         HoverEvent hoverEvent = new HoverEvent( HoverEvent.Action.SHOW_ITEM, contentItem ); | ||||
|         component1.setHoverEvent( hoverEvent ); | ||||
|         json = ComponentSerializer.toString( component1 ); | ||||
|         components = ComponentSerializer.parse( json ); | ||||
|         Item parsedContentItem = ( (Item) components[0].getHoverEvent().getContents().get( 0 ) ); | ||||
|         Assert.assertEquals( contentItem, parsedContentItem ); | ||||
|         Assert.assertEquals( contentItem.getCount(), parsedContentItem.getCount() ); | ||||
|         Assert.assertEquals( contentItem.getId(), parsedContentItem.getId() ); | ||||
|         Assert.assertEquals( nbt, parsedContentItem.getTag().getNbt() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testEmptyComponentBuilder() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|  | ||||
|         BaseComponent[] parts = builder.create(); | ||||
|         Assert.assertEquals( parts.length, 0 ); | ||||
|  | ||||
|         for ( int i = 0; i < 3; i++ ) | ||||
|         { | ||||
|             builder.append( "part:" + i ); | ||||
|             parts = builder.create(); | ||||
|             Assert.assertEquals( parts.length, i + 1 ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testDummyRetaining() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|         Assert.assertNotNull( builder.getCurrentComponent() ); | ||||
|         builder.color( ChatColor.GREEN ); | ||||
|         builder.append( "test ", ComponentBuilder.FormatRetention.ALL ); | ||||
|         Assert.assertEquals( builder.getCurrentComponent().getColor(), ChatColor.GREEN ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = IndexOutOfBoundsException.class) | ||||
|     public void testComponentGettingExceptions() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|         builder.getComponent( -1 ); | ||||
|         builder.getComponent( 0 ); | ||||
|         builder.getComponent( 1 ); | ||||
|         BaseComponent component = new TextComponent( "Hello" ); | ||||
|         builder.append( component ); | ||||
|         Assert.assertEquals( builder.getComponent( 0 ), component ); | ||||
|         builder.getComponent( 1 ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testComponentParting() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|         TextComponent apple = new TextComponent( "apple" ); | ||||
|         builder.append( apple ); | ||||
|         Assert.assertEquals( builder.getCurrentComponent(), apple ); | ||||
|         Assert.assertEquals( builder.getComponent( 0 ), apple ); | ||||
|  | ||||
|         TextComponent mango = new TextComponent( "mango" ); | ||||
|         TextComponent orange = new TextComponent( "orange" ); | ||||
|         builder.append( mango ); | ||||
|         builder.append( orange ); | ||||
|         builder.removeComponent( 1 ); // Removing mango | ||||
|         Assert.assertEquals( builder.getComponent( 0 ), apple ); | ||||
|         Assert.assertEquals( builder.getComponent( 1 ), orange ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testToLegacyFromLegacy() | ||||
|     { | ||||
|         String text = "§a§lHello §f§kworld§7!"; | ||||
|         Assert.assertEquals( text, TextComponent.toLegacyText( TextComponent.fromLegacyText( text ) ) ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = IndexOutOfBoundsException.class) | ||||
|     public void testComponentBuilderCursorInvalidPos() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|         builder.append( new TextComponent( "Apple, " ) ); | ||||
|         builder.append( new TextComponent( "Orange, " ) ); | ||||
|         builder.setCursor( -1 ); | ||||
|         builder.setCursor( 2 ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testComponentBuilderCursor() | ||||
|     { | ||||
|         TextComponent t1, t2, t3; | ||||
|         ComponentBuilder builder = new ComponentBuilder(); | ||||
|         Assert.assertEquals( builder.getCursor(), -1 ); | ||||
|         builder.append( t1 = new TextComponent( "Apple, " ) ); | ||||
|         Assert.assertEquals( builder.getCursor(), 0 ); | ||||
|         builder.append( t2 = new TextComponent( "Orange, " ) ); | ||||
|         builder.append( t3 = new TextComponent( "Mango, " ) ); | ||||
|         Assert.assertEquals( builder.getCursor(), 2 ); | ||||
|  | ||||
|         builder.setCursor( 0 ); | ||||
|         Assert.assertEquals( builder.getCurrentComponent(), t1 ); | ||||
|  | ||||
|         // Test that appending new components updates the position to the new list size | ||||
|         // after having previously set it to 0 (first component) | ||||
|         builder.append( new TextComponent( "and Grapefruit" ) ); | ||||
|         Assert.assertEquals( builder.getCursor(), 3 ); | ||||
|  | ||||
|         builder.setCursor( 0 ); | ||||
|         builder.resetCursor(); | ||||
|         Assert.assertEquals( builder.getCursor(), 3 ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testLegacyComponentBuilderAppend() | ||||
|     { | ||||
|         String text = "§a§lHello §r§kworld§7!"; | ||||
|         BaseComponent[] components = TextComponent.fromLegacyText( text ); | ||||
|         BaseComponent[] builderComponents = new ComponentBuilder().append( components ).create(); | ||||
|         Assert.assertArrayEquals( components, builderComponents ); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|     @Test | ||||
|     public void testItemTag() | ||||
|     { | ||||
|         TextComponent component = new TextComponent( "Hello world" ); | ||||
|         HoverEvent.ContentItem content = new HoverEvent.ContentItem(); | ||||
|         content.setId( "minecraft:diamond_sword" ); | ||||
|         content.setCount( 1 ); | ||||
|         content.setTag( ItemTag.builder() | ||||
|                 .ench( new ItemTag.Enchantment( 5, 16 ) ) | ||||
|                 .name( new TextComponent( "Sharp Sword" ) ) | ||||
|                 .unbreakable( true ) | ||||
|                 .lore( new ComponentBuilder( "Line1" ).create() ) | ||||
|                 .lore( new ComponentBuilder( "Line2" ).create() ) | ||||
|                 .build() ); | ||||
|         HoverEvent event = new HoverEvent( HoverEvent.Action.SHOW_ITEM, content ); | ||||
|         component.setHoverEvent( event ); | ||||
|         String serialised = ComponentSerializer.toString( component ); | ||||
|         BaseComponent[] deserialised = ComponentSerializer.parse( serialised ); | ||||
|         Assert.assertEquals( TextComponent.toLegacyText( deserialised ), TextComponent.toLegacyText( component ) ); | ||||
|     } | ||||
|      */ | ||||
|  | ||||
|     @Test | ||||
|     public void testModernShowAdvancement() | ||||
|     { | ||||
|         String advancement = "achievement.openInventory"; | ||||
|         // First do the text using the newer contents system | ||||
|         HoverEvent hoverEvent = new HoverEvent( | ||||
|                 HoverEvent.Action.SHOW_TEXT, | ||||
|                 new Text( advancement ) | ||||
|         ); | ||||
|         TextComponent component = new TextComponent( "test" ); | ||||
|         component.setHoverEvent( hoverEvent ); | ||||
|         Assert.assertEquals( component.getHoverEvent().getContents().size(), 1 ); | ||||
|         Assert.assertTrue( component.getHoverEvent().getContents().get( 0 ) instanceof Text ); | ||||
|         Assert.assertEquals( ( (Text) component.getHoverEvent().getContents().get( 0 ) ).getValue(), advancement ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testHoverEventContents() | ||||
|     { | ||||
|         // First do the text using the newer contents system | ||||
|         HoverEvent hoverEvent = new HoverEvent( | ||||
|                 HoverEvent.Action.SHOW_TEXT, | ||||
|                 new Text( new ComponentBuilder( "First" ).create() ), | ||||
|                 new Text( new ComponentBuilder( "Second" ).create() ) | ||||
|         ); | ||||
|  | ||||
|         TextComponent component = new TextComponent( "Sample text" ); | ||||
|         component.setHoverEvent( hoverEvent ); | ||||
|         Assert.assertEquals( hoverEvent.getContents().size(), 2 ); | ||||
|         Assert.assertFalse( hoverEvent.isLegacy() ); | ||||
|         String serialized = ComponentSerializer.toString( component ); | ||||
|         BaseComponent[] deserialized = ComponentSerializer.parse( serialized ); | ||||
|         Assert.assertEquals( component.getHoverEvent(), deserialized[0].getHoverEvent() ); | ||||
|  | ||||
|         // check the test still works with the value method | ||||
|         hoverEvent = new HoverEvent( HoverEvent.Action.SHOW_TEXT, new ComponentBuilder( "Sample text" ).create() ); | ||||
|         Assert.assertEquals( hoverEvent.getContents().size(), 1 ); | ||||
|         Assert.assertTrue( hoverEvent.isLegacy() ); | ||||
|         serialized = ComponentSerializer.toString( component ); | ||||
|         deserialized = ComponentSerializer.parse( serialized ); | ||||
|         Assert.assertEquals( component.getHoverEvent(), deserialized[0].getHoverEvent() ); | ||||
|  | ||||
|         // Test single content: | ||||
|         String json = "{\"italic\":true,\"color\":\"gray\",\"translate\":\"chat.type.admin\",\"with\":[{\"text\":\"@\"}" | ||||
|                 + ",{\"translate\":\"commands.give.success.single\",\"with\":[\"1\",{\"color\":\"white\"" | ||||
|                 + ",\"hoverEvent\":{\"action\":\"show_item\",\"contents\":{\"id\":\"minecraft:diamond_sword\",\"tag\":\"" | ||||
|                 + "{Damage:0,display:{Lore:['\\\"test lore'!\\\"'],Name:'\\\"test\\\"'}}\"}}," | ||||
|                 + "\"extra\":[{\"italic\":true,\"extra\":[{\"text\":\"test\"}],\"text\":\"\"},{\"text\":\"]\"}]," | ||||
|                 + "\"text\":\"[\"},{\"insertion\":\"Name\",\"clickEvent\":{\"action\":\"suggest_command\",\"value\":" | ||||
|                 + "\"/tell Name \"},\"hoverEvent\":{\"action\":\"show_entity\",\"contents\":" | ||||
|                 + "{\"type\":\"minecraft:player\",\"id\":\"00000000-0000-0000-0000-00000000000000\",\"name\":" | ||||
|                 + "{\"text\":\"Name\"}}},\"text\":\"Name\"}]}]}"; | ||||
|         testDissembleReassemble( ComponentSerializer.parse( json ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testFormatRetentionCopyFormatting() | ||||
|     { | ||||
|         TextComponent first = new TextComponent( "Hello" ); | ||||
|         first.setBold( true ); | ||||
|         first.setColor( ChatColor.RED ); | ||||
|         first.setClickEvent( new ClickEvent( ClickEvent.Action.RUN_COMMAND, "test" ) ); | ||||
|         first.setHoverEvent( new HoverEvent( HoverEvent.Action.SHOW_TEXT, new ComponentBuilder( "Test" ).create() ) ); | ||||
|  | ||||
|         TextComponent second = new TextComponent( " world" ); | ||||
|         second.copyFormatting( first, ComponentBuilder.FormatRetention.ALL, true ); | ||||
|         Assert.assertEquals( first.isBold(), second.isBold() ); | ||||
|         Assert.assertEquals( first.getColor(), second.getColor() ); | ||||
|         Assert.assertEquals( first.getClickEvent(), second.getClickEvent() ); | ||||
|         Assert.assertEquals( first.getHoverEvent(), second.getHoverEvent() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderClone() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder( "Hello " ).color( ChatColor.RED ).append( "world" ).color( ChatColor.DARK_RED ); | ||||
|         ComponentBuilder cloned = new ComponentBuilder( builder ); | ||||
|  | ||||
|         Assert.assertEquals( TextComponent.toLegacyText( builder.create() ), TextComponent.toLegacyText( cloned.create() ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderAppendMixedComponents() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder( "Hello " ); | ||||
|         TextComponent textComponent = new TextComponent( "world " ); | ||||
|         TranslatableComponent translatableComponent = new TranslatableComponent( "item.swordGold.name" ); | ||||
|         // array based BaseComponent append | ||||
|         builder.append( new BaseComponent[] | ||||
|         { | ||||
|             textComponent, | ||||
|             translatableComponent | ||||
|         } ); | ||||
|         ScoreComponent scoreComponent = new ScoreComponent( "myscore", "myobjective" ); | ||||
|         builder.append( scoreComponent ); // non array based BaseComponent append | ||||
|         BaseComponent[] components = builder.create(); | ||||
|         Assert.assertEquals( "Hello ", components[0].toPlainText() ); | ||||
|         Assert.assertEquals( textComponent.toPlainText(), components[1].toPlainText() ); | ||||
|         Assert.assertEquals( translatableComponent.toPlainText(), components[2].toPlainText() ); | ||||
|         Assert.assertEquals( scoreComponent.toPlainText(), components[3].toPlainText() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testScore() | ||||
|     { | ||||
|         BaseComponent[] component = ComponentSerializer.parse( "{\"score\":{\"name\":\"@p\",\"objective\":\"TEST\",\"value\":\"hello\"}}" ); | ||||
|         String text = ComponentSerializer.toString( component ); | ||||
|         BaseComponent[] reparsed = ComponentSerializer.parse( text ); | ||||
|  | ||||
|         Assert.assertArrayEquals( component, reparsed ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderAppend() | ||||
|     { | ||||
|         ClickEvent clickEvent = new ClickEvent( ClickEvent.Action.RUN_COMMAND, "/help " ); | ||||
|         HoverEvent hoverEvent = new HoverEvent( HoverEvent.Action.SHOW_TEXT, new ComponentBuilder( "Hello world" ).create() ); | ||||
|  | ||||
|         ComponentBuilder builder = new ComponentBuilder( "Hello " ).color( ChatColor.YELLOW ); | ||||
|         builder.append( new ComponentBuilder( "world!" ).color( ChatColor.GREEN ).event( hoverEvent ).event( clickEvent ).create() ); | ||||
|  | ||||
|         BaseComponent[] components = builder.create(); | ||||
|  | ||||
|         Assert.assertEquals( components[1].getHoverEvent(), hoverEvent ); | ||||
|         Assert.assertEquals( components[1].getClickEvent(), clickEvent ); | ||||
|         Assert.assertEquals( "Hello world!", BaseComponent.toPlainText( components ) ); | ||||
|         Assert.assertEquals( ChatColor.YELLOW + "Hello " + ChatColor.GREEN + "world!", BaseComponent.toLegacyText( components ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderAppendLegacy() | ||||
|     { | ||||
|         ComponentBuilder builder = new ComponentBuilder( "Hello " ).color( ChatColor.YELLOW ); | ||||
|         builder.appendLegacy( "§aworld!" ); | ||||
|  | ||||
|         BaseComponent[] components = builder.create(); | ||||
|  | ||||
|         Assert.assertEquals( "Hello world!", BaseComponent.toPlainText( components ) ); | ||||
|         Assert.assertEquals( ChatColor.YELLOW + "Hello " + ChatColor.GREEN + "world!", BaseComponent.toLegacyText( components ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBasicComponent() | ||||
|     { | ||||
|         TextComponent textComponent = new TextComponent( "Hello world" ); | ||||
|         textComponent.setColor( ChatColor.RED ); | ||||
|  | ||||
|         Assert.assertEquals( "Hello world", textComponent.toPlainText() ); | ||||
|         Assert.assertEquals( ChatColor.RED + "Hello world", textComponent.toLegacyText() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testLegacyConverter() | ||||
|     { | ||||
|         BaseComponent[] test1 = TextComponent.fromLegacyText( ChatColor.AQUA + "Aqua " + ChatColor.RED + ChatColor.BOLD + "RedBold" ); | ||||
|  | ||||
|         Assert.assertEquals( "Aqua RedBold", BaseComponent.toPlainText( test1 ) ); | ||||
|         Assert.assertEquals( ChatColor.AQUA + "Aqua " + ChatColor.RED + ChatColor.BOLD + "RedBold", BaseComponent.toLegacyText( test1 ) ); | ||||
|  | ||||
|         BaseComponent[] test2 = TextComponent.fromLegacyText( "Text http://spigotmc.org " + ChatColor.GREEN + "google.com/test" ); | ||||
|  | ||||
|         Assert.assertEquals( "Text http://spigotmc.org google.com/test", BaseComponent.toPlainText( test2 ) ); | ||||
|         //The extra ChatColor instances are sometimes inserted when not needed but it doesn't change the result | ||||
|         Assert.assertEquals( ChatColor.WHITE + "Text " + ChatColor.WHITE + "http://spigotmc.org" + ChatColor.WHITE | ||||
|                 + " " + ChatColor.GREEN + "google.com/test" + ChatColor.GREEN, BaseComponent.toLegacyText( test2 ) ); | ||||
|  | ||||
|         ClickEvent url1 = test2[1].getClickEvent(); | ||||
|         Assert.assertNotNull( url1 ); | ||||
|         Assert.assertTrue( url1.getAction() == ClickEvent.Action.OPEN_URL ); | ||||
|         Assert.assertEquals( "http://spigotmc.org", url1.getValue() ); | ||||
|  | ||||
|         ClickEvent url2 = test2[3].getClickEvent(); | ||||
|         Assert.assertNotNull( url2 ); | ||||
|         Assert.assertTrue( url2.getAction() == ClickEvent.Action.OPEN_URL ); | ||||
|         Assert.assertEquals( "http://google.com/test", url2.getValue() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testTranslateComponent() | ||||
|     { | ||||
|         TranslatableComponent item = new TranslatableComponent( "item.swordGold.name" ); | ||||
|         item.setColor( ChatColor.AQUA ); | ||||
|         TranslatableComponent translatableComponent = new TranslatableComponent( "commands.give.success", | ||||
|                 item, "5", | ||||
|                 "thinkofdeath" ); | ||||
|  | ||||
|         Assert.assertEquals( "Given Golden Sword * 5 to thinkofdeath", translatableComponent.toPlainText() ); | ||||
|         Assert.assertEquals( ChatColor.WHITE + "Given " + ChatColor.AQUA + "Golden Sword" + ChatColor.WHITE | ||||
|                 + " * " + ChatColor.WHITE + "5" + ChatColor.WHITE + " to " + ChatColor.WHITE + "thinkofdeath", | ||||
|                 translatableComponent.toLegacyText() ); | ||||
|  | ||||
|         TranslatableComponent positional = new TranslatableComponent( "book.pageIndicator", "5", "50" ); | ||||
|  | ||||
|         Assert.assertEquals( "Page 5 of 50", positional.toPlainText() ); | ||||
|         Assert.assertEquals( ChatColor.WHITE + "Page " + ChatColor.WHITE + "5" + ChatColor.WHITE + " of " + ChatColor.WHITE + "50", positional.toLegacyText() ); | ||||
|  | ||||
|         TranslatableComponent one_four_two = new TranslatableComponent( "filled_map.buried_treasure" ); | ||||
|         Assert.assertEquals( "Buried Treasure Map", one_four_two.toPlainText() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilder() | ||||
|     { | ||||
|         BaseComponent[] components = new ComponentBuilder( "Hello " ).color( ChatColor.RED ). | ||||
|                 append( "World" ).bold( true ).color( ChatColor.BLUE ). | ||||
|                 append( "!" ).color( ChatColor.YELLOW ).create(); | ||||
|  | ||||
|         Assert.assertEquals( "Hello World!", BaseComponent.toPlainText( components ) ); | ||||
|         Assert.assertEquals( ChatColor.RED + "Hello " + ChatColor.BLUE + ChatColor.BOLD | ||||
|                 + "World" + ChatColor.YELLOW + ChatColor.BOLD + "!", BaseComponent.toLegacyText( components ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderReset() | ||||
|     { | ||||
|         BaseComponent[] components = new ComponentBuilder( "Hello " ).color( ChatColor.RED ) | ||||
|                 .append( "World" ).reset().create(); | ||||
|  | ||||
|         Assert.assertEquals( components[0].getColor(), ChatColor.RED ); | ||||
|         Assert.assertEquals( components[1].getColor(), ChatColor.WHITE ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testBuilderFormatRetention() | ||||
|     { | ||||
|         BaseComponent[] noneRetention = new ComponentBuilder( "Hello " ).color( ChatColor.RED ) | ||||
|                 .append( "World", ComponentBuilder.FormatRetention.NONE ).create(); | ||||
|  | ||||
|         Assert.assertEquals( noneRetention[0].getColor(), ChatColor.RED ); | ||||
|         Assert.assertEquals( noneRetention[1].getColor(), ChatColor.WHITE ); | ||||
|  | ||||
|         HoverEvent testEvent = new HoverEvent( HoverEvent.Action.SHOW_TEXT, new ComponentBuilder( "test" ).create() ); | ||||
|  | ||||
|         BaseComponent[] formattingRetention = new ComponentBuilder( "Hello " ).color( ChatColor.RED ) | ||||
|                 .event( testEvent ).append( "World", ComponentBuilder.FormatRetention.FORMATTING ).create(); | ||||
|  | ||||
|         Assert.assertEquals( formattingRetention[0].getColor(), ChatColor.RED ); | ||||
|         Assert.assertEquals( formattingRetention[0].getHoverEvent(), testEvent ); | ||||
|         Assert.assertEquals( formattingRetention[1].getColor(), ChatColor.RED ); | ||||
|         Assert.assertNull( formattingRetention[1].getHoverEvent() ); | ||||
|  | ||||
|         ClickEvent testClickEvent = new ClickEvent( ClickEvent.Action.OPEN_URL, "http://www.example.com" ); | ||||
|  | ||||
|         BaseComponent[] eventRetention = new ComponentBuilder( "Hello " ).color( ChatColor.RED ) | ||||
|                 .event( testEvent ).event( testClickEvent ).append( "World", ComponentBuilder.FormatRetention.EVENTS ).create(); | ||||
|  | ||||
|         Assert.assertEquals( eventRetention[0].getColor(), ChatColor.RED ); | ||||
|         Assert.assertEquals( eventRetention[0].getHoverEvent(), testEvent ); | ||||
|         Assert.assertEquals( eventRetention[0].getClickEvent(), testClickEvent ); | ||||
|         Assert.assertEquals( eventRetention[1].getColor(), ChatColor.WHITE ); | ||||
|         Assert.assertEquals( eventRetention[1].getHoverEvent(), testEvent ); | ||||
|         Assert.assertEquals( eventRetention[1].getClickEvent(), testClickEvent ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = IllegalArgumentException.class) | ||||
|     public void testLoopSimple() | ||||
|     { | ||||
|         TextComponent component = new TextComponent( "Testing" ); | ||||
|         component.addExtra( component ); | ||||
|         ComponentSerializer.toString( component ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = IllegalArgumentException.class) | ||||
|     public void testLoopComplex() | ||||
|     { | ||||
|         TextComponent a = new TextComponent( "A" ); | ||||
|         TextComponent b = new TextComponent( "B" ); | ||||
|         b.setColor( ChatColor.AQUA ); | ||||
|         TextComponent c = new TextComponent( "C" ); | ||||
|         c.setColor( ChatColor.RED ); | ||||
|         a.addExtra( b ); | ||||
|         b.addExtra( c ); | ||||
|         c.addExtra( a ); | ||||
|         ComponentSerializer.toString( a ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testRepeated() | ||||
|     { | ||||
|         TextComponent a = new TextComponent( "A" ); | ||||
|         TextComponent b = new TextComponent( "B" ); | ||||
|         b.setColor( ChatColor.AQUA ); | ||||
|         a.addExtra( b ); | ||||
|         a.addExtra( b ); | ||||
|         ComponentSerializer.toString( a ); | ||||
|     } | ||||
|  | ||||
|     @Test(expected = IllegalArgumentException.class) | ||||
|     public void testRepeatedError() | ||||
|     { | ||||
|         TextComponent a = new TextComponent( "A" ); | ||||
|         TextComponent b = new TextComponent( "B" ); | ||||
|         b.setColor( ChatColor.AQUA ); | ||||
|         TextComponent c = new TextComponent( "C" ); | ||||
|         c.setColor( ChatColor.RED ); | ||||
|         a.addExtra( b ); | ||||
|         a.addExtra( c ); | ||||
|         c.addExtra( a ); | ||||
|         a.addExtra( b ); | ||||
|         ComponentSerializer.toString( a ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testInvalidColorCodes() | ||||
|     { | ||||
|         StringBuilder allInvalidColorCodes = new StringBuilder(); | ||||
|  | ||||
|         // collect all invalid color codes (e.g. §z, §g, ...) | ||||
|         for ( char alphChar : "0123456789abcdefghijklmnopqrstuvwxyz".toCharArray() ) | ||||
|         { | ||||
|             if ( ChatColor.ALL_CODES.indexOf( alphChar ) == -1 ) | ||||
|             { | ||||
|                 allInvalidColorCodes.append( ChatColor.COLOR_CHAR ); | ||||
|                 allInvalidColorCodes.append( alphChar ); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // last char is a single '§' | ||||
|         allInvalidColorCodes.append( ChatColor.COLOR_CHAR ); | ||||
|  | ||||
|         String invalidColorCodesLegacyText = fromAndToLegacyText( allInvalidColorCodes.toString() ); | ||||
|         String emptyLegacyText = fromAndToLegacyText( "" ); | ||||
|  | ||||
|         // all invalid color codes and the trailing '§' should be ignored | ||||
|         Assert.assertEquals( emptyLegacyText, invalidColorCodesLegacyText ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testFormattingOnlyTextConversion() | ||||
|     { | ||||
|         String text = "§a"; | ||||
|  | ||||
|         BaseComponent[] converted = TextComponent.fromLegacyText( text ); | ||||
|         Assert.assertEquals( ChatColor.GREEN, converted[0].getColor() ); | ||||
|  | ||||
|         String roundtripLegacyText = BaseComponent.toLegacyText( converted ); | ||||
|  | ||||
|         // color code should not be lost during conversion | ||||
|         Assert.assertEquals( text, roundtripLegacyText ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testEquals() | ||||
|     { | ||||
|         TextComponent first = new TextComponent( "Hello, " ); | ||||
|         first.addExtra( new TextComponent( "World!" ) ); | ||||
|  | ||||
|         TextComponent second = new TextComponent( "Hello, " ); | ||||
|         second.addExtra( new TextComponent( "World!" ) ); | ||||
|  | ||||
|         Assert.assertEquals( first, second ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testNotEquals() | ||||
|     { | ||||
|         TextComponent first = new TextComponent( "Hello, " ); | ||||
|         first.addExtra( new TextComponent( "World." ) ); | ||||
|  | ||||
|         TextComponent second = new TextComponent( "Hello, " ); | ||||
|         second.addExtra( new TextComponent( "World!" ) ); | ||||
|  | ||||
|         Assert.assertNotEquals( first, second ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testLegacyHack() | ||||
|     { | ||||
|         BaseComponent[] hexColored = new ComponentBuilder().color( ChatColor.of( Color.GRAY ) ).append( "Test" ).create(); | ||||
|         String legacy = TextComponent.toLegacyText( hexColored ); | ||||
|  | ||||
|         BaseComponent[] reColored = TextComponent.fromLegacyText( legacy ); | ||||
|  | ||||
|         Assert.assertArrayEquals( hexColored, reColored ); | ||||
|     } | ||||
|  | ||||
|     private String fromAndToLegacyText(String legacyText) | ||||
|     { | ||||
|         return BaseComponent.toLegacyText( TextComponent.fromLegacyText( legacyText ) ); | ||||
|     } | ||||
| } | ||||
| @@ -1,28 +0,0 @@ | ||||
| package net.md_5.bungee.api.chat; | ||||
|  | ||||
| import net.md_5.bungee.chat.ComponentSerializer; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
|  | ||||
| public class TranslatableComponentTest | ||||
| { | ||||
|  | ||||
|     @Test | ||||
|     public void testMissingPlaceholdersAdded() | ||||
|     { | ||||
|         TranslatableComponent testComponent = new TranslatableComponent( "Test string with %s placeholders: %s", 2, "aoeu" ); | ||||
|         Assert.assertEquals( "Test string with 2 placeholders: aoeu", testComponent.toPlainText() ); | ||||
|         Assert.assertEquals( "§fTest string with §f2§f placeholders: §faoeu", testComponent.toLegacyText() ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testJsonSerialisation() | ||||
|     { | ||||
|         TranslatableComponent testComponent = new TranslatableComponent( "Test string with %s placeholder", "a" ); | ||||
|         String jsonString = ComponentSerializer.toString( testComponent ); | ||||
|         BaseComponent[] baseComponents = ComponentSerializer.parse( jsonString ); | ||||
|  | ||||
|         Assert.assertEquals( "Test string with a placeholder", TextComponent.toPlainText( baseComponents ) ); | ||||
|         Assert.assertEquals( "§fTest string with §fa§f placeholder", TextComponent.toLegacyText( baseComponents ) ); | ||||
|     } | ||||
| } | ||||
| @@ -33,9 +33,9 @@ | ||||
|  | ||||
|         <!-- See http://checkstyle.sourceforge.net/config_filters.html --> | ||||
|         <module name="SuppressionCommentFilter"/> | ||||
|         <module name="SuppressWarningsHolder"/> | ||||
|  | ||||
|         <!-- See http://checkstyle.sourceforge.net/config_imports.html --> | ||||
|         <module name="AvoidStarImport"/> | ||||
|         <module name="ImportOrder"> | ||||
|             <property name="option" value="above"/> | ||||
|             <property name="ordered" value="true"/> | ||||
| @@ -54,11 +54,11 @@ | ||||
|         <module name="OperatorWrap"/> | ||||
|         <module name="ParenPad"> | ||||
|             <property name="option" value="nospace"/> | ||||
|             <property name="tokens" value="ANNOTATION, CTOR_DEF, METHOD_DEF"/> | ||||
|             <property name="tokens" value="ANNOTATION, CTOR_DEF, METHOD_DEF, LAMBDA"/> | ||||
|         </module> | ||||
|         <module name="ParenPad"> | ||||
|             <property name="option" value="space"/> | ||||
|             <property name="tokens" value="ANNOTATION_FIELD_DEF, CTOR_CALL, DOT, ENUM_CONSTANT_DEF, EXPR, LITERAL_CATCH, LITERAL_DO, LITERAL_FOR, LITERAL_IF, LITERAL_NEW, LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_WHILE, METHOD_CALL, QUESTION, RESOURCE_SPECIFICATION, SUPER_CTOR_CALL, LAMBDA"/> | ||||
|             <property name="tokens" value="ANNOTATION_FIELD_DEF, CTOR_CALL, DOT, ENUM_CONSTANT_DEF, EXPR, LITERAL_CATCH, LITERAL_DO, LITERAL_FOR, LITERAL_IF, LITERAL_NEW, LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_WHILE, METHOD_CALL, QUESTION, RESOURCE_SPECIFICATION, SUPER_CTOR_CALL, RECORD_DEF"/> | ||||
|         </module> | ||||
|         <module name="SingleSpaceSeparator"/> | ||||
|         <module name="TypecastParenPad"/> | ||||
| @@ -84,4 +84,6 @@ | ||||
|         <module name="Indentation"/> | ||||
|         <module name="UpperEll"/> | ||||
|     </module> | ||||
|  | ||||
|     <module name="SuppressWarningsFilter"/> | ||||
| </module> | ||||
|   | ||||
| @@ -4,15 +4,14 @@ | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <parent> | ||||
|         <groupId>net.md-5</groupId> | ||||
|         <groupId>fr.pandacube.bungeecord</groupId> | ||||
|         <artifactId>bungeecord-parent</artifactId> | ||||
|         <version>1.16-R0.4-SNAPSHOT</version> | ||||
|         <version>1.21-R0.5-SNAPSHOT</version> | ||||
|         <relativePath>../pom.xml</relativePath> | ||||
|     </parent> | ||||
|  | ||||
|     <groupId>net.md-5</groupId> | ||||
|     <artifactId>bungeecord-config</artifactId> | ||||
|     <version>1.16-R0.4-SNAPSHOT</version> | ||||
|     <version>1.21-R0.5-SNAPSHOT</version> | ||||
|     <packaging>jar</packaging> | ||||
|  | ||||
|     <name>BungeeCord-Config</name> | ||||
| @@ -22,14 +21,14 @@ | ||||
|         <dependency> | ||||
|             <groupId>com.google.code.gson</groupId> | ||||
|             <artifactId>gson</artifactId> | ||||
|             <version>2.8.0</version> | ||||
|             <version>2.11.0</version> | ||||
|             <scope>compile</scope> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.yaml</groupId> | ||||
|             <artifactId>snakeyaml</artifactId> | ||||
|             <version>1.26</version> | ||||
|             <version>2.2</version> | ||||
|             <scope>compile</scope> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| package net.md_5.bungee.config; | ||||
|  | ||||
| import com.google.common.base.Charsets; | ||||
| import com.google.gson.Gson; | ||||
| import com.google.gson.GsonBuilder; | ||||
| import com.google.gson.JsonElement; | ||||
| @@ -16,6 +15,7 @@ import java.io.OutputStreamWriter; | ||||
| import java.io.Reader; | ||||
| import java.io.Writer; | ||||
| import java.lang.reflect.Type; | ||||
| import java.nio.charset.StandardCharsets; | ||||
| import java.util.LinkedHashMap; | ||||
| import java.util.Map; | ||||
| import lombok.AccessLevel; | ||||
| @@ -37,7 +37,7 @@ public class JsonConfiguration extends ConfigurationProvider | ||||
|     @Override | ||||
|     public void save(Configuration config, File file) throws IOException | ||||
|     { | ||||
|         try ( Writer writer = new OutputStreamWriter( new FileOutputStream( file ), Charsets.UTF_8 ) ) | ||||
|         try ( Writer writer = new OutputStreamWriter( new FileOutputStream( file ), StandardCharsets.UTF_8 ) ) | ||||
|         { | ||||
|             save( config, writer ); | ||||
|         } | ||||
| @@ -91,7 +91,7 @@ public class JsonConfiguration extends ConfigurationProvider | ||||
|     @Override | ||||
|     public Configuration load(InputStream is, Configuration defaults) | ||||
|     { | ||||
|         return load( new InputStreamReader( is, Charsets.UTF_8 ), defaults ); | ||||
|         return load( new InputStreamReader( is, StandardCharsets.UTF_8 ), defaults ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| package net.md_5.bungee.config; | ||||
|  | ||||
| import com.google.common.base.Charsets; | ||||
| import java.io.File; | ||||
| import java.io.FileInputStream; | ||||
| import java.io.FileOutputStream; | ||||
| @@ -9,11 +8,13 @@ import java.io.InputStream; | ||||
| import java.io.OutputStreamWriter; | ||||
| import java.io.Reader; | ||||
| import java.io.Writer; | ||||
| import java.nio.charset.StandardCharsets; | ||||
| import java.util.LinkedHashMap; | ||||
| import java.util.Map; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.yaml.snakeyaml.DumperOptions; | ||||
| import org.yaml.snakeyaml.LoaderOptions; | ||||
| import org.yaml.snakeyaml.Yaml; | ||||
| import org.yaml.snakeyaml.constructor.Constructor; | ||||
| import org.yaml.snakeyaml.nodes.Node; | ||||
| @@ -29,7 +30,10 @@ public class YamlConfiguration extends ConfigurationProvider | ||||
|         @Override | ||||
|         protected Yaml initialValue() | ||||
|         { | ||||
|             Representer representer = new Representer() | ||||
|             DumperOptions options = new DumperOptions(); | ||||
|             options.setDefaultFlowStyle( DumperOptions.FlowStyle.BLOCK ); | ||||
|  | ||||
|             Representer representer = new Representer( options ) | ||||
|             { | ||||
|                 { | ||||
|                     representers.put( Configuration.class, new Represent() | ||||
| @@ -43,17 +47,14 @@ public class YamlConfiguration extends ConfigurationProvider | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             DumperOptions options = new DumperOptions(); | ||||
|             options.setDefaultFlowStyle( DumperOptions.FlowStyle.BLOCK ); | ||||
|  | ||||
|             return new Yaml( new Constructor(), representer, options ); | ||||
|             return new Yaml( new Constructor( new LoaderOptions() ), representer, options ); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     @Override | ||||
|     public void save(Configuration config, File file) throws IOException | ||||
|     { | ||||
|         try ( Writer writer = new OutputStreamWriter( new FileOutputStream( file ), Charsets.UTF_8 ) ) | ||||
|         try ( Writer writer = new OutputStreamWriter( new FileOutputStream( file ), StandardCharsets.UTF_8 ) ) | ||||
|         { | ||||
|             save( config, writer ); | ||||
|         } | ||||
|   | ||||
| @@ -1,30 +1,25 @@ | ||||
| package net.md_5.bungee.config; | ||||
|  | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import java.io.StringReader; | ||||
| import java.io.StringWriter; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.stream.Stream; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
| import org.junit.runner.RunWith; | ||||
| import org.junit.runners.Parameterized; | ||||
| import org.junit.runners.Parameterized.Parameters; | ||||
| import org.junit.jupiter.params.ParameterizedTest; | ||||
| import org.junit.jupiter.params.provider.Arguments; | ||||
| import org.junit.jupiter.params.provider.MethodSource; | ||||
|  | ||||
| @RequiredArgsConstructor | ||||
| @RunWith(Parameterized.class) | ||||
| public class CompoundConfigurationTest | ||||
| { | ||||
|  | ||||
|     @Parameters(name = "{0}") | ||||
|     public static Iterable<Object[]> data() | ||||
|     { | ||||
|         // CHECKSTYLE:OFF | ||||
|         return Arrays.asList( new Object[][] | ||||
|         { | ||||
|     public static Stream<Arguments> data() | ||||
|     { | ||||
|         return Stream.of( | ||||
|                 Arguments.of( | ||||
|                         // provider | ||||
|                         YamlConfiguration.class, | ||||
|                         // testDocument | ||||
| @@ -73,8 +68,8 @@ public class CompoundConfigurationTest | ||||
|                         + "null:\n" | ||||
|                         + "    null: object\n" | ||||
|                         + "    object: null\n" | ||||
|             }, | ||||
|             { | ||||
|                 ), | ||||
|                 Arguments.of( | ||||
|                         // provider | ||||
|                         JsonConfiguration.class, | ||||
|                         // testDocument | ||||
| @@ -131,18 +126,13 @@ public class CompoundConfigurationTest | ||||
|                         + "    \"object\": null\n" | ||||
|                         + "  }\n" | ||||
|                         + "}" | ||||
|                 ) | ||||
|         ); | ||||
|     } | ||||
|         } ); | ||||
|         // CHECKSTYLE:ON | ||||
|     } | ||||
|     // | ||||
|     private final Class<? extends ConfigurationProvider> provider; | ||||
|     private final String testDocument; | ||||
|     private final String numberTest; | ||||
|     private final String nullTest; | ||||
|  | ||||
|     @Test | ||||
|     public void testConfig() throws Exception | ||||
|     @ParameterizedTest | ||||
|     @MethodSource("data") | ||||
|     public void testConfig(Class<? extends ConfigurationProvider> provider, String testDocument, String numberTest, String nullTest) throws Exception | ||||
|     { | ||||
|         Configuration conf = ConfigurationProvider.getProvider( provider ).load( testDocument ); | ||||
|         testSection( conf ); | ||||
| @@ -151,7 +141,7 @@ public class CompoundConfigurationTest | ||||
|         ConfigurationProvider.getProvider( provider ).save( conf, sw ); | ||||
|  | ||||
|         // Check nulls were saved, see #1094 | ||||
|         Assert.assertFalse( "Config contains null", sw.toString().contains( "null" ) ); | ||||
|         assertFalse( sw.toString().contains( "null" ), "Config contains null" ); | ||||
|  | ||||
|         conf = ConfigurationProvider.getProvider( provider ).load( new StringReader( sw.toString() ) ); | ||||
|         conf.set( "receipt", "Oz-Ware Purchase Invoice" ); // Add it back | ||||
| @@ -160,37 +150,38 @@ public class CompoundConfigurationTest | ||||
|  | ||||
|     private void testSection(Configuration conf) | ||||
|     { | ||||
|         Assert.assertEquals( "receipt", "Oz-Ware Purchase Invoice", conf.getString( "receipt" ) ); | ||||
|         // Assert.assertEquals( "date", "2012-08-06", conf.get( "date" ).toString() ); | ||||
|         assertEquals( "Oz-Ware Purchase Invoice", conf.getString( "receipt" ), "receipt" ); | ||||
|         // assertEquals( "2012-08-06", conf.get( "date" ).toString(), "date" ); | ||||
|  | ||||
|         Configuration customer = conf.getSection( "customer" ); | ||||
|         Assert.assertEquals( "customer.given", "Dorothy", customer.getString( "given" ) ); | ||||
|         Assert.assertEquals( "customer.given", "Dorothy", conf.getString( "customer.given" ) ); | ||||
|         assertEquals( "Dorothy", customer.getString( "given" ), "customer.given" ); | ||||
|         assertEquals( "Dorothy", conf.getString( "customer.given" ), "customer.given" ); | ||||
|  | ||||
|         List items = conf.getList( "items" ); | ||||
|         Map item = (Map) items.get( 0 ); | ||||
|         Assert.assertEquals( "items[0].part_no", "A4786", item.get( "part_no" ) ); | ||||
|         assertEquals( "A4786", item.get( "part_no" ), "items[0].part_no" ); | ||||
|  | ||||
|         conf.set( "receipt", null ); | ||||
|         Assert.assertEquals( null, conf.get( "receipt" ) ); | ||||
|         Assert.assertEquals( "foo", conf.get( "receipt", "foo" ) ); | ||||
|         assertEquals( null, conf.get( "receipt" ) ); | ||||
|         assertEquals( "foo", conf.get( "receipt", "foo" ) ); | ||||
|  | ||||
|         Configuration newSection = conf.getSection( "new.section" ); | ||||
|         newSection.set( "value", "foo" ); | ||||
|         Assert.assertEquals( "foo", conf.get( "new.section.value" ) ); | ||||
|         assertEquals( "foo", conf.get( "new.section.value" ) ); | ||||
|  | ||||
|         conf.set( "other.new.section", "bar" ); | ||||
|         Assert.assertEquals( "bar", conf.get( "other.new.section" ) ); | ||||
|         assertEquals( "bar", conf.get( "other.new.section" ) ); | ||||
|  | ||||
|         Assert.assertTrue( conf.contains( "customer.given" ) ); | ||||
|         Assert.assertTrue( customer.contains( "given" ) ); | ||||
|         assertTrue( conf.contains( "customer.given" ) ); | ||||
|         assertTrue( customer.contains( "given" ) ); | ||||
|  | ||||
|         Assert.assertFalse( conf.contains( "customer.foo" ) ); | ||||
|         Assert.assertFalse( customer.contains( "foo" ) ); | ||||
|         assertFalse( conf.contains( "customer.foo" ) ); | ||||
|         assertFalse( customer.contains( "foo" ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testNumberedKeys() | ||||
|     @ParameterizedTest | ||||
|     @MethodSource("data") | ||||
|     public void testNumberedKeys(Class<? extends ConfigurationProvider> provider, String testDocument, String numberTest, String nullTest) | ||||
|     { | ||||
|         Configuration conf = ConfigurationProvider.getProvider( provider ).load( numberTest ); | ||||
|  | ||||
| @@ -201,29 +192,31 @@ public class CompoundConfigurationTest | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testNull() | ||||
|     @ParameterizedTest | ||||
|     @MethodSource("data") | ||||
|     public void testNull(Class<? extends ConfigurationProvider> provider, String testDocument, String numberTest, String nullTest) | ||||
|     { | ||||
|         Configuration conf = ConfigurationProvider.getProvider( provider ).load( nullTest ); | ||||
|  | ||||
|         Assert.assertEquals( "object", conf.get( "null.null" ) ); | ||||
|         Assert.assertEquals( "object", conf.getSection( "null" ).get( "null" ) ); | ||||
|         assertEquals( "object", conf.get( "null.null" ) ); | ||||
|         assertEquals( "object", conf.getSection( "null" ).get( "null" ) ); | ||||
|  | ||||
|         Assert.assertEquals( null, conf.get( "null.object" ) ); | ||||
|         Assert.assertEquals( "", conf.getString( "null.object" ) ); | ||||
|         assertEquals( null, conf.get( "null.object" ) ); | ||||
|         assertEquals( "", conf.getString( "null.object" ) ); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testMapAddition() | ||||
|     @ParameterizedTest | ||||
|     @MethodSource("data") | ||||
|     public void testMapAddition(Class<? extends ConfigurationProvider> provider, String testDocument, String numberTest, String nullTest) | ||||
|     { | ||||
|         Configuration conf = ConfigurationProvider.getProvider( provider ).load( testDocument ); | ||||
|  | ||||
|         conf.set( "addition", Collections.singletonMap( "foo", "bar" ) ); | ||||
|  | ||||
|         // Order matters | ||||
|         Assert.assertEquals( "bar", conf.getSection( "addition" ).getString( "foo" ) ); | ||||
|         Assert.assertEquals( "bar", conf.getString( "addition.foo" ) ); | ||||
|         assertEquals( "bar", conf.getSection( "addition" ).getString( "foo" ) ); | ||||
|         assertEquals( "bar", conf.getString( "addition.foo" ) ); | ||||
|  | ||||
|         Assert.assertTrue( conf.get( "addition" ) instanceof Configuration ); | ||||
|         assertTrue( conf.get( "addition" ) instanceof Configuration ); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package net.md_5.bungee.config; | ||||
|  | ||||
| import org.junit.Assert; | ||||
| import org.junit.Test; | ||||
| import static org.junit.jupiter.api.Assertions.*; | ||||
| import org.junit.jupiter.api.Test; | ||||
|  | ||||
| public class DefaultConfigurationTest | ||||
| { | ||||
| @@ -16,8 +16,8 @@ public class DefaultConfigurationTest | ||||
|  | ||||
|         Configuration actualConfig = new Configuration( defaultConfig ); | ||||
|  | ||||
|         Assert.assertEquals( 10, actualConfig.getInt( "setting" ) ); | ||||
|         Assert.assertEquals( 11, actualConfig.getInt( "nested.setting" ) ); | ||||
|         Assert.assertEquals( 12, actualConfig.getInt( "double.nested.setting" ) ); | ||||
|         assertEquals( 10, actualConfig.getInt( "setting" ) ); | ||||
|         assertEquals( 11, actualConfig.getInt( "nested.setting" ) ); | ||||
|         assertEquals( 12, actualConfig.getInt( "double.nested.setting" ) ); | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										28
									
								
								dialog/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								dialog/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| BSD 3-Clause License | ||||
|  | ||||
| Copyright (c) 2025, SpigotMC Pty. Ltd. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are met: | ||||
|  | ||||
| 1. Redistributions of source code must retain the above copyright notice, this | ||||
|    list of conditions and the following disclaimer. | ||||
|  | ||||
| 2. Redistributions in binary form must reproduce the above copyright notice, | ||||
|    this list of conditions and the following disclaimer in the documentation | ||||
|    and/or other materials provided with the distribution. | ||||
|  | ||||
| 3. Neither the name of the copyright holder nor the names of its | ||||
|    contributors may be used to endorse or promote products derived from | ||||
|    this software without specific prior written permission. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
							
								
								
									
										38
									
								
								dialog/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								dialog/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| BungeeCord-Dialog | ||||
| ================= | ||||
|  | ||||
| Highly experimental API, subject to breakage. All contributions welcome, including major refactors/design changes. | ||||
|  | ||||
| Sample Plugin | ||||
| ------------- | ||||
|  | ||||
| ```java | ||||
|     private class TestCommand extends Command | ||||
|     { | ||||
|  | ||||
|         public TestCommand() | ||||
|         { | ||||
|             super( "btest" ); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void execute(CommandSender sender, String[] args) | ||||
|         { | ||||
|             ProxiedPlayer player = (ProxiedPlayer) sender; | ||||
|  | ||||
|             Dialog notice = new NoticeDialog( new DialogBase( new ComponentBuilder( "Hello" ).color( ChatColor.RED ).build() ) ); | ||||
|             player.showDialog( notice ); | ||||
|  | ||||
|             notice = new NoticeDialog( | ||||
|                     new DialogBase( new ComponentBuilder( "Hello" ).color( ChatColor.RED ).build() ) | ||||
|                             .inputs( | ||||
|                                     Arrays.asList( new TextInput( "first", new ComponentBuilder( "First" ).build() ), | ||||
|                                             new TextInput( "second", new ComponentBuilder( "Second" ).build() ) | ||||
|                                     ) | ||||
|                             ) ) | ||||
|                     .action( new ActionButton( new ComponentBuilder( "Submit Button" ).build(), new CustomClickAction( "customform" ) ) ); | ||||
|  | ||||
|             player.sendMessage( new ComponentBuilder( "click me" ).event( new ShowDialogClickEvent( notice ) ).build() ); | ||||
|         } | ||||
|     } | ||||
| ``` | ||||
							
								
								
									
										35
									
								
								dialog/pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								dialog/pom.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
|  | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <parent> | ||||
|         <groupId>fr.pandacube.bungeecord</groupId> | ||||
|         <artifactId>bungeecord-parent</artifactId> | ||||
|         <version>1.21-R0.5-SNAPSHOT</version> | ||||
|         <relativePath>../pom.xml</relativePath> | ||||
|     </parent> | ||||
|  | ||||
|     <artifactId>bungeecord-dialog</artifactId> | ||||
|     <version>1.21-R0.5-SNAPSHOT</version> | ||||
|     <packaging>jar</packaging> | ||||
|  | ||||
|     <name>BungeeCord-Dialog</name> | ||||
|     <description>Minecraft dialog API intended for use with BungeeCord</description> | ||||
|     <licenses> | ||||
|         <license> | ||||
|             <name>BSD-3-Clause</name> | ||||
|             <url>https://github.com/SpigotMC/BungeeCord/blob/master/dialog/LICENSE</url> | ||||
|             <distribution>repo</distribution> | ||||
|         </license> | ||||
|     </licenses> | ||||
|  | ||||
|     <dependencies> | ||||
|         <dependency> | ||||
|             <groupId>fr.pandacube.bungeecord</groupId> | ||||
|             <artifactId>bungeecord-chat</artifactId> | ||||
|             <version>${project.version}</version> | ||||
|             <scope>compile</scope> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
| </project> | ||||
| @@ -0,0 +1,39 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.dialog.action.ActionButton; | ||||
|  | ||||
| /** | ||||
|  * Represents a simple dialog with text and two actions at the bottom (default: | ||||
|  * "yes", "no"). | ||||
|  */ | ||||
| @Data | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @AllArgsConstructor | ||||
| @Accessors(fluent = true) | ||||
| public final class ConfirmationDialog implements Dialog | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     @Accessors(fluent = false) | ||||
|     private DialogBase base; | ||||
|     /** | ||||
|      * The "yes" click action / bottom (appears on the left). | ||||
|      */ | ||||
|     private ActionButton yes; | ||||
|     /** | ||||
|      * The "no" click action / bottom (appears on the right). | ||||
|      */ | ||||
|     private ActionButton no; | ||||
|  | ||||
|     public ConfirmationDialog(@NonNull DialogBase base) | ||||
|     { | ||||
|         this( base, null, null ); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										29
									
								
								dialog/src/main/java/net/md_5/bungee/api/dialog/Dialog.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								dialog/src/main/java/net/md_5/bungee/api/dialog/Dialog.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog GUI. | ||||
|  */ | ||||
| public interface Dialog | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * Gets the dialog base which contains the dialog title and other options | ||||
|      * common to all types of dialogs. | ||||
|      * | ||||
|      * @return mutable reference to the dialog base | ||||
|      */ | ||||
|     DialogBase getBase(); | ||||
|  | ||||
|     /** | ||||
|      * Sets the dialog base. | ||||
|      * <br> | ||||
|      * For internal use only as this is mandatory and should be specified in the | ||||
|      * constructor. | ||||
|      * | ||||
|      * @param base the new dialog base | ||||
|      */ | ||||
|     @ApiStatus.Internal | ||||
|     void setBase(DialogBase base); | ||||
| } | ||||
| @@ -0,0 +1,84 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import java.util.List; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
| import net.md_5.bungee.api.dialog.body.DialogBody; | ||||
| import net.md_5.bungee.api.dialog.input.DialogInput; | ||||
|  | ||||
| /** | ||||
|  * Represents the title and other options common to all dialogs. | ||||
|  */ | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| @Accessors(fluent = true) | ||||
| public final class DialogBase | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The mandatory dialog title. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private BaseComponent title; | ||||
|     /** | ||||
|      * The name which is used for any buttons leading to this dialog (eg from a | ||||
|      * {@link DialogListDialog}). Otherwise defaults to {@link #title}. | ||||
|      */ | ||||
|     @SerializedName("external_title") | ||||
|     private BaseComponent externalTitle; | ||||
|     /** | ||||
|      * The inputs to the dialog. | ||||
|      */ | ||||
|     private List<DialogInput> inputs; | ||||
|     /** | ||||
|      * The body elements which make up this dialog. | ||||
|      */ | ||||
|     private List<DialogBody> body; | ||||
|     /** | ||||
|      * Whether this dialog can be closed with the escape key (default: true). | ||||
|      */ | ||||
|     @SerializedName("can_close_with_escape") | ||||
|     private Boolean canCloseWithEscape; | ||||
|     /** | ||||
|      * Whether this dialog should pause the game in single-player mode (default: | ||||
|      * true). | ||||
|      */ | ||||
|     private Boolean pause; | ||||
|     /** | ||||
|      * Action to take after the a click or submit action is performed on the | ||||
|      * dialog (default: close). | ||||
|      */ | ||||
|     @SerializedName("after_action") | ||||
|     private AfterAction afterAction; | ||||
|  | ||||
|     public DialogBase(@NonNull BaseComponent title) | ||||
|     { | ||||
|         this( title, null, null, null, null, null, null ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Types of action which may be taken after the dialog. | ||||
|      */ | ||||
|     public enum AfterAction | ||||
|     { | ||||
|         /** | ||||
|          * Close the dialog. | ||||
|          */ | ||||
|         @SerializedName("close") | ||||
|         CLOSE, | ||||
|         /** | ||||
|          * Do nothing. | ||||
|          */ | ||||
|         @SerializedName("none") | ||||
|         NONE, | ||||
|         /** | ||||
|          * Show a waiting for response screen. | ||||
|          */ | ||||
|         @SerializedName("wait_for_response") | ||||
|         WAIT_FOR_RESPONSE; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,74 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.dialog.action.ActionButton; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog which contains buttons that link to other dialogs. | ||||
|  */ | ||||
| @Data | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @Accessors(fluent = true) | ||||
| public final class DialogListDialog implements Dialog | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     @Accessors(fluent = false) | ||||
|     private DialogBase base; | ||||
|     /** | ||||
|      * The child dialogs behind each button. | ||||
|      */ | ||||
|     private List<Dialog> dialogs; | ||||
|     /** | ||||
|      * The {@link ActionButton} activated when the dialog is exited. | ||||
|      */ | ||||
|     @SerializedName("exit_action") | ||||
|     private ActionButton exitAction; | ||||
|     /** | ||||
|      * The number of columns for the dialog buttons (default: 2). | ||||
|      */ | ||||
|     private Integer columns; | ||||
|     /** | ||||
|      * The width of the dialog buttons (default: 150, minimum: 1, maximum: | ||||
|      * 1024). | ||||
|      */ | ||||
|     @SerializedName("button_width") | ||||
|     private Integer buttonWidth; | ||||
|  | ||||
|     public DialogListDialog(@NonNull DialogBase base, Dialog... dialogs) | ||||
|     { | ||||
|         this( base, Arrays.asList( dialogs ), null, null, null ); | ||||
|     } | ||||
|  | ||||
|     public DialogListDialog(@NonNull DialogBase base, List<Dialog> dialogs, ActionButton exitAction, Integer columns, Integer buttonWidth) | ||||
|     { | ||||
|         this.base = base; | ||||
|         this.dialogs = dialogs; | ||||
|         this.exitAction = exitAction; | ||||
|         columns( columns ); | ||||
|         buttonWidth( buttonWidth ); | ||||
|     } | ||||
|  | ||||
|     public DialogListDialog columns(Integer columns) | ||||
|     { | ||||
|         Preconditions.checkArgument( columns == null || columns > 0, "At least one column is required" ); | ||||
|         this.columns = columns; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     public DialogListDialog buttonWidth(Integer buttonWidth) | ||||
|     { | ||||
|         Preconditions.checkArgument( buttonWidth == null || ( buttonWidth >= 1 && buttonWidth <= 1024 ), "buttonWidth must be between 1 and 1024" ); | ||||
|         this.buttonWidth = buttonWidth; | ||||
|         return this; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,64 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.dialog.action.ActionButton; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog with text a list of action buttons grouped into columns | ||||
|  * and scrollable if necessary. | ||||
|  */ | ||||
| @Data | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @Accessors(fluent = true) | ||||
| public final class MultiActionDialog implements Dialog | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     @Accessors(fluent = false) | ||||
|     private DialogBase base; | ||||
|     /** | ||||
|      * The action buttons in the dialog. At least one must be provided. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private List<ActionButton> actions; | ||||
|     /** | ||||
|      * The number of columns for the dialog buttons (default: 2). | ||||
|      */ | ||||
|     private Integer columns; | ||||
|     /** | ||||
|      * The {@link ActionButton} activated when the dialog is exited. | ||||
|      */ | ||||
|     @SerializedName("exit_action") | ||||
|     private ActionButton exitAction; | ||||
|  | ||||
|     public MultiActionDialog(@NonNull DialogBase base, @NonNull ActionButton... actions) | ||||
|     { | ||||
|         this( base, Arrays.asList( actions ), null, null ); | ||||
|     } | ||||
|  | ||||
|     public MultiActionDialog(@NonNull DialogBase base, @NonNull List<ActionButton> actions, Integer columns, ActionButton exitAction) | ||||
|     { | ||||
|         Preconditions.checkArgument( !actions.isEmpty(), "At least one action must be provided" ); | ||||
|  | ||||
|         this.base = base; | ||||
|         this.actions = actions; | ||||
|         columns( columns ); | ||||
|         this.exitAction = exitAction; | ||||
|     } | ||||
|  | ||||
|     public MultiActionDialog columns(Integer columns) | ||||
|     { | ||||
|         Preconditions.checkArgument( columns == null || columns > 0, "At least one column is required" ); | ||||
|         this.columns = columns; | ||||
|         return this; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,35 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.dialog.action.ActionButton; | ||||
|  | ||||
| /** | ||||
|  * Represents a simple dialog with text and one action at the bottom (default: | ||||
|  * "OK"). | ||||
|  */ | ||||
| @Data | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @AllArgsConstructor | ||||
| @Accessors(fluent = true) | ||||
| public final class NoticeDialog implements Dialog | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     @Accessors(fluent = false) | ||||
|     private DialogBase base; | ||||
|     /** | ||||
|      * The "OK" action button for the dialog. | ||||
|      */ | ||||
|     private ActionButton action; | ||||
|  | ||||
|     public NoticeDialog(DialogBase base) | ||||
|     { | ||||
|         this( base, null ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,72 @@ | ||||
| package net.md_5.bungee.api.dialog; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.dialog.action.ActionButton; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog which shows the links configured/sent from the server. | ||||
|  */ | ||||
| @Data | ||||
| @ToString | ||||
| @EqualsAndHashCode | ||||
| @Accessors(fluent = true) | ||||
| public final class ServerLinksDialog implements Dialog | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     @Accessors(fluent = false) | ||||
|     private DialogBase base; | ||||
|     /** | ||||
|      * The optional {@link ActionButton} for this dialog. | ||||
|      */ | ||||
|     @SerializedName("action") | ||||
|     private ActionButton action; | ||||
|     /** | ||||
|      * The {@link ActionButton} activated when the dialog is exited. | ||||
|      */ | ||||
|     @SerializedName("exit_action") | ||||
|     private ActionButton exitAction; | ||||
|     /** | ||||
|      * The number of columns for the dialog buttons (default: 2). | ||||
|      */ | ||||
|     private Integer columns; | ||||
|     /** | ||||
|      * The width of the dialog buttons (default: 150, minimum: 1, maximum: | ||||
|      * 1024). | ||||
|      */ | ||||
|     @SerializedName("button_width") | ||||
|     private Integer buttonWidth; | ||||
|  | ||||
|     public ServerLinksDialog(@NonNull DialogBase base) | ||||
|     { | ||||
|         this( base, null, null, null ); | ||||
|     } | ||||
|  | ||||
|     public ServerLinksDialog(@NonNull DialogBase base, ActionButton action, Integer columns, Integer buttonWidth) | ||||
|     { | ||||
|         this.base = base; | ||||
|         this.action = action; | ||||
|         columns( columns ); | ||||
|         buttonWidth( buttonWidth ); | ||||
|     } | ||||
|  | ||||
|     public ServerLinksDialog columns(Integer columns) | ||||
|     { | ||||
|         Preconditions.checkArgument( columns == null || columns > 0, "At least one column is required" ); | ||||
|         this.columns = columns; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     public ServerLinksDialog buttonWidth(Integer buttonWidth) | ||||
|     { | ||||
|         Preconditions.checkArgument( buttonWidth == null || ( buttonWidth >= 1 && buttonWidth <= 1024 ), "buttonWidth must be between 1 and 1024" ); | ||||
|         this.buttonWidth = buttonWidth; | ||||
|         return this; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,6 @@ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
|  | ||||
| public interface Action | ||||
| { | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,54 @@ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog action which will usually appear as a button. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| public class ActionButton | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The text label of the button, mandatory. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private BaseComponent label; | ||||
|     /** | ||||
|      * The hover tooltip of the button. | ||||
|      */ | ||||
|     private BaseComponent tooltip; | ||||
|     /** | ||||
|      * The width of the button (default: 150, minimum: 1, maximum: 1024). | ||||
|      */ | ||||
|     private Integer width; | ||||
|     /** | ||||
|      * The action to take. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private Action action; | ||||
|  | ||||
|     public ActionButton(@NonNull BaseComponent label, BaseComponent tooltip, Integer width, @NonNull Action action) | ||||
|     { | ||||
|         this.label = label; | ||||
|         this.tooltip = tooltip; | ||||
|         setWidth( width ); | ||||
|         this.action = action; | ||||
|     } | ||||
|  | ||||
|     public ActionButton(@NonNull BaseComponent label, @NonNull Action action) | ||||
|     { | ||||
|         this( label, null, null, action ); | ||||
|     } | ||||
|  | ||||
|     public void setWidth(Integer width) | ||||
|     { | ||||
|         Preconditions.checkArgument( width == null || ( width >= 1 && width <= 1024 ), "width must be between 1 and 1024" ); | ||||
|         this.width = width; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,27 @@ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
|  | ||||
| import com.google.gson.JsonElement; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| /** | ||||
|  * Submits the dialog with the given ID and values as a payload. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| @ToString(callSuper = true) | ||||
| public class CustomClickAction implements Action | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The namespaced key of the submission. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private String id; | ||||
|     /** | ||||
|      * Fields to be added to the submission payload. | ||||
|      */ | ||||
|     private JsonElement additions; | ||||
| } | ||||
| @@ -0,0 +1,25 @@ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| /** | ||||
|  * Executes a command. If the command requires a permission higher than 0, a | ||||
|  * confirmation dialog will be shown by the client. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| @ToString(callSuper = true) | ||||
| public class RunCommandAction implements Action | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The template to be applied, where variables of the form | ||||
|      * <code>$(key)</code> will be replaced by their | ||||
|      * {@link net.md_5.bungee.api.dialog.input.DialogInput#key} value. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private String template; | ||||
| } | ||||
| @@ -0,0 +1,20 @@ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.ClickEvent; | ||||
|  | ||||
| /** | ||||
|  * Represents a static dialog action. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| @ToString(callSuper = true) | ||||
| public class StaticAction implements Action | ||||
| { | ||||
|  | ||||
|     @NonNull | ||||
|     private ClickEvent clickEvent; | ||||
| } | ||||
| @@ -0,0 +1,4 @@ | ||||
| /** | ||||
|  * Contains the different actions/buttons for a {@link net.md_5.bungee.api.dialog.Dialog}. | ||||
|  */ | ||||
| package net.md_5.bungee.api.dialog.action; | ||||
| @@ -0,0 +1,20 @@ | ||||
| package net.md_5.bungee.api.dialog.body; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents the body content of a {@link net.md_5.bungee.api.dialog.Dialog}. | ||||
|  */ | ||||
| @Data | ||||
| public abstract class DialogBody | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The internal body type. | ||||
|      */ | ||||
|     @NonNull | ||||
|     @ApiStatus.Internal | ||||
|     private final String type; | ||||
| } | ||||
| @@ -0,0 +1,49 @@ | ||||
| package net.md_5.bungee.api.dialog.body; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| /** | ||||
|  * Represents a dialog body which consists of text constrained to a certain | ||||
|  * width. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public class PlainMessageBody extends DialogBody | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The text body. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private BaseComponent contents; | ||||
|     /** | ||||
|      * The maximum width (default: 200, minimum: 1, maximum: 1024). | ||||
|      */ | ||||
|     private Integer width; | ||||
|  | ||||
|     public PlainMessageBody(@NonNull BaseComponent contents) | ||||
|     { | ||||
|         this( contents, null ); | ||||
|     } | ||||
|  | ||||
|     public PlainMessageBody(@NonNull BaseComponent contents, Integer width) | ||||
|     { | ||||
|         super( "minecraft:plain_message" ); | ||||
|         this.contents = contents; | ||||
|         width( width ); | ||||
|     } | ||||
|  | ||||
|     public void width(Integer width) | ||||
|     { | ||||
|         Preconditions.checkArgument( width == null || ( width >= 1 && width <= 1024 ), "width must be between 1 and 1024" ); | ||||
|         this.width = width; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,5 @@ | ||||
| /** | ||||
|  * Contains the different {@link net.md_5.bungee.api.dialog.Dialog} body content | ||||
|  * types. | ||||
|  */ | ||||
| package net.md_5.bungee.api.dialog.body; | ||||
| @@ -0,0 +1,42 @@ | ||||
| package net.md_5.bungee.api.dialog.chat; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import net.md_5.bungee.api.chat.ClickEvent; | ||||
| import net.md_5.bungee.api.dialog.Dialog; | ||||
|  | ||||
| /** | ||||
|  * Click event which displays either a pre-existing dialog by key or a custom | ||||
|  * dialog. | ||||
|  */ | ||||
| @Data | ||||
| @EqualsAndHashCode(callSuper = false) | ||||
| public class ShowDialogClickEvent extends ClickEvent | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * Key for a pre-existing dialog to show. | ||||
|      */ | ||||
|     private String reference; | ||||
|     /** | ||||
|      * Dialog to show. | ||||
|      */ | ||||
|     private Dialog dialog; | ||||
|  | ||||
|     public ShowDialogClickEvent(String reference) | ||||
|     { | ||||
|         this( reference, null ); | ||||
|     } | ||||
|  | ||||
|     public ShowDialogClickEvent(Dialog dialog) | ||||
|     { | ||||
|         this( null, dialog ); | ||||
|     } | ||||
|  | ||||
|     private ShowDialogClickEvent(String reference, Dialog dialog) | ||||
|     { | ||||
|         super( Action.SHOW_DIALOG, null ); | ||||
|         this.reference = reference; | ||||
|         this.dialog = dialog; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,4 @@ | ||||
| /** | ||||
|  * Contains dialog extensions to the chat API. | ||||
|  */ | ||||
| package net.md_5.bungee.api.dialog.chat; | ||||
| @@ -0,0 +1,54 @@ | ||||
| package net.md_5.bungee.api.dialog.input; | ||||
|  | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NonNull; | ||||
| import lombok.ToString; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| /** | ||||
|  * Represents a checkbox input control. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| @ToString(callSuper = true) | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| public class BooleanInput extends DialogInput | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The input label. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private BaseComponent label; | ||||
|     /** | ||||
|      * The initial value (default: false/unchecked). | ||||
|      */ | ||||
|     private Boolean initial; | ||||
|     /** | ||||
|      * The string value to be submitted when true/checked (default: "true"). | ||||
|      */ | ||||
|     @SerializedName("on_true") | ||||
|     private String onTrue; | ||||
|     /** | ||||
|      * The string value to be submitted when false/unchecked (default: "false"). | ||||
|      */ | ||||
|     @SerializedName("on_false") | ||||
|     private String onFalse; | ||||
|  | ||||
|     public BooleanInput(@NonNull String key, @NonNull BaseComponent label) | ||||
|     { | ||||
|         this( key, label, null, "true", "false" ); | ||||
|     } | ||||
|  | ||||
|     public BooleanInput(@NonNull String key, @NonNull BaseComponent label, Boolean initial, String onTrue, String onFalse) | ||||
|     { | ||||
|         super( "minecraft:boolean", key ); | ||||
|         this.label = label; | ||||
|         this.initial = initial; | ||||
|         this.onTrue = onTrue; | ||||
|         this.onFalse = onFalse; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,29 @@ | ||||
| package net.md_5.bungee.api.dialog.input; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.experimental.Accessors; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| /** | ||||
|  * Represents a type of input which may be displayed/submitted with a form | ||||
|  * dialog. | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(fluent = true) | ||||
| public class DialogInput | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The internal input type. | ||||
|      */ | ||||
|     @NonNull | ||||
|     @ApiStatus.Internal | ||||
|     private final String type; | ||||
|     /** | ||||
|      * The key corresponding to this input and associated with the value | ||||
|      * submitted. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private final String key; | ||||
| } | ||||
| @@ -0,0 +1,39 @@ | ||||
| package net.md_5.bungee.api.dialog.input; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NonNull; | ||||
| import lombok.experimental.Accessors; | ||||
| import net.md_5.bungee.api.chat.BaseComponent; | ||||
|  | ||||
| /** | ||||
|  * Represents an option choice which may form part of a | ||||
|  * {@link SingleOptionInput}. | ||||
|  */ | ||||
| @Data | ||||
| @AllArgsConstructor | ||||
| @Accessors(fluent = true) | ||||
| public class InputOption | ||||
| { | ||||
|  | ||||
|     /** | ||||
|      * The string value associated with this option, to be submitted when | ||||
|      * selected. | ||||
|      */ | ||||
|     @NonNull | ||||
|     private String id; | ||||
|     /** | ||||
|      * The text to display for this option. | ||||
|      */ | ||||
|     private BaseComponent display; | ||||
|     /** | ||||
|      * Whether this option is the one initially selected. Only one option may | ||||
|      * have this value as true (default: first option). | ||||
|      */ | ||||
|     private Boolean initial; | ||||
|  | ||||
|     public InputOption(@NonNull String id) | ||||
|     { | ||||
|         this( id, null, null ); | ||||
|     } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user