index.blade.php 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614
  1. @extends('admin.layout')
  2. @section('header')
  3. <style type="text/css">
  4. .side-class-bar ul li a {
  5. padding: 5px 10px !important;
  6. }
  7. .side-class-bar .a_selected {
  8. background-color: #eeeeee;
  9. }
  10. #file-upload-modal .modal-dialog {
  11. width: 80%;
  12. }
  13. #crop-file-modal .modal-dialog {
  14. width: 80%;
  15. }
  16. .img-check {
  17. position: absolute;
  18. right: 5px;
  19. bottom: 5px;
  20. margin: 0;
  21. padding: 0;
  22. width: 22px;
  23. height: 22px;
  24. background: url(/base/img/check_blue_1.png) no-repeat;
  25. border: none;
  26. background-position: -144px 0;
  27. }
  28. .img-check.checked {
  29. background-position: -168px 0;
  30. }
  31. .img-card .content {
  32. text-align: center;
  33. word-wrap: break-word;
  34. }
  35. .img-card .image {
  36. position: relative;
  37. }
  38. .sg-divider {
  39. margin: 1rem 0rem;
  40. line-height: 1;
  41. height: 0em;
  42. font-weight: bold;
  43. text-transform: uppercase;
  44. letter-spacing: 0.05em;
  45. color: rgba(0, 0, 0, 0.85);
  46. }
  47. </style>
  48. <script type="text/javascript" src="/base/neditor-1.5.3/dialogs/image/image.js"></script>
  49. <script type="text/javascript" src="/base/neditor-1.5.3/dialogs/internal.js"></script>
  50. @endsection
  51. @section('content')
  52. <div class="wrapper wrapper-content animated fadeInRight">
  53. <div class="row">
  54. <div class="col-sm-12">
  55. @if(isset($errors) && !$errors->isEmpty())
  56. <div class="alert alert-danger alert-dismissable">
  57. <button type="button" class="close" data-dismiss="alert"
  58. aria-hidden="true">
  59. &times;
  60. </button>
  61. @foreach($errors->keys() as $key)
  62. {{ $errors->first($key) }}
  63. @endforeach
  64. </div>
  65. @endif
  66. <div class="ibox float-e-margins">
  67. <div class="ibox-title" style="padding: 7px 14px 15px;">
  68. <h5 style="margin-top: 8px">图片管理</h5>
  69. <div class="ibox-tools" style="margin-top: 8px">
  70. <a class="collapse-link"> <i class="fa fa-chevron-up"></i>
  71. </a>
  72. </div>
  73. <div class="btn-group pull-right">
  74. <div id="select-toogle" class="btn btn-sm btn-default">全选图片</div>
  75. @if($status != 'search')
  76. {!! widget('Tools.ImgUpload')->upload('上传', '/images','upload_files', ['position'=>'local','class' => $a_class->class, 'url' => U('Base/Photos/index', ['class' => Request::get('class')?:1])]) !!}
  77. @endif
  78. <div id="edit-file" class="btn btn-sm btn-success">编辑</div>
  79. @if($status != 'search')
  80. <div id="move-file" class="btn btn-sm btn-info">移动</div>
  81. @endif
  82. <div id="crop-file" class="btn btn-sm btn-warning">裁剪</div>
  83. {{--<div id="download-file" class="btn btn-sm btn-primary">下载</div>--}}
  84. <div id="delete-file" class="btn btn-sm btn-danger">删除图片</div>
  85. @if($a_class->class != '未分类' && $status != 'search')
  86. <div id="delete-class" class="btn btn-sm btn-danger">删除分类</div>
  87. @endif
  88. </div>
  89. </div>
  90. <div class="ibox-content">
  91. <div class="row">
  92. <div class="col-sm-2">
  93. <div class="side-class-bar">
  94. <div class="btn btn-sm btn-primary btn-block" id="add-class-btn">添加分类</div>
  95. <ul class="nav">
  96. @foreach($classes as $class)
  97. @if($class->class == $a_class->class && $status != 'search')
  98. <li class="a_selected">
  99. <a href="{{ U('Base/Photos/index', ['class' => $class->id]) }}">
  100. <i class="fa fa-folder-open" aria-hidden="true"></i>
  101. <span class="nav-label">
  102. {{ $class->class }}
  103. </span>
  104. </a>
  105. </li>
  106. @else
  107. <li>
  108. <a href="{{ U('Base/Photos/index', ['class' => $class->id]) }}">
  109. <i class="fa fa-folder" aria-hidden="true"></i>
  110. <span class="nav-label">
  111. {{ $class->class }}
  112. </span>
  113. </a>
  114. </li>
  115. @endif
  116. @endforeach
  117. </ul>
  118. </div>
  119. </div>
  120. <div class="col-sm-10">
  121. <div class="row">
  122. <form method="GET" accept-charset="UTF-8">
  123. <div class="col-sm-4">
  124. <input type="text" value="{{Request::get('search') ? : ''}}" placeholder="请输入关键词" name="keyword"
  125. class="input-sm form-control" id="search-file-input">
  126. </div>
  127. <div class="col-sm-3">
  128. <input name="start" id="start"
  129. class="laydate-icon form-control layer-date"
  130. value="{{Request::get('start') ? : ''}}" placeholder="开始日期">
  131. </div>
  132. <div class="col-sm-3">
  133. <input name="end" id="end" class="laydate-icon form-control layer-date"
  134. value="{{Request::get('end') ? : ''}}" placeholder="结束日期">
  135. </div>
  136. <div class="col-sm-2">
  137. <button type="submit" class="btn btn-sm btn-primary"
  138. id="search-file">搜索
  139. </button>
  140. </div>
  141. </form>
  142. </div>
  143. <div class="photos" style="margin-top: 10px;padding-top: 15px;border-top: 1px solid #ddd">
  144. @foreach($photos as $photo)
  145. @if($loop->index % 6 == 0)
  146. <div class="row">
  147. @endif
  148. <div class="img-card col-sm-2">
  149. <div class="image">
  150. <img src="{{ $photo->url }}" class="img-thumbnail">
  151. <span class="img-check" data-id="{{ $photo->id }}"
  152. data-name="{{ $photo->name }}"
  153. data-src="{{ $photo->url }}"
  154. data-class="{{ $photo->class }}"
  155. data-md5="{{ $photo->md5 }}"></span>
  156. </div>
  157. <div class="content">
  158. {{ $photo->name }}
  159. </div>
  160. </div>
  161. @if($loop->index % 6 == 5)
  162. </div>
  163. @endif
  164. @endforeach
  165. </div>
  166. </div>
  167. </div>
  168. </div>
  169. <div class="row">
  170. <div class="col-sm-6">
  171. <div class="dataTables_info" id="DataTables_Table_0_info"
  172. role="alert" aria-live="polite" aria-relevant="all">每页{{ $photos->count() }}
  173. 条,共{{ $photos->lastPage() }}页,总{{ $photos->total() }}条。
  174. </div>
  175. </div>
  176. <div class="col-sm-6">
  177. <div class="dataTables_paginate paging_simple_numbers" id="DataTables_Table_0_paginate">
  178. {!! $photos->setPath('')->appends(Request::all())->render() !!}
  179. </div>
  180. </div>
  181. </div>
  182. </div>
  183. </div>
  184. </div>
  185. </div>
  186. <div class="modal fade" id="add-class-modal" tabindex="-1" role="dialog" aria-labelledby="add-class-modal-label"
  187. aria-hidden="true">
  188. <div class="modal-dialog">
  189. <div class="modal-content">
  190. <div class="modal-header">
  191. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  192. <h4 class="modal-title" id="add-class-modal-label">添加分类</h4>
  193. </div>
  194. <div class="modal-body">
  195. <form class="form-horizontal" role="form" method="POST" id="add-class-form"
  196. action="{{ U('Base/Class/add') }}">
  197. {{ csrf_field() }}
  198. <div class="form-group">
  199. <label for="class" class="col-md-2 col-md-offset-2 control-label">分类</label>
  200. <div class="col-md-6">
  201. <input id="class" type="text" class="form-control" name="class" required autofocus>
  202. </div>
  203. </div>
  204. <div class="form-group">
  205. <div class="col-md-6 col-md-offset-4">
  206. <button type="submit" class="btn btn-primary" id="edit-project-confirm">
  207. 添加
  208. </button>
  209. </div>
  210. </div>
  211. </form>
  212. </div>
  213. </div>
  214. </div>
  215. </div>
  216. <div class="modal fade" id="edit-file-modal" tabindex="-1" role="dialog" aria-labelledby="edit-file-modal-label"
  217. aria-hidden="true">
  218. <div class="modal-dialog">
  219. <div class="modal-content">
  220. <div class="modal-header">
  221. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  222. <h4 class="modal-title" id="edit-file-modal-label">编辑图片</h4>
  223. </div>
  224. <div class="modal-body">
  225. <form class="form-horizontal" role="form" method="POST" id="edit-file-form"
  226. action="{{ U('Base/Photos/edit') }}">
  227. {{ csrf_field() }}
  228. <div class="form-group">
  229. <label for="img-name" class="col-md-2 col-md-offset-2 control-label">图片名</label>
  230. <div class="col-md-6">
  231. <input type="text" class="img-name form-control" name="img-name" required autofocus>
  232. </div>
  233. </div>
  234. <input class="img-id" type="text" name="img-id" style="display: none">
  235. <div class="form-group">
  236. <div class="col-md-6 col-md-offset-4">
  237. <button type="submit" class="btn btn-primary">
  238. 确认
  239. </button>
  240. </div>
  241. </div>
  242. </form>
  243. </div>
  244. </div>
  245. </div>
  246. </div>
  247. <div class="modal fade" id="move-file-modal" tabindex="-1" role="dialog" aria-labelledby="move-file-modal-label"
  248. aria-hidden="true">
  249. <div class="modal-dialog">
  250. <div class="modal-content">
  251. <div class="modal-header">
  252. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  253. <h4 class="modal-title" id="move-file-modal-label">移动图片</h4>
  254. </div>
  255. <div class="modal-body">
  256. <form class="form-horizontal" role="form" method="POST" id="move-file-form"
  257. action="{{ U('Base/Photos/move') }}">
  258. {{ csrf_field() }}
  259. <div class="form-group">
  260. <label for="class" class="col-md-2 col-md-offset-2 control-label">移动到</label>
  261. <div class="col-md-6">
  262. <select class="form-control" name="class">
  263. @foreach($classes as $class)
  264. <option value="{{ $class->id }}">
  265. {{ $class->class }}
  266. </option>
  267. @endforeach
  268. </select>
  269. </div>
  270. </div>
  271. <input class="ids" type="text" name="ids" style="display: none">
  272. <div class="form-group">
  273. <div class="col-md-6 col-md-offset-4">
  274. <button type="submit" class="btn btn-primary">
  275. 确认
  276. </button>
  277. </div>
  278. </div>
  279. </form>
  280. </div>
  281. </div>
  282. </div>
  283. </div>
  284. <div class="modal fade" id="delete-file-modal" tabindex="-1" role="dialog"
  285. aria-labelledby="delete-file-modal-label"
  286. aria-hidden="true">
  287. <div class="modal-dialog">
  288. <div class="modal-content">
  289. <div class="modal-header">
  290. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  291. <h4 class="modal-title" id="delete-file-modal-label">确定要删除以下图片吗?</h4>
  292. </div>
  293. <div class="modal-body">
  294. <div id="delete-files"></div>
  295. <form class="form-horizontal" role="form" method="POST" id="delete-file-form"
  296. action="{{ U('Base/Photos/delete') }}">
  297. {{ csrf_field() }}
  298. <input class="ids" type="text" name="ids" style="display: none">
  299. <div class="form-group">
  300. <div class="col-md-6 col-md-offset-4">
  301. <button type="submit" class="btn btn-primary">
  302. 确认
  303. </button>
  304. </div>
  305. </div>
  306. </form>
  307. </div>
  308. </div>
  309. </div>
  310. </div>
  311. <div class="modal fade" id="delete-class-modal" tabindex="-1" role="dialog"
  312. aria-labelledby="delete-class-modal-label" aria-hidden="true">
  313. <div class="modal-dialog">
  314. <div class="modal-content">
  315. <div class="modal-header">
  316. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  317. <h4 class="modal-title" id="delete-class-modal-label">
  318. {{ '确定要删除分类《' . $a_class->class . '》及其分类下所有的图片吗?' }}
  319. </h4>
  320. </div>
  321. <div class="modal-body">
  322. <form class="form-horizontal" role="form" method="POST" id="delete-class-form"
  323. action="{{ U('Base/Class/delete', ['class' => $a_class->id]) }}">
  324. {{ csrf_field() }}
  325. <div class="form-group">
  326. <div class="col-md-6 col-md-offset-4">
  327. <button type="submit" class="btn btn-primary">
  328. 确认
  329. </button>
  330. </div>
  331. </div>
  332. </form>
  333. </div>
  334. </div>
  335. </div>
  336. </div>
  337. <div class="modal fade" id="crop-file-modal" tabindex="-1" role="dialog" aria-labelledby="crop-file-modal-label"
  338. aria-hidden="true">
  339. <div class="modal-dialog">
  340. <div class="modal-content">
  341. <div class="modal-header">
  342. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
  343. <h4 class="modal-title" id="crop-file-modal-label">
  344. 裁剪图片
  345. </h4>
  346. </div>
  347. <div class="modal-body">
  348. <div class="row">
  349. <div class="col-sm-6">
  350. <img id="crop-img">
  351. </div>
  352. <div class="col-sm-6">
  353. <div class="row">
  354. <div class="btn-group ratio-group">
  355. <div class="btn btn-sm btn-success ratio-16-9">16:9</div>
  356. <div class="btn btn-sm btn-success ratio-4-3">4:3</div>
  357. <div class="btn btn-sm btn-success ratio-1-1">1:1</div>
  358. <div class="btn btn-sm btn-success ratio-free">自定义</div>
  359. </div>
  360. </div>
  361. <div class="sg-divider"></div>
  362. <div class="row">
  363. <div class="form-group">
  364. <div class="col-md-6">
  365. <input id="crop-img-name" type="text" class="form-control" name="img-name"
  366. placeholder="文件名,扩展名为png">
  367. </div>
  368. </div>
  369. </div>
  370. <div class="sg-divider"></div>
  371. <div class="row">
  372. <div class="btn btn-sm btn-success" id="crop-confirm">确认裁剪</div>
  373. </div>
  374. </div>
  375. </div>
  376. </div>
  377. </div>
  378. </div>
  379. </div>
  380. <div class="modal fade modal-margin-top" id="file-upload-modal" tabindex="-1" role="dialog"
  381. aria-labelledby="file-upload-modal-label" aria-hidden="true">
  382. <div class="modal-dialog">
  383. <div class="modal-content">
  384. <div class="modal-body">
  385. <div id="fine-uploader-manual-trigger"></div>
  386. </div>
  387. </div>
  388. </div>
  389. </div>
  390. @endsection
  391. @section('footer')
  392. <script type="text/javascript">
  393. $(function () {
  394. // var uploader = new qq.FineUploader({
  395. // element: document.getElementById("fine-uploader-manual-trigger"),
  396. // request: {
  397. // endpoint: "/admin/Base/Attachment/webupload?folder={{ urlencode('/upload/user') }}&class={{ $a_class->class }}",
  398. // customHeaders: {
  399. // 'X-CSRF-TOKEN': $('meta[name="csrf-token"]').attr('content')
  400. // },
  401. // inputName: 'file'
  402. // },
  403. // template: 'qq-template-manual-trigger',
  404. // thumbnails: {
  405. // placeholders: {
  406. // waitingPath: "/base/plugins/fine-uploader/placeholders/waiting-generic.png",
  407. // notAvailablePath: "/base/plugins/fine-uploader/placeholders/not_available-generic.png"
  408. // }
  409. // },
  410. // autoUpload: false,
  411. // callbacks: {
  412. // onComplete: function(id, name, response) {
  413. // },
  414. // onAllComplete: function(succeeded, failed)
  415. // {
  416. // if(failed.length == 0) {
  417. // window.location = "{{ U('Base/Photos/index', ['class' => $a_class->id]) }}";
  418. // }
  419. // }
  420. // },
  421. // validation: {
  422. // allowedExtensions: ['jpg', 'jpeg', 'png', 'bmp', 'gif'],
  423. // allowEmpty: true,
  424. // },
  425. // messages: {
  426. // noFilesError: '没有需要上传的文件',
  427. // sizeError: '{file}过大',
  428. // typeError: '{file}上传错误!支持的文件类型:{extensions}',
  429. // }
  430. // });
  431. // $('#trigger-upload').click(function() {
  432. // ratio = parseInt($('#compress_ratio').val());
  433. // if(ratio >= 1 && ratio <= 50) {
  434. // uploader.setEndpoint("/admin/Base/Attachment/webupload?folder={{ urlencode('/upload/user') }}&class={{ $a_class->class }}&compress_ratio=" + ratio);
  435. // }
  436. // uploader.uploadStoredFiles();
  437. // });
  438. // $('#upload-file').on('click', function() {
  439. // $('#file-upload-modal').modal('show');
  440. // });
  441. $('#add-class-btn').on('click', function () {
  442. $('#add-class-modal').modal('show');
  443. });
  444. $('#add-class-confirm-btn').on('click', function () {
  445. $('#add-class-form').submit();
  446. });
  447. $('.img-card', '.photos').on('click', function () {
  448. $(this).find('.img-check').toggleClass('checked');
  449. });
  450. $('#edit-file').on('click', function () {
  451. edit = $('.photos').find('.checked')[0];
  452. if (edit != undefined) {
  453. $edit = $(edit);
  454. $('.img-name', '#edit-file-modal').val($edit.attr('data-name'));
  455. $('.img-id', '#edit-file-modal').val($edit.attr('data-id'));
  456. $('#edit-file-modal').modal('show');
  457. }
  458. });
  459. $('#download-file').on('click', function () {
  460. edit = $('.photos').find('.checked')[0];
  461. if (edit != undefined) {
  462. window.location = "{{ url('/') }}" + "/attachment/" + $(edit).attr('data-md5');
  463. }
  464. });
  465. $('#select-toogle').on('click', function () {
  466. if ($(this).hasClass('selected')) {
  467. $('.photos .img-check').removeClass('checked');
  468. $(this).removeClass('selected').text('全选图片');
  469. } else {
  470. $('.photos .img-check').removeClass('checked').addClass('checked');
  471. $(this).addClass('selected').text('取消全选');
  472. }
  473. });
  474. $('#move-file').on('click', function () {
  475. photos = $('.photos').find('.checked');
  476. if (photos.length != 0) {
  477. ids = $(photos[0]).attr('data-id');
  478. for (i = 1; i < photos.length; ++i) {
  479. ids += ',' + $(photos[i]).attr('data-id');
  480. }
  481. $('.ids', '#move-file-modal').val(ids);
  482. $('#move-file-modal').modal('show');
  483. }
  484. });
  485. $('#delete-file').on('click', function () {
  486. photos = $('.photos').find('.checked');
  487. info = '';
  488. if (photos.length != 0) {
  489. ids = $(photos[0]).attr('data-id');
  490. info += $(photos[0]).attr('data-name');
  491. for (i = 1; i < photos.length; ++i) {
  492. ids += ',' + $(photos[i]).attr('data-id');
  493. info += '<br>' + $(photos[i]).attr('data-name');
  494. }
  495. $('.ids', '#delete-file-modal').val(ids);
  496. $('#delete-files').html(info);
  497. $('#delete-file-modal').modal('show');
  498. }
  499. });
  500. $('#delete-class').on('click', function () {
  501. $('#delete-class-modal').modal('show');
  502. });
  503. $('#search-file').on('click', function (e) {
  504. e.preventDefault();
  505. $input = $('#search-file-input');
  506. $start = $('#start');
  507. $end = $('#end');
  508. if ($input.val().trim() != '' || $start.val().trim() != '' || $end.val().trim() != '') {
  509. window.location = "{{ U('Base/Photos/index') }}" + '?search=' + $input.val().trim() + '&start=' + $start.val() + '&end=' + $end.val();
  510. }
  511. });
  512. var crop_image = document.getElementById('crop-img');
  513. var cropper = new Cropper(crop_image, {
  514. minContainerWidth: 400,
  515. minContainerHeight: 300,
  516. });
  517. $('#crop-file').on('click', function () {
  518. edit = $('.photos').find('.checked')[0];
  519. if (edit != undefined) {
  520. src = $(edit).attr('data-src');
  521. cropper.replace(src);
  522. cropper.reset();
  523. $('#crop-img').attr('src', src);
  524. $('#crop-file-modal').attr('data-class', $(edit).attr('data-class'));
  525. $('#crop-file-modal').modal('show');
  526. }
  527. });
  528. $('#crop-confirm').on('click', function () {
  529. cropper.getCroppedCanvas().toBlob(function (blob) {
  530. var formData = new FormData();
  531. a_class = $('#crop-file-modal').attr('data-class').trim();
  532. name = $('#crop-img-name').val().trim();
  533. if (name == '') {
  534. name = '裁剪.png';
  535. } else if (name.lastIndexOf('.png') == -1) {
  536. name = name + '.png';
  537. }
  538. formData.append('file', blob);
  539. formData.append('folder', '/upload/images/crop');
  540. formData.append('class', a_class);
  541. formData.append('from', 'crop');
  542. formData.append('name', name);
  543. $.ajax({
  544. url: "{{ U('Base/Photos/crop') }}",
  545. method: 'POST',
  546. data: formData,
  547. processData: false,
  548. contentType: false,
  549. header: {
  550. 'X-CSRF-TOKEN': "{{ csrf_token() }}"
  551. },
  552. success: function (data) {
  553. window.location = "{{ U('Base/Photos/index', ['class' => $a_class->class]) }}";
  554. }
  555. });
  556. });
  557. });
  558. $('.ratio-group').on('click', '.ratio-16-9', function () {
  559. cropper.setAspectRatio(16 / 9);
  560. }).on('click', '.ratio-4-3', function () {
  561. cropper.setAspectRatio(4 / 3);
  562. }).on('click', '.ratio-1-1', function () {
  563. cropper.setAspectRatio(1);
  564. }).on('click', '.ratio-free', function () {
  565. cropper.setAspectRatio(NaN);
  566. });
  567. });
  568. </script>
  569. <script>
  570. laydate({elem: "#start", event: "focus"});
  571. laydate({elem: "#end", event: "focus"});
  572. </script>
  573. @endsection