textfiles.ts 9.8 KB
Newer Older
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

J
Johannes Rieken 已提交
6
import { TPromise } from 'vs/base/common/winjs.base';
7
import { URI } from 'vs/base/common/uri';
M
Matt Bierner 已提交
8
import { Event } from 'vs/base/common/event';
J
Johannes Rieken 已提交
9
import { IDisposable } from 'vs/base/common/lifecycle';
10
import { IEncodingSupport, ConfirmResult, IRevertOptions } from 'vs/workbench/common/editor';
11
import { IBaseStat, IResolveContentOptions, ITextSnapshot } from 'vs/platform/files/common/files';
J
Johannes Rieken 已提交
12
import { createDecorator } from 'vs/platform/instantiation/common/instantiation';
J
Joao Moreno 已提交
13
import { ITextEditorModel } from 'vs/editor/common/services/resolverService';
14
import { ITextBufferFactory } from 'vs/editor/common/model';
I
isidor 已提交
15
import { RawContextKey } from 'vs/platform/contextkey/common/contextkey';
16 17

/**
N
Nick Schonning 已提交
18
 * The save error handler can be installed on the text file editor model to install code that executes when save errors occur.
19 20 21 22 23 24
 */
export interface ISaveErrorHandler {

	/**
	 * Called whenever a save fails.
	 */
B
Benjamin Pasero 已提交
25
	onSaveError(error: Error, model: ITextFileEditorModel): void;
26 27 28 29 30 31 32
}

export interface ISaveParticipant {

	/**
	 * Participate in a save of a model. Allows to change the model before it is being saved to disk.
	 */
B
Benjamin Pasero 已提交
33
	participate(model: ITextFileEditorModel, env: { reason: SaveReason }): Thenable<void>;
34 35 36
}

/**
N
Nick Schonning 已提交
37
 * States the text file editor model can be in.
38
 */
39
export const enum ModelState {
40 41 42
	SAVED,
	DIRTY,
	PENDING_SAVE,
43 44 45 46 47

	/**
	 * A model is in conflict mode when changes cannot be saved because the
	 * underlying file has changed. Models in conflict mode are always dirty.
	 */
48
	CONFLICT,
49 50 51 52 53 54 55 56 57 58

	/**
	 * A model is in orphan state when the underlying file has been deleted.
	 */
	ORPHAN,

	/**
	 * Any error that happens during a save that is not causing the CONFLICT state.
	 * Models in error mode are always diry.
	 */
59 60 61
	ERROR
}

62
export const enum StateChange {
63 64 65 66 67
	DIRTY,
	SAVING,
	SAVE_ERROR,
	SAVED,
	REVERTED,
68
	ENCODING,
69 70
	CONTENT_CHANGE,
	ORPHANED_CHANGE
71 72 73 74 75 76 77 78 79 80 81
}

export class TextFileModelChangeEvent {
	private _resource: URI;
	private _kind: StateChange;

	constructor(model: ITextFileEditorModel, kind: StateChange) {
		this._resource = model.getResource();
		this._kind = kind;
	}

B
Benjamin Pasero 已提交
82
	get resource(): URI {
83 84 85
		return this._resource;
	}

B
Benjamin Pasero 已提交
86
	get kind(): StateChange {
87 88 89 90 91
		return this._kind;
	}
}

export const TEXT_FILE_SERVICE_ID = 'textFileService';
I
isidor 已提交
92
export const AutoSaveContext = new RawContextKey<string>('config.files.autoSave', undefined);
93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109

export interface ITextFileOperationResult {
	results: IResult[];
}

export interface IResult {
	source: URI;
	target?: URI;
	success?: boolean;
}

export interface IAutoSaveConfiguration {
	autoSaveDelay: number;
	autoSaveFocusChange: boolean;
	autoSaveApplicationChange: boolean;
}

110
export const enum AutoSaveMode {
111 112 113 114 115 116 117
	OFF,
	AFTER_SHORT_DELAY,
	AFTER_LONG_DELAY,
	ON_FOCUS_CHANGE,
	ON_WINDOW_CHANGE
}

118
export const enum SaveReason {
119 120 121 122 123 124
	EXPLICIT = 1,
	AUTO = 2,
	FOCUS_CHANGE = 3,
	WINDOW_CHANGE = 4
}

125
export const enum LoadReason {
126 127 128 129 130
	EDITOR = 1,
	REFERENCE = 2,
	OTHER = 3
}

