Rework deck editor page and allow drag/drop of decks and cards

This commit is contained in:
Andrio Celos 2023-08-31 12:26:03 +10:00
parent 252b326262
commit 5dfd7b6a44
11 changed files with 311 additions and 75 deletions

3
.gitmodules vendored
View File

@ -1,3 +1,6 @@
[submodule "TableturfBattleClient/qrcodejs"]
path = TableturfBattleClient/qrcodejs
url = https://github.com/davidshimjs/qrcodejs.git
[submodule "TableturfBattleClient/mobile-drag-drop"]
path = TableturfBattleClient/mobile-drag-drop
url = https://github.com/timruffles/mobile-drag-drop.git

View File

@ -0,0 +1,3 @@
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="white" class="bi bi-grip-horizontal" viewBox="0 0 16 16">
<path d="M2 8a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm0-3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm3 3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm0-3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm3 3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm0-3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm3 3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm0-3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm3 3a1 1 0 1 1 0 2 1 1 0 0 1 0-2zm0-3a1 1 0 1 1 0 2 1 1 0 0 1 0-2z"/>
</svg>

After

Width:  |  Height:  |  Size: 479 B

View File

@ -0,0 +1,4 @@
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="grey" class="bi bi-plus-circle" viewBox="0 0 16 16">
<path d="M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z"/>
<path d="M8 4a.5.5 0 0 1 .5.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3A.5.5 0 0 1 8 4z"/>
</svg>

After

Width:  |  Height:  |  Size: 329 B

View File

@ -15,6 +15,11 @@
<link rel="stylesheet" href="tableturf.css"/>
<script src="config/config.js"></script>
<script src="qrcodejs/qrcode.js"></script>
<script src="mobile-drag-drop/release/index.min.js"></script>
<script>
// Enable a polyfill for drag/drop support on mobile Firefox.
MobileDragDrop.polyfill({ dragImageTranslateOverride: MobileDragDrop.scrollBehaviourDragImageTranslateOverride });
</script>
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta property="og:type" content="website"/>
<meta name="title" property="og:title" content="Tableturf Battle"/>
@ -402,27 +407,27 @@
<div id="testPlacementList"></div>
</div>
<div id="deckListPage" hidden>
<section id="deckEditordeckListPage">
<section id="deckEditorDeckListSection">
<a id="deckListBackButton" href=".">Back</a>
<h3>Deck list</h3>
<button id="deckExportAllButton">Export all</button>
<div id="deckList" class="deckList">
<div id="addDeckControls">
<button id="newDeckButton">New deck</button>
<button id="importDeckButton">Import deck</button>
</div>
</div>
<div id="addDeckControls">
<button id="newDeckButton">New deck</button>
<button id="importDeckButton">Import deck</button>
</div>
</section>
<section id="deckEditorDeckViewSection" hidden>
<section id="deckEditorDeckViewSection">
<a id="deckViewBackButton" href="#">Back</a>
<h3 id="deckName">Deck</h3>
<div>
<button type="button" id="deckEditButton">Edit</button>
<button type="button" id="deckListTestButton">Test</button>
<button type="button" id="deckExportButton">Export</button>
<button type="button" id="deckRenameButton">Rename</button>
<button type="button" id="deckCopyButton">Copy</button>
<button type="button" id="deckDeleteButton" class="danger">Delete</button>
<h3 id="deckName">&nbsp;</h3>
<div id="deckListToolbar">
<button type="button" id="deckListTestButton" disabled>Test</button>
<button type="button" id="deckExportButton" disabled>Export</button>
<button type="button" id="deckCopyButton" disabled>Copy</button>
<button type="button" id="deckEditButton" disabled>Edit</button>
<button type="button" id="deckRenameButton" disabled>Rename</button>
<button type="button" id="deckDeleteButton" class="danger" disabled>Delete</button>
</div>
<div class="deckSizeContainer">Total: <div id="deckViewSize">0</div></div>
<div id="deckCardListView">
@ -490,15 +495,15 @@
</dialog>
</div>
<div id="deckEditPage" hidden>
<section id="deckEditordeckEditPage">
<section id="deckEditorDeckEditPage">
<h3 id="deckName2">Deck</h3>
<div>
<div id="deckEditToolbar">
<button type="button" id="deckSortButton">Sort</button>
<button type="button" id="deckTestButton">Test</button>
<button type="button" id="deckSaveButton">Save</button>
<button type="button" id="deckCancelButton">Cancel</button>
</div>
<div class="deckSizeContainer">Total: <div id="deckEditSize">0</div></div>
<div class="deckSizeContainer">Total <div id="deckEditSize">0</div></div>
<div id="deckCardListEdit">
</div>
</section>

