srecord noeq JTextFileEditor(File file) is Swingable { settable bool autoSave; JTextArea textArea; JButton btnReload, btnSave; JLabel lblChanged; volatile S fileContents, savedText; FileWatchService fileWatcher; Q q = startQ(); RSTOverQ rstLoad = rstWithPreDelay(1.0, q, r _load); RSTOverQ rstSave; cachedVisualize { assertNotNull(file); rstSave = rstWithPreDelay(autoSave ? 1.0 : 0.0, q, r _save); savedText = fileContents = loadTextFile(file); textArea = wordWrapTypeWriterTextArea(savedText); onChange(textArea, -> { updateButtons(); if (autoSave) rstSave!; }); bindToComponent(textArea, -> { fileWatcher = new FileWatchService; fileWatcher.addNonRecursiveListener(dirOfFile(file), f -> { if (sameFile(f, file)) rstLoad!; }); rstLoad!; }, -> { dispose fileWatcher; }); ret northAndCenterWithMargin( centerAndEastWithMargin( swing(-> JFilePathLabel(file).visualize()), jline( autoSave ? (lblChanged = jlabel()) : (btnSave = disableButton(jbutton("Save", rstSave))), btnReload = jbutton("Reload", r reload) )), textArea ); } void updateButtons { bool changed = !eq(savedText, getText(textArea)); setEnabled(btnSave, changed); setEnabled(btnReload, !eq(fileContents, savedText)); setText(lblChanged, changed ? "*" : ""); } void _save { saveTextFile(file, savedText = getText(textArea)); } void reload { savedText = fileContents; setText(textArea, savedText); updateButtons(); } void _load { print("Loading " + file); fileContents = loadTextFile(file); updateButtons(); } }