traxxx/assets/components/actors/actors.vue

656 lines
14 KiB
Vue
Raw Normal View History

<template>
2020-05-17 23:22:56 +00:00
<div class="actors">
<nav class="filter">
<ul class="genders nolist">
2020-08-02 01:51:52 +00:00
<li class="gender">
<router-link
:to="{ name: 'actors', params: { gender: 'all', letter, pageNumber: 1 } }"
:class="{ selected: gender === 'all' }"
class="gender-link all"
>all</router-link>
</li>
2020-05-17 23:22:56 +00:00
<li class="gender">
<router-link
:to="{ name: 'actors', params: { gender: 'female', letter, pageNumber: 1 } }"
2020-05-17 23:22:56 +00:00
:class="{ selected: gender === 'female' }"
class="gender-link female"
><Gender gender="female" /></router-link>
</li>
<li class="gender">
<router-link
:to="{ name: 'actors', params: { gender: 'male', letter, pageNumber: 1 } }"
2020-05-17 23:22:56 +00:00
:class="{ selected: gender === 'male' }"
class="gender-link male"
replace
2020-05-17 23:22:56 +00:00
><Gender gender="male" /></router-link>
</li>
<li class="gender">
<router-link
:to="{ name: 'actors', params: { gender: 'trans', letter, pageNumber: 1 } }"
2020-05-17 23:22:56 +00:00
:class="{ selected: gender === 'trans' }"
class="gender-link transsexual"
replace
2020-05-17 23:22:56 +00:00
><Gender gender="transsexual" /></router-link>
</li>
<li class="gender">
<router-link
:to="{ name: 'actors', params: { gender: 'other', letter, pageNumber: 1 } }"
2020-05-17 23:22:56 +00:00
:class="{ selected: gender === 'other' }"
class="gender-link other"
replace
2020-05-17 23:22:56 +00:00
><Icon icon="question5" /></router-link>
</li>
</ul>
<ul class="letters nolist">
<li
v-for="letterX in letters"
:key="letterX"
class="letter"
>
<router-link
:to="{ name: 'actors', params: { gender, letter: letterX, pageNumber: 1 } }"
2020-05-17 23:22:56 +00:00
:class="{ selected: letterX === letter }"
class="letter-link"
replace
2020-05-17 23:22:56 +00:00
>{{ letterX || 'All' }}</router-link>
</li>
</ul>
</nav>
2021-02-28 02:38:54 +00:00
<nav class="filter">
<ul class="nolist">
<li>
<Tooltip class="filter boobs">
<span class="filter-trigger"><Icon icon="boobs" />Boobs</span>
<template v-slot:tooltip>
<div class="filter-section boobsize">
<label class="filter-label off noselect">
<span class="label">
<Checkbox
:checked="boobSizeRequired"
class="checkbox"
@change="(checked) => updateValue('boobSizeRequired', checked)"
/>Size
</span>
2021-02-28 02:38:54 +00:00
<span
v-if="boobSizeRequired"
class="label-values"
>({{ boobSize[0] }} - {{ boobSize[1] }})</span>
</label>
<span class="filter-split">
<Range
:min="0"
:max="boobSizes.length - 1"
:value="boobSize"
2021-03-03 12:47:27 +00:00
:values="boobSizes"
:disabled="!boobSizeRequired"
:allow-enable="true"
@enable="boobSizeRequired = true"
@input="(sizeRange) => updateValue('boobSize', sizeRange, false)"
@change="(sizeRange) => updateValue('boobSize', sizeRange, true)"
2021-02-28 02:38:54 +00:00
>
<template v-slot:start>
<span class="range-label"><Icon icon="boobs-small" /></span>
</template>
<template v-slot:end>
<span class="range-label on"><Icon icon="boobs-big" /></span>
</template>
</Range>
2021-02-28 02:38:54 +00:00
</span>
</div>
<div class="filter-section">
<span class="filter-label">Enhanced</span>
2021-02-28 02:38:54 +00:00
<span
:class="{ [['off', 'default', 'on'][naturalBoobs]]: true }"
class="toggle-container noclick"
2021-02-28 02:38:54 +00:00
>
<span
class="toggle-label off"
2021-02-28 02:38:54 +00:00
@click="updateValue('naturalBoobs', 0)"
><Icon icon="leaf" /></span>
2021-02-28 02:38:54 +00:00
<input
v-model="naturalBoobs"
class="toggle"
2021-02-28 02:38:54 +00:00
type="range"
min="0"
max="2"
@change="updateFilters"
>
<span
class="toggle-label on"
2021-02-28 02:38:54 +00:00
@click="updateValue('naturalBoobs', 2)"
><Icon icon="magic-wand2" /></span>
2021-02-28 02:38:54 +00:00
</span>
</div>
</template>
</Tooltip>
</li>
<li>
<Tooltip class="filter boobs">
<span class="filter-trigger"><Icon icon="rulers" />Physique</span>
<template v-slot:tooltip>
<div class="filter-section">
<label class="filter-label noselect">
<span class="label">
<Checkbox
:checked="heightRequired"
class="checkbox"
@change="(checked) => updateValue('heightRequired', checked)"
/>Height
</span>
<span
v-if="heightRequired"
class="label-values"
>({{ height[0] }} - {{ height[1] }} cm)</span>
</label>
<span class="filter-split">
<Range
:min="50"
:max="250"
:value="height"
:disabled="!heightRequired"
:allow-enable="true"
@enable="heightRequired = true"
@input="(heightRange) => updateValue('height', heightRange, false)"
@change="(heightRange) => updateValue('height', heightRange, true)"
>
<template v-slot:start><Icon icon="height-short" /></template>
<template v-slot:end><Icon icon="height" /></template>
</Range>
</span>
</div>
<div class="filter-section">
<label class="filter-label noselect">
<span class="label">
<Checkbox
:checked="weightRequired"
class="checkbox"
@change="(checked) => updateValue('weightRequired', checked)"
/>Weight
</span>
<span
v-if="weightRequired"
class="label-values"
>({{ weight[0] }} - {{ weight[1] }} kg)</span>
</label>
<span class="filter-split">
<Range
:min="30"
:max="200"
:value="weight"
:disabled="!weightRequired"
:allow-enable="true"
@enable="weightRequired = true"
@input="(weightRange) => updateValue('weight', weightRange, false)"
@change="(weightRange) => updateValue('weight', weightRange, true)"
>
<template v-slot:start><Icon icon="meter-slow" /></template>
<template v-slot:end><Icon icon="meter-fast" /></template>
</Range>
</span>
</div>
</template>
</Tooltip>
</li>
2021-02-28 02:38:54 +00:00
</ul>
</nav>
<div class="tiles">
2020-05-17 23:22:56 +00:00
<Actor
v-for="actor in actors"
:key="`actor-${actor.id}`"
:actor="actor"
/>
</div>
<Pagination
2020-05-25 00:02:28 +00:00
v-if="totalCount > 0"
:items-total="totalCount"
:items-per-page="limit"
class="pagination-top"
/>
2020-08-15 17:04:33 +00:00
<Footer />
2020-05-17 23:22:56 +00:00
</div>
</template>
<script>
2020-06-29 23:07:48 +00:00
import Actor from './tile.vue';
import Gender from './gender.vue';
import Checkbox from '../form/checkbox.vue';
import Range from '../form/range.vue';
import Pagination from '../pagination/pagination.vue';
2021-02-28 02:38:54 +00:00
const toggleValues = [true, undefined, false];
2021-03-03 12:53:18 +00:00
const boobSizes = 'ABCDEFGHZ'.split('');
2021-02-28 02:38:54 +00:00
function updateFilters() {
this.$router.push({
name: 'actors',
params: this.$route.params,
query: {
naturalBoobs: toggleValues[this.naturalBoobs],
bs: this.boobSizeRequired ? this.boobSize.join(',') : undefined,
h: this.heightRequired ? this.height.join(',') : undefined,
w: this.weightRequired ? this.weight.join(',') : undefined,
2021-02-28 02:38:54 +00:00
},
});
}
function updateValue(prop, value, load = true) {
2021-02-28 02:38:54 +00:00
this[prop] = value;
if (load) {
this.updateFilters();
}
2021-02-28 02:38:54 +00:00
}
async function fetchActors(scroll) {
2020-05-17 23:22:56 +00:00
const curatedGender = this.gender.replace('trans', 'transsexual');
2020-02-23 04:23:07 +00:00
const { actors, totalCount } = await this.$store.dispatch('fetchActors', {
limit: this.limit,
pageNumber: Number(this.$route.params.pageNumber) || 1,
2020-05-17 23:22:56 +00:00
letter: this.letter.replace('all', ''),
gender: curatedGender === 'other' ? null : curatedGender,
2021-03-03 12:47:27 +00:00
boobSize: this.boobSizeRequired && this.boobSize,
2021-02-28 02:38:54 +00:00
naturalBoobs: toggleValues[this.naturalBoobs] ?? null,
height: this.heightRequired && this.height,
weight: this.weightRequired && this.weight,
2020-05-17 23:22:56 +00:00
});
this.actors = actors;
this.totalCount = totalCount;
2021-01-17 20:24:20 +00:00
2021-02-28 02:38:54 +00:00
if (scroll) {
this.$refs.pagination.$el.scrollIntoView();
}
}
function letter() {
2020-05-17 23:22:56 +00:00
return this.$route.params.letter || 'all';
}
function gender() {
2020-08-02 01:51:52 +00:00
return this.$route.params.gender || 'all';
}
2021-02-28 02:38:54 +00:00
async function route(to, from) {
const scroll = to.params.pageNumber !== from.params.pageNumber
|| to.params.gender !== from.params.gender
|| to.params.letter !== from.params.letter;
await this.fetchActors(scroll);
}
async function mounted() {
2020-05-17 23:22:56 +00:00
this.pageTitle = 'Actors';
2020-05-17 23:22:56 +00:00
await this.fetchActors();
}
export default {
2020-05-17 23:22:56 +00:00
components: {
Actor,
Checkbox,
2020-05-17 23:22:56 +00:00
Gender,
Range,
Pagination,
2020-05-17 23:22:56 +00:00
},
data() {
const naturalBoobs = ['true', undefined, 'false'].indexOf(this.$route.query.nb);
2021-02-28 02:38:54 +00:00
2020-05-17 23:22:56 +00:00
return {
actors: [],
pageTitle: null,
totalCount: 0,
2021-01-17 20:24:20 +00:00
limit: 50,
2020-05-17 23:22:56 +00:00
letters: ['all'].concat(Array.from({ length: 26 }, (value, index) => String.fromCharCode(index + 97).toUpperCase())),
boobSizes,
2021-03-03 12:47:27 +00:00
boobSize: this.$route.query.bs?.split(',') || ['A', 'Z'],
boobSizeRequired: !!this.$route.query.bs,
2021-02-28 02:38:54 +00:00
naturalBoobs: naturalBoobs > -1 ? naturalBoobs : 1,
height: this.$route.query.h?.split(',').map(Number) || [50, 250],
heightRequired: !!this.$route.query.h,
weight: this.$route.query.w?.split(',').map(Number) || [30, 200],
weightRequired: !!this.$route.query.w,
2020-05-17 23:22:56 +00:00
};
},
computed: {
letter,
gender,
},
watch: {
$route: route,
},
mounted,
methods: {
fetchActors,
2021-02-28 02:38:54 +00:00
updateFilters,
updateValue,
2020-05-17 23:22:56 +00:00
},
};
</script>
<style lang="scss">
.gender-link {
&.selected .gender .icon {
fill: var(--text-light);
filter: none;
}
&:hover:not(.selected) {
.gender .icon {
fill: var(--text-light);
}
.male .icon {
filter: drop-shadow(0 0 1px var(--male));
}
.female .icon {
filter: drop-shadow(0 0 1px var(--female));
}
}
&:hover:not(.selected) .transsexual .icon {
fill: var(--female);
filter: drop-shadow(1px 0 0 var(--text-light)) drop-shadow(-1px 0 0 var(--text-light)) drop-shadow(0 1px 0 var(--text-light)) drop-shadow(0 -1px 0 var(--text-light)) drop-shadow(1px 0 0 var(--male)) drop-shadow(-1px 0 0 var(--male)) drop-shadow(0 1px 0 var(--male)) drop-shadow(0 -1px 0 var(--male)) drop-shadow(0 0 1px rgba(0, 0, 0, 0.5));
}
}
</style>
<style lang="scss" scoped>
@import 'breakpoints';
.actors {
display: flex;
flex-direction: column;
}
.tiles {
display: grid;
2020-05-21 22:55:11 +00:00
grid-template-columns: repeat(auto-fill, minmax(11rem, 1fr));
2020-06-29 23:07:48 +00:00
grid-gap: .5rem;
padding: 1rem;
flex-grow: 1;
}
.filter {
display: flex;
justify-content: center;
align-items: center;
padding: 0 1rem;
2021-02-28 02:38:54 +00:00
margin: 0 0 1rem 0;
&:first-child {
margin: 1rem 0;
}
}
.genders {
2020-02-23 04:23:07 +00:00
display: flex;
flex-shrink: 0;
padding: 0 .5rem 0 0;
2020-03-23 00:43:49 +00:00
border-right: solid 1px var(--shadow-hint);
margin: 0 1rem 0 0;
}
.letter,
.gender {
display: inline-block;
}
.letter-link,
.gender-link {
width: 2.5rem;
height: 2.5rem;
display: flex;
align-items: center;
justify-content: center;
box-sizing: border-box;
margin: .25rem .5rem .25rem 0;
2020-03-23 00:43:49 +00:00
color: var(--shadow);
background: var(--background);
font-weight: bold;
text-decoration: none;
box-shadow: 0 0 3px var(--darken-weak);
.male,
.female,
.transsexual {
padding: .2rem 0 0 0;
}
.icon {
fill: var(--shadow);
}
&:hover {
color: var(--text);
cursor: pointer;
.icon {
fill: var(--text);
}
}
&.selected {
2020-03-23 00:43:49 +00:00
background: var(--primary);
color: var(--text-light);
&.other .icon {
fill: var(--text-light);
}
}
}
2021-02-28 02:38:54 +00:00
.filter-trigger {
display: inline-flex;
align-items: center;
color: var(--shadow);
font-weight: bold;
.icon {
fill: var(--shadow);
width: 1.5rem;
height: 1.5rem;
margin: 0 .5rem 0 0;
}
&:hover {
color: var(--shadow-strong);
cursor: pointer;
.icon {
fill: var(--shadow-strong);
}
}
}
.filter-section {
width: 15rem;
max-width: 100%;
border-bottom: solid 1px var(--darken-hint);
}
2021-02-28 02:38:54 +00:00
.filter-label {
display: flex;
justify-content: space-between;
padding: .75rem .5rem .5rem .5rem;
color: var(--darken);
2021-02-28 02:38:54 +00:00
font-weight: bold;
font-size: .9rem;
.label {
display: inline-flex;
align-items: center;
}
.checkbox {
margin: 0 .75rem 0 0;
}
2021-02-28 02:38:54 +00:00
.icon {
margin: 0 .5rem 0 0;
}
}
.label-values {
font-weight: normal;
}
.filter-split {
display: flex;
align-items: center;
}
.toggle-container,
2021-02-28 02:38:54 +00:00
.range-container {
display: flex;
flex-grow: 1;
2021-02-28 02:38:54 +00:00
align-items: center;
padding: .5rem 0;
&.on {
.toggle-label.on {
2021-02-28 02:38:54 +00:00
color: var(--enabled);
.icon {
fill: var(--enabled);
}
2021-02-28 02:38:54 +00:00
}
.toggle {
2021-03-03 03:05:27 +00:00
background-color: var(--enabled-background);
&::-webkit-slider-thumb {
background: var(--enabled);
}
&::-moz-range-thumb {
background: var(--enabled);
}
2021-02-28 02:38:54 +00:00
}
}
&.off {
.toggle-label.off {
2021-02-28 02:38:54 +00:00
color: var(--disabled);
.icon {
fill: var(--disabled);
}
2021-02-28 02:38:54 +00:00
}
.toggle {
2021-03-03 03:05:27 +00:00
background-color: var(--disabled-background);
&::-webkit-slider-thumb {
background: var(--disabled);
}
2021-02-28 02:38:54 +00:00
&::-moz-range-thumb {
background: var(--disabled);
}
2021-02-28 02:38:54 +00:00
}
}
}
.toggle-label {
display: inline-flex;
justify-content: center;
min-width: 1.5rem;
flex-shrink: 0;
2021-02-28 02:38:54 +00:00
padding: 0 .5rem;
color: var(--darken);
2021-02-28 02:38:54 +00:00
font-weight: bold;
font-size: .9rem;
&.on {
text-align: right;
}
.icon {
fill: var(--darken);
}
2021-02-28 02:38:54 +00:00
&:hover {
cursor: pointer;
&.on {
color: var(--enabled);
.icon {
fill: var(--enabled);
}
2021-02-28 02:38:54 +00:00
}
&.off {
color: var(--disabled);
.icon {
fill: var(--disabled);
}
2021-02-28 02:38:54 +00:00
}
}
}
.toggle {
width: 0;
2021-02-28 02:38:54 +00:00
flex-grow: 1;
height: 1.25rem;
2021-02-28 02:38:54 +00:00
appearance: none;
2021-03-01 18:45:29 +00:00
border-radius: 1rem;
2021-03-03 03:05:27 +00:00
background-color: var(--darken-hint);
background-image: radial-gradient(circle, var(--shadow-weak) .3rem, transparent calc(.3rem + 1px));
2021-02-28 02:38:54 +00:00
cursor: pointer;
&::-webkit-slider-thumb {
appearance: none;
2021-03-03 03:05:27 +00:00
background: var(--disabled-handle);
width: 1.25rem;
height: 1.25rem;
border-radius: .625rem;
2021-03-03 03:05:27 +00:00
box-shadow: 0 0 3px var(--darken-weak);
2021-02-28 02:38:54 +00:00
}
&::-moz-range-thumb {
appearance: none;
2021-03-03 03:05:27 +00:00
background: var(--disabled-handle);
width: 1.25rem;
height: 1.25rem;
border: none;
border-radius: .625rem;
2021-03-03 03:05:27 +00:00
box-shadow: 0 0 3px var(--darken-weak);
2021-02-28 02:38:54 +00:00
}
}
@media(max-width: $breakpoint) {
2020-02-23 04:23:07 +00:00
.genders {
flex-direction: column;
}
}
@media(max-width: $breakpoint-micro) {
.tiles {
grid-template-columns: repeat(auto-fill, minmax(8rem, 1fr));
}
}
</style>