self::BLACK, self::DARK_BLUE => self::DARK_BLUE, self::DARK_GREEN => self::DARK_GREEN, self::DARK_AQUA => self::DARK_AQUA, self::DARK_RED => self::DARK_RED, self::DARK_PURPLE => self::DARK_PURPLE, self::GOLD => self::GOLD, self::GRAY => self::GRAY, self::DARK_GRAY => self::DARK_GRAY, self::BLUE => self::BLUE, self::GREEN => self::GREEN, self::AQUA => self::AQUA, self::RED => self::RED, self::LIGHT_PURPLE => self::LIGHT_PURPLE, self::YELLOW => self::YELLOW, self::WHITE => self::WHITE, self::MINECOIN_GOLD => self::MINECOIN_GOLD, ]; public const OBFUSCATED = TextFormat::ESCAPE . "k"; public const BOLD = TextFormat::ESCAPE . "l"; public const STRIKETHROUGH = TextFormat::ESCAPE . "m"; public const UNDERLINE = TextFormat::ESCAPE . "n"; public const ITALIC = TextFormat::ESCAPE . "o"; public const FORMATS = [ self::OBFUSCATED => self::OBFUSCATED, self::BOLD => self::BOLD, self::STRIKETHROUGH => self::STRIKETHROUGH, self::UNDERLINE => self::UNDERLINE, self::ITALIC => self::ITALIC, ]; public const RESET = TextFormat::ESCAPE . "r"; private static function makePcreError() : \InvalidArgumentException{ $errorCode = preg_last_error(); $message = [ PREG_INTERNAL_ERROR => "Internal error", PREG_BACKTRACK_LIMIT_ERROR => "Backtrack limit reached", PREG_RECURSION_LIMIT_ERROR => "Recursion limit reached", PREG_BAD_UTF8_ERROR => "Malformed UTF-8", PREG_BAD_UTF8_OFFSET_ERROR => "Bad UTF-8 offset", PREG_JIT_STACKLIMIT_ERROR => "PCRE JIT stack limit reached" ][$errorCode] ?? "Unknown (code $errorCode)"; throw new \InvalidArgumentException("PCRE error: $message"); } /** * @throws \InvalidArgumentException */ private static function preg_replace(string $pattern, string $replacement, string $string) : string{ $result = preg_replace($pattern, $replacement, $string); if($result === null){ throw self::makePcreError(); } return $result; } /** * Splits the string by Format tokens * * @return string[] */ public static function tokenize(string $string) : array{ $result = preg_split("/(" . TextFormat::ESCAPE . "[0-9a-gk-or])/u", $string, -1, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE); if($result === false) throw self::makePcreError(); return $result; } /** * Cleans the string from Minecraft codes, ANSI Escape Codes and invalid UTF-8 characters * * @return string valid clean UTF-8 */ public static function clean(string $string, bool $removeFormat = true) : string{ $string = mb_scrub($string, 'UTF-8'); $string = self::preg_replace("/[\x{E000}-\x{F8FF}]/u", "", $string); //remove unicode private-use-area characters (they might break the console) if($removeFormat){ $string = str_replace(TextFormat::ESCAPE, "", self::preg_replace("/" . TextFormat::ESCAPE . "[0-9a-gk-or]/u", "", $string)); } return str_replace("\x1b", "", self::preg_replace("/\x1b[\\(\\][[0-9;\\[\\(]+[Bm]/u", "", $string)); } /** * Replaces placeholders of § with the correct character. Only valid codes (as in the constants of the TextFormat class) will be converted. * * @param string $placeholder default "&" */ public static function colorize(string $string, string $placeholder = "&") : string{ return self::preg_replace('/' . preg_quote($placeholder, "/") . '([0-9a-gk-or])/u', TextFormat::ESCAPE . '$1', $string); } /** * Returns an HTML-formatted string with colors/markup */ public static function toHTML(string $string) : string{ $newString = ""; $tokens = 0; foreach(self::tokenize($string) as $token){ switch($token){ case TextFormat::BOLD: $newString .= ""; ++$tokens; break; case TextFormat::OBFUSCATED: //$newString .= ""; //++$tokens; break; case TextFormat::ITALIC: $newString .= ""; ++$tokens; break; case TextFormat::UNDERLINE: $newString .= ""; ++$tokens; break; case TextFormat::STRIKETHROUGH: $newString .= ""; ++$tokens; break; case TextFormat::RESET: $newString .= str_repeat("", $tokens); $tokens = 0; break; //Colors case TextFormat::BLACK: $newString .= ""; ++$tokens; break; case TextFormat::DARK_BLUE: $newString .= ""; ++$tokens; break; case TextFormat::DARK_GREEN: $newString .= ""; ++$tokens; break; case TextFormat::DARK_AQUA: $newString .= ""; ++$tokens; break; case TextFormat::DARK_RED: $newString .= ""; ++$tokens; break; case TextFormat::DARK_PURPLE: $newString .= ""; ++$tokens; break; case TextFormat::GOLD: $newString .= ""; ++$tokens; break; case TextFormat::GRAY: $newString .= ""; ++$tokens; break; case TextFormat::DARK_GRAY: $newString .= ""; ++$tokens; break; case TextFormat::BLUE: $newString .= ""; ++$tokens; break; case TextFormat::GREEN: $newString .= ""; ++$tokens; break; case TextFormat::AQUA: $newString .= ""; ++$tokens; break; case TextFormat::RED: $newString .= ""; ++$tokens; break; case TextFormat::LIGHT_PURPLE: $newString .= ""; ++$tokens; break; case TextFormat::YELLOW: $newString .= ""; ++$tokens; break; case TextFormat::WHITE: $newString .= ""; ++$tokens; break; case TextFormat::MINECOIN_GOLD: $newString .= ""; ++$tokens; break; default: $newString .= $token; break; } } $newString .= str_repeat("", $tokens); return $newString; } }