Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improve snapping behavior and add undo in EditableTemplate #6687

Merged
merged 6 commits into from
Apr 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Binary file added doc/_static/images/builder_undo.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
6 changes: 6 additions & 0 deletions doc/how_to/notebook/layout_builder.md
Original file line number Diff line number Diff line change
Expand Up @@ -114,6 +114,12 @@ You might want to reorder the cards for a more logical flow. To do this, drag th

![Builder Rearrange View](../../_static/images/builder_rearrange.png)

### Undo

You might want to undo previous remove, resize and reorder actions:

![Builder Rearrange View](../../_static/images/builder_undo.png)

### Resetting Layout

If you're not satisfied with the changes, you can start over by clicking the reset button at the top right.
Expand Down
12 changes: 8 additions & 4 deletions examples/reference/templates/EditableTemplate.ipynb
Original file line number Diff line number Diff line change
Expand Up @@ -128,15 +128,19 @@
"\n",
"<img src=\"\"></img>\n",
"\n",
"In edit mode we additionally have reset and save icons in the header:\n",
"In edit mode we additionally have undo, reset and save icons in the header:\n",
"\n",
"- Save: Persists the current layout into the browser storage overriding the layout persisted on the server.\n",
"- Undo: Undo previous delete, resize and re-arrange actions.\n",
"\n",
"<img src=\"\"></img>\n",
"<img src=\"\" width=\"24\" height=\"24\"></img>\n",
"\n",
"- Reset: Resets the arrangement and order of the individual components to the default.\n",
"\n",
"<img src=\"\" width=\"24\" height=\"24\"></img>"
"<img src=\"\" width=\"24\" height=\"24\"></img>\n",
"\n",
"- Save: Persists the current layout into the browser storage overriding the layout persisted on the server.\n",
"\n",
"<img src=\"\"></img>\n"
]
}
],
Expand Down
2 changes: 1 addition & 1 deletion panel/io/handlers.py
Original file line number Diff line number Diff line change
Expand Up @@ -568,7 +568,7 @@ def _render_template(self, doc, path):
reset = 'reset' in state.session_args
if not (editable or persist):
state.template.editable = False
state.template.title = os.path.splitext(os.path.basename(path))[0].title()
state.template.title = os.path.splitext(os.path.basename(path))[0].replace('_', ' ').title()

layouts, outputs, cells = {}, {}, {}
for cell_id, objects in state._cell_outputs.items():
Expand Down
15 changes: 11 additions & 4 deletions panel/template/editable/editable.css
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@
background-color: currentColor;
cursor: pointer;
mask-size: cover;
height: 18px;
width: 18px;
height: 20px;
width: 20px;
margin-left: 10px;
-webkit-mask-size: cover;
}
Expand All @@ -22,8 +22,15 @@
}

#grid-reset {
mask-image: url('');
-webkit-mask-image: url('');
mask-image: url('');
-webkit-mask-image: url('');
}

#grid-undo {
mask-image: url('');
-webkit-mask-image: url('');
height: 25px;
width: 25px;
}

