index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678
  1. <template>
  2. <div class="app">
  3. <div class="admin-area">
  4. <main-header :class="{ 'admin-sidebar-active': sidebarActive }" />
  5. <div class="admin-content">
  6. <div
  7. class="admin-sidebar"
  8. :class="{ minimised: !sidebarActive }"
  9. >
  10. <div class="inner">
  11. <div
  12. class="bottom"
  13. :style="`padding-bottom: ${sidebarPadding}px`"
  14. >
  15. <div
  16. class="sidebar-item toggle-sidebar"
  17. @click="toggleSidebar()"
  18. content="Expand"
  19. v-tippy="{ onShow: () => !sidebarActive }"
  20. >
  21. <i class="material-icons">menu_open</i>
  22. <span>Minimise</span>
  23. </div>
  24. <router-link
  25. class="sidebar-item songs"
  26. to="/admin/songs"
  27. content="Songs"
  28. v-tippy="{
  29. theme: 'info',
  30. onShow: () => !sidebarActive
  31. }"
  32. >
  33. <i class="material-icons">music_note</i>
  34. <span>Songs</span>
  35. </router-link>
  36. <router-link
  37. class="sidebar-item reports"
  38. to="/admin/reports"
  39. content="Reports"
  40. v-tippy="{
  41. theme: 'info',
  42. onShow: () => !sidebarActive
  43. }"
  44. >
  45. <i class="material-icons">flag</i>
  46. <span>Reports</span>
  47. </router-link>
  48. <router-link
  49. class="sidebar-item stations"
  50. to="/admin/stations"
  51. content="Stations"
  52. v-tippy="{
  53. theme: 'info',
  54. onShow: () => !sidebarActive
  55. }"
  56. >
  57. <i class="material-icons">radio</i>
  58. <span>Stations</span>
  59. </router-link>
  60. <router-link
  61. class="sidebar-item playlists"
  62. to="/admin/playlists"
  63. content="Playlists"
  64. v-tippy="{
  65. theme: 'info',
  66. onShow: () => !sidebarActive
  67. }"
  68. >
  69. <i class="material-icons">library_music</i>
  70. <span>Playlists</span>
  71. </router-link>
  72. <div
  73. v-if="sidebarActive"
  74. class="sidebar-item with-children"
  75. :class="{ 'is-active': childrenActive.users }"
  76. >
  77. <span>
  78. <router-link to="/admin/users">
  79. <i class="material-icons">people</i>
  80. <span>Users</span>
  81. </router-link>
  82. <i
  83. class="material-icons toggle-sidebar-children"
  84. @click="
  85. toggleChildren({ child: 'users' })
  86. "
  87. >
  88. {{
  89. childrenActive.users
  90. ? "expand_less"
  91. : "expand_more"
  92. }}
  93. </i>
  94. </span>
  95. <div class="sidebar-item-children">
  96. <router-link
  97. class="sidebar-item-child"
  98. to="/admin/users"
  99. >
  100. Users
  101. </router-link>
  102. <router-link
  103. class="sidebar-item-child"
  104. to="/admin/users/data-requests"
  105. >
  106. Data Requests
  107. </router-link>
  108. </div>
  109. </div>
  110. <router-link
  111. v-else
  112. class="sidebar-item users"
  113. to="/admin/users"
  114. content="Users"
  115. v-tippy="{
  116. theme: 'info',
  117. onShow: () => !sidebarActive
  118. }"
  119. >
  120. <i class="material-icons">people</i>
  121. <span>Users</span>
  122. </router-link>
  123. <router-link
  124. class="sidebar-item punishments"
  125. to="/admin/punishments"
  126. content="Punishments"
  127. v-tippy="{
  128. theme: 'info',
  129. onShow: () => !sidebarActive
  130. }"
  131. >
  132. <i class="material-icons">gavel</i>
  133. <span>Punishments</span>
  134. </router-link>
  135. <router-link
  136. class="sidebar-item news"
  137. to="/admin/news"
  138. content="News"
  139. v-tippy="{
  140. theme: 'info',
  141. onShow: () => !sidebarActive
  142. }"
  143. >
  144. <i class="material-icons">chrome_reader_mode</i>
  145. <span>News</span>
  146. </router-link>
  147. <router-link
  148. class="sidebar-item statistics"
  149. to="/admin/statistics"
  150. content="Statistics"
  151. v-tippy="{
  152. theme: 'info',
  153. onShow: () => !sidebarActive
  154. }"
  155. >
  156. <i class="material-icons">show_chart</i>
  157. <span>Statistics</span>
  158. </router-link>
  159. </div>
  160. </div>
  161. </div>
  162. <div class="admin-container">
  163. <div class="admin-tab-container">
  164. <router-view></router-view>
  165. </div>
  166. <main-footer />
  167. </div>
  168. </div>
  169. </div>
  170. <floating-box
  171. id="keyboardShortcutsHelper"
  172. ref="keyboardShortcutsHelper"
  173. >
  174. <template #body>
  175. <div>
  176. <div>
  177. <span class="biggest"
  178. ><b>Keyboard shortcuts helper</b></span
  179. >
  180. <span
  181. ><b>Ctrl + /</b> - Toggles this keyboard shortcuts
  182. helper</span
  183. >
  184. <span
  185. ><b>Ctrl + Shift + /</b> - Resets the position of
  186. this keyboard shortcuts helper</span
  187. >
  188. <hr />
  189. </div>
  190. <div>
  191. <span class="biggest"><b>Table</b></span>
  192. <span class="bigger"><b>Navigation</b></span>
  193. <span
  194. ><b>Up / Down arrow keys</b> - Move between
  195. rows</span
  196. >
  197. <hr />
  198. </div>
  199. <div>
  200. <span class="bigger"><b>Page navigation</b></span>
  201. <span
  202. ><b>Ctrl + Left/Right arrow keys</b> - Previous/next
  203. page</span
  204. >
  205. <span
  206. ><b>Ctrl + Shift + Left/Right arrow keys</b> -
  207. First/last page</span
  208. >
  209. <hr />
  210. </div>
  211. <div>
  212. <span class="bigger"><b>Reset localStorage</b></span>
  213. <span><b>Ctrl + F5</b> - Resets localStorage</span>
  214. <hr />
  215. </div>
  216. <div>
  217. <span class="bigger"><b>Selecting</b></span>
  218. <span><b>Space</b> - Selects/unselects a row</span>
  219. <span><b>Ctrl + A</b> - Selects all rows</span>
  220. <span
  221. ><b>Shift + Up/Down arrow keys</b> - Selects all
  222. rows in between</span
  223. >
  224. <span
  225. ><b>Ctrl + Up/Down arrow keys</b> - Unselects all
  226. rows in between</span
  227. >
  228. <hr />
  229. </div>
  230. <div>
  231. <span class="bigger"><b>Popup actions</b></span>
  232. <span><b>Ctrl + 1-9</b> - Execute action 1-9</span>
  233. <span><b>Ctrl + 0</b> - Select action 1</span>
  234. <hr />
  235. </div>
  236. </div>
  237. </template>
  238. </floating-box>
  239. <modal-manager />
  240. </div>
  241. </template>
  242. <script>
  243. import { mapState, mapActions, mapGetters } from "vuex";
  244. import keyboardShortcuts from "@/keyboardShortcuts";
  245. import FloatingBox from "@/components/FloatingBox.vue";
  246. import ModalManager from "@/components/ModalManager.vue";
  247. export default {
  248. components: {
  249. FloatingBox,
  250. ModalManager
  251. },
  252. data() {
  253. return {
  254. currentTab: "",
  255. siteSettings: {
  256. logo: "",
  257. sitename: ""
  258. },
  259. sidebarActive: true,
  260. sidebarPadding: 0
  261. };
  262. },
  263. computed: {
  264. ...mapGetters({
  265. socket: "websockets/getSocket"
  266. }),
  267. ...mapState("admin", { childrenActive: state => state.childrenActive })
  268. },
  269. watch: {
  270. $route(route) {
  271. if (this.getTabFromPath(route.path)) this.onRouteChange();
  272. }
  273. },
  274. async mounted() {
  275. if (this.getTabFromPath()) {
  276. this.onRouteChange();
  277. } else if (localStorage.getItem("lastAdminPage")) {
  278. this.$router.push(
  279. `/admin/${localStorage.getItem("lastAdminPage")}`
  280. );
  281. } else {
  282. this.$router.push(`/admin/songs`);
  283. }
  284. this.siteSettings = await lofig.get("siteSettings");
  285. this.sidebarActive = JSON.parse(
  286. localStorage.getItem("admin-sidebar-active")
  287. );
  288. if (this.sidebarActive === null)
  289. this.sidebarActive = !(document.body.clientWidth <= 768);
  290. this.calculateSidebarPadding();
  291. window.addEventListener("scroll", this.calculateSidebarPadding);
  292. keyboardShortcuts.registerShortcut(
  293. "admin.toggleKeyboardShortcutsHelper",
  294. {
  295. keyCode: 191, // '/' key
  296. ctrl: true,
  297. preventDefault: true,
  298. handler: () => {
  299. this.toggleKeyboardShortcutsHelper();
  300. }
  301. }
  302. );
  303. keyboardShortcuts.registerShortcut(
  304. "admin.resetKeyboardShortcutsHelper",
  305. {
  306. keyCode: 191, // '/' key
  307. ctrl: true,
  308. shift: true,
  309. preventDefault: true,
  310. handler: () => {
  311. this.resetKeyboardShortcutsHelper();
  312. }
  313. }
  314. );
  315. },
  316. beforeUnmount() {
  317. this.socket.dispatch("apis.leaveRooms");
  318. window.removeEventListener("scroll", this.calculateSidebarPadding);
  319. const shortcutNames = [
  320. "admin.toggleKeyboardShortcutsHelper",
  321. "admin.resetKeyboardShortcutsHelper"
  322. ];
  323. shortcutNames.forEach(shortcutName => {
  324. keyboardShortcuts.unregisterShortcut(shortcutName);
  325. });
  326. },
  327. methods: {
  328. onRouteChange() {
  329. if (this.currentTab.startsWith("songs")) {
  330. this.toggleChildren({ child: "songs", force: false });
  331. } else if (this.currentTab.startsWith("users")) {
  332. this.toggleChildren({ child: "users", force: false });
  333. }
  334. this.currentTab = this.getTabFromPath();
  335. if (this.$refs[`${this.currentTab}-tab`])
  336. this.$refs[`${this.currentTab}-tab`].scrollIntoView({
  337. inline: "center",
  338. block: "nearest"
  339. });
  340. localStorage.setItem("lastAdminPage", this.currentTab);
  341. if (this.currentTab.startsWith("songs"))
  342. this.toggleChildren({ child: "songs", force: true });
  343. else if (this.currentTab.startsWith("users"))
  344. this.toggleChildren({ child: "users", force: true });
  345. },
  346. toggleKeyboardShortcutsHelper() {
  347. this.$refs.keyboardShortcutsHelper.toggleBox();
  348. },
  349. resetKeyboardShortcutsHelper() {
  350. this.$refs.keyboardShortcutsHelper.resetBox();
  351. },
  352. toggleSidebar() {
  353. this.sidebarActive = !this.sidebarActive;
  354. localStorage.setItem("admin-sidebar-active", this.sidebarActive);
  355. },
  356. getTabFromPath(path) {
  357. const localPath = path || this.$route.path;
  358. return localPath.substr(0, 7) === "/admin/"
  359. ? localPath.substr(7, localPath.length)
  360. : null;
  361. },
  362. calculateSidebarPadding() {
  363. const scrollTop =
  364. document.documentElement.scrollTop || document.scrollTop || 0;
  365. if (scrollTop <= 64) this.sidebarPadding = 64 - scrollTop;
  366. else this.sidebarPadding = 0;
  367. },
  368. ...mapActions("admin", ["toggleChildren"])
  369. }
  370. };
  371. </script>
  372. <style lang="less" scoped>
  373. .night-mode {
  374. .main-container .admin-area .admin-sidebar .inner {
  375. .top {
  376. background-color: var(--dark-grey-3);
  377. }
  378. .bottom {
  379. background-color: var(--dark-grey-2);
  380. .sidebar-item {
  381. background-color: var(--dark-grey-2);
  382. border-color: var(--dark-grey-3);
  383. &,
  384. &.with-children .sidebar-item-child,
  385. &.with-children > span > a {
  386. color: var(--white);
  387. }
  388. }
  389. }
  390. }
  391. }
  392. .main-container {
  393. height: auto;
  394. .admin-area {
  395. display: flex;
  396. flex-direction: column;
  397. min-height: 100vh;
  398. :deep(.nav) {
  399. .nav-menu.is-active {
  400. left: 45px;
  401. }
  402. &.admin-sidebar-active .nav-menu.is-active {
  403. left: 200px;
  404. }
  405. }
  406. .admin-sidebar {
  407. display: flex;
  408. min-width: 200px;
  409. width: 200px;
  410. @media screen and (max-width: 768px) {
  411. min-width: 45px;
  412. width: 45px;
  413. }
  414. .inner {
  415. display: flex;
  416. flex-direction: column;
  417. max-height: 100vh;
  418. width: 200px;
  419. position: sticky;
  420. top: 0;
  421. bottom: 0;
  422. left: 0;
  423. z-index: 5;
  424. box-shadow: @box-shadow;
  425. .bottom {
  426. overflow-y: auto;
  427. height: 100%;
  428. max-height: 100%;
  429. display: flex;
  430. flex-direction: column;
  431. flex: 1 0 auto;
  432. background-color: var(--white);
  433. .sidebar-item {
  434. display: flex;
  435. padding: 0 20px;
  436. line-height: 40px;
  437. font-size: 16px;
  438. font-weight: 600;
  439. color: var(--primary-color);
  440. background-color: var(--white);
  441. border-bottom: 1px solid var(--light-grey-2);
  442. transition: filter 0.2s ease-in-out;
  443. & > .material-icons {
  444. line-height: 40px;
  445. margin-right: 5px;
  446. }
  447. &:hover,
  448. &:focus,
  449. &.router-link-active,
  450. &.is-active {
  451. filter: brightness(95%);
  452. }
  453. &.toggle-sidebar {
  454. cursor: pointer;
  455. font-weight: 400;
  456. }
  457. &.with-children {
  458. flex-direction: column;
  459. & > span {
  460. display: flex;
  461. line-height: 40px;
  462. cursor: pointer;
  463. & > a {
  464. display: flex;
  465. }
  466. & > .material-icons,
  467. & > a > .material-icons {
  468. line-height: 40px;
  469. margin-right: 5px;
  470. }
  471. }
  472. .toggle-sidebar-children {
  473. margin-left: auto;
  474. }
  475. .sidebar-item-children {
  476. display: none;
  477. }
  478. &.is-active .sidebar-item-children {
  479. display: flex;
  480. flex-direction: column;
  481. .sidebar-item-child {
  482. display: flex;
  483. flex-direction: column;
  484. margin-left: 30px;
  485. font-size: 14px;
  486. line-height: 30px;
  487. position: relative;
  488. &::before {
  489. content: "";
  490. position: absolute;
  491. width: 1px;
  492. height: 30px;
  493. top: 0;
  494. left: -20px;
  495. background-color: var(--light-grey-3);
  496. }
  497. &:last-child::before {
  498. height: 16px;
  499. }
  500. &::after {
  501. content: "";
  502. position: absolute;
  503. width: 15px;
  504. height: 1px;
  505. top: 15px;
  506. left: -20px;
  507. background-color: var(--light-grey-3);
  508. }
  509. &.router-link-active {
  510. filter: brightness(95%);
  511. }
  512. }
  513. }
  514. }
  515. }
  516. }
  517. }
  518. &.minimised {
  519. min-width: 45px;
  520. width: 45px;
  521. .inner {
  522. max-width: 45px;
  523. .top {
  524. justify-content: center;
  525. .full-logo {
  526. display: none;
  527. }
  528. .minimised-logo {
  529. display: flex;
  530. }
  531. }
  532. .sidebar-item {
  533. justify-content: center;
  534. padding: 0;
  535. & > span {
  536. display: none;
  537. }
  538. }
  539. }
  540. }
  541. }
  542. .admin-content {
  543. display: flex;
  544. flex-direction: row;
  545. flex-grow: 1;
  546. .admin-container {
  547. display: flex;
  548. flex-direction: column;
  549. flex-grow: 1;
  550. overflow: hidden;
  551. :deep(.admin-tab-container) {
  552. display: flex;
  553. flex-direction: column;
  554. flex: 1 0 auto;
  555. padding: 10px 10px 20px 10px;
  556. .admin-tab {
  557. max-width: 1900px;
  558. margin: 0 auto;
  559. padding: 0 10px;
  560. }
  561. .admin-tab,
  562. .container {
  563. .button-row {
  564. display: flex;
  565. flex-direction: row;
  566. flex-wrap: wrap;
  567. justify-content: center;
  568. margin-bottom: 5px;
  569. & > .button,
  570. & > span {
  571. margin: 5px 0;
  572. &:not(:first-child) {
  573. margin-left: 5px;
  574. }
  575. }
  576. }
  577. }
  578. }
  579. }
  580. }
  581. }
  582. }
  583. :deep(.container) {
  584. position: relative;
  585. }
  586. :deep(.box) {
  587. box-shadow: @box-shadow;
  588. display: block;
  589. &:not(:last-child) {
  590. margin-bottom: 20px;
  591. }
  592. }
  593. #keyboardShortcutsHelper {
  594. .box-body {
  595. .biggest {
  596. font-size: 18px;
  597. }
  598. .bigger {
  599. font-size: 16px;
  600. }
  601. span {
  602. display: block;
  603. }
  604. }
  605. }
  606. @media screen and (min-width: 980px) {
  607. :deep(.container) {
  608. margin: 0 auto;
  609. max-width: 960px;
  610. }
  611. }
  612. @media screen and (min-width: 1180px) {
  613. :deep(.container) {
  614. max-width: 1200px;
  615. }
  616. }
  617. </style>