diff --git a/frontend/app/element/markdown.tsx b/frontend/app/element/markdown.tsx
index 59925fdeb..44b0e280e 100644
--- a/frontend/app/element/markdown.tsx
+++ b/frontend/app/element/markdown.tsx
@@ -178,7 +178,7 @@ type MarkdownProps = {
 };
 
 const Markdown = ({ text, textAtom, showTocAtom, style, className, resolveOpts, onClickExecute }: MarkdownProps) => {
-    const textAtomValue = useAtomValueSafe(textAtom);
+    const textAtomValue = useAtomValueSafe<string>(textAtom);
     const tocRef = useRef<TocItem[]>([]);
     const showToc = useAtomValueSafe(showTocAtom) ?? false;
     const contentsOsRef = useRef<OverlayScrollbarsComponentRef>(null);
diff --git a/frontend/app/view/codeeditor/codeeditor.tsx b/frontend/app/view/codeeditor/codeeditor.tsx
index dc544678d..ae9d56b98 100644
--- a/frontend/app/view/codeeditor/codeeditor.tsx
+++ b/frontend/app/view/codeeditor/codeeditor.tsx
@@ -2,11 +2,12 @@
 // SPDX-License-Identifier: Apache-2.0
 
 import { atoms } from "@/app/store/global";
+import { useAtomValueSafe } from "@/util/util";
 import loader from "@monaco-editor/loader";
 import { Editor, Monaco } from "@monaco-editor/react";
-import { atom, useAtomValue } from "jotai";
+import { Atom, atom, useAtomValue } from "jotai";
 import type * as MonacoTypes from "monaco-editor/esm/vs/editor/editor.api";
-import React, { useMemo, useRef } from "react";
+import React, { useMemo, useRef, useState } from "react";
 import "./codeeditor.less";
 
 // there is a global monaco variable (TODO get the correct TS type)
@@ -70,7 +71,8 @@ function defaultEditorOptions(): MonacoTypes.editor.IEditorOptions {
 }
 
 interface CodeEditorProps {
-    text: string;
+    text?: string;
+    textAtom?: Atom<string> | Atom<Promise<string>>;
     filename: string;
     language?: string;
     onChange?: (text: string) => void;
@@ -87,11 +89,13 @@ const stickyScrollEnabledAtom = atom((get) => {
     return settings["editor:stickyscrollenabled"] ?? false;
 });
 
-export function CodeEditor({ text, language, filename, onChange, onMount }: CodeEditorProps) {
+export function CodeEditor({ text, textAtom, language, filename, onChange, onMount }: CodeEditorProps) {
     const divRef = useRef<HTMLDivElement>(null);
     const unmountRef = useRef<() => void>(null);
     const minimapEnabled = useAtomValue(minimapEnabledAtom);
     const stickyScrollEnabled = useAtomValue(stickyScrollEnabledAtom);
+    const textAtomValue = useAtomValueSafe<string>(textAtom);
+    const [textValue] = useState(() => textAtomValue ?? text);
     const theme = "wave-theme-dark";
 
     React.useEffect(() => {
@@ -127,7 +131,7 @@ export function CodeEditor({ text, language, filename, onChange, onMount }: Code
             <div className="code-editor" ref={divRef}>
                 <Editor
                     theme={theme}
-                    value={text}
+                    value={textValue}
                     options={editorOpts}
                     onChange={handleEditorChange}
                     onMount={handleEditorOnMount}
diff --git a/frontend/app/view/preview/preview.tsx b/frontend/app/view/preview/preview.tsx
index ea671403d..4a2ecc6a5 100644
--- a/frontend/app/view/preview/preview.tsx
+++ b/frontend/app/view/preview/preview.tsx
@@ -126,7 +126,9 @@ export class PreviewModel implements ViewModel {
     fileMimeType: jotai.Atom<Promise<string>>;
     fileMimeTypeLoadable: jotai.Atom<Loadable<string>>;
     fileContent: jotai.Atom<Promise<string>>;
-    newFileContent: jotai.PrimitiveAtom<string | null>;
+    fileContentLastSaved: jotai.PrimitiveAtom<string>;
+    fileContentModified: jotai.PrimitiveAtom<string>;
+    previewFileContent: jotai.WritableAtom<Promise<string>, [value: string], void>;
 
     openFileModal: jotai.PrimitiveAtom<boolean>;
     openFileError: jotai.PrimitiveAtom<string>;
@@ -208,6 +210,7 @@ export class PreviewModel implements ViewModel {
             const blockData = get(this.blockAtom);
             return blockData?.meta?.edit ?? false;
         });
+
         this.viewName = jotai.atom("Preview");
         this.viewText = jotai.atom((get) => {
             let headerPath = get(this.metaFilePath);
@@ -241,7 +244,7 @@ export class PreviewModel implements ViewModel {
                 },
             ];
             let saveClassName = "secondary";
-            if (get(this.newFileContent) !== null) {
+            if (get(this.fileContentModified) !== null) {
                 saveClassName = "primary";
             }
             if (isCeView) {
@@ -368,7 +371,6 @@ export class PreviewModel implements ViewModel {
             return fileInfo?.mimetype;
         });
         this.fileMimeTypeLoadable = loadable(this.fileMimeType);
-        this.newFileContent = jotai.atom(null) as jotai.PrimitiveAtom<string | null>;
         this.goParentDirectory = this.goParentDirectory.bind(this);
 
         const fullFileAtom = jotai.atom<Promise<FullFile>>(async (get) => {
@@ -389,6 +391,23 @@ export class PreviewModel implements ViewModel {
         this.fullFile = fullFileAtom;
         this.fileContent = fileContentAtom;
 
+        this.fileContentModified = jotai.atom();
+        this.fileContentLastSaved = jotai.atom();
+        this.previewFileContent = jotai.atom(
+            async (get) => {
+                const fileContentModified = get(this.fileContentModified);
+                const fileContentLastSaved = get(this.fileContentLastSaved);
+                if (!fileContentModified) {
+                    return fileContentLastSaved ?? (await get(this.fileContent));
+                } else {
+                    return fileContentModified;
+                }
+            },
+            (_get, set, value) => {
+                set(this.fileContentModified, value);
+            }
+        );
+
         this.specializedView = jotai.atom<Promise<{ specializedView?: string; errorStr?: string }>>(async (get) => {
             return this.getSpecializedView(get);
         });
@@ -533,7 +552,7 @@ export class PreviewModel implements ViewModel {
         if (filePath == null) {
             return;
         }
-        const newFileContent = globalStore.get(this.newFileContent);
+        const newFileContent = globalStore.get(this.fileContentModified);
         if (newFileContent == null) {
             console.log("not saving file, newFileContent is null");
             return;
@@ -541,7 +560,8 @@ export class PreviewModel implements ViewModel {
         const conn = globalStore.get(this.connection) ?? "";
         try {
             services.FileService.SaveFile(conn, filePath, util.stringToBase64(newFileContent));
-            globalStore.set(this.newFileContent, null);
+            globalStore.set(this.fileContentModified, null);
+            globalStore.set(this.fileContentLastSaved, newFileContent);
             console.log("saved file", filePath);
         } catch (error) {
             console.error("Error saving file:", error);
@@ -549,9 +569,7 @@ export class PreviewModel implements ViewModel {
     }
 
     async handleFileRevert() {
-        const fileContent = await globalStore.get(this.fileContent);
-        this.monacoRef.current?.setValue(fileContent);
-        globalStore.set(this.newFileContent, null);
+        globalStore.set(this.fileContentModified, null);
     }
 
     async handleOpenFile(filePath: string) {
@@ -621,7 +639,7 @@ export class PreviewModel implements ViewModel {
         const loadableSV = globalStore.get(this.loadableSpecializedView);
         if (loadableSV.state == "hasData") {
             if (loadableSV.data.specializedView == "codeedit") {
-                if (globalStore.get(this.newFileContent) != null) {
+                if (globalStore.get(this.fileContentModified) != null) {
                     menuItems.push({ type: "separator" });
                     menuItems.push({
                         label: "Save File",
@@ -711,7 +729,11 @@ function MarkdownPreview({ model }: SpecializedViewProps) {
     }, [connName, fileInfo.dir]);
     return (
         <div className="view-preview view-preview-markdown">
-            <Markdown textAtom={model.fileContent} showTocAtom={model.markdownShowToc} resolveOpts={resolveOpts} />
+            <Markdown
+                textAtom={model.previewFileContent}
+                showTocAtom={model.markdownShowToc}
+                resolveOpts={resolveOpts}
+            />
         </div>
     );
 }
@@ -762,8 +784,7 @@ function StreamingPreview({ model }: SpecializedViewProps) {
 }
 
 function CodeEditPreview({ model }: SpecializedViewProps) {
-    const fileContent = jotai.useAtomValue(model.fileContent);
-    const setNewFileContent = jotai.useSetAtom(model.newFileContent);
+    const setNewFileContent = jotai.useSetAtom(model.previewFileContent);
     const fileName = jotai.useAtomValue(model.statFilePath);
 
     function codeEditKeyDownHandler(e: WaveKeyboardEvent): boolean {
@@ -812,16 +833,16 @@ function CodeEditPreview({ model }: SpecializedViewProps) {
 
     return (
         <CodeEditor
-            text={fileContent}
+            textAtom={model.previewFileContent}
             filename={fileName}
-            onChange={(text) => setNewFileContent(text)}
+            onChange={setNewFileContent}
             onMount={onMount}
         />
     );
 }
 
 function CSVViewPreview({ model, parentRef }: SpecializedViewProps) {
-    const fileContent = jotai.useAtomValue(model.fileContent);
+    const fileContent = jotai.useAtomValue(model.previewFileContent);
     const fileName = jotai.useAtomValue(model.statFilePath);
     return <CSVView parentRef={parentRef} readonly={true} content={fileContent} filename={fileName} />;
 }