Compare commits

..

No commits in common. "2cff603e11f54ee46742fb6dca939070ac375b49" and "1cf9ed0c8e0847f1ccfa9af324c7df4ab61d2869" have entirely different histories.

21 changed files with 442 additions and 743 deletions

View file

@ -228,7 +228,6 @@ final class AuthPresenter extends OpenVKPresenter
return; return;
} }
$this->template->disable_ajax = 1;
$this->template->is2faEnabled = $request->getUser()->is2faEnabled(); $this->template->is2faEnabled = $request->getUser()->is2faEnabled();
if($_SERVER["REQUEST_METHOD"] === "POST") { if($_SERVER["REQUEST_METHOD"] === "POST") {

View file

@ -38,7 +38,6 @@ final class NoSpamPresenter extends OpenVKPresenter
if ($mode === "form") { if ($mode === "form") {
$this->template->_template = "NoSpam/Index"; $this->template->_template = "NoSpam/Index";
$this->template->disable_ajax = 1;
$foundClasses = []; $foundClasses = [];
foreach (Finder::findFiles('*.php')->from($targetDir) as $file) { foreach (Finder::findFiles('*.php')->from($targetDir) as $file) {
$content = file_get_contents($file->getPathname()); $content = file_get_contents($file->getPathname());
@ -68,7 +67,6 @@ final class NoSpamPresenter extends OpenVKPresenter
$this->template->models = $models; $this->template->models = $models;
} else if ($mode === "templates") { } else if ($mode === "templates") {
$this->template->_template = "NoSpam/Templates.xml"; $this->template->_template = "NoSpam/Templates.xml";
$this->template->disable_ajax = 1;
$filter = []; $filter = [];
if ($this->queryParam("id")) { if ($this->queryParam("id")) {
$filter["id"] = (int)$this->queryParam("id"); $filter["id"] = (int)$this->queryParam("id");

View file

@ -283,8 +283,9 @@ abstract class OpenVKPresenter extends SimplePresenter
} }
} }
if($_SERVER['HTTP_X_OPENVK_AJAX_QUERY'] == '1' && $this->user->identity) { if($this->queryParam('al') == '1') {
error_reporting(0); error_reporting(0);
$this->assertNoCSRF();
header('Content-Type: text/plain; charset=UTF-8'); header('Content-Type: text/plain; charset=UTF-8');
} }

View file

@ -43,7 +43,6 @@ final class ReportPresenter extends OpenVKPresenter
"perPage" => 15, "perPage" => 15,
]; ];
$this->template->mode = $act ?? "all"; $this->template->mode = $act ?? "all";
$this->template->disable_ajax = 1;
if ($_SERVER["REQUEST_METHOD"] === "POST") { if ($_SERVER["REQUEST_METHOD"] === "POST") {
$reports = []; $reports = [];
@ -79,7 +78,6 @@ final class ReportPresenter extends OpenVKPresenter
$this->notFound(); $this->notFound();
$this->template->report = $report; $this->template->report = $report;
$this->template->disable_ajax = 1;
} }
function renderCreate(int $id): void function renderCreate(int $id): void

View file

@ -198,18 +198,18 @@
{var $canAccessHelpdesk = $thisUser->getChandlerUser()->can("write")->model('openvk\Web\Models\Entities\TicketReply')->whichBelongsTo(0)} {var $canAccessHelpdesk = $thisUser->getChandlerUser()->can("write")->model('openvk\Web\Models\Entities\TicketReply')->whichBelongsTo(0)}
{var $menuLinksAvaiable = sizeof(OPENVK_ROOT_CONF['openvk']['preferences']['menu']['links']) > 0 && $thisUser->getLeftMenuItemStatus('links')} {var $menuLinksAvaiable = sizeof(OPENVK_ROOT_CONF['openvk']['preferences']['menu']['links']) > 0 && $thisUser->getLeftMenuItemStatus('links')}
<div n:if="$canAccessAdminPanel || $canAccessHelpdesk || $menuLinksAvaiable" class="menu_divider"></div> <div n:if="$canAccessAdminPanel || $canAccessHelpdesk || $menuLinksAvaiable" class="menu_divider"></div>
<a href="/admin" class="link" n:if="$canAccessAdminPanel" title="{_admin} [Alt+Shift+A]" accesskey="a" rel="nofollow">{_admin}</a> <a href="/admin" class="link" n:if="$canAccessAdminPanel" title="{_admin} [Alt+Shift+A]" accesskey="a">{_admin}</a>
<a href="/support/tickets" class="link" n:if="$canAccessHelpdesk" rel="nofollow">{_helpdesk} <a href="/support/tickets" class="link" n:if="$canAccessHelpdesk">{_helpdesk}
{if $helpdeskTicketNotAnsweredCount > 0} {if $helpdeskTicketNotAnsweredCount > 0}
(<b>{$helpdeskTicketNotAnsweredCount}</b>) (<b>{$helpdeskTicketNotAnsweredCount}</b>)
{/if} {/if}
</a> </a>
<a n:if="$canAccessHelpdesk" href="/scumfeed" class="link" rel="nofollow">{tr("reports")} <a n:if="$canAccessHelpdesk" href="/scumfeed" class="link">{tr("reports")}
{if $reportNotAnsweredCount > 0} {if $reportNotAnsweredCount > 0}
(<b>{$reportNotAnsweredCount}</b>) (<b>{$reportNotAnsweredCount}</b>)
{/if} {/if}
</a> </a>
<a n:if="$canAccessHelpdesk" href="/noSpam" class="link" rel="nofollow"> <a n:if="$canAccessHelpdesk" href="/noSpam" class="link">
noSpam noSpam
</a> </a>
<a <a
@ -392,7 +392,6 @@
{script "js/al_navigation.js"} {script "js/al_navigation.js"}
{script "js/al_comments.js"} {script "js/al_comments.js"}
{script "js/al_music.js"} {script "js/al_music.js"}
{script "js/al_despacito_wall.js"}
{ifset $thisUser} {ifset $thisUser}
{script "js/al_notifs.js"} {script "js/al_notifs.js"}
@ -434,14 +433,13 @@
//]]> //]]>
</script> </script>
<script id='_js_ep_script'> <script>
window.openvk = { window.openvk = {
"audio_genres": {\openvk\Web\Models\Entities\Audio::genres}, "audio_genres": {\openvk\Web\Models\Entities\Audio::genres},
"at_search": {$atSearch ?? false}, "at_search": {$atSearch ?? false},
"max_attachments": {\OPENVK_ROOT_CONF["openvk"]["preferences"]["wall"]["postSizes"]["maxAttachments"] ?? 10}, "max_attachments": {\OPENVK_ROOT_CONF["openvk"]["preferences"]["wall"]["postSizes"]["maxAttachments"] ?? 10},
"max_filesize_mb": 5, "max_filesize_mb": 5,
"current_id": {$thisUser ? $thisUser->getId() : 0}, "current_id": {$thisUser ? $thisUser->getId() : 0},
"disable_ajax": {$disable_ajax ? $disable_ajax : 0},
} }
</script> </script>

View file

@ -70,4 +70,33 @@
<button class="button" type="submit">{_save}</button> <button class="button" type="submit">{_save}</button>
</div> </div>
</form> </form>
<script>
document.querySelector("input[name='audios']").value = {$audiosIds}
u("#editPlaylistForm").on("submit", (e) => {
document.querySelector("#editPlaylistForm input[name='title']").value = document.querySelector(".playlistInfo input[name='title']").value
document.querySelector("#editPlaylistForm textarea[name='description']").value = document.querySelector(".playlistBlock textarea[name='description']").value
document.querySelector("#editPlaylistForm input[name='is_unlisted']").value = document.querySelector("input[name='is_unlisted']").checked ? 1 : 0
})
u("#editPlaylistForm input[name='new_cover']").on("change", (e) => {
if(!e.currentTarget.files[0].type.startsWith("image/")) {
fastError(tr("not_a_photo"))
return
}
let image = URL.createObjectURL(e.currentTarget.files[0])
document.querySelector(".playlistCover img").src = image
})
u(".playlistCover img").on("click", (e) => {
document.querySelector("input[name='new_cover']").click()
})
document.querySelector("#editPlaylistForm input[name='new_cover']").value = ""
</script>
{script "js/al_playlists.js"}
{/block} {/block}

View file

@ -51,6 +51,24 @@
</div> </div>
</div> </div>
<div style="margin-top: 19px;">
<input id="playlist_query" type="text" style="height: 26px;" placeholder="{_header_search}">
<div class="playlistAudiosContainer editContainer">
<div id="newPlaylistAudios" n:foreach="$audios as $audio">
<div style="width: 78%;float: left;">
{include "player.xml", audio => $audio, hideButtons => true}
</div>
<div class="attachAudio addToPlaylist" data-id="{$audio->getId()}">
<span>{_add_to_playlist}</span>
</div>
</div>
</div>
<div class="showMoreAudiosPlaylist" data-page="2" {if !is_null($_GET["gid"])}data-club="{abs($_GET['gid'])}"{/if} n:if="$pagesCount > 1">
{_show_more_audios}
</div>
</div>
<form method="post" id="newPlaylistForm" enctype="multipart/form-data"> <form method="post" id="newPlaylistForm" enctype="multipart/form-data">
<input type="hidden" name="title" maxlength="125" /> <input type="hidden" name="title" maxlength="125" />
<input type="hidden" name="hash" value="{$csrfToken}" /> <input type="hidden" name="hash" value="{$csrfToken}" />
@ -63,4 +81,35 @@
<button class="button" type="submit">{_create}</button> <button class="button" type="submit">{_create}</button>
</div> </div>
</form> </form>
<script>
document.querySelector("input[name='audios']").value = ""
u("#newPlaylistForm").on("submit", (e) => {
document.querySelector("#newPlaylistForm input[name='title']").value = document.querySelector(".plinfo input[name='title']").value
document.querySelector("#newPlaylistForm textarea[name='description']").value = document.querySelector(".plinfo textarea[name='description']").value
document.querySelector("#newPlaylistForm input[name='is_unlisted']").value = document.querySelector(".plinfo input[name='is_unlisted']").checked ? 1 : 0
})
u("#newPlaylistForm input[name='cover']").on("change", (e) => {
if(!e.currentTarget.files[0].type.startsWith("image/")) {
fastError(tr("not_a_photo"))
return
}
let image = URL.createObjectURL(e.currentTarget.files[0])
document.querySelector(".playlistCover img").src = image
document.querySelector(".playlistCover img").style.display = "block"
})
u(".playlistCover img").on("click", (e) => {
document.querySelector("#newPlaylistForm input[name='cover']").value = ""
e.currentTarget.href = ""
})
document.querySelector("#newPlaylistForm input[name='cover']").value = ""
</script>
{script "js/al_playlists.js"}
{/block} {/block}

View file

@ -13,7 +13,7 @@
{block content} {block content}
<div class="gift_grid scroll_container"> <div class="gift_grid scroll_container">
<a href='/gifts?act=confirm&user={$user->getId()}&pack={$cat->getId()}&elid={$gift->getId()}' n:foreach="$gifts as $gift" n:class="scroll_node, gift_sel, !$gift->canUse($thisUser) ? disabled" data-gift="{$gift->getId()}"> <div n:foreach="$gifts as $gift" n:class="scroll_node, gift_sel, !$gift->canUse($thisUser) ? disabled" data-gift="{$gift->getId()}">
<img class="gift_pic" src="{$gift->getImage(2)}" alt="{_gift}" loading=lazy /> <img class="gift_pic" src="{$gift->getImage(2)}" alt="{_gift}" loading=lazy />
<strong class="gift_price"> <strong class="gift_price">
@ -29,7 +29,7 @@
{tr("gifts_left", $gift->getUsagesLeft($thisUser))} {tr("gifts_left", $gift->getUsagesLeft($thisUser))}
{/if}&nbsp; {/if}&nbsp;
</strong> </strong>
</a> </div>
</div> </div>
<div style="padding: 8px;"> <div style="padding: 8px;">
@ -41,3 +41,18 @@
]} ]}
</div> </div>
{/block} {/block}
{block bodyScripts}
<script>
$(".gift_sel").click(function() {
let el = $(this);
if(el.hasClass("disabled"))
return false;
let link = "/gifts?act=confirm&user={$user->getId()}&pack={$cat->getId()}&elid=";
let gift = el.data("gift");
window.router.route(link + gift);
});
</script>
{/block}

