Skip to content
Snippets Groups Projects
Radios.vue 7.96 KiB
Newer Older
<template>
  <main v-title="labels.title">
    <section class="ui vertical stripe segment">
Eliot Berriot's avatar
Eliot Berriot committed
      <h2 class="ui header">
Georg Krause's avatar
Georg Krause committed
        <translate translate-context="Content/Radio/Title">
          Browsing radios
        </translate>
Eliot Berriot's avatar
Eliot Berriot committed
      </h2>
Georg Krause's avatar
Georg Krause committed
      <div class="ui hidden divider" />
      <div class="ui row">
        <h3 class="ui header">
Georg Krause's avatar
Georg Krause committed
          <translate translate-context="Content/Radio/Title">
            Instance radios
          </translate>
        </h3>
        <div class="ui cards">
Georg Krause's avatar
Georg Krause committed
          <radio-card
            v-if="isAuthenticated"
            :type="'actor-content'"
            :object-id="$store.state.auth.fullUsername"
          />
          <radio-card
            v-if="isAuthenticated && hasFavorites"
            :type="'favorites'"
          />
          <radio-card :type="'random'" />
          <radio-card :type="'recently-added'" />
          <radio-card
            v-if="$store.state.auth.authenticated"
            :type="'less-listened'"
          />
Georg Krause's avatar
Georg Krause committed
      <div class="ui hidden divider" />
      <h3 class="ui header">
Georg Krause's avatar
Georg Krause committed
        <translate translate-context="Content/Radio/Title">
          User radios
        </translate>
Georg Krause's avatar
Georg Krause committed
      <router-link
        v-if="isAuthenticated"
Georg Krause's avatar
Georg Krause committed
        class="ui success button"
        to="/library/radios/build"
        exact
      >
        <translate translate-context="Content/Radio/Button.Label/Verb">
          Create your own radio
        </translate>
Bat's avatar
Bat committed
      </router-link>
Georg Krause's avatar
Georg Krause committed
      <div class="ui hidden divider" />
      <form
        :class="['ui', {'loading': isLoading}, 'form']"
        @submit.prevent="updateQueryString();fetchData()"
      >
        <div class="fields">
          <div class="field">
            <label for="radios-search"><translate translate-context="Content/Search/Input.Label/Noun">Search</translate></label>
Georg Krause's avatar
Georg Krause committed
              <input
                id="radios-search"
                v-model="query"
                type="text"
                name="search"
                :placeholder="labels.searchPlaceholder"
              >
              <button
                class="ui icon button"
                type="submit"
                :aria-label="$pgettext('Content/Search/Input.Label/Noun', 'Search')"
              >
                <i class="search icon" />
          </div>
          <div class="field">
            <label for="radios-ordering"><translate translate-context="Content/Search/Dropdown.Label/Noun">Ordering</translate></label>
Georg Krause's avatar
Georg Krause committed
            <select
              id="radios-ordering"
              v-model="ordering"
              class="ui dropdown"
            >
              <option
                v-for="(option, key) in orderingOptions"
                :key="key"
                :value="option[0]"
              >
                {{ sharedLabels.filters[option[1]] }}
              </option>
            </select>
          </div>
          <div class="field">
            <label for="radios-ordering-direction"><translate translate-context="Content/Search/Dropdown.Label/Noun">Order</translate></label>
Georg Krause's avatar
Georg Krause committed
            <select
              id="radios-ordering-direction"
              v-model="orderingDirection"
              class="ui dropdown"
            >
Eliot Berriot's avatar
Eliot Berriot committed
              <option value="+">
Georg Krause's avatar
Georg Krause committed
                <translate translate-context="Content/Search/Dropdown">
                  Ascending
                </translate>
Eliot Berriot's avatar
Eliot Berriot committed
              </option>
              <option value="-">
Georg Krause's avatar
Georg Krause committed
                <translate translate-context="Content/Search/Dropdown">
                  Descending
                </translate>
Eliot Berriot's avatar
Eliot Berriot committed
              </option>
            </select>
          </div>
          <div class="field">
            <label for="radios-results"><translate translate-context="Content/Search/Dropdown.Label/Noun">Results per page</translate></label>
Georg Krause's avatar
Georg Krause committed
            <select
              id="radios-results"
              v-model="paginateBy"
              class="ui dropdown"
            >
              <option :value="parseInt(12)">
                12
              </option>
              <option :value="parseInt(25)">
                25
              </option>
              <option :value="parseInt(50)">
                50
              </option>
            </select>
          </div>
        </div>