#grid-save {
Expand Down
67 changes: 64 additions & 3 deletions panel/template/editable/editable.html
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
{% block header_end %}
{% if editable %}
<div id="header-indicators">
<div id="grid-undo" class="header-icon disabled-button" title="Undo the last action"></div>
<div id="grid-reset" class="header-icon" title="Reset the Layout"></div>
{% if local_save %}
<div id="grid-save" class="header-icon disabled-button" title="Save the Layout to Local Storage"></div>
Expand Down Expand Up @@ -170,7 +171,21 @@
dragEnabled: {{ editable|json }},
dragHandle: '.muuri-handle.drag',
layout: {fillGaps: true}
});
}).on('dragInit', function () {
undo_stack.push({action: "drag", items: grid.getItems()})
}).on('dragEnd', function () {
const last_event = undo_stack.pop()
if (last_event.action == "move") {
undo_stack.push(last_event)
}
}).on('move', function () {
const last_event = undo_stack.pop()
if (last_event.action === "drag") {
const event = {...last_event, action: "move"}
undo_stack.push(event)
document.getElementById('grid-undo').classList.remove('disabled-button')
}
})
for (const spec of layout) {
const el = document.querySelector(`[data-id="${spec.id}"]`);
if (el && spec.width && spec.height) {
Expand All @@ -196,6 +211,28 @@

// Set up reset, hide and resize actions
{% if editable %}
const undo_stack = []
const undo_button = document.getElementById("grid-undo")
undo_button.addEventListener("click", function() {
const action = undo_stack.pop()
if (!action) {
return
}
if (undo_stack.length === 0) {
undo_button.classList.add('disabled-button')
}
switch(action.action) {
case "hide":
grid.show([action.item], {instant: true});
case "resize":
resize_item(action.item.getElement(), action.width, action.height, false);
grid.refreshItems()
grid.layout()
case "move":
grid.sort(action.items)
}
})

const reset_button = document.getElementById("grid-reset")
reset_button.addEventListener("click", function() {
if (window.localStorage) {
Expand All @@ -209,7 +246,10 @@

for (const handle of document.querySelectorAll('.muuri-handle.delete')) {
handle.addEventListener("click", (event) => {
grid.hide([grid.getItem(event.target.parentElement)], {instant: true});
const item = grid.getItem(event.target.parentElement);
grid.hide([item], {instant: true});
undo_stack.push({action: "hide", item: item})
undo_button.classList.remove('disabled-button')
{% if local_save %}
document.getElementById('grid-save').classList.remove('disabled-button')
{% endif %}
Expand Down Expand Up @@ -243,6 +283,24 @@
// resize from all edges and corners
edges: { right: '.muuri-handle.resize', bottom: '.muuri-handle.resize' },
listeners: {
start (event) {
const el = event.target
const item = grid.getItem(el);
let height = el.style.height.slice(null, -2);
if (!height) {
const {top} = item.getMargin();
height = item.getHeight()-top;
} else {
height = parseFloat(height);
}
let width;
if (el.style.width.length) {
width = parseFloat(el.style.width.split('(')[1].split('%')[0]);
} else {
width = 100;
}
undo_stack.push({action: "resize", item, width, height})
},
move (event) {
const item = grid.getItem(event.target);
const {top, bottom} = item.getMargin();
Expand All @@ -253,12 +311,14 @@
resize_item(event.target, width, height, false);
grid.refreshItems()
grid.layout()
window.dispatchEvent(new Event('resize'));
},
end (event) {
event.target.style.removeProperty('z-index')
grid.refreshItems()
grid.layout()
window.dispatchEvent(new Event('resize'));
undo_button.classList.remove('disabled-button')
{% if local_save %}
save_button.classList.remove('disabled-button')
{% endif %}
Expand All @@ -276,13 +336,14 @@
const target = {range: 25}
const grid_bbox = grid.getElement().getBoundingClientRect()
for (const item of grid.getItems()) {
if (item === interaction.element) {
if ((item.getElement() === interaction.element) && item.isVisible()) {
continue
}
const {top, left} = item.getPosition();
const margin = item.getMargin()
const bottom = (top + item.getHeight()) + margin.top + margin.bottom;
const right = (left + item.getWidth()) + margin.left + margin.right;

if (Math.abs(right - x) < target.range) {
target.x = right + margin.right
}
Expand Down
57 changes: 57 additions & 0 deletions panel/tests/ui/template/test_editabletemplate.py
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,24 @@ def test_editable_template_delete_item(page):
wait_until(lambda: tmpl.layout.get(id(md2), {}).get('visible') == False, page)


def test_editable_template_undo_delete_item(page):
tmpl = EditableTemplate()
md1 = Markdown('1')
md2 = Markdown('2')

tmpl.main[:] = [md1, md2]

serve_component(page, tmpl)

page.locator(".muuri-handle.delete").nth(1).click()

wait_until(lambda: tmpl.layout.get(id(md2), {}).get('visible') == False, page)

page.locator('#grid-undo').click()

wait_until(lambda: tmpl.layout.get(id(md2), {}).get('visible') == True, page)


def test_editable_template_drag_item(page):
tmpl = EditableTemplate()
md1 = Markdown('1')
Expand All @@ -160,6 +178,24 @@ def test_editable_template_drag_item(page):

wait_until(lambda: list(tmpl.layout) == [id(md2), id(md1)], page)

def test_editable_template_undo_drag_item(page):
tmpl = EditableTemplate()
md1 = Markdown('1')
md2 = Markdown('2')

tmpl.main[:] = [md1, md2]

serve_component(page, tmpl)

md2_handle = page.locator(".muuri-handle.drag").nth(1)

md2_handle.drag_to(md2_handle, target_position={'x': 0, 'y': -50}, force=True)

wait_until(lambda: list(tmpl.layout) == [id(md2), id(md1)], page)

page.locator('#grid-undo').click()

wait_until(lambda: list(tmpl.layout) == [id(md1), id(md2)], page)

def test_editable_template_resize_item(page):
md1 = Markdown('1')
Expand All @@ -177,3 +213,24 @@ def test_editable_template_resize_item(page):
md2_handle.drag_to(md2_handle, target_position={'x': -50, 'y': -30}, force=True)

wait_until(lambda: tmpl.layout.get(id(md2), {}).get('width') < 45, page)

def test_editable_template_undo_resize_item(page):
md1 = Markdown('1')
md2 = Markdown('2')

tmpl = EditableTemplate(layout={id(md2): {'width': 50, 'height': 80}})

tmpl.main[:] = [md1, md2]

serve_component(page, tmpl)

md2_handle = page.locator(".muuri-handle.resize").nth(1)

md2_handle.hover()
md2_handle.drag_to(md2_handle, target_position={'x': -50, 'y': -30}, force=True)

wait_until(lambda: tmpl.layout.get(id(md2), {}).get('width') < 45, page)

page.locator('#grid-undo').click()

wait_until(lambda: tmpl.layout.get(id(md2), {}).get('width') == 50, page)
Loading