View file

@ -82,25 +82,23 @@
</table> </table>
<script n:if="$club->getOwner()->getId() != $user->getId() && $manager && $thisUser->getId() == $club->getOwner()->getId()"> <script n:if="$club->getOwner()->getId() != $user->getId() && $manager && $thisUser->getId() == $club->getOwner()->getId()">
if(typeof changeOwner == 'undefined') { function changeOwner(club, newOwner) {
function changeOwner(club, newOwner) { const action = "/groups/" + club + "/setNewOwner/" + newOwner;
const action = "/groups/" + club + "/setNewOwner/" + newOwner;
MessageBox({_group_changeowner_modal_title}, ` MessageBox({_group_changeowner_modal_title}, `
{tr("group_changeowner_modal_text", htmlentities($user->getFullName()))|noescape} {tr("group_changeowner_modal_text", htmlentities($user->getFullName()))|noescape}
<br/><br/> <br/><br/>
<form id="transfer-owner-permissions-form" method="post"> <form id="transfer-owner-permissions-form" method="post">
<label for="password">{_password|noescape}</label> <label for="password">{_password|noescape}</label>
<input type="password" id="password" name="password" required /> <input type="password" id="password" name="password" required />
<input type="hidden" name="hash" value={$csrfToken} /> <input type="hidden" name="hash" value={$csrfToken} />
</form> </form>
`, [{_transfer}, {_cancel}], [ `, [{_transfer}, {_cancel}], [
() => { () => {
$("#transfer-owner-permissions-form").attr("action", action); $("#transfer-owner-permissions-form").attr("action", action);
document.querySelector("#transfer-owner-permissions-form").submit(); document.querySelector("#transfer-owner-permissions-form").submit();
}, Function.noop }, Function.noop
]); ]);
}
} }
</script> </script>
{/block} {/block}

View file

@ -318,3 +318,7 @@
</div> </div>
{/block} {/block}
{block bodyScripts}
{script "js/al_despacito_wall.js"}
{/block}

View file

