diff --git a/app/assets/javascripts/stories.js b/app/assets/javascripts/stories.js index 5e7da51b..f06b95e6 100644 --- a/app/assets/javascripts/stories.js +++ b/app/assets/javascripts/stories.js @@ -25,8 +25,57 @@ document.addEventListener("DOMContentLoaded", () => { debounceTimer = window.setTimeout(updateMarkdown, 300); }); }); + + const form = document.querySelector(".edit_story"); + const backButton = document.getElementById("back"); + const logo = document.getElementById("logo"); + let isDirty = false; + + if (form) { + // Mark the form as dirty when any input changes + form.addEventListener("input", function () { + isDirty = true; + addBeforeUnloadEventListener(isDirty); + }); + + // Reset isDirty on form submission + form.addEventListener("submit", function () { + isDirty = false; + addBeforeUnloadEventListener(isDirty); + }); + } + + // Attach a click event to the custom back button + [backButton, logo].forEach(element => { + element.addEventListener("click", function (event) { + if (isDirty) { + const confirmLeave = confirm("You have unsaved changes. Are you sure you want to go back?"); + if (!confirmLeave) { + // Prevent navigation if the user chooses not to leave + event.preventDefault(); + } else { + // Optionally, reset isDirty if leaving + isDirty = false; + addBeforeUnloadEventListener(isDirty) + } + } + }) + }); }); +function addBeforeUnloadEventListener(isDirty) { + if (isDirty) { + window.addEventListener("beforeunload", warnUserifUnsavedEdits); + } else { + window.removeEventListener("beforeunload", warnUserifUnsavedEdits); + } +} + +function warnUserifUnsavedEdits(event) { + event.preventDefault(); + event.returnValue = ''; +} + function updateStatusButton(color, status) { const button = document.querySelector(".story-title .dropdown-wrapper > button"); button.className = `button ${color}`; diff --git a/spec/features/stories_manage_spec.rb b/spec/features/stories_manage_spec.rb index 5048201f..22cf0c98 100644 --- a/spec/features/stories_manage_spec.rb +++ b/spec/features/stories_manage_spec.rb @@ -68,6 +68,17 @@ expect(page).to have_content "Story updated!" end + it "alerts me when I try to navigate away from the page without saving my edits", js: true do + visit project_path(id: project.id) + click_button "More actions" + click_link "Edit" + fill_in "story[title]", with: "As a user, I want to edit stories" + click_link "Back" + alert_text = page.driver.browser.switch_to.alert.text + + expect(alert_text).to eq "You have unsaved changes. Are you sure you want to go back?" + end + it "allows me to delete a story" do visit project_path(id: project.id)