forked from DebaucheryLibrarian/traxxx
624 lines
12 KiB
Vue
Executable File
624 lines
12 KiB
Vue
Executable File
<template>
|
|
<Dialog
|
|
title="Add alert"
|
|
@close="$emit('close')"
|
|
>
|
|
<div
|
|
v-if="error"
|
|
class="dialog-error"
|
|
>{{ error }}</div>
|
|
|
|
<form
|
|
class="dialog-body"
|
|
@submit.prevent="addAlert"
|
|
>
|
|
<div class="dialog-section">
|
|
<h3 class="dialog-heading">
|
|
When
|
|
|
|
<label class="dialog-description noselect">
|
|
<template v-if="all">Scene must match <strong>all</strong> fields</template>
|
|
<template v-else>Scene must match <strong>any</strong> field</template>
|
|
|
|
<Toggle
|
|
:checked="all"
|
|
@change="(checked) => all = checked"
|
|
/>
|
|
</label>
|
|
</h3>
|
|
|
|
<div class="alert-section">
|
|
<h4
|
|
v-if="actors.length > 1"
|
|
class="alert-heading"
|
|
>Actors</h4>
|
|
|
|
<h4
|
|
v-else
|
|
class="alert-heading"
|
|
>Actor</h4>
|
|
|
|
<ul class="actors nolist">
|
|
<li
|
|
v-for="actor in actors"
|
|
:key="`actor-${actor.id}`"
|
|
class="actor"
|
|
>
|
|
<ActorPreview
|
|
:actor="actor"
|
|
target="_blank"
|
|
/>
|
|
|
|
<Icon
|
|
icon="cross3"
|
|
class="remove"
|
|
@click.native="removeActor(actor)"
|
|
/>
|
|
</li>
|
|
|
|
<Tooltip>
|
|
<li class="actor placeholder"><template v-if="actors.length === 0">Any actor</template>
|
|
<Icon
|
|
icon="plus3"
|
|
class="add"
|
|
/>
|
|
</li>
|
|
|
|
<template #tooltip>
|
|
<Search
|
|
content="actors"
|
|
@select="actor => addActor(actor)"
|
|
/>
|
|
</template>
|
|
</Tooltip>
|
|
</ul>
|
|
</div>
|
|
|
|
<div class="alert-section">
|
|
<h4
|
|
v-if="actors.length > 1"
|
|
class="alert-heading"
|
|
>Do</h4>
|
|
|
|
<h4
|
|
v-else
|
|
class="alert-heading"
|
|
>Does</h4>
|
|
|
|
<ul class="tags nolist">
|
|
<li
|
|
v-for="tag in tags"
|
|
:key="`tag-${tag.id}`"
|
|
class="tag"
|
|
>{{ tag.name }}
|
|
<Icon
|
|
icon="cross3"
|
|
class="remove"
|
|
@click.native="removeTag(tag)"
|
|
/>
|
|
</li>
|
|
|
|
<Tooltip>
|
|
<li class="tag placeholder"><template v-if="tags.length === 0">Any type of scene</template>
|
|
<Icon
|
|
icon="plus3"
|
|
class="add"
|
|
/>
|
|
</li>
|
|
|
|
<template #tooltip>
|
|
<Search
|
|
content="tags"
|
|
:defaults="['anal', 'blowbang', 'mfm', 'dp', 'gangbang', 'airtight']"
|
|
@select="tag => addTag(tag)"
|
|
/>
|
|
</template>
|
|
</Tooltip>
|
|
</ul>
|
|
</div>
|
|
|
|
<div class="alert-section">
|
|
<h4 class="alert-heading">For</h4>
|
|
|
|
<div class="entities">
|
|
<div
|
|
v-for="(entity, index) in entities"
|
|
:key="`entity-${entity.id}`"
|
|
:class="{ invalid: all && index > 0 }"
|
|
class="entity"
|
|
>
|
|
<Entity
|
|
:entity="entity"
|
|
target="_blank"
|
|
/>
|
|
|
|
<Icon
|
|
icon="cross3"
|
|
class="remove"
|
|
@click.native="removeEntity(entity)"
|
|
/>
|
|
</div>
|
|
|
|
<Tooltip v-if="entities.length < 1 || !all">
|
|
<div class="entity placeholder">
|
|
Any channel
|
|
|
|
<Icon
|
|
icon="plus3"
|
|
class="add"
|
|
/>
|
|
</div>
|
|
|
|
<template #tooltip>
|
|
<Search
|
|
label="Search channels"
|
|
content="entities"
|
|
@select="entity => addEntity(entity)"
|
|
/>
|
|
</template>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="alert-section">
|
|
<h4 class="alert-heading">Matching</h4>
|
|
|
|
<ul class="matches nolist">
|
|
<li
|
|
v-for="(match, index) in matches"
|
|
:key="`match-${index}`"
|
|
class="match"
|
|
>
|
|
<span class="match-property">{{ match.property }}: </span>
|
|
|
|
<span
|
|
v-if="match.expression.slice(0, 1) === '/' && match.expression.slice(-1) === '/'"
|
|
class="match-expression"
|
|
><span class="match-slash">/</span>{{ match.expression.slice(1, -1) }}<span class="match-slash">/</span></span>
|
|
|
|
<span
|
|
v-else
|
|
class="match-expression"
|
|
>{{ match.expression }}</span>
|
|
|
|
<Icon
|
|
icon="cross3"
|
|
class="remove"
|
|
@click.native="removeMatch(index)"
|
|
/>
|
|
</li>
|
|
|
|
<Tooltip
|
|
v-if="entities.length === 0"
|
|
@open="$refs.expression?.focus()"
|
|
>
|
|
<li class="match placeholder">
|
|
Anything
|
|
|
|
<Icon
|
|
icon="plus3"
|
|
class="add"
|
|
/>
|
|
</li>
|
|
|
|
<template #tooltip>
|
|
<form
|
|
class="pattern-tooltip"
|
|
@submit.prevent="addMatch"
|
|
>
|
|
<select
|
|
v-model="matchProperty"
|
|
class="input"
|
|
>
|
|
<option value="title">Title</option>
|
|
<option value="description">Description</option>
|
|
</select>
|
|
|
|
<input
|
|
ref="expression"
|
|
v-model="matchExpression"
|
|
class="input"
|
|
placeholder="Expression, // for RegExp"
|
|
>
|
|
</form>
|
|
</template>
|
|
</Tooltip>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="dialog-section">
|
|
<h3 class="dialog-heading">Then</h3>
|
|
|
|
<label class="alert-label">
|
|
<Checkbox
|
|
:checked="notify"
|
|
@change="checked => notify = checked"
|
|
/>Notify me in traxxx
|
|
</label>
|
|
|
|
<!--
|
|
<label class="alert-label">
|
|
<Checkbox
|
|
:checked="email"
|
|
@change="checked => email = checked"
|
|
/>Send me an e-mail
|
|
</label>
|
|
-->
|
|
|
|
<div class="stashes-container">
|
|
<ul class="stashes nolist">
|
|
<li
|
|
v-for="stash in stashes"
|
|
:key="`stash-${stash.id}`"
|
|
class="stash"
|
|
>{{ stash.name }}
|
|
|
|
<Icon
|
|
icon="cross3"
|
|
class="remove"
|
|
@click.native="removeStash(stash)"
|
|
/>
|
|
</li>
|
|
|
|
<Tooltip>
|
|
<li class="stash placeholder">
|
|
Add to stash
|
|
<Icon
|
|
icon="plus3"
|
|
class="add"
|
|
/>
|
|
</li>
|
|
|
|
<template #tooltip>
|
|
<Search
|
|
content="stashes"
|
|
@select="stash => addStash(stash)"
|
|
/>
|
|
</template>
|
|
</Tooltip>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="dialog-actions right">
|
|
<button
|
|
:disabled="actors.length === 0 && tags.length === 0 && entities.length === 0 && matches.length === 0"
|
|
type="submit"
|
|
class="button button-primary"
|
|
>Add alert</button>
|
|
</div>
|
|
</form>
|
|
</Dialog>
|
|
</template>
|
|
|
|
<script>
|
|
import ActorPreview from '../actors/preview.vue';
|
|
import Entity from '../entities/tile.vue';
|
|
import Checkbox from '../form/checkbox.vue';
|
|
import Toggle from '../form/toggle.vue';
|
|
import Search from './search.vue';
|
|
|
|
async function addAlert() {
|
|
this.error = null;
|
|
|
|
try {
|
|
await this.$store.dispatch('addAlert', {
|
|
all: this.all,
|
|
actors: this.actors.map((actor) => actor.id),
|
|
tags: this.tags.map((tag) => tag.id),
|
|
matches: this.matches,
|
|
entities: this.entities.map((entity) => entity.id),
|
|
notify: this.notify,
|
|
email: this.email,
|
|
stashes: this.stashes.map((stash) => stash.id),
|
|
});
|
|
|
|
this.$emit('close', true);
|
|
} catch (error) {
|
|
this.error = error.message;
|
|
}
|
|
}
|
|
|
|
function addActor(actor) {
|
|
if (!this.actors.some((selectedActor) => selectedActor.id === actor.id)) {
|
|
this.actors = this.actors.concat(actor);
|
|
}
|
|
|
|
this.events.emit('blur');
|
|
}
|
|
|
|
function addEntity(entity) {
|
|
this.entities = this.entities.concat(entity);
|
|
this.events.emit('blur');
|
|
}
|
|
|
|
function addTag(tag) {
|
|
if (!this.tags.some((selectedTag) => selectedTag.id === tag.id)) {
|
|
this.tags = this.tags.concat(tag);
|
|
}
|
|
|
|
this.events.emit('blur');
|
|
}
|
|
|
|
function removeActor(actor) {
|
|
this.actors = this.actors.filter((listedActor) => listedActor.id !== actor.id);
|
|
}
|
|
|
|
function removeEntity(entity) {
|
|
this.entities = this.entities.filter((alertEntity) => alertEntity.id !== entity.id);
|
|
}
|
|
|
|
function removeTag(tag) {
|
|
this.tags = this.tags.filter((listedTag) => listedTag.id !== tag.id);
|
|
}
|
|
|
|
function addMatch() {
|
|
if (!this.matchExpression) {
|
|
return;
|
|
}
|
|
|
|
this.matches = this.matches.concat({
|
|
property: this.matchProperty,
|
|
expression: this.matchExpression,
|
|
});
|
|
|
|
this.matchProperty = 'title';
|
|
this.matchExpression = null;
|
|
|
|
this.events.emit('blur');
|
|
}
|
|
|
|
function removeMatch(removeIndex) {
|
|
this.matches = this.matches.filter((match, matchIndex) => matchIndex !== removeIndex);
|
|
}
|
|
|
|
function addStash(stash) {
|
|
if (!this.stashes.some((selectedStash) => selectedStash.id === stash.id)) {
|
|
this.stashes = this.stashes.concat(stash);
|
|
}
|
|
|
|
this.events.emit('blur');
|
|
}
|
|
|
|
function removeStash(stash) {
|
|
this.stashes = this.stashes.filter((listedStash) => listedStash.id !== stash.id);
|
|
}
|
|
|
|
export default {
|
|
components: {
|
|
ActorPreview,
|
|
Checkbox,
|
|
Entity,
|
|
Search,
|
|
Toggle,
|
|
},
|
|
emits: ['close'],
|
|
data() {
|
|
return {
|
|
error: null,
|
|
actors: [],
|
|
tags: [],
|
|
all: true,
|
|
entities: [],
|
|
matches: [],
|
|
matchProperty: 'title',
|
|
matchExpression: null,
|
|
notify: true,
|
|
email: false,
|
|
stashes: [],
|
|
availableStashes: this.$store.state.auth.user.stashes,
|
|
};
|
|
},
|
|
methods: {
|
|
addActor,
|
|
addAlert,
|
|
addEntity,
|
|
addMatch,
|
|
addTag,
|
|
addStash,
|
|
removeActor,
|
|
removeEntity,
|
|
removeMatch,
|
|
removeTag,
|
|
removeStash,
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.dialog-section {
|
|
width: 30rem;
|
|
max-width: 100%;
|
|
|
|
&:first-child {
|
|
border-bottom: solid 1px var(--shadow-hint);
|
|
margin: 0 0 1rem 0;
|
|
}
|
|
}
|
|
|
|
.dialog-heading {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
margin: 0 0 .25rem 0;
|
|
color: var(--primary);
|
|
}
|
|
|
|
.dialog-description {
|
|
display: flex;
|
|
align-items: center;
|
|
color: var(--shadow);
|
|
font-size: .9rem;
|
|
font-weight: normal;
|
|
|
|
.toggle-container {
|
|
margin-left: .5rem;
|
|
}
|
|
}
|
|
|
|
.dialog-error {
|
|
padding: 1rem;
|
|
margin-bottom: 1rem;
|
|
background: var(--error);
|
|
color: var(--text-light);
|
|
font-weight: bold;
|
|
text-align: center;
|
|
}
|
|
|
|
.alert-heading {
|
|
margin: .75rem 0 .25rem 0;
|
|
}
|
|
|
|
.actors,
|
|
.entities,
|
|
.tags {
|
|
display: flex;
|
|
align-items: center;
|
|
flex-wrap: wrap;
|
|
font-size: 0;
|
|
}
|
|
|
|
.match {
|
|
display: flex;
|
|
align-items: center;
|
|
padding: .25rem 0;
|
|
font-family: inherit;
|
|
|
|
.remove {
|
|
position: relative;
|
|
top: -.1rem;
|
|
right: 0;
|
|
}
|
|
}
|
|
|
|
.match-property {
|
|
text-transform: capitalize;
|
|
color: var(--shadow);
|
|
}
|
|
|
|
.match-expression {
|
|
flex-grow: 1;
|
|
}
|
|
|
|
.match-slash {
|
|
padding: 0 .1rem;
|
|
color: var(--primary);
|
|
font-weight: bold;
|
|
}
|
|
|
|
.actors > .actor,
|
|
.entity,
|
|
.tag,
|
|
.stash {
|
|
position: relative;
|
|
font-size: 1rem;
|
|
margin: 0 .5rem .5rem 0;
|
|
}
|
|
|
|
.entity.invalid {
|
|
opacity: .5;
|
|
pointer-events: none;
|
|
}
|
|
|
|
.entity .tile {
|
|
width: 10rem;
|
|
height: 2.5rem;
|
|
}
|
|
|
|
.tag:not(.placeholder),
|
|
.stash:not(.placeholder) {
|
|
padding: .5rem .75rem;
|
|
border: solid 1px var(--shadow-hint);
|
|
margin: 0 .75rem 0 0;
|
|
font-size: .9rem;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.stashes {
|
|
margin: 0 0 0 .25rem;
|
|
color: var(--text);
|
|
}
|
|
|
|
.pattern-tooltip {
|
|
display: flex;
|
|
gap: .5rem;
|
|
position: relative;
|
|
padding: .5rem;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.remove {
|
|
width: 1rem;
|
|
height: 1rem;
|
|
position: absolute;
|
|
top: -.35rem;
|
|
right: -.35rem;
|
|
z-index: 1;
|
|
border: solid 1px var(--darken-hint);
|
|
border-radius: 50%;
|
|
background: var(--background);
|
|
fill: var(--shadow-weak);
|
|
box-shadow: 0 0 1px var(--shadow);
|
|
|
|
&:hover {
|
|
fill: var(--text-light);
|
|
background: var(--primary);
|
|
border: solid 1px var(--primary);
|
|
cursor: pointer;
|
|
}
|
|
}
|
|
|
|
.placeholder {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
color: var(--shadow-strong);
|
|
padding: .75rem 0;
|
|
margin: 0 0 .5rem 0;
|
|
font-size: 1rem;
|
|
|
|
.add {
|
|
fill: var(--shadow);
|
|
margin: 0 0 0 .5rem;
|
|
}
|
|
|
|
&:hover {
|
|
color: var(--primary);
|
|
cursor: pointer;
|
|
|
|
.add {
|
|
fill: var(--primary);
|
|
}
|
|
}
|
|
}
|
|
|
|
.alert-label {
|
|
display: flex;
|
|
align-items: center;
|
|
padding: .5rem 0;
|
|
margin: 0 0 .25rem 0;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.stashes-heading {
|
|
display: flex;
|
|
align-items: center;
|
|
margin: 0 0 .5rem 0;
|
|
font-size: 1rem;
|
|
|
|
.alert-label {
|
|
display: inline-block;
|
|
margin: 0;
|
|
}
|
|
}
|
|
|
|
.tooltip-container {
|
|
display: inline-block;
|
|
}
|
|
|
|
.check-container {
|
|
display: inline-block;
|
|
margin: 0 .5rem 0 0;
|
|
}
|
|
</style>
|