Georg Krause's avatar
Georg Krause committed
      <div class="ui hidden divider" />
      <div
        v-if="result && !result.results.length > 0"
        class="ui placeholder segment"
      >
        <div class="ui icon header">
Georg Krause's avatar
Georg Krause committed
          <i class="feed icon" />
          <translate translate-context="Content/Radios/Placeholder">
            No results matching your query
          </translate>
        </div>
        <router-link
Georg Krause's avatar
Georg Krause committed
          v-if="$store.state.auth.authenticated"
          :to="{name: 'library.radios.build'}"
          class="ui success button labeled icon"
        >
          <i class="rss icon" />
          <translate translate-context="Content/*/Verb">
            Create a radio
          </translate>
        </router-link>
      </div>
Eliot Berriot's avatar
Eliot Berriot committed
      <div
        v-if="result && result.results.length > 0"
Georg Krause's avatar
Georg Krause committed
        class="ui cards"
      >
        <radio-card
          v-for="radio in result.results"
          :key="radio.id"
Georg Krause's avatar
Georg Krause committed
          type="custom"
          :custom-radio="radio"
        />
      </div>
      <div class="ui center aligned basic segment">
        <pagination
          v-if="result && result.count > paginateBy"
          :current="page"
          :paginate-by="paginateBy"
          :total="result.count"
Georg Krause's avatar
Georg Krause committed
          @page-changed="selectPage"
        />
      </div>
</template>

<script>
Georg Krause's avatar
Georg Krause committed
import axios from 'axios'
import $ from 'jquery'
Ciaran Ainsworth's avatar
Ciaran Ainsworth committed
import logger from '@/logging.js'
Georg Krause's avatar
Georg Krause committed
import OrderingMixin from '@/components/mixins/Ordering.vue'
import PaginationMixin from '@/components/mixins/Pagination.vue'
import TranslationsMixin from '@/components/mixins/Translations.vue'
import RadioCard from '@/components/radios/Card.vue'
import Pagination from '@/components/Pagination.vue'
Georg Krause's avatar
Georg Krause committed
const FETCH_URL = 'radios/radios/'

export default {
  components: {
    RadioCard,
    Pagination
  },
Georg Krause's avatar
Georg Krause committed
  mixins: [OrderingMixin, PaginationMixin, TranslationsMixin],
  props: {
    defaultQuery: { type: String, required: false, default: '' },
    scope: { type: String, required: false, default: 'all' }
  },
  data () {
    return {
      isLoading: true,
      result: null,
      page: parseInt(this.defaultPage),
      query: this.defaultQuery,
Georg Krause's avatar
Georg Krause committed
      orderingOptions: [['creation_date', 'creation_date'], ['name', 'name']]
Eliot Berriot's avatar
Eliot Berriot committed
  computed: {
Georg Krause's avatar
Georg Krause committed
    labels () {
      const searchPlaceholder = this.$pgettext('Content/Search/Input.Placeholder', 'Enter a radio name…')
      const title = this.$pgettext('*/*/*', 'Radios')
Eliot Berriot's avatar
Eliot Berriot committed
      return {
        searchPlaceholder,
        title
      }
    },
    isAuthenticated () {
      return this.$store.state.auth.authenticated
    },
    hasFavorites () {
      return this.$store.state.favorites.count > 0
Georg Krause's avatar
Georg Krause committed
    }
  },
  watch: {
    page () {
      this.updateQueryString()
      this.fetchData()
    }
  },
  created () {
    this.fetchData()
  },
  mounted () {
    $('.ui.dropdown').dropdown()
Eliot Berriot's avatar
Eliot Berriot committed
  },
  methods: {
Georg Krause's avatar
Georg Krause committed
    updateQueryString: function () {
      history.pushState(
        {},
        null,
        this.$route.path + '?' + new URLSearchParams(
          {
Georg Krause's avatar
Georg Krause committed
            query: this.query,
            page: this.page,
            paginateBy: this.paginateBy,
            ordering: this.getOrderingAsString()
          }).toString()
Georg Krause's avatar
Georg Krause committed
    fetchData: function () {
      const self = this
      this.isLoading = true
Georg Krause's avatar
Georg Krause committed
      const url = FETCH_URL
      const params = {
        page: this.page,
        page_size: this.paginateBy,
        name__icontains: this.query,
Georg Krause's avatar
Georg Krause committed
        ordering: this.getOrderingAsString()
Georg Krause's avatar
Georg Krause committed
      logger.default.debug('Fetching radios')
      axios.get(url, { params: params }).then(response => {
        self.result = response.data
        self.isLoading = false
      })
Georg Krause's avatar
Georg Krause committed
    selectPage: function (page) {
      this.page = page
    }
  }
}
</script>