index.blade.php 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. @extends('user.layouts')
  2. @section('css')
  3. <link href="/assets/global/vendor/bootstrap-select/bootstrap-select.min.css" rel="stylesheet">
  4. <link href="/assets/global/fonts/font-awesome/css/all.min.css" rel="stylesheet">
  5. <link href="/assets/global/vendor/aspieprogress/asPieProgress.min.css" rel="stylesheet">
  6. <style>
  7. .panel-info .announcement-content {
  8. overflow-y: auto !important;
  9. }
  10. .pie-container {
  11. width: var(--min-size, 100px);
  12. margin: 0 auto;
  13. }
  14. .list-group-dividered .list-group-item:last-child {
  15. border-bottom: none;
  16. }
  17. </style>
  18. @endsection
  19. @section('content')
  20. <div class="page-content container-fluid">
  21. <div class="row">
  22. @if (Session::has('successMsg'))
  23. <x-alert class="col-md-12" :message="Session::pull('successMsg')" />
  24. @endif
  25. <div class="col-xxl-3 col-xl-4 col-lg-5 col-12">
  26. <div class="card card-shadow">
  27. <div class="card-block p-20">
  28. <button class="btn btn-floating btn-sm btn-pure" type="button">
  29. <i class="wb-heart red-500" aria-hidden="true"></i>
  30. </button>
  31. <span class="font-weight-400">{{ trans('user.account.status') }}</span>
  32. @if (sysConfig('checkin_interval'))
  33. <button class="btn btn-md btn-round btn-info float-right" onclick="checkIn()">
  34. <i class="wb-star yellow-400 mr-5" aria-hidden="true"></i>
  35. {{ trans('user.home.attendance.attribute') }}
  36. </button>
  37. @endif
  38. <div class="content-text text-center mb-0">
  39. @if (!$paying_user)
  40. <p class="ml-15 mt-15 text-left">{{ trans('common.more') }}
  41. <code>{{ ucfirst(trans('validation.attributes.time')) }}</code>
  42. </p>
  43. <p class="text-center">{{ trans('common.more') }}
  44. <code>{{ trans('user.attribute.data') }}</code>
  45. </p>
  46. <p class="mb-15 mr-15 text-right">{{ trans('common.more') }}
  47. <code>{{ trans('user.attribute.node') }}</code>
  48. </p>
  49. <a class="btn btn-block btn-danger" href="{{ route('shop.index') }}">{{ trans('user.purchase.promotion') }}</a>
  50. @elseif(auth()->user()->enable)
  51. <i class="wb-check green-400 font-size-40 mr-10" aria-hidden="true"></i>
  52. <span class="font-size-40 font-weight-100">{{ trans('common.status.normal') }}</span>
  53. <p class="font-weight-300 m-0 green-500">{{ trans('user.account.reason.normal') }}</p>
  54. @elseif($remainDays < 0)
  55. <i class="wb-close red-400 font-size-40 mr-10" aria-hidden="true"></i>
  56. <span class="font-size-40 font-weight-100">{{ trans('common.status.expire') }}</span>
  57. <p class="font-weight-300 m-0 red-500">{{ trans('user.account.reason.expired') }}</p>
  58. @elseif($user['unused_traffic'] === 0)
  59. <i class="wb-close red-400 font-size-40 mr-10"></i>
  60. <span class="font-size-40 font-weight-100">{{ trans('common.status.disabled') }}</span>
  61. <p class="font-weight-300 m-0 red-500">{{ trans('user.account.reason.traffic_exhausted') }}</p>
  62. @elseif($user['ban_time'])
  63. <i class="wb-alert orange-400 font-size-40 mr-10"></i>
  64. <span class="font-size-40 font-weight-100">{{ trans('common.status.limited') }}</span>
  65. <p class="font-weight-300 m-0 orange-500">{!! trans('user.account.reason.overused', ['data' => sysConfig('traffic_abuse_limit')]) !!}</p>
  66. @else
  67. <i class="wb-help red-400 font-size-40 mr-10"></i>
  68. <span class="font-size-40 font-weight-100">{{ trans('common.status.disabled') }}</span>
  69. <p class="font-weight-300 m-0 red-500">{{ trans('user.account.reason.unknown') }}</p>
  70. @endif
  71. </div>
  72. </div>
  73. </div>
  74. <div class="card card-shadow">
  75. <div class="card-block p-20">
  76. <div class="row">
  77. <div class="col-sm-7">
  78. <button class="btn btn-floating btn-sm btn-pure" type="button">
  79. <i class="wb-stats-bars cyan-500"></i>
  80. </button>
  81. <span class="font-weight-400">{{ trans('user.account.remain') }}</span>
  82. <div class="text-center font-weight-100 font-size-40">
  83. @if ($user['unused_traffic'] === 0)
  84. {{ trans('common.status.run_out') }}
  85. @else
  86. {{ formatBytes($user['unused_traffic']) }}
  87. @endif
  88. <br />
  89. <h4>{{ trans('user.account.level') }}:
  90. <code class="font-size-20">{{ Auth::user()->level }}</code>
  91. </h4>
  92. </div>
  93. <div class="text-center font-weight-300 blue-grey-500 mb-10">
  94. @if (isset($resetDays) && $resetDays >= 0)
  95. {!! trans_choice('user.account.reset', $resetDays, ['days' => $resetDays]) !!}
  96. @endif
  97. </div>
  98. </div>
  99. <div class="col-sm-5 d-flex align-items-center justify-content-center">
  100. <div class="pie-container">
  101. <div data-plugin="pieProgress" data-valuemax="100" data-barcolor="#96A3FA" data-size="100" data-barsize="10"
  102. data-goal="{{ $unusedPercent }}" role="progressbar" aria-valuenow="{{ $unusedPercent }}">
  103. <span class="pie-progress-number blue-grey-700 font-size-20">{{ $unusedPercent }}%</span>
  104. </div>
  105. </div>
  106. </div>
  107. </div>
  108. </div>
  109. </div>
  110. <div class="card card-shadow">
  111. <div class="card-block p-20">
  112. <button class="btn btn-floating btn-sm btn-pure" type="button">
  113. <i class="wb-calendar green-500"></i>
  114. </button>
  115. <span class="font-weight-400">{{ trans('user.account.time') }}</span>
  116. <div class="content-text text-center mb-0">
  117. @if ($remainDays >= 0)
  118. <span class="font-size-40 font-weight-100">{{ $remainDays . ' ' . trans_choice('common.days.attribute', 1) }}</span>
  119. <p class="blue-grey-500 font-weight-300 m-0">{{ $user['expiration_date'] }}</p>
  120. @else
  121. <span class="font-size-40 font-weight-100">{{ trans('common.status.expire') }}</span>
  122. <br />
  123. <a class="btn btn-danger" href="{{ route('shop.index') }}">{{ trans('user.shop.buy') }}</a>
  124. @endif
  125. </div>
  126. </div>
  127. </div>
  128. @if ($userLoginLog)
  129. <div class="card card-shadow">
  130. <div class="card-block p-20">
  131. <button class="btn btn-floating btn-sm btn-pure" type="button">
  132. <i class="wb-globe purple-500"></i>
  133. </button>
  134. <span class="font-weight-400 mb-10">{{ trans('user.account.last_login') }}</span>
  135. <ul class="list-group list-group-dividered">
  136. <li class="list-group-item">
  137. <i class="icon wb-time"></i> {{ ucfirst(trans('validation.attributes.time')) }}
  138. <span class="float-right">{{ date_format($userLoginLog->created_at, 'Y/m/d H:i') }}</span>
  139. </li>
  140. <li class="list-group-item">
  141. <i class="icon wb-code"></i> {{ trans('user.attribute.ip') }}
  142. <span class="float-right">{{ $userLoginLog->ip }}</span>
  143. </li>
  144. <li class="list-group-item">
  145. <i class="icon wb-cloud"></i> {{ trans('user.attribute.isp') }}
  146. <span class="float-right">{{ $userLoginLog->isp }}</span>
  147. </li>
  148. <li class="list-group-item">
  149. <i class="icon wb-map"></i> {{ trans('user.attribute.address') }}
  150. <span class="float-right">
  151. {{ $userLoginLog->country . ' ' . $userLoginLog->province . ' ' . $userLoginLog->city . ' ' . $userLoginLog->area }}
  152. </span>
  153. </li>
  154. </ul>
  155. </div>
  156. </div>
  157. @endif
  158. </div>
  159. <div class="col-xxl-9 col-xl-8 col-lg-7 col-12">
  160. <div class="row" data-plugin="matchHeight" data-by-row="true">
  161. <div class="col-xl-4 mb-30">
  162. <div class="card card-shadow h-full">
  163. <div class="card-block text-center">
  164. <h3 class="card-header-transparent">
  165. <i class="icon wb-link-intact"></i>{{ trans('user.subscribe.link') }}
  166. </h3>
  167. @if ($subscribe['status'])
  168. <div class="card-body p-lg-0">
  169. @if (count($subType) > 1)
  170. <div class="form-group row">
  171. <label class="col-md-auto col-form-label" for="subType">{{ trans('common.customize') }}</label>
  172. <div class="col">
  173. <select class="form-control" id="subType" name="subType" data-plugin="selectpicker"
  174. data-style="btn-outline btn-primary" title="{{ trans('common.all') }}">
  175. @foreach ($subType as $type => $protocol)
  176. <option value="{{ $type }}">
  177. {{ trans('user.subscribe.protocol_only', ['protocol' => $protocol]) }}
  178. </option>
  179. @endforeach
  180. </select>
  181. </div>
  182. </div>
  183. @endif
  184. <div class="form-group row">
  185. <label class="col-md-auto col-form-label" for="client">{{ trans('user.clients') }}</label>
  186. <div class="col">
  187. <select class="form-control" id="client" name="client" data-plugin="selectpicker"
  188. data-style="btn-primary btn-outline" title="{{ trans('common.default') }}">
  189. <option value="quantumult">Quantumult</option>
  190. <option value="quantumult%20x">QuantumultX</option>
  191. <option value="clash">Clash</option>
  192. <option value="surfboard">Surfboard</option>
  193. <option value="surge">Surge</option>
  194. <option value="shadowrocket">Shadowrocket</option>
  195. <option value="v2rayn">v2rayN</option>
  196. {{-- <option value="shadowsocks">SS路由器</option> --}}
  197. </select>
  198. </div>
  199. </div>
  200. </div>
  201. <div class="card-footer-transparent btn-group">
  202. <button class="btn btn-outline-danger" onclick="exchangeSubscribe();">
  203. <i class="icon wb-refresh" aria-hidden="true"></i>
  204. {{ trans('common.change') }}</button>
  205. <button class="btn btn-outline-info mt-clipboard">
  206. <i class="icon wb-copy" aria-hidden="true"></i>
  207. {{ trans('common.copy.attribute') }}</button>
  208. </div>
  209. @else
  210. <x-alert type="danger" :message="__($subscribe['ban_desc'])" />
  211. @endif
  212. </div>
  213. </div>
  214. </div>
  215. <div class="col-xl-4 mb-30">
  216. <div class="card card-shadow h-full">
  217. <div class="card-block text-center">
  218. <i class="font-size-40 wb-wrench"></i>
  219. <h4 class="card-title">{{ trans('user.clients') }}</h4>
  220. <div class="card-body">
  221. <p>{{ trans('common.download') . ' & ' . trans('user.tutorials') }}</p>
  222. <a class="btn btn-primary mb-10" href="{{ route('knowledge.index') }}">{{ trans('common.goto') }}</a>
  223. </div>
  224. </div>
  225. </div>
  226. </div>
  227. @if (config('common.contact.telegram') || config('common.contact.qq'))
  228. <div class="col-xl-4 mb-30">
  229. <div class="card card-shadow text-center h-full">
  230. <div class="card-block">
  231. <h4 class="card-title">
  232. <i class="wb-bell mr-10 yellow-600"></i>{{ trans('user.home.chat_group') }}
  233. </h4>
  234. @if ($paying_user)
  235. <div class="btn-group">
  236. @if (config('common.contact.qq'))
  237. <a class="card-link btn btn-sm btn-pill-left btn-info" href="{{ config('common.contact.qq') }}" target="_blank"
  238. rel="noopener">
  239. <i class="fa-brands fa-qq"></i> QQ{{ trans('user.home.chat_group') }}
  240. </a>
  241. @endif
  242. @if (config('common.contact.telegram'))
  243. <a class="card-link btn btn-sm btn-pill-right btn-success" href="{{ config('common.contact.telegram') }}"
  244. target="_blank" rel="noopener">
  245. TG{{ trans('user.home.chat_group') }}
  246. <i class="fa-brands fa-telegram"></i></a>
  247. @endif
  248. </div>
  249. @else
  250. <p class="card-link btn btn-sm btn-primary">
  251. <i class="wb-lock mr-5"></i>{{ trans('user.purchase.to_unlock') }}
  252. </p>
  253. @endif
  254. </div>
  255. </div>
  256. </div>
  257. @endif
  258. </div>
  259. <div class="row">
  260. <div class="col-xxl-6 mb-30 mb-xl-0">
  261. <div class="panel panel-info panel-line h-full">
  262. <div class="panel-heading">
  263. <h2 class="panel-title">
  264. <i class="wb-volume-high mr-10"></i>
  265. {{ trans('user.home.announcement') }}
  266. </h2>
  267. <div class="panel-actions pagination-no-border pagination-sm">
  268. {{ $announcements->links() }}
  269. </div>
  270. </div>
  271. <div class="panel-body">
  272. @forelse($announcements as $announcement)
  273. <h2 class="text-center">{!! $announcement->title !!}</h2>
  274. <p class="text-right"><small>{{ trans('common.updated_at') }}
  275. <code>{{ $announcement->updated_at }}</code></small></p>
  276. <div class="announcement-content">
  277. {!! $announcement->content !!}
  278. </div>
  279. @empty
  280. <p class="text-center font-size-40">{{ trans('user.home.empty_announcement') }}</p>
  281. @endforelse
  282. </div>
  283. </div>
  284. </div>
  285. <div class="col-xxl-6">
  286. <div class="panel panel-primary panel-line h-full">
  287. <div class="panel-heading">
  288. <h1 class="panel-title">
  289. <i class="wb-pie-chart mr-10"></i>{{ trans('user.home.traffic_logs') }}
  290. </h1>
  291. <div class="panel-actions">
  292. <ul class="nav nav-pills" role="tablist">
  293. <li class="nav-item">
  294. <a class="nav-link active" data-toggle="tab" href="#daily" role="tab" aria-controls="daily"
  295. aria-expanded="true" aria-selected="false">{{ trans_choice('common.days.attribute', 1) }}</a>
  296. </li>
  297. <li class="nav-item">
  298. <a class="nav-link" data-toggle="tab" href="#monthly" role="tab" aria-controls="monthly"
  299. aria-selected="true">{{ ucfirst(trans('validation.attributes.month')) }}</a>
  300. </li>
  301. </ul>
  302. </div>
  303. </div>
  304. <x-alert type="danger" :message="trans('user.traffic_logs.tips')" />
  305. <div class="panel-body">
  306. <div class="tab-content">
  307. <div class="tab-pane active" id="daily" role="tabpanel">
  308. <canvas id="dailyChart" role="img" aria-label="{{ trans('user.traffic_logs.hourly') }}"></canvas>
  309. </div>
  310. <div class="tab-pane" id="monthly" role="tabpanel">
  311. <canvas id="monthlyChart" role="img" aria-label="{{ trans('user.traffic_logs.daily') }}"></canvas>
  312. </div>
  313. </div>
  314. </div>
  315. </div>
  316. </div>
  317. </div>
  318. </div>
  319. </div>
  320. </div>
  321. @endsection
  322. @section('javascript')
  323. <script src="/assets/global/vendor/aspieprogress/jquery-asPieProgress.min.js"></script>
  324. <script src="/assets/global/vendor/matchheight/jquery.matchHeight-min.js"></script>
  325. <script src="/assets/global/vendor/chart-js/chart.umd.min.js"></script>
  326. <script src="/assets/global/vendor/bootstrap-select/bootstrap-select.min.js"></script>
  327. <script src="/assets/global/js/Plugin/aspieprogress.js"></script>
  328. <script src="/assets/global/js/Plugin/matchheight.js"></script>
  329. <script src="/assets/global/js/Plugin/bootstrap-select.js"></script>
  330. <script>
  331. function exchangeSubscribe() { // 更换订阅地址
  332. showConfirm({
  333. title: '{{ trans('common.warning') }}',
  334. html: `{!! trans('user.subscribe.exchange_warning') !!}`,
  335. icon: "warning",
  336. onConfirm: function() {
  337. ajaxPost('{{ route('changeSub') }}');
  338. }
  339. });
  340. }
  341. $(".mt-clipboard").on("click", function() {
  342. let base = @json($user['sub_url']);
  343. const client = $("#client").val();
  344. const subType = $("#subType").val();
  345. const params = [];
  346. if (client) {
  347. params.push("target=" + client);
  348. }
  349. if (subType) {
  350. params.push("type=" + subType);
  351. }
  352. if (params.length > 0) {
  353. base += "?" + params.join("&");
  354. }
  355. copyToClipboard(base);
  356. });
  357. @if (sysConfig('checkin_interval'))
  358. function checkIn() { // 签到
  359. ajaxPost('{{ route('checkIn') }}');
  360. }
  361. @endif
  362. function common_options(tail) {
  363. return {
  364. responsive: true,
  365. scales: {
  366. x: {
  367. ticks: {
  368. callback: function(value) {
  369. return this.getLabelForValue(value) + " " + tail;
  370. }
  371. },
  372. grid: {
  373. display: false
  374. }
  375. },
  376. y: {
  377. ticks: {
  378. callback: function(value) {
  379. return this.getLabelForValue(value) + " GB";
  380. }
  381. },
  382. grid: {
  383. display: false
  384. },
  385. min: 0
  386. }
  387. },
  388. plugins: {
  389. legend: false,
  390. tooltip: {
  391. mode: "index",
  392. intersect: false,
  393. callbacks: {
  394. title: function(context) {
  395. return context[0].label + " " + tail;
  396. },
  397. label: function(context) {
  398. return context.parsed.y + " GB";
  399. }
  400. }
  401. }
  402. }
  403. };
  404. }
  405. function datasets(label, data) {
  406. return {
  407. labels: label,
  408. datasets: [{
  409. backgroundColor: "rgba(184, 215, 255)",
  410. borderColor: "rgba(184, 215, 255)",
  411. data: data,
  412. tension: 0.4
  413. }]
  414. };
  415. }
  416. let dailyChartInstance = new Chart(document.getElementById("dailyChart"), {
  417. type: "line",
  418. data: datasets(@json($dayHours), @json($trafficHourly)),
  419. options: common_options(@json(trans_choice('common.hour', 2)))
  420. });
  421. let monthlyChartInstance = new Chart(document.getElementById("monthlyChart"), {
  422. type: "line",
  423. data: datasets(@json($monthDays), @json($trafficDaily)),
  424. options: common_options(@json(trans_choice('common.days.attribute', 2)))
  425. });
  426. @if ($user['ban_time']) // 封禁倒计时
  427. const banedTime = new Date("{{ $user['ban_time'] }}").getTime();
  428. countDown(banedTime, "banedTime", true);
  429. setInterval(function() {
  430. countDown(banedTime, "banedTime", true);
  431. }, 1000);
  432. @endif
  433. @if (isset($resetDays) && $resetDays === 0) // 重置日倒计时
  434. const resetTime = new Date("{{ date('Y-m-d 00:00', strtotime('tomorrow')) }}").getTime();
  435. countDown(resetTime, "restTime");
  436. setInterval(function() {
  437. countDown(resetTime, "restTime");
  438. }, 60000);
  439. @endif
  440. function countDown(endTime, id, seconds = false) { // 计时器主题逻辑
  441. const distance = endTime - new Date().getTime();
  442. const hour = Math.floor(distance % 86400000 / 3600000);
  443. const minute = Math.floor((distance % 3600000) / 60000);
  444. let string = "";
  445. if (hour) {
  446. string += hour + " " + @json(trans_choice('common.hour', 1));
  447. }
  448. if (minute) {
  449. string += " " + minute + " " + @json(ucfirst(trans('validation.attributes.minute')));
  450. }
  451. if (seconds) {
  452. string += " " + Math.floor((distance % 60000) / 1000) + " " + @json(ucfirst(trans('validation.attributes.second')));
  453. }
  454. document.getElementById(id).innerHTML = string;
  455. if (distance <= 0 && distance.abs > 1000) {
  456. window.location.reload();
  457. }
  458. }
  459. function syncAnnouncementHeight() {
  460. // 获取右侧面板的总高度
  461. const rightPanelHeight = $(".panel-primary .panel-body").outerHeight();
  462. if (rightPanelHeight > 0) {
  463. $('.panel-info .announcement-content').css({
  464. 'max-height': rightPanelHeight - 20 + 'px',
  465. });
  466. }
  467. }
  468. function syncPieSize() {
  469. const $container = $(".pie-container");
  470. const $parent = $container.parent();
  471. if ($parent.length) {
  472. let minSize = Math.min($parent.width(), $parent.height());
  473. $container.css('--min-size', minSize + 'px');
  474. // 仅在必要时重绘插件,避免动画闪烁
  475. const $pie = $('[data-plugin="pieProgress"]');
  476. if ($pie.data('asPieProgress')) {
  477. $pie.asPieProgress('update');
  478. }
  479. }
  480. }
  481. $(document).ready(function() {
  482. syncAnnouncementHeight();
  483. syncPieSize();
  484. });
  485. // 窗口缩放时重新计算
  486. $(window).on('resize', function() {
  487. if (dailyChartInstance) dailyChartInstance.resize();
  488. if (monthlyChartInstance) monthlyChartInstance.resize();
  489. syncPieSize();
  490. syncAnnouncementHeight();
  491. });
  492. </script>
  493. @endsection