untitledEditorModel.ts 6.6 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5 6
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/
'use strict';

J
Johannes Rieken 已提交
7 8 9 10
import { IDisposable } from 'vs/base/common/lifecycle';
import { TPromise } from 'vs/base/common/winjs.base';
import { EditorModel, IEncodingSupport } from 'vs/workbench/common/editor';
import { StringEditorModel } from 'vs/workbench/common/editor/stringEditorModel';
E
Erich Gamma 已提交
11
import URI from 'vs/base/common/uri';
J
Johannes Rieken 已提交
12 13
import { PLAINTEXT_MODE_ID } from 'vs/editor/common/modes/modesRegistry';
import { EndOfLinePreference } from 'vs/editor/common/editorCommon';
14
import { IFileService, IFilesConfiguration } from 'vs/platform/files/common/files';
J
Johannes Rieken 已提交
15 16 17 18 19
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { IModeService } from 'vs/editor/common/services/modeService';
import { IModelService } from 'vs/editor/common/services/modelService';
import { IMode } from 'vs/editor/common/modes';
import Event, { Emitter } from 'vs/base/common/event';
E
Erich Gamma 已提交
20 21

export class UntitledEditorModel extends StringEditorModel implements IEncodingSupport {
A
Alex Dima 已提交
22
	private textModelChangeListener: IDisposable;
23
	private configurationChangeListener: IDisposable;
E
Erich Gamma 已提交
24 25

	private dirty: boolean;
26
	private _onDidChangeDirty: Emitter<void>;
27
	private _onDidChangeEncoding: Emitter<void>;
28

E
Erich Gamma 已提交
29 30 31
	private configuredEncoding: string;
	private preferredEncoding: string;

32 33
	private hasAssociatedFilePath: boolean;

D
Daniel Imms 已提交
34 35
	private backupPromises: TPromise<void>[];

E
Erich Gamma 已提交
36 37
	constructor(
		value: string,
38
		modeId: string,
E
Erich Gamma 已提交
39 40 41 42
		resource: URI,
		hasAssociatedFilePath: boolean,
		@IModeService modeService: IModeService,
		@IModelService modelService: IModelService,
D
Daniel Imms 已提交
43
		@IFileService private fileService: IFileService,
E
Erich Gamma 已提交
44 45
		@IConfigurationService private configurationService: IConfigurationService
	) {
46
		super(value, modeId, resource, modeService, modelService);
E
Erich Gamma 已提交
47

48
		this.hasAssociatedFilePath = hasAssociatedFilePath;
49
		this.dirty = hasAssociatedFilePath || value !== ''; // untitled associated to file path are dirty right away
E
Erich Gamma 已提交
50

51
		this._onDidChangeDirty = new Emitter<void>();
52
		this._onDidChangeEncoding = new Emitter<void>();
53

D
Daniel Imms 已提交
54
		this.backupPromises = [];
E
Erich Gamma 已提交
55

D
Daniel Imms 已提交
56
		this.registerListeners();
57 58
	}

59 60 61 62
	public get onDidChangeDirty(): Event<void> {
		return this._onDidChangeDirty.event;
	}

63 64 65 66
	public get onDidChangeEncoding(): Event<void> {
		return this._onDidChangeEncoding.event;
	}

67 68 69
	protected getOrCreateMode(modeService: IModeService, modeId: string, firstLineText?: string): TPromise<IMode> {
		if (!modeId || modeId === PLAINTEXT_MODE_ID) {
			return modeService.getOrCreateModeByFilenameOrFirstLine(this.resource.fsPath, firstLineText); // lookup mode via resource path if the provided modeId is unspecific
70 71
		}

72
		return super.getOrCreateMode(modeService, modeId, firstLineText);
73 74
	}

E
Erich Gamma 已提交
75 76 77
	private registerListeners(): void {

		// Config Changes
78
		this.configurationChangeListener = this.configurationService.onDidUpdateConfiguration(e => this.onConfigurationChange(e.config));
E
Erich Gamma 已提交
79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
	}

	private onConfigurationChange(configuration: IFilesConfiguration): void {
		this.configuredEncoding = configuration && configuration.files && configuration.files.encoding;
	}

	public getValue(): string {
		if (this.textEditorModel) {
			return this.textEditorModel.getValue(EndOfLinePreference.TextDefined, true /* Preserve BOM */);
		}

		return null;
	}

