x
1
2
3
4
5
6
7
8
9
10
11
<dialog class="dialog" data-controller="turbo-confirm" data-action="close->turbo-confirm#submit" role="alertdialog" aria-labelledby="turbo_confirm_title" aria-describedby="turbo_confirm_text"> <form method="dialog" class="dialog__content"> <h2 class="text-lg font-semibold leading-none" id="turbo_confirm_title" data-turbo-confirm-target="title"></h2> <p class="text-sm text-subtle mbs-2" id="turbo_confirm_text" data-turbo-confirm-target="text"></p> <div class="flex items-center justify-end gap mbs-4"> <button value="cancel" class="btn">Cancel</button> <button value="confirm" class="btn btn--primary">Continue</button> </div> </form></dialog><form data-turbo-confirm="Are you absolutely sure?" data-turbo-confirm-text="This action cannot be undone. This will permanently delete your account and remove your data from our servers." class="button_to" method="post" action="/lookbook/home/index"><input type="hidden" name="_method" value="delete" autocomplete="off" /><button class="btn" type="submit">Click to destroy</button><input type="hidden" name="authenticity_token" value="bOUhAHITydFZyFy2OpQy59dGNOpEhXdB_qKk_XR1to3ni7sAhDEx3uhwMwXIzSHZN8nJZkYkcgbkKUed1uO3sQ" autocomplete="off" /></form>
1
2
3
4
5
6
7
8
9
10
11
12
13
<dialog class="dialog" data-controller="turbo-confirm" data-action="close->turbo-confirm#submit" role="alertdialog" aria-labelledby="turbo_confirm_title" aria-describedby="turbo_confirm_text"> <form method="dialog" class="dialog__content"> <h2 class="text-lg font-semibold leading-none" id="turbo_confirm_title" data-turbo-confirm-target="title"></h2> <p class="text-sm text-subtle mbs-2" id="turbo_confirm_text" data-turbo-confirm-target="text"></p> <div class="flex items-center justify-end gap mbs-4"> <button value="cancel" class="btn">Cancel</button> <button value="confirm" class="btn btn--primary">Continue</button> </div> </form></dialog><%= button_to "Click to destroy", nil, class: "btn", method: :delete, form: { data: { turbo_confirm: "Are you absolutely sure?", turbo_confirm_text: "This action cannot be undone. This will permanently delete your account and remove your data from our servers." } } %>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
.dialog { background-color: var(--color-bg); border-radius: var(--rounded-lg); border-width: var(--border); box-shadow: var(--shadow-lg); color: var(--color-text); inline-size: var(--size-full); margin: auto; max-inline-size: var(--dialog-size, var(--max-i-lg)); &::backdrop { background-color: rgba(0, 0, 0, .8); } /* Final state of exit animation and setup */ opacity: 0; transform: var(--scale-95); transition-behavior: allow-discrete; transition-duration: var(--time-200); transition-property: display, overlay, opacity, transform; &::backdrop { opacity: 0; transition-behavior: allow-discrete; transition-duration: var(--time-200); transition-property: display, overlay, opacity; } /* Final state of entry animation */ &[open] { opacity: 1; transform: var(--scale-100); } &[open]::backdrop { opacity: 1; } /* Initial state of entry animation */ @starting-style { &[open] { opacity: 0; transform: var(--scale-95); } &[open]::backdrop { opacity: 0; } } /* Drawer component on mobile */ @media (width < 40rem) { border-end-end-radius: 0; border-end-start-radius: 0; margin-block-end: 0; max-inline-size: none; }}.dialog__content { padding: var(--size-6);}.dialog__close { inset-block-start: var(--size-3); inset-inline-end: var(--size-3); position: absolute;}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import { Controller } from "@hotwired/stimulus"export default class extends Controller { static targets = [ "title", "text" ] static values = { defaultConfirmText: { type: String, default: "This action cannot be undone and may have permanent consequences." } } #confirmCallback = null connect() { Turbo.config.forms.confirm = this.#confirm.bind(this) } disconnect() { Turbo.config.forms.confirm = null } submit({ target }) { this.#confirmCallback?.(target.returnValue === "confirm") } #confirm(message, target) { this.titleTarget.textContent = message this.textTarget.textContent = target.dataset.turboConfirmText || this.defaultConfirmTextValue this.element.showModal() return new Promise(resolve => (this.#confirmCallback = resolve)) }}
You must have only one turbo confirm dialog per page.