LibraryOptions.vue 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <template>
  2. <div class="library-options">
  3. <b-form-group
  4. class="mode-select-group"
  5. :label="$t('options.display.title')"
  6. v-slot="{ ariaDescribedby }"
  7. >
  8. <b-form-radio-group
  9. id="mode-select"
  10. v-model="activeMode" :options="modes"
  11. name="mode-select" :aria-describedby="ariaDescribedby"
  12. buttons button-variant="outline-dark"
  13. />
  14. <b-button
  15. v-if="activeMode === 'map'"
  16. variant="outline-dark" class="d-block mt-2"
  17. @click="$store.dispatch('ROLL_RANDOM_NODES')"
  18. >
  19. {{ $t('options.display.shuffle') }}
  20. </b-button>
  21. <b-dropdown
  22. v-if="activeMode === 'list'"
  23. :text="$t('options.display.go-to-char')"
  24. variant="outline-dark" class="char-select d-block mt-2"
  25. dropright
  26. >
  27. <b-dropdown-item-button
  28. class="text-center"
  29. v-for="char in availableCharacters" :key="char"
  30. @click="scrollToChar(char)"
  31. >
  32. {{ char }}
  33. </b-dropdown-item-button>
  34. </b-dropdown>
  35. </b-form-group>
  36. <!-- LIST + MAP ONLY -->
  37. <b-form-group
  38. class="filters-group"
  39. v-if="mode !== 'tree'"
  40. :label="$t('options.filters.title')"
  41. >
  42. <div class="filters-mini-group mb-1">
  43. <b-form-checkbox
  44. v-model="tagsIsActive" name="tags-button"
  45. button class="toggle-btn"
  46. :class="tagsIsActive ? null : 'collapsed'"
  47. :aria-expanded="tagsIsActive ? 'true' : 'false'"
  48. aria-controls="tags-select"
  49. button-variant="outline-dark"
  50. >
  51. {{ $t('options.filters.choices.tags') }}
  52. </b-form-checkbox>
  53. <multiple-tags-select
  54. v-if="tagsIsActive"
  55. id="tags-select" :button-text="$t('options.filters.add')"
  56. v-model="activeTags" :options="tagsOptions"
  57. />
  58. </div>
  59. <div class="filters-mini-group align-items-center">
  60. <b-form-checkbox
  61. v-model="strangenessIsActive" name="tags-button"
  62. button class="toggle-btn"
  63. :class="strangenessIsActive ? null : 'collapsed'"
  64. :aria-expanded="strangenessIsActive ? 'true' : 'false'"
  65. aria-controls="strangeness-input"
  66. button-variant="outline-dark"
  67. >
  68. {{ $t('options.filters.choices.strangeness') }}
  69. </b-form-checkbox>
  70. <div v-if="strangenessIsActive" class="strangeness-input-wrapper">
  71. <b-form-input
  72. id="strangeness-input"
  73. type="range" min="1" max="5"
  74. v-model="activeStrangeness"
  75. />
  76. </div>
  77. </div>
  78. </b-form-group>
  79. <!-- LIST + MAP ONLY -->
  80. <search-input v-if="mode !== 'tree'" v-model="activeSearch" />
  81. <!-- TREE ONLY -->
  82. <b-form-group
  83. v-else
  84. label="Texte de départ :"
  85. label-for="text-depart-select"
  86. >
  87. <b-form-select
  88. id="text-depart-select"
  89. v-model="activeNodeId"
  90. :options="nodesDepartsOptions"
  91. class="border"
  92. />
  93. </b-form-group>
  94. </div>
  95. </template>
  96. <script>
  97. import { mapGetters } from 'vuex'
  98. import { MultipleTagsSelect, SearchInput } from '@/components/formItems'
  99. export default {
  100. name: 'LibraryOptions',
  101. components: {
  102. MultipleTagsSelect,
  103. SearchInput
  104. },
  105. data () {
  106. return {
  107. modes: [
  108. { text: this.$t('options.display.choices.tree-map'), value: 'tree' },
  109. { text: this.$t('options.display.choices.card-map'), value: 'map' },
  110. { text: this.$t('options.display.choices.card-list'), value: 'list' }
  111. ],
  112. selectedTags: [],
  113. tagsIsActive: false,
  114. strangenessIsActive: false
  115. }
  116. },
  117. computed: {
  118. ...mapGetters([
  119. 'nodesDepartsOptions',
  120. 'orderedTextsDepart',
  121. 'tagsOptions',
  122. 'nodebook',
  123. 'mode',
  124. 'nodeDepartId',
  125. 'search',
  126. 'tags',
  127. 'strangeness'
  128. ]),
  129. availableCharacters () {
  130. if (!this.orderedTextsDepart) return []
  131. return this.orderedTextsDepart.map(group => group[0])
  132. },
  133. activeNodeId: {
  134. get () { return this.nodeDepartId },
  135. set (value) {
  136. this.$store.dispatch('SET_NODE_DEPART_ID', value)
  137. }
  138. },
  139. activeMode: {
  140. get () { return this.mode },
  141. set (value) {
  142. this.$store.dispatch('UPDATE_QUERY_MODE', value)
  143. }
  144. },
  145. activeSearch: {
  146. get () { return this.search },
  147. set (value) {
  148. this.$store.commit('SET_SEARCH', value)
  149. }
  150. },
  151. activeTags: {
  152. get () { return this.tags },
  153. set (value) {
  154. this.$store.commit('UPDATE_TAGS', value)
  155. }
  156. },
  157. activeStrangeness: {
  158. get () { return this.strangeness },
  159. set (value) {
  160. this.$store.commit('UPDATE_STRANGENESS', parseInt(value))
  161. }
  162. }
  163. },
  164. watch: {
  165. strangenessIsActive (activate) {
  166. if (activate) {
  167. this.$store.commit('UPDATE_STRANGENESS', 1)
  168. } else {
  169. this.$store.commit('UPDATE_STRANGENESS', undefined)
  170. }
  171. },
  172. tagsIsActive () {
  173. this.$store.commit('UPDATE_TAGS', [])
  174. }
  175. },
  176. methods: {
  177. scrollToChar (char) {
  178. const h = document.querySelector(`h3[id=${char}]`)
  179. if (h) {
  180. h.scrollIntoView({ behavior: 'smooth' })
  181. }
  182. }
  183. }
  184. }
  185. </script>
  186. <style lang="scss" scoped>
  187. .library-options {
  188. padding: 0 $header-spacer-sm $header-spacer-sm;
  189. background-color: $body-bg;
  190. width: 100%;
  191. @include media-breakpoint-down($layout-bp-down) {
  192. font-size: 0.9rem;
  193. ::v-deep {
  194. .btn, .form-control, .input-group-text {
  195. font-size: inherit;
  196. }
  197. }
  198. }
  199. @include media-breakpoint-up($size-bp) {
  200. padding: 0 $header-spacer $header-spacer;
  201. display: flex;
  202. justify-content: space-between;
  203. }
  204. ::v-deep {
  205. @include media-breakpoint-up($size-bp) {
  206. flex-wrap: wrap;
  207. fieldset {
  208. margin-bottom: 0;
  209. &:last-child {
  210. flex-shrink: 2;
  211. }
  212. }
  213. }
  214. > :last-child {
  215. margin-bottom: 0;
  216. }
  217. .mode-select-group {
  218. > div {
  219. line-height: 0;
  220. }
  221. label {
  222. border-color: transparent;
  223. &:not(:disabled):not(.disabled):active,
  224. &:not(:disabled):not(.disabled).active {
  225. background-color: transparent;
  226. color: $black;
  227. }
  228. }
  229. .char-select {
  230. line-height: 1;
  231. .dropdown-menu {
  232. max-height: 200px;
  233. overflow: auto;
  234. min-width: 3rem;
  235. }
  236. }
  237. }
  238. .filters-group {
  239. flex-grow: 2;
  240. @include media-breakpoint-up($size-bp) {
  241. padding: 0 2rem;
  242. max-width: 40%;
  243. }
  244. .filters-mini-group {
  245. display: flex;
  246. line-height: 0;
  247. }
  248. .toggle-btn label {
  249. border-color: transparent;
  250. margin-right: .5rem;
  251. &:not(:disabled):not(.disabled):active,
  252. &:not(:disabled):not(.disabled).active {
  253. border-color: $black;
  254. background-color: transparent;
  255. color: $black;
  256. }
  257. }
  258. .strangeness-input-wrapper {
  259. min-width: 150px;
  260. &::before,
  261. &::after {
  262. content: '';
  263. display: inline-block;
  264. width: 3px;
  265. height: 20px;
  266. background-color: $black;
  267. position: absolute;
  268. }
  269. &::after {
  270. margin-left: -2px;
  271. }
  272. }
  273. }
  274. }
  275. }
  276. </style>