Skip to content

Commit bb16e49

Browse files
authored
Added Gradients (#99)
## Description Expanded the Color markdown by adding Gradients! Users can apply gradients by using -#hex1,#hex2,#hex3-(text) aka -#000000,#FFFFFF,#0000FF-(this message goes from Black to White to Blue) ## Changes - **Change 1**: Added Gradiants - **Change 2**: Added Respective Test Cases - ## How has this been tested? I wrote Automated Tests in the `Test` Package and i have run them all ## Screenshots (if applicable) ![image](https://github.com/user-attachments/assets/72b47d53-3b8f-4c73-bf7d-140ac39e3cd7) ## Type of Change - [ ] Bug fix - [x] New feature - [ ] Cleanup - [ ] Breaking change - [ ] Documentation update ## Checklist - [x] My code follows the style guidelines of this project. [Contributing](https://github.com/MeAlam1/BlueLib/blob/1.21/CONTRIBUTING.md) - [x] I have performed a self-review of my own code. - [x] I have commented my code following the guidelines. [Contributing](https://github.com/MeAlam1/BlueLib/blob/1.21/CONTRIBUTING.md) - [x] My changes generate no new warnings. ## Related Issues #61
1 parent 74c58a8 commit bb16e49

11 files changed

+222
-19
lines changed

common/src/main/java/software/bluelib/markdown/syntax/Color.java

+110-15
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,9 @@
22

33
package software.bluelib.markdown.syntax;
44

5+
import java.util.ArrayList;
6+
import java.util.List;
7+
import java.util.regex.Matcher;
58
import java.util.regex.Pattern;
69
import net.minecraft.network.chat.Component;
710
import net.minecraft.network.chat.MutableComponent;
@@ -26,7 +29,9 @@
2629
* <li>{@link #apply(MutableComponent)} - Applies the color feature to a given component.</li>
2730
* <li>{@link #processComponentTextWithColors(String, Style, MutableComponent, Pattern)} - Processes text with color formatting.</li>
2831
* <li>{@link #processSiblingsWithColors(MutableComponent, Pattern)} - Processes siblings with color formatting.</li>
29-
* <li>{@link #appendColor(String, String, Style, MutableComponent)} - Appends color formatted text to a component.</li>
32+
* <li>{@link #appendColor(String, List, Style, MutableComponent)} - Appends color formatted text to a component.</li>
33+
* <li>{@link #interpolateColor(int, int, float)} - Interpolates a color between two colors.</li>
34+
* <li>{@link #extractColorsFromMatcher(Matcher)} - Extracts colors from a matcher.</li>
3035
* <li>{@link #isFeatureEnabled()} - Checks if the feature is enabled.</li>
3136
* <li>{@link #getFeatureName()} - Gets the name of the feature.</li>
3237
* <li>{@link #setPrefixSuffix(String, String)} - Sets the prefix and suffix for color formatting.</li>
@@ -125,7 +130,9 @@ public MutableComponent apply(MutableComponent pComponent) {
125130
return pComponent;
126131
}
127132

128-
Pattern pattern = Pattern.compile(Pattern.quote(getPrefix()) + "(#[0-9A-Fa-f]{6})" + Pattern.quote(getSuffix()) + "\\((.*?)\\)");
133+
Pattern pattern = Pattern.compile(Pattern.quote(getPrefix()) +
134+
"#([0-9A-Fa-f]{6}(?:,#([0-9A-Fa-f]{6}))*)" +
135+
Pattern.quote(getSuffix()) + "\\((.*?)\\)");
129136

130137
MutableComponent result = Component.empty();
131138

@@ -162,14 +169,43 @@ public MutableComponent apply(MutableComponent pComponent) {
162169
protected void processComponentTextWithColors(String pText, Style pOriginalStyle, MutableComponent pResult, Pattern pPattern) {
163170
processComponentText(pText, pOriginalStyle, pResult, pPattern,
164171
(matcher, res) -> {
165-
String color = matcher.group(1);
166-
String colorText = matcher.group(2);
167-
if (color != null && !color.isEmpty()) {
168-
appendColor(colorText, color, pOriginalStyle, res);
169-
}
172+
List<Integer> colors = extractColorsFromMatcher(matcher);
173+
String gradientText = matcher.group(matcher.groupCount());
174+
175+
appendColor(gradientText, colors, pOriginalStyle, res);
170176
});
171177
}
172178

179+
/**
180+
* Extracts colors from a matcher.
181+
* <p>
182+
* Purpose: This method extracts colors from a matcher and returns them as a list of integers.<br>
183+
* When: It is called to extract colors from a matcher.<br>
184+
* Where: It is invoked in {@link #processComponentTextWithColors} to extract colors from a matcher.<br>
185+
* Additional Info: The method ensures that only valid colors are extracted and converted to hexadecimal format.<br>
186+
* </p>
187+
*
188+
* @param matcher The matcher to extract colors from.
189+
* @return A list of integers representing the extracted colors in hexadecimal format.
190+
* @see #processComponentTextWithColors
191+
* @since 1.7.0
192+
*/
193+
private List<Integer> extractColorsFromMatcher(Matcher matcher) {
194+
List<Integer> colors = new ArrayList<>();
195+
196+
String colorGroup = matcher.group(1);
197+
String[] colorArray = colorGroup.split(",");
198+
for (String color : colorArray) {
199+
if (IsValidUtils.isValidColor(color)) {
200+
colors.add(ColorConversionUtils.parseColorToHexString(color));
201+
}
202+
}
203+
204+
System.out.println("Colors: " + colors);
205+
206+
return colors;
207+
}
208+
173209
/**
174210
* Appends color formatted text to a component.
175211
* <p>
@@ -179,8 +215,8 @@ protected void processComponentTextWithColors(String pText, Style pOriginalStyle
179215
* Additional Info: The method ensures that the appropriate style is applied to the appended text.<br>
180216
* </p>
181217
*
182-
* @param colorText The text to be appended.
183-
* @param pColor The color to be applied to the text.
218+
* @param pColorText The text to be appended.
219+
* @param pColors List of all colors that will be applied to the text.
184220
* @param pOriginalStyle The original style of the component.
185221
* @param pResult The component to append the formatted text to.
186222
* @author MeAlam
@@ -192,13 +228,72 @@ protected void processComponentTextWithColors(String pText, Style pOriginalStyle
192228
* @see TextColor#fromRgb(int)
193229
* @since 1.6.0
194230
*/
195-
private void appendColor(String colorText, String pColor, Style pOriginalStyle, MutableComponent pResult) {
196-
if (IsValidUtils.isValidColor(pColor)) {
197-
pResult.append(Component.literal(colorText)
198-
.setStyle(pOriginalStyle.withColor(TextColor.fromRgb(ColorConversionUtils.parseColorToHexString(pColor)))));
199-
} else {
200-
pResult.append(Component.literal(colorText).setStyle(pOriginalStyle));
231+
private void appendColor(String pColorText, List<Integer> pColors, Style pOriginalStyle, MutableComponent pResult) {
232+
if (pColors.isEmpty()) {
233+
pResult.append(Component.literal(pColorText).setStyle(pOriginalStyle));
234+
return;
201235
}
236+
237+
if (pColors.size() == 1) {
238+
int color = pColors.get(0);
239+
pResult.append(Component.literal(pColorText).setStyle(pOriginalStyle.withColor(TextColor.fromRgb(color))));
240+
return;
241+
}
242+
243+
char[] characters = pColorText.toCharArray();
244+
int textLength = characters.length;
245+
int colorCount = pColors.size();
246+
int segmentLength = textLength / (colorCount - 1);
247+
int remainder = textLength % (colorCount - 1);
248+
249+
int charIndex = 0;
250+
251+
for (int colorIndex = 0; colorIndex < colorCount - 1; colorIndex++) {
252+
int startColor = pColors.get(colorIndex);
253+
int endColor = pColors.get(colorIndex + 1);
254+
255+
int currentSegmentLength = segmentLength + (colorIndex < remainder ? 1 : 0);
256+
257+
for (int i = 0; i < currentSegmentLength && charIndex < textLength; i++, charIndex++) {
258+
float positionRatio = (float) i / (currentSegmentLength - 1);
259+
int interpolatedColor = interpolateColor(startColor, endColor, positionRatio);
260+
261+
pResult.append(Component.literal(String.valueOf(characters[charIndex]))
262+
.setStyle(pOriginalStyle.withColor(TextColor.fromRgb(interpolatedColor))));
263+
}
264+
}
265+
}
266+
267+
/**
268+
* Interpolates a color between two colors.
269+
* <p>
270+
* Purpose: This method interpolates a color between two colors based on a given ratio.<br>
271+
* When: It is called to interpolate a color between two colors.<br>
272+
* Where: It is invoked in {@link #appendColor} to interpolate a color between two colors.<br>
273+
* Additional Info: The method calculates the interpolated color based on the start and end colors and the given ratio.<br>
274+
* </p>
275+
*
276+
* @param startColor The starting color.
277+
* @param endColor The ending color.
278+
* @param ratio The ratio used to interpolate the color.
279+
* @return The interpolated color between the start and end colors based on the ratio.
280+
* @see #appendColor
281+
* @since 1.6.0
282+
*/
283+
private int interpolateColor(int startColor, int endColor, float ratio) {
284+
int startR = (startColor >> 16) & 0xFF;
285+
int startG = (startColor >> 8) & 0xFF;
286+
int startB = startColor & 0xFF;
287+
288+
int endR = (endColor >> 16) & 0xFF;
289+
int endG = (endColor >> 8) & 0xFF;
290+
int endB = endColor & 0xFF;
291+
292+
int r = (int) (startR + (endR - startR) * ratio);
293+
int g = (int) (startG + (endG - startG) * ratio);
294+
int b = (int) (startB + (endB - startB) * ratio);
295+
296+
return (r << 16) | (g << 8) | b;
202297
}
203298

204299
/**

common/src/main/java/software/bluelib/test/markdown/Markdown.java

+23
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ public static void bold(GameTestHelper pHelper) {
1717
BoldTest.boldHyperlink(pHelper);
1818
BoldTest.boldColor(pHelper);
1919
BoldTest.boldSpoiler(pHelper);
20+
BoldTest.boldGradient(pHelper);
2021
BoldTest.boldCancel(pHelper);
2122
});
2223
}
@@ -32,6 +33,7 @@ public static void italic(GameTestHelper pHelper) {
3233
ItalicTest.italicHyperlink(pHelper);
3334
ItalicTest.italicColor(pHelper);
3435
ItalicTest.italicSpoiler(pHelper);
36+
ItalicTest.italicGradient(pHelper);
3537
ItalicTest.italicCancel(pHelper);
3638
});
3739
}
@@ -47,6 +49,7 @@ public static void underline(GameTestHelper pHelper) {
4749
UnderlineTest.underlineHyperlink(pHelper);
4850
UnderlineTest.underlineColor(pHelper);
4951
UnderlineTest.underlineSpoiler(pHelper);
52+
UnderlineTest.underlineGradient(pHelper);
5053
UnderlineTest.underlineCancel(pHelper);
5154
});
5255
}
@@ -62,6 +65,7 @@ public static void strikethrough(GameTestHelper pHelper) {
6265
StrikethroughTest.strikethroughHyperlink(pHelper);
6366
StrikethroughTest.strikethroughColor(pHelper);
6467
StrikethroughTest.strikethroughSpoiler(pHelper);
68+
StrikethroughTest.strikethroughGradient(pHelper);
6569
StrikethroughTest.strikethroughCancel(pHelper);
6670
});
6771
}
@@ -77,6 +81,7 @@ public static void hyperlink(GameTestHelper pHelper) {
7781
HyperlinkTest.hyperlinkHyperlink(pHelper);
7882
HyperlinkTest.hyperlinkColor(pHelper);
7983
HyperlinkTest.hyperlinkSpoiler(pHelper);
84+
HyperlinkTest.hyperlinkGradient(pHelper);
8085
HyperlinkTest.hyperlinkCancel(pHelper);
8186
HyperlinkTest.hyperlinkInvalid(pHelper);
8287
});
@@ -93,6 +98,7 @@ public static void color(GameTestHelper pHelper) {
9398
ColorTest.colorHyperlink(pHelper);
9499
ColorTest.colorColor(pHelper);
95100
ColorTest.colorSpoiler(pHelper);
101+
ColorTest.colorGradient(pHelper);
96102
ColorTest.colorCancel(pHelper);
97103
ColorTest.colorInvalid(pHelper);
98104
});
@@ -109,10 +115,27 @@ public static void spoiler(GameTestHelper pHelper) {
109115
SpoilerTest.spoilerHyperlink(pHelper);
110116
SpoilerTest.spoilerColor(pHelper);
111117
SpoilerTest.spoilerSpoiler(pHelper);
118+
SpoilerTest.spoilerGradient(pHelper);
112119
SpoilerTest.spoilerCancel(pHelper);
113120
});
114121
}
115122

123+
@GameTest
124+
public static void gradient(GameTestHelper pHelper) {
125+
pHelper.succeedIf(() -> {
126+
GradientTest.gradient(pHelper);
127+
GradientTest.gradientBold(pHelper);
128+
GradientTest.gradientItalic(pHelper);
129+
GradientTest.gradientUnderline(pHelper);
130+
GradientTest.gradientStrikethrough(pHelper);
131+
GradientTest.gradientHyperlink(pHelper);
132+
GradientTest.gradientColor(pHelper);
133+
GradientTest.gradientSpoiler(pHelper);
134+
GradientTest.gradientGradient(pHelper);
135+
GradientTest.gradientCancel(pHelper);
136+
});
137+
}
138+
116139
@GameTest
117140
public static void all(GameTestHelper pHelper) {
118141
pHelper.succeedIf(() -> {

common/src/main/java/software/bluelib/test/markdown/MarkdownStyleTest.java

+10-4
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,9 @@ public class MarkdownStyleTest {
2121
"-#" + MessageUtils.getRandomHex() + "-(Color)", // Color
2222
"\\-#" + MessageUtils.getRandomHex() + "-(Color)", // Color Canceled
2323
"||Spoiler||", // Spoiler
24-
"\\||Spoiler||" // Spoiler Canceled
24+
"\\||Spoiler||", // Spoiler Canceled
25+
"-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)", // Gradient
26+
"\\-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)" // Gradient Canceled
2527
);
2628

2729
private static final List<String> CANCEL_ONLY_STYLES = List.of(
@@ -31,7 +33,8 @@ public class MarkdownStyleTest {
3133
"\\~~Strikethrough~~", // Strikethrough Canceled
3234
"\\[Hyperlink](https://modrinth.com/mod/bluelib)", // Hyperlink Canceled
3335
"\\-#" + MessageUtils.getRandomHex() + "-(Color)", // Color Canceled
34-
"\\||Spoiler||" // Spoiler Canceled
36+
"\\||Spoiler||", // Spoiler Canceled
37+
"\\-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)" // Gradient Canceled
3538
);
3639

3740
private static final List<String> ALL_TEST_STYLES = List.of(
@@ -41,7 +44,8 @@ public class MarkdownStyleTest {
4144
"~~Strikethrough~~", // Strikethrough
4245
"[Hyperlink](https://www.curseforge.com/minecraft/mc-mods/bluelib)", // Hyperlink
4346
"-#" + MessageUtils.getRandomHex() + "-(Color)", // Color
44-
"||Spoiler||" // Spoiler
47+
"||Spoiler||", // Spoiler
48+
"-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)" // Gradient
4549
);
4650

4751
private static final List<String> ALL_TWICE_STYLES = List.of(
@@ -58,7 +62,9 @@ public class MarkdownStyleTest {
5862
"-#" + MessageUtils.getRandomHex() + "-(Color)", // Color
5963
"-#" + MessageUtils.getRandomHex() + "-(Color)", // Color
6064
"||Spoiler||", // Spoiler
61-
"||Spoiler||" // Spoiler
65+
"||Spoiler||", // Spoiler
66+
"-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)", // Gradient
67+
"-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)" // Gradient
6268
);
6369

6470
public static void testAllStyles(GameTestHelper pHelper) {

common/src/main/java/software/bluelib/test/markdown/syntax/BoldTest.java

+4
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,10 @@ public static void boldSpoiler(GameTestHelper pHelper) {
3737
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a bold/spoiler test: §r **bold** ||spoiler||");
3838
}
3939

40+
public static void boldGradient(GameTestHelper pHelper) {
41+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a bold/gradient test: §r **bold** -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
42+
}
43+
4044
public static void boldCancel(GameTestHelper pHelper) {
4145
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a canceled bold test: §r \\**bold**");
4246
}

common/src/main/java/software/bluelib/test/markdown/syntax/ColorTest.java

+4
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,10 @@ public static void colorSpoiler(GameTestHelper pHelper) {
3737
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a color/spoiler test: §r -#" + MessageUtils.getRandomHex() + "-(Color) ||spoiler||");
3838
}
3939

40+
public static void colorGradient(GameTestHelper pHelper) {
41+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a color/gradient test: §r -#" + MessageUtils.getRandomHex() + "-(Color) -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
42+
}
43+
4044
public static void colorCancel(GameTestHelper pHelper) {
4145
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a canceled color test: §r \\-#" + MessageUtils.getRandomHex() + "-(Color)");
4246
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
package software.bluelib.test.markdown.syntax;
2+
3+
import net.minecraft.gametest.framework.GameTestHelper;
4+
import software.bluelib.test.utils.MessageUtils;
5+
6+
public class GradientTest {
7+
8+
public static void gradient(GameTestHelper pHelper) {
9+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
10+
}
11+
12+
public static void gradientBold(GameTestHelper pHelper) {
13+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/bold test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) **bold**");
14+
}
15+
16+
public static void gradientItalic(GameTestHelper pHelper) {
17+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/italic test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) *italic*");
18+
}
19+
20+
public static void gradientUnderline(GameTestHelper pHelper) {
21+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/underline test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) __Underline__");
22+
}
23+
24+
public static void gradientStrikethrough(GameTestHelper pHelper) {
25+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/strikethrough test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) ~~Strikethrough~~");
26+
}
27+
28+
public static void gradientHyperlink(GameTestHelper pHelper) {
29+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/hyperlink test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) [Hyperlink](https://www.curseforge.com/minecraft/mc-mods/bluelib)");
30+
}
31+
32+
public static void gradientColor(GameTestHelper pHelper) {
33+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/color test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) -#" + MessageUtils.getRandomHex() + "-(color)");
34+
}
35+
36+
public static void gradientSpoiler(GameTestHelper pHelper) {
37+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/spoiler test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) ||spoiler||");
38+
}
39+
40+
public static void gradientGradient(GameTestHelper pHelper) {
41+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a gradient/gradient test: §r -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient) -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
42+
}
43+
44+
public static void gradientCancel(GameTestHelper pHelper) {
45+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a canceled gradient test: §r \\-#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
46+
}
47+
48+
public static void gradientInvalid(GameTestHelper pHelper) {
49+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is an invalid gradient test: §r -#invalidgradient-(gradient)");
50+
}
51+
}

common/src/main/java/software/bluelib/test/markdown/syntax/HyperlinkTest.java

+4
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,10 @@ public static void hyperlinkSpoiler(GameTestHelper pHelper) {
3737
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a hyperlink/spoiler test: §r [Hyperlink](https://github.com/MeAlam1/BlueLib/issues) ||spoiler||");
3838
}
3939

40+
public static void hyperlinkGradient(GameTestHelper pHelper) {
41+
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a hyperlink/gradient test: §r [Hyperlink](https://www.curseforge.com/minecraft/mc-mods/bluelib) -#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + ",#" + MessageUtils.getRandomHex() + "-(Gradient)");
42+
}
43+
4044
public static void hyperlinkCancel(GameTestHelper pHelper) {
4145
MessageUtils.sendMessageToPlayers(pHelper, "§6 This is a canceled hyperlink test: §r \\[Hyperlink](https://www.curseforge.com/minecraft/mc-mods/bluelib)");
4246
}

0 commit comments

Comments
 (0)