@ -165,10 +165,33 @@
</form> </form>
{/if} {/if}
<a class="profile_link" style="display:block;width:96%;" href="javascript:reportUser({$user->getId()})">{_report}</a> <a class="profile_link" style="display:block;width:96%;" href="javascript:reportUser()">{_report}</a>
<a n:if="!$user->isHideFromGlobalFeedEnabled()" class="profile_link" style="display:block;width:96%;" id="__ignoreSomeone" data-val='{!$ignore_status ? 1 : 0}' data-id="{$user->getId()}"> <a n:if="!$user->isHideFromGlobalFeedEnabled()" class="profile_link" style="display:block;width:96%;" id="__ignoreSomeone" data-val='{!$ignore_status ? 1 : 0}' data-id="{$user->getId()}">
{if !$ignore_status}{_ignore_user}{else}{_unignore_user}{/if} {if !$ignore_status}{_ignore_user}{else}{_unignore_user}{/if}
</a> </a>
<script>
function reportUser() {
uReportMsgTxt = tr("going_to_report_user");
uReportMsgTxt += "<br/>"+tr("report_question_text");
uReportMsgTxt += "<br/><br/><b>"+tr("report_reason")+"</b>: <input type='text' id='uReportMsgInput' placeholder='" + tr("reason") + "' />"
MessageBox(tr("report_question"), uReportMsgTxt, [tr("confirm_m"), tr("cancel")], [
(function() {
res = document.querySelector("#uReportMsgInput").value;
xhr = new XMLHttpRequest();
xhr.open("GET", "/report/" + {$user->getId()} + "?reason=" + res + "&type=user", 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
]);
}
</script>
{/if} {/if}
<a style="width: 194px;" n:if="$user->getFollowersCount() > 0" href="/friends{$user->getId()}?act=incoming" class="profile_link">{tr("followers", $user->getFollowersCount())}</a> <a style="width: 194px;" n:if="$user->getFollowersCount() > 0" href="/friends{$user->getId()}?act=incoming" class="profile_link">{tr("followers", $user->getFollowersCount())}</a>
</div> </div>
@ -737,6 +760,51 @@
} }
{/if} {/if}
</script> </script>
<script n:if="isset($thisUser) && $user->getId() == $thisUser->getId()" n:syntax="off">
function setStatusEditorShown(shown) {
document.getElementById("status_editor").style.display = shown ? "block" : "none";
}
document.addEventListener("click", event => {
if(!event.target.closest("#status_editor") && !event.target.closest("#page_status_text"))
setStatusEditorShown(false);
});
document.getElementById("page_status_text").onclick = setStatusEditorShown.bind(this, true);
async function changeStatus() {
const status = document.status_popup_form.status.value;
const broadcast = document.status_popup_form.broadcast.checked;
document.status_popup_form.submit.innerHTML = "<div class=\"button-loading\"></div>";
document.status_popup_form.submit.disabled = true;
const formData = new FormData();
formData.append("status", status);
formData.append("broadcast", Number(broadcast));
formData.append("hash", document.status_popup_form.hash.value);
const response = await ky.post("/edit?act=status", {body: formData});
if(!parseAjaxResponse(await response.text())) {
document.status_popup_form.submit.innerHTML = tr("send");
document.status_popup_form.submit.disabled = false;
return;
}
if(document.status_popup_form.status.value === "") {
document.querySelector("#page_status_text").innerHTML = `[ ${tr("change_status")} ]`;
document.querySelector("#page_status_text").className = "edit_link page_status_edit_button";
} else {
document.querySelector("#page_status_text").innerHTML = status;
document.querySelector("#page_status_text").className = "page_status page_status_edit_button";
}
setStatusEditorShown(false);
document.status_popup_form.submit.innerHTML = tr("send");
document.status_popup_form.submit.disabled = false;
}
</script>
</div> </div>
{/if} {/if}
@ -745,3 +813,7 @@
{include "banned.xml"} {include "banned.xml"}
{/if} {/if}
{/block} {/block}
{block bodyScripts}
{script "js/al_despacito_wall.js"}
{/block}

View file

@ -54,11 +54,11 @@
<input type="checkbox" name="as_group" /> {_comment_as_group} <input type="checkbox" name="as_group" /> {_comment_as_group}
</label> </label>
</div> </div>
<input type="hidden" name="horizontal_attachments" value="" autocomplete="off" /> <input type="hidden" name="horizontal_attachments" value="" />
<input type="hidden" name="vertical_attachments" value="" autocomplete="off" /> <input type="hidden" name="vertical_attachments" value="" />
<input type="hidden" name="poll" value="none" autocomplete="off" /> <input type="hidden" name="poll" value="none" />
<input type="hidden" id="source" name="source" value="none" autocomplete="off" /> <input type="hidden" id="source" name="source" value="none" />
<input type="hidden" name="type" value="1" autocomplete="off" /> <input type="hidden" name="type" value="1" />
<input type="hidden" name="hash" value="{$csrfToken}" /> <input type="hidden" name="hash" value="{$csrfToken}" />
<br/> <br/>
<input type="submit" value="{_write}" class="button" /> <input type="submit" value="{_write}" class="button" />
@ -105,6 +105,11 @@
</form> </form>
</div> </div>
<script>
u("#post-buttons{$textAreaId} input[name='horizontal_attachments']")["nodes"].at(0).value = ""
u("#post-buttons{$textAreaId} input[name='vertical_attachments']")["nodes"].at(0).value = ""
</script>
{if $graffiti} {if $graffiti}
{script "js/node_modules/react/dist/react-with-addons.min.js"} {script "js/node_modules/react/dist/react-with-addons.min.js"}
{script "js/node_modules/react-dom/dist/react-dom.min.js"} {script "js/node_modules/react-dom/dist/react-dom.min.js"}

View file

@ -13,7 +13,7 @@
} }
.musicIcon { .musicIcon {
background-image: url('/assets/packages/static/openvk/img/audios_controls.png?v=6'); background-image: url('/assets/packages/static/openvk/img/audios_controls.png?v=2');
background-repeat: no-repeat; background-repeat: no-repeat;
cursor: pointer; cursor: pointer;
} }
@ -150,7 +150,7 @@
} }
.bigPlayer .slider, .audioEmbed .track .slider { .bigPlayer .slider, .audioEmbed .track .slider {
width: 15px; width: 18px;
height: 7px; height: 7px;
background: #606060; background: #606060;
position: absolute; position: absolute;
@ -231,13 +231,13 @@
} }
.bigPlayer .trackPanel .track .selectableTrack > div { .bigPlayer .trackPanel .track .selectableTrack > div {
width: 96%; width: 95%;
position: relative; position: relative;
} }
.bigPlayer .volumePanel .selectableTrack > div { .bigPlayer .volumePanel .selectableTrack > div {
position: relative; position: relative;
width: 77%; width: 72%
} }
.audioEmbed .track > .selectableTrack, .bigPlayer .selectableTrack { .audioEmbed .track > .selectableTrack, .bigPlayer .selectableTrack {
@ -881,137 +881,3 @@
margin-top: 6px; margin-top: 6px;
padding: 1px; padding: 1px;
} }
/* AJAX player */
#ajax_audio_player {
transition: background .1s ease-out;
background: rgba(44, 44, 44, 0.7);
padding: 1px;
width: 500px;
height: 37px;
position: fixed;
z-index: 99;
border-radius: 3px;
}
#ajax_audio_player.hidden {
display: none;
}
#ajax_audio_player #aj_player {
position: relative;
height: 100%;
}
#ajax_audio_player #aj_player #aj_player_internal_controls {
padding: 7px 8px;
display: flex;
gap: 7px;
}
#ajax_audio_player.ui-draggable-dragging {
background: rgba(20, 20, 20, 0.9);
}
#ajax_audio_player #aj_player_close_btn, #ajax_audio_player #aj_player_play {
padding: 3px 0px;
}
#ajax_audio_player #aj_player_close_btn,
#ajax_audio_player #aj_player_play #aj_player_play_btn,
#ajax_audio_player #aj_player_buttons > div {
background: url('/assets/packages/static/openvk/img/audios_controls.png?v=6');
background-repeat: no-repeat;
cursor: pointer;
}
#ajax_audio_player #aj_player_close_btn {
position: absolute;
top: 0;
right: 0;
width: 10px;
height: 9px;
background-position: 0px -77px;
opacity: 0.6;
}
#ajax_audio_player #aj_player_close_btn:hover {
opacity: 1;
}
#ajax_audio_player #aj_player_play #aj_player_play_btn {
width: 16px;
height: 16px;
background-position: -147px -28px;
}
#ajax_audio_player #aj_player_play #aj_player_play_btn.paused {
background-position: -165px -28px;
}
#ajax_audio_player #aj_player_track {
width: 100%;
position: relative;
}
#ajax_audio_player #aj_player_track #aj_player_track_name {
display: flex;
justify-content: space-between;
}
#ajax_audio_player #aj_player_track #aj_player_track_name #aj_player_track_title,
#ajax_audio_player #aj_player_track #aj_player_track_name #aj_player_track_title b,
#ajax_audio_player #aj_player_track #aj_player_track_name #aj_player_track_title span,
#ajax_audio_player #aj_player_track #aj_player_track_name span {
color: white;
}
#ajax_audio_player .selectableTrack {
width: 100%;
position: relative;
height: 6px;
border-top: #ffffff 1px solid;
user-select: none;
}
#ajax_audio_player .selectableTrack .slider {
width: 11px;
height: 6px;
background: #ffffff;
position: absolute;
}
#ajax_audio_player #aj_player_volume {
width: 60px;
padding-top: 15px;
position: relative;
}
#ajax_audio_player #aj_player_buttons {
display: flex;
flex-direction: row;
align-items: center;
gap: 10px;
}
#ajax_audio_player #aj_player_buttons #aj_player_previous {
width: 14px;
height: 12px;
background-position: -194px -5px;
}
#ajax_audio_player #aj_player_buttons #aj_player_repeat {
width: 14px;
height: 13px;
background-position: -233px -5px;
}
#ajax_audio_player #aj_player_buttons #aj_player_repeat.pressed {
opacity: 0.6;
}
#ajax_audio_player #aj_player_buttons #aj_player_next {
width: 15px;
height: 12px;
background-position: -214px -5px;
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.5 KiB

After

Width:  |  Height:  |  Size: 4.2 KiB

View file

