index.blade.php 29 KB

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