Helpers.php 127 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285
  1. <?php
  2. use GuzzleHttp\Client;
  3. use Illuminate\Support\Facades\DB;
  4. use Illuminate\Support\Facades\Log;
  5. use Monolog\Handler\StreamHandler;
  6. use Monolog\Logger;
  7. use OSS\OssClient;
  8. use OSS\Core\OssException;
  9. use Tos\TosClient;
  10. use Tos\Exception\TosClientException;
  11. use Tos\Exception\TosServerException;
  12. use Tos\Model\PutObjectInput;
  13. // use Imagick;
  14. /**
  15. * 截取字符串
  16. *
  17. * @param $string
  18. * @param $length
  19. * @param bool $append
  20. * @return array|string
  21. */
  22. function sysSubStr($string, $length, $append = false)
  23. {
  24. if (strlen($string) <= $length) {
  25. return $string;
  26. }
  27. $i = 0;
  28. $stringLast = [];
  29. while ($i < $length) {
  30. $stringTMP = mb_substr($string, $i, 1);
  31. if (ord($stringTMP) >= 224) {
  32. $stringTMP = mb_substr($string, $i, 3);
  33. $i += 3;
  34. } elseif (ord($stringTMP) >= 192) {
  35. $stringTMP = mb_substr($string, $i, 2);
  36. $i += 2;
  37. } else {
  38. ++$i;
  39. }
  40. $stringLast[] = $stringTMP;
  41. }
  42. $stringLast = implode('', $stringLast);
  43. if ($append) {
  44. $stringLast .= '...';
  45. }
  46. return $stringLast;
  47. }
  48. /**
  49. * 自定义日志
  50. *
  51. * @param $name
  52. * @param string $filename
  53. * @return \Illuminate\Log\Writer
  54. */
  55. function myLog($name, $filename = '')
  56. {
  57. if (!$filename) {
  58. $filename = $name;
  59. }
  60. $filename = $filename . '.log';
  61. $logger = new Logger($name);
  62. $writer = new \Illuminate\Log\Writer($logger);
  63. $writer->useDailyFiles(storage_path('logs/' . $filename));
  64. return $writer;
  65. }
  66. /**
  67. * 业务调试日志
  68. *
  69. * @return Logger
  70. */
  71. function commonLog()
  72. {
  73. $name = 'common';
  74. $filename = $name . '.log';
  75. $logger = new Logger($name);
  76. $logger->pushHandler(new StreamHandler(storage_path('logs/' . $filename), 'debug'));
  77. return $logger;
  78. }
  79. /**
  80. * @param $name
  81. * @return Logger
  82. */
  83. function dLog($name)
  84. {
  85. $filename = $name . '-' . date('Y-m-d') . '.log';
  86. $logger = new Logger($name);
  87. $logger->pushHandler(new StreamHandler(storage_path('logs/' . $filename), 'debug'));
  88. return $logger;
  89. }
  90. /**
  91. * 获取对象或数组的属性值
  92. *
  93. * @param $param
  94. * @param $key
  95. * @param string $default
  96. * @return mixed|string
  97. */
  98. function getProp($param, $key, $default = '')
  99. {
  100. $result = $default;
  101. if (is_object($param) && isset($param->$key)) {
  102. $result = $param->$key;
  103. }
  104. if (is_array($param) && isset($param[$key])) {
  105. $result = $param[$key];
  106. }
  107. return $result;
  108. }
  109. /**
  110. * @param $data
  111. * @param array $trans
  112. * @return array
  113. */
  114. function assetData($data, $trans = []): array
  115. {
  116. $result = [];
  117. if (empty($data)) {
  118. return $result;
  119. }
  120. if (empty($trans)) {
  121. return $data;
  122. }
  123. foreach ($trans as $tran) {
  124. [$originKey, $newKey, $conv, $default] = [$tran['o'], $tran['n'], $tran['conv'], $tran['default']];
  125. if (isset($data[$originKey])) {
  126. $result[$newKey] = $conv(getProp($data, $originKey, $default));
  127. }
  128. }
  129. return $result;
  130. }
  131. /**
  132. * 随机数
  133. *
  134. * @param int $num
  135. * @return string
  136. * @throws Exception
  137. */
  138. function random($num = 16)
  139. {
  140. $bytes = random_bytes($num);
  141. return bin2hex($bytes);
  142. }
  143. /**
  144. * 生成订单
  145. *
  146. * @param string $prefix 订单前缀
  147. * @return string
  148. */
  149. function generateOrderSn($prefix = '')
  150. {
  151. return $prefix . date('YmdHis') . getMillisecond() . rand(1000, 9999);
  152. }
  153. /**
  154. * 分页数据
  155. *
  156. * @param $data
  157. * @return array
  158. */
  159. function getMeta($data)
  160. {
  161. $currentPage = (int)$data->currentPage();
  162. $lastPage = (int)$data->lastPage();
  163. return [
  164. 'current_page' => $currentPage,
  165. 'next_page' => $currentPage >= $lastPage ? $lastPage : ++$currentPage,
  166. 'last_page' => $lastPage,
  167. 'per_page' => (int)$data->perPage(),
  168. 'total' => (int)$data->total(),
  169. 'is_end' => !$data->hasMorePages(),
  170. 'next_page_url' => (string)$data->nextPageUrl(),
  171. 'prev_page_url' => (string)$data->previousPageUrl()
  172. ];
  173. }
  174. /**
  175. * 钉钉通知异常
  176. *
  177. * @param $message
  178. */
  179. function sendNotice($message)
  180. {
  181. $webHook = env('DD_WEB_HOOK');
  182. $data = [
  183. 'msgtype' => 'text',
  184. 'text' => [
  185. 'content' => $message
  186. ],
  187. 'at' => [
  188. 'isAll' => true
  189. ]
  190. ];
  191. // 异步发送
  192. $client = new GuzzleHttp\Client();
  193. $client->post($webHook, ['json' => $data]);
  194. }
  195. /**
  196. * 判断数据是合法的json数据
  197. *
  198. * @param $string
  199. * @return bool
  200. */
  201. function is_json($string)
  202. {
  203. json_decode($string);
  204. return (json_last_error() == JSON_ERROR_NONE);
  205. }
  206. /**
  207. * 获取ip地址
  208. *
  209. * @return mixed|string
  210. */
  211. function getIpAddr()
  212. {
  213. $ipaddress = '';
  214. if (isset($_SERVER['HTTP_CLIENT_IP']))
  215. $ipaddress = $_SERVER['HTTP_CLIENT_IP'];
  216. else if (isset($_SERVER['HTTP_X_FORWARDED_FOR']))
  217. $ipaddress = $_SERVER['HTTP_X_FORWARDED_FOR'];
  218. else if (isset($_SERVER['HTTP_X_FORWARDED']))
  219. $ipaddress = $_SERVER['HTTP_X_FORWARDED'];
  220. else if (isset($_SERVER['HTTP_FORWARDED_FOR']))
  221. $ipaddress = $_SERVER['HTTP_FORWARDED_FOR'];
  222. else if (isset($_SERVER['HTTP_FORWARDED']))
  223. $ipaddress = $_SERVER['HTTP_FORWARDED'];
  224. else if (isset($_SERVER['REMOTE_ADDR']))
  225. $ipaddress = $_SERVER['REMOTE_ADDR'];
  226. else
  227. $ipaddress = 'UNKNOWN';
  228. return $ipaddress;
  229. }
  230. /**
  231. * @param int $num
  232. * @return bool|string
  233. * @throws Exception
  234. */
  235. function randStr($num = 5)
  236. {
  237. $str = 'QWERTYUIOPASDFGHJKLZXCVBNM1234567890qwertyuiopasdfghjklzxcvbnm';
  238. return substr(str_shuffle($str), random_int(0, strlen($str) - 11), $num);
  239. }
  240. /**
  241. * 上传文件(火山云tos)
  242. *
  243. * @param $prefix 文件夹前缀
  244. * @param $file 文件二进制
  245. * @param $filename 文件名(不带后缀)
  246. * @return mixed
  247. */
  248. function uploadFileByTos($prefix, $file, $filename='')
  249. {
  250. if (!$filename) $filename = randStr(10) . '.' . $file->getClientOriginalExtension();
  251. else $filename = $filename . '.' . $file->getClientOriginalExtension();
  252. try {
  253. $client = new TosClient([
  254. 'region' => env('VOLC_REGION'),
  255. 'endpoint' => env('VOLC_END_POINT'),
  256. 'ak' => env('VOLC_AK'),
  257. 'sk' => env('VOLC_SK'),
  258. ]);
  259. $stream = fopen($file->getRealPath(), 'r');
  260. $input = new PutObjectInput(env('VOLC_BUCKET'), "$prefix/$filename", $stream);
  261. $output = $client->putObject($input);
  262. if ($output->getRequestId()) {
  263. return "https://".env('VOLC_BUCKET').'.'.env('VOLC_END_POINT')."/$prefix/$filename";
  264. }
  265. } catch (TosClientException $ex) {
  266. dLog('exception')->info('saveToTos', [$ex->getMessage()]);
  267. } catch (TosServerException $ex) {
  268. dLog('exception')->info('saveToTos', [$ex->getRequestId(), $ex->getStatusCode(), $ex->getErrorCode()]);
  269. }
  270. return '';
  271. }
  272. /**
  273. * 上传文件流(火山云tos)
  274. *
  275. * @param $prefix 文件夹前缀
  276. * @param $file 文件二进制
  277. * @param $filename 文件名(带后缀)
  278. * @return mixed
  279. */
  280. function uploadStreamByTos($prefix, $stream, $filename)
  281. {
  282. try {
  283. $client = new TosClient([
  284. 'region' => env('VOLC_REGION'),
  285. 'endpoint' => env('VOLC_END_POINT'),
  286. 'ak' => env('VOLC_AK'),
  287. 'sk' => env('VOLC_SK'),
  288. ]);
  289. $input = new PutObjectInput(env('VOLC_BUCKET'), "$prefix/$filename", $stream);
  290. $output = $client->putObject($input);
  291. if ($output->getRequestId()) {
  292. return "https://".env('VOLC_BUCKET').'.'.env('VOLC_END_POINT')."/$prefix/$filename";
  293. }
  294. } catch (TosClientException $ex) {
  295. dLog('exception')->info('saveToTos', [$ex->getMessage()]);
  296. } catch (TosServerException $ex) {
  297. dLog('exception')->info('saveToTos', [$ex->getRequestId(), $ex->getStatusCode(), $ex->getErrorCode()]);
  298. }
  299. return '';
  300. }
  301. // 从图片URL获取图片扩展名
  302. function getImgExtFromUrl(string $url): ?string {
  303. $imageInfo = @getimagesize($url);
  304. if ($imageInfo !== false) {
  305. // 返回MIME类型
  306. return mimeToExt($imageInfo['mime']);
  307. // 或者返回文件扩展名
  308. // return image_type_to_extension($imageInfo[2], false);
  309. }
  310. return null;
  311. }
  312. /**
  313. * 从视频URL获取视频扩展名
  314. * 通过下载文件头部字节来识别真实的视频格式
  315. *
  316. * @param string $url 视频URL
  317. * @return string 返回扩展名(如 .mp4),默认 .mp4
  318. */
  319. function getVideoExtFromUrl(string $url): string {
  320. try {
  321. // 方法1:从URL路径中提取扩展名
  322. $parsedUrl = parse_url($url);
  323. $path = $parsedUrl['path'] ?? '';
  324. $pathInfo = pathinfo($path);
  325. $extension = $pathInfo['extension'] ?? '';
  326. if ($extension) {
  327. // 验证是否为常见视频格式
  328. $validVideoExts = ['mp4', 'avi', 'mov', 'wmv', 'flv', 'mkv', 'webm', 'm4v', 'mpeg', 'mpg'];
  329. if (in_array(strtolower($extension), $validVideoExts)) {
  330. return '.' . strtolower($extension);
  331. }
  332. }
  333. // 方法2:下载文件头部字节来识别格式
  334. $client = new \GuzzleHttp\Client(['timeout' => 30]);
  335. // 先尝试HEAD请求获取Content-Type
  336. try {
  337. $response = $client->head($url);
  338. $contentType = $response->getHeader('Content-Type')[0] ?? '';
  339. // 根据Content-Type映射扩展名
  340. $mimeMap = [
  341. 'video/mp4' => '.mp4',
  342. 'video/x-msvideo' => '.avi',
  343. 'video/quicktime' => '.mov',
  344. 'video/x-ms-wmv' => '.wmv',
  345. 'video/x-flv' => '.flv',
  346. 'video/x-matroska' => '.mkv',
  347. 'video/webm' => '.webm',
  348. 'video/mpeg' => '.mpeg',
  349. ];
  350. if (isset($mimeMap[$contentType])) {
  351. return $mimeMap[$contentType];
  352. }
  353. } catch (\Exception $e) {
  354. // HEAD请求失败,继续尝试其他方法
  355. }
  356. // 方法3:下载前几个字节分析文件魔数(Magic Number)
  357. $response = $client->get($url, [
  358. 'stream' => true,
  359. 'headers' => [
  360. 'Range' => 'bytes=0-31' // 只下载前32字节
  361. ]
  362. ]);
  363. $header = $response->getBody()->read(32);
  364. // 根据文件魔数判断格式
  365. // MP4/M4V: 以 ftyp 开头(偏移4-8字节)
  366. if (strpos($header, 'ftyp') !== false) {
  367. // 进一步判断是否为 M4V
  368. if (strpos($header, 'M4V') !== false || strpos($header, 'm4v') !== false) {
  369. return '.m4v';
  370. }
  371. return '.mp4';
  372. }
  373. // AVI: 以 RIFF 开头,包含 AVI
  374. if (substr($header, 0, 4) === 'RIFF' && strpos($header, 'AVI') !== false) {
  375. return '.avi';
  376. }
  377. // MOV: 包含 moov 或 mdat
  378. if (strpos($header, 'moov') !== false || strpos($header, 'mdat') !== false) {
  379. return '.mov';
  380. }
  381. // FLV: 以 FLV 开头
  382. if (substr($header, 0, 3) === 'FLV') {
  383. return '.flv';
  384. }
  385. // WebM: 以 0x1A 0x45 0xDF 0xA3 开头
  386. if (ord($header[0]) === 0x1A && ord($header[1]) === 0x45 &&
  387. ord($header[2]) === 0xDF && ord($header[3]) === 0xA3) {
  388. return '.webm';
  389. }
  390. // MKV: 与WebM类似,但包含 matroska
  391. if (strpos($header, 'matroska') !== false) {
  392. return '.mkv';
  393. }
  394. // MPEG: 以 0x00 0x00 0x01 开头
  395. if (ord($header[0]) === 0x00 && ord($header[1]) === 0x00 && ord($header[2]) === 0x01) {
  396. return '.mpeg';
  397. }
  398. // 默认返回 .mp4(最常见的格式)
  399. return '.mp4';
  400. } catch (\Exception $e) {
  401. // 出错时返回默认格式
  402. return '.mp4';
  403. }
  404. }
  405. // 辅助 MIME 映射函数
  406. function mimeToExt(string $mime): ?string {
  407. $map = [
  408. 'audio/wav' => '.wav',
  409. 'audio/x-ms-wma' => '.wma',
  410. 'video/x-ms-wmv' => '.wmv',
  411. 'video/mp4' => '.mp4',
  412. 'audio/mpeg' => '.mp3',
  413. 'audio/amr' => '.amr',
  414. 'application/vnd.rn-realmedia' => '.rm',
  415. 'audio/mid' => '.mid',
  416. 'image/bmp' => '.bmp',
  417. 'image/gif' => '.gif',
  418. 'image/png' => '.png',
  419. 'image/tiff' => '.tiff',
  420. 'image/jpeg' => '.jpg',
  421. 'application/pdf' => '.pdf',
  422. ];
  423. return $map[$mime] ?? null;
  424. }
  425. /**
  426. * 上传文件(阿里云oss)
  427. *
  428. * @param $prefix 文件夹前缀
  429. * @param $file 文件二进制
  430. * @param $filename 文件名(不带后缀)
  431. * @return mixed
  432. */
  433. function uploadFile($prefix, $file, $filename='')
  434. {
  435. // 阿里云主账号
  436. $accessKeyId = env('OSS_ACCESS_ID');
  437. $accessKeySecret = env('OSS_ACCESS_KEY');
  438. $endpoint = env('OSS_END_POINT');
  439. $bucket = env('OSS_BUCKET');
  440. if (!$filename) $filename = randStr(10) . '.' . $file->getClientOriginalExtension();
  441. else $filename = $filename . '.' . $file->getClientOriginalExtension();
  442. // 设置文件名称。
  443. $object = env('OSS_DIRECTORY') . '/' . $prefix . '/' . $filename;
  444. $provider = new \OSS\Credentials\StaticCredentialsProvider($accessKeyId, $accessKeySecret);
  445. try {
  446. $configs = [
  447. "provider" => $provider,
  448. "endpoint" => $endpoint,
  449. "signatureVersion" => 'v4',
  450. "region" => "cn-hangzhou"
  451. ];
  452. $ossClient = new OssClient($configs);
  453. $uploadRes = $ossClient->uploadFile($bucket, $object, $file->path());
  454. } catch (OssException $e) {
  455. dLog('exception')->info('saveImageToOss', [$e->getMessage()]);
  456. return '';
  457. }
  458. // 替换域名
  459. if (!isset($uploadRes['oss-request-url'])) return '';
  460. $url = str_ireplace('zw-ai.oss-cn-hangzhou.aliyuncs.com', 'cdn-zwai.ycsd.cn', $uploadRes['oss-request-url']);
  461. $url = urldecode($url);
  462. return str_ireplace('http://', 'https://', $url);
  463. }
  464. /**
  465. * 上传文件流(阿里云oss)
  466. *
  467. * @param $prefix 文件夹前缀
  468. * @param $file 文件二进制
  469. * @param $filename 文件名(带后缀)
  470. * @return mixed
  471. */
  472. function uploadStreamToOss($prefix, $stream, $filename) {
  473. $accessKeyId = env('OSS_ACCESS_ID');
  474. $accessKeySecret = env('OSS_ACCESS_KEY');
  475. $endpoint = env('OSS_END_POINT');
  476. $bucket = env('OSS_BUCKET');
  477. // if (!$filename) $filename = randStr(10) . '.' . $file->getClientOriginalExtension();
  478. // 设置文件名称。
  479. $object = env('OSS_DIRECTORY') . '/' . $prefix . '/' . $filename;
  480. $provider = new \OSS\Credentials\StaticCredentialsProvider($accessKeyId, $accessKeySecret);
  481. try{
  482. $configs = [
  483. "provider" => $provider,
  484. "endpoint" => $endpoint,
  485. "signatureVersion" => 'v4',
  486. "region"=> "cn-hangzhou"
  487. ];
  488. $ossClient = new OssClient($configs);
  489. $uploadRes = $ossClient->putObject($bucket, $object, $stream);
  490. } catch (OssException $e) {
  491. dLog('exception')->info('saveImageToOss', [$e->getMessage()]);
  492. return '';
  493. }
  494. // 替换域名
  495. if (!isset($uploadRes['oss-request-url'])) return '';
  496. $url = str_ireplace('zw-ai.oss-cn-hangzhou.aliyuncs.com', 'cdn-zwai.ycsd.cn', $uploadRes['oss-request-url']);
  497. $url = urldecode($url);
  498. return str_ireplace('http://', 'https://', $url);
  499. }
  500. /**
  501. * 下载文件到本地
  502. *
  503. * @param $object
  504. * @param $localFile
  505. * @return string
  506. */
  507. function downloadFile($object, $localFile)
  508. {
  509. // 阿里云主账号
  510. $accessKeyId = env('OSS_ACCESS_ID');
  511. $accessKeySecret = env('OSS_ACCESS_KEY');
  512. $endpoint = env('OSS_END_POINT');
  513. $bucket = env('OSS_BUCKET');
  514. try {
  515. $ossClient = new OssClient($accessKeyId, $accessKeySecret, $endpoint);
  516. $ossClient->getObject($bucket, $object, [
  517. OssClient::OSS_FILE_DOWNLOAD => $localFile
  518. ]);
  519. } catch (OssException $e) {
  520. printf($e->getMessage());
  521. return '';
  522. }
  523. }
  524. /**
  525. * 上传封面
  526. *
  527. * @param $file
  528. * @return string
  529. * @throws Exception
  530. */
  531. function uploadCoverFile($file)
  532. {
  533. // 阿里云主账号
  534. $accessKeyId = env('OSS_ACCESS_ID');
  535. $accessKeySecret = env('OSS_ACCESS_KEY');
  536. $endpoint = env('OSS_END_POINT');
  537. $bucket = env('OSS_BUCKET');
  538. // 设置文件名称。
  539. $object = 'books/cover/' . randStr(10) . '.' . $file->getClientOriginalExtension();
  540. try {
  541. $ossClient = new OssClient($accessKeyId, $accessKeySecret, $endpoint);
  542. $ossImgBackData = $ossClient->uploadFile($bucket, $object, $file->path());
  543. } catch (OssException $e) {
  544. printf($e->getMessage());
  545. return '';
  546. }
  547. $urlArr = parse_url($ossImgBackData['oss-request-url']);
  548. return getProp($urlArr, 'path') ? 'http://' . $bucket . '.' . $endpoint . getProp($urlArr, 'path') : '';
  549. }
  550. function uploadAgreementFile($file)
  551. {
  552. // 阿里云主账号
  553. $accessKeyId = env('OSS_ACCESS_ID');
  554. $accessKeySecret = env('OSS_ACCESS_KEY');
  555. $endpoint = env('OSS_END_POINT');
  556. $bucket = env('OSS_BUCKET');
  557. $file_name = $file->getClientOriginalName();
  558. $file_name = str_replace('.' . $file->getClientOriginalExtension(), '', $file_name);
  559. // 设置文件名称。
  560. $object = 'books/contract/' . $file_name . '-' . date('YmdHi') . '.' . $file->getClientOriginalExtension();
  561. try {
  562. $ossClient = new OssClient($accessKeyId, $accessKeySecret, $endpoint);
  563. $ossImgBackData = $ossClient->uploadFile($bucket, $object, $file);
  564. } catch (OssException $e) {
  565. printf($e->getMessage());
  566. return '';
  567. }
  568. $urlArr = parse_url($ossImgBackData['oss-request-url']);
  569. return getProp($urlArr, 'path') ? 'http://' . $bucket . '.' . $endpoint . urldecode(getProp($urlArr, 'path')) : '';
  570. }
  571. /**
  572. * 上传wap推荐图片
  573. *
  574. * @param $file
  575. * @param $filename
  576. * @return string
  577. * @throws Exception
  578. */
  579. function uploadWapRecommendPic($file, $filename)
  580. {
  581. // 阿里云主账号
  582. $accessKeyId = env('OSS_ACCESS_ID');
  583. $accessKeySecret = env('OSS_ACCESS_KEY');
  584. $endpoint = env('OSS_END_POINT');
  585. $bucket = env('OSS_BUCKET_YCSD');
  586. // 设置文件名称。
  587. $object = 'ycsd_web_3nd/images/homebanners/' . $filename . '.' . $file->getClientOriginalExtension();
  588. try {
  589. $ossClient = new OssClient($accessKeyId, $accessKeySecret, $endpoint);
  590. $options = array(
  591. OssClient::OSS_HEADERS => array(
  592. 'Content-Type' => 'image/jpeg',
  593. 'Content-Disposition' => 'inline'
  594. ),
  595. );
  596. $ossImgBackData = $ossClient->uploadFile($bucket, $object, $file->path(), $options);
  597. } catch (OssException $e) {
  598. printf($e->getMessage());
  599. return '';
  600. }
  601. $urlArr = parse_url($ossImgBackData['oss-request-url']);
  602. return getProp($urlArr, 'path') ? 'http://' . $bucket . '.' . $endpoint . getProp($urlArr, 'path') : '';
  603. }
  604. /**
  605. * @param $path
  606. * @return Generator
  607. */
  608. function readFileContent($path)
  609. {
  610. if ($handle = fopen($path, 'r')) {
  611. while (!feof($handle)) {
  612. yield trim(fgets($handle));
  613. }
  614. fclose($handle);
  615. }
  616. }
  617. function exportFileCsv(array $headers, array $data, string $filename)
  618. {
  619. header("Content-type:application/vnd.ms-excel");
  620. header("Content-Disposition:attachment;filename=" . $filename . ".csv");
  621. $headers = collect($headers)->map(function ($item) {
  622. return "\"" . mb_convert_encoding($item, "GBK", "UTF-8") . "\"";
  623. })->all();
  624. echo implode(",", $headers);
  625. echo "\r\n";
  626. foreach ($data as $item) {
  627. $rows = collect($item)->map(function ($row) {
  628. return "\"" . mb_convert_encoding(is_numeric($row) && strlen($row) > 12 ? "'" . $row : $row, "GBK", "UTF-8") . "\"";
  629. })->all();
  630. echo implode(",", $rows);
  631. echo "\r\n";
  632. }
  633. exit();
  634. }
  635. //function exportFileCsv(array $header, array $data, string $filename) {
  636. // header('Content-Encoding: UTF-8');
  637. // header("Content-type:application/vnd.ms-excel;charset=UTF-8");
  638. // header('Content-Disposition: attachment;filename="' . $filename . '.csv"');
  639. //
  640. // //打开php标准输出流
  641. // $fp = fopen('php://output', 'a');
  642. //
  643. // //添加BOM头,以UTF8编码导出CSV文件,如果文件头未添加BOM头,打开会出现乱码。
  644. // fwrite($fp, chr(0xEF).chr(0xBB).chr(0xBF));
  645. // //添加导出标题
  646. // fputcsv($fp, $header);
  647. //
  648. // foreach ($data as $k => $item) {
  649. // fputcsv($fp, $item);
  650. // if ($k % 5000 == 0) {
  651. // //每1万条数据就刷新缓冲区
  652. // ob_flush();
  653. // flush();
  654. // }
  655. // }
  656. // exit();
  657. //}
  658. //function exportFileCsv(array $header, array $data, string $filename) {
  659. //
  660. // header('Content-Type: application/vnd.ms-excel');
  661. // header('Content-Disposition: attachment;filename="'.$filename.'.csv"');
  662. // header('Cache-Control: max-age=0');
  663. //
  664. // //打开PHP文件句柄,php://output 表示直接输出到浏览器
  665. // $fp = fopen('php://output', 'a');
  666. //
  667. // //输出Excel列名信息
  668. // foreach ($header as $key => $value) {
  669. // //CSV的Excel支持GBK编码,一定要转换,否则乱码
  670. // $header[$key] = iconv('utf-8', 'gbk', $value);
  671. // }
  672. //
  673. // //将数据通过fputcsv写到文件句柄
  674. // fputcsv($fp, $header);
  675. //
  676. // //计数器
  677. // $num = 0;
  678. //
  679. // //每隔$limit行,刷新一下输出buffer,不要太大,也不要太小
  680. // $limit = 10000;
  681. //
  682. // //逐行取出数据,不浪费内存
  683. // $count = count($data);
  684. // for ($i = 0; $i < $count; $i++) {
  685. //
  686. // $num++;
  687. //
  688. // //刷新一下输出buffer,防止由于数据过多造成问题
  689. // if ($limit == $num) {
  690. // ob_flush();
  691. // flush();
  692. // $num = 0;
  693. // }
  694. //
  695. // $row = $data[$i];
  696. // foreach ($row as $key => $value) {
  697. // $row[$key] = iconv('utf-8', 'gbk', $value);
  698. // }
  699. //
  700. // fputcsv($fp, $row);
  701. // }
  702. // exit();
  703. //}
  704. // 获取除空格外的字数
  705. function getSize(string $content)
  706. {
  707. $content = preg_replace('/\s+/', '', $content);
  708. return mb_strlen($content, 'utf-8');
  709. }
  710. // 获取word字符数(不计空格)
  711. function getChargeSize(string $content)
  712. {
  713. //判断是否存在替换字符
  714. $is_replace_count = substr_count($content, "龘");
  715. try {
  716. //先将回车换行符做特殊处理
  717. $str = preg_replace('/(\r\n+|\s+| +)/', "龘", $content);
  718. //处理英文字符数字,连续字母、数字、英文符号视为一个单词
  719. $str = preg_replace('/[a-z_A-Z0-9-\.!@#\$%\\\^&\*\)\(\+=\{\}\[\]\/",\'<>~`\?:;|]/', "m", $str);
  720. //合并字符m,连续字母、数字、英文符号视为一个单词
  721. $str = preg_replace('/m+/', "*", $str);
  722. //去掉回车换行符
  723. $str = preg_replace('/龘+/', "", $str);
  724. //返回字数
  725. return mb_strlen($str) + $is_replace_count;
  726. } catch (\Exception $e) {
  727. return 0;
  728. }
  729. }
  730. // 将阿拉伯数字转换成中文
  731. function chineseNum($figure, $capital = false, $mode = true)
  732. {
  733. if ($figure == '0') return '零';
  734. $numberChar = ['零', '一', '二', '三', '四', '五', '六', '七', '八', '九'];
  735. $unitChar = ['', '十', '百', '千', '', '万', '亿', '兆', '京', '垓', '秭', '穣', '沟', '涧', '正', '载', '极', '恒河沙', '阿僧祇', '那由他', '不可思议', '无量大数'];
  736. if ($capital !== false) {
  737. $numberChar = ['零', '壹', '贰', '叁', '肆', '伍', '陆', '柒', '捌', '玖'];
  738. $unitChar = ['', '拾', '佰', '仟', '', '万', '亿', '兆', '京', '垓', '秭', '穣', '沟', '涧', '正', '载', '极', '恒河沙', '阿僧祇', '那由他', '不可思议', '无量大数'];
  739. }
  740. $dec = "点";
  741. $target = '';
  742. $matches = [];
  743. if ($mode) {
  744. preg_match("/^0*(\d*)\.?(\d*)/", $figure, $matches);
  745. } else {
  746. preg_match("/(\d*)\.?(\d*)/", $figure, $matches);
  747. }
  748. list(, $number, $point) = $matches;
  749. if ($point) {
  750. $target = $dec . chineseNum($point, $capital, false);
  751. }
  752. if (!$number) {
  753. return $target;
  754. }
  755. $str = strrev($number);
  756. for ($i = 0; $i < strlen($str); $i++) {
  757. $out[$i] = $numberChar[$str[$i]];
  758. if ($mode === false) {
  759. continue;
  760. }
  761. $out[$i] .= $str[$i] != '0' ? $unitChar[$i % 4] : '';
  762. if ($i > 0 && $str[$i] + $str[$i - 1] == 0) {
  763. $out[$i] = '';
  764. }
  765. if ($i % 4 == 0) {
  766. $temp = substr($str, $i, 4);
  767. $out[$i] = str_replace($numberChar[0], '', $out[$i]);
  768. if (strrev($temp) > 0) {
  769. $out[$i] .= $unitChar[4 + floor($i / 4)];
  770. } else {
  771. $out[$i] .= $numberChar[0];
  772. }
  773. }
  774. }
  775. $result = join('', array_reverse($out)) . $target;
  776. return mb_substr($result, 0, 2) == '一十' ? mb_substr($result, 1) : $result;
  777. }
  778. function addPrefix($str)
  779. {
  780. if (!$str) return '';
  781. if (mb_substr($str, 0, 4) == 'http') return $str;
  782. if (mb_substr($str, 0, 5) == 'https') return $str;
  783. if (mb_substr($str, 0, 7) == '/books/') return 'http://zwcontent.oss-cn-hangzhou.aliyuncs.com' . $str;
  784. if (mb_substr($str, 0, 6) == '/card/') return 'http://zwcontent.oss-cn-hangzhou.aliyuncs.com' . $str;
  785. if (mb_substr($str, 0, 15) == 'uploader/idcard') return 'http://zwcontent.oss-cn-hangzhou.aliyuncs.com/' . $str;
  786. if (mb_substr($str, 0, 6) == '/cover') return 'https://cdn-newyc.ycsd.cn/ycsd_cover/covermiddle' . mb_substr($str, 6);
  787. if (mb_substr($str, 0, 8) == '/images/') return 'https://cdn-newyc.ycsd.cn/ycsd_cover' . $str;
  788. }
  789. /**
  790. * 章节内容排版
  791. *
  792. * @param $content
  793. * @return string
  794. */
  795. function filterContent($content)
  796. {
  797. if (!$content) return '';
  798. $content = str_replace(
  799. ['&nbsp;&nbsp;', '<br /><br />', '<br>', '<br />', '&nbsp;', '<p>', '</p>', '&ldquo;', '&rdquo;', '&hellip;', '&lsquo;', '&rsquo;', '&mdash;'],
  800. [' ', PHP_EOL, PHP_EOL, PHP_EOL, ' ', '', PHP_EOL, '“', '”', '...', '‘', '’', '-'],
  801. $content);
  802. $content = preg_replace('/(\r\n)+/', PHP_EOL, $content);
  803. // 段落首字母前加两个中文空格
  804. $string = explode(PHP_EOL, $content);
  805. foreach ($string as $line => $text) {
  806. $string[$line] = str_replace([' ', "\r\n", "\r", "\n", ' '], '', $string[$line]);
  807. if (!$string[$line]) {
  808. unset($string[$line]);
  809. } else {
  810. $string[$line] = $string[$line] . PHP_EOL;
  811. // if (mb_substr($string[$line], 0, 1) == ' ') {
  812. // $string[$line] = str_replace(' ', '', $string[$line]); // 去除多个空格
  813. // $string[$line] = '  ' . $string[$line].PHP_EOL;
  814. // }
  815. // if (mb_substr($string[$line], 0, 2) != '  ') {
  816. // $string[$line] = '  ' . $string[$line].PHP_EOL;
  817. // }
  818. // if (mb_substr($string[$line], 0, 2) == '  ' && str_replace(' ', '', $string[$line])) {
  819. // $string[$line] .= PHP_EOL;
  820. // }
  821. }
  822. }
  823. $content = implode(PHP_EOL, $string);
  824. return $content;
  825. }
  826. /**
  827. * 书籍简介排版
  828. *
  829. * @param $content
  830. * @return string
  831. */
  832. function filterContent2($content)
  833. {
  834. if (!$content) return '';
  835. $content = str_replace(
  836. ['&nbsp;&nbsp;', '<br /><br />', '<br>', '<br />', '&nbsp;', '<p>', '</p>', '&ldquo;', '&rdquo;', '&hellip;', '&lsquo;', '&rsquo;', '&mdash;'],
  837. [' ', PHP_EOL, PHP_EOL, PHP_EOL, ' ', '', PHP_EOL, '“', '”', '...', '‘', '’', '-'],
  838. $content);
  839. $content = preg_replace('/(\r\n)+/', PHP_EOL, $content);
  840. // 段落首字母前加两个中文空格
  841. $string = explode(PHP_EOL, $content);
  842. $content = '';
  843. foreach ($string as $line => $text) {
  844. $string[$line] = str_replace([' ', "\r\n", "\r", "\n", ' ', '<br />'], '', $string[$line]);
  845. if (!$string[$line]) {
  846. unset($string[$line]);
  847. } else {
  848. $string[$line] = '  ' . $string[$line] . '<br />';
  849. $content .= $string[$line];
  850. }
  851. }
  852. $content = trim($content, '<br />');
  853. return $content;
  854. }
  855. /**
  856. * 书籍简介排版-抖音版
  857. *
  858. * @param $content
  859. * @return string
  860. */
  861. function filterIntro($content)
  862. {
  863. if (!$content) return '';
  864. $content = str_replace(
  865. ['&nbsp;&nbsp;', '<br /><br />', '<br>', '<br />', '&nbsp;', '<p>', '</p>', '&ldquo;', '&rdquo;', '&hellip;', '&lsquo;', '&rsquo;', '&mdash;'],
  866. [' ', ' ', ' ', ' ', ' ', '', ' ', '“', '”', '...', '‘', '’', '-'],
  867. $content);
  868. $content = preg_replace('/(\r\n)+/', PHP_EOL, $content);
  869. // 段落首字母前加两个中文空格
  870. $string = explode(PHP_EOL, $content);
  871. $content = '';
  872. foreach ($string as $line => $text) {
  873. $string[$line] = str_replace([' ', "\r\n", "\r", "\n", ' ', '<br />'], '', $string[$line]);
  874. if (!$string[$line]) {
  875. unset($string[$line]);
  876. } else {
  877. $content .= $string[$line];
  878. }
  879. }
  880. $content = trim($content, '<br />');
  881. return $content;
  882. }
  883. function sensitiveStr($list, $string)
  884. {
  885. $count = 0; //违规词的个数
  886. $sensitiveWord = ''; //违规词
  887. $stringAfter = $string; //替换后的内容
  888. $total = count($list);
  889. $size = 500;
  890. $last = ceil($total / $size);
  891. $patternList = [];
  892. for ($page = 1; $page <= $last; $page++) {
  893. $arr = array_slice($list, $size * ($page - 1), $size);
  894. $filter = [];
  895. foreach ($arr as $v) {
  896. if (preg_match('/[^a-zA-Z0-9\|\p{Han}\·]/u', $v)) continue;
  897. $filter[] = $v;
  898. // $a = preg_replace('/[^a-zA-Z0-9\|\p{Han}]/u', '', $v);
  899. // if ($a) $filter[] = $a;
  900. }
  901. $pattern = "/" . implode("|", $filter) . "/i"; //定义正则表达式
  902. if (preg_match_all($pattern, $string, $matches)) { //匹配到了结果
  903. $patternList = array_merge($patternList, $matches[0]); //匹配到的数组
  904. }
  905. }
  906. $sensitiveWord = '';
  907. if ($patternList) {
  908. $count = count($patternList);
  909. // $sensitiveWord = implode(',', $patternList); //敏感词数组转字符串
  910. $replaceArray = array_combine($patternList, array_fill(0, count($patternList), '*')); //把匹配到的数组进行合并,替换使用
  911. $stringAfter = strtr($string, $replaceArray); //结果替换
  912. // 将敏感词合并
  913. $return_pattern = [];
  914. foreach ($patternList as $v) {
  915. if (!isset($return_pattern[$v])) {
  916. $return_pattern[$v] = [
  917. 'word' => $v,
  918. 'count' => 1,
  919. ];
  920. } else {
  921. $return_pattern[$v]['count'] += 1;
  922. }
  923. }
  924. foreach ($return_pattern as $v) {
  925. $sensitiveWord .= $v['word'] . ',';
  926. }
  927. }
  928. return [
  929. 'count' => $count,
  930. 'sensitive_words' => trim($sensitiveWord, ','),
  931. 'content' => $stringAfter
  932. ];
  933. }
  934. /**
  935. * 转换时间格式
  936. *
  937. * @param $date
  938. * @param string $format
  939. * @return false|string
  940. */
  941. function transDate($date, $format = 'Y-m-d H:i:s')
  942. {
  943. return strtotime($date) > 0 ? date($format, strtotime($date)) : '';
  944. }
  945. /**
  946. * 根据网段获取计算所有IP
  947. *
  948. * @param string $segment 网段 '139.217.0.1/24'
  949. * @return array [网络地址:139.217.0.1 广播地址:139.217.0.255 IP列表: ['139.217.0.2','139.217.0.3'……'139.217.0.254']]
  950. */
  951. function getIpBySegment($segment)
  952. {
  953. $segmentInfo = explode("/", $segment);
  954. $beginIpArray = explode(".", $segmentInfo[0]);
  955. $mask = intval($segmentInfo['1']);
  956. $endIp = array();
  957. foreach ($beginIpArray as $ipKey => $item) {
  958. $beginFlag = 8 * ($ipKey); //0 8 16 24
  959. $endFlag = 8 * ($ipKey + 1);//8 16 24 32
  960. $decbinItem = str_pad(decbin($item), 8, "0", STR_PAD_LEFT);
  961. $endIp[] = $mask >= $endFlag ? $item : ($mask > $beginFlag ? bindec(str_pad(substr($decbinItem, 0, $mask - $beginFlag), 8, "1", STR_PAD_RIGHT)) : ($ipKey <= 2 ? pow(2, 8) - 1 : pow(2, 8) - 1));
  962. }
  963. $ipArray = array();
  964. for ($beginIp[0] = $beginIpArray[0]; $beginIp[0] <= $endIp[0]; $beginIp[0]++) {
  965. for ($beginIp[1] = $beginIpArray[1]; $beginIp[1] <= $endIp[1]; $beginIp[1]++) {
  966. for ($beginIp[2] = $beginIpArray[2]; $beginIp[2] <= $endIp[2]; $beginIp[2]++) {
  967. for ($beginIp[3] = $beginIpArray[3]; $beginIp[3] <= $endIp[3]; $beginIp[3]++) {
  968. $ipArray[] = implode(".", $beginIp);
  969. }
  970. }
  971. }
  972. }
  973. $network_ip_addr = $beginIpArray[0] . '.' . $beginIpArray[1] . '.' . $beginIpArray[2] . '.' . '0'; // 网络地址
  974. $broadcast_ip_addr = end($ipArray); // 广播地址
  975. if ($ipArray[0] == $network_ip_addr) { // 如果是网络地址则删掉
  976. unset($ipArray[0]);
  977. }
  978. $last = count($ipArray);
  979. unset($ipArray[$last]);
  980. return [$network_ip_addr, $broadcast_ip_addr, $ipArray];
  981. }
  982. /**
  983. * 在指定网段中分配子网段
  984. *
  985. * @param string $segment 指定网段
  986. * @param int $ipNum 需要的IP数
  987. * @param array $usedIpArray 不可用(已经使用)的IP,默认为空数组
  988. * @return bool|string 成功则返回分配的网段
  989. */
  990. function allocateSegment($segment, $ipNum, $usedIpArray = [])
  991. {
  992. $usedIpArray = empty($usedIpArray) ? [] : array_flip($usedIpArray);
  993. //计算需要多少个IP
  994. $i = 0;
  995. $ipCount = pow(2, $i);
  996. while ($ipCount < $ipNum) {
  997. $i++;
  998. $ipCount = pow(2, $i);
  999. }
  1000. $newMask = 32 - $i;
  1001. //大网段的开始和结束IP
  1002. $segmentInfo = explode("/", $segment); //['139.217.0.1',24]
  1003. $beginIpArray = explode(".", $segmentInfo[0]);//[139,217,0,1]
  1004. $mask = intval($segmentInfo['1']); //24
  1005. if ($newMask < $mask) {
  1006. return false;
  1007. }
  1008. $endIp = array();
  1009. $step = [];
  1010. foreach ($beginIpArray as $ipKey => $item) {
  1011. $beginFlag = 8 * ($ipKey); //0 8 16 24
  1012. $endFlag = 8 * ($ipKey + 1);//8 16 24 32
  1013. $step[$ipKey] = $newMask > $endFlag ? 1 : ($endFlag - $newMask < 8 ? pow(2, $endFlag - $newMask) : pow(2, 8));
  1014. $decbinItem = str_pad(decbin($item), 8, "0", STR_PAD_LEFT);
  1015. $endIp[] = $mask >= $endFlag ? $item : ($mask > $beginFlag ? bindec(str_pad(substr($decbinItem, 0, $mask - $beginFlag), 8, "1", STR_PAD_RIGHT)) : ($ipKey <= 2 ? pow(2, 8) - 1 : pow(2, 8) - 1));
  1016. }
  1017. //遍历生成网段
  1018. for ($beginIp[0] = $beginIpArray[0]; $beginIp[0] <= $endIp[0]; $beginIp[0] += $step[0]) {
  1019. for ($beginIp[1] = $beginIpArray[1]; $beginIp[1] <= $endIp[1]; $beginIp[1] += $step[1]) {
  1020. for ($beginIp[2] = $beginIpArray[2]; $beginIp[2] <= $endIp[2]; $beginIp[2] += $step[2]) {
  1021. for ($beginIp[3] = $beginIpArray[3]; $beginIp[3] <= $endIp[3]; $beginIp[3] += $step[3]) {
  1022. $newSegment = implode('.', $beginIp) . '/' . $newMask;
  1023. //获取该网段所有的IP
  1024. $ipArray = getIpBySegment($newSegment);
  1025. $canUse = true;
  1026. //判断该网段是否可用
  1027. if (!empty($usedIpArray)) {
  1028. foreach ($ipArray as $ip) {
  1029. if (isset($usedIpArray[$ip])) {
  1030. $canUse = false;
  1031. break;
  1032. }
  1033. }
  1034. }
  1035. if ($canUse) {
  1036. return $newSegment;
  1037. }
  1038. }
  1039. }
  1040. }
  1041. }
  1042. return false;
  1043. }
  1044. function remove_xss($val)
  1045. {
  1046. // remove all non-printable characters. CR(0a) and LF(0b) and TAB(9) are allowed
  1047. // this prevents some character re-spacing such as <java\0script>
  1048. // note that you have to handle splits with \n, \r, and \t later since they *are* allowed in some inputs
  1049. $val = preg_replace('/([\x00-\x08,\x0b-\x0c,\x0e-\x19])/', '', $val);
  1050. // straight replacements, the user should never need these since they're normal characters
  1051. // this prevents like <IMG SRC=@avascript:alert('XSS')>
  1052. $search = 'abcdefghijklmnopqrstuvwxyz';
  1053. $search .= 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
  1054. $search .= '1234567890!@#$%^&*()';
  1055. $search .= '~`";:?+/={}[]-_|\'\\';
  1056. for ($i = 0; $i < strlen($search); $i++) {
  1057. // ;? matches the ;, which is optional
  1058. // 0{0,7} matches any padded zeros, which are optional and go up to 8 chars
  1059. // @ @ search for the hex values
  1060. $val = preg_replace('/(&#[xX]0{0,8}' . dechex(ord($search[$i])) . ';?)/i', $search[$i], $val); // with a ;
  1061. // @ @ 0{0,7} matches '0' zero to seven times
  1062. $val = preg_replace('/(�{0,8}' . ord($search[$i]) . ';?)/', $search[$i], $val); // with a ;
  1063. }
  1064. // now the only remaining whitespace attacks are \t, \n, and \r
  1065. $ra1 = array('javascript', 'vbscript', 'expression', 'applet', 'meta', 'xml', 'blink', 'link', 'style', 'script', 'embed', 'object', 'iframe', 'frame', 'frameset', 'ilayer', 'layer', 'bgsound', 'title', 'base');
  1066. $ra2 = array(
  1067. 'onabort', 'onactivate', 'onafterprint', 'onafterupdate', 'onbeforeactivate', 'onbeforecopy', 'onbeforecut', 'onbeforedeactivate', 'onbeforeeditfocus', 'onbeforepaste', 'onbeforeprint', 'onbeforeunload', 'onbeforeupdate', 'onblur', 'onbounce', 'oncellchange', 'onchange', 'onclick', 'oncontextmenu', 'oncontrolselect', 'oncopy', 'oncut', 'ondataavailable', 'ondatasetchanged', 'ondatasetcomplete', 'ondblclick', 'ondeactivate', 'ondrag', 'ondragend', 'ondragenter', 'ondragleave', 'ondragover', 'ondragstart', 'ondrop', 'onerror', 'onerrorupdate', 'onfilterchange', 'onfinish', 'onfocus', 'onfocusin', 'onfocusout', 'onhelp', 'onkeydown', 'onkeypress', 'onkeyup', 'onlayoutcomplete', 'onload', 'onlosecapture', 'onmousedown', 'onmouseenter', 'onmouseleave', 'onmousemove', 'onmouseout', 'onmouseover', 'onmouseup', 'onmousewheel', 'onmove', 'onmoveend', 'onmovestart', 'onpaste', 'onpropertychange', 'onreadystatechange', 'onreset', 'onresize', 'onresizeend', 'onresizestart', 'onrowenter', 'onrowexit', 'onrowsdelete', 'onrowsinserted', 'onscroll', 'onselect', 'onselectionchange', 'onselectstart', 'onstart', 'onstop', 'onsubmit', 'onunload'
  1068. );
  1069. $ra = array_merge($ra1, $ra2);
  1070. $found = true; // keep replacing as long as the previous round replaced something
  1071. while ($found == true) {
  1072. $val_before = $val;
  1073. for ($i = 0; $i < sizeof($ra); $i++) {
  1074. $pattern = '/';
  1075. for ($j = 0; $j < strlen($ra[$i]); $j++) {
  1076. if ($j > 0) {
  1077. $pattern .= '(';
  1078. $pattern .= '(&#[xX]0{0,8}([9ab]);)';
  1079. $pattern .= '|';
  1080. $pattern .= '|(�{0,8}([9|10|13]);)';
  1081. $pattern .= ')*';
  1082. }
  1083. $pattern .= $ra[$i][$j];
  1084. }
  1085. $pattern .= '/i';
  1086. $replacement = substr($ra[$i], 0, 2) . '<x>' . substr($ra[$i], 2); // add in <> to nerf the tag
  1087. $val = preg_replace($pattern, $replacement, $val); // filter out the hex tags
  1088. if ($val_before == $val) {
  1089. // no replacements were made, so exit the loop
  1090. $found = false;
  1091. }
  1092. }
  1093. }
  1094. return $val;
  1095. }
  1096. /**
  1097. * 计算作者积分等级
  1098. *
  1099. * @param $score
  1100. */
  1101. function calcAuthorLevel($score): int
  1102. {
  1103. switch (true) {
  1104. case $score <= 0:
  1105. $level = 0;
  1106. break;
  1107. case $score <= 5000:
  1108. $level = 1;
  1109. break;
  1110. case $score <= 50000:
  1111. $level = 2;
  1112. break;
  1113. case $score <= 100000:
  1114. $level = 3;
  1115. break;
  1116. case $score <= 300000:
  1117. $level = 4;
  1118. break;
  1119. case $score <= 800000:
  1120. $level = 5;
  1121. break;
  1122. case $score <= 1500000:
  1123. $level = 6;
  1124. break;
  1125. case $score <= 2500000:
  1126. $level = 7;
  1127. break;
  1128. case $score <= 5000000:
  1129. $level = 8;
  1130. break;
  1131. case $score <= 10000000:
  1132. $level = 9;
  1133. break;
  1134. default:
  1135. $level = 10;
  1136. break;
  1137. }
  1138. return $level;
  1139. }
  1140. /**
  1141. * 运营数据(上传附件)
  1142. *
  1143. * @param $file
  1144. * @return string
  1145. * @throws Exception
  1146. */
  1147. function uploadEnclosureFile($file)
  1148. {
  1149. // 阿里云主账号
  1150. $accessKeyId = env('OSS_ACCESS_ID');
  1151. $accessKeySecret = env('OSS_ACCESS_KEY');
  1152. $endpoint = env('OSS_END_POINT');
  1153. $bucket = env('OSS_BUCKET');
  1154. // 设置文件名称。
  1155. $object = 'books/enclosure/' . randStr(10) . '--' . $file->getClientOriginalName();
  1156. try {
  1157. $ossClient = new OssClient($accessKeyId, $accessKeySecret, $endpoint);
  1158. $ossImgBackData = $ossClient->uploadFile($bucket, $object, $file->path());
  1159. } catch (OssException $e) {
  1160. printf($e->getMessage());
  1161. return '';
  1162. }
  1163. $urlArr = parse_url($ossImgBackData['oss-request-url']);
  1164. return getProp($urlArr, 'path') ? 'http://' . $bucket . '.' . $endpoint . getProp($urlArr, 'path') : '';
  1165. }
  1166. // 获取当前域名是http还是https
  1167. function getHttpType()
  1168. {
  1169. return ((isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on') || (isset($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] == 'https')) ? 'https://' : 'http://';
  1170. }
  1171. // 根据id生成唯一邀请码
  1172. function enCodeId($user_id)
  1173. {
  1174. $key = 'XzeTdSPQc1uYHRBVWmUE6x94q25g3krfCGhb8FjtDZvMNKJpnayw7s';
  1175. $num = strlen($key);
  1176. $code = ''; // 邀请码
  1177. while ($user_id > 0) { // 转进制
  1178. $mod = $user_id % $num; // 求模
  1179. $user_id = ($user_id - $mod) / $num;
  1180. $code = $key[$mod] . $code;
  1181. }
  1182. $code = str_pad($code, 6, 'A', STR_PAD_LEFT); // 不足用0补充
  1183. return $code;
  1184. }
  1185. // 根据邀请码解密为id
  1186. function deCodeId($code)
  1187. {
  1188. $key = 'XzeTdSPQc1uYHRBVWmUE6x94q25g3krfCGhb8FjtDZvMNKJpnayw7s';
  1189. $num = strlen($key);
  1190. if (strrpos($code, '0') !== false) $code = substr($code, strrpos($code, '0') + 1);
  1191. $len = strlen($code);
  1192. $code = strrev($code);
  1193. $user_id = 0;
  1194. for ($i = 0; $i < $len; $i++) {
  1195. $user_id += strpos($key, $code[$i]) * pow($num, $i);
  1196. }
  1197. return $user_id;
  1198. }
  1199. /**
  1200. * 将二维数组按其中的某个数组排序(此方法适用于将数据库数据按数组取出后自动按ID排序的情况 ps:即未按该数组排序)
  1201. *
  1202. * @param array $array 二维数组
  1203. * @param array $sort 排序数组
  1204. * @param string $field 排序字段(二维数组和排序数组相同的字段)
  1205. * @return array
  1206. */
  1207. function sortByArray(array $array, array $sort, string $field): array
  1208. {
  1209. $data = [];
  1210. if (is_array($array) && is_array($sort)) {
  1211. foreach ($sort as $v) {
  1212. foreach ($array as $key => $val) {
  1213. if ($v == $val[$field]) {
  1214. array_push($data, $array[$key]);
  1215. }
  1216. }
  1217. }
  1218. }
  1219. return $data;
  1220. }
  1221. /**
  1222. * 将二维数组按其中的字段排序(正序或倒序)
  1223. *
  1224. * @param array $array 二维数组
  1225. * @param string $field 排序字段
  1226. * @param mixed $type 排序方式(3倒序,4正序)
  1227. * @return array|mixed
  1228. */
  1229. function sortByField(array $array, string $field, $type): array
  1230. {
  1231. if (is_array($array)) {
  1232. array_multisort(array_column($array, $field), $type, $array);
  1233. }
  1234. return $array;
  1235. }
  1236. // 生成用户邀请码
  1237. function setUserInviteCode($id)
  1238. {
  1239. return \Vinkla\Hashids\Facades\Hashids::connection('invite')->encode($id);
  1240. }
  1241. // 解密用户邀请码
  1242. function decodeUserInviteCode($code)
  1243. {
  1244. return \Vinkla\Hashids\Facades\Hashids::connection('invite')->decode($code);
  1245. }
  1246. function getMillisecond()
  1247. {
  1248. list($microsecond, $time) = explode(' ', microtime());
  1249. return (float)sprintf('%.0f', (floatval($microsecond) + floatval($time)) * 1000);
  1250. }
  1251. function get_client_ip($type = 0, $adv = false)
  1252. {
  1253. $type = $type ? 1 : 0;
  1254. static $ip = null;
  1255. if (null !== $ip) {
  1256. return $ip[$type];
  1257. }
  1258. if ($adv) {
  1259. if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
  1260. $arr = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
  1261. $pos = array_search('unknown', $arr);
  1262. if (false !== $pos) {
  1263. unset($arr[$pos]);
  1264. }
  1265. $ip = trim($arr[0]);
  1266. } elseif (isset($_SERVER['HTTP_CLIENT_IP'])) {
  1267. $ip = $_SERVER['HTTP_CLIENT_IP'];
  1268. } elseif (isset($_SERVER['REMOTE_ADDR'])) {
  1269. $ip = $_SERVER['REMOTE_ADDR'];
  1270. }
  1271. } elseif (isset($_SERVER['REMOTE_ADDR'])) {
  1272. $ip = $_SERVER['REMOTE_ADDR'];
  1273. }
  1274. // IP地址合法验证
  1275. $long = sprintf("%u", ip2long($ip));
  1276. $ip = $long ? array($ip, $long) : array('0.0.0.0', 0);
  1277. return $ip[$type];
  1278. }
  1279. /**
  1280. * 获取真实IP
  1281. */
  1282. function _getIp()
  1283. {
  1284. if (getenv('HTTP_X_FORWARDED_FOR')) {
  1285. $ip = getenv('HTTP_X_FORWARDED_FOR');
  1286. } else if (getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"), "unknown"))
  1287. $ip = getenv("HTTP_CLIENT_IP");
  1288. else if (getenv("HTTP_X_FORWARD_FOR") && strcasecmp(getenv("HTTP_X_FORWARD_FOR"), "unknown"))
  1289. $ip = getenv("HTTP_X_FORWARD_FOR");
  1290. else if (getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"), "unknown"))
  1291. $ip = getenv("REMOTE_ADDR");
  1292. else if (isset($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'], "unknown"))
  1293. $ip = $_SERVER['REMOTE_ADDR'];
  1294. else
  1295. $ip = "unknown";
  1296. return ($ip);
  1297. }
  1298. /**
  1299. * 数组 转 对象
  1300. *
  1301. * @param array $arr 数组
  1302. * @return object
  1303. */
  1304. function array_to_object($arr)
  1305. {
  1306. if (gettype($arr) != 'array') {
  1307. return;
  1308. }
  1309. foreach ($arr as $k => $v) {
  1310. if (gettype($v) == 'array' || getType($v) == 'object') {
  1311. $arr[$k] = (object)array_to_object($v);
  1312. }
  1313. }
  1314. return (object)$arr;
  1315. }
  1316. /**
  1317. * 对象 转 数组
  1318. *
  1319. * @param object $obj 对象
  1320. * @return array
  1321. */
  1322. function object_to_array($obj)
  1323. {
  1324. $obj = (array)$obj;
  1325. foreach ($obj as $k => $v) {
  1326. if (gettype($v) == 'resource') {
  1327. return;
  1328. }
  1329. if (gettype($v) == 'object' || gettype($v) == 'array') {
  1330. $obj[$k] = (array)object_to_array($v);
  1331. }
  1332. }
  1333. return $obj;
  1334. }
  1335. /**
  1336. * 检查是否为手机号码
  1337. */
  1338. function _isPhone($number)
  1339. {
  1340. return preg_match("/^1[34578][0-9]{9}$/", $number);
  1341. }
  1342. /**
  1343. * 判断所传的参数是否缺少,如果缺少返回渠道的字段,正确返回0
  1344. *
  1345. * @param array $param
  1346. * @param array $must
  1347. * @return int|mixed
  1348. */
  1349. function checkParam(array $param, array $must)
  1350. {
  1351. foreach ($must as $item) {
  1352. if (array_key_exists($item, $param) && $param[$item] != '') {
  1353. } else {
  1354. return $item;
  1355. }
  1356. }
  1357. return 0;
  1358. }
  1359. /**
  1360. * 对象 转 数组
  1361. *
  1362. * @param object $obj 对象
  1363. * @return array
  1364. */
  1365. function ignoreKeyInArray($targetArray, $delete_keys = [], $changes = [])
  1366. {
  1367. $change_keys = array_keys($changes);
  1368. foreach ($targetArray as $key => $value) {
  1369. if (in_array($key, $delete_keys) && isset($targetArray[$key])) unset($targetArray[$key]);
  1370. if (in_array($key, $change_keys) && isset($targetArray[$key])) $targetArray[$key] = $changes[$key];
  1371. if (is_array($value)) ignoreKeyInArray($value, $delete_keys, $change_keys);
  1372. }
  1373. return $targetArray;
  1374. }
  1375. function itemTransform($trans, $data)
  1376. {
  1377. if ($data) {
  1378. return $trans->transform($data);
  1379. } else {
  1380. return [];
  1381. }
  1382. }
  1383. function collectionTransform($trans, $data)
  1384. {
  1385. $ret_data = [];
  1386. if ($data) {
  1387. foreach ($data as $item) {
  1388. $ret_data[] = $trans->transform($item);
  1389. }
  1390. }
  1391. return $ret_data;
  1392. }
  1393. function paginationTransform($trans, $paginator)
  1394. {
  1395. $ret = [];
  1396. $ret['list'] = [];
  1397. if ($paginator) {
  1398. foreach ($paginator as $item) {
  1399. $ret['list'][] = $trans->transform($item);
  1400. }
  1401. $ret['meta'] = [
  1402. 'total' => (int)$paginator->total(),
  1403. 'per_page' => (int)$paginator->perPage(),
  1404. 'current_page' => (int)$paginator->currentPage(),
  1405. 'last_page' => (int)$paginator->lastPage(),
  1406. 'next_page_url' => (string)$paginator->nextPageUrl(),
  1407. 'prev_page_url' => (string)$paginator->previousPageUrl()
  1408. ];
  1409. }
  1410. return $ret;
  1411. }
  1412. /**
  1413. * 加密site id
  1414. */
  1415. function encodeDistributionChannelId($id)
  1416. {
  1417. $encrypt_pool = [
  1418. ];
  1419. if (isset($encrypt_pool[$id])) {
  1420. return $encrypt_pool[$id];
  1421. }
  1422. $hashids = new \Hashids\Hashids('', 16, 'abcdefghjklmnopqrstuvwxyz1234567890');
  1423. return $hashids->encode($id);
  1424. }
  1425. /**
  1426. * 解密密site id
  1427. */
  1428. function decodeDistributionChannelId($code)
  1429. {
  1430. $encrypt_pool = [
  1431. ];
  1432. if (isset($encrypt_pool[$code])) {
  1433. return $encrypt_pool[$code];
  1434. }
  1435. $hashids = new \Hashids\Hashids('', 16, 'abcdefghjklmnopqrstuvwxyz1234567890');
  1436. $res = $hashids->decode($code);
  1437. if ($res && isset($res[0])) {
  1438. return $res[0];
  1439. }
  1440. return null;
  1441. }
  1442. //bid加密
  1443. function book_hash_encode($bid)
  1444. {
  1445. return Vinkla\Hashids\Facades\Hashids::encode($bid);
  1446. }
  1447. function decodeBid($encode_bid)
  1448. {
  1449. $bid = 0;
  1450. try {
  1451. $bid_arr = \Hashids::decode($encode_bid);
  1452. if (isset($bid_arr[0])) {
  1453. $bid = $bid_arr[0];
  1454. }
  1455. } catch (\Exception $e) {
  1456. return null;
  1457. }
  1458. return $bid;
  1459. }
  1460. /**
  1461. * 获取当前域名
  1462. */
  1463. function _domain()
  1464. {
  1465. return str_replace('https://', '', str_replace('http://', '', url('/')));
  1466. }
  1467. /**
  1468. * 字符串转*
  1469. *
  1470. * @param $str // 待转的字符串
  1471. * @param $start // 转*起始位置
  1472. * @param int $end // 转*结束位置
  1473. * @param string $dot // 转换的字符(必须是单字符,默认是*)
  1474. * @param string $charset // 编码方式
  1475. * @param string $end_char // 特殊字符(碰到此字符则确定end位置)
  1476. * @return string
  1477. */
  1478. function trans_pass($str, $start, $end = 0, $dot = "*", $charset = "UTF-8", $end_char = '@'): string
  1479. {
  1480. $len = mb_strlen($str, $charset);
  1481. if ($start == 0 || $start > $len) {
  1482. $start = 1;
  1483. }
  1484. if ($end != 0 && $end > $len) {
  1485. $end = $len - 2;
  1486. }
  1487. if (strstr($str, $end_char)) {
  1488. $end = $len - strrpos($str, $end_char);
  1489. }
  1490. $endStart = $len - $end;
  1491. $top = mb_substr($str, 0, $start, $charset);
  1492. $bottom = "";
  1493. if ($endStart > 0) {
  1494. $bottom = mb_substr($str, $endStart, $end, $charset);
  1495. }
  1496. $len -= mb_strlen($top, $charset);
  1497. $len -= mb_strlen($bottom, $charset);
  1498. $newStr = $top;
  1499. for ($i = 0; $i < $len; $i++) {
  1500. $newStr .= $dot;
  1501. }
  1502. $newStr .= $bottom;
  1503. return $newStr;
  1504. }
  1505. /**
  1506. * 格式化章节内容
  1507. *
  1508. * @param $content
  1509. * @return false|string
  1510. */
  1511. function formatContent($content)
  1512. {
  1513. if (!$content) return '';
  1514. $content = str_replace(
  1515. ['&nbsp;&nbsp;', '<br /><br />', '<br>', '<br />', '&nbsp;', '<p>', '</p >', '&ldquo;', '&rdquo;', '&hellip;'],
  1516. [' ', PHP_EOL, PHP_EOL, PHP_EOL, ' ', '', PHP_EOL, '“', '”', '...'],
  1517. $content);
  1518. $content = str_replace(["&nbsp;", '&ldquo;', '&hellip;', '&rdquo;', '<p>'], '', $content);
  1519. // 段落首字母前加两个中文空格
  1520. $string = explode(PHP_EOL, $content);
  1521. foreach ($string as $line => $text) {
  1522. if (mb_substr($text, 0, 2) != '  ') $string[$line] = '  ' . $text;
  1523. }
  1524. $content = implode(PHP_EOL, $string);
  1525. $content = mb_convert_encoding($content, 'UTF-8', 'UTF-8,GBK,GB2312');
  1526. $content = iconv('UTF-8', 'UTF-8//IGNORE', $content);
  1527. return $content;
  1528. }
  1529. /**
  1530. * 筛选出有效的id集合
  1531. *
  1532. * @param array $ids
  1533. * @return array
  1534. */
  1535. function filterValidIds(array $ids): array
  1536. {
  1537. // 传参
  1538. if (empty($ids)) {
  1539. return [];
  1540. }
  1541. $result = [];
  1542. foreach ($ids as $id) {
  1543. if (in_array($id, $result) || !is_numeric($id) || (int)$id < 1) {
  1544. continue;
  1545. }
  1546. $result[] = (int)$id;
  1547. }
  1548. return $result;
  1549. }
  1550. function arrayToStr($map)
  1551. {
  1552. $isMap = isArrMap($map);
  1553. $result = "";
  1554. if ($isMap) {
  1555. $result = "map[";
  1556. }
  1557. $keyArr = array_keys($map);
  1558. if ($isMap) {
  1559. sort($keyArr);
  1560. }
  1561. $paramsArr = array();
  1562. foreach ($keyArr as $k) {
  1563. $v = $map[$k];
  1564. if ($isMap) {
  1565. if (is_array($v)) {
  1566. $paramsArr[] = sprintf("%s:%s", $k, arrayToStr($v));
  1567. } else {
  1568. $paramsArr[] = sprintf("%s:%s", $k, trim(strval($v)));
  1569. }
  1570. } else {
  1571. if (is_array($v)) {
  1572. $paramsArr[] = arrayToStr($v);
  1573. } else {
  1574. $paramsArr[] = trim(strval($v));
  1575. }
  1576. }
  1577. }
  1578. $result = sprintf("%s%s", $result, join(" ", $paramsArr));
  1579. if (!$isMap) {
  1580. $result = sprintf("[%s]", $result);
  1581. } else {
  1582. $result = sprintf("%s]", $result);
  1583. }
  1584. return $result;
  1585. }
  1586. function isArrMap($map)
  1587. {
  1588. foreach ($map as $k => $v) {
  1589. if (is_string($k)) {
  1590. return true;
  1591. }
  1592. }
  1593. return false;
  1594. }
  1595. /**
  1596. * 随机字符串
  1597. *
  1598. * @param $length
  1599. * @return string
  1600. */
  1601. function makeRandStr($length): string
  1602. {
  1603. // 密码字符集,可任意添加你需要的字符
  1604. $str = [
  1605. 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h',
  1606. 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's',
  1607. 't', 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D',
  1608. 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
  1609. 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
  1610. '0', '1', '2', '3', '4', '5', '6', '7', '8', '9'
  1611. ];
  1612. // 在 $str 中随机取 $length 个数组元素键名
  1613. $keys = array_rand($str, $length);
  1614. $password = '';
  1615. for ($i = 0; $i < $length; $i++) {
  1616. // 将 $length 个数组元素连接成字符串
  1617. $password .= $str[$keys[$i]];
  1618. }
  1619. return $password;
  1620. }
  1621. /**
  1622. * 导出数据为excel表格
  1623. *
  1624. * @param $data 一个二维数组,结构如同从数据库查出来的数组
  1625. * @param $title excel的第一行标题,一个数组,如果为空则没有标题
  1626. * @param $filename 下载的文件名
  1627. * @examlpe10
  1628. */
  1629. function exportExcel($data = [], $title = [], $filename = 'report')
  1630. {
  1631. ob_end_clean();
  1632. ob_start();
  1633. header("Content-type:application/octet-stream");
  1634. header("Accept-Ranges:bytes");
  1635. header("Content-type:application/vnd.ms-excel");
  1636. header("Content-Disposition:attachment;filename=" . $filename . ".xls");
  1637. header("Pragma: no-cache");
  1638. header("Expires: 0");
  1639. //导出xls 开始
  1640. if (!empty($title)) {
  1641. foreach ($title as $k => $v) {
  1642. $title[$k] = iconv("UTF-8", "GB2312", $v);
  1643. }
  1644. $title = implode("\t", $title);
  1645. echo "$title\n";
  1646. }
  1647. if (!empty($data)) {
  1648. foreach ($data as $key => $val) {
  1649. foreach ($val as $ck => $cv) {
  1650. $data[$key][$ck] = iconv("UTF-8", "GB2312", $cv);
  1651. }
  1652. $data[$key] = implode("\t", $data[$key]);
  1653. }
  1654. echo implode("\n", $data);
  1655. }
  1656. }
  1657. /**
  1658. * 导出csv文件
  1659. * @param string $name
  1660. * @param array $headers
  1661. * @param array $data
  1662. * @return void
  1663. */
  1664. function exportCsv(string $name, array $headers, array $data = [])
  1665. {
  1666. header('Content-Description: File Transfer');
  1667. header('Content-Type: application/csv');
  1668. header("Content-Disposition: attachment; filename=".$name.".csv");
  1669. header('Cache-Control: must-revalidate, post-check=0, pre-check=0');
  1670. $handle = fopen('php://output', 'w');
  1671. ob_clean();
  1672. fputcsv($handle, $headers);
  1673. if ($data) {
  1674. foreach ($data as $row) {
  1675. fputcsv($handle, $row);
  1676. }
  1677. }
  1678. ob_flush();
  1679. fclose($handle);
  1680. die();
  1681. }
  1682. // 树状分类
  1683. function buildCategoryTree($categories, $pid = 0) {
  1684. $tree = [];
  1685. foreach ($categories as $category) {
  1686. if ($category['pid'] == $pid) {
  1687. $children = buildCategoryTree($categories, $category['category_id']);
  1688. if ($children) {
  1689. $category['children'] = $children;
  1690. }
  1691. $tree[] = $category;
  1692. }
  1693. }
  1694. return $tree;
  1695. }
  1696. // 获取文件的md5值
  1697. function getFileContentMD5($filePath){
  1698. //获取文件MD5的128位二进制数组
  1699. $md5Bytes = md5_file($filePath,true);
  1700. //计算文件的Content-MD5
  1701. $contentMD5 = base64_encode($md5Bytes);
  1702. return $contentMD5;
  1703. }
  1704. function getTextTokens($text) {
  1705. // 方法1:按空格分词(西文较准)
  1706. $words = preg_split('/\s+/', $text);
  1707. $wordCount = count($words);
  1708. // 方法2:按字符数估算(中文较准:1个汉字 ≈ 1.5-2 tokens)
  1709. $charCount = mb_strlen($text);
  1710. // 综合估算(根据语言调整权重)
  1711. $tokenCount = $wordCount + $charCount * 0.5; // 示例公式
  1712. // 更简单:直接按字符数 * 系数(中文推荐)
  1713. // $tokenCount = $charCount * 1.8; // 经验系数
  1714. return (int)ceil($tokenCount);
  1715. }
  1716. // 处理小说剧本文本
  1717. function handleScriptWords($text, $enable_emotion=1) {
  1718. $text = preg_replace('/[\r\n]+/', PHP_EOL, $text);
  1719. $text_arr = explode(PHP_EOL, $text);
  1720. $roles = [];
  1721. $words = [];
  1722. $role_gender = [];
  1723. // $sequence = 0;
  1724. foreach ($text_arr as $line) {
  1725. $line = trim($line);
  1726. if ($enable_emotion) {
  1727. $match_rule = '/^(.*?)\:(.*?)\{(.*?)\}$/';
  1728. $count = 4;
  1729. } else {
  1730. $match_rule = '/^(.*?)\:(.*?)$/';
  1731. $count = 3;
  1732. }
  1733. preg_match($match_rule, $line, $matches);
  1734. if (count($matches) == $count) {
  1735. $gender = '0';
  1736. // 角色部分拆分
  1737. preg_match('/^(.*?)\((.*?)\)$/', $matches[1], $matches2);
  1738. if (count($matches2) == 3) {
  1739. $role = $matches2[1];
  1740. $gender_arr = ['男'=>'1', '女'=>'2'];
  1741. $gender = isset($gender_arr[$matches2[2]]) ? $gender_arr[$matches2[2]] : '0';
  1742. }else {
  1743. $role = $matches[1];
  1744. }
  1745. if (!in_array($role, $roles)) {
  1746. $roles[] = $role; // 记录角色
  1747. $role_gender[$role] = $gender;
  1748. }
  1749. $words[] = [
  1750. 'role' => $role,
  1751. 'gender' => $gender,
  1752. 'text' => $matches[2],
  1753. 'emotion' => $enable_emotion ? $matches[3] : '中性',
  1754. ];
  1755. }
  1756. }
  1757. $new_words = [];
  1758. $tmp = '';
  1759. $tmp_arr = [];
  1760. $tmp_text = '';
  1761. // 将words数组按照role和emotion合并相邻的text内容,不相邻则跳过合并
  1762. foreach ($words as $word) {
  1763. if(!$tmp) $tmp = $word['role'].'-'.$word['emotion'];
  1764. if($tmp == $word['role'].'-'.$word['emotion']) {
  1765. $tmp_text .= PHP_EOL.$word['text'];
  1766. $tmp_arr = [
  1767. 'role' => $word['role'],
  1768. 'gender' => $word['gender'],
  1769. 'text' => trim($tmp_text, PHP_EOL),
  1770. 'emotion' => $word['emotion'],
  1771. ];
  1772. }else {
  1773. // $sequence++;
  1774. // $tmp_arr['sequence'] = $sequence;
  1775. $new_words[] = $tmp_arr;
  1776. $tmp = $word['role'].'-'.$word['emotion'];
  1777. $tmp_text = $word['text'];
  1778. $tmp_arr = [
  1779. 'role' => $word['role'],
  1780. 'gender' => $word['gender'],
  1781. 'text' => trim($tmp_text, PHP_EOL),
  1782. 'emotion' => $word['emotion'],
  1783. ];
  1784. }
  1785. }
  1786. if ($tmp_arr) {
  1787. // $sequence++;
  1788. // $tmp_arr['sequence'] = $sequence;
  1789. $new_words[] = $tmp_arr;
  1790. }
  1791. return [
  1792. 'roles' => $roles,
  1793. 'role_gender' => $role_gender,
  1794. 'words' => $new_words,
  1795. ];
  1796. }
  1797. /**
  1798. * 处理剧本内容
  1799. */
  1800. function handleScriptContent($originalContent) {
  1801. if (!$originalContent) return [];
  1802. // 定义原始内容
  1803. // $originalContent = <<<'EOD'
  1804. // ###剧本名:西昆仑
  1805. // ###故事梗概
  1806. // 内容梗概:少女许芸重生回到悲剧发生前一个月,她本是许家假千金,真千金徐清遥因在徐家遭受表哥卞大伟侵犯及养母徐母包庇而心怀怨恨,最终导致许芸被逼联姻、家暴惨死。重生后,许芸提前让身份归位,试图避开悲剧。但她发现一个因暗恋她而与系统交易、令她重生的“空气人”始终在暗中保护她,她却因系统惩罚永远无法看见、听见他。许芸在躲避卞大伟侵害、应对原生家庭矛盾的过程中,逐渐感知到“空气人”的存在与深情。她努力追寻他的身份,得知他是她曾资助的学霸程景。两人在“看不见”的困境中相互靠近,经历生死危机与误解分离后,最终冲破系统束缚,得以相见相守,共度余生。
  1807. // ###剧本亮点
  1808. // 亮点1:重生救赎与无形守护
  1809. // 开篇以许芸死亡回忆切入,画面阴郁沉重,转场重生瞬间光影骤亮。许芸躲避悲剧时,“空气人”多次在危急关头无形相助——如车祸前拉扯卫衣、对抗卞大伟时扭打空气、深夜喂水亲吻等。镜头以许芸主观视角呈现,观众只见她反应却不见施救者,营造悬疑与浪漫交织的张力,突出“被深爱却不自知”的宿命感。
  1810. // 亮点2:系统惩罚下的情感博弈
  1811. // 许芸与“空气人”程景因系统设定无法直接接触,却通过“血脚印”“消失的面条”“深夜触碰”等细节传递暧昧与温情。许芸大胆挑逗、程景克制回避,两人在“看不见”的局限中试探彼此心意,既有荒诞喜感,又充满情感拉扯的酸涩与甜蜜,颠覆传统恋爱叙事。
  1812. // 亮点3:身份谜团与校园暗恋
  1813. // 许芸通过重点班花名册、志愿填报等线索,拼凑程景身份。场景在校园、阁楼间切换,许芸努力“听不见”程景信息时的茫然与执着,与夏颜的八卦互动形成反差,逐步揭开“资助者与被资助者”的前缘。观众随许芸一起解谜,沉浸式体验暗恋回溯的感动。
  1814. // 亮点4:生死危机与真相爆发
  1815. // 卞大伟持刀复仇段落紧张激烈,程景为救许芸受重伤“显形”血迹,却因伤情自卑选择分手。许芸在法院外闻到皂角味、假意跳楼逼其现身,将情感推向高潮。最终阁楼相见,系统束缚破除,两人记忆补全,过往守护细节——浮现,情感爆发力十足。
  1816. // 亮点5:跨越苦难的终成眷属
  1817. // 结局以十年后怀孕收尾,画面温暖明亮。许芸与程景回顾重生与守护历程,对话温情而释然。悲剧循环被打破,强调“坚持与爱可冲破既定命运”的主题,给予观众情感治愈与希望感。
  1818. // ###人物关系
  1819. // 许芸与程景:重生前为资助者与被资助者,程景长期暗恋许芸;重生后程景化为“空气人”默默守护,许芸逐渐感知并主动靠近,最终成为恋人、夫妻。
  1820. // 许芸与徐清遥:被抱错的真假千金,徐清遥因遭遇迁怒许芸,重生后许芸主动归位身份避免冲突。
  1821. // 许芸与徐母、卞大伟:徐母重男轻女、包庇侄子;卞大伟是侵犯者加害者,对许芸屡次施暴。
  1822. // 许芸与夏颜:同学兼闺蜜,夏颜单恋程景,一度误导许芸,后和解。
  1823. // 程景与系统:交易关系,程景以“许芸永远看不见他”为代价换她重生。
  1824. // ###核心矛盾
  1825. // 1. 许芸与既定悲剧命运的矛盾:她需避开卞大伟侵害、徐母压迫及徐清遥怨恨,改变惨死结局。
  1826. // 2. 许芸与程景“不可见”的系统惩罚矛盾:两人相爱却无法直接接触、沟通,情感传递受阻。
  1827. // 3. 程景的自卑与守护矛盾:他因出身、伤势自觉不配,一度逃避,与许芸的坚定追求形成拉扯。
  1828. // 4. 许芸与原生家庭的矛盾:徐母的苛待、卞大伟的威胁,迫使她反抗、逃离。
  1829. // ###主体列表
  1830. // 许芸-重生者:18岁少女,原许家假千金,惨死后重生,决心改变命运,性格从乖顺转向果敢。
  1831. // 许芸-追寻者:感知程景存在后,主动探寻其身份,大胆表达爱意,坚韧执着。
  1832. // 程景/“空气人”/1号同学:学霸,父母早逝,受许芸资助长大,暗恋许芸至深,以系统交易换她重生,默默守护。
  1833. // 卞大伟:徐母侄子,混混,侵犯徐清遥未遂后多次企图伤害许芸,最终入狱。
  1834. // 徐母:许芸生母,重男轻女,包庇卞大伟,苛待女儿。
  1835. // 徐父:怯懦但逐渐觉醒,最终与徐母离婚支持女儿。
  1836. // 徐清遥:真千金,因遭遇怨恨许芸,重生后未直接报复。
  1837. // 夏颜:许芸同学,暗恋程景,曾误导许芸,后和解。
  1838. // 系统:无形存在,设定“许芸永远看不见程景”的惩罚。
  1839. // ###美术风格
  1840. // 基础画风风格词:现代写实略带奇幻色调
  1841. // 整体采用细腻写实风格,色彩随情绪与剧情变化。重生前回忆片段采用冷灰、暗蓝色调,氛围压抑;重生后日常场景以自然光、生活化色彩为主,突出真实感。程景作为“空气人”存在时,镜头多以许芸主观视角拍摄,画面中他“缺席”却留有痕迹(如血脚印、水面波动),辅以微妙光影与音效暗示其存在。两人情感互动时,色调转为暖黄、柔光,烘托暧昧与温情。危机场景如遇袭、跳楼等,采用快速剪辑、手持晃动镜头与对比色,强化紧张感。结局部分画面明亮温暖,象征新生与希望。
  1842. // ###场景列表
  1843. // 徐家馄饨店:老旧街边小店,锅炉、案板杂乱,象征许芸压抑的原生家庭。
  1844. // 网吧:烟雾缭绕,光线昏暗,许芸在此熬夜躲避悲剧。
  1845. // 街道/路口:日常街景,车祸救人处,程景首次“无形”相助。
  1846. // 程景阁楼:狭小但整洁,书桌、床、矮几,两人主要相处空间,充满私密与温情。
  1847. // 学校教室/走廊:重点班与普通班场景,许芸探寻程景身份的主要场所。
  1848. // 医院:程景受伤住院处,许芸因看不见而无助。
  1849. // 法院外台阶:卞大伟宣判后,许芸与徐母冲突,程景暗中出现。
  1850. // 平台/草坪:许芸假意跳楼逼程景现身之处。
  1851. // ###分集剧本
  1852. // ##分集02
  1853. // 分集名: 街头相遇
  1854. // 场景描述:街道 / 日 / 外
  1855. // 出场角色:许芸、程景(无形)、老奶奶
  1856. // 台词内容:
  1857. // 许芸:(内心独白,画外音)我重生了。回到悲剧发生前一个月。这一次,我绝不会再走上那条绝路。
  1858. // (许芸熬夜后恍惚过马路,汽车疾驰而来)
  1859. // 许芸:(惊愕瞪大眼)啊!
  1860. // (许芸被无形力量猛地向后拉扯,踉跄站稳)
  1861. // 老奶奶:(拄拐走近)这司机开车横冲直撞,要不是小姑娘你反应快,一定会被撞到。
  1862. // 许芸:(环顾四周,茫然)不是我自己躲的……
  1863. // (许芸急切在人群中寻找,风吹过她的发梢)
  1864. // 许芸:(低声,眼眶微红)是你吗?傻瓜……你看我惨死的时候,很痛吧。
  1865. // ##分集03
  1866. // 分集名: 重生
  1867. // 场景描述:街道 / 日 / 外
  1868. // 出场角色:许芸、程景(无形)、老奶奶
  1869. // 台词内容:
  1870. // 许芸:(内心独白,画外音)我重生了。回到悲剧发生前一个月。这一次,我绝不会再走上那条绝路。
  1871. // (许芸熬夜后恍惚过马路,汽车疾驰而来)
  1872. // 许芸:(惊愕瞪大眼)啊!
  1873. // (许芸被无形力量猛地向后拉扯,踉跄站稳)
  1874. // 老奶奶:(拄拐走近)这司机开车横冲直撞,要不是小姑娘你反应快,一定会被撞到。
  1875. // 许芸:(环顾四周,茫然)不是我自己躲的……
  1876. // (许芸急切在人群中寻找,风吹过她的发梢)
  1877. // 许芸:(低声,眼眶微红)是你吗?傻瓜……你看我惨死的时候,很痛吧。
  1878. // 场景描述:徐家馄饨店 / 日 / 内
  1879. // 出场角色:许芸、徐母、卞大伟
  1880. // 台词内容:
  1881. // 徐母:(挥舞扫帚打许芸)夜不归宿死哪儿去了?!
  1882. // 许芸:(躲闪)去许家让清遥给我补课了。
  1883. // 徐母:(扔下扫帚,指案板)去包馄饨!
  1884. // (许芸沉默洗手,余光瞥向锅炉后的卞大伟)
  1885. // 卞大伟:(阴沉盯着许芸,无声冷笑)
  1886. // 许芸:(内心独白)上辈子,就是他在今天毁了徐清遥。幸好,我躲过去了。
  1887. // ##分集04
  1888. // 分集名: 馄饨店
  1889. // 场景描述:徐家馄饨店 / 傍晚 / 内
  1890. // 出场角色:许芸、卞大伟、程景(无形)
  1891. // 台词内容:
  1892. // (许芸放学进门,卞大伟猛地拉住她)
  1893. // 卞大伟:今天姑姑不在,看谁还能救你!
  1894. // 许芸:(挣扎)放开我!
  1895. // (桌椅撞倒,许芸掏刀,卞大伟突然怪异地摔倒)
  1896. // 卞大伟:(朝空气怒吼)你他妈谁啊?!
  1897. // 许芸:(愣住,随即眼睛一亮)是你!
  1898. // (许芸趁机跑出,回头看见卞大伟与“空气”扭打)
  1899. // 许芸:(朝屋内喊)我们走!
  1900. // (许芸跑出,看到臭水沟溅起水花,地上出现血脚印)
  1901. // 许芸:(追上)你受伤了!
  1902. // (血脚印停住)
  1903. // 许芸:(固执)我要跟你在一起。
  1904. // 场景描述:程景阁楼 / 夜 / 内
  1905. // 出场角色:许芸、程景(无形)
  1906. // 台词内容:
  1907. // (许芸坐在床边,矮几上出现两碗面,她对面的面条自动减少)
  1908. // 许芸:(看着空气)你一直在看着我,对吧?
  1909. // (许芸眼泪滴落)
  1910. // 许芸:(哽咽)你太惨了……换我重生,你却连名字都不能让我知道。
  1911. // (许芸擦泪吃面)
  1912. // 许芸:我今晚住你这。
  1913. // (许芸洗澡后穿T恤躺下,关灯)
  1914. // 许芸:(黑暗中)试试看我能不能碰到你。
  1915. // (许芸伸手摸索,只触到空气)
  1916. // 许芸:(沮丧)睡觉了。
  1917. // EOD;
  1918. // $originalContent = <<<'EOD'
  1919. // ###剧本名:看不见的守护者
  1920. // ###故事梗概
  1921. // 少女许芸重生回到悲剧发生前一个月,她本是豪门假千金,上辈子被真千金徐清遥报复,被迫联姻后惨死。一个暗恋她至深的陌生男人程景,用与系统交易的代价换她重生,代价是她永远看不见他。许芸利用先知改变命运,躲避了表哥卞大伟的侵犯,并试图回归平凡生活。在危机四伏的重生路上,那个“看不见”的守护者程景一次次救她于危难。许芸从恐惧、好奇到依赖,并主动追寻这个神秘存在。两人在“一个看得见,一个看不见”的荒诞设定下,发展出一段跨越生死、充满拉扯与救赎的深情。最终,许芸的执着冲破了系统规则,两人得以相见相守,共度余生。
  1922. // ###剧本亮点
  1923. // 亮点1:重生设定与“隐形”守护者的双重奇幻
  1924. // 开篇即用快速剪辑展现许芸上辈子的惨死与陌生男人程景为她下葬、与系统交易的画面,建立强烈的戏剧冲突与悬念。重生后,许芸视角中程景的“不存在”与观众视角(或通过环境互动)能感知的“存在”形成奇妙反差,如被无形力量救下车、凭空出现的面条、地上的血脚印等,视觉呈现新颖,情感张力十足。
  1925. // 亮点2:“看见”与“看不见”的极致情感拉扯
  1926. // 许芸对着一团空气说话、试探、甚至调情的场景,充满了戏剧性的孤独与浪漫。她努力感知程景的存在,通过细微痕迹(如血迹、水渍、移动的物品)与他“交流”。这种单向可见的互动,将暗恋的卑微与守护的深情推到极致,观众能深切共情两人近在咫尺却无法触及的煎熬与甜蜜。
  1927. // 亮点3:悬疑推进与身份揭秘的层层递进
  1928. // 剧本并非单纯恋爱,穿插着许芸对抗原生家庭伤害、智斗恶毒表哥卞大伟的主线。程景的“隐形”能力成为她破局的关键外挂,但也带来了新的危险。随着许芸通过学校花名册、资助信等线索一步步接近程景的真实身份,悬疑感与情感共鸣同步加深,直到最终高潮的“相见”时刻,情感得到彻底释放。
  1929. // ###人物关系
  1930. // 许芸与程景:被守护者与隐形守护者,后成为恋人、夫妻。许芸从依赖、好奇到深爱并主动追寻程景;程景则因感恩与深爱,默默守护许芸两世。
  1931. // 许芸与徐家父母:亲生父母,关系疏离冷漠。母亲重男轻女,偏袒侄子卞大伟;父亲怯懦,后期在许芸影响下有所改变。
  1932. // 许芸与卞大伟:表哥,施暴者与主要反派,对许芸有侵犯意图,是前期主要威胁。
  1933. // 许芸与徐清遥:身份互换的真假千金,上辈子是仇敌,这辈子因许芸提前归位身份而关系未彻底恶化,但存在潜在矛盾。
  1934. // 许芸与夏颜:高中同桌兼闺蜜,后期因同时喜欢程景而产生嫉妒与隔阂,是推动程景“分手”谎言的工具人。
  1935. // ###核心矛盾
  1936. // 1. **生存矛盾**:重生后的许芸需躲避上辈子的悲剧命运(被卞大伟侵犯、被家庭出卖联姻),在恶劣原生环境中挣扎求生。
  1937. // 2. **感知矛盾**:系统规则导致许芸无法看见、听见程景,而程景深爱并守护着她。两人之间存在极致的“接触”渴望与“隔绝”现实的矛盾。
  1938. // 3. **情感矛盾**:许芸在依赖与追寻程景的过程中渐生情愫,但程景因自卑(身份差距)、身体创伤(以为不育)及系统限制,一度选择逃避和“分手”,造成两人的情感拉扯。
  1939. // 4. **身份矛盾**:真假千金的错位人生带来的家庭伦理与社会地位冲突,是许芸前世悲剧的根源,也是她今生需要面对和化解的阴影。
  1940. // ###主体列表
  1941. // 许芸-重生迷茫:刚重生时,心怀恐惧与决绝,试图改变命运。
  1942. // 许芸-坚韧求生:面对原生家庭压迫,冷静周旋,努力读书改变命运。
  1943. // 许芸-主动追寻:对“空气人”程景从好奇到依赖,再到大胆主动地试探与追求。
  1944. // 程景-隐形守护:始终在许芸身边,默默保护、照顾她,情感深沉而克制。
  1945. // 程景-隐忍挣扎:因系统限制和自身顾虑,承受着无法被感知的孤独与爱而不得的痛苦。
  1946. // 卞大伟-阴狠施暴:对许芸心怀不轨,是具象化的危险来源。
  1947. // 徐母-偏执刻薄:重男轻女,维护侄子,是家庭伤害的主要施加者。
  1948. // 徐父-怯懦转变:前期麻木,后期在女儿影响下逐渐觉醒。
  1949. // 夏颜-闺蜜变情敌:前期提供帮助,后期因嫉妒程景而说谎离间。
  1950. // “系统”:无形的规则设定,制造了主角间“看不见”的核心障碍。
  1951. // ###美术风格
  1952. // 基础画风风格词:写实细腻的现代都市情感风格,带有轻微悬疑色彩。
  1953. // 整体采用电影质感的写实风格,色彩随情绪和剧情变化。许芸重生前的回忆片段使用冷峻、压抑的蓝灰色调,对比强烈。重生后的现实生活,色调偏向自然光感,但徐家馄饨店等场景色调偏暗、拥挤,体现压抑感。当“隐形”的程景互动时,画面注重细节特写(如自动移动的物体、凹陷的床垫、凭空出现的水杯),并辅以微妙的光影变化或轻柔的焦点虚化来暗示他的“存在”。许芸与程景的独处时光,尤其是阁楼场景,色调转为温暖、柔和的橙黄色,营造私密与暧昧氛围。最终相见时刻,光线明亮饱满,色彩鲜活,象征隔阂的打破与希望降临。
  1954. // ###场景列表
  1955. // 徐家馄饨店:拥挤、老旧、充满烟火气的小店,是许芸重生后的主要生活场景,也是危机发生地。
  1956. // 学校教室/走廊:青春校园环境,是许芸努力学习、与夏颜交流的主要场景。
  1957. // 网吧包间:许芸躲避第一夜危机的地点,嘈杂、昏暗。
  1958. // 城市街道/路口:多次发生危机(车祸、被卞大伟堵截)和程景隐形救援的场景。
  1959. // 程景的阁楼:狭小但整洁,是两人的秘密基地和情感发酵的主要空间,充满生活气息与私密感。
  1960. // 医院:程景受伤后治疗的地点,也是许芸感到无助与寻找他的地方。
  1961. // 法院门口:卞大伟宣判后,徐母闹事、许芸再次感知到程景存在的场景。
  1962. // ###分集剧本
  1963. // ##分集01
  1964. // 分集名:重生与隐形救援
  1965. // 分镜01
  1966. // 场景描述:徐家馄饨店后院卧室,清晨,光线昏暗。许芸从噩梦中惊醒,额头上全是冷汗。
  1967. // 运镜:特写许芸惊恐睁大的双眼,快速闪回上辈子被家暴、惨死、以及一个模糊男人背影为她下葬的碎片画面。
  1968. // 出场角色:许芸
  1969. // 台词内容:
  1970. // 许芸(内心独白,带着颤音):我重生了…源于一个男人偏执的暗恋。
  1971. // 分镜02
  1972. // 场景描述:馄饨店前厅,清晨。徐母穿戴整齐,对正在收拾书包的许芸吩咐。卞大伟在锅炉后假装忙碌,眼神阴鸷地偷瞄许芸。
  1973. // 运镜:中景,先聚焦徐母严厉的脸,然后镜头平移,扫过乖巧点头的许芸,最后落在卞大伟身上,给他一个眼神特写。
  1974. // 出场角色:许芸,徐母,卞大伟
  1975. // 台词内容:
  1976. // 徐母:今天店就交给你表哥看着,晚上一放学就回来帮他干活。
  1977. // 许芸:(低头)嗯。
  1978. // 徐母:(转身离开)
  1979. // (许芸目光扫向卞大伟,内心独白):上辈子,就是他侮辱了徐清遥…也是我一切悲剧的开始。
  1980. // 分镜03
  1981. // 场景描述:网吧包间,夜。许芸坐在电脑前,屏幕光映着她苍白的脸。周围烟雾缭绕,嘈杂。
  1982. // 运镜:固定机位,侧面拍摄许芸蜷缩在椅子里的身影,环境音突出键盘声和隐约的游戏音效。
  1983. // 出场角色:许芸
  1984. // 台词内容:
  1985. // 许芸(内心独白):我在这里躲了一夜…可脑子里全是上辈子的事。徐清遥的恨,养父母的冷漠,那个老畜生的折磨…还有,那个给我收尸的陌生男人…
  1986. // 分镜04
  1987. // 场景描述:城市街道,清晨。许芸背着书包,困倦地过马路。一辆车急速驶来。
  1988. // 运镜:主观镜头,许芸视线模糊地看向冲来的车。紧接着第三视角,许芸的卫衣后领突然向后一紧,她整个人被“拽”离原地,车子擦身而过。
  1989. // 出场角色:许芸,路人老奶奶
  1990. // 台词内容:
  1991. // (急刹车声)
  1992. // 老奶奶:(拄着拐走近)这司机开车横冲直撞,要不是小姑娘你反应快…
  1993. // 许芸:(惊魂未定,四下张望)不是我自己躲的…
  1994. // (内心独白):是他!他就在我身边!
  1995. // 分镜05
  1996. // 场景描述:馄饨店后屋,白天。许芸刚进门,徐母拿着扫帚劈头盖脸打来。卞大伟在门口阴沉地看着。
  1997. // 运镜:手持晃动镜头,模拟挨打的混乱感。穿插卞大伟站在逆光门口的阴沉中景。
  1998. // 出场角色:许芸,徐母,卞大伟
  1999. // 台词内容:
  2000. // 徐母:死丫头!夜不归宿死哪儿去了?
  2001. // 许芸:(躲闪)去许家让清遥给我补课了!
  2002. // 徐母:(打累了,扔下扫帚)去包馄饨!
  2003. // (许芸沉默地洗手,开始包馄饨,余光警惕着卞大伟)
  2004. // ##分集02
  2005. // 分集名:危机再现与阁楼初夜
  2006. // 分镜01
  2007. // 场景描述:馄饨店后屋,傍晚。许芸放学推门进来,被埋伏的卞大伟一把抓住手腕。
  2008. // 运镜:快速推近,特写许芸惊愕的脸和卞大伟狞笑的手。镜头跟随挣扎的两人摇晃,撞倒桌椅。
  2009. // 出场角色:许芸,卞大伟
  2010. // 台词内容:
  2011. // 许芸:你干什么?放开我!
  2012. // 卞大伟:(喘着粗气)今天可没人救你了!
  2013. // (许芸挣扎中摸出兜里的小刀)
  2014. // 分镜02
  2015. // 场景描述:同上场景。卞大伟正要压倒许芸,突然怪叫着向后仰倒,仿佛被无形力量击中。
  2016. // 运镜:慢镜头,卞大伟身体不自然地扭曲、摔倒。许芸趁机爬起,震惊地看着“空气”。
  2017. // 出场角色:许芸,卞大伟
  2018. // 台词内容:
  2019. // 卞大伟:(对着空气怒吼)你他妈谁啊?!
  2020. // 许芸:(心口狂跳,看向卞大伟扭打的方向)是他…只能是他!
  2021. // 许芸:(朝“空气”大喊)我们走!
  2022. // 分镜03
  2023. // 场景描述:老旧居民楼楼梯,傍晚。许芸跟着一串突然出现在地上的“血脚印”奔跑。
  2024. // 运镜:俯拍镜头,跟随许芸上楼的脚步和地上断断续续的血脚印。
  2025. // 出场角色:许芸
  2026. // 台词内容:
  2027. // 许芸:(对着前方的空气)你受伤了!
  2028. // (血脚印停住)
  2029. // 许芸:(执拗地)我要跟你在一起。
  2030. // 分镜04
  2031. // 场景描述:狭小阁楼内,夜。许芸坐在床边,好奇地打量房间。小矮几上凭空出现两碗热气腾腾的面条。
  2032. // 运镜:全景展示阁楼布局,然后聚焦到矮几上“自动出现”的面碗。对面碗里的面条开始“自动”减少。
  2033. // 出场角色:许芸
  2034. // 台词内容:
  2035. // 许芸:(看着对面“自动”减少的面条,眼泪突然掉下来)啪嗒…好惨…这男的太惨了…
  2036. // 分镜05
  2037. // 场景描述:阁楼床上,夜。许芸穿着宽大T恤躺在床上。她突然睁开眼,感觉嘴唇上有温热柔软的触感。
  2038. // 运镜:特写许芸猛然睁大的眼睛,和微微湿润的嘴唇。背景是浴室隐约的水声。
  2039. // 出场角色:许芸
  2040. // 台词内容:
  2041. // 许芸:(内心惊愕)他在偷吻我?我…感觉到了?
  2042. // (她闭眼假装睡着,感觉脖颈又被亲了一下,忍不住轻颤)
  2043. // (浴室水声再次响起)
  2044. // 许芸:(嘴角微微勾起)好像…发现了系统的BUG?
  2045. // ##分集03
  2046. // 分集名:身份追寻与最终相见
  2047. // 分镜01
  2048. // 场景描述:学校教室,白天。许芸拿着重点班花名册,请同桌夏颜念名字。夏颜念到第一个名字时,许芸一脸茫然。
  2049. // 运镜:过肩镜头,从许芸视角看夏颜的嘴在动,但配以表示“无声”或“雪花噪音”的音效和画面扭曲效果。
  2050. // 出场角色:许芸,夏颜
  2051. // 台词内容:
  2052. // 夏颜:(念名单)…
  2053. // 许芸:(打断,急切地)夏夏,等高考后,你能帮我打听一下1号同学报哪所大学吗?我想跟他一起。
  2054. // 夏颜:(皱眉,嘴巴张合——许芸听不清)
  2055. // 许芸:(自顾自点头,眼冒星星)对,我喜欢1号同学!
  2056. // 分镜02
  2057. // 场景描述:法院外台阶,白天。卞大伟宣判后被带走,徐母发疯般冲过来要打许芸,被徐父拦住。许芸下台阶时,突然停下,嗅了嗅空气。
  2058. // 运镜:中景拍摄混乱场面,然后特写许芸突然愣住、微微抽动鼻翼的表情。
  2059. // 出场角色:许芸,徐母,徐父
  2060. // 台词内容:
  2061. // 徐母:(尖叫)许芸!你毁了我的一切!
  2062. // 许芸:(对徐父)爸,拉住她!(下台阶,脚步一顿)…皂角味?
  2063. // (内心独白):他来了?为什么…
  2064. // 分镜03
  2065. // 场景描述:程景家楼下小平台,傍晚。许芸站在平台边缘,下面是草坪。
  2066. // 运镜:全景,许芸孤身站在平台边,风吹动她的头发。然后切换为她主观视角,看向下方。
  2067. // 出场角色:许芸
  2068. // 台词内容:
  2069. // 许芸:(环顾四周,大声说)1号同学,没有你,我不想活了!
  2070. // (作势要跳)
  2071. // 分镜04
  2072. // 场景描述:同上场景。许芸身体前倾跌落的瞬间,被一股无形的力量拦腰抱住,然后轻轻“丢”在平台地面上。
  2073. // 运镜:高速摄影,捕捉许芸跌落瞬间被“空气”拦截并缓缓放下的超现实画面。
  2074. // 出场角色:许芸
  2075. // 台词内容:
  2076. // 许芸:(被抱住时惊喜)1号同学!我就知道…(被丢到地上,屁股疼)哎哟!
  2077. // 许芸:(爬起来,对着空气喊)你非要这么折磨我吗?还是你和夏颜在一起了?!
  2078. // 分镜05
  2079. // 场景描述:阁楼内,夜。许芸被浴室水声吵醒,迷迷糊糊走过去,赫然看见一个裸身男人(程景)在洗澡。
  2080. // 运镜:从许芸迷糊的主观镜头,到看清浴室景象时猛地拉近、定格在程景背影的震惊视角。
  2081. // 出场角色:许芸,程景
  2082. // 台词内容:
  2083. // 许芸:(惊吓后退)啊!你…你是谁?!
  2084. // (程景转身,两人对视)
  2085. // 程景:(震惊,试探)许芸?你能看见我?
  2086. // 许芸:(警惕)我当然能看见你!变-态!
  2087. // (程景扯掉浴巾,许芸惊呼转身又转回)
  2088. // 程景:(快速穿裤子)我就是你的1号同学。
  2089. // 分镜06
  2090. // 场景描述:阁楼内,片刻后。许芸接着夏颜的电话,程景站在一旁。接完电话,许芸走向程景。
  2091. // 运镜:双人镜头,许芸一边接电话,一边目光复杂地看着程景。挂电话后,她主动靠近,镜头柔和。
  2092. // 出场角色:许芸,程景
  2093. // 台词内容:
  2094. // 许芸:(对电话)…这挺好,可以毫无顾忌了。(挂电话,走向程景)
  2095. // 许芸:(圈住程景脖子)夏颜说,你受伤了,可能做不了爸爸?
  2096. // 程景:(眼神一暗)…嗯。
  2097. // 许芸:(微笑)没关系。我只想跟你好好地在一起。
  2098. // (两人相拥,窗外天色渐亮)
  2099. // EOD;
  2100. // 使用更精确的正则表达式分割内容
  2101. $parts = [];
  2102. // // 确保内容使用UTF-8编码
  2103. // if (!mb_check_encoding($originalContent, 'UTF-8')) {
  2104. // $originalContent = mb_convert_encoding($originalContent, 'UTF-8', 'auto');
  2105. // }
  2106. // 提取剧本名(###剧本名:后面的内容)
  2107. if (preg_match('/###剧本名[::]\s*(.*?)(?=\n|$)/u', $originalContent, $summaryMatch)) {
  2108. $parts['script_name'] = isset($summaryMatch[1]) ? trim($summaryMatch[1]) : '';
  2109. }
  2110. // 提取故事梗概(直到遇到下一个###标记)
  2111. preg_match('/###故事梗概\s*\n(.*?)(?=\n###|$)/s', $originalContent, $summaryMatch);
  2112. $parts['intro'] = isset($summaryMatch[1]) ? trim($summaryMatch[1]) : '';
  2113. // 提取剧本亮点
  2114. preg_match('/###剧本亮点\s*\n(.*?)(?=\n###|$)/s', $originalContent, $highlightsMatch);
  2115. $parts['highlights'] = isset($highlightsMatch[1]) ? trim($highlightsMatch[1]) : '';
  2116. // 提取人物关系
  2117. preg_match('/###人物关系\s*\n(.*?)(?=\n###|$)/s', $originalContent, $relationsMatch);
  2118. $parts['role_relationship'] = isset($relationsMatch[1]) ? trim($relationsMatch[1]) : '';
  2119. // 提取核心矛盾
  2120. preg_match('/###核心矛盾\s*\n(.*?)(?=\n###|$)/s', $originalContent, $contradictionsMatch);
  2121. $parts['core_contradiction'] = isset($contradictionsMatch[1]) ? trim($contradictionsMatch[1]) : '';
  2122. // 提取主体列表
  2123. preg_match('/###主体列表\s*\n(.*?)(?=\n###|$)/s', $originalContent, $subjectsMatch);
  2124. $rolesText = isset($subjectsMatch[1]) ? trim($subjectsMatch[1]) : '';
  2125. $parts['roles'] = parseRolesFromText($rolesText);
  2126. // 提取美术风格
  2127. preg_match('/###美术风格\s*\n(.*?)(?=\n###|$)/s', $originalContent, $artStyleMatch);
  2128. $parts['art_style'] = isset($artStyleMatch[1]) ? trim($artStyleMatch[1]) : '';
  2129. // 提取场景列表
  2130. preg_match('/###场景列表\s*\n(.*?)(?=\n###|$)/s', $originalContent, $scenesMatch);
  2131. $scenesText = isset($scenesMatch[1]) ? trim($scenesMatch[1]) : '';
  2132. $parts['scenes'] = parseScenesFromText($scenesText);
  2133. // 提取分集详细内容
  2134. preg_match('/###分集详细内容\s*\n(.*?)(?=\n###|$)/s', $originalContent, $contentMatch);
  2135. $parts['content'] = isset($contentMatch[1]) ? trim($contentMatch[1]) : '';
  2136. $parts['episode_title'] = '';
  2137. $parts['acts'] = [];
  2138. // 单剧集格式:顶层是大纲信息,分镜部分采用 handleEpisodeContent 的结构标准
  2139. $singleStoryboard = '';
  2140. if (preg_match('/###\s*分镜剧本\s*\n(.*?)(?=\n\s*###[^#]|\z)/su', $originalContent, $singleStoryboardMatch)) {
  2141. $singleStoryboard = trim($singleStoryboardMatch[1]);
  2142. }
  2143. if ($singleStoryboard !== '') {
  2144. $episodeContent = $singleStoryboard;
  2145. if (!preg_match('/第\d+集[::\s]+/u', $episodeContent)) {
  2146. $defaultEpisodeTitle = '第1集:' . ($parts['script_name'] ?? '未命名');
  2147. $episodeContent = $defaultEpisodeTitle . "\n\n" . $episodeContent;
  2148. }
  2149. $episodeSections = [];
  2150. $episodeSections[] = $episodeContent;
  2151. if ($parts['intro'] !== '') {
  2152. $episodeSections[] = "###故事梗概\n" . $parts['intro'];
  2153. }
  2154. if ($parts['art_style'] !== '') {
  2155. $episodeSections[] = "###美术风格\n" . $parts['art_style'];
  2156. }
  2157. if ($rolesText !== '') {
  2158. $episodeSections[] = "###主体列表\n" . $rolesText;
  2159. }
  2160. if ($scenesText !== '') {
  2161. $episodeSections[] = "###场景列表\n" . $scenesText;
  2162. }
  2163. if (strpos($episodeContent, '###分镜剧本') === false) {
  2164. $episodeSections[] = "###分镜剧本\n" . $singleStoryboard;
  2165. }
  2166. $episode_arr = handleEpisodeContent(implode("\n\n", $episodeSections));
  2167. if (!empty($episode_arr['acts'])) {
  2168. $parts['episode_title'] = getProp($episode_arr, 'episode_title');
  2169. $parts['acts'] = getProp($episode_arr, 'acts', []);
  2170. if (empty($parts['roles'])) {
  2171. $parts['roles'] = getProp($episode_arr, 'roles', []);
  2172. }
  2173. if (empty($parts['scenes'])) {
  2174. $parts['scenes'] = getProp($episode_arr, 'scenes', []);
  2175. }
  2176. }
  2177. }
  2178. if (empty($parts['acts'])) {
  2179. $fallbackEpisodeArr = handleEpisodeContent($originalContent);
  2180. if (!empty($fallbackEpisodeArr['acts'])) {
  2181. $parts['episode_title'] = getProp($fallbackEpisodeArr, 'episode_title');
  2182. $parts['acts'] = getProp($fallbackEpisodeArr, 'acts', []);
  2183. if (empty($parts['intro'])) {
  2184. $parts['intro'] = getProp($fallbackEpisodeArr, 'intro');
  2185. }
  2186. if (empty($parts['art_style'])) {
  2187. $parts['art_style'] = getProp($fallbackEpisodeArr, 'art_style');
  2188. }
  2189. if (empty($parts['roles'])) {
  2190. $parts['roles'] = getProp($fallbackEpisodeArr, 'roles', []);
  2191. }
  2192. if (empty($parts['scenes'])) {
  2193. $parts['scenes'] = getProp($fallbackEpisodeArr, 'scenes', []);
  2194. }
  2195. }
  2196. }
  2197. // 多剧集格式:继续兼容旧的分集剧本结构
  2198. $fullScript = '';
  2199. if (preg_match('/###分集剧本\s*\n(.*)/su', $originalContent, $storyboardMatch)) {
  2200. $fullScript = trim($storyboardMatch[1]);
  2201. } elseif (preg_match('/##分集剧本\s*\n(.*)/su', $originalContent, $storyboardMatch)) {
  2202. $fullScript = trim($storyboardMatch[1]);
  2203. } elseif (preg_match('/分集剧本[::]\s*\n(.*)/su', $originalContent, $storyboardMatch)) {
  2204. $fullScript = trim($storyboardMatch[1]);
  2205. }
  2206. if (empty($fullScript) && preg_match('/(##分集.*)/su', $originalContent, $fallbackMatch)) {
  2207. $fullScript = trim($fallbackMatch[1]);
  2208. }
  2209. $episodes = [];
  2210. preg_match_all('/##分集(\d+)\s*\n分集名[::]\s*(.*?)\s*\n(?:场景描述[::].*?\n出场角色[::].*?\n台词内容[::]\s*\n)?(.*?)(?=\n##分集|$)/su', $fullScript, $matches, PREG_SET_ORDER);
  2211. foreach ($matches as $match) {
  2212. $episodeNumber = (int)trim($match[1]);
  2213. $episodeName = trim($match[2]);
  2214. $episodeContent = trim($match[3]);
  2215. $segments = [];
  2216. preg_match_all('/分镜(\d+)\s*\n(.*?)(?=\n分镜\d+|\z)/s', $episodeContent, $segmentMatches, PREG_SET_ORDER);
  2217. foreach ($segmentMatches as $segMatch) {
  2218. $segmentNumber = (int)trim($segMatch[1]);
  2219. $segmentContent = trim($segMatch[2]);
  2220. $segments[] = [
  2221. 'segment_number' => $segmentNumber,
  2222. 'segment_content' => $segmentContent,
  2223. ];
  2224. }
  2225. $episodes[] = [
  2226. 'episode_number' => $episodeNumber,
  2227. 'title' => $episodeName,
  2228. 'content' => $episodeContent,
  2229. 'segments' => $segments,
  2230. ];
  2231. }
  2232. if (empty($episodes) && !empty($parts['acts'])) {
  2233. $singleSegments = [];
  2234. foreach ($parts['acts'] as $act) {
  2235. $actSegments = getProp($act, 'segments', []);
  2236. if (!is_array($actSegments)) {
  2237. continue;
  2238. }
  2239. foreach ($actSegments as $segment) {
  2240. $singleSegments[] = [
  2241. 'segment_number' => getProp($segment, 'segment_number'),
  2242. 'segment_content' => getProp($segment, 'segment_content'),
  2243. ];
  2244. }
  2245. }
  2246. $episodeTitle = $parts['episode_title'] ?: ('第1集:' . ($parts['script_name'] ?? '未命名'));
  2247. $episodeName = preg_replace('/^第\d+集[::]\s*/u', '', $episodeTitle);
  2248. $episodes[] = [
  2249. 'episode_number' => 1,
  2250. 'title' => $episodeName,
  2251. 'content' => $singleStoryboard,
  2252. 'segments' => $singleSegments,
  2253. ];
  2254. }
  2255. $parts['episodes'] = $episodes;
  2256. return $parts;
  2257. }
  2258. function handleEpisodeContent($originalContent) {
  2259. if (!$originalContent) return [];
  2260. // $originalContent = '好的,作为您的专业文档分析助手及资深编剧,我将根据您提供的本章剧情内容、历史对话记录以及本次修改要求,严格按照您的要求完成第1集分镜剧本的创作。
  2261. // 第1集:重生与隐形的拯救
  2262. // ###故事梗概
  2263. // 内容梗概:重生在悲剧发生前一个月的徐清遥,提前与许家真千金交换了身份,回到了原生家庭徐家。在悲剧发生日(双十一),她听从徐母安排,放学后没有回家,而是选择在网吧包夜以避开危险。在网吧煎熬一夜后,次日清晨因疲惫过马路时险些遭遇车祸,千钧一发之际被一股无形的力量拉开。她意识到是上辈子那个为她收尸、为她复仇、并付出代价换她重生的“隐形守护者”再次出手相救。劫后余生的徐清遥在人群中徒劳地寻找他的踪迹,心中涌起复杂的情感。
  2264. // ###美术风格
  2265. // 基础画风风格词:韩漫二次元
  2266. // 视觉风格描述:采用细腻的韩漫厚涂风格,以冷灰色调和低饱和度为主,营造重生后的疏离感与命运未卜的紧张氛围。回忆片段采用高对比度的黑白或单色处理,强调过往的惨痛与不真实感。现实场景中,网吧的昏暗浑浊与清晨街道的清冷形成对比,车祸瞬间运用动态模糊和慢镜头特写,突出“隐形守护者”介入时的超现实感与宿命牵引。人物面部表情刻画精细,着重展现徐清遥眼神中的沧桑、坚韧以及对无形守护的复杂情感。
  2267. // ###主体列表
  2268. // 徐清遥-重生后(17岁): 身高约162cm,面容清秀但带着长期营养不良的苍白与疲惫。眼神坚韧,偶尔流露出超越年龄的沧桑。常穿着黄色T恤加蓝色牛仔裤。回到徐家后,手上常有干活留下的细微伤痕。{{甜心小美}}
  2269. // “空气人”(隐形守护者): 存在但不可见。通过间接信息推测为同龄或稍长男性,身材应较为高大(能提起女主角躲避车祸),经济拮据(住阁楼),爱干净(阁楼有皂角味),可能为学生(桌上有高中课本试卷)。{{阳光青年}}
  2270. // 卞大伟(20岁): 身高约175cm,体型偏瘦但力气不小。眼神阴鸷,面色因长期混迹网吧而泛着油光与不健康的青白。常穿廉价的紧身T恤和牛仔裤,身上有烟味和厨房的油腻味。{{广州德哥}}
  2271. // 徐母(45岁左右): 身材干瘦,颧骨高突,眉眼间总带着不耐烦与算计。穿着廉价的花衬衫和黑裤子,嗓门洪亮,动作粗鲁。{{邻居阿姨}}
  2272. // 徐父(48岁左右): 背微驼,面容愁苦,眼神躲闪。穿着沾有污渍的工装,身上有淡淡的酒气,整体气质怯懦。{{广州德哥}}
  2273. // 老奶奶(70岁左右): 头发花白,身形佝偻,拄着拐杖,面容慈祥但带着老年人特有的迟缓与关切。{{邻居阿姨}}
  2274. // 许芸-大学生/成年: 20-30岁,气质逐渐成熟温婉,衣着简约大方,笑容增多,眼中有了被爱滋养的光彩。{{爽快思思}}
  2275. // ###场景列表
  2276. // 徐家馄饨店后间: 通往后院的狭窄房间,兼做厨房和部分家庭成员活动区。墙面油腻发黄,堆满杂物。中间一张旧方桌,几张塑料凳。灯光昏暗,空气中弥漫着馄饨馅和煤球炉的味道。
  2277. // 网吧包夜区: 环境昏暗,空气混浊,弥漫着烟味与泡面味。一排排电脑屏幕闪着幽光,键盘鼠标声杂乱。徐清遥所在的角落相对安静。
  2278. // 车祸路口: 城市普通街道十字路口,人行道狭窄,车流匆忙。背景是略显老旧的居民楼商铺。
  2279. // 学校门口: 普通高中校门,上下学时间人流密集。有家长、学生和小贩,背景是教学楼。
  2280. // ###分镜剧本
  2281. // ##第1幕:徐家老旧厨房 白天 室内
  2282. // 分镜1
  2283. // 画面描述:许芸-校服装站在厨房门口,书包背在肩上,徐母手持扫帚,正对着许芸的背部猛烈挥打。卞大伟站在锅炉后方,双手抱胸,阴沉的目光落在许芸身上。
  2284. // 场景:徐家老旧厨房
  2285. // 构图设计:全景,低角度仰视
  2286. // 运镜调度:手持拍摄
  2287. // 配音角色:旁白
  2288. // 出镜角色:许芸-校服装、徐母
  2289. // 台词内容:我一到家,徐母的扫帚就落在我身上,嘴里一个劲地骂我,
  2290. // 画面类型:普通画面
  2291. // 尾帧描述:手持拍摄微晃镜头,中年妇女猛力挥动扫帚击打少女背部,少女身体因冲击微颤,阴影中的肥胖男子目光冷冷跟随。
  2292. // 分镜2
  2293. // 画面描述:徐母面容扭曲,嘴唇快速开合,扫帚挥舞在空中。
  2294. // 场景:徐家老旧厨房
  2295. // 构图设计:近景,徐母面部特写
  2296. // 运镜调度:固定镜头
  2297. // 配音角色:徐母
  2298. // 出镜角色:徐母
  2299. // 台词内容:问我夜不归宿死哪儿去了。
  2300. // 画面类型:对口型
  2301. // 尾帧描述:固定镜头,中年妇女面部肌肉剧烈抖动,嘴唇快速开合进行辱骂,背景中的扫帚在空中上下挥动。
  2302. // ##第2幕:徐家简陋客厅 黄昏 室内
  2303. // 分镜3
  2304. // 画面描述:客厅窗户透出昏黄的夕阳余晖,空荡荡的客厅中央只有几把木凳和一张方桌,显得异常安静。
  2305. // 场景:徐家简陋客厅
  2306. // 构图设计:全景,景深虚化
  2307. // 运镜调度:固定镜头
  2308. // 配音角色:旁白
  2309. // 出镜角色:无
  2310. // 台词内容:可我没想到,半个月后,徐母居然再一次外出,
  2311. // 画面类型:普通画面
  2312. // 尾帧描述:固定镜头。昏黄的夕阳余晖在地面上缓慢移动,空气中的浮尘在光束中跳动,室内维持着死寂的安静。
  2313. // 分镜4
  2314. // 画面描述:一张泛黄的火车票特写,上面显示着隔壁市的站名和日期。
  2315. // 场景:徐家简陋客厅
  2316. // 构图设计:特写,火车票
  2317. // 运镜调度:固定镜头
  2318. // 配音角色:旁白
  2319. // 出镜角色:无
  2320. // 台词内容:这次是去隔壁市看望她生病的好姐妹。
  2321. // 画面类型:普通画面
  2322. // 尾帧描述:固定镜头。光影在火车票表面微微晃动,强调其陈旧的纸张纹理与关键的行程信息。
  2323. // \n\n';
  2324. // 解析剧集内容
  2325. $result = [
  2326. 'episode_title' => '',
  2327. 'intro' => '',
  2328. 'art_style' => '',
  2329. 'roles' => [],
  2330. 'scenes' => [],
  2331. 'acts' => []
  2332. ];
  2333. // 提取剧集标题 - 匹配"第xx集:标题"或"第xx集 标题"格式,排除###标记
  2334. if (preg_match('/第(\d+)集[::\s]+([^#\n]+?)(?=\s*###|\s*$|\n)/u', $originalContent, $titleMatch)) {
  2335. $result['episode_title'] = '第' . $titleMatch[1] . '集:' . trim($titleMatch[2]);
  2336. }
  2337. // 提取故事梗概 - 兼容多种格式:###故事梗概、### 故事梗概、### 故事梗概
  2338. if (preg_match('/###\s*故事梗概\s*\n(.*?)(?=\n\s*###[^#]|\z)/s', $originalContent, $summaryMatch)) {
  2339. $result['intro'] = trim($summaryMatch[1]);
  2340. }
  2341. // 提取美术风格 - 兼容多种格式
  2342. if (preg_match('/###\s*美术风格\s*\n(.*?)(?=\n\s*###[^#]|\z)/s', $originalContent, $styleMatch)) {
  2343. $result['art_style'] = trim($styleMatch[1]);
  2344. }
  2345. // 提取主体列表 - 兼容多种格式
  2346. if (preg_match('/###\s*主体列表\s*\n(.*?)(?=\n\s*###[^#]|\z)/s', $originalContent, $charactersMatch)) {
  2347. $charactersText = trim($charactersMatch[1]);
  2348. $characterLines = explode("\n", $charactersText);
  2349. foreach ($characterLines as $line) {
  2350. $line = trim($line);
  2351. if (empty($line)) continue;
  2352. // 兼容中文冒号:和英文冒号:,同时提取音色信息
  2353. if (preg_match('/^([^::]+)[::](.+)$/u', $line, $charMatch)) {
  2354. $role = trim($charMatch[1]);
  2355. $description = trim($charMatch[2]);
  2356. $timbreName = null;
  2357. // 检查描述末尾是否有{{音色名}}格式
  2358. if (preg_match('/^(.*?)\{\{([^}]+)\}\}\s*$/u', $description, $timbreMatch)) {
  2359. $description = trim($timbreMatch[1]);
  2360. $timbreName = trim($timbreMatch[2]);
  2361. }
  2362. $roleData = [
  2363. 'role' => $role,
  2364. 'description' => $description
  2365. ];
  2366. // 如果有音色信息,添加到数组中
  2367. if ($timbreName) {
  2368. $timbre = DB::table('mp_timbres')->where('is_enabled', 1)->where('timbre_name', 'like', "%{$timbreName}%")->orderBy('id')->select('timbre_type', 'audio_url')->first();
  2369. if ($timbre) {
  2370. $roleData['voice_name'] = $timbreName;
  2371. $roleData['voice_type'] = getProp($timbre, 'timbre_type');
  2372. $roleData['voice_audio_url'] = getProp($timbre, 'audio_url');;
  2373. }
  2374. }
  2375. $result['roles'][] = $roleData;
  2376. }
  2377. }
  2378. // 加入旁白角色
  2379. $result['roles'][] = [
  2380. 'role' => '旁白',
  2381. 'description' => '负责叙述剧情、补充说明和情感渲染的非视觉角色。',
  2382. 'voice_name' => '旁白',
  2383. 'voice_type' => 'zh_male_linjiananhai_moon_bigtts',
  2384. 'voice_audio_url' => 'https://zw-audiobook.tos-cn-beijing.volces.com/demonstrate/zh_male_linjiananhai_moon_bigtts.wav'
  2385. ];
  2386. }
  2387. // 提取场景列表 - 兼容多种格式
  2388. if (preg_match('/###\s*场景列表\s*\n(.*?)(?=\n\s*###[^#]|\z)/s', $originalContent, $scenesMatch)) {
  2389. $scenesText = trim($scenesMatch[1]);
  2390. $sceneLines = explode("\n", $scenesText);
  2391. foreach ($sceneLines as $line) {
  2392. $line = trim($line);
  2393. if (empty($line)) continue;
  2394. // 兼容中文冒号:和英文冒号:
  2395. if (preg_match('/^([^::]+)[::](.+)$/u', $line, $sceneMatch)) {
  2396. $result['scenes'][] = [
  2397. 'scene' => trim($sceneMatch[1]),
  2398. 'description' => trim($sceneMatch[2])
  2399. ];
  2400. }
  2401. }
  2402. }
  2403. // 提取分镜剧本 - 兼容多种格式
  2404. if (preg_match('/###\s*分镜剧本\s*\n(.*?)(?=\n\s*###[^#]|\z)/s', $originalContent, $storyboardMatch)) {
  2405. $storyboardText = trim($storyboardMatch[1]);
  2406. // 按幕分割 - 修复第1幕识别和乱码问题
  2407. $acts = [];
  2408. // 先在开头添加换行符,确保第1幕也能被正确分割
  2409. $normalizedText = "\n" . $storyboardText;
  2410. $parts = preg_split('/\n\s*##/', $normalizedText);
  2411. foreach ($parts as $part) {
  2412. $part = trim($part);
  2413. if (empty($part)) continue;
  2414. // 如果不是以"第"开头,跳过
  2415. if (!preg_match('/^第\d+幕/', $part)) {
  2416. continue;
  2417. }
  2418. // 分离标题和内容
  2419. $lines = explode("\n", $part, 2);
  2420. $actTitle = trim($lines[0]);
  2421. $actContent = isset($lines[1]) ? trim($lines[1]) : '';
  2422. // 解析幕标题,提取序号和详细信息 - 修复乱码问题
  2423. if (preg_match('/^第(\d+)幕[::]?\s*(.*)$/u', $actTitle, $actTitleMatch)) {
  2424. $actNumber = intval($actTitleMatch[1]);
  2425. $actDetails = trim($actTitleMatch[2]);
  2426. // 如果详细信息为空或者就是冒号,使用完整标题
  2427. if (empty($actDetails) || $actDetails === ':' || $actDetails === ':') {
  2428. $actDetails = $actTitle;
  2429. }
  2430. } else {
  2431. $actNumber = count($acts) + 1;
  2432. $actDetails = $actTitle;
  2433. }
  2434. // 解析该幕下的分镜
  2435. $segments = [];
  2436. $segmentPattern = '/分镜(\d+)\s*\n(.*?)(?=\n+\s*分镜\d+|\z)/s';
  2437. preg_match_all($segmentPattern, $actContent, $segmentMatches, PREG_SET_ORDER);
  2438. foreach ($segmentMatches as $segmentMatch) {
  2439. $segmentNumber = intval($segmentMatch[1]);
  2440. $segmentContent = trim($segmentMatch[2]);
  2441. // 解析分镜详细信息
  2442. $segmentData = [
  2443. 'segment_id' => date('YmdHis') . mt_rand(1000, 9999) . str_pad($segmentNumber, 3, "0", STR_PAD_LEFT), // 生成唯一ID(后续可在生成任务里查看历史版本)
  2444. 'segment_number' => $segmentNumber,
  2445. 'segment_content' => $segmentContent,
  2446. 'description' => '',
  2447. 'composition' => '',
  2448. 'camera_movement' => '',
  2449. 'voice_actor' => '',
  2450. 'dialogue' => '',
  2451. 'frame_type' => '',
  2452. 'scene' => '', // 新增:场景
  2453. 'characters' => '', // 新增:出镜角色
  2454. 'tail_frame' => '', // 新增:尾帧描述
  2455. // 新增字段
  2456. 'emotion' => '中性', // 新增: 情感
  2457. 'gender' => '0', // 新增: 性别(0未知,1男,2女)
  2458. 'speed_ratio' => 0, // 新增: 语速
  2459. 'loudness_ratio' => 0, // 新增: 音量
  2460. 'emotion_scale' => 4, // 新增: 语调
  2461. 'pitch' => 0, // 新增: 音调
  2462. ];
  2463. // 提取各个字段 - 兼容中文冒号和英文冒号,支持多种表达方式
  2464. if (preg_match('/(?:画面描述|镜头描述|场景描述)[::]\s*([^\n]+)/u', $segmentContent, $descMatch)) {
  2465. $segmentData['description'] = trim($descMatch[1]);
  2466. }
  2467. if (preg_match('/(?:构图设计|构图|镜头构图)[::]\s*([^\n]+)/u', $segmentContent, $compMatch)) {
  2468. $segmentData['composition'] = trim($compMatch[1]);
  2469. }
  2470. if (preg_match('/(?:运镜调度|运镜|镜头运动|摄影机运动)[::]\s*([^\n]+)/u', $segmentContent, $cameraMatch)) {
  2471. $segmentData['camera_movement'] = trim($cameraMatch[1]);
  2472. }
  2473. if (preg_match('/(?:配音角色|配音|角色|声优)[::]\s*([^\n]+)/u', $segmentContent, $voiceMatch)) {
  2474. $segmentData['voice_actor'] = trim($voiceMatch[1]);
  2475. }
  2476. if (preg_match('/(?:台词内容|台词|对白|对话)[::]\s*([^\n]+)/u', $segmentContent, $dialogueMatch)) {
  2477. $segmentData['dialogue'] = trim($dialogueMatch[1]);
  2478. }
  2479. if (preg_match('/(?:画面类型|镜头类型|类型)[::]\s*([^\n]+)/u', $segmentContent, $frameMatch)) {
  2480. $segmentData['frame_type'] = trim($frameMatch[1]);
  2481. }
  2482. // 新增:场景字段
  2483. if (preg_match('/(?:场景|拍摄场景|背景场景|环境)[::]\s*([^\n]+)/u', $segmentContent, $sceneMatch)) {
  2484. $segmentData['scene'] = trim($sceneMatch[1]);
  2485. }
  2486. // 新增:出镜角色字段
  2487. if (preg_match('/(?:出镜角色|角色出镜|登场角色|人物)[::]\s*([^\n]+)/u', $segmentContent, $charactersMatch)) {
  2488. $segmentData['characters'] = trim($charactersMatch[1]);
  2489. }
  2490. // 新增:尾帧描述字段
  2491. if (preg_match('/(?:尾帧描述|尾帧|结束帧|最后一帧|结尾画面|结束画面)[::]\s*([^\n]+)/u', $segmentContent, $tailFrameMatch)) {
  2492. $segmentData['tail_frame'] = trim($tailFrameMatch[1]);
  2493. }
  2494. $replaceEmptyArr = [];
  2495. // 新增:情感字段
  2496. if (preg_match('/(?:情感|情绪|感情)[::]\s*([^\n]+)/u', $segmentContent, $emotionMatch)) {
  2497. $replaceEmptyArr[] = trim($emotionMatch[0]);
  2498. $segmentData['emotion'] = trim($emotionMatch[1]);
  2499. }
  2500. // 新增:性别字段
  2501. if (preg_match('/(?:性别)[::]\s*([^\n]+)/u', $segmentContent, $genderMatch)) {
  2502. $replaceEmptyArr[] = trim($genderMatch[0]);
  2503. $genderStr = trim($genderMatch[1]);
  2504. if (strpos($genderStr, '男') !== false || $genderStr === '1') {
  2505. $segmentData['gender'] = '1';
  2506. } elseif (strpos($genderStr, '女') !== false || $genderStr === '2') {
  2507. $segmentData['gender'] = '2';
  2508. } else {
  2509. $segmentData['gender'] = '0';
  2510. }
  2511. }
  2512. // 新增:语速字段
  2513. if (preg_match('/(?:语速|说话速度)[::]\s*([-+]?[0-9]*\.?[0-9]+)/u', $segmentContent, $speedMatch)) {
  2514. $replaceEmptyArr[] = trim($speedMatch[0]);
  2515. $segmentData['speed_ratio'] = (float)trim($speedMatch[1]);
  2516. }
  2517. // 新增:音量字段
  2518. if (preg_match('/(?:音量|声音大小)[::]\s*([-+]?[0-9]*\.?[0-9]+)/u', $segmentContent, $loudnessMatch)) {
  2519. $replaceEmptyArr[] = trim($loudnessMatch[0]);
  2520. $segmentData['loudness_ratio'] = (float)trim($loudnessMatch[1]);
  2521. }
  2522. // 新增:情感强度字段
  2523. if (preg_match('/(?:情感强度|情绪强度)[::]\s*([0-9]+)/u', $segmentContent, $scaleMatch)) {
  2524. $replaceEmptyArr[] = trim($scaleMatch[0]);
  2525. $segmentData['emotion_scale'] = (int)trim($scaleMatch[1]);
  2526. }
  2527. // 新增:音调字段
  2528. if (preg_match('/(?:音调|音高)[::]\s*([-+]?[0-9]+)/u', $segmentContent, $pitchMatch)) {
  2529. $replaceEmptyArr[] = trim($pitchMatch[0]);
  2530. $segmentData['pitch'] = (int)trim($pitchMatch[1]);
  2531. }
  2532. $segmentData['segment_content'] = str_replace($replaceEmptyArr, '', $segmentContent);
  2533. $segments[] = $segmentData;
  2534. }
  2535. $acts[] = [
  2536. 'act_number' => $actNumber,
  2537. 'act_title' => $actTitle,
  2538. 'act_details' => $actDetails,
  2539. 'segments' => $segments
  2540. ];
  2541. }
  2542. $result['acts'] = $acts;
  2543. }
  2544. return $result;
  2545. }
  2546. /**
  2547. * 远程图片压缩(尽可能保持原图 fidelity,压缩到不超过 maxBytes 字节)
  2548. * 采用有损/无损组合策略:保留原始格式尽量不失真,若需要则通过尺寸缩放和质量调节来降低体积。
  2549. *
  2550. * @param string $url 远程图片 URL
  2551. * @param int $maxBytes 最大字节数,默认 3MB
  2552. * @param string|null $aspectRatio 目标长宽比,如 "16:9"、"4:3"、"1:1" 等,null 则保持原图比例
  2553. * @return string|null 返回压缩后的图片二进制数据,失败时返回 null
  2554. */
  2555. function compressRemoteImageUrlToSize(string $url, int $maxBytes = 3 * 1024 * 1024, ?string $aspectRatio = null): ?string
  2556. {
  2557. // 1) 下载图片数据(使用 Guzzle 以避免 allow_url_fopen 依赖)
  2558. try {
  2559. $client = new Client(['timeout' => 30]);
  2560. $response = $client->get($url, ['stream' => true]);
  2561. if ($response->getStatusCode() !== 200) {
  2562. return null;
  2563. }
  2564. $data = $response->getBody()->getContents();
  2565. } catch (\Exception $e) {
  2566. return null;
  2567. }
  2568. if (!$data) return null;
  2569. // 2) 识别图片类型
  2570. $imgInfo = @getimagesizefromstring($data);
  2571. $mime = $imgInfo['mime'] ?? '';
  2572. if ($mime == 'image/jpeg') return null; // JPEG 不做有损压缩
  2573. // 3) 载入图像对象
  2574. $srcImg = @imagecreatefromstring($data);
  2575. if (!$srcImg) {
  2576. return null;
  2577. }
  2578. $origW = imagesx($srcImg);
  2579. $origH = imagesy($srcImg);
  2580. // 3.2) 计算目标尺寸(根据长宽比调整)
  2581. $targetW = $origW;
  2582. $targetH = $origH;
  2583. if ($aspectRatio !== null) {
  2584. // 解析长宽比,如 "16:9" -> [16, 9]
  2585. $validRatios = ["16:9", "4:3", "1:1", "3:4", "9:16", "21:9"];
  2586. if (in_array($aspectRatio, $validRatios)) {
  2587. list($ratioW, $ratioH) = explode(':', $aspectRatio);
  2588. $ratioW = (float)$ratioW;
  2589. $ratioH = (float)$ratioH;
  2590. $targetRatio = $ratioW / $ratioH;
  2591. $currentRatio = $origW / $origH;
  2592. // 根据目标比例裁剪图片
  2593. if ($currentRatio > $targetRatio) {
  2594. // 原图更宽,需要裁剪宽度
  2595. $targetW = (int)round($origH * $targetRatio);
  2596. $targetH = $origH;
  2597. } else {
  2598. // 原图更高,需要裁剪高度
  2599. $targetW = $origW;
  2600. $targetH = (int)round($origW / $targetRatio);
  2601. }
  2602. }
  2603. }
  2604. // 如果需要裁剪,创建裁剪后的图像
  2605. if ($targetW !== $origW || $targetH !== $origH) {
  2606. $croppedImg = imagecreatetruecolor($targetW, $targetH);
  2607. imagealphablending($croppedImg, false);
  2608. imagesavealpha($croppedImg, true);
  2609. // 处理透明背景
  2610. if (in_array(strtolower($mime), ['image/png','image/webp'])) {
  2611. $transparent = imagecolorallocatealpha($croppedImg, 0, 0, 0, 127);
  2612. imagefill($croppedImg, 0, 0, $transparent);
  2613. }
  2614. // 居中裁剪
  2615. $srcX = (int)(($origW - $targetW) / 2);
  2616. $srcY = (int)(($origH - $targetH) / 2);
  2617. imagecopy($croppedImg, $srcImg, 0, 0, $srcX, $srcY, $targetW, $targetH);
  2618. // 替换原图
  2619. safeDestroyImage($srcImg);
  2620. $srcImg = $croppedImg;
  2621. $origW = $targetW;
  2622. $origH = $targetH;
  2623. }
  2624. // 3.1) 内部渲染成不同格式的字符串
  2625. $render = function($srcRes, string $mimeType, int $quality) {
  2626. ob_start();
  2627. switch (strtolower($mimeType)) {
  2628. case 'image/jpeg':
  2629. case 'image/jpg':
  2630. case 'image/pjpeg':
  2631. // 输出 JPEG
  2632. imagejpeg($srcRes, null, $quality);
  2633. break;
  2634. case 'image/png':
  2635. // 将 quality 映射到 PNG 的 compression level (0-9)
  2636. $level = (int)round((100 - $quality) / 11.11);
  2637. if ($level < 0) $level = 0;
  2638. if ($level > 9) $level = 9;
  2639. imagepng($srcRes, null, $level);
  2640. break;
  2641. case 'image/webp':
  2642. if (function_exists('imagewebp')) {
  2643. imagewebp($srcRes, null, $quality);
  2644. } else {
  2645. imagejpeg($srcRes, null, $quality);
  2646. }
  2647. break;
  2648. default:
  2649. imagejpeg($srcRes, null, $quality);
  2650. break;
  2651. }
  2652. $out = ob_get_contents();
  2653. ob_end_clean();
  2654. return $out;
  2655. };
  2656. // 4) 尝试策略:尽量保留原图尺寸,逐步降维/降质量,直到 <= maxBytes
  2657. $tryList = [];
  2658. // 原始尺寸,尽量保留
  2659. $tryList[] = ['scale'=>1.0, 'mime'=>$mime, 'quality'=>100];
  2660. // 逐步缩小尺寸
  2661. for ($s = 0.9; $s >= 0.2; $s -= 0.1) {
  2662. $tryList[] = ['scale'=>$s, 'mime'=>$mime, 'quality'=>90];
  2663. }
  2664. // 一系列质量等级(用于 JPEG/WebP)
  2665. $qualityLevels = [95, 90, 85, 75, 60, 50, 40, 30];
  2666. foreach ($qualityLevels as $q) {
  2667. $tryList[] = ['scale'=>1.0, 'mime'=>$mime, 'quality'=>$q];
  2668. }
  2669. $bestData = null;
  2670. foreach ($tryList as $cand) {
  2671. $scale = isset($cand['scale']) ? (float)$cand['scale'] : 1.0;
  2672. $mimeT = $cand['mime'] ?? $mime;
  2673. $quality = isset($cand['quality']) ? (int)$cand['quality'] : 90;
  2674. $w = (int)round($origW * $scale);
  2675. $h = (int)round($origH * $scale);
  2676. $src = $srcImg;
  2677. $tempImg = null;
  2678. if ($scale < 1.0) {
  2679. $tempImg = imagecreatetruecolor($w, $h);
  2680. // 处理透明通道
  2681. imagealphablending($tempImg, false);
  2682. imagesavealpha($tempImg, true);
  2683. if (in_array(strtolower($mime), ['image/png','image/webp'])) {
  2684. $transparent = imagecolorallocatealpha($tempImg, 0, 0, 0, 127);
  2685. imagefill($tempImg, 0, 0, $transparent);
  2686. }
  2687. // 确保 $srcImg 有效
  2688. if (!$srcImg || (!is_resource($srcImg) && !($srcImg instanceof \GdImage))) {
  2689. safeDestroyImage($tempImg);
  2690. continue;
  2691. }
  2692. imagecopyresampled($tempImg, $srcImg, 0, 0, 0, 0, $w, $h, $origW, $origH);
  2693. $src = $tempImg;
  2694. }
  2695. $imageBytes = $render($src, $mimeT, $quality);
  2696. if ($tempImg !== null) {
  2697. safeDestroyImage($tempImg);
  2698. }
  2699. if ($imageBytes !== false && strlen($imageBytes) <= $maxBytes) {
  2700. $bestData = $imageBytes;
  2701. break;
  2702. }
  2703. }
  2704. // 5) 回退策略:若仍未达到要求,尝试更大幅度降解到一个合理的小尺寸 JPEG
  2705. if ($bestData === null) {
  2706. $tmpW = max(1, (int)round($origW * 0.5));
  2707. $tmpH = max(1, (int)round($origH * 0.5));
  2708. $tmpImg = imagecreatetruecolor($tmpW, $tmpH);
  2709. imagealphablending($tmpImg, false);
  2710. imagesavealpha($tmpImg, true);
  2711. if (in_array(strtolower($mime), ['image/png','image/webp'])) {
  2712. $transparent = imagecolorallocatealpha($tmpImg, 0, 0, 0, 127);
  2713. imagefill($tmpImg, 0, 0, $transparent);
  2714. }
  2715. imagecopyresampled($tmpImg, $srcImg, 0, 0, 0, 0, $tmpW, $tmpH, $origW, $origH);
  2716. ob_start();
  2717. if (in_array(strtolower($mime), ['image/jpeg','image/jpg','image/pjpeg'])) {
  2718. imagejpeg($tmpImg, null, 75);
  2719. } elseif (strtolower($mime) === 'image/png') {
  2720. imagepng($tmpImg, null, 6);
  2721. } elseif (function_exists('imagewebp')) {
  2722. imagewebp($tmpImg, null, 75);
  2723. } else {
  2724. imagejpeg($tmpImg, null, 75);
  2725. }
  2726. $tmpBytes = ob_get_contents();
  2727. ob_end_clean();
  2728. safeDestroyImage($tmpImg);
  2729. if ($tmpBytes !== '' && strlen($tmpBytes) <= $maxBytes) {
  2730. $bestData = $tmpBytes;
  2731. }
  2732. }
  2733. // 6) 清理
  2734. safeDestroyImage($srcImg);
  2735. return $bestData;
  2736. }
  2737. /**
  2738. * 安全销毁 GD 图像资源(兼容 PHP7.4+ 的资源管理)
  2739. * 通过引用传递并在销毁后置空变量,避免未定义变量的问题
  2740. *
  2741. * @param resource|\GdImage|null &$img GD 图像资源或对象(PHP7.4 为 resource,PHP8.0+ 为 GdImage)
  2742. */
  2743. function safeDestroyImage(&$img)
  2744. {
  2745. if (is_resource($img) || (is_object($img) && $img instanceof \GdImage)) {
  2746. @imagedestroy($img);
  2747. }
  2748. $img = null;
  2749. }
  2750. // /**
  2751. // * 使用 Imagick 将远程图片压缩至不超过 maxBytes(默认3MB)内
  2752. // * 目标:在尽量保留原始格式和质量的前提下进行压缩,避免内存暴涨
  2753. // * 依赖:Imagick 扩展与 ImageMagick 库,下载使用 Guzzle 获取图片 blob
  2754. // *
  2755. // * @param string $url
  2756. // * @param int $maxBytes 最大字节数,默认 3*1024*1024
  2757. // * @return string|null 经过压缩后的图片 blob,失败返回 null
  2758. // */
  2759. // function compressRemoteImageUrlToSizeImagick(string $url, int $maxBytes = 3 * 1024 * 1024): ?string
  2760. // {
  2761. // // 兼容性检查
  2762. // if (!extension_loaded('imagick') || !class_exists('Imagick')) {
  2763. // return null;
  2764. // }
  2765. // try {
  2766. // $client = new Client(['timeout' => 30]);
  2767. // // 使用流式下载,将数据写入临时文件,避免一次性加载到内存
  2768. // $response = $client->get($url, ['stream' => true]);
  2769. // if ($response->getStatusCode() !== 200) {
  2770. // return null;
  2771. // }
  2772. // $body = $response->getBody();
  2773. // $tmp = tmpfile();
  2774. // if ($tmp === false) {
  2775. // return null;
  2776. // }
  2777. // stream_copy_to_stream($body, $tmp);
  2778. // rewind($tmp);
  2779. // $imagick = new Imagick();
  2780. // $imagick->readImageFile($tmp);
  2781. // $origW = $imagick->getImageWidth();
  2782. // $origH = $imagick->getImageHeight();
  2783. // $format = strtolower($imagick->getImageFormat());
  2784. // // 尝试序列:尺寸缩放 + 质量调节
  2785. // $scales = [1.0, 0.9, 0.8, 0.6, 0.4, 0.25];
  2786. // $qualities = [95, 90, 85, 75, 60, 40, 30, 20];
  2787. // $bestBlob = null;
  2788. // foreach ($scales as $scale) {
  2789. // foreach ($qualities as $q) {
  2790. // $clone = clone $imagick;
  2791. // if ($scale < 1.0) {
  2792. // $w = (int)round($origW * $scale);
  2793. // $h = (int)round($origH * $scale);
  2794. // if ($w < 1 || $h < 1) {
  2795. // $clone->destroy();
  2796. // continue;
  2797. // }
  2798. // $clone->resizeImage($w, $h, Imagick::FILTER_LANCZOS, 1);
  2799. // }
  2800. // $clone->setImageFormat($format);
  2801. // $clone->setImageCompressionQuality($q);
  2802. // $blob = $clone->getImageBlob();
  2803. // $clone->destroy();
  2804. // if ($blob !== false && strlen($blob) <= $maxBytes) {
  2805. // $bestBlob = $blob;
  2806. // break 2;
  2807. // }
  2808. // }
  2809. // }
  2810. // if ($bestBlob !== null) {
  2811. // $imagick->destroy();
  2812. // fclose($tmp);
  2813. // return $bestBlob;
  2814. // }
  2815. // // 回退:尝试强制输出为当前格式的一个中等质量版本
  2816. // $fallback = clone $imagick;
  2817. // $fallback->setImageFormat($format);
  2818. // $fallback->setImageCompressionQuality(75);
  2819. // $blob = $fallback->getImageBlob();
  2820. // $fallback->destroy();
  2821. // $imagick->destroy();
  2822. // fclose($tmp);
  2823. // if ($blob !== false && strlen($blob) <= $maxBytes) {
  2824. // return $blob;
  2825. // }
  2826. // } catch (\Exception $e) {
  2827. // return null;
  2828. // }
  2829. // return null;
  2830. // }
  2831. /**
  2832. * 将主体列表文本拆分为主体数组
  2833. *
  2834. * @param string $rolesText 主体列表文本内容
  2835. * @return array 主体名称数组
  2836. */
  2837. function parseRolesFromText(string $rolesText): array
  2838. {
  2839. if (empty($rolesText)) {
  2840. return [];
  2841. }
  2842. $roles = [];
  2843. // 按行分割文本
  2844. $lines = explode("\n", $rolesText);
  2845. foreach ($lines as $line) {
  2846. $line = trim($line);
  2847. if (empty($line)) {
  2848. continue;
  2849. }
  2850. if (preg_match('/^([^::]+)[::](.+)$/u', $line, $charMatch)) {
  2851. $role = trim($charMatch[1]);
  2852. $description = trim($charMatch[2]);
  2853. $timbreName = null;
  2854. if (preg_match('/^(.*?)\{\{([^}]+)\}\}\s*$/u', $description, $timbreMatch)) {
  2855. $description = trim($timbreMatch[1]);
  2856. $timbreName = trim($timbreMatch[2]);
  2857. }
  2858. $roleData = [
  2859. 'role' => $role,
  2860. 'description' => $description,
  2861. ];
  2862. if ($timbreName) {
  2863. $timbre = DB::table('mp_timbres')
  2864. ->where('is_enabled', 1)
  2865. ->where('timbre_name', 'like', "%{$timbreName}%")
  2866. ->orderBy('id')
  2867. ->select('timbre_type', 'audio_url')
  2868. ->first();
  2869. if ($timbre) {
  2870. $roleData['voice_name'] = $timbreName;
  2871. $roleData['voice_type'] = getProp($timbre, 'timbre_type');
  2872. $roleData['voice_audio_url'] = getProp($timbre, 'audio_url');
  2873. }
  2874. }
  2875. $roles[] = $roleData;
  2876. }
  2877. }
  2878. $hasNarrator = false;
  2879. foreach ($roles as $role) {
  2880. if (getProp($role, 'role') === '旁白') {
  2881. $hasNarrator = true;
  2882. break;
  2883. }
  2884. }
  2885. if (!$hasNarrator) {
  2886. $roles[] = [
  2887. 'role' => '旁白',
  2888. 'description' => '负责叙述剧情、补充说明和情感渲染的非视觉角色。',
  2889. 'voice_name' => '旁白',
  2890. 'voice_type' => 'zh_male_linjiananhai_moon_bigtts',
  2891. 'voice_audio_url' => 'https://zw-audiobook.tos-cn-beijing.volces.com/demonstrate/zh_male_linjiananhai_moon_bigtts.wav'
  2892. ];
  2893. }
  2894. return $roles;
  2895. }
  2896. /**
  2897. * 将场景列表文本拆分为主体数组
  2898. *
  2899. * @param string $rolesText 场景列表列表文本内容
  2900. * @return array 场景名称数组
  2901. */
  2902. function parseScenesFromText(string $scenesText): array
  2903. {
  2904. if (empty($scenesText)) {
  2905. return [];
  2906. }
  2907. $scenes = [];
  2908. // 按行分割文本
  2909. $lines = explode("\n", $scenesText);
  2910. foreach ($lines as $line) {
  2911. $line = trim($line);
  2912. if (empty($line)) {
  2913. continue;
  2914. }
  2915. $line = str_replace(':', ':', $line);
  2916. if (strstr($line, ':')) {
  2917. $line_arr = explode(':', $line);
  2918. if (count($line_arr) == 2) {
  2919. $scenes[] = [
  2920. 'scene' => $line_arr[0],
  2921. 'description' => $line_arr[1],
  2922. ];
  2923. }
  2924. }
  2925. }
  2926. return $scenes;
  2927. }