class playersSearcher { constructor(context_type, context_id) { this.context_type = context_type this.context_id = context_id this.searchType = "by_name" this.query = "" this.page = 1 this.successCallback = () => {} this.errorCallback = () => {} this.beforesendCallback = () => {} this.clearContainer = () => {} } execute() { $.ajax({ type: "POST", url: "/audios/context", data: { context: this.context_type, hash: u("meta[name=csrf]").attr("value"), page: this.page, query: this.query, context_entity: this.context_id, type: this.searchType, returnPlayers: 1, }, beforeSend: () => { this.beforesendCallback() }, error: () => { this.errorCallback() }, success: (response) => { this.successCallback(response, this) } }) } movePage(page) { this.page = page this.execute() } } window.player = new class { context = { object: {}, pagesCount: 0, count: 0, playedPages: [], } __linked_player_id = null current_track_id = 0 tracks = [] get timeType() { return localStorage.getItem('audio.timeType') ?? 0 } set timeType(value) { localStorage.setItem('audio.timeType', value) } get audioPlayer() { return this.__realAudioPlayer } get linkedInlinePlayer() { if(!this.__linked_player_id) { return null; } return u('#' + this.__linked_player_id) } get ajaxPlayer() { return u('#ajax_audio_player') } get uiPlayer() { return u('.bigPlayer') } get currentTrack() { return this.__findTrack(this.current_track_id) } get previousTrack() { const current = this.__findTrack(this.current_track_id, true) return this.__findByIndex(current - 1) } get nextTrack() { const current = this.__findTrack(this.current_track_id, true) return this.__findByIndex(current + 1) } async init(input_context) { let context = Object.assign({ url: location.pathname + location.search }, input_context) this.context.object = !input_context ? null : context this.__realAudioPlayer = document.createElement("audio") this.dashPlayer = dashjs.MediaPlayer().create() await this.loadContext(input_context ? context.page : 0) this.initEvents() this.__setMediaSessionActions() } initEvents() { this.audioPlayer.ontimeupdate = () => { const current_track = this.currentTrack if(!current_track) { return } const time = this.audioPlayer.currentTime const ps = ((time * 100) / current_track.length).toFixed(3) this.uiPlayer.find(".time").html(fmtTime(time)) this.__updateTime(time) if (ps <= 100) { this.uiPlayer.find(".track .selectableTrack .slider").attr('style', `left:${ ps}%`); if(this.linkedInlinePlayer) { this.linkedInlinePlayer.find(".subTracks .lengthTrackWrapper .slider").attr('style', `left:${ ps}%`) this.linkedInlinePlayer.find('.mini_timer .nobold').html(fmtTime(time)) } if(this.ajaxPlayer) { this.ajaxPlayer.find('#aj_player_track_length .slider').attr('style', `left:${ ps}%`) this.ajaxPlayer.find('#aj_player_track_name #aj_time').html(fmtTime(time)) } } } this.audioPlayer.onvolumechange = () => { const volume = this.audioPlayer.volume; const ps = Math.ceil((volume * 100) / 1); if (ps <= 100) { this.uiPlayer.find(".volumePanel .selectableTrack .slider").attr('style', `left:${ ps}%`); if(this.linkedInlinePlayer) { this.linkedInlinePlayer.find(".subTracks .volumeTrackWrapper .slider").attr('style', `left:${ ps}%`) } if(this.ajaxPlayer) { this.ajaxPlayer.find('#aj_player_volume .slider').attr('style', `left:${ ps}%`) } } localStorage.setItem('audio.volume', volume) } this.audioPlayer.onprogress = (e) => { u('.loaded_chunk').remove() const buffered = this.audioPlayer.buffered if (buffered.length > 0) { this.listen_coef += 1.25 const end = buffered.end(buffered.length - 1) const percentage = (end / window.player.audioPlayer.duration) * 100 if(this.uiPlayer.length > 0) { this.uiPlayer.find('.track .selectableTrackLoadProgress .load_bar').attr('style', `width:${percentage.toFixed(2)}%`) } if(this.linkedInlinePlayer) { this.linkedInlinePlayer.find('.lengthTrackWrapper .selectableTrackLoadProgress .load_bar').attr('style', `width:${percentage.toFixed(2)}%`) } } if(window.player.listen_coef > 10) { this.__countListen() window.player.listen_coef = -10 } } this.audioPlayer.onended = async (e) => { e.preventDefault() if(!this.nextTrack && window.player.context.playedPages.indexOf(1) == -1) { this.loadContext(1, false) await this.setTrack(this.__findByIndex(0).id) } else { this.playNextTrack() } if(this.linkedInlinePlayer && this.connectionType) { const parent = this.linkedInlinePlayer.closest(this.connectionType) const real_current = parent.find(`.audioEmbed[data-realid='${this.current_track_id}']`) u('.audioEntry .playerButton .playIcon.paused').removeClass('paused') u('.audioEntry .subTracks.shown').removeClass('shown') real_current.find('.subTracks').addClass('shown') this.linkPlayer(real_current) } } this.audioPlayer.volume = Number(localStorage.getItem('audio.volume') ?? 1) } async loadContext(page = 1, after = true) { if(!this.context.object) { return } if(this.context.playedPages.indexOf(page) != -1) { return } const form_data = new FormData switch(this.context.object.name) { case 'entity_audios': form_data.append('context', this.context.object.name) form_data.append('context_entity', this.context.object.entity_id) break case 'playlist_context': form_data.append('context', this.context.object.name) form_data.append('context_entity', this.context.object.entity_id) break case 'classic_search_context': // tidi riwriti form_data.append('context', this.context.object.name) form_data.append('context_entity', JSON.stringify({ 'order': this.context.object.order, 'invert': this.context.object.invert, 'genre': this.context.object.genre, 'only_performers': this.context.object.only_performers, 'with_lyrics': this.context.object.with_lyrics, 'query': this.context.object.query, })) break case 'alone_audio': form_data.append('context', this.context.object.name) form_data.append('context_entity', this.context.object.entity_id) } form_data.append('page', page) form_data.append("hash", u("meta[name=csrf]").attr("value")) this.context.playedPages.push(page) const req = await fetch('/audios/context', { method: 'POST', body: form_data }) const res = await req.json() if(!res.success) { makeError(tr("unable_to_load_queue")) return } this.context.pagesCount = res.pagesCount this.context.count = res.count this.__appendTracks(res.items, after) } linkPlayer(node) { this.__linked_player_id = node.attr('id') u(this.audioPlayer).trigger('volumechange') } async setTrack(id, ref = null) { if(!this.tracks || this.tracks.length < 1) { makeError('Context is not loaded yet', 'Red', 5000, 1489) return } if(window.__current_page_audio_context && (!this.context.object || this.context.object.url != location.pathname + location.search)) { console.log('Audio | Resetting context because of ajax :3') this.__renewContext() await this.loadContext(window.__current_page_audio_context.page ?? 1) if(!isNaN(parseInt(location.hash.replace('#', '')))) { const adp = parseInt(location.hash.replace('#', '')) await this.loadContext(adp) } else if((new URL(location.href)).searchParams.p) { const adp = (new URL(location.href)).searchParams.p await this.loadContext(adp) } this.__updateFace() } this.listen_coef = 0.0 const old_id = this.current_track_id this.current_track_id = id const c_track = this.currentTrack if(!c_track) { this.current_track_id = old_id makeError('Error playing audio: track not found') return } const protData = { "org.w3.clearkey": { "clearkeys": c_track.keys } }; u('.nowPlaying').removeClass('nowPlaying') this.__highlightActiveTrack() navigator.mediaSession.setPositionState({ duration: this.currentTrack.length }) this.__updateMediaSession() this.dashPlayer.initialize(this.audioPlayer, c_track.url, false); this.dashPlayer.setProtectionData(protData) if(!this.nextTrack && Math.max(...this.context["playedPages"]) < this.context["pagesCount"]) { await this.loadContext(Number(Math.max(...this.context["playedPages"])) + 1, true) } if(!this.previousTrack && (Math.min(...this.context["playedPages"]) > 1)) { await this.loadContext(Math.min(...this.context["playedPages"]) - 1, false) } this.is_closed = false if(ref != 'localstorage') { this.dump() } this.__updateFace() u(this.audioPlayer).trigger('volumechange') } switchTracks(id1, id2) { const first_audio = this.__findTrack(id1) const first_audio_index = this.__findTrack(id1, true) const second_audio = this.__findTrack(id2) const second_audio_index = this.__findTrack(id2, true) this.tracks[first_audio_index] = second_audio this.tracks[second_audio_index] = first_audio this.__updateFace() } appendTrack(object, after = true) { this.__appendTracks([object], after) } hasTrackWithId(id) { return this.__findTrack(id, true) != -1 } async play() { if(!this.currentTrack) { return } document.querySelectorAll('audio').forEach(el => el.pause()) await this.audioPlayer.play() this.__setFavicon() this.__updateFace() navigator.mediaSession.playbackState = "playing" } pause() { if(!this.currentTrack) { return } this.audioPlayer.pause() this.__setFavicon('paused') this.__updateFace() navigator.mediaSession.playbackState = "paused" } async playPreviousTrack() { if(!this.currentTrack || !this.previousTrack) { return } await this.setTrack(this.previousTrack.id) if(!this.currentTrack.available || this.currentTrack.withdrawn) { if(!this.previousTrack) { return } this.playPreviousTrack() } await this.play() } async playNextTrack() { if(!this.currentTrack || !this.nextTrack) { return } await this.setTrack(this.nextTrack.id) if(!this.currentTrack.available || this.currentTrack.withdrawn) { if(!this.nextTrack) { return } this.playNextTrack() } await this.play() } // fake shuffle async shuffle() { this.tracks.sort(() => Math.random() - 0.59) await this.setTrack(this.tracks.at(0).id) await this.play() } isAtAudiosPage() { return u('.bigPlayer').length > 0 } isAtCurrentContextPage() { return this.isAtAudiosPage() && this.context.object.url == location.pathname + location.search } dump() { const final = { context: this.context, current_track_id: this.current_track_id, tracks: this.tracks, time: this.audioPlayer.currentTime, } localStorage.setItem('audio.lastDump', JSON.stringify(final)) console.log('Audio | Tracks was dumped') } undump() { localStorage.setItem('audio.lastDump', null) } async loadDump(dump_object) { this.context = dump_object.context this.current_track_id = dump_object.current_track_id this.tracks = dump_object.tracks if(this.current_track_id) { await this.setTrack(this.current_track_id, 'localstorage') } if(dump_object.time) { setTimeout(() => { this.audioPlayer.currentTime = dump_object.time }, 1000) } } // Добавляем ощущение продуманности. __highlightActiveTrack() { u(`.audiosContainer .audioEmbed[data-realid='${this.current_track_id}'] .audioEntry, .audios_padding .audioEmbed[data-realid='${this.current_track_id}'] .audioEntry`).addClass('nowPlaying') } __renewContext() { let context = Object.assign({ url: location.pathname + location.search }, window.__current_page_audio_context) this.pause() this.__resetContext() this.context.object = context } __resetContext() { this.context = { object: {}, pagesCount: 0, count: 0, playedPages: [], } this.tracks = [] //this.__realAudioPlayer = document.createElement("audio") this.listen_coef = 0 } async _handlePageTransition() { const state = this.isAtAudiosPage() if(!state) { // AJAX audio player if(this.is_closed) { return } this.ajaxPlayer.removeClass('hidden') if(this.ajaxPlayer.length > 0) { return } else { if(this.audioPlayer.paused) { return } this.ajCreate() this.__updateFace() } } else { this.ajClose(false) this.is_closed = false if(this.tracks.length < 1) { if(window.__current_page_audio_context) { await this.init(window.__current_page_audio_context) } } } this.__linked_player_id = null if(this.currentTrack) { this.__updateFace() } this.__highlightActiveTrack() u(this.audioPlayer).trigger('volumechange') } __setFavicon(state = 'playing') { if(state == 'playing') { document.querySelector('link[rel="icon"], link[rel="shortcut icon"]').setAttribute("href", "/assets/packages/static/openvk/img/favicons/favicon24_paused.png") } else { document.querySelector('link[rel="icon"], link[rel="shortcut icon"]').setAttribute("href", "/assets/packages/static/openvk/img/favicons/favicon24_playing.png") } } __findTrack(id, return_index = false) { if(return_index) { return this.tracks.indexOf(this.tracks.find(item => item.id == id)) } return this.tracks.find(item => item.id == id) } __findByIndex(index) { return this.tracks[index] } __setMediaSessionActions() { navigator.mediaSession.setActionHandler('play', async () => { await window.player.play() }); navigator.mediaSession.setActionHandler('pause', () => { window.player.pause() }); navigator.mediaSession.setActionHandler('previoustrack', async () => { await window.player.playPreviousTrack() }); navigator.mediaSession.setActionHandler('nexttrack', async () => { await window.player.playNextTrack() }); navigator.mediaSession.setActionHandler("seekto", async (details) => { window.player.audioPlayer.currentTime = details.seekTime }); } __appendTracks(list, after = true) { if(after) { this.tracks = this.tracks.concat(list) } else { this.tracks = list.concat(this.tracks) } } __updateFace() { // Во второй раз перепутал next и back, но фиксить смысла уже нет. const _c = this.currentTrack const prev_button = this.uiPlayer.find('.nextButton') const next_button = this.uiPlayer.find('.backButton') if(!this.previousTrack) { prev_button.addClass('lagged') if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_previous').addClass('lagged') } } else { prev_button.removeClass('lagged') prev_button.attr('data-title', ovk_proc_strtr(escapeHtml(this.previousTrack.name), 50)) if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_previous').removeClass('lagged') } } if(!this.nextTrack) { next_button.addClass('lagged') if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_next').addClass('lagged') } } else { next_button.removeClass('lagged') next_button.attr('data-title', ovk_proc_strtr(escapeHtml(this.nextTrack.name), 50)) if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_next').removeClass('lagged') } } if(!this.audioPlayer.paused) { this.uiPlayer.find('.playButton').addClass('pause') if(this.linkedInlinePlayer) { this.linkedInlinePlayer.find('.playerButton .playIcon').addClass('paused') } if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_play_btn').addClass('paused') } } else { this.uiPlayer.find('.playButton').removeClass('pause') if(this.linkedInlinePlayer) { this.linkedInlinePlayer.find('.playerButton .playIcon').removeClass('paused') } if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_play_btn').removeClass('paused') } } if(_c) { this.uiPlayer.find('.trackInfo .trackName span').html(escapeHtml(_c.name)) this.uiPlayer.find('.trackInfo .trackPerformers').html('') const performers = _c.performer.split(', ') const lastPerformer = performers[performers.length - 1] performers.forEach(performer => { this.uiPlayer.find('.trackInfo .trackPerformers').append( `${performer.escapeHtml()}${(performer != lastPerformer ? ', ' : '')}`) }) } else { this.uiPlayer.find('.trackInfo .trackName span').html(tr('track_noname')) this.uiPlayer.find('.trackInfo .trackPerformers').html(`${tr('track_unknown')}`) } if(this.ajaxPlayer.length > 0) { this.ajaxPlayer.find('#aj_player_track_title b').html(escapeHtml(_c.performer)) this.ajaxPlayer.find('#aj_player_track_title span').html(escapeHtml(_c.name)) } u(`.tip_result`).remove() } __updateTime(new_time) { this.uiPlayer.find(".trackInfo .time").html(fmtTime(new_time)) if(this.timeType == 1) { this.uiPlayer.find(".trackInfo .elapsedTime").html(fmtTime(this.currentTrack.length)) } else { this.uiPlayer.find(".trackInfo .elapsedTime").html(getRemainingTime(this.currentTrack.length, new_time)) } } __updateMediaSession() { const album = document.querySelector(".playlistBlock") const cur = this.currentTrack navigator.mediaSession.metadata = new MediaMetadata({ title: escapeHtml(cur.name), artist: escapeHtml(cur.performer), album: album == null ? "OpenVK Audios" : escapeHtml(album.querySelector(".playlistInfo h4").innerHTML), artwork: [{ src: album == null ? "/assets/packages/static/openvk/img/song.jpg" : album.querySelector(".playlistCover img").src }], }) } async __countListen() { let playlist = 0 if(!this.listen_coef) { return false } if(this.context.object && this.context.object.name == 'playlist_context') { playlist = this.context.object.entity_id } const form_data = new FormData form_data.append('hash', u("meta[name=csrf]").attr("value")) form_data.append('playlist', playlist) if(this.context.object) { form_data.append('listened_from', this.context.object.url) } const req = await fetch(`/audio${this.currentTrack.id}/listen`, { method: 'POST', body: form_data }) const res = await req.json() if(res.success) { console.log('Listen is counted') } else { console.log('Listen is not counted ! ! !') } } ajClose(pause = true) { this.is_closed = true if(pause) { this.pause() } u('#ajax_audio_player').addClass('hidden') } ajReveal() { this.is_closed = false if(u('#ajax_audio_player').length == 0) { this.ajCreate() } u('#ajax_audio_player').removeClass('hidden') } ajCreate() { const previous_time_x = localStorage.getItem('audio.lastX') ?? 100 const previous_time_y = localStorage.getItem('audio.lastY') ?? scrollY const miniplayer_template = u(`
`) u('body').append(miniplayer_template) miniplayer_template.attr('style', `left:${previous_time_x}px;top:${previous_time_y}px`) miniplayer_template.find('#aj_player_close_btn').on('click', (e) => { this.ajClose() }) miniplayer_template.find('#aj_time').on('click', (e) => { if(window.player && window.player.context && window.player.context.object) { window.router.route(window.player.context.object.url) } }) $('#ajax_audio_player').draggable({ cursor: 'grabbing', containment: 'window', cancel: '#aj_player_track .selectableTrack, #aj_player_volume .selectableTrack, #aj_player_buttons', stop: function(e) { if(window.player.ajaxPlayer.length > 0) { const left = parseInt(window.player.ajaxPlayer.nodes[0].style.left) const top = parseInt(window.player.ajaxPlayer.nodes[0].style.top) localStorage.setItem('audio.lastX', left) localStorage.setItem('audio.lastY', top) } } }) } } document.addEventListener("DOMContentLoaded", async () => { let parsed = null try { parsed = JSON.parse(localStorage.getItem('audio.lastDump')) } catch(e) {} if(window.openvk.current_id == 0) { return } if(parsed) { await window.player.init(null) await window.player.loadDump(parsed) if(!window.player.isAtAudiosPage() && parsed.current_track_id) { window.player.ajReveal() window.player.__updateFace() u(window.player.audioPlayer).trigger('timeupdate') } } else { await window.player.init(window.__current_page_audio_context) } }) window.addEventListener('beforeunload', (e) => { window.player.dump() }) u(document).on('click', '.audioEntry .playerButton > .playIcon', async (e) => { const audioPlayer = u(e.target).closest('.audioEmbed') const id = Number(audioPlayer.attr('data-realid')) if(!window.player) { return } if(window.openvk.current_id == 0) { return } if(!window.player.hasTrackWithId(id) && !window.player.isAtAudiosPage()) { let _nodes = null if(u(e.target).closest('.attachments').length > 0) { window.player.connectionType = '.attachments' _nodes = u(e.target).closest('.attachments').find('.audioEmbed').nodes } else if(u(e.target).closest('.content_list').length > 0) { window.player.connectionType = '.content_list' _nodes = u(e.target).closest('.content_list').find('.audioEmbed').nodes } else if(u(e.target).closest('.generic_audio_list').length > 0) { window.player.connectionType = '.generic_audio_list' _nodes = u(e.target).closest('.generic_audio_list').find('.audioEmbed').nodes } else if(u(e.target).closest('.audiosInsert').length > 0) { window.player.connectionType = '.audiosInsert' _nodes = u(e.target).closest('.audiosInsert').find('.audioEmbed').nodes } window.player.tracks = [] _nodes.forEach(el => { const tempAudio = u(el) const name = tempAudio.attr('data-name').split(' — ') window.player.appendTrack({ 'id': Number(tempAudio.attr('data-realid')), 'available': true, // , судя по всему 'keys': JSON.parse(tempAudio.attr('data-keys')), 'length': Number(tempAudio.attr('data-length')), 'url': tempAudio.attr('data-url'), 'name': name[1], 'performer': name[0] }) }) } else if(!window.player.hasTrackWithId(id) && window.player.isAtAudiosPage()) { window.player.__renewContext() await window.player.loadContext(window.__current_page_audio_context.page ?? 1) if(!isNaN(parseInt(location.hash.replace('#', '')))) { const adp = parseInt(location.hash.replace('#', '')) await window.player.loadContext(adp) } else if((new URL(location.href)).searchParams.p) { const adp = (new URL(location.href)).searchParams.p await window.player.loadContext(adp) } } if(window.player.current_track_id != id) { await window.player.setTrack(id) } if(window.player.audioPlayer.paused) { await window.player.play() if(!window.player.isAtAudiosPage()) { u('.audioEntry .playerButton .playIcon.paused').removeClass('paused') u(e.target).addClass('paused') } } else { window.player.pause() } if(window.player.isAtAudiosPage()) { } else { window.player.linkPlayer(audioPlayer) u('.audioEntry .subTracks.shown').removeClass('shown') audioPlayer.find('.subTracks').addClass('shown') } }) u(document).on('click', '.bigPlayer .playButton, #ajax_audio_player #aj_player_play_btn', async (e) => { if(window.player.audioPlayer.paused) { await window.player.play() } else { window.player.pause() } }) u(document).on('click', '.bigPlayer .backButton, #ajax_audio_player #aj_player_next', async (e) => { await window.player.playNextTrack() }) u(document).on('click', '.bigPlayer .nextButton, #ajax_audio_player #aj_player_previous', async (e) => { await window.player.playPreviousTrack() }) u(document).on("click", ".bigPlayer .elapsedTime", (e) => { if(window.player.current_track_id == 0) return const res = window.player.timeType == 0 ? 1 : 0 window.player.timeType = res window.player.__updateTime(window.player.audioPlayer.currentTime) }) u(document).on("click", ".bigPlayer .additionalButtons .repeatButton, #ajax_audio_player #aj_player_repeat", (e) => { if(window.player.current_track_id == 0) return const targ = u(e.target) targ.toggleClass("pressed") if(targ.hasClass("pressed")) window.player.audioPlayer.loop = true else window.player.audioPlayer.loop = false }) u(document).on("click", ".bigPlayer .additionalButtons .shuffleButton", async (e) => { if(window.player.current_track_id == 0) return await window.player.shuffle() }) u(document).on("click", ".bigPlayer .additionalButtons .deviceButton", (e) => { if(window.player.current_track_id == 0) return e.target.classList.toggle("pressed") window.player.audioPlayer.muted = e.target.classList.contains("pressed") }) u(document).on('keydown', (e) => { if(document.activeElement.closest('.page_header')) { return } if(!window.player || !window.player.isAtAudiosPage()) { return } if([32, 37, 39, 107, 109].includes(e.keyCode)) { if(window.messagebox_stack.length > 0) return e.preventDefault() } const arrow_click_offset = 3 const volume_offset = 0.1 switch(e.keyCode) { case 32: window.player.listen_coef -= 0.1 window.player.uiPlayer.find('.playButton').trigger('click') break case 37: window.player.listen_coef -= 0.5 window.player.audioPlayer.currentTime = window.player.audioPlayer.currentTime - arrow_click_offset break case 39: window.player.listen_coef -= 0.5 window.player.audioPlayer.currentTime = window.player.audioPlayer.currentTime + arrow_click_offset break case 107: window.player.audioPlayer.volume = Math.max(Math.min(window.player.audioPlayer.volume + volume_offset, 1), 0) break case 109: window.player.audioPlayer.volume = Math.max(Math.min(window.player.audioPlayer.volume - volume_offset, 1), 0) break default: //console.log(e.keyCode) break } }) u(document).on('keyup', (e) => { if(document.activeElement.closest('.page_header')) { return } if(!window.player || !window.player.isAtAudiosPage()) { return } if([87, 65, 83, 68, 82, 77].includes(e.keyCode)) { if(window.messagebox_stack.length > 0) return e.preventDefault() } switch(e.keyCode) { case 87: case 65: window.player.playPreviousTrack() break case 83: case 68: window.player.playNextTrack() break case 82: document.querySelector(".bigPlayer .additionalButtons .repeatButton").click() break case 77: document.querySelector(".bigPlayer .additionalButtons .deviceButton").click() break } }) u(document).on("mousemove click mouseup", ".bigPlayer .trackPanel .selectableTrack, .audioEntry .subTracks .lengthTrackWrapper .selectableTrack, #aj_player_track_length .selectableTrack", (e) => { if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) return if(u('.ui-draggable-dragging').length > 0) { return } function __defaultAction(i_time) { window.player.listen_coef -= 0.5 window.player.audioPlayer.currentTime = i_time } const taggart = u(e.target).closest('.selectableTrack') const parent = taggart.parent() const rect = taggart.nodes[0].getBoundingClientRect() const width = e.clientX - rect.left const time = Math.ceil((width * window.player.currentTrack.length) / (rect.right - rect.left)) if(e.type == "mousemove") { let buttonsPresseed = _bsdnUnwrapBitMask(e.buttons) if(buttonsPresseed[0]) __defaultAction(time) } if(e.type == 'click' || e.type == 'mouseup') { __defaultAction(time) } if(parent.find('.tip_result').length < 1) { parent.append(`
`) } parent.find('.tip_result').html(fmtTime(time)).attr('style', `left:min(${width - 15}px, 315.5px)`) }) u(document).on("mouseout", ".bigPlayer .trackPanel .selectableTrack, .audioEntry .subTracks .lengthTrackWrapper .selectableTrack, #aj_player_track_length .selectableTrack", (e) => { if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) return u(e.target).closest('.selectableTrack').parent().find('.tip_result').remove() }) u(document).on("mousemove click mouseup", ".bigPlayer .volumePanelTrack .selectableTrack, .audioEntry .subTracks .volumeTrack .selectableTrack, #aj_player_volume .selectableTrack", (e) => { if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) return if(u('.ui-draggable-dragging').length > 0) { return } function __defaultAction(i_volume) { window.player.audioPlayer.volume = i_volume } const rect = u(e.target).closest(".selectableTrack").nodes[0].getBoundingClientRect(); const taggart = u(e.target).closest('.selectableTrack') const parent = taggart.parent() const width = e.clientX - rect.left const volume = Math.max(0, (width * 1) / (rect.right - rect.left)) if(e.type == "mousemove") { let buttonsPresseed = _bsdnUnwrapBitMask(e.buttons) if(buttonsPresseed[0]) __defaultAction(volume) } if(e.type == 'click' || e.type == 'mouseup') { __defaultAction(volume) } if(parent.find('.tip_result').length < 1) { parent.append(`
`) } parent.find('.tip_result').html((volume * 100).toFixed(0) + '%').attr('style', `left:${width - 15}px`) }) u(document).on("mouseout", ".bigPlayer .volumePanelTrack .selectableTrack, .audioEntry .subTracks .volumeTrack .selectableTrack, #aj_player_volume .selectableTrack", (e) => { if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) return u(e.target).closest('.selectableTrack').parent().find('.tip_result').remove() }) u(document).on('dragstart', '.audiosContainer .audioEmbed', (e) => { u(e.target).closest('.audioEmbed').addClass('currently_dragging') return }) u(document).on('dragover', '.audiosContainer .audioEmbed', (e) => { e.preventDefault() const target = u(e.target).closest('.audioEmbed') const current = u('.audioEmbed.currently_dragging') if(current.length < 1) { return } if(target.nodes[0].dataset.id != current.nodes[0].dataset.id) { target.addClass('dragged') } return }) u(document).on('dragend', '.audiosContainer .audioEmbed', (e) => { //console.log(e) u(e.target).closest('.audioEmbed').removeClass('dragged') return }) // TODO: write changes on server side (audio.reorder) u(document).on("drop", '.audiosContainer', function(e) { const current = u('.audioEmbed.currently_dragging') if(e.dataTransfer.types.includes('Files')) { e.preventDefault() e.dataTransfer.dropEffect = 'move' } else if(e.dataTransfer.types.length < 1 || e.dataTransfer.types.includes('text/uri-list')) { e.preventDefault() u('.audioEmbed.currently_dragging').removeClass('currently_dragging') const target = u(e.target).closest('.audioEmbed') const first_id = Number(current.attr('data-realid')) const second_id = Number(target.attr('data-realid')) const first_html = target.nodes[0].outerHTML const second_html = current.nodes[0].outerHTML current.nodes[0].outerHTML = first_html target.nodes[0].outerHTML = second_html window.player.switchTracks(first_id, second_id) } }) u(document).on('contextmenu', '.bigPlayer, .audioEmbed, #ajax_audio_player', (e) => { e.preventDefault() u('#ctx_menu').remove() const ctx_type = u(e.target).closest('.bigPlayer, #ajax_audio_player').length > 0 ? 'main_player' : 'mini_player' const parent = e.target.closest('.ctx_place') if(!parent) { return } const rect = parent.getBoundingClientRect() let x, y; let rx = rect.x + window.scrollX, ry = rect.y + window.scrollY x = e.pageX - rx y = e.pageY - ry const ctx_u = u(`
${tr('copy_link_to_audio')} ${ctx_type == 'main_player' ? ` ${tr('repeat_tip')} ${tr('shuffle_tip')} ${tr('mute_tip_noun')} ` : ''} ${ctx_type == 'mini_player' ? ` ${tr('audio_ctx_play_next')} ` : ''} ${tr('audio_ctx_add_to_group')} ${tr('audio_ctx_add_to_playlist')} ${ctx_type == 'main_player' ? ` ${tr('audio_ctx_clear_context')}` : ''} ${ctx_type == 'main_player' ? `BigPlayer v1.1 by MrIlyew` : ''}
`) u(parent).append(ctx_u) ctx_u.find('#audio_ctx_copy').on('click', async (e) => { if(ctx_type == 'main_player') { if(window.player.current_track_id == 0) { makeError(tr('copy_link_to_audio_error_not_selected_track'), 'Red', 4000, 80) return } const url = location.origin + `/audio${window.openvk.current_id}_${window.player.current_track_id}` await copyToClipboard(url) } else { const url = location.origin + `/audio${window.openvk.current_id}_${u(e.target).closest('.audioEmbed').attr('data-realid')}` await copyToClipboard(url) } }) ctx_u.find('#audio_ctx_repeat').on('click', () => { if(window.player.current_track_id == 0) { return } if(!window.player.audioPlayer.loop) { window.player.audioPlayer.loop = true window.player.uiPlayer.find('.repeatButton').addClass('pressed') } else { window.player.audioPlayer.loop = false window.player.uiPlayer.find('.repeatButton').removeClass('pressed') } }) ctx_u.find('#audio_ctx_shuffle').on('click', async () => { if(window.player.current_track_id == 0) { return } await window.player.shuffle() }) ctx_u.find('#audio_ctx_mute').on('click', async () => { if(window.player.current_track_id == 0) { return } window.player.uiPlayer.find('.deviceButton').toggleClass('pressed') window.player.audioPlayer.muted = window.player.uiPlayer.find('.deviceButton').hasClass('pressed') }) ctx_u.find('#audio_ctx_add_to_group').on('click', async () => { if(ctx_type == 'main_player') { if(window.player.current_track_id == 0) { return } __showAudioAddDialog(window.player.current_track_id) } else { __showAudioAddDialog(Number(u(e.target).closest('.audioEmbed').attr('data-realid'))) } }) ctx_u.find('#audio_ctx_add_to_playlist').on('click', async () => { if(ctx_type == 'main_player') { if(window.player.current_track_id == 0) { return } __showAudioAddDialog(window.player.current_track_id, 'playlist') } else { __showAudioAddDialog(Number(u(e.target).closest('.audioEmbed').attr('data-realid')), 'playlist') } }) ctx_u.find('#audio_ctx_play_next').on('click', (ev) => { const current_id = window.player.current_track_id const move_id = Number(u(e.target).closest('.audioEmbed').attr('data-realid')) if(current_id == 0) { return } if(current_id == move_id) { return } const current_index = window.player.__findTrack(current_id, true) const next_track = window.player.__findTrack(move_id) const next_track_player = u(`.audioEmbed[data-realid='${window.player.nextTrack.id}']`) const moving_track_player = u(`.audioEmbed[data-realid='${move_id}']`) window.player.tracks.splice(current_index + 1, 0, next_track) if(next_track_player.length > 0 && moving_track_player.length > 0) { next_track_player.nodes[0].outerHTML = moving_track_player.nodes[0].outerHTML + next_track_player.nodes[0].outerHTML moving_track_player.remove() } }) ctx_u.find('#audio_ctx_clear_context').on('click', (ev) => { const old_url = window.player.context.object.url window.player.pause() window.player.ajClose() window.player.__resetContext() window.player.__updateFace() window.player.undump() window.router.route(old_url) }) }) u(document).on("click", ".musicIcon.edit-icon", (e) => { const player = e.target.closest(".audioEmbed") const id = Number(player.dataset.realid) const performer = e.target.dataset.performer const name = e.target.dataset.title const genre = player.dataset.genre const lyrics = e.target.dataset.lyrics MessageBox(tr("edit_audio"), `
${tr("performer")}
${tr("audio_name")}
${tr("genre")}
${tr("lyrics")}


${tr("fully_delete_audio")}
`, [tr("save"), tr("cancel")], [ function() { const t_name = $(".ovk-diag-body input[name=name]").val(); const t_perf = $(".ovk-diag-body input[name=performer]").val(); const t_genre = $(".ovk-diag-body select[name=genre]").val(); const t_lyrics = $(".ovk-diag-body textarea[name=lyrics]").val(); const t_explicit = document.querySelector(".ovk-diag-body input[name=explicit]").checked; const t_unlisted = document.querySelector(".ovk-diag-body input[name=searchable]").checked; $.ajax({ type: "POST", url: `/audio${id}/action?act=edit`, data: { name: t_name, performer: t_perf, genre: t_genre, lyrics: t_lyrics, unlisted: Number(t_unlisted), explicit: Number(t_explicit), hash: u("meta[name=csrf]").attr("value") }, success: (response) => { if(response.success) { const perf = player.querySelector(".performer a") perf.innerHTML = escapeHtml(response.new_info.performer) perf.setAttribute("href", "/search?q=§ion=audios&order=listens&only_performers=on&q="+response.new_info.performer) e.target.setAttribute("data-performer", escapeHtml(response.new_info.performer)) e.target.setAttribute("data-title", escapeHtml(response.new_info.name)) e.target.setAttribute("data-lyrics", response.new_info.lyrics_unformatted) e.target.setAttribute("data-explicit", Number(response.new_info.explicit)) e.target.setAttribute("data-searchable", Number(!response.new_info.unlisted)) player.setAttribute("data-genre", response.new_info.genre) let name = player.querySelector(".title") name.innerHTML = escapeHtml(response.new_info.name) if(response.new_info.lyrics_unformatted != "") { if(player.querySelector(".lyrics") != null) { player.querySelector(".lyrics").innerHTML = response.new_info.lyrics player.querySelector(".title").classList.add("withLyrics") } else { player.insertAdjacentHTML("beforeend", `
${response.new_info.lyrics}
`) player.querySelector(".title").classList.add("withLyrics") } } else { $(player.querySelector(".lyrics")).remove() player.querySelector(".title").classList.remove("withLyrics") } if(Number(response.new_info.explicit) == 1) { if(!player.querySelector(".mediaInfo .explicitMark")) player.querySelector(".mediaInfo").insertAdjacentHTML("beforeend", `
`) } else { $(player.querySelector(".mediaInfo .explicitMark")).remove() } let url = new URL(location.href) let page = "1" if(url.searchParams.p != null) page = String(url.searchParams.p) } else fastError(response.flash.message) } }); }, Function.noop ]); window.openvk.audio_genres.forEach(elGenre => { document.querySelector(".ovk-diag-body select[name=genre]").insertAdjacentHTML("beforeend", ` `) }) u(".ovk-diag-body #_fullyDeleteAudio").on("click", (e) => { MessageBox(tr('confirm'), tr('confirm_deleting_audio'), [tr('yes'), tr('no')], [() => { $.ajax({ type: "POST", url: `/audio${id}/action?act=delete`, data: { hash: u("meta[name=csrf]").attr("value") }, success: (response) => { u("body").removeClass("dimmed"); u(".ovk-diag-cont").remove(); document.querySelector("html").style.overflowY = "scroll" if(response.success) u(player).remove() else fastError(response.flash.message) } }); }, () => {Function.noop}]) }) }) u(document).on("click", ".title.withLyrics", (e) => { const parent = e.currentTarget.closest(".audioEmbed") parent.querySelector(".lyrics").classList.toggle("showed") }) $(document).on("click", ".musicIcon.remove-icon", (e) => { e.stopImmediatePropagation() const id = e.currentTarget.dataset.id if(e.detail > 1 || e.altKey) { const player = e.target.closest('.audioEmbed') player.querySelector('.add-icon-group').click() return } let formdata = new FormData() formdata.append("hash", u("meta[name=csrf]").attr("value")) ky.post(`/audio${id}/action?act=remove`, { hooks: { beforeRequest: [ (_request) => { e.target.classList.add("lagged") } ], afterResponse: [ async (_request, _options, response) => { let json = await response.json() if(json.success) { e.target.classList.remove("remove-icon") e.target.classList.add("add-icon") e.target.classList.remove("lagged") let withd = e.target.closest(".audioEmbed.withdrawn") if(withd != null) u(withd).remove() } else fastError(json.flash.message) } ] }, body: formdata }) }) $(document).on("click", ".musicIcon.remove-icon-group", (e) => { e.stopImmediatePropagation() let id = e.currentTarget.dataset.id let formdata = new FormData() formdata.append("hash", u("meta[name=csrf]").attr("value")) formdata.append("club", e.currentTarget.dataset.club) ky.post(`/audio${id}/action?act=remove_club`, { hooks: { beforeRequest: [ (_request) => { e.currentTarget.classList.add("lagged") } ], afterResponse: [ async (_request, _options, response) => { let json = await response.json() if(json.success) $(e.currentTarget.closest(".audioEmbed")).remove() else fastError(json.flash.message) } ] }, body: formdata }) }) function __showAudioAddDialog(id, current_tab = 'club') { const body = `
${tr('add_audio_limitations')}
` MessageBox(tr("add_audio"), body, [tr("cancel"), tr("add")], [Function.noop, () => { const ids = [] u('#_content .entity_vertical_list_item').nodes.forEach(item => { const _checkbox = item.querySelector(`input[type='checkbox'][name='add_to']`) if(_checkbox.checked) { ids.push(item.dataset.id) } }) if(ids.length < 1 || ids.length > 10) { return } switch(current_tab) { case 'club': $.ajax({ type: "POST", url: `/audio${id}/action?act=add_to_club`, data: { hash: u("meta[name=csrf]").attr("value"), clubs: ids.join(',') }, success: (response) => { if(!response.success) fastError(response.flash.message) else NewNotification(tr("audio_was_successfully_added"), '') } }) break case 'playlist': $.ajax({ type: "POST", url: `/audio${id}/action?act=add_to_playlist`, data: { hash: u("meta[name=csrf]").attr("value"), playlists: ids.join(',') }, success: (response) => { if(!response.success) fastError(response.flash.message) else NewNotification(tr("audio_was_successfully_added"), '') } }) break } }]) u(".ovk-diag-body").attr('style', 'padding:0px;height: 260px;') async function switchTab(tab = 'club') { current_tab = tab u(`#_addAudioAdditional .mb_tab`).attr('id', 'ki') u(`#_addAudioAdditional .mb_tab[data-name='${tab}']`).attr('id', 'active') switch(tab) { case 'club': u("#_content").html(`
`) if(window.openvk.writeableClubs == null) { u('.entity_vertical_list').append(`
`) try { window.openvk.writeableClubs = await API.Groups.getWriteableClubs() } catch (e) { u("#_content").html(tr("no_access_clubs")) return } u('.entity_vertical_list #gif_loader').remove() } window.openvk.writeableClubs.forEach(el => { u("#_content .entity_vertical_list").append(` `) }) break case 'playlist': const per_page = 10 let page = 0 u("#_content").html(`
`) async function recievePlaylists(s_page) { res = await fetch(`/method/audio.searchAlbums?auth_mechanism=roaming&query=&limit=10&offset=${s_page * per_page}&from_me=1`) res = await res.json() return res } function appendPlaylists(response) { response.items.forEach(el => { u("#_content .entity_vertical_list").append(` `) }) if(response.count > per_page * page) { u("#_content .entity_vertical_list").append(`${tr('show_more')}`) } } if(window.openvk.writeablePlaylists == null) { u('.entity_vertical_list').append(`
`) try { res = await recievePlaylists(page) page += 1 window.openvk.writeablePlaylists = res.response if(!window.openvk.writeablePlaylists || window.openvk.writeablePlaylists.count < 1) { throw new Error } } catch (e) { u("#_content").html(tr("no_access_playlists")) return } u('.entity_vertical_list #gif_loader').remove() } appendPlaylists(window.openvk.writeablePlaylists) u('#_addAudioAdditional').on('click', '#_pladdwinshowmore', async (e) => { e.target.outerHTML = '' res = await recievePlaylists(page) page += 1 appendPlaylists(res.response) }) break } } switchTab(current_tab) u("#_addAudioAdditional").on("click", ".mb_tab a", async (e) => { await switchTab(u(e.target).closest('.mb_tab').attr('data-name')) }) u("#_addAudioAdditional").on("click", "input[name='add_to']", async (e) => { if(u(`input[name='add_to']:checked`).length > 10) { e.preventDefault() } }) } $(document).on("click", ".musicIcon.add-icon-group", async (ev) => { const id = Number(ev.target.dataset.id) __showAudioAddDialog(id) }) $(document).on("click", ".musicIcon.add-icon", (e) => { const id = e.currentTarget.dataset.id if(e.detail > 1 || e.altKey) { const player = e.target.closest('.audioEmbed') player.querySelector('.add-icon-group').click() return } let formdata = new FormData() formdata.append("hash", u("meta[name=csrf]").attr("value")) ky.post(`/audio${id}/action?act=add`, { hooks: { beforeRequest: [ (_request) => { e.target.classList.add("lagged") } ], afterResponse: [ async (_request, _options, response) => { let json = await response.json() if(json.success) { e.target.classList.remove("add-icon") e.target.classList.add("remove-icon") e.target.classList.remove("lagged") } else fastError(json.flash.message) } ] }, body: formdata }) }) $(document).on("click", "#_deletePlaylist", (e) => { let id = e.currentTarget.dataset.id MessageBox(tr("warning"), tr("sure_delete_playlist"), [tr("yes"), tr("no")], [() => { $.ajax({ type: "POST", url: `/playlist${id}/action?act=delete`, data: { hash: u("meta[name=csrf]").attr("value"), }, beforeSend: () => { e.currentTarget.classList.add("lagged") }, success: (response) => { if(response.success) { window.router.route("/playlists" + response.id) } else { fastError(response.flash.message) } } }) }, Function.noop]) }) function showAudioAttachment(type = 'form', form = null) { const msg = new CMessageBox({ title: tr("select_audio"), body: `
`, buttons: [tr('close')], callbacks: [Function.noop], }) msg.getNode().find('.ovk-diag-body').attr('style', 'padding:0px;height:335px') msg.getNode().attr('style', 'width:580px') let searcher = new playersSearcher("entity_audios", 0) searcher.successCallback = (response, thisc) => { let domparser = new DOMParser() let result = domparser.parseFromString(response, "text/html") let pagesCount = result.querySelector("input[name='pagesCount']").value let count = Number(result.querySelector("input[name='count']").value) if(count < 1) { document.querySelector(".audiosInsert").innerHTML = thisc.context_type == "entity_audios" ? tr("no_audios_thisuser") : tr("no_results") return } result.querySelectorAll(".audioEmbed").forEach(el => { let id = 0 if(type == 'form') { id = el.dataset.prettyid } else { id = el.dataset.realid } let is_attached = false if(type == 'form') { is_attached = (u(form).find(`.post-vertical .vertical-attachment[data-id='${id}']`)).length > 0 } else { is_attached = (u(form).find(`.PE_audios .vertical-attachment[data-id='${id}']`)).length > 0 } document.querySelector(".audiosInsert").insertAdjacentHTML("beforeend", `
${el.outerHTML}
${is_attached ? tr("detach_audio") : tr("attach_audio")}
`) }) u("#loader").remove() u('#show_more').remove() if(thisc.page < pagesCount) { document.querySelector(".audiosInsert").insertAdjacentHTML("beforeend", `
${tr("show_more_audios")}
`) } } searcher.errorCallback = () => { fastError("Error when loading players.") } searcher.beforesendCallback = () => { document.querySelector(".audiosInsert").insertAdjacentHTML("beforeend", ``) } searcher.clearContainer = () => { document.querySelector(".audiosInsert").innerHTML = "" } searcher.movePage(1) u(".audiosInsert").on("click", "#show_more", async (e) => { u(e.target).closest('#show_more').addClass('lagged') searcher.movePage(Number(e.currentTarget.dataset.page)) }) u(".searchBox input").on("change", async (e) => { if(e.currentTarget.value === document.querySelector(".searchBox input").value) { searcher.clearContainer() if(e.currentTarget.value == "") { searcher.context_type = "entity_audios" searcher.context_id = 0 searcher.query = "" searcher.movePage(1) return } searcher.context_type = "search_context" searcher.context_id = 0 searcher.query = e.currentTarget.value searcher.movePage(1) return; } }) u(".searchBox select").on("change", async (e) => { searcher.clearContainer() searcher.searchType = e.currentTarget.value $(".searchBox input").trigger("change") return; }) u(".audiosInsert").on("click", ".attachAudio", (ev) => { const id = ev.currentTarget.dataset.attachmentdata let is_attached = false if(type == 'form') { is_attached = u(form).find(`.post-vertical .vertical-attachment[data-id='${id}']`).length > 0 } else { is_attached = u(form).find(`.PE_audios .vertical-attachment[data-id='${id}']`).length > 0 } // 04.11.2024 19:03 // 30.11.2024 19:03 if(is_attached) { if(type == 'form') { u(form).find(`.post-vertical .vertical-attachment[data-id='${id}']`).remove() } else { u(form).find(`.PE_audios .vertical-attachment[data-id='${id}']`).remove() } u(ev.currentTarget).find("span").html(tr("attach_audio")) } else { if(type == 'form' && u(form).find(`.upload-item`).length > window.openvk.max_attachments) { makeError(tr('too_many_attachments'), 'Red', 10000, 1) return } u(ev.currentTarget).find("span").html(tr("detach_audio")) const header = u(ev.currentTarget).closest('.audio_attachment_header') const player = header.find('.player_part') u(form).find(type == 'form' ? ".post-vertical" : '.PE_audios').append(`
${player.html()}
`) } }) } $(document).on("click", "#__audioAttachment", (e) => { const form = e.target.closest("form") showAudioAttachment('form', form) }) $(document).on("click", ".audioEmbed.processed .playerButton", (e) => { const msg = new CMessageBox({ title: tr('error'), body: tr('audio_embed_processing'), unique_name: 'processing_notify', buttons: [tr('ok')], callbacks: [Function.noop] }) }) $(document).on("click", ".audioEmbed.withdrawn", (e) => { const msg = new CMessageBox({ title: tr('error'), body: tr('audio_embed_withdrawn'), unique_name: 'withdrawn_notify', buttons: [tr('ok')], callbacks: [Function.noop] }) }) $(document).on("click", ".musicIcon.report-icon", (e) => { MessageBox(tr("report_question"), ` ${tr("going_to_report_audio")}
${tr("report_question_text")}

${tr("report_reason")}: `, [tr("confirm_m"), tr("cancel")], [(function() { res = document.querySelector("#uReportMsgInput").value; xhr = new XMLHttpRequest(); xhr.open("GET", "/report/" + e.target.dataset.id + "?reason=" + res + "&type=audio", true); xhr.onload = (function() { if(xhr.responseText.indexOf("reason") === -1) MessageBox(tr("error"), tr("error_sending_report"), ["OK"], [Function.noop]); else MessageBox(tr("action_successfully"), tr("will_be_watched"), ["OK"], [Function.noop]); }); xhr.send(null) }), Function.noop]) }) $(document).on("click", ".addToPlaylist", (e) => { let audios = document.querySelector("input[name='audios']") let id = e.currentTarget.dataset.id if(!audios.value.includes(id + ",")) { document.querySelector("input[name='audios']").value += (id + ",") e.currentTarget.querySelector("span").innerHTML = tr("remove_from_playlist") } else { document.querySelector("input[name='audios']").value = document.querySelector("input[name='audios']").value.replace(id + ",", "") e.currentTarget.querySelector("span").innerHTML = tr("add_to_playlist") } }) $(document).on("click", "#bookmarkPlaylist, #unbookmarkPlaylist", (e) => { let target = e.currentTarget let id = target.id $.ajax({ type: "POST", url: `/playlist${e.currentTarget.dataset.id}/action?act=${id == "unbookmarkPlaylist" ? "unbookmark" : "bookmark"}`, data: { hash: u("meta[name=csrf]").attr("value"), }, beforeSend: () => { e.currentTarget.classList.add("lagged") }, success: (response) => { if(response.success) { e.currentTarget.setAttribute("id", id == "unbookmarkPlaylist" ? "bookmarkPlaylist" : "unbookmarkPlaylist") e.currentTarget.innerHTML = id == "unbookmarkPlaylist" ? tr("bookmark") : tr("unbookmark") e.currentTarget.classList.remove("lagged") } else fastError(response.flash.message) } }) }) u(document).on('click', '.upload_container_element #small_remove_button', (e) => { if(u('.uploading').length > 0) { return } // 1984 const element = u(e.target).closest('.upload_container_element') const element_index = Number(element.attr('data-index')) element.remove() window.__audio_upload_page.files_list[element_index] = null if(u('#lastStep .upload_container_element').length < 1) { window.__audio_upload_page.showFirstPage() } }) u(document).on('click', `#upload_container #uploadMusic`, async (e) => { const current_upload_page = location.href let error = null let end_redir = '' u('#lastStepButtons').addClass('lagged') for(const elem of u('#lastStepContainers .upload_container_element').nodes) { if(!elem) { return } const elem_u = u(elem) const index = elem.dataset.index const file = window.__audio_upload_page.files_list[index] if(!file || !index) { return } elem_u.addClass('lagged').find('.upload_container_name').addClass('uploading') // Upload process const fd = serializeForm(elem) fd.append('blob', file.file) fd.append('ajax', 1) fd.append('hash', window.router.csrf) const result = await fetch(current_upload_page, { method: 'POST', body: fd, }) const result_text = await result.json() if(result_text.success) { end_redir = result_text.redirect_link } else { makeError(escapeHtml(result_text.flash.message)) } await sleep(6000) elem_u.remove() } if(!end_redir) { u('#lastStepButtons').removeClass('lagged') window.__audio_upload_page.showFirstPage() return } if(current_upload_page == location.href) { window.router.route(end_redir) } }) u(document).on("drop", "#upload_container", function (e) { e.preventDefault() e.dataTransfer.dropEffect = 'move'; document.getElementById("audio_input").files = e.dataTransfer.files u("#audio_input").trigger("change") }) u(document).on('click', '#_playlistAppendTracks', (e) => { showAudioAttachment('playlist', u('.PE_wrapper').nodes[0]) }) u(document).on('drop', '.PE_audios .vertical-attachment', (e) => { const current = u('.upload-item.currently_dragging') if(e.dataTransfer.types.length < 1 || e.dataTransfer.types.includes('text/uri-list')) { e.preventDefault() const target = u(e.target).closest('.upload-item') u('.dragged').removeClass('dragged') current.removeClass('currently_dragging') if(!current.closest('.vertical-attachment').length < 1 && target.closest('.vertical-attachment').length < 1 || current.closest('.vertical-attachment').length < 1 && !target.closest('.vertical-attachment').length < 1) { return } const first_html = target.nodes[0].outerHTML const second_html = current.nodes[0].outerHTML current.nodes[0].outerHTML = first_html target.nodes[0].outerHTML = second_html } }) u(document).on("change", `input[name='cover']`, (e) => { const file = e.target.files[0] if(!file.type.startsWith("image/")) { makeError(tr("not_a_photo")) return } const image = URL.createObjectURL(file) u(".playlistCover img").attr('src', image).attr('style', 'display:block') }) u(document).on("drop", `.playlistCover`, (e) => { e.preventDefault() e.dataTransfer.dropEffect = 'move'; document.querySelector(`input[name='cover']`).files = e.dataTransfer.files u(`input[name='cover']`).trigger("change") }) u(document).on('click', '.PE_end #playlist_create, .PE_end #playlist_edit', async (e) => { const ids = [] u('.PE_audios .vertical-attachment').nodes.forEach(vatch => { ids.push(vatch.dataset.id) }) if(!ids || ids.length < 1) { makeError(tr('error_playlist_creating_too_small'), 'Red', 5000, 77) return } u(e.target).addClass('lagged') const fd = serializeForm(u('.PE_playlistEditPage').nodes[0]) fd.append('hash', window.router.csrf) fd.append('ajax', 1) fd.append('audios', ids) const req = await fetch(location.href, { method: 'POST', body: fd, }) const req_json = await req.json() if(req_json.success) { window.router.route(req_json.redirect) } else { makeError(req_json.flash.message) } u(e.target).removeClass('lagged') })