
.page-link {
    color: black;
}
.page-item.active .page-link {
    background-color: #6c757d;
    border-color: #6c757d;
}
.border-3 {
    border-width: 3px !important;
}


/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./frontend/vue/components/SearchableObjectsTable.vue"],"names":[],"mappings":";AAkUA;IACA;AACA;AACA;IACA;IACA;AACA;AACA;IACA;AACA","file":"css/vue-0ebcfddd8a8328422592.css","sourcesContent":["<template>\n<div class=\"mt-0\">\n  <b-card no-body>\n    <b-tabs\n      v-model=\"searchTabIndex\"\n      pills card\n      vertical\n      >\n      <b-tab title=\"Basic Search\" :title-link-class=\"linkClass(0)\">\n        <b-form-group>\n          <b-input-group>\n            <b-form-input\n              v-model=\"basicFilter\"\n              debounce=\"250\"\n              id=\"basicfilter\"\n              :placeholder=\"basicSearchPlaceholder\"\n              >\n            </b-form-input>\n            <b-input-group-append size=\"sm\">\n              <b-button\n                :disabled=\"!basicFilter\"\n                @click=\"basicFilter = ''\"\n                >\n                Clear\n              </b-button>\n            </b-input-group-append>\n          </b-input-group>\n        </b-form-group>\n      </b-tab>\n      <b-tab title=\"Advanced Query\" :title-link-class=\"linkClass(1)\">\n        <div>\n          <b-form-group\n            v-for=\"filteringField in filteringFieldsAdvanced\"\n            :key=\"filteringField.field\"\n            >\n            <b-input-group>\n              <template v-slot:prepend>\n                <b-input-group-text><strong>{{ filteringField.label }}</strong></b-input-group-text>\n                <b-input-group-text class=\"bg-light\"><em>{{ filteringField.lookup }}</em></b-input-group-text>\n              </template>\n              <b-form-input\n                v-model=\"filteringField.filter\"\n                debounce=\"250\"\n                id=\"filteringField.key\"\n                >\n              </b-form-input>\n              <b-input-group-append size=\"sm\">\n                <b-button\n                  :disabled=\"!filteringField.filter\"\n                  @click=\"filteringField.filter = ''\"\n                  >\n                  Clear\n                </b-button>\n              </b-input-group-append>\n            </b-input-group>\n          </b-form-group>\n        </div>\n      </b-tab>\n    </b-tabs>\n  </b-card>\n  <b-row :v-if=\"paginated\" class=\"mt-4\">\n    <b-col class=\"col-2\">\n      <b-badge variant=\"secondary\" class=\"mt-2 px-3 py-2\">{{ totalRows }}&emsp;item<span v-if=\"totalRows != 1\">s</span></b-badge>\n    </b-col>\n    <b-col class=\"col-8\">\n      <b-pagination\n        v-model=\"currentPage\"\n        :total-rows=\"totalRows\"\n        :per-page=\"perPage\"\n        class=\"m-2\"\n        align=\"center\"\n        aria-controls=\"objects-table\"\n        >\n      </b-pagination>\n    </b-col>\n    <b-col class=\"col-2\">\n      <item-detail-links\n        :url_spx_api=\"queryURL\"\n        >\n      </item-detail-links>\n    </b-col>\n  </b-row>\n  <br>\n  <b-table\n    id=\"objects-table\"\n    show-empty\n    :items=\"objectsProvider\"\n    :fields=\"computedDisplayFields\"\n    :tbody-tr-class=\"tbodyTrClass\"\n    :per-page=\"perPage\"\n    :current-page=\"currentPage\"\n    @row-clicked=\"onRowClicked\"\n    >\n    <template v-slot:table-busy>\n      <div class=\"text-center text-primary my-2\">\n        <b-spinner class=\"align-middle\"></b-spinner>\n        <strong>Loading...</strong>\n      </div>\n    </template>\n    <template v-slot:cell(spxlinks)=\"row\">\n      <item-detail-links\n        :url_spx=\"row.item.url_spx\"\n        :url_spx_api=\"row.item.url_spx_api\"\n        >\n      </item-detail-links>\n    </template>\n    <template v-slot:row-details=\"row\">\n      <object-row-details\n        :object_type=\"object_type\"\n        :object=\"row.item\"\n        >\n      </object-row-details>\n    </template>\n  </b-table>\n</div>\n</template>\n\n<script>\nconst headers = new Headers();\nheaders.append('Accept', 'application/json; version=0')\n\nimport ItemDetailLinks from './ItemDetailLinks.vue'\nimport ObjectRowDetails from './ObjectRowDetails/ObjectRowDetails.vue'\n\nexport default {\n    name: \"searchable-objects-table\",\n    components: {\n        ItemDetailLinks,\n        ObjectRowDetails,\n    },\n    props: {\n        object_type: {\n            type: String,\n            required: true\n        },\n        displayfields: {\n            type: Array,\n            required: true\n        },\n        paginated: true,\n        numbered: false,\n        url: {\n            type: String,\n            required: true\n        },\n        initial_filter: {\n            type: String,\n            required: false\n        },\n        excluded_filter_fields: {\n            type: Array,\n            required: false\n        },\n    },\n    data() {\n        return {\n            searchTabIndex: 0,\n            basicFilter: null,\n            filteringFieldsBasic: [],\n            filteringFieldsAdvanced: [],\n            objects: [],\n            perPage: 10,\n            currentPage: 1,\n            totalRows: 1,\n            filter: null,\n            filterOn: [],\n        }\n    },\n    \n    computed: {\n        basicSearchPlaceholder: function() {\n            var placeholder = 'Search in: '\n            var counter = 0\n            this.filteringFieldsBasic.forEach(\n                (filteringField) => {\n                    if (counter > 0) {\n                        placeholder += ', '\n                    }\n                    counter += 1\n                    placeholder += filteringField\n                }\n            )\n            return placeholder\n        },\n        computedDisplayFields: function () {\n            let cdf = this.displayfields.slice()\n            cdf.push({ key: 'spxlinks', label: ''})\n            if (this.numbered) {\n                //this.displayfields.unshift({ key: '_number', label: ''})\n                cdf.unshift({ key: '_number', label: ''})\n            }\n            return cdf\n        },\n        queryParameters: function() {\n            var parameters = ''\n            if (this.searchTabIndex == 0) { // basic search\n                if (this.basicFilter) {\n                    // this.filteringFields.forEach((filteringField) => {\n                    //     if (filteringField.isDefault) {\n                    //         parameters += ('&' + filteringField.key + '__'\n                    //                        + filteringField.lookup + '=' + this.basicFilter)\n                    //     }\n                    // })\n                    parameters += '&search=' + this.basicFilter\n                }\n            }\n            else {\n                this.filteringFieldsAdvanced.forEach((filteringField) => {\n                    if (filteringField.filter) {\n                        parameters += ('&' + filteringField.key + '__'\n                                       + filteringField.lookup + '=' + filteringField.filter)\n                    }\n                })\n            }\n            if (this.initial_filter) {\n                parameters += ('&' + this.initial_filter)\n            }\n            return parameters\n        },\n        queryURL: function() {\n            return ('/api/' + this.url + '/' + '?limit=' + this.perPage\n                    + '&offset=' + this.perPage * (this.currentPage - 1)\n                    + this.queryParameters)\n        }\n    },\n    methods: {\n        fetchFilteringFields () {\n            fetch('/api/' + this.object_type + '/filtering_options', {headers: headers})\n                .then(stream => stream.json())\n                .then(data => {\n                    this.filteringFieldsBasic = data.basic\n                    this.filteringFieldsAdvanced = data.advanced.map(\n                        function(option) {\n                            return {\n                                key: option.field,\n                                label: option.label,\n                                lookup: option.lookup,\n                                isDefault: option.default,\n                                filter: null\n                            }\n                        })\n                    if (this.excluded_filter_fields) {\n                        this.filteringFieldsBasic = this.filteringFieldsBasic.filter(\n                            option => {\n                                return !(this.excluded_filter_fields.includes(option))\n                            })\n                        this.filteringFieldsAdvanced = this.filteringFieldsAdvanced.filter(\n                            option => {\n                                return !(this.excluded_filter_fields.includes(option.label))\n                            })\n                    }\n                })\n                .catch(error => console.error(error))\n        },\n        fetchObjects () {\n            fetch('/api/' + this.url, {headers: headers})\n                .then(stream => stream.json())\n                .then(data => this.objects = data.results || data) // to handle paginated or flat list, or empty list\n                .catch(error => console.error(error))\n        },\n        linkClass(idx) {\n            if (this.searchTabIndex === idx) {\n                return ['mx-2', 'bg-secondary', 'text-white', 'font-weight-bold']\n            } else {\n                return ['mx-2', 'bg-light', 'text-dark']\n            }\n        },\n        objectsProvider (ctx) {\n            // Our API uses limit/offset pagination\n            var params = '?limit=' + ctx.perPage + '&offset=' + ctx.perPage * (ctx.currentPage - 1) + this.queryParameters\n            const promise = fetch('/api/' + this.url + '/' + params, {headers: headers})\n            return promise.then(stream => stream.json())\n                .then(data => {\n                    var items = data.results || data // to handle paginated or flat list\n                    this.totalRows = data.count || items.length\n                    items = items.map(function(item, index) {\n                        item['_number'] = index + 1\n                        item['_showDetails'] = true  // needed to enable the row-details prop\n                        return item\n                    })\n                    return items || []\n                })\n        },\n        async onRowClicked (item, index) {\n            this.item = await item\n            if (this.object_type == \"affiliations\") {\n                window.open(\"/persons/\" + this.item.person, \"_self\")\n            } else if (this.object_type == \"publications\") {\n                window.open(\"/sources/\" + item.uuid, \"_self\")\n            } else if (this.object_type == \"sources\") {\n                window.open(\"/sources/\" + item.uuid, \"_self\")\n            } else {\n                window.open('/' + this.url + '/' + item.uuid, '_self')\n            }\n        },\n        tbodyTrClass (item, type) {\n            if (type == 'row') {\n                return 'bg-light border-top border-dark border-3'\n            }\n            else if (type == 'row-details') {\n                return ''\n             }\n         }\n     },\n     mounted () {\n         this.fetchFilteringFields()\n         this.fetchObjects()\n         if (this.numbered) {\n             this.displayfields.unshift({ key: '_number', label: ''})\n         }\n         this.displayfields.push({ key: 'spxlinks', label: ''})\n     },\n     watch: {\n         queryParameters: function () {\n             this.$root.$emit('bv::refresh::table', 'objects-table')\n         }\n     }\n }\n</script>\n\n\n<style>\n  .page-link {\n      color: black;\n  }\n  .page-item.active .page-link {\n      background-color: #6c757d;\n      border-color: #6c757d;\n  }\n  .border-3 {\n      border-width: 3px !important;\n  }\n</style>\n"],"sourceRoot":""}*/