	public getModeId(): string {
		if (this.textEditorModel) {
			return this.textEditorModel.getModeId();
		}

		return null;
	}

	public getEncoding(): string {
		return this.preferredEncoding || this.configuredEncoding;
	}

	public setEncoding(encoding: string): void {
106
		const oldEncoding = this.getEncoding();
E
Erich Gamma 已提交
107 108 109 110
		this.preferredEncoding = encoding;

		// Emit if it changed
		if (oldEncoding !== this.preferredEncoding) {
111
			this._onDidChangeEncoding.fire();
E
Erich Gamma 已提交
112 113 114 115 116 117 118
		}
	}

	public isDirty(): boolean {
		return this.dirty;
	}

119 120 121
	public revert(): void {
		this.dirty = false;

122
		this._onDidChangeDirty.fire();
123 124
	}

E
Erich Gamma 已提交
125 126
	public load(): TPromise<EditorModel> {
		return super.load().then((model) => {
127
			const configuration = this.configurationService.getConfiguration<IFilesConfiguration>();
E
Erich Gamma 已提交
128

129 130
			// Encoding
			this.configuredEncoding = configuration && configuration.files && configuration.files.encoding;
E
Erich Gamma 已提交
131

132
			// Listen to content changes
B
Benjamin Pasero 已提交
133
			this.textModelChangeListener = this.textEditorModel.onDidChangeContent(e => this.onModelContentChanged());
E
Erich Gamma 已提交
134

135 136 137
			// Emit initial dirty event if we are
			if (this.dirty) {
				setTimeout(() => {
138
					this._onDidChangeDirty.fire();
139 140
				}, 0 /* prevent race condition between creating model and emitting dirty event */);
			}
E
Erich Gamma 已提交
141

142
			return model;
E
Erich Gamma 已提交
143 144 145
		});
	}

B
Benjamin Pasero 已提交
146
	private onModelContentChanged(): void {
147 148

		// mark the untitled editor as non-dirty once its content becomes empty and we do
B
Benjamin Pasero 已提交
149 150 151 152 153 154
		// not have an associated path set. we never want dirty indicator in that case.
		if (!this.hasAssociatedFilePath && this.textEditorModel.getLineCount() === 1 && this.textEditorModel.getLineContent(1) === '') {
			if (this.dirty) {
				this.dirty = false;
				this._onDidChangeDirty.fire();
			}
E
Erich Gamma 已提交
155
		}
156

B
Benjamin Pasero 已提交
157 158 159
		// turn dirty if we were not
		else if (!this.dirty) {
			this.dirty = true;
160
			this._onDidChangeDirty.fire();
D
Daniel Imms 已提交
161

162
		}
163

D
Daniel Imms 已提交
164 165 166 167 168 169 170 171 172
		if (this.fileService.isHotExitEnabled()) {
			if (this.dirty) {
				console.log('backup');
				this.doBackup();
			} else {
				console.log('discard');
				this.fileService.discardBackup(this.resource);
			}
		}
E
Erich Gamma 已提交
173 174 175 176 177 178
	}

	public dispose(): void {
		super.dispose();

		if (this.textModelChangeListener) {
A
Alex Dima 已提交
179
			this.textModelChangeListener.dispose();
E
Erich Gamma 已提交
180 181 182
			this.textModelChangeListener = null;
		}

183 184 185
		if (this.configurationChangeListener) {
			this.configurationChangeListener.dispose();
			this.configurationChangeListener = null;
E
Erich Gamma 已提交
186
		}
187 188 189

		this._onDidChangeDirty.dispose();
		this._onDidChangeEncoding.dispose();
D
Daniel Imms 已提交
190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213

		this.cancelBackupPromises();
		console.log('discard');
		this.fileService.discardBackup(this.resource);
	}

	private doBackup(): TPromise<void> {
		// Cancel any currently running backups to make this the one that succeeds
		this.cancelBackupPromises();

		// Create new backup promise and keep it
		const promise = TPromise.timeout(1000).then(() => {
			this.fileService.backupFile(this.resource, this.getValue()); // Very important here to not return the promise because if the timeout promise is canceled it will bubble up the error otherwise - do not change
		});

		this.backupPromises.push(promise);

		return promise;
	}

	private cancelBackupPromises(): void {
		while (this.backupPromises.length) {
			this.backupPromises.pop().cancel();
		}
E
Erich Gamma 已提交
214 215
	}
}