@ -1,3 +1,4 @@
const contentPage = document.querySelector(".page_content");
const rootElement = document.documentElement; const rootElement = document.documentElement;
// охуенное название файла, КТО ЭТО ПРИДУМАЛ КРАСАВА Я ИЗ КОМНАТЫ С ЭТОГО УЛЕТЕЛ НАХУЙ // охуенное название файла, КТО ЭТО ПРИДУМАЛ КРАСАВА Я ИЗ КОМНАТЫ С ЭТОГО УЛЕТЕЛ НАХУЙ
@ -7,11 +8,11 @@ let scrolledAndHidden = false;
let smallBlockObserver = new IntersectionObserver(entries => { let smallBlockObserver = new IntersectionObserver(entries => {
entries.forEach(x => { entries.forEach(x => {
window.requestAnimationFrame(() => { window.requestAnimationFrame(() => {
let pastHeight = u('.page_content').nodes[0].getBoundingClientRect().height; let pastHeight = contentPage.getBoundingClientRect().height;
if(x.isIntersecting) if(x.isIntersecting)
u('.page_content').nodes[0].classList.remove("overscrolled"); contentPage.classList.remove("overscrolled");
else else
u('.page_content').nodes[0].classList.add("overscrolled"); contentPage.classList.add("overscrolled");
// let currentHeight = contentPage.getBoundingClientRect().height; // let currentHeight = contentPage.getBoundingClientRect().height;
// let ratio = currentHeight / pastHeight; // let ratio = currentHeight / pastHeight;
@ -21,7 +22,7 @@ let smallBlockObserver = new IntersectionObserver(entries => {
// То что я задокументировал - работает мегакриво. // То что я задокументировал - работает мегакриво.
// Пусть юзер и проскролливает какую-то часть контента, зато не получит // Пусть юзер и проскролливает какую-то часть контента, зато не получит
// эпилепсии при использовании :) // эпилепсии при использовании :)
}, u('.page_content').nodes[0]); }, contentPage);
}); });
}, { }, {
root: null, // screen root: null, // screen

View file

@ -1,3 +1,5 @@
window.savedAudiosPages = {}
class playersSearcher { class playersSearcher {
constructor(context_type, context_id) { constructor(context_type, context_id) {
this.context_type = context_type this.context_type = context_type
@ -66,18 +68,6 @@ window.player = new class {
return this.__realAudioPlayer 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() { get uiPlayer() {
return u('.bigPlayer') return u('.bigPlayer')
} }
@ -96,9 +86,17 @@ window.player = new class {
return this.__findByIndex(current + 1) return this.__findByIndex(current + 1)
} }
get linkedInlinePlayer() {
if(!this.__linked_player_id) {
return null;
}
return u('#' + this.__linked_player_id)
}
async init(input_context) { async init(input_context) {
let context = Object.assign({ let context = Object.assign({
url: location.pathname + location.search url: location.pathname
}, input_context) }, input_context)
this.context.object = !input_context ? null : context this.context.object = !input_context ? null : context
this.__realAudioPlayer = document.createElement("audio") this.__realAudioPlayer = document.createElement("audio")
@ -111,13 +109,8 @@ window.player = new class {
initEvents() { initEvents() {
this.audioPlayer.ontimeupdate = () => { this.audioPlayer.ontimeupdate = () => {
const current_track = this.currentTrack const time = this.audioPlayer.currentTime;
if(!current_track) { const ps = ((time * 100) / this.currentTrack.length).toFixed(3)
return
}
const time = this.audioPlayer.currentTime
const ps = ((time * 100) / current_track.length).toFixed(3)
this.uiPlayer.find(".time").html(fmtTime(time)) this.uiPlayer.find(".time").html(fmtTime(time))
this.__updateTime(time) this.__updateTime(time)
@ -128,11 +121,6 @@ window.player = new class {
this.linkedInlinePlayer.find(".subTracks .lengthTrackWrapper .slider").attr('style', `left:${ ps}%`) this.linkedInlinePlayer.find(".subTracks .lengthTrackWrapper .slider").attr('style', `left:${ ps}%`)
this.linkedInlinePlayer.find('.mini_timer .nobold').html(fmtTime(time)) 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))
}
} }
} }
@ -146,10 +134,6 @@ window.player = new class {
if(this.linkedInlinePlayer) { if(this.linkedInlinePlayer) {
this.linkedInlinePlayer.find(".subTracks .volumeTrackWrapper .slider").attr('style', `left:${ ps}%`) 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) localStorage.setItem('audio.volume', volume)
@ -178,12 +162,12 @@ window.player = new class {
} }
} }
this.audioPlayer.onended = async (e) => { this.audioPlayer.onended = (e) => {
e.preventDefault() e.preventDefault()
if(!this.nextTrack && window.player.context.playedPages.indexOf(1) == -1) { if(!this.nextTrack && window.player.context.playedPages.indexOf(1) == -1) {
this.loadContext(1, false) this.loadContext(1, false)
await this.setTrack(this.__findByIndex(0).id) this.setTrack(this.__findByIndex(0).id)
} else { } else {
this.playNextTrack() this.playNextTrack()
} }
@ -259,28 +243,10 @@ window.player = new class {
return 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 this.listen_coef = 0.0
const old_id = this.current_track_id
this.current_track_id = id this.current_track_id = id
const c_track = this.currentTrack const c_track = this.currentTrack
if(!c_track) { if(!c_track) {
this.current_track_id = old_id
makeError('Error playing audio: track not found') makeError('Error playing audio: track not found')
return return
} }
@ -292,7 +258,9 @@ window.player = new class {
}; };
u('.nowPlaying').removeClass('nowPlaying') u('.nowPlaying').removeClass('nowPlaying')
this.__highlightActiveTrack() if(this.isAtAudiosPage()) {
u(`.audioEmbed[data-realid='${id}'] .audioEntry`).addClass('nowPlaying')
}
navigator.mediaSession.setPositionState({ navigator.mediaSession.setPositionState({
duration: this.currentTrack.length duration: this.currentTrack.length
@ -309,7 +277,6 @@ window.player = new class {
await this.loadContext(Math.min(...this.context["playedPages"]) - 1, false) await this.loadContext(Math.min(...this.context["playedPages"]) - 1, false)
} }
this.is_closed = false
this.__updateFace() this.__updateFace()
u(this.audioPlayer).trigger('volumechange') u(this.audioPlayer).trigger('volumechange')
} }
@ -357,12 +324,12 @@ window.player = new class {
navigator.mediaSession.playbackState = "paused" navigator.mediaSession.playbackState = "paused"
} }
async playPreviousTrack() { playPreviousTrack() {
if(!this.currentTrack || !this.previousTrack) { if(!this.currentTrack || !this.previousTrack) {
return return
} }
await this.setTrack(this.previousTrack.id) this.setTrack(this.previousTrack.id)
if(!this.currentTrack.available || this.currentTrack.withdrawn) { if(!this.currentTrack.available || this.currentTrack.withdrawn) {
if(!this.previousTrack) { if(!this.previousTrack) {
return return
@ -374,12 +341,12 @@ window.player = new class {
this.play() this.play()
} }
async playNextTrack() { playNextTrack() {
if(!this.currentTrack || !this.nextTrack) { if(!this.currentTrack || !this.nextTrack) {
return return
} }
await this.setTrack(this.nextTrack.id) this.setTrack(this.nextTrack.id)
if(!this.currentTrack.available || this.currentTrack.withdrawn) { if(!this.currentTrack.available || this.currentTrack.withdrawn) {
if(!this.nextTrack) { if(!this.nextTrack) {
return return
@ -392,78 +359,15 @@ window.player = new class {
} }
// fake shuffle // fake shuffle
async shuffle() { shuffle() {
this.tracks.sort(() => Math.random() - 0.59) this.tracks.sort(() => Math.random() - 0.59)
await this.setTrack(this.tracks.at(0).id) this.setTrack(this.tracks.at(0).id)
} }
isAtAudiosPage() { isAtAudiosPage() {
return u('.bigPlayer').length > 0 return u('.bigPlayer').length > 0
} }
// Добавляем ощущение продуманности.
__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() {
console.log('Audio | Switched page :3')
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()
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') { __setFavicon(state = 'playing') {
if(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") document.querySelector('link[rel="icon"], link[rel="shortcut icon"]').setAttribute("href", "/assets/packages/static/openvk/img/favicons/favicon24_paused.png")
@ -514,28 +418,16 @@ window.player = new class {
if(!this.previousTrack) { if(!this.previousTrack) {
prev_button.addClass('lagged') prev_button.addClass('lagged')
if(this.ajaxPlayer.length > 0) {
this.ajaxPlayer.find('#aj_player_previous').addClass('lagged')
}
} else { } else {
prev_button.removeClass('lagged') prev_button.removeClass('lagged')
prev_button.attr('data-title', ovk_proc_strtr(escapeHtml(this.previousTrack.name), 50)) 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) { if(!this.nextTrack) {
next_button.addClass('lagged') next_button.addClass('lagged')
if(this.ajaxPlayer.length > 0) {
this.ajaxPlayer.find('#aj_player_next').addClass('lagged')
}
} else { } else {
next_button.removeClass('lagged') next_button.removeClass('lagged')
next_button.attr('data-title', ovk_proc_strtr(escapeHtml(this.nextTrack.name), 50)) 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) { if(!this.audioPlayer.paused) {
@ -543,17 +435,11 @@ window.player = new class {
if(this.linkedInlinePlayer) { if(this.linkedInlinePlayer) {
this.linkedInlinePlayer.find('.playerButton .playIcon').addClass('paused') this.linkedInlinePlayer.find('.playerButton .playIcon').addClass('paused')
} }
if(this.ajaxPlayer.length > 0) {
this.ajaxPlayer.find('#aj_player_play_btn').addClass('paused')
}
} else { } else {
this.uiPlayer.find('.playButton').removeClass('pause') this.uiPlayer.find('.playButton').removeClass('pause')
if(this.linkedInlinePlayer) { if(this.linkedInlinePlayer) {
this.linkedInlinePlayer.find('.playerButton .playIcon').removeClass('paused') this.linkedInlinePlayer.find('.playerButton .playIcon').removeClass('paused')
} }
if(this.ajaxPlayer.length > 0) {
this.ajaxPlayer.find('#aj_player_play_btn').removeClass('paused')
}
} }
this.uiPlayer.find('.trackInfo .trackName span').html(escapeHtml(_c.name)) this.uiPlayer.find('.trackInfo .trackName span').html(escapeHtml(_c.name))
@ -564,12 +450,6 @@ window.player = new class {
this.uiPlayer.find('.trackInfo .trackPerformers').append( this.uiPlayer.find('.trackInfo .trackPerformers').append(
`<a href='/search?section=audios&order=listens&only_performers=on&q=${encodeURIComponent(performer.escapeHtml())}'>${performer.escapeHtml()}${(performer != lastPerformer ? ', ' : '')}</a>`) `<a href='/search?section=audios&order=listens&only_performers=on&q=${encodeURIComponent(performer.escapeHtml())}'>${performer.escapeHtml()}${(performer != lastPerformer ? ', ' : '')}</a>`)
}) })
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() u(`.tip_result`).remove()
} }
@ -621,95 +501,13 @@ window.player = new class {
console.log('Listen is not counted ! ! !') console.log('Listen is not counted ! ! !')
} }
} }
ajClose() {
this.is_closed = true
this.pause()
u('#ajax_audio_player').addClass('hidden')
}
ajReveal() {
this.is_closed = false
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(`
<div id='ajax_audio_player'>
<div id='aj_player'>
<div id='aj_player_internal_controls'>
<div id='aj_player_play'>
<div id='aj_player_play_btn'></div>
</div>
<div id='aj_player_track'>
<div id='aj_player_track_name'>
<a id='aj_player_track_title' class='noOverflow' style='width: 300px;'>
<b>Unknown</b>
<span>Untitled</span>
</a>
<span id='aj_time'>00:00</span>
</div>
<div id='aj_player_track_length'>
<div class="selectableTrack">
<div style='width: 97%;position: relative;'>
<div class="slider"></div>
</div>
</div>
</div>
</div>
<div id='aj_player_volume'>
<div class="selectableTrack">
<div style='width: 75%;position: relative;'>
<div class="slider"></div>
</div>
</div>
</div>
<div id='aj_player_buttons'>
<div id='aj_player_previous'></div>
<div id='aj_player_repeat'></div>
<div id='aj_player_next'></div>
</div>
</div>
<div id='aj_player_close_btn'></div>
</div>
</div>
`)
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_player_track_title').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',
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 () => { document.addEventListener("DOMContentLoaded", async () => {
await window.player.init(window.__current_page_audio_context) await window.player.init(window.__current_page_audio_context)
}) })
u(document).on('click', '.audioEntry .playerButton > .playIcon', async (e) => { u(document).on('click', '.audioEntry .playerButton > .playIcon', (e) => {
const audioPlayer = u(e.target).closest('.audioEmbed') const audioPlayer = u(e.target).closest('.audioEmbed')
const id = Number(audioPlayer.attr('data-realid')) const id = Number(audioPlayer.attr('data-realid'))
if(!window.player) { if(!window.player) {
@ -742,7 +540,7 @@ u(document).on('click', '.audioEntry .playerButton > .playIcon', async (e) => {
} }
if(window.player.current_track_id != id) { if(window.player.current_track_id != id) {
await window.player.setTrack(id) window.player.setTrack(id)
} }
if(window.player.audioPlayer.paused) { if(window.player.audioPlayer.paused) {
@ -765,7 +563,7 @@ u(document).on('click', '.audioEntry .playerButton > .playIcon', async (e) => {
} }
}) })
u(document).on('click', '.bigPlayer .playButton, #ajax_audio_player #aj_player_play_btn', (e) => { u(document).on('click', '.bigPlayer .playButton', (e) => {
if(window.player.audioPlayer.paused) { if(window.player.audioPlayer.paused) {
window.player.play() window.player.play()
} else { } else {
@ -773,12 +571,12 @@ u(document).on('click', '.bigPlayer .playButton, #ajax_audio_player #aj_player_p
} }
}) })
u(document).on('click', '.bigPlayer .backButton, #ajax_audio_player #aj_player_next', async (e) => { u(document).on('click', '.bigPlayer .backButton', (e) => {
await window.player.playNextTrack() window.player.playNextTrack()
}) })
u(document).on('click', '.bigPlayer .nextButton, #ajax_audio_player #aj_player_previous', async (e) => { u(document).on('click', '.bigPlayer .nextButton', (e) => {
await window.player.playPreviousTrack() window.player.playPreviousTrack()
}) })
u(document).on("click", ".bigPlayer .elapsedTime", (e) => { u(document).on("click", ".bigPlayer .elapsedTime", (e) => {
@ -790,7 +588,7 @@ u(document).on("click", ".bigPlayer .elapsedTime", (e) => {
window.player.__updateTime(window.player.audioPlayer.currentTime) window.player.__updateTime(window.player.audioPlayer.currentTime)
}) })
u(document).on("click", ".bigPlayer .additionalButtons .repeatButton, #ajax_audio_player #aj_player_repeat", (e) => { u(document).on("click", ".bigPlayer .additionalButtons .repeatButton", (e) => {
if(window.player.current_track_id == 0) if(window.player.current_track_id == 0)
return return
@ -803,11 +601,11 @@ u(document).on("click", ".bigPlayer .additionalButtons .repeatButton, #ajax_audi
window.player.audioPlayer.loop = false window.player.audioPlayer.loop = false
}) })
u(document).on("click", ".bigPlayer .additionalButtons .shuffleButton", async (e) => { u(document).on("click", ".bigPlayer .additionalButtons .shuffleButton", (e) => {
if(window.player.current_track_id == 0) if(window.player.current_track_id == 0)
return return
await window.player.shuffle() window.player.shuffle()
}) })
u(document).on("click", ".bigPlayer .additionalButtons .deviceButton", (e) => { u(document).on("click", ".bigPlayer .additionalButtons .deviceButton", (e) => {
@ -823,7 +621,7 @@ u(document).on('keydown', (e) => {
return return
} }
if(!window.player || !window.player.isAtAudiosPage()) { if(!window.player) {
return return
} }
@ -895,7 +693,7 @@ u(document).on('keyup', (e) => {
} }
}) })
u(document).on("mousemove click mouseup", ".bigPlayer .trackPanel .selectableTrack, .audioEntry .subTracks .lengthTrackWrapper .selectableTrack, #aj_player_track_length .selectableTrack", (e) => { u(document).on("mousemove click mouseup", ".bigPlayer .trackPanel .selectableTrack, .audioEntry .subTracks .lengthTrackWrapper .selectableTrack", (e) => {
if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) if(window.player.isAtAudiosPage() && window.player.current_track_id == 0)
return return
@ -926,14 +724,14 @@ u(document).on("mousemove click mouseup", ".bigPlayer .trackPanel .selectableTra
parent.find('.tip_result').html(fmtTime(time)).attr('style', `left:min(${width - 15}px, 315.5px)`) 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) => { u(document).on("mouseout", ".bigPlayer .trackPanel .selectableTrack, .audioEntry .subTracks .lengthTrackWrapper .selectableTrack", (e) => {
if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) if(window.player.isAtAudiosPage() && window.player.current_track_id == 0)
return return
u(e.target).closest('.selectableTrack').parent().find('.tip_result').remove() 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) => { u(document).on("mousemove click mouseup", ".bigPlayer .volumePanelTrack .selectableTrack, .audioEntry .subTracks .volumeTrack .selectableTrack", (e) => {
if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) if(window.player.isAtAudiosPage() && window.player.current_track_id == 0)
return return
@ -963,7 +761,7 @@ u(document).on("mousemove click mouseup", ".bigPlayer .volumePanelTrack .selecta
parent.find('.tip_result').html((volume * 100).toFixed(0) + '%').attr('style', `left:${width - 15}px`) 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) => { u(document).on("mouseout", ".bigPlayer .volumePanelTrack .selectableTrack, .audioEntry .subTracks .volumeTrack .selectableTrack", (e) => {
if(window.player.isAtAudiosPage() && window.player.current_track_id == 0) if(window.player.isAtAudiosPage() && window.player.current_track_id == 0)
return return
@ -1126,6 +924,8 @@ u(document).on("click", ".musicIcon.edit-icon", (e) => {
if(url.searchParams.p != null) if(url.searchParams.p != null)
page = String(url.searchParams.p) page = String(url.searchParams.p)
window.savedAudiosPages[page] = null
} else } else
fastError(response.flash.message) fastError(response.flash.message)
} }
@ -1683,6 +1483,47 @@ $(document).on("click", ".musicIcon.report-icon", (e) => {
Function.noop]) Function.noop])
}) })
u(document).on("click", ".audiosContainer .paginator a", (e) => {
e.preventDefault()
let url = new URL(e.currentTarget.href)
let page = url.searchParams.get("p")
function searchNode(id) {
let node = document.querySelector(`.audioEmbed[data-realid='${id}'] .audioEntry`)
if(node != null) {
node.classList.add("nowPlaying")
}
}
if(window.savedAudiosPages[page] != null) {
history.pushState({}, "", e.currentTarget.href)
document.querySelector(".audiosContainer").innerHTML = window.savedAudiosPages[page].innerHTML
searchNode(window.player.currentTrack != null ? window.player.currentTrack.id : 0)
return
}
e.currentTarget.parentNode.classList.add("lagged")
$.ajax({
type: "GET",
url: e.currentTarget.href,
success: (response) => {
let domparser = new DOMParser()
let result = domparser.parseFromString(response, "text/html")
document.querySelector(".audiosContainer").innerHTML = result.querySelector(".audiosContainer").innerHTML
history.pushState({}, "", e.currentTarget.href)
window.savedAudiosPages[page] = result.querySelector(".audiosContainer")
searchNode(window.player.currentTrack != null ? window.player.currentTrack.id : 0)
if(!window.player.context["playedPages"].includes(page)) {
window.player.loadContext(page)
}
}
})
})
$(document).on("click", ".addToPlaylist", (e) => { $(document).on("click", ".addToPlaylist", (e) => {
let audios = document.querySelector("input[name='audios']") let audios = document.querySelector("input[name='audios']")
let id = e.currentTarget.dataset.id let id = e.currentTarget.dataset.id

View file

@ -161,7 +161,6 @@ function loadMoreSuggestedPosts() {
// нажатие на "x предложенных записей" // нажатие на "x предложенных записей"
$(document).on("click", ".sugglist a", (e) => { $(document).on("click", ".sugglist a", (e) => {
e.preventDefault() e.preventDefault()
e.stopPropagation()
if(e.currentTarget.getAttribute("data-toogled") == null || e.currentTarget.getAttribute("data-toogled") == "false") { if(e.currentTarget.getAttribute("data-toogled") == null || e.currentTarget.getAttribute("data-toogled") == "false") {
e.currentTarget.setAttribute("data-toogled", "true") e.currentTarget.setAttribute("data-toogled", "true")

View file

@ -91,7 +91,6 @@ async function OpenMiniature(e, photo, post, photo_id, type = "post") {
костыли но смешные однако костыли но смешные однако
*/ */
e.preventDefault(); e.preventDefault();
e.stopPropagation()
// Значения для переключения фоток // Значения для переключения фоток
@ -450,7 +449,6 @@ async function OpenVideo(video_arr = [], init_player = true)
u(document).on('click', '#videoOpen', (e) => { u(document).on('click', '#videoOpen', (e) => {
e.preventDefault() e.preventDefault()
e.stopPropagation()
try { try {
const target = e.target.closest('#videoOpen') const target = e.target.closest('#videoOpen')
@ -473,7 +471,6 @@ u(document).on('keydown', '.edit_menu #write', (e) => {
e.target.closest('.edit_menu').querySelector('#__edit_save').click() e.target.closest('.edit_menu').querySelector('#__edit_save').click()
}) })
// Migrated from inline start
function reportPhoto(photo_id) { function reportPhoto(photo_id) {
uReportMsgTxt = tr("going_to_report_photo"); uReportMsgTxt = tr("going_to_report_photo");
uReportMsgTxt += "<br/>"+tr("report_question_text"); uReportMsgTxt += "<br/>"+tr("report_question_text");
@ -518,154 +515,6 @@ function reportVideo(video_id) {
]); ]);
} }
function reportUser(user_id) {
uReportMsgTxt = tr("going_to_report_user");
uReportMsgTxt += "<br/>"+tr("report_question_text");
uReportMsgTxt += "<br/><br/><b>"+tr("report_reason")+"</b>: <input type='text' id='uReportMsgInput' placeholder='" + tr("reason") + "' />"
MessageBox(tr("report_question"), uReportMsgTxt, [tr("confirm_m"), tr("cancel")], [
(function() {
res = document.querySelector("#uReportMsgInput").value;
xhr = new XMLHttpRequest();
xhr.open("GET", "/report/" + user_id + "?reason=" + res + "&type=user", 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", "#_photoDelete, #_videoDelete", function(e) {
var formHtml = "<form id='tmpPhDelF' action='" + u(this).attr("href") + "' >";
formHtml += "<input type='hidden' name='hash' value='" + u("meta[name=csrf]").attr("value") + "' />";
formHtml += "</form>";
u("body").append(formHtml);
MessageBox(tr('warning'), tr('question_confirm'), [
tr('yes'),
tr('no')
], [
(function() {
u("#tmpPhDelF").nodes[0].submit();
}),
(function() {
u("#tmpPhDelF").remove();
}),
]);
e.stopPropagation()
return e.preventDefault();
});
/* @rem-pai why this func wasn't named as "#_deleteDialog"? It looks universal IMO */
u(document).on("click", "#_noteDelete", function(e) {
var formHtml = "<form id='tmpPhDelF' action='" + u(this).attr("href") + "' >";
formHtml += "<input type='hidden' name='hash' value='" + u("meta[name=csrf]").attr("value") + "' />";
formHtml += "</form>";
u("body").append(formHtml);
MessageBox(tr('warning'), tr('question_confirm'), [
tr('yes'),
tr('no')
], [
(function() {
u("#tmpPhDelF").nodes[0].submit();
}),
(function() {
u("#tmpPhDelF").remove();
}),
]);
e.stopPropagation()
return e.preventDefault();
});
// TODO REWRITE cuz its a little broken
u(document).on("click", "#_pinGroup", async function(e) {
e.preventDefault();
e.stopPropagation()
let link = u(this).attr("href");
let thisButton = u(this);
let groupName = u(this).attr("data-group-name");
let groupUrl = u(this).attr("data-group-url");
let list = u('#_groupListPinnedGroups');
thisButton.nodes[0].classList.add('loading');
thisButton.nodes[0].classList.add('disable');
let req = await ky(link);
if(req.ok == false) {
NewNotification(tr('error'), tr('error_1'), null);
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return;
}
if(!parseAjaxResponse(await req.text())) {
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return;
}
// Adding a divider if not already there
if(list.nodes[0].children.length == 0) {
list.nodes[0].append(u('<div class="menu_divider"></div>').first());
}
// Changing the button name
if(thisButton.html().trim() == tr('remove_from_left_menu')) {
thisButton.html(tr('add_to_left_menu'));
for(let i = 0; i < list.nodes[0].children.length; i++) {
let element = list.nodes[0].children[i];
if(element.pathname == groupUrl) {
element.remove();
}
}
}else{
thisButton.html(tr('remove_from_left_menu'));
list.nodes[0].append(u('<a href="' + groupUrl + '" class="link group_link">' + groupName + '</a>').first());
}
// Adding the group to the left group list
if(list.nodes[0].children[0].className != "menu_divider" || list.nodes[0].children.length == 1) {
list.nodes[0].children[0].remove();
}
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return false;
});
u(document).handle("submit", "#_submitUserSubscriptionAction", async function(e) {
e.preventDefault()
e.stopPropagation()
u(this).nodes[0].parentElement.classList.add('loading');
u(this).nodes[0].parentElement.classList.add('disable');
console.log(e.target);
const data = await fetch(u(this).attr('action'), { method: 'POST', body: new FormData(e.target) });
if (data.ok) {
u(this).nodes[0].parentElement.classList.remove('loading');
u(this).nodes[0].parentElement.classList.remove('disable');
if (e.target[0].value == "add") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_add_msg");
} else if (e.target[0].value == "rej") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_rej_msg");
} else if (e.target[0].value == "rem") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_rem_msg");
}
}
})
// Migrated from inline end
var tooltipClientTemplate = Handlebars.compile(` var tooltipClientTemplate = Handlebars.compile(`
<table> <table>
<tr> <tr>
@ -2234,9 +2083,7 @@ async function __processPaginatorNextPage(page)
window.player.loadContext(page) window.player.loadContext(page)
} }
const new_url = new URL(location.href) location.hash = 'pages/'+page
new_url.hash = page
history.replaceState(null, null, new_url)
if(typeof __scrollHook != 'undefined') { if(typeof __scrollHook != 'undefined') {
__scrollHook(page) __scrollHook(page)
@ -2400,52 +2247,3 @@ u(document).on('mouseover mousemove mouseout', `div[data-tip='simple']`, (e) =>
</div> </div>
`) `)
}) })
function setStatusEditorShown(shown) {
document.getElementById("status_editor").style.display = shown ? "block" : "none";
}
document.addEventListener("click", event => {
if(u('#status_editor').length < 1) {
return
}
if(!event.target.closest("#status_editor") && !event.target.closest("#page_status_text"))
setStatusEditorShown(false);
});
u(document).on('click', '#page_status_text', (e) => {
setStatusEditorShown(true)
})
async function changeStatus() {
const status = document.status_popup_form.status.value;
const broadcast = document.status_popup_form.broadcast.checked;
document.status_popup_form.submit.innerHTML = "<div class=\"button-loading\"></div>";
document.status_popup_form.submit.disabled = true;
const formData = new FormData();
formData.append("status", status);
formData.append("broadcast", Number(broadcast));
formData.append("hash", document.status_popup_form.hash.value);
const response = await ky.post("/edit?act=status", {body: formData});
if(!parseAjaxResponse(await response.text())) {
document.status_popup_form.submit.innerHTML = tr("send");
document.status_popup_form.submit.disabled = false;
return;
}
if(document.status_popup_form.status.value === "") {
document.querySelector("#page_status_text").innerHTML = `[ ${tr("change_status")} ]`;
document.querySelector("#page_status_text").className = "edit_link page_status_edit_button";
} else {
document.querySelector("#page_status_text").innerHTML = status;
document.querySelector("#page_status_text").className = "page_status page_status_edit_button";
}
setStatusEditorShown(false);
document.status_popup_form.submit.innerHTML = tr("send");
document.status_popup_form.submit.disabled = false;
}

View file

@ -39,6 +39,126 @@ function parseAjaxResponse(responseString) {
} }
} }
document.addEventListener("DOMContentLoaded", function() { //BEGIN
$(document).on("click", "#_photoDelete, #_videoDelete", function(e) {
var formHtml = "<form id='tmpPhDelF' action='" + u(this).attr("href") + "' >";
formHtml += "<input type='hidden' name='hash' value='" + u("meta[name=csrf]").attr("value") + "' />";
formHtml += "</form>";
u("body").append(formHtml);
MessageBox(tr('warning'), tr('question_confirm'), [
tr('yes'),
tr('no')
], [
(function() {
u("#tmpPhDelF").nodes[0].submit();
}),
(function() {
u("#tmpPhDelF").remove();
}),
]);
return e.preventDefault();
});
/* @rem-pai why this func wasn't named as "#_deleteDialog"? It looks universal IMO */
u(document).on("click", "#_noteDelete", function(e) {
var formHtml = "<form id='tmpPhDelF' action='" + u(this).attr("href") + "' >";
formHtml += "<input type='hidden' name='hash' value='" + u("meta[name=csrf]").attr("value") + "' />";
formHtml += "</form>";
u("body").append(formHtml);
MessageBox(tr('warning'), tr('question_confirm'), [
tr('yes'),
tr('no')
], [
(function() {
u("#tmpPhDelF").nodes[0].submit();
}),
(function() {
u("#tmpPhDelF").remove();
}),
]);
return e.preventDefault();
});
u("#_pinGroup").on("click", async function(e) {
e.preventDefault();
let link = u(this).attr("href");
let thisButton = u(this);
let groupName = u(this).attr("data-group-name");
let groupUrl = u(this).attr("data-group-url");
let list = u('#_groupListPinnedGroups');
thisButton.nodes[0].classList.add('loading');
thisButton.nodes[0].classList.add('disable');
let req = await ky(link);
if(req.ok == false) {
NewNotification(tr('error'), tr('error_1'), null);
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return;
}
if(!parseAjaxResponse(await req.text())) {
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return;
}
// Adding a divider if not already there
if(list.nodes[0].children.length == 0) {
list.nodes[0].append(u('<div class="menu_divider"></div>').first());
}
// Changing the button name
if(thisButton.html().trim() == tr('remove_from_left_menu')) {
thisButton.html(tr('add_to_left_menu'));
for(let i = 0; i < list.nodes[0].children.length; i++) {
let element = list.nodes[0].children[i];
if(element.pathname == groupUrl) {
element.remove();
}
}
}else{
thisButton.html(tr('remove_from_left_menu'));
list.nodes[0].append(u('<a href="' + groupUrl + '" class="link group_link">' + groupName + '</a>').first());
}
// Adding the group to the left group list
if(list.nodes[0].children[0].className != "menu_divider" || list.nodes[0].children.length == 1) {
list.nodes[0].children[0].remove();
}
thisButton.nodes[0].classList.remove('loading');
thisButton.nodes[0].classList.remove('disable');
return false;
});
u(document).handle("submit", "#_submitUserSubscriptionAction", async function(e) {
u(this).nodes[0].parentElement.classList.add('loading');
u(this).nodes[0].parentElement.classList.add('disable');
console.log(e.target);
const data = await fetch(u(this).attr('action'), { method: 'POST', body: new FormData(e.target) });
if (data.ok) {
u(this).nodes[0].parentElement.classList.remove('loading');
u(this).nodes[0].parentElement.classList.remove('disable');
if (e.target[0].value == "add") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_add_msg");
} else if (e.target[0].value == "rej") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_rej_msg");
} else if (e.target[0].value == "rem") {
u(this).nodes[0].parentElement.innerHTML = tr("friends_rem_msg");
}
}
})
}); //END ONREADY DECLS
function setClubAdminComment(clubId, adminId, hash) { function setClubAdminComment(clubId, adminId, hash) {
MessageBox("Изменить комментарий к администратору", ` MessageBox("Изменить комментарий к администратору", `
<form action="/club${clubId}/setAdmin" method="post" id="uClubAdminCommentForm_${clubId}_${adminId}"> <form action="/club${clubId}/setAdmin" method="post" id="uClubAdminCommentForm_${clubId}_${adminId}">

View file

@ -1,4 +1,8 @@
window.router = new class { window.router = new class {
skeletons = {
}
get csrf() { get csrf() {
return u("meta[name=csrf]").attr("value") return u("meta[name=csrf]").attr("value")
} }
@ -16,30 +20,12 @@ window.router = new class {
__appendScript(script) { __appendScript(script) {
const _t_scr = document.createElement('script') const _t_scr = document.createElement('script')
_t_scr.crossorigin = 'anonymous'
if(script.getAttribute('integrity')) {
_t_scr.setAttribute('integrity', script.getAttribute('integrity'))
}
if(script.getAttribute('id')) {
_t_scr.id = script.id
}
if(script.getAttribute('type')) {
_t_scr.type = script.type
}
//const parent = script.parentNode
//const idx = Array.from(parent.children).indexOf(script)
if(script.src) { if(script.src) {
_t_scr.src = script.src _t_scr.src = script.src
} else { } else {
_t_scr.async = false
_t_scr.textContent = script.textContent _t_scr.textContent = script.textContent
} }
//parent.children[idx].before(script)
document.body.appendChild(_t_scr) document.body.appendChild(_t_scr)
} }
@ -47,26 +33,25 @@ window.router = new class {
u(`script:not([src])`).remove() u(`script:not([src])`).remove()
} }
__closeMsgs() {
window.messagebox_stack.forEach(msg => msg.close())
}
__appendPage(parsed_content) { __appendPage(parsed_content) {
const scripts_to_append = [] if(u('.paginator:not(.paginator-at-top)').length > 0) {
showMoreObserver.unobserve(u('.paginator:not(.paginator-at-top)').nodes[0])
}
const page_body = u(parsed_content.querySelector('.page_body')) const page_body = u(parsed_content.querySelector('.page_body'))
const sidebar = u(parsed_content.querySelector('.sidebar')) const sidebar = u(parsed_content.querySelector('.sidebar'))
const page_header = u(parsed_content.querySelector('.page_header')) const page_header = u(parsed_content.querySelector('.page_header'))
if(page_body.length < 1) { if(page_body.length < 1) {
throw new Error('Invalid page has been loaded') makeError('Err')
return return
} }
window.__current_page_audio_context = null
this.__clearScripts() this.__clearScripts()
parsed_content.querySelectorAll('.page_body script, #_js_ep_script').forEach(script => { parsed_content.querySelectorAll('script').forEach(script => {
console.log(script)
if(!this.__isScriptAlreadyLoaded(script)) { if(!this.__isScriptAlreadyLoaded(script)) {
scripts_to_append.push(script) this.__appendScript(script)
script.parentNode.removeChild(script)
} }
}) })
u('.page_body').html(page_body.html()) u('.page_body').html(page_body.html())
@ -86,43 +71,15 @@ window.router = new class {
u("meta[name=csrf]").attr("value", u(parsed_content.querySelector('meta[name=csrf]')).attr('value')) u("meta[name=csrf]").attr("value", u(parsed_content.querySelector('meta[name=csrf]')).attr('value'))
document.title = parsed_content.title document.title = parsed_content.title
scripts_to_append.forEach(append_me => {
this.__appendScript(append_me)
})
}
async __integratePage() {
window.scrollTo(0, 0) window.scrollTo(0, 0)
bsdnHydrate() bsdnHydrate()
if(u('.paginator:not(.paginator-at-top)').length > 0) { if(u('.paginator:not(.paginator-at-top)').length > 0) {
showMoreObserver.observe(u('.paginator:not(.paginator-at-top)').nodes[0]) showMoreObserver.observe(u('.paginator:not(.paginator-at-top)').nodes[0])
} }
if(u(`div[class$="_small_block"]`).length > 0 && typeof smallBlockObserver != 'undefined') {
smallBlockObserver.observe(u(`div[class$="_small_block"]`).nodes[0])
}
if(window.player) {
await window.player._handlePageTransition()
}
}
__unlinkObservers() {
if(u('.paginator:not(.paginator-at-top)').length > 0) {
showMoreObserver.unobserve(u('.paginator:not(.paginator-at-top)').nodes[0])
}
if(u(`div[class$="_small_block"]`).length > 0 && typeof smallBlockObserver != 'undefined') {
smallBlockObserver.unobserve(u(`div[class$="_small_block"]`).nodes[0])
}
} }
checkUrl(url) { checkUrl(url) {
if(window.openvk.disable_ajax == 1) {
return false
}
if((localStorage.getItem('ux.disable_ajax_routing') ?? 0) == 1 || window.openvk.current_id == 0) { if((localStorage.getItem('ux.disable_ajax_routing') ?? 0) == 1 || window.openvk.current_id == 0) {
return false return false
} }
@ -135,10 +92,6 @@ window.router = new class {
return false return false
} }
if(url.indexOf('hash=') != -1) {
return false
}
return true return true
} }
@ -162,6 +115,8 @@ window.router = new class {
const push_url = params.push_state ?? true const push_url = params.push_state ?? true
const next_page_url = new URL(url) const next_page_url = new URL(url)
next_page_url.searchParams.append('al', 1)
next_page_url.searchParams.append('hash', this.csrf)
if(push_url) { if(push_url) {
history.pushState({'from_router': 1}, '', url) history.pushState({'from_router': 1}, '', url)
} else { } else {
@ -170,10 +125,7 @@ window.router = new class {
const parser = new DOMParser const parser = new DOMParser
const next_page_request = await fetch(next_page_url, { const next_page_request = await fetch(next_page_url, {
method: 'GET', method: 'GET'
headers: {
'X-OpenVK-Ajax-Query': '1',
}
}) })
const next_page_text = await next_page_request.text() const next_page_text = await next_page_request.text()
const parsed_content = parser.parseFromString(next_page_text, 'text/html') const parsed_content = parser.parseFromString(next_page_text, 'text/html')
@ -181,17 +133,7 @@ window.router = new class {
history.replaceState({'from_router': 1}, '', next_page_request.url) history.replaceState({'from_router': 1}, '', next_page_request.url)
} }
this.__closeMsgs() this.__appendPage(parsed_content)
this.__unlinkObservers()
try {
this.__appendPage(parsed_content)
await this.__integratePage()
} catch(e) {
console.error(e)
next_page_url.searchParams.delete('al', 1)
location.assign(next_page_url)
}
} }
} }
@ -202,29 +144,19 @@ u(document).on('click', 'a', async (e) => {
let url = target.nodes[0].href let url = target.nodes[0].href
if(id) { if(id) {
if(['act_tab_a', 'ki', 'used', '_pinGroup', 'profile_link'].indexOf(id) == -1) { if(['act_tab_a', 'ki'].indexOf(id) == -1) {
console.log('AJAX | Skipping cuz maybe its function call link.') console.log('AJAX | Skipping cuz maybe its function call link.')
return return
} }
} }
/*if(url.indexOf('hash=') != -1) {
e.preventDefault()
return false
}*/
if(target.rel == 'nofollow') {
console.log('AJAX | Skipped because its nofollow')
return
}
if(!dom_url || dom_url == '#' || dom_url.indexOf('javascript:') != -1) { if(!dom_url || dom_url == '#' || dom_url.indexOf('javascript:') != -1) {
console.log('AJAX | Skipped because its anchor or function call') console.log('AJAX | Skipped cuz its anchor or function call')
return return
} }
if(target.attr('target') == '_blank') { if(target.attr('target') == '_blank') {
console.log('AJAX | Skipping because its _blank.') console.log('AJAX | Skipping cuz its _blank.')
return return
} }
@ -256,12 +188,9 @@ u(document).on('submit', 'form', async (e) => {
const form = e.target const form = e.target
const method = form.method ?? 'get' const method = form.method ?? 'get'
const url = form.action const url = form.action
if(form.onsubmit) {
u('#ajloader').removeClass('shown')
return false
}
const url_object = new URL(url) const url_object = new URL(url)
url_object.searchParams.append('al', 1)
if(method == 'get' || method == 'GET') { if(method == 'get' || method == 'GET') {
url_object.searchParams.append('hash', window.router.csrf) url_object.searchParams.append('hash', window.router.csrf)
$(form).serializeArray().forEach(param => { $(form).serializeArray().forEach(param => {
@ -270,16 +199,12 @@ u(document).on('submit', 'form', async (e) => {
} }
if(!url) { if(!url) {
u('#ajloader').removeClass('shown')
return return
} }
const form_data = serializeForm(form) const form_data = serializeForm(form)
const request_object = { const request_object = {
method: method, method: method
headers: {
'X-OpenVK-Ajax-Query': '1',
}
} }
if(method != 'GET' && method != 'get') { if(method != 'GET' && method != 'get') {
@ -290,7 +215,6 @@ u(document).on('submit', 'form', async (e) => {
const form_result = await form_res.text() const form_result = await form_res.text()
switch(form_res.status) { switch(form_res.status) {
case 500: case 500:
case 502:
makeError(form_res.statusText) makeError(form_res.statusText)
break break
} }
@ -310,15 +234,6 @@ u(document).on('submit', 'form', async (e) => {
console.log(form_res) console.log(form_res)
window.router.__appendPage(parsed_content) window.router.__appendPage(parsed_content)
await window.router.__integratePage()
u('#ajloader').removeClass('shown') u('#ajloader').removeClass('shown')
}) })
window.addEventListener('popstate', (e) => {
e.preventDefault();
window.router.route({
url: location.href,
push_state: false,
})
})

View file

@ -186,7 +186,7 @@ function serializeForm(form)
{ {
const u_ = u(form) const u_ = u(form)
const inputs = u_.find('input, textarea') const inputs = u_.find('input, textarea')
console.log(inputs)
let fd = new FormData() let fd = new FormData()
inputs.nodes.forEach(inp => { inputs.nodes.forEach(inp => {
if(!inp || !inp.name) { if(!inp || !inp.name) {
@ -203,14 +203,7 @@ function serializeForm(form)
case 'hidden': case 'hidden':
case 'text': case 'text':
case 'textarea': case 'textarea':
case 'select':
fd.append(inp.name, inp.value) fd.append(inp.name, inp.value)
break
case 'checkbox':
if(inp.checked) {
fd.append(inp.name, inp.value)
}
break break
case 'file': case 'file':
for(const __file of inp.files) { for(const __file of inp.files) {