ClassFilter.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. <template>
  2. <div class="flex gap-2">
  3. <select
  4. v-if="showType"
  5. v-model="ruleset.column"
  6. class="w-2/6 md:w-3/12 h-10 px-2"
  7. name="searchType"
  8. @change="filterTypeChanged"
  9. >
  10. <option :value="ClassSelectorColumn.Module" default>Modul</option>
  11. <option :value="ClassSelectorColumn.Degree" default>Studiengang</option>
  12. <option :value="ClassSelectorColumn.Room">Raum</option>
  13. <option :value="ClassSelectorColumn.Class">Klasse</option>
  14. <option :value="ClassSelectorColumn.Lecturer">Dozent</option>
  15. <option :value="ClassSelectorColumn.Time">Zeit</option>
  16. <option :value="ClassSelectorColumn.TeachingType">Art</option>
  17. </select>
  18. <div class="w-full justify-center">
  19. <div
  20. v-if="ruleset.column == ClassSelectorColumn.Time"
  21. class="w-full gap-2 grid grid-cols-1"
  22. >
  23. <div class="flex items-center">
  24. <span class="hidden md:inline w-12">Von:</span>
  25. <WeekdayTimePicker
  26. v-model="(ruleset.filterData as Record<string, number>).startTime"
  27. :disable-greater-than="disableUpper"
  28. class="w-full h-10"
  29. />
  30. </div>
  31. <div class="flex items-center">
  32. <span class="hidden md:inline w-12">Bis:</span>
  33. <WeekdayTimePicker
  34. v-model="(ruleset.filterData as Record<string, number>).endTime"
  35. :disable-lower-than="disableLower"
  36. class="w-full h-10"
  37. />
  38. </div>
  39. </div>
  40. <select
  41. v-else-if="ruleset.column == ClassSelectorColumn.Degree"
  42. id="degreeSelector"
  43. v-model="ruleset.filterData.degree"
  44. class="w-full h-10"
  45. @change="
  46. () =>
  47. stateStore.updateLastSelectedDegreeProgram(
  48. ruleset.filterData.degree ?? '',
  49. )
  50. "
  51. >
  52. <option default value="">Alle</option>
  53. <option
  54. v-for="name in classesStore.degreePrograms"
  55. :key="name"
  56. :value="name"
  57. >
  58. {{ name }}
  59. </option>
  60. </select>
  61. <select
  62. v-else-if="ruleset.column == ClassSelectorColumn.TeachingType"
  63. id="teachingTypeSelector"
  64. v-model="ruleset.filterData.teachingType"
  65. class="w-full capitalize h-10"
  66. >
  67. <option default value="">Alle</option>
  68. <option v-for="name in TeachingType" :key="name" :value="name">
  69. {{ name }}
  70. </option>
  71. </select>
  72. <input
  73. v-else
  74. v-model="ruleset.filterData.term"
  75. class="w-full h-10"
  76. name="search"
  77. type="text"
  78. :placeholder="searchPlaceholderText"
  79. autocomplete="off"
  80. />
  81. </div>
  82. <div
  83. v-if="showAddRemove"
  84. class="w-32 h-10 flex justify-center items-center"
  85. >
  86. <button
  87. class="action-button"
  88. title="Neuen Filter nach diesem einfügen"
  89. @click="$emit('addFilter', ruleset)"
  90. >
  91. <font-awesome-icon icon="fa-solid fa-plus" />
  92. </button>
  93. <button
  94. :disabled="isLastFilter"
  95. class="action-button"
  96. title="Diesen Filter entfernen"
  97. @click="$emit('removeFilter', ruleset)"
  98. >
  99. <font-awesome-icon icon="fa-solid fa-minus" />
  100. </button>
  101. <div
  102. title="Filtering ein/ausschalten"
  103. class="action-checkbox"
  104. @click="filterEnabledClick(false)"
  105. >
  106. <input
  107. v-model="ruleset.enabled"
  108. type="checkbox"
  109. @change="filterEnabledClick(true)"
  110. />
  111. </div>
  112. </div>
  113. </div>
  114. </template>
  115. <script lang="ts">
  116. import { PropType } from "vue";
  117. import { useClassesStore } from "../../stores/classes";
  118. import { useStateStore } from "../../stores/state";
  119. import { ClassSelectorColumn, FilterRule, TeachingType } from "../../types";
  120. import WeekdayTimePicker from "../general/WeekdayTimePicker.vue";
  121. export default {
  122. name: "ClassFilter",
  123. components: { WeekdayTimePicker },
  124. props: {
  125. modelValue: {
  126. required: true,
  127. type: Object as PropType<FilterRule>,
  128. },
  129. isLastFilter: {
  130. required: false,
  131. type: Boolean,
  132. default: true,
  133. },
  134. showType: {
  135. required: false,
  136. type: Boolean,
  137. default: true,
  138. },
  139. showAddRemove: {
  140. required: false,
  141. type: Boolean,
  142. default: true,
  143. },
  144. },
  145. emits: ["update:modelValue", "addFilter", "removeFilter"],
  146. setup() {
  147. const classesStore = useClassesStore();
  148. const stateStore = useStateStore();
  149. return {
  150. ClassSelectorColumn,
  151. classesStore,
  152. stateStore,
  153. TeachingType,
  154. };
  155. },
  156. data() {
  157. return {
  158. ruleset: this.modelValue,
  159. };
  160. },
  161. computed: {
  162. disableUpper(): number {
  163. return Math.floor(
  164. ((this.ruleset.filterData as Record<string, number>).endTime ??
  165. 86400 * 7) / 86400,
  166. );
  167. },
  168. disableLower(): number {
  169. return Math.floor(
  170. ((this.ruleset.filterData as Record<string, number>).startTime ?? 0) /
  171. 86400,
  172. );
  173. },
  174. searchPlaceholderText(): string {
  175. switch (this.ruleset.column) {
  176. case ClassSelectorColumn.Module:
  177. return "Suche nach Modulkürzel oder Name";
  178. case ClassSelectorColumn.Room:
  179. return "Suche nach Raumbezeichnung";
  180. case ClassSelectorColumn.Class:
  181. return "Suche nach Klassenbezeichnung";
  182. case ClassSelectorColumn.Lecturer:
  183. return "Suche nach Dozentenkürzel";
  184. case ClassSelectorColumn.Time:
  185. break;
  186. }
  187. return "";
  188. },
  189. },
  190. methods: {
  191. filterEnabledClick(direct: boolean) {
  192. if (!direct) {
  193. // Someone pressed the box around the checkbox.
  194. this.ruleset.enabled = !this.ruleset.enabled;
  195. } else {
  196. // Only notify changes when the value change is detected directly
  197. this.filterChanged();
  198. }
  199. },
  200. filterChanged() {
  201. this.$emit("update:modelValue", this.ruleset);
  202. },
  203. filterTypeChanged() {
  204. if (this.ruleset.column == ClassSelectorColumn.Degree) {
  205. if (!this.ruleset.filterData.degree)
  206. this.ruleset.filterData.degree = this.classesStore.degreePrograms[0];
  207. }
  208. if (this.ruleset.column == ClassSelectorColumn.TeachingType) {
  209. if (!this.ruleset.filterData.teachingType)
  210. this.ruleset.filterData.teachingType = "";
  211. }
  212. },
  213. },
  214. };
  215. </script>
  216. <style scoped>
  217. @reference "../../style.css";
  218. .action-button {
  219. @apply h-5
  220. text-xs
  221. md:h-7
  222. md:text-base
  223. mx-0.5
  224. md:mx-1
  225. block
  226. bg-gray-200
  227. hover:bg-gray-300
  228. active:bg-gray-400
  229. disabled:bg-gray-200
  230. disabled:text-gray-400
  231. dark:bg-gray-700
  232. dark:hover:bg-gray-600
  233. dark:active:bg-gray-700
  234. dark:disabled:bg-gray-800
  235. dark:disabled:text-gray-600
  236. disabled:pointer-events-none
  237. aspect-square
  238. rounded-sm
  239. transition-all
  240. duration-200;
  241. }
  242. .action-checkbox {
  243. @apply h-5
  244. text-xs
  245. m-2
  246. md:h-7
  247. md:text-base
  248. mx-0.5
  249. md:mx-1
  250. bg-gray-200
  251. hover:bg-gray-300
  252. active:bg-gray-400
  253. disabled:bg-gray-200
  254. disabled:text-gray-400
  255. dark:bg-gray-700
  256. dark:hover:bg-gray-600
  257. dark:active:bg-gray-700
  258. dark:disabled:bg-gray-800
  259. dark:disabled:text-gray-600
  260. disabled:pointer-events-none
  261. aspect-square
  262. rounded-sm
  263. transition-all
  264. duration-200
  265. items-center
  266. flex
  267. cursor-pointer;
  268. }
  269. .action-checkbox input {
  270. @apply cursor-pointer;
  271. }
  272. </style>