1
0
mirror of https://github.com/danog/MadelineProto.git synced 2025-01-12 10:18:18 +01:00
MadelineProto/src/StrTools.php

241 lines
9.1 KiB
PHP
Raw Normal View History

2022-12-30 21:54:44 +01:00
<?php
declare(strict_types=1);
2020-06-16 17:52:55 +02:00
/**
* Tools module.
*
* This file is part of MadelineProto.
* MadelineProto is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
* MadelineProto is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU Affero General Public License for more details.
* You should have received a copy of the GNU General Public License along with MadelineProto.
* If not, see <http://www.gnu.org/licenses/>.
*
* @author Daniil Gentili <daniil@daniil.it>
2023-01-04 12:43:01 +01:00
* @copyright 2016-2023 Daniil Gentili <daniil@daniil.it>
2020-06-16 17:52:55 +02:00
* @license https://opensource.org/licenses/AGPL-3.0 AGPLv3
* @link https://docs.madelineproto.xyz MadelineProto documentation
*/
namespace danog\MadelineProto;
2022-12-29 20:40:06 +01:00
use danog\MadelineProto\TL\Conversion\DOMEntities;
use danog\MadelineProto\TL\Conversion\Extension;
2022-05-01 20:17:16 +02:00
use Parsedown;
2023-01-11 18:47:27 +01:00
use Webmozart\Assert\Assert;
2020-06-16 17:52:55 +02:00
/**
* Some tools.
*/
abstract class StrTools extends Extension
2020-06-16 17:52:55 +02:00
{
2022-05-17 22:17:41 +02:00
/**
* Get Telegram UTF-8 length of string.
*
* @param string $text Text
*/
2023-01-11 18:47:27 +01:00
public static function mbStrlen(string $text): int
2022-05-17 22:17:41 +02:00
{
$length = 0;
$textlength = \strlen($text);
for ($x = 0; $x < $textlength; $x++) {
$char = \ord($text[$x]);
if (($char & 0xc0) != 0x80) {
2023-01-14 19:51:23 +01:00
$length += 1 + ($char >= 0xf0 ? 1 : 0);
2022-05-17 22:17:41 +02:00
}
}
return $length;
}
/**
* Telegram UTF-8 multibyte substring.
*
* @param string $text Text to substring
* @param integer $offset Offset
2023-01-20 15:30:13 +01:00
* @param null|int $length Length
2022-05-17 22:17:41 +02:00
*/
2022-12-30 19:21:36 +01:00
public static function mbSubstr(string $text, int $offset, ?int $length = null): string
2022-05-17 22:17:41 +02:00
{
return \mb_convert_encoding(
\substr(
\mb_convert_encoding($text, 'UTF-16'),
$offset<<1,
2022-12-30 19:21:36 +01:00
$length === null ? null : ($length<<1),
),
'UTF-8',
2022-12-30 19:21:36 +01:00
'UTF-16',
);
2022-05-17 22:17:41 +02:00
}
/**
* Telegram UTF-8 multibyte split.
*
* @param string $text Text
* @param integer $length Length
* @return array<string>
2022-05-17 22:17:41 +02:00
*/
public static function mbStrSplit(string $text, int $length): array
{
$result = [];
foreach (\str_split(\mb_convert_encoding($text, 'UTF-16'), $length<<1) as $chunk) {
2023-01-14 19:18:23 +01:00
$chunk = \mb_convert_encoding($chunk, 'UTF-8', 'UTF-16');
Assert::string($chunk);
$result []= $chunk;
2022-05-17 22:17:41 +02:00
}
return $result;
}
/**
2023-07-01 14:39:33 +02:00
* Manually convert HTML to a message and a set of entities.
*
* NOTE: You don't have to use this method to send HTML messages.
2023-07-01 18:04:20 +02:00
*
2023-07-01 14:39:33 +02:00
* This method is already called automatically by using parse_mode: "HTML" in messages.sendMessage, messages.sendMedia, et cetera...
*
* @see https://docs.madelineproto.xyz/API_docs/methods/messages.sendMessage.html#usage-of-parse_mode
*
2023-07-05 21:28:17 +02:00
* @return \danog\MadelineProto\TL\Conversion\DOMEntities Object containing message and entities
*/
2023-07-05 21:28:17 +02:00
public static function htmlToMessageEntities(string $html): \danog\MadelineProto\TL\Conversion\DOMEntities
{
return new DOMEntities($html);
}
/**
2023-07-01 14:39:33 +02:00
* Manually convert markdown to a message and a set of entities.
*
* NOTE: You don't have to use this method to send Markdown messages.
2023-07-01 18:04:20 +02:00
*
2023-07-01 14:39:33 +02:00
* This method is already called automatically by using parse_mode: "Markdown" in messages.sendMessage, messages.sendMedia, et cetera...
*
* @see https://docs.madelineproto.xyz/API_docs/methods/messages.sendMessage.html#usage-of-parse_mode
*
2023-07-05 21:28:17 +02:00
* @return \danog\MadelineProto\TL\Conversion\DOMEntities Object containing message and entities
*/
2023-07-05 21:28:17 +02:00
public static function markdownToMessageEntities(string $markdown): \danog\MadelineProto\TL\Conversion\DOMEntities
{
return new DOMEntities(Parsedown::instance()->line($markdown));
}
/**
* Convert a message and a set of entities to HTML.
*
2023-07-01 14:39:33 +02:00
* @param bool $allowTelegramTags Whether to allow telegram-specific tags like tg-spoiler, tg-emoji, mention links and so on...
*/
public static function messageEntitiesToHtml(string $message, array $entities, bool $allowTelegramTags = false): string
{
$insertions = [];
foreach ($entities as $entity) {
['_' => $type, 'offset' => $offset, 'length' => $length] = $entity;
$insertions[$offset] ??= '';
$insertions[$offset] .= match ($type) {
'messageEntityBold' => '<b>',
'messageEntityItalic' => '<i>',
'messageEntityCode' => '<code>',
'messageEntityPre' => $entity['language'] !== '' ? '<pre language="'.$entity['language'].'">' : '<pre>',
'messageEntityTextUrl' => '<a href="'.$entity['url'].'">',
'messageEntityStrike' => '<s>',
'messageEntityUnderline' => '<u>',
'messageEntityBlockquote' => '<blockquote>',
2023-07-01 17:53:29 +02:00
'messageEntityUrl' => '<a href="'.\htmlspecialchars(self::mbSubstr($message, $offset, $length)).'">',
2023-07-01 18:04:20 +02:00
'messageEntityEmail' => '<a href="mailto:'.\htmlspecialchars(self::mbSubstr($message, $offset, $length)).'">',
'messageEntityPhone' => '<a href="phone:'.\htmlspecialchars(self::mbSubstr($message, $offset, $length)).'">',
'messageEntityMention' => '<a href="https://t.me/'.\htmlspecialchars(self::mbSubstr($message, $offset+1, $length-1)).'">',
'messageEntitySpoiler' => $allowTelegramTags ? '<tg-spoiler>' : '',
'messageEntityCustomEmoji' => $allowTelegramTags ? '<tg-emoji emoji-id="'.$entity['document_id'].'">' : '',
'messageEntityMentionName' => $allowTelegramTags ? '<a href="tg://user?id='.$entity['user_id'].'">' : '',
default => '',
};
$offset += $length;
$insertions[$offset] = match ($type) {
'messageEntityBold' => '</b>',
'messageEntityItalic' => '</i>',
'messageEntityCode' => '</code>',
'messageEntityPre' => '</pre>',
'messageEntityTextUrl', 'messageEntityUrl', 'messageEntityEmail', 'messageEntityMention', 'messageEntityPhone' => '</a>',
'messageEntityStrike' => '</s>',
'messageEntityUnderline' => '</u>',
'messageEntityBlockquote' => '</blockquote>',
'messageEntitySpoiler' => $allowTelegramTags ? '</tg-spoiler>' : '',
'messageEntityCustomEmoji' => $allowTelegramTags ? "</tg-emoji>" : '',
'messageEntityMentionName' => $allowTelegramTags ? '</a>' : '',
default => '',
} . ($insertions[$offset] ?? '');
}
\ksort($insertions);
$final = '';
$pos = 0;
foreach ($insertions as $offset => $insertion) {
2023-07-01 14:24:50 +02:00
$final .= \htmlspecialchars(StrTools::mbSubstr($message, $pos, $offset-$pos));
$final .= $insertion;
$pos = $offset;
}
2023-07-01 14:24:50 +02:00
return \str_replace("\n", "<br>", $final.\htmlspecialchars(StrTools::mbSubstr($message, $pos)));
}
2020-06-16 17:52:55 +02:00
/**
* Convert to camelCase.
*
* @param string $input String
*/
public static function toCamelCase(string $input): string
{
return \lcfirst(\str_replace('_', '', \ucwords($input, '_')));
}
/**
* Convert to snake_case.
*
* @param string $input String
*/
public static function toSnakeCase(string $input): string
{
\preg_match_all('!([A-Z][A-Z0-9]*(?=$|[A-Z][a-z0-9])|[A-Za-z][a-z0-9]+)!', $input, $matches);
$ret = $matches[0];
foreach ($ret as &$match) {
$match = $match == \strtoupper($match) ? \strtolower($match) : \lcfirst($match);
}
return \implode('_', $ret);
}
/**
* Escape string for markdown.
*
2023-06-13 22:43:56 +02:00
* @internal
*
2020-06-16 17:52:55 +02:00
* @param string $hwat String to escape
*/
public static function markdownEscape(string $hwat): string
{
return \str_replace('_', '\\_', $hwat);
}
/**
* Escape type name.
*
2023-06-13 22:43:56 +02:00
* @internal
*
2020-06-16 17:52:55 +02:00
* @param string $type String to escape
*/
public static function typeEscape(string $type): string
{
$type = \str_replace(['<', '>'], ['_of_', ''], $type);
return \preg_replace('/.*_of_/', '', $type);
}
/**
* Escape method name.
*
2023-06-13 22:43:56 +02:00
* @internal
*
2020-06-16 17:52:55 +02:00
* @param string $method Method name
*/
public static function methodEscape(string $method): string
{
return \str_replace('.', '->', $method);
}
2022-05-01 20:17:16 +02:00
/**
* Strip markdown tags.
*
* @internal
*/
public static function toString(string $markdown): string
{
if ($markdown === '') {
return $markdown;
}
2022-12-29 20:40:06 +01:00
return (new DOMEntities(Parsedown::instance()->text($markdown)))->message;
2022-05-01 20:17:16 +02:00
}
2020-06-16 17:52:55 +02:00
}