@ -0,0 +1 @@
Subproject commit e30fa878adfb669745aa52e4bb6de8f31fbd1828

View File

@ -31,9 +31,33 @@ class CheckButtonGroup<TValue> {
}
replace(index: number, button: CheckButton, value: TValue) {
const existingChild = this.entries[index].button.buttonElement;
this.entries[index] = { button, value };
this.setupButton(button, value);
// The caller is responsible for adding the button to the DOM.
this.parentElement?.insertBefore(button.buttonElement, existingChild);
this.parentElement?.removeChild(existingChild);
}
insert(index: number, button: CheckButton, value: TValue) {
const existingChild = this.entries[index].button.buttonElement;
this.entries.splice(index, 0, { button, value });
this.setupButton(button, value);
this.parentElement?.insertBefore(button.buttonElement, existingChild);
}
removeAt(index: number) {
const existingChild = this.entries[index].button.buttonElement;
this.entries.splice(index, 1)
this.parentElement?.removeChild(existingChild);
}
move(index: number, newIndex: number | null) {
const entry = this.entries[index];
this.entries.splice(index, 1)
const existingChild = newIndex == null || newIndex >= this.entries.length ? null : this.entries[newIndex].button.buttonElement;
if (newIndex == null) this.entries.push(entry);
else this.entries.splice(newIndex, 0, entry);
this.parentElement?.insertBefore(entry.button.buttonElement, existingChild);
}
clear() {

View File

@ -7,3 +7,4 @@ interface Config {
}
declare var config: Config;
declare var polyfillActive: boolean;

View File

@ -17,7 +17,7 @@ const testStageSelectionDialog = document.getElementById('testStageSelectionDial
const deckEditCardButtons = new CheckButtonGroup<number>(deckCardListEdit);
let selectedDeckCardIndex: number | null = null;
let draggingCardButton: Element | null = null;
function deckEditInitCardDatabase(cards: Card[]) {
for (const card of cards) {
@ -32,8 +32,9 @@ function deckEditInitCardDatabase(cards: Card[]) {
button2.checked = false;
}
if (selectedDeckCardIndex == null) return;
const oldEntry = deckEditCardButtons.entries[selectedDeckCardIndex];
const index = deckEditCardButtons.entries.findIndex(el => el.button.checked);
if (index < 0) return;
const oldEntry = deckEditCardButtons.entries[index];
const oldCardNumber = oldEntry.value;
if (oldCardNumber != 0)
@ -43,11 +44,7 @@ function deckEditInitCardDatabase(cards: Card[]) {
const button3 = createDeckEditCardButton(card.number);
button3.checked = true;
const oldElement = oldEntry.button.buttonElement;
deckCardListEdit.insertBefore(button3.buttonElement, oldElement);
deckCardListEdit.removeChild(oldElement);
deckEditCardButtons.replace(selectedDeckCardIndex, button3, card.number);
deckEditCardButtons.replace(index, button3, card.number);
deckEditUpdateSize();
cardList.listElement.parentElement!.classList.remove('selecting');
@ -94,7 +91,6 @@ function editDeck() {
deckNameLabel2.innerText = selectedDeck.name;
deckEditCardButtons.clear();
selectedDeckCardIndex = null;
for (let i = 0; i < 15; i++) {
if (selectedDeck.cards[i]) {
@ -117,27 +113,88 @@ function editDeck() {
function createDeckEditCardButton(cardNumber: number) {
const button = new CardButton(cardDatabase.get(cardNumber));
button.buttonElement.draggable = true;
button.buttonElement.addEventListener('click', () => {
selectedDeckCardIndex = deckEditCardButtons.entries.findIndex(e => e.button == button);
for (const button2 of cardList.cardButtons) {
button2.checked = button2.card.number == cardNumber;
}
cardList.listElement.parentElement!.classList.add('selecting');
});
button.buttonElement.addEventListener('dragstart', e => {
if (e.dataTransfer == null) return;
const index = deckEditCardButtons.entries.findIndex(el => el.button.buttonElement == e.currentTarget);
draggingCardButton = button.buttonElement;
e.dataTransfer.effectAllowed = 'move';
e.dataTransfer.setData('application/tableturf-card-index', index.toString());
button.buttonElement.classList.add('dragging');
});
button.buttonElement.addEventListener('dragend', e => {
button.buttonElement.classList.remove('dragging');
if (draggingCardButton != null && e.currentTarget == draggingCardButton) {
const index = deckEditCardButtons.entries.findIndex(el => el.button.buttonElement == e.currentTarget) + 1;
deckCardListEdit.insertBefore(draggingCardButton, index >= deckEditCardButtons.entries.length ? null : deckEditCardButtons.entries[index].button.buttonElement);
draggingCardButton = null;
}
});
button.buttonElement.addEventListener('dragenter', e => e.preventDefault());
button.buttonElement.addEventListener('dragover', deckEditCardButton_dragover);
button.buttonElement.addEventListener('drop', deckEditCardButton_drop);
return button;
}
function deckEditCardButton_dragover(e: DragEvent) {
e.preventDefault();
if (e.dataTransfer == null) return;
const indexString = e.dataTransfer.getData('application/tableturf-card-index');
if (indexString != '' && draggingCardButton != null) {
e.dataTransfer.dropEffect = 'move';
if (e.currentTarget != draggingCardButton && e.currentTarget != deckCardListEdit) {
// Move the card being dragged into the new position as a preview.
for (let el = draggingCardButton.nextElementSibling; el != null; el = el.nextElementSibling) {
if (el == e.currentTarget) {
deckCardListEdit.insertBefore(draggingCardButton, el.nextElementSibling);
return;
}
}
deckCardListEdit.insertBefore(draggingCardButton, e.currentTarget as Node);
}
} else if (e.dataTransfer.getData('text/plain'))
e.dataTransfer.dropEffect = 'copy';
}
function deckEditCardButton_drop(e: DragEvent) {
e.preventDefault();
if (e.dataTransfer == null) return;
const indexString = e.dataTransfer.getData('application/tableturf-card-index');
if (indexString) {
const index = parseInt(indexString);
let newIndex = 0;
for (let el = deckCardListEdit.firstElementChild; el != null; el = el.nextElementSibling) {
if (el == draggingCardButton) break;
newIndex++;
}
if (newIndex == index) return;
console.log(`Moving card ${index} to ${newIndex}.`);
deckEditCardButtons.move(index, newIndex);
draggingCardButton = null;
}
}
function createDeckEditEmptySlotButton() {
const buttonElement = document.createElement('button');
const button = new CheckButton(buttonElement);
buttonElement.type = 'button';
buttonElement.className = 'card emptySlot';
buttonElement.addEventListener('click', () => {
selectedDeckCardIndex = deckEditCardButtons.entries.findIndex(e => e.button == button);
for (const button2 of cardList.cardButtons)
button2.checked = false;
cardList.listElement.parentElement!.classList.add('selecting');
});
buttonElement.addEventListener('dragenter', e => e.preventDefault());
buttonElement.addEventListener('dragover', deckEditCardButton_dragover);
buttonElement.addEventListener('drop', deckEditCardButton_drop);
return button;
}

View File

@ -1,3 +1,4 @@
const deckListPage = document.getElementById('deckListPage')!;
const deckListBackButton = document.getElementById('deckListBackButton') as HTMLLinkElement;
const deckViewBackButton = document.getElementById('deckViewBackButton') as HTMLLinkElement;
const deckEditorDeckViewSection = document.getElementById('deckEditorDeckViewSection')!;
@ -38,7 +39,10 @@ const deckImportFileBox = document.getElementById('deckImportFileBox') as HTMLIn
const deckImportErrorBox = document.getElementById('deckImportErrorBox')!;
const deckImportOkButton = document.getElementById('deckImportOkButton') as HTMLButtonElement;
const deckButtons = new CheckButtonGroup<Deck>();
const deckButtons = new CheckButtonGroup<Deck>(deckList);
let deckListTouchMode = false;
let draggingDeckButton: Element | null = null;
function showDeckList() {
showPage('deckList');
@ -46,6 +50,18 @@ function showDeckList() {
deckButtons.deselect();
}
deckList.addEventListener('touchstart', deckListEnableTouchMode);
function deckListEnableTouchMode() {
if (deckListTouchMode) return;
deckListTouchMode = true;
deckListPage.classList.add('touchmode');
for (var b of deckButtons.buttons) {
b.buttonElement.draggable = false;
(b.buttonElement.getElementsByClassName('handle')[0] as HTMLElement).draggable = true;
}
}
deckListBackButton.addEventListener('click', e => {
e.preventDefault();
showPage('preGame');
@ -65,7 +81,7 @@ deckViewBackButton.addEventListener('click', e => {
e.preventDefault();
clearChildren(deckCardListView);
deselectDeck();
deckEditorDeckViewSection.hidden = true;
deckListPage.classList.remove('showingDeck');
});
function saveDecks() {
@ -90,25 +106,99 @@ function saveDecks() {
}
for (let i = 0; i < decks.length; i++) {
createDeckButton(i, decks[i]);
createDeckButton(decks[i]);
}
}
function createDeckButton(index: number, deck: Deck) {
function createDeckButton(deck: Deck) {
const buttonElement = document.createElement('button');
buttonElement.type = 'button';
buttonElement.draggable = true;
const button = new CheckButton(buttonElement);
deckButtons.add(button, deck);
buttonElement.addEventListener('click', () => {
selectedDeck = deckButtons.value;
selectDeck();
});
buttonElement.innerText = deck.name;
buttonElement.addEventListener('dragstart', e => {
if (e.dataTransfer == null) return;
const index = decks.indexOf(deck);
draggingDeckButton = buttonElement;
e.dataTransfer.effectAllowed = 'copyMove';
e.dataTransfer.setData('text/plain', JSON.stringify(deck, [ 'name', 'cards' ]));
e.dataTransfer.setData('application/tableturf-deck-index', index.toString());
buttonElement.classList.add('dragging');
});
buttonElement.addEventListener('dragend', e => {
buttonElement.classList.remove('dragging');
if (draggingDeckButton != null && e.currentTarget == draggingDeckButton) {
const index = deckButtons.entries.findIndex(el => el.button.buttonElement == e.currentTarget) + 1;
deckList.insertBefore(draggingDeckButton, index >= deckButtons.entries.length ? null : deckButtons.entries[index].button.buttonElement);
draggingDeckButton = null;
}
});
buttonElement.addEventListener('dragenter', e => e.preventDefault());
buttonElement.addEventListener('dragover', deckButton_dragover);
buttonElement.addEventListener('drop', deckButton_drop);
const handle = document.createElement('div');
handle.className = 'handle';
buttonElement.appendChild(handle);
buttonElement.appendChild(document.createTextNode(deck.name));
deckList.insertBefore(buttonElement, addDeckControls);
return button;
}
function deckButton_dragover(e: DragEvent) {
e.preventDefault();
if (e.dataTransfer == null) return;
const indexString = e.dataTransfer.getData('application/tableturf-deck-index');
if (indexString != '' && draggingDeckButton != null) {
e.dataTransfer.dropEffect = 'move';
if (e.currentTarget != draggingDeckButton && e.currentTarget != deckList) {
// Move the deck being dragged into the new position as a preview.
for (let el = draggingDeckButton.nextElementSibling; el != null; el = el.nextElementSibling) {
if (el == e.currentTarget) {
deckList.insertBefore(draggingDeckButton, el.nextElementSibling);
return;
}
}
deckList.insertBefore(draggingDeckButton, e.currentTarget as Node);
}
} else if (e.dataTransfer.getData('text/plain'))
e.dataTransfer.dropEffect = 'copy';
}
function deckButton_drop(e: DragEvent) {
e.preventDefault();
if (e.dataTransfer == null) return;
const indexString = e.dataTransfer.getData('application/tableturf-deck-index');
if (indexString) {
const index = parseInt(indexString);
let newIndex = 0;
for (let el = deckList.firstElementChild; el != null; el = el.nextElementSibling) {
if (el == draggingDeckButton) break;
newIndex++;
}
if (newIndex == index) return;
console.log(`Moving deck ${index} to ${newIndex}.`);
deckButtons.move(index, newIndex);
const deck = decks[index];
decks.splice(index, 1);
decks.splice(newIndex, 0, deck);
saveDecks();
draggingDeckButton = null;
return;
}
const text = e.dataTransfer.getData('text/plain');
if (text) {
const data = JSON.parse(text);
const decks = (data instanceof Array ? data : [ data ]) as Deck[];
importDecks(decks);
}
}
function importDecks(decksToImport: (Deck | number[])[]) {
let newSelectedDeck: Deck | null = null;
for (const el of decksToImport) {
@ -119,7 +209,7 @@ function importDecks(decksToImport: (Deck | number[])[]) {
deck = el;
if (!deck.name) deck.name = `Imported Deck ${decks.length + 1}`;
}
createDeckButton(decks.length, deck);
createDeckButton(deck);
decks.push(deck);
newSelectedDeck ??= deck;
}
@ -134,7 +224,7 @@ function importDecks(decksToImport: (Deck | number[])[]) {
newDeckButton.addEventListener('click', () => {
selectedDeck = new Deck(`Deck ${decks.length + 1}`, [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ], false);
createDeckButton(decks.length, selectedDeck);
createDeckButton(selectedDeck);
decks.push(selectedDeck);
editDeck();
});
@ -196,20 +286,29 @@ function selectDeck() {
}
}
deckListTestButton.disabled = false;
deckExportButton.disabled = false;
deckCopyButton.disabled = false;
deckEditButton.disabled = selectedDeck.isReadOnly;
deckRenameButton.disabled = selectedDeck.isReadOnly;
deckCopyButton.disabled = false;
deckDeleteButton.disabled = selectedDeck.isReadOnly;
deckViewSize.innerText = size.toString();
deckEditorDeckViewSection.hidden = false;
deckListPage.classList.add('showingDeck');
}
function deselectDeck() {
selectedDeck = null;
deckButtons.deselect();
clearChildren(deckCardListView);
deckNameLabel.innerText = '\u00a0';
deckViewSize.innerText = '0';
deckListTestButton.disabled = true;
deckExportButton.disabled = true;
deckCopyButton.disabled = true;
deckEditButton.disabled = true;
deckRenameButton.disabled = true;
deckCopyButton.disabled = true;
deckDeleteButton.disabled = true;
deckListPage.classList.remove('showingDeck');
}
deckExportButton.addEventListener('click', () => {
@ -244,19 +343,10 @@ deckDeleteButton.addEventListener('click', () => {
if (selectedDeck == null) return;
if (!confirm(`Are you sure you want to delete ${selectedDeck.name}?`)) return;
for (const el of deckButtons.entries) {
if (el.value == selectedDeck) {
deckList.removeChild(el.button.buttonElement);
break;
}
}
const index = decks.indexOf(selectedDeck);
if (index >= 0) decks.splice(index, 1);
deckButtons.entries.splice(index, 1);
deckButtons.deselect();
selectedDeck = null;
deckEditorDeckViewSection.hidden = true;
deckButtons.removeAt(index);
deselectDeck();
saveDecks();
});

View File

@ -195,6 +195,8 @@ document.getElementById('preGameBackButton')!.addEventListener('click', e => {
backPreGameForm(true);
})
preGameDeckEditorButton.addEventListener('touchstart', deckListEnableTouchMode);
preGameDeckEditorButton.addEventListener('click', e => {
e.preventDefault();
showDeckList();

View File

@ -1328,11 +1328,16 @@ dialog::backdrop {
:is(#deckListPage, #deckEditPage):not([hidden]) {
height: 100vh;
display: grid;
grid-template-columns: auto 1fr;
display: flex;
justify-content: center;
}
.deckList button {
.deckList {
display: flex;
flex-flow: column;
}
.deckList button, #addDeckControls button {
display: flex;
justify-content: center;
align-items: center;
@ -1341,12 +1346,15 @@ dialog::backdrop {
border: inherit;
text-shadow: 0 0 4px black;
}
.deckList > * {
button.dragging {
opacity: 0.5;
}
.deckList > button, #addDeckControls {
width: 20rem;
height: 3rem;
margin: 0.5em;
}
.deckList > button { background: var(--primary-colour-2); }
.deckList > button { background: var(--primary-colour-2); position: relative;}
.deckList > button:hover { background: var(--special-colour-2); }
.deckList > button:focus-within { outline: 2px solid var(--special-accent-colour-2); }
.deckList > button:is(:active, .checked) { background: var(--special-accent-colour-2); }
@ -1357,6 +1365,7 @@ dialog::backdrop {
gap: 1em;
}
#addDeckControls > button {
flex-basis: 8em;
flex-grow: 1;
background: black;
border: 0.25em solid var(--primary-colour-2);
@ -1365,13 +1374,22 @@ dialog::backdrop {
#addDeckControls > button:focus-within { outline: 2px solid var(--special-accent-colour-2); }
#addDeckControls > button:is(:active, .checked) { border-color: var(--special-accent-colour-2); }
.deckList {
display: flex;
flex-flow: column;
.deckList .handle {
position: absolute;
background: url('assets/grip-horizontal.svg') 0.5em center/1em no-repeat;
left: 0;
right: 0;
top: 0;
bottom: 0;
}
.touchmode .handle {
width: 2em;
}
.card.emptySlot {
--colour: dimgrey;
background: url('assets/plus-circle.svg') center/2em no-repeat, black;
}
#deckCardListView, #deckCardListEdit {
@ -1379,14 +1397,15 @@ dialog::backdrop {
grid-template-columns: auto auto auto;
justify-items: center;
grid-column: 2;
grid-row: 1 / 5;
grid-row: 2 / -1;
overflow-y: scroll;
width: 33em;
}
#deckEditorDeckViewSection:not([hidden]), #deckEditordeckEditPage {
#deckEditorDeckViewSection:not([hidden]), #deckEditorDeckEditPage {
display: grid;
grid-template-columns: minmax(min-content, auto) min-content;
grid-template-rows: auto auto auto 1fr;
grid-template-columns: 1fr auto;
grid-template-rows: auto auto 1fr;
height: 100vh;
}
@ -1395,21 +1414,35 @@ dialog::backdrop {
}
#deckName, #deckName2 {
grid-column: 1;
grid-column: 1 / -1;
grid-row: 1;
}
:is(#deckEditorDeckViewSection, #deckEditordeckEditPage) > h3 + div {
grid-row: 2;
#deckEditToolbar, #deckListToolbar {
grid-column: 1;
grid-row: 3;
display: flex;
flex-flow: column;
}
:is(#deckListToolbar, #deckEditToolbar) button {
margin: 0.25em;
width: 5em;
height: 4em;
}
.deckSizeContainer {
grid-column: 1;
grid-row: 3;
grid-row: 2;
background: dimgrey;
text-align: center;
line-height: 1.5em;
margin: 0.25em;
min-width: 4em;
}
#deckEditorCardListSection {
flex-grow: 1;
background: #00000080;
height: 100vh;
display: grid;
grid-template-rows: auto 1fr;
@ -1684,16 +1717,21 @@ dialog::backdrop {
:is(#deckListPage, #deckEditPage) section {
position: absolute;
background: black;
left: 0;
right: 0;
top: 0;
}
#deckListPage:not(.showingDeck) #deckEditorDeckViewSection {
display: none;
}
#deckListPage.showingDeck #deckEditorDeckListSection {
display: none;
}
#deckEditorDeckViewSection:not([hidden]) {
z-index: 10;
grid-template-columns: auto 1fr 1fr;
grid-template-rows: auto auto 1fr;
grid-template-columns: auto 1fr auto;
}
#deckViewBackButton {
@ -1709,16 +1747,23 @@ dialog::backdrop {
}
#deckName {
grid-column: 2;
grid-column: 2 / -1;
}
:is(#deckEditorDeckViewSection, #deckEditordeckEditPage) > h3 + div {
grid-row: 1;
grid-column: -2;
#deckListToolbar, #deckEditToolbar {
grid-row: 2;
grid-column: 1 / -2;
flex-flow: row;
}
:is(#deckListToolbar, #deckEditToolbar) button {
width: auto;
flex-grow: 1;
flex-basis: 4em;
}
.deckSizeContainer {
grid-column: 1 / -1;
grid-column: -2;
grid-row: 2;
justify-self: end;
}
@ -1729,6 +1774,7 @@ dialog::backdrop {
overflow-y: scroll;
font-size: 80%;
grid-template-columns: 1fr 1fr 1fr;
width: 100%;
}
#deckEditorCardListSection:not(.selecting) {