1
0
Fork 0
mirror of https://github.com/jellyfin/jellyfin-web synced 2025-03-30 19:56:21 +00:00
jellyfin-web/src/components/multiselect/multiselect.js

591 lines
19 KiB
JavaScript
Raw Normal View History

2019-11-05 18:50:41 +03:00
define(["browser", "appStorage", "apphost", "loading", "connectionManager", "globalize", "appRouter", "dom", "css!./multiselect"], function (browser, appStorage, appHost, loading, connectionManager, globalize, appRouter, dom) {
"use strict";
var selectedItems = [];
var selectedElements = [];
var currentSelectionCommandsPanel;
2018-10-23 01:05:09 +03:00
function hideSelections() {
2018-10-23 01:05:09 +03:00
var selectionCommandsPanel = currentSelectionCommandsPanel;
if (selectionCommandsPanel) {
selectionCommandsPanel.parentNode.removeChild(selectionCommandsPanel);
currentSelectionCommandsPanel = null;
selectedItems = [];
selectedElements = [];
2019-11-05 18:50:41 +03:00
var elems = document.querySelectorAll(".itemSelectionPanel");
for (var i = 0, length = elems.length; i < length; i++) {
2018-10-23 01:05:09 +03:00
var parent = elems[i].parentNode;
parent.removeChild(elems[i]);
2019-11-05 18:50:41 +03:00
parent.classList.remove("withMultiSelect");
2018-10-23 01:05:09 +03:00
}
}
}
function onItemSelectionPanelClick(e, itemSelectionPanel) {
// toggle the checkbox, if it wasn't clicked on
2019-11-05 18:50:41 +03:00
if (!dom.parentWithClass(e.target, "chkItemSelect")) {
var chkItemSelect = itemSelectionPanel.querySelector(".chkItemSelect");
if (chkItemSelect) {
2019-11-05 18:50:41 +03:00
if (chkItemSelect.classList.contains("checkedInitial")) {
chkItemSelect.classList.remove("checkedInitial");
} else {
2018-10-23 01:05:09 +03:00
var newValue = !chkItemSelect.checked;
chkItemSelect.checked = newValue;
updateItemSelection(chkItemSelect, newValue);
2018-10-23 01:05:09 +03:00
}
}
2018-10-23 01:05:09 +03:00
}
e.preventDefault();
e.stopPropagation();
return false;
2018-10-23 01:05:09 +03:00
}
function updateItemSelection(chkItemSelect, selected) {
2019-11-05 18:50:41 +03:00
var id = dom.parentWithAttribute(chkItemSelect, "data-id").getAttribute("data-id");
2018-10-23 01:05:09 +03:00
if (selected) {
var current = selectedItems.filter(function (i) {
return i === id;
});
if (!current.length) {
selectedItems.push(id);
selectedElements.push(chkItemSelect);
}
} else {
selectedItems = selectedItems.filter(function (i) {
return i !== id;
});
selectedElements = selectedElements.filter(function (i) {
return i !== chkItemSelect;
});
}
2018-10-23 01:05:09 +03:00
if (selectedItems.length) {
2019-11-05 18:50:41 +03:00
var itemSelectionCount = document.querySelector(".itemSelectionCount");
if (itemSelectionCount) {
itemSelectionCount.innerHTML = selectedItems.length;
}
} else {
hideSelections();
}
2018-10-23 01:05:09 +03:00
}
function onSelectionChange(e) {
updateItemSelection(this, this.checked);
2018-10-23 01:05:09 +03:00
}
function showSelection(item, isChecked) {
2019-11-05 18:50:41 +03:00
var itemSelectionPanel = item.querySelector(".itemSelectionPanel");
2018-10-23 01:05:09 +03:00
if (!itemSelectionPanel) {
2019-11-05 18:50:41 +03:00
itemSelectionPanel = document.createElement("div");
itemSelectionPanel.classList.add("itemSelectionPanel");
2019-11-05 18:50:41 +03:00
var parent = item.querySelector(".cardBox") || item.querySelector(".cardContent");
parent.classList.add("withMultiSelect");
parent.appendChild(itemSelectionPanel);
2019-11-05 18:50:41 +03:00
var cssClass = "chkItemSelect";
if (isChecked && !browser.firefox) {
// In firefox, the initial tap hold doesnt' get treated as a click
// In other browsers it does, so we need to make sure that initial click is ignored
2019-11-05 18:50:41 +03:00
cssClass += " checkedInitial";
}
2019-11-05 18:50:41 +03:00
var checkedAttribute = isChecked ? " checked" : "";
itemSelectionPanel.innerHTML = '<label class="checkboxContainer"><input type="checkbox" is="emby-checkbox" data-outlineclass="multiSelectCheckboxOutline" class="' + cssClass + '"' + checkedAttribute + '/><span></span></label>';
2019-11-05 18:50:41 +03:00
var chkItemSelect = itemSelectionPanel.querySelector(".chkItemSelect");
chkItemSelect.addEventListener("change", onSelectionChange);
2018-10-23 01:05:09 +03:00
}
}
function showSelectionCommands() {
2018-10-23 01:05:09 +03:00
var selectionCommandsPanel = currentSelectionCommandsPanel;
2018-10-23 01:05:09 +03:00
if (!selectionCommandsPanel) {
2019-11-05 18:50:41 +03:00
selectionCommandsPanel = document.createElement("div");
selectionCommandsPanel.classList.add("selectionCommandsPanel");
document.body.appendChild(selectionCommandsPanel);
currentSelectionCommandsPanel = selectionCommandsPanel;
2019-11-05 18:50:41 +03:00
var html = "";
2020-04-26 02:37:28 +03:00
html += '<button is="paper-icon-button-light" class="btnCloseSelectionPanel autoSize"><span class="material-icons close"></span></button>';
html += '<h1 class="itemSelectionCount"></h1>';
2020-04-25 10:00:20 +03:00
const moreIcon = "more_horiz";
2020-04-26 02:37:28 +03:00
html += '<button is="paper-icon-button-light" class="btnSelectionPanelOptions autoSize" style="margin-left:auto;"><span class="material-icons ' + moreIcon + '"></span></button>';
selectionCommandsPanel.innerHTML = html;
2019-11-05 18:50:41 +03:00
selectionCommandsPanel.querySelector(".btnCloseSelectionPanel").addEventListener("click", hideSelections);
2019-11-05 18:50:41 +03:00
var btnSelectionPanelOptions = selectionCommandsPanel.querySelector(".btnSelectionPanelOptions");
2019-11-05 18:50:41 +03:00
dom.addEventListener(btnSelectionPanelOptions, "click", showMenuForSelectedItems, { passive: true });
2018-10-23 01:05:09 +03:00
}
}
function alertText(options) {
return new Promise(function (resolve, reject) {
2019-11-05 18:50:41 +03:00
require(["alert"], function (alert) {
alert(options).then(resolve, resolve);
});
});
2018-10-23 01:05:09 +03:00
}
function deleteItems(apiClient, itemIds) {
return new Promise(function (resolve, reject) {
2019-11-05 18:50:41 +03:00
var msg = globalize.translate("ConfirmDeleteItem");
var title = globalize.translate("HeaderDeleteItem");
if (itemIds.length > 1) {
2019-11-05 18:50:41 +03:00
msg = globalize.translate("ConfirmDeleteItems");
title = globalize.translate("HeaderDeleteItems");
}
2019-11-05 18:50:41 +03:00
require(["confirm"], function (confirm) {
confirm(msg, title).then(function () {
var promises = itemIds.map(function (itemId) {
apiClient.deleteItem(itemId);
2018-10-23 01:05:09 +03:00
});
Promise.all(promises).then(resolve, function () {
2019-11-05 18:50:41 +03:00
alertText(globalize.translate("ErrorDeletingItem")).then(reject, reject);
});
}, reject);
});
});
2018-10-23 01:05:09 +03:00
}
function showMenuForSelectedItems(e) {
2018-10-23 01:05:09 +03:00
var apiClient = connectionManager.currentApiClient();
apiClient.getCurrentUser().then(function (user) {
2018-10-23 01:05:09 +03:00
var menuItems = [];
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("AddToCollection"),
id: "addtocollection",
icon: "add"
});
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("AddToPlaylist"),
id: "playlist",
icon: "playlist_add"
});
// TODO: Be more dynamic based on what is selected
if (user.Policy.EnableContentDeletion) {
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("Delete"),
id: "delete",
icon: "delete"
});
}
2019-11-05 18:50:41 +03:00
if (user.Policy.EnableContentDownloading && appHost.supports("filedownload")) {
menuItems.push({
2020-03-26 14:25:16 +01:00
name: globalize.translate("ButtonDownload"),
2019-11-05 18:50:41 +03:00
id: "download",
icon: "file_download"
});
}
2019-08-15 18:37:37 +03:00
if (user.Policy.IsAdministrator) {
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("GroupVersions"),
id: "groupvideos",
icon: "call_merge"
2019-08-15 18:37:37 +03:00
});
}
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("MarkPlayed"),
id: "markplayed",
2019-11-05 18:43:10 +03:00
icon: "check_box"
});
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("MarkUnplayed"),
id: "markunplayed",
2019-11-05 18:43:10 +03:00
icon: "check_box_outline_blank"
});
2018-10-23 01:05:09 +03:00
menuItems.push({
2019-11-05 18:50:41 +03:00
name: globalize.translate("RefreshMetadata"),
id: "refresh",
2019-11-05 18:43:10 +03:00
icon: "refresh"
});
require(['actionsheet'], function (actionsheet) {
2018-10-23 01:05:09 +03:00
actionsheet.show({
items: menuItems,
positionTo: e.target,
callback: function (id) {
var items = selectedItems.slice(0);
var serverId = apiClient.serverInfo().Id;
2018-10-23 01:05:09 +03:00
switch (id) {
2019-11-05 18:50:41 +03:00
case "addtocollection":
require(["collectionEditor"], function (collectionEditor) {
new collectionEditor().show({
2018-10-23 01:05:09 +03:00
items: items,
serverId: serverId
});
});
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
2019-11-05 18:50:41 +03:00
case "playlist":
require(["playlistEditor"], function (playlistEditor) {
new playlistEditor().show({
2018-10-23 01:05:09 +03:00
items: items,
serverId: serverId
});
});
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
2019-11-05 18:50:41 +03:00
case "delete":
deleteItems(apiClient, items).then(dispatchNeedsRefresh);
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
2019-11-05 18:50:41 +03:00
case "groupvideos":
2018-10-23 01:05:09 +03:00
combineVersions(apiClient, items);
break;
2019-11-05 18:50:41 +03:00
case "markplayed":
items.forEach(function (itemId) {
apiClient.markPlayed(apiClient.getCurrentUserId(), itemId);
});
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
2019-11-05 18:50:41 +03:00
case "markunplayed":
items.forEach(function (itemId) {
apiClient.markUnplayed(apiClient.getCurrentUserId(), itemId);
});
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
2019-11-05 18:50:41 +03:00
case "refresh":
require(["refreshDialog"], function (refreshDialog) {
2018-10-23 01:05:09 +03:00
new refreshDialog({
itemIds: items,
serverId: serverId
}).show();
});
hideSelections();
dispatchNeedsRefresh();
2018-10-23 01:05:09 +03:00
break;
default:
break;
2018-10-23 01:05:09 +03:00
}
}
});
});
});
2018-10-23 01:05:09 +03:00
}
function dispatchNeedsRefresh() {
2018-10-23 01:05:09 +03:00
var elems = [];
[].forEach.call(selectedElements, function (i) {
2019-11-05 18:50:41 +03:00
var container = dom.parentWithAttribute(i, "is", "emby-itemscontainer");
if (container && elems.indexOf(container) === -1) {
elems.push(container);
}
2018-10-23 01:05:09 +03:00
});
for (var i = 0, length = elems.length; i < length; i++) {
elems[i].notifyRefreshNeeded(true);
}
2018-10-23 01:05:09 +03:00
}
function combineVersions(apiClient, selection) {
if (selection.length < 2) {
2019-11-05 18:50:41 +03:00
require(["alert"], function (alert) {
alert({
2019-11-05 18:50:41 +03:00
text: globalize.translate("PleaseSelectTwoItems")
});
});
return;
}
loading.show();
apiClient.ajax({
2018-10-23 01:05:09 +03:00
type: "POST",
2019-11-05 18:50:41 +03:00
url: apiClient.getUrl("Videos/MergeVersions", { Ids: selection.join(",") })
}).then(function () {
loading.hide();
hideSelections();
dispatchNeedsRefresh();
});
2018-10-23 01:05:09 +03:00
}
function showSelections(initialCard) {
2019-11-05 18:50:41 +03:00
require(["emby-checkbox"], function () {
var cards = document.querySelectorAll(".card");
for (var i = 0, length = cards.length; i < length; i++) {
showSelection(cards[i], initialCard === cards[i]);
}
showSelectionCommands();
updateItemSelection(initialCard, true);
});
2018-10-23 01:05:09 +03:00
}
function onContainerClick(e) {
2018-10-23 01:05:09 +03:00
var target = e.target;
2018-10-23 01:05:09 +03:00
if (selectedItems.length) {
2019-11-05 18:50:41 +03:00
var card = dom.parentWithClass(target, "card");
2018-10-23 01:05:09 +03:00
if (card) {
2019-11-05 18:50:41 +03:00
var itemSelectionPanel = card.querySelector(".itemSelectionPanel");
if (itemSelectionPanel) {
return onItemSelectionPanelClick(e, itemSelectionPanel);
}
2018-10-23 01:05:09 +03:00
}
e.preventDefault();
e.stopPropagation();
return false;
2018-10-23 01:05:09 +03:00
}
}
2019-11-05 18:50:41 +03:00
document.addEventListener("viewbeforehide", hideSelections);
return function (options) {
var self = this;
var container = options.container;
function onTapHold(e) {
2019-11-05 18:50:41 +03:00
var card = dom.parentWithClass(e.target, "card");
if (card) {
showSelections(card);
2018-10-23 01:05:09 +03:00
}
e.preventDefault();
// It won't have this if it's a hammer event
if (e.stopPropagation) {
e.stopPropagation();
2018-10-23 01:05:09 +03:00
}
return false;
}
2018-10-23 01:05:09 +03:00
function getTouches(e) {
return e.changedTouches || e.targetTouches || e.touches;
}
var touchTarget;
var touchStartTimeout;
var touchStartX;
var touchStartY;
function onTouchStart(e) {
var touch = getTouches(e)[0];
touchTarget = null;
touchStartX = 0;
touchStartY = 0;
if (touch) {
touchStartX = touch.clientX;
touchStartY = touch.clientY;
var element = touch.target;
if (element) {
2019-11-05 18:50:41 +03:00
var card = dom.parentWithClass(element, "card");
if (card) {
if (touchStartTimeout) {
clearTimeout(touchStartTimeout);
touchStartTimeout = null;
}
touchTarget = card;
touchStartTimeout = setTimeout(onTouchStartTimerFired, 550);
2018-10-23 01:05:09 +03:00
}
}
}
}
function onTouchMove(e) {
2018-10-23 01:05:09 +03:00
if (touchTarget) {
var touch = getTouches(e)[0];
var deltaX;
var deltaY;
if (touch) {
var touchEndX = touch.clientX || 0;
var touchEndY = touch.clientY || 0;
deltaX = Math.abs(touchEndX - (touchStartX || 0));
deltaY = Math.abs(touchEndY - (touchStartY || 0));
} else {
deltaX = 100;
deltaY = 100;
}
if (deltaX >= 5 || deltaY >= 5) {
onMouseOut(e);
2018-10-23 01:05:09 +03:00
}
}
}
function onTouchEnd(e) {
onMouseOut(e);
}
function onMouseDown(e) {
2018-10-23 01:05:09 +03:00
if (touchStartTimeout) {
clearTimeout(touchStartTimeout);
touchStartTimeout = null;
2018-10-23 01:05:09 +03:00
}
touchTarget = e.target;
touchStartTimeout = setTimeout(onTouchStartTimerFired, 550);
}
function onMouseOut(e) {
if (touchStartTimeout) {
clearTimeout(touchStartTimeout);
touchStartTimeout = null;
2018-10-23 01:05:09 +03:00
}
touchTarget = null;
}
function onTouchStartTimerFired() {
2018-10-23 01:05:09 +03:00
if (!touchTarget) {
return;
2018-10-23 01:05:09 +03:00
}
2019-11-05 18:50:41 +03:00
var card = dom.parentWithClass(touchTarget, "card");
touchTarget = null;
if (card) {
showSelections(card);
2018-10-23 01:05:09 +03:00
}
}
function initTapHold(element) {
// mobile safari doesn't allow contextmenu override
if (browser.touch && !browser.safari) {
2019-11-05 18:50:41 +03:00
element.addEventListener("contextmenu", onTapHold);
} else {
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "touchstart", onTouchStart, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "touchmove", onTouchMove, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "touchend", onTouchEnd, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "touchcancel", onTouchEnd, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "mousedown", onMouseDown, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "mouseleave", onMouseOut, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.addEventListener(element, "mouseup", onMouseOut, {
passive: true
});
2018-10-23 01:05:09 +03:00
}
}
initTapHold(container);
if (options.bindOnClick !== false) {
2019-11-05 18:50:41 +03:00
container.addEventListener("click", onContainerClick);
}
self.onContainerClick = onContainerClick;
self.destroy = function () {
2019-11-05 18:50:41 +03:00
container.removeEventListener("click", onContainerClick);
container.removeEventListener("contextmenu", onTapHold);
var element = container;
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "touchstart", onTouchStart, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "touchmove", onTouchMove, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "touchend", onTouchEnd, {
passive: true
});
// this fires in safari due to magnifying class
2019-11-05 18:50:41 +03:00
//dom.removeEventListener(element, "touchcancel", onTouchEnd, {
// passive: true
//});
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "mousedown", onMouseDown, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "mouseleave", onMouseOut, {
passive: true
});
2019-11-05 18:50:41 +03:00
dom.removeEventListener(element, "mouseup", onMouseOut, {
passive: true
});
};
};
});