131 132 133 134 135 136 137
export const ITextFileService = createDecorator<ITextFileService>(TEXT_FILE_SERVICE_ID);

export interface IRawTextContent extends IBaseStat {

	/**
	 * The line grouped content of a text file.
	 */
138
	value: ITextBufferFactory;
139 140 141 142 143 144 145

	/**
	 * The encoding of the content if known.
	 */
	encoding: string;
}

146
export interface IModelLoadOrCreateOptions {
147

148 149 150 151
	/**
	 * Context why the model is being loaded or created.
	 */
	reason?: LoadReason;
152 153 154 155

	/**
	 * The encoding to use when resolving the model text content.
	 */
156
	encoding?: string;
157 158

	/**
159 160 161 162 163 164
	 * If the model was already loaded before, allows to trigger
	 * a reload of it to fetch the latest contents:
	 * - async: loadOrCreate() will return immediately and trigger
	 * a reload that will run in the background.
	 * - sync: loadOrCreate() will only return resolved when the
	 * model has finished reloading.
165
	 */
166 167 168
	reload?: {
		async: boolean
	};
169 170 171 172 173

	/**
	 * Allow to load a model even if we think it is a binary file.
	 */
	allowBinary?: boolean;
174 175
}

176 177
export interface ITextFileEditorModelManager {

178
	onModelDisposed: Event<URI>;
179
	onModelContentChanged: Event<TextFileModelChangeEvent>;
180 181
	onModelEncodingChanged: Event<TextFileModelChangeEvent>;

182 183 184 185
	onModelDirty: Event<TextFileModelChangeEvent>;
	onModelSaveError: Event<TextFileModelChangeEvent>;
	onModelSaved: Event<TextFileModelChangeEvent>;
	onModelReverted: Event<TextFileModelChangeEvent>;
186
	onModelOrphanedChanged: Event<TextFileModelChangeEvent>;
187 188 189 190 191

	onModelsDirty: Event<TextFileModelChangeEvent[]>;
	onModelsSaveError: Event<TextFileModelChangeEvent[]>;
	onModelsSaved: Event<TextFileModelChangeEvent[]>;
	onModelsReverted: Event<TextFileModelChangeEvent[]>;
192 193 194 195 196

	get(resource: URI): ITextFileEditorModel;

	getAll(resource?: URI): ITextFileEditorModel[];

B
Benjamin Pasero 已提交
197
	loadOrCreate(resource: URI, options?: IModelLoadOrCreateOptions): TPromise<ITextFileEditorModel>;
198 199

	disposeModel(model: ITextFileEditorModel): void;
200 201
}

202
export interface ISaveOptions {
203
	force?: boolean;
204 205 206
	reason?: SaveReason;
	overwriteReadonly?: boolean;
	overwriteEncoding?: boolean;
207
	skipSaveParticipants?: boolean;
208
	writeElevated?: boolean;
209 210
}

211 212 213 214 215 216 217 218 219 220 221
export interface ILoadOptions {

	/**
	 * Go to disk bypassing any cache of the model if any.
	 */
	forceReadFromDisk?: boolean;

	/**
	 * Allow to load a model even if we think it is a binary file.
	 */
	allowBinary?: boolean;
222 223 224 225 226

	/**
	 * Context why the model is being loaded.
	 */
	reason?: LoadReason;
227 228
}

229 230
export interface ITextFileEditorModel extends ITextEditorModel, IEncodingSupport {

231
	onDidContentChange: Event<StateChange>;
232 233
	onDidStateChange: Event<StateChange>;

234 235
	getVersionId(): number;

236 237
	getResource(): URI;

238
	hasState(state: ModelState): boolean;
239

240 241
	getETag(): string;

242 243
	updatePreferredEncoding(encoding: string): void;

244
	save(options?: ISaveOptions): TPromise<void>;
245

B
Benjamin Pasero 已提交
246
	load(options?: ILoadOptions): Thenable<ITextFileEditorModel>;
B
Benjamin Pasero 已提交
247

248
	revert(soft?: boolean): TPromise<void>;
249

250 251
	createSnapshot(): ITextSnapshot;

252 253
	isDirty(): boolean;

254 255
	isResolved(): boolean;

256 257 258
	isDisposed(): boolean;
}

259 260 261 262

export interface IWillMoveEvent {
	oldResource: URI;
	newResource: URI;
263
	waitUntil(p: Thenable<any>): void;
264 265
}

266 267
export interface ITextFileService extends IDisposable {
	_serviceBrand: any;
B
Benjamin Pasero 已提交
268 269 270 271

	readonly onAutoSaveConfigurationChange: Event<IAutoSaveConfiguration>;
	readonly onFilesAssociationChange: Event<void>;

272 273
	onWillMove: Event<IWillMoveEvent>;

B
Benjamin Pasero 已提交
274
	readonly isHotExitEnabled: boolean;
275 276 277 278

	/**
	 * Access to the manager of text file editor models providing further methods to work with them.
	 */
B
Benjamin Pasero 已提交
279
	readonly models: ITextFileEditorModelManager;
280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305

	/**
	 * Resolve the contents of a file identified by the resource.
	 */
	resolveTextContent(resource: URI, options?: IResolveContentOptions): TPromise<IRawTextContent>;

	/**
	 * A resource is dirty if it has unsaved changes or is an untitled file not yet saved.
	 *
	 * @param resource the resource to check for being dirty. If it is not specified, will check for
	 * all dirty resources.
	 */
	isDirty(resource?: URI): boolean;

	/**
	 * Returns all resources that are currently dirty matching the provided resources or all dirty resources.
	 *
	 * @param resources the resources to check for being dirty. If it is not specified, will check for
	 * all dirty resources.
	 */
	getDirty(resources?: URI[]): URI[];

	/**
	 * Saves the resource.
	 *
	 * @param resource the resource to save
306
	 * @param options optional save options
307
	 * @return true if the resource was saved.
308 309 310 311
	 */
	save(resource: URI, options?: ISaveOptions): TPromise<boolean>;

	/**
312
	 * Saves the provided resource asking the user for a file name or using the provided one.
313 314
	 *
	 * @param resource the resource to save as.
315 316
	 * @param targetResource the optional target to save to.
	 * @param options optional save options
317
	 * @return true if the file was saved.
318
	 */
319
	saveAs(resource: URI, targetResource?: URI, options?: ISaveOptions): TPromise<URI>;
320 321 322 323 324 325 326

	/**
	 * Saves the set of resources and returns a promise with the operation result.
	 *
	 * @param resources can be null to save all.
	 * @param includeUntitled to save all resources and optionally exclude untitled ones.
	 */
327 328
	saveAll(includeUntitled?: boolean, options?: ISaveOptions): TPromise<ITextFileOperationResult>;
	saveAll(resources: URI[], options?: ISaveOptions): TPromise<ITextFileOperationResult>;
329 330 331 332 333 334 335

	/**
	 * Reverts the provided resource.
	 *
	 * @param resource the resource of the file to revert.
	 * @param force to force revert even when the file is not dirty
	 */
336
	revert(resource: URI, options?: IRevertOptions): TPromise<boolean>;
337 338 339 340

	/**
	 * Reverts all the provided resources and returns a promise with the operation result.
	 */
341
	revertAll(resources?: URI[], options?: IRevertOptions): TPromise<ITextFileOperationResult>;
342

B
Benjamin Pasero 已提交
343 344 345 346 347 348
	/**
	 * Create a file. If the file exists it will be overwritten with the contents if
	 * the options enable to overwrite.
	 */
	create(resource: URI, contents?: string, options?: { overwrite?: boolean }): TPromise<void>;

B
Benjamin Pasero 已提交
349 350 351
	/**
	 * Delete a file. If the file is dirty, it will get reverted and then deleted from disk.
	 */
352
	delete(resource: URI, options?: { useTrash?: boolean, recursive?: boolean }): TPromise<void>;
B
Benjamin Pasero 已提交
353 354 355 356 357 358

	/**
	 * Move a file. If the file is dirty, its contents will be preserved and restored.
	 */
	move(source: URI, target: URI, overwrite?: boolean): TPromise<void>;

359 360 361 362 363 364
	/**
	 * Brings up the confirm dialog to either save, don't save or cancel.
	 *
	 * @param resources the resources of the files to ask for confirmation or null if
	 * confirming for all dirty resources.
	 */
365
	confirmSave(resources?: URI[]): TPromise<ConfirmResult>;
366 367 368 369 370 371 372 373 374 375

	/**
	 * Convinient fast access to the current auto save mode.
	 */
	getAutoSaveMode(): AutoSaveMode;

	/**
	 * Convinient fast access to the raw configured auto save settings.
	 */
	getAutoSaveConfiguration(): IAutoSaveConfiguration;
I
isidor 已提交
376
}