| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261 | <template>	<modal title="Edit Station" class="edit-station-modal">		<template v-slot:body>			<!--  Station Preferences -->			<div class="section left-section">				<div class="col col-2">					<div>						<label class="label">Name</label>						<p class="control">							<input								class="input"								type="text"								v-model="editing.name"							/>						</p>					</div>					<div>						<label class="label">Display name</label>						<p class="control">							<input								class="input"								type="text"								v-model="editing.displayName"							/>						</p>					</div>				</div>				<div class="col col-1">					<div>						<label class="label">Description</label>						<p class="control">							<input								class="input"								type="text"								v-model="editing.description"							/>						</p>					</div>				</div>				<div class="col col-2" v-if="editing.genres">					<div>						<label class="label">Genre(s)</label>						<p class="control has-addons">							<input								class="input"								type="text"								id="new-genre"								v-model="genreInputValue"								v-on:blur="blurGenreInput()"								v-on:focus="focusGenreInput()"								v-on:keydown="keydownGenreInput()"								v-on:keyup.enter="addTag('genres')"							/>							<button								class="button is-info add-button blue"								v-on:click="addTag('genres')"							>								<i class="material-icons">add</i>							</button>						</p>						<div							class="autosuggest-container"							v-if="								(genreInputFocussed ||									genreAutosuggestContainerFocussed) &&									genreAutosuggestItems.length > 0							"							@mouseover="focusGenreContainer()"							@mouseleave="blurGenreContainer()"						>							<span								class="autosuggest-item"								tabindex="0"								v-on:click="selectGenreAutosuggest(item)"								v-for="(item, index) in genreAutosuggestItems"								:key="index"								>{{ item }}</span							>						</div>						<div class="list-container">							<div								class="list-item"								v-for="(genre, index) in editing.genres"								:key="index"							>								<div									class="list-item-circle blue"									v-on:click="removeTag('genres', index)"								>									<i class="material-icons">close</i>								</div>								<p>{{ genre }}</p>							</div>						</div>					</div>					<div>						<label class="label">Blacklist genre(s)</label>						<p class="control has-addons">							<input								class="input"								type="text"								v-model="blacklistGenreInputValue"								v-on:blur="blurBlacklistGenreInput()"								v-on:focus="focusBlacklistGenreInput()"								v-on:keydown="keydownBlacklistGenreInput()"								v-on:keyup.enter="addTag('blacklist-genres')"							/>							<button								class="button is-info add-button red"								v-on:click="addTag('blacklist-genres')"							>								<i class="material-icons">add</i>							</button>						</p>						<div							class="autosuggest-container"							v-if="								(blacklistGenreInputFocussed ||									blacklistGenreAutosuggestContainerFocussed) &&									blacklistGenreAutosuggestItems.length > 0							"							@mouseover="focusBlacklistGenreContainer()"							@mouseleave="blurBlacklistGenreContainer()"						>							<span								class="autosuggest-item"								tabindex="0"								v-on:click="									selectBlacklistGenreAutosuggest(item)								"								v-for="(item,								index) in blacklistGenreAutosuggestItems"								:key="index"								>{{ item }}</span							>						</div>						<div class="list-container">							<div								class="list-item"								v-for="(genre,								index) in editing.blacklistedGenres"								:key="index"							>								<div									class="list-item-circle red"									v-on:click="										removeTag('blacklist-genres', index)									"								>									<i class="material-icons">close</i>								</div>								<p>{{ genre }}</p>							</div>						</div>					</div>				</div>				<!--  Choose a playlist -->				<div v-if="!editing.partyMode && playlists.length > 0">					<hr style="margin: 10px 0 20px 0;" />					<h4 class="modal-section-title">Choose a playlist</h4>					<p class="modal-section-description">						Choose one of your playlists to add to the queue.					</p>					<br />					<div id="playlists">						<div							class="playlist"							v-for="(playlist, index) in playlists"							:key="index"						>							<playlist-item :playlist="playlist">								<div slot="actions">									<!-- <a										class="button is-danger"										href="#"										@click="											togglePlaylistSelection(												playlist._id											)										"										v-if="isPlaylistSelected(playlist._id)"									>										<i											class="material-icons icon-with-button"											>stop</i										>										Stop playing									</a> -->									<a										class="button is-success"										href="#"										@click="selectPlaylist(playlist._id)"										><i											class="material-icons icon-with-button"											>play_arrow</i										>Play in queue									</a>								</div>							</playlist-item>						</div>					</div>				</div>			</div>			<!--  Buttons changing the privacy settings -->			<div class="section right-section">				<div>					<label class="label">Privacy</label>					<div						@mouseenter="privacyDropdownActive = true"						@mouseleave="privacyDropdownActive = false"						class="button-wrapper"					>						<button							v-bind:class="privacyButtons[editing.privacy].style"							style="text-transform: capitalize"							@click="updatePrivacyLocal(editing.privacy)"						>							<i class="material-icons">{{								privacyButtons[editing.privacy].iconName							}}</i>							{{ editing.privacy }}						</button>						<transition name="slide-down">							<button								class="green"								v-if="									privacyDropdownActive &&										editing.privacy !== 'public'								"								@click="updatePrivacyLocal('public')"							>								<i class="material-icons">{{									privacyButtons["public"].iconName								}}</i>								Public							</button>						</transition>						<transition name="slide-down">							<button								class="orange"								v-if="									privacyDropdownActive &&										editing.privacy !== 'unlisted'								"								@click="updatePrivacyLocal('unlisted')"							>								<i class="material-icons">{{									privacyButtons["unlisted"].iconName								}}</i>								Unlisted							</button>						</transition>						<transition name="slide-down">							<button								class="red"								v-if="									privacyDropdownActive &&										editing.privacy !== 'private'								"								@click="updatePrivacyLocal('private')"							>								<i class="material-icons">{{									privacyButtons["private"].iconName								}}</i>								Private							</button>						</transition>					</div>				</div>				<!--  Buttons changing the mode of the station -->				<div v-if="editing.type === 'community'">					<label class="label">Mode</label>					<div						@mouseenter="modeDropdownActive = true"						@mouseleave="modeDropdownActive = false"						class="button-wrapper"					>						<button							v-bind:class="{								blue: !editing.partyMode,								yellow: editing.partyMode							}"							@click="								editing.partyMode									? updatePartyModeLocal(true)									: updatePartyModeLocal(false)							"						>							<i class="material-icons">{{								editing.partyMode									? "emoji_people"									: "playlist_play"							}}</i>							{{ editing.partyMode ? "Party" : "Playlist" }}						</button>						<transition name="slide-down">							<button								class="blue"								v-if="modeDropdownActive && editing.partyMode"								@click="updatePartyModeLocal(false)"							>								<i class="material-icons">playlist_play</i>								Playlist							</button>						</transition>						<transition name="slide-down">							<button								class="yellow"								v-if="modeDropdownActive && !editing.partyMode"								@click="updatePartyModeLocal(true)"							>								<i class="material-icons">emoji_people</i>								Party							</button>						</transition>					</div>				</div>				<div					v-if="						editing.type === 'community' &&							editing.partyMode === true					"				>					<label class="label">Queue lock</label>					<div						@mouseenter="queueLockDropdownActive = true"						@mouseleave="queueLockDropdownActive = false"						class="button-wrapper"					>						<button							v-bind:class="{								green: editing.locked,								red: !editing.locked							}"							@click="								editing.locked									? updateQueueLockLocal(true)									: updateQueueLockLocal(false)							"						>							<i class="material-icons">{{								editing.locked ? "lock" : "lock_open"							}}</i>							{{ editing.locked ? "Locked" : "Unlocked" }}						</button>						<transition name="slide-down">							<button								class="green"								v-if="									queueLockDropdownActive && !editing.locked								"								@click="updateQueueLockLocal(true)"							>								<i class="material-icons">lock</i>								Locked							</button>						</transition>						<transition name="slide-down">							<button								class="red"								v-if="queueLockDropdownActive && editing.locked"								@click="updateQueueLockLocal(false)"							>								<i class="material-icons">lock_open</i>								Unlocked							</button>						</transition>					</div>				</div>			</div>		</template>		<template v-slot:footer>			<button class="button is-success" v-on:click="update()">				Update Settings			</button>			<button				v-if="station.type === 'community'"				class="button is-danger"				@click="deleteStation()"			>				Delete station			</button>		</template>	</modal></template><script>import { mapState, mapActions } from "vuex";import Toast from "toasters";import PlaylistItem from "../PlaylistItem.vue";import Modal from "./Modal.vue";import io from "../../io";import validation from "../../validation";export default {	computed: {		...mapState("admin/station", {			stations: state => state.stations		}),		...mapState({			editing(state) {				return this.$props.store					.split("/")					.reduce((a, v) => a[v], state).editing;			},			station(state) {				return this.$props.store					.split("/")					.reduce((a, v) => a[v], state).station;			}		})	},	mounted() {		io.getSocket(socket => {			this.socket = socket;			this.socket.emit("playlists.indexForUser", res => {				if (res.status === "success") this.playlists = res.data;			});			this.socket.on("event:playlist.create", playlist => {				this.playlists.push(playlist);			});			this.socket.on("event:playlist.delete", playlistId => {				this.playlists.forEach((playlist, index) => {					if (playlist._id === playlistId) {						this.playlists.splice(index, 1);					}				});			});			this.socket.on("event:playlist.addSong", data => {				this.playlists.forEach((playlist, index) => {					if (playlist._id === data.playlistId) {						this.playlists[index].songs.push(data.song);					}				});			});			this.socket.on("event:playlist.removeSong", data => {				this.playlists.forEach((playlist, index) => {					if (playlist._id === data.playlistId) {						this.playlists[index].songs.forEach((song, index2) => {							if (song._id === data.songId) {								this.playlists[index].songs.splice(index2, 1);							}						});					}				});			});			this.socket.on("event:playlist.updateDisplayName", data => {				this.playlists.forEach((playlist, index) => {					if (playlist._id === data.playlistId) {						this.playlists[index].displayName = data.displayName;					}				});			});			return socket;		});	},	data() {		return {			genreInputValue: "",			genreInputFocussed: false,			genreAutosuggestContainerFocussed: false,			keydownGenreInputTimeout: 0,			genreAutosuggestItems: [],			blacklistGenreInputValue: "",			blacklistGenreInputFocussed: false,			blacklistGenreAutosuggestContainerFocussed: false,			blacklistKeydownGenreInputTimeout: 0,			blacklistGenreAutosuggestItems: [],			privacyDropdownActive: false,			modeDropdownActive: false,			queueLockDropdownActive: false,			genres: [				"Blues",				"Country",				"Disco",				"Funk",				"Hip-Hop",				"Jazz",				"Metal",				"Oldies",				"Other",				"Pop",				"Rap",				"Reggae",				"Rock",				"Techno",				"Trance",				"Classical",				"Instrumental",				"House",				"Electronic",				"Christian Rap",				"Lo-Fi",				"Musical",				"Rock 'n' Roll",				"Opera",				"Drum & Bass",				"Club-House",				"Indie",				"Heavy Metal",				"Christian rock",				"Dubstep"			],			privacyButtons: {				public: {					style: "green",					iconName: "public"				},				private: {					style: "red",					iconName: "lock"				},				unlisted: {					style: "orange",					iconName: "link"				}			},			playlists: []		};	},	props: ["store"],	methods: {		isPlaylistSelected(id) {			// TODO Also change this once it changes for a station			if (this.station && this.station.privatePlaylist === id)				return true;			return false;		},		selectPlaylist(playlistId) {			this.socket.emit(				"stations.selectPrivatePlaylist",				this.station._id,				playlistId,				res => {					if (res.status === "failure")						return new Toast({							content: res.message,							timeout: 8000						});					return new Toast({ content: res.message, timeout: 4000 });				}			);		},		update() {			if (this.station.name !== this.editing.name) this.updateName();			if (this.station.displayName !== this.editing.displayName)				this.updateDisplayName();			if (this.station.description !== this.editing.description)				this.updateDescription();			if (this.station.privacy !== this.editing.privacy)				this.updatePrivacy();			if (				this.station.type === "community" &&				this.station.partyMode !== this.editing.partyMode			)				this.updatePartyMode();			if (				this.station.type === "community" &&				this.editing.partyMode &&				this.station.locked !== this.editing.locked			)				this.updateQueueLock();			if (this.$props.store !== "station") {				if (					this.station.genres.toString() !==					this.editing.genres.toString()				)					this.updateGenres();				if (					this.station.blacklistedGenres.toString() !==					this.editing.blacklistedGenres.toString()				)					this.updateBlacklistedGenres();			}		},		updateName() {			const { name } = this.editing;			if (!validation.isLength(name, 2, 16))				return new Toast({					content: "Name must have between 2 and 16 characters.",					timeout: 8000				});			if (!validation.regex.az09_.test(name))				return new Toast({					content:						"Invalid name format. Allowed characters: a-z, 0-9 and _.",					timeout: 8000				});			return this.socket.emit(				"stations.updateName",				this.editing._id,				name,				res => {					if (res.status === "success") {						if (this.station) this.station.name = name;						else {							this.stations.forEach((station, index) => {								if (station._id === this.editing._id) {									this.stations[index].name = name;									return name;								}								return false;							});						}					}					new Toast({ content: res.message, timeout: 8000 });				}			);		},		updateDisplayName() {			const { displayName } = this.editing;			if (!validation.isLength(displayName, 2, 32))				return new Toast({					content:						"Display name must have between 2 and 32 characters.",					timeout: 8000				});			if (!validation.regex.ascii.test(displayName))				return new Toast({					content:						"Invalid display name format. Only ASCII characters are allowed.",					timeout: 8000				});			return this.socket.emit(				"stations.updateDisplayName",				this.editing._id,				displayName,				res => {					if (res.status === "success") {						if (this.station)							this.station.displayName = displayName;						else {							this.stations.forEach((station, index) => {								if (station._id === this.editing._id) {									this.stations[										index									].displayName = displayName;									return displayName;								}								return false;							});						}					}					new Toast({ content: res.message, timeout: 8000 });				}			);		},		updateDescription() {			const { description } = this.editing;			if (!validation.isLength(description, 2, 200))				return new Toast({					content:						"Description must have between 2 and 200 characters.",					timeout: 8000				});			let characters = description.split("");			characters = characters.filter(character => {				return character.charCodeAt(0) === 21328;			});			if (characters.length !== 0)				return new Toast({					content: "Invalid description format.",					timeout: 8000				});			return this.socket.emit(				"stations.updateDescription",				this.editing._id,				description,				res => {					if (res.status === "success") {						if (this.station)							this.station.description = description;						else {							this.stations.forEach((station, index) => {								if (station._id === this.editing._id) {									this.stations[										index									].description = description;									return description;								}								return false;							});						}						return new Toast({							content: res.message,							timeout: 4000						});					}					return new Toast({ content: res.message, timeout: 8000 });				}			);		},		updatePrivacyLocal(privacy) {			if (this.editing.privacy === privacy) return;			this.editing.privacy = privacy;			this.privacyDropdownActive = false;		},		updatePrivacy() {			this.socket.emit(				"stations.updatePrivacy",				this.editing._id,				this.editing.privacy,				res => {					if (res.status === "success") {						if (this.station)							this.station.privacy = this.editing.privacy;						else {							this.stations.forEach((station, index) => {								if (station._id === this.editing._id) {									this.stations[										index									].privacy = this.editing.privacy;									return this.editing.privacy;								}								return false;							});						}						return new Toast({							content: res.message,							timeout: 4000						});					}					return new Toast({ content: res.message, timeout: 8000 });				}			);		},		updateGenres() {			this.socket.emit(				"stations.updateGenres",				this.editing._id,				this.editing.genres,				res => {					if (res.status === "success") {						const genres = JSON.parse(							JSON.stringify(this.editing.genres)						);						if (this.station) this.station.genres = genres;						this.stations.forEach((station, index) => {							if (station._id === this.editing._id) {								this.stations[index].genres = genres;								return genres;							}							return false;						});						return new Toast({							content: res.message,							timeout: 4000						});					}					return new Toast({ content: res.message, timeout: 8000 });				}			);		},		updateBlacklistedGenres() {			this.socket.emit(				"stations.updateBlacklistedGenres",				this.editing._id,				this.editing.blacklistedGenres,				res => {					if (res.status === "success") {						const blacklistedGenres = JSON.parse(							JSON.stringify(this.editing.blacklistedGenres)						);						if (this.station)							this.station.blacklistedGenres = blacklistedGenres;						this.stations.forEach((station, index) => {							if (station._id === this.editing._id) {								this.stations[									index								].blacklistedGenres = blacklistedGenres;								return blacklistedGenres;							}							return false;						});						return new Toast({							content: res.message,							timeout: 4000						});					}					return new Toast({ content: res.message, timeout: 8000 });				}			);		},		updatePartyModeLocal(partyMode) {			if (this.editing.partyMode === partyMode) return;			this.editing.partyMode = partyMode;			this.modeDropdownActive = false;		},		updatePartyMode() {			this.socket.emit(				"stations.updatePartyMode",				this.editing._id,				this.editing.partyMode,				res => {					if (res.status === "success") {						if (this.station)							this.station.partyMode = this.editing.partyMode;						// if (this.station)						// 	this.station.partyMode = this.editing.partyMode;						// this.stations.forEach((station, index) => {						// 	if (station._id === this.editing._id) {						// 		this.stations[						// 			index						// 		].partyMode = this.editing.partyMode;						// 		return this.editing.partyMode;						// 	}						// 	return false;						// });						return new Toast({							content: res.message,							timeout: 4000						});					}					return new Toast({ content: res.message, timeout: 8000 });				}			);		},		updateQueueLockLocal(locked) {			if (this.editing.locked === locked) return;			this.editing.locked = locked;			this.queueLockDropdownActive = false;		},		updateQueueLock() {			this.socket.emit("stations.toggleLock", this.editing._id, res => {				console.log(res);				if (res.status === "success") {					if (this.station) this.station.locked = res.data;					return new Toast({						content: `Toggled queue lock succesfully to ${res.data}`,						timeout: 4000					});				}				return new Toast({					content: "Failed to toggle queue lock.",					timeout: 8000				});			});		},		deleteStation() {			this.socket.emit("stations.remove", this.editing._id, res => {				if (res.status === "success")					this.closeModal({						sector: "station",						modal: "editStation"					});				return new Toast({ content: res.message, timeout: 8000 });			});		},		blurGenreInput() {			this.genreInputFocussed = false;		},		focusGenreInput() {			this.genreInputFocussed = true;		},		keydownGenreInput() {			clearTimeout(this.keydownGenreInputTimeout);			this.keydownGenreInputTimeout = setTimeout(() => {				if (this.genreInputValue.length > 1) {					this.genreAutosuggestItems = this.genres.filter(genre => {						return genre							.toLowerCase()							.startsWith(this.genreInputValue.toLowerCase());					});				} else this.genreAutosuggestItems = [];			}, 1000);		},		focusGenreContainer() {			this.genreAutosuggestContainerFocussed = true;		},		blurGenreContainer() {			this.genreAutosuggestContainerFocussed = false;		},		selectGenreAutosuggest(value) {			this.genreInputValue = value;		},		blurBlacklistGenreInput() {			this.blacklistGenreInputFocussed = false;		},		focusBlacklistGenreInput() {			this.blacklistGenreInputFocussed = true;		},		keydownBlacklistGenreInput() {			clearTimeout(this.keydownBlacklistGenreInputTimeout);			this.keydownBlacklistGenreInputTimeout = setTimeout(() => {				if (this.blacklistGenreInputValue.length > 1) {					this.blacklistGenreAutosuggestItems = this.genres.filter(						genre => {							return genre								.toLowerCase()								.startsWith(									this.blacklistGenreInputValue.toLowerCase()								);						}					);				} else this.blacklistGenreAutosuggestItems = [];			}, 1000);		},		focusBlacklistGenreContainer() {			this.blacklistGenreAutosuggestContainerFocussed = true;		},		blurBlacklistGenreContainer() {			this.blacklistGenreAutosuggestContainerFocussed = false;		},		selectBlacklistGenreAutosuggest(value) {			this.blacklistGenreInputValue = value;		},		addTag(type) {			if (type === "genres") {				const genre = this.genreInputValue.toLowerCase().trim();				if (this.editing.genres.indexOf(genre) !== -1)					return new Toast({						content: "Genre already exists",						timeout: 3000					});				if (genre) {					this.editing.genres.push(genre);					this.genreInputValue = "";					return false;				}				return new Toast({					content: "Genre cannot be empty",					timeout: 3000				});			}			if (type === "blacklist-genres") {				const genre = this.blacklistGenreInputValue					.toLowerCase()					.trim();				if (this.editing.blacklistedGenres.indexOf(genre) !== -1)					return new Toast({						content: "Blacklist genre already exists",						timeout: 3000					});				if (genre) {					this.editing.blacklistedGenres.push(genre);					this.blacklistGenreInputValue = "";					return false;				}				return new Toast({					content: "Blacklist genre cannot be empty",					timeout: 3000				});			}			return false;		},		removeTag(type, index) {			if (type === "genres") this.editing.genres.splice(index, 1);			else if (type === "blacklist-genres")				this.editing.blacklistedGenres.splice(index, 1);		},		...mapActions("modals", ["closeModal"])	},	components: { Modal, PlaylistItem }};</script><style lang="scss">@import "styles/global.scss";.night-mode {	.modal-card,	.modal-card-head,	.modal-card-body,	.modal-card-foot {		background-color: $night-mode-secondary;	}	.section {		background-color: #111 !important;		border: 0 !important;	}	.label,	p,	strong {		color: #ddd;	}}.edit-station-modal {	.modal-card-title {		text-align: center;		margin-left: 24px;	}	.modal-card {		width: 800px;		font-size: 16px;		.modal-card-body {			padding: 16px;			display: flex;		}	}}.section {	border: 1px solid #a3e0ff;	background-color: #f4f4f4;	border-radius: 5px;	padding: 16px;}.left-section {	width: 595px;	display: grid;	gap: 16px;	grid-template-rows: min-content min-content auto;	.control {		input {			width: 100%;			height: 36px;		}		.add-button {			width: 32px;			&.blue {				background-color: $musareBlue !important;			}			&.red {				background-color: $red !important;			}			i {				font-size: 32px;			}		}	}	.col {		> div {			position: relative;		}	}	.list-item-circle {		width: 16px;		height: 16px;		border-radius: 8px;		cursor: pointer;		margin-right: 8px;		float: left;		-webkit-touch-callout: none;		-webkit-user-select: none;		-khtml-user-select: none;		-moz-user-select: none;		-ms-user-select: none;		user-select: none;		&.blue {			background-color: $musareBlue;			i {				color: $musareBlue;			}		}		&.red {			background-color: $red;			i {				color: $red;			}		}		i {			font-size: 14px;			margin-left: 1px;		}	}	.list-item-circle:hover,	.list-item-circle:focus {		i {			color: white;		}	}	.list-item > p {		line-height: 16px;		word-wrap: break-word;		width: calc(100% - 24px);		left: 24px;		float: left;		margin-bottom: 8px;	}	.list-item:last-child > p {		margin-bottom: 0;	}	.autosuggest-container {		position: absolute;		background: white;		width: calc(100% + 1px);		top: 57px;		z-index: 200;		overflow: auto;		max-height: 100%;		clear: both;		.autosuggest-item {			padding: 8px;			display: block;			border: 1px solid #dbdbdb;			margin-top: -1px;			line-height: 16px;			cursor: pointer;			-webkit-user-select: none;			-ms-user-select: none;			-moz-user-select: none;			user-select: none;		}		.autosuggest-item:hover,		.autosuggest-item:focus {			background-color: #eee;		}		.autosuggest-item:first-child {			border-top: none;		}		.autosuggest-item:last-child {			border-radius: 0 0 3px 3px;		}	}}.right-section {	width: 157px;	min-height: 375px;	margin-left: 16px;	display: grid;	gap: 16px;	grid-template-rows: min-content min-content min-content;	.button-wrapper {		display: flex;		flex-direction: column;	}	button {		width: 100%;		height: 36px;		border: 0;		border-radius: 3px;		font-size: 18px;		color: white;		box-shadow: 0px 1px 3px rgba(0, 0, 0, 0.25);		display: block;		text-align: center;		justify-content: center;		display: inline-flex;		-ms-flex-align: center;		align-items: center;		-moz-user-select: none;		user-select: none;		cursor: pointer;		margin-bottom: 10px;		padding: 0;		&.red {			background-color: $red;		}		&.green {			background-color: $green;		}		&.blue {			background-color: $musareBlue;		}		&.orange {			background-color: $light-orange;		}		&.yellow {			background-color: $yellow;		}		i {			font-size: 20px;			margin-right: 4px;		}	}}.col {	display: grid;	grid-column-gap: 16px;}.col-1 {	grid-template-columns: auto;}.col-2 {	grid-template-columns: auto auto;}.slide-down-enter-active {	transition: transform 0.25s;}.slide-down-enter {	transform: translateY(-10px);}#playlists {	height: 168px;	overflow: auto;}.modal-card {	overflow: auto;}.modal-card-body {	overflow: unset;}</style>
 |