Atom/src/workspace.js

2269 lines
75 KiB
JavaScript

const _ = require('underscore-plus');
const url = require('url');
const path = require('path');
const { Emitter, Disposable, CompositeDisposable } = require('event-kit');
const fs = require('fs-plus');
const { Directory } = require('pathwatcher');
const Grim = require('grim');
const DefaultDirectorySearcher = require('./default-directory-searcher');
const RipgrepDirectorySearcher = require('./ripgrep-directory-searcher');
const Dock = require('./dock');
const Model = require('./model');
const StateStore = require('./state-store');
const TextEditor = require('./text-editor');
const Panel = require('./panel');
const PanelContainer = require('./panel-container');
const Task = require('./task');
const WorkspaceCenter = require('./workspace-center');
const WorkspaceElement = require('./workspace-element');
const STOPPED_CHANGING_ACTIVE_PANE_ITEM_DELAY = 100;
const ALL_LOCATIONS = ['center', 'left', 'right', 'bottom'];
// Essential: Represents the state of the user interface for the entire window.
// An instance of this class is available via the `atom.workspace` global.
//
// Interact with this object to open files, be notified of current and future
// editors, and manipulate panes. To add panels, use {Workspace::addTopPanel}
// and friends.
//
// ## Workspace Items
//
// The term "item" refers to anything that can be displayed
// in a pane within the workspace, either in the {WorkspaceCenter} or in one
// of the three {Dock}s. The workspace expects items to conform to the
// following interface:
//
// ### Required Methods
//
// #### `getTitle()`
//
// Returns a {String} containing the title of the item to display on its
// associated tab.
//
// ### Optional Methods
//
// #### `getElement()`
//
// If your item already *is* a DOM element, you do not need to implement this
// method. Otherwise it should return the element you want to display to
// represent this item.
//
// #### `destroy()`
//
// Destroys the item. This will be called when the item is removed from its
// parent pane.
//
// #### `onDidDestroy(callback)`
//
// Called by the workspace so it can be notified when the item is destroyed.
// Must return a {Disposable}.
//
// #### `serialize()`
//
// Serialize the state of the item. Must return an object that can be passed to
// `JSON.stringify`. The state should include a field called `deserializer`,
// which names a deserializer declared in your `package.json`. This method is
// invoked on items when serializing the workspace so they can be restored to
// the same location later.
//
// #### `getURI()`
//
// Returns the URI associated with the item.
//
// #### `getLongTitle()`
//
// Returns a {String} containing a longer version of the title to display in
// places like the window title or on tabs their short titles are ambiguous.
//
// #### `onDidChangeTitle(callback)`
//
// Called by the workspace so it can be notified when the item's title changes.
// Must return a {Disposable}.
//
// #### `getIconName()`
//
// Return a {String} with the name of an icon. If this method is defined and
// returns a string, the item's tab element will be rendered with the `icon` and
// `icon-${iconName}` CSS classes.
//
// ### `onDidChangeIcon(callback)`
//
// Called by the workspace so it can be notified when the item's icon changes.
// Must return a {Disposable}.
//
// #### `getDefaultLocation()`
//
// Tells the workspace where your item should be opened in absence of a user
// override. Items can appear in the center or in a dock on the left, right, or
// bottom of the workspace.
//
// Returns a {String} with one of the following values: `'center'`, `'left'`,
// `'right'`, `'bottom'`. If this method is not defined, `'center'` is the
// default.
//
// #### `getAllowedLocations()`
//
// Tells the workspace where this item can be moved. Returns an {Array} of one
// or more of the following values: `'center'`, `'left'`, `'right'`, or
// `'bottom'`.
//
// #### `isPermanentDockItem()`
//
// Tells the workspace whether or not this item can be closed by the user by
// clicking an `x` on its tab. Use of this feature is discouraged unless there's
// a very good reason not to allow users to close your item. Items can be made
// permanent *only* when they are contained in docks. Center pane items can
// always be removed. Note that it is currently still possible to close dock
// items via the `Close Pane` option in the context menu and via Atom APIs, so
// you should still be prepared to handle your dock items being destroyed by the
// user even if you implement this method.
//
// #### `save()`
//
// Saves the item.
//
// #### `saveAs(path)`
//
// Saves the item to the specified path.
//
// #### `getPath()`
//
// Returns the local path associated with this item. This is only used to set
// the initial location of the "save as" dialog.
//
// #### `isModified()`
//
// Returns whether or not the item is modified to reflect modification in the
// UI.
//
// #### `onDidChangeModified()`
//
// Called by the workspace so it can be notified when item's modified status
// changes. Must return a {Disposable}.
//
// #### `copy()`
//
// Create a copy of the item. If defined, the workspace will call this method to
// duplicate the item when splitting panes via certain split commands.
//
// #### `getPreferredHeight()`
//
// If this item is displayed in the bottom {Dock}, called by the workspace when
// initially displaying the dock to set its height. Once the dock has been
// resized by the user, their height will override this value.
//
// Returns a {Number}.
//
// #### `getPreferredWidth()`
//
// If this item is displayed in the left or right {Dock}, called by the
// workspace when initially displaying the dock to set its width. Once the dock
// has been resized by the user, their width will override this value.
//
// Returns a {Number}.
//
// #### `onDidTerminatePendingState(callback)`
//
// If the workspace is configured to use *pending pane items*, the workspace
// will subscribe to this method to terminate the pending state of the item.
// Must return a {Disposable}.
//
// #### `shouldPromptToSave()`
//
// This method indicates whether Atom should prompt the user to save this item
// when the user closes or reloads the window. Returns a boolean.
module.exports = class Workspace extends Model {
constructor(params) {
super(...arguments);
this.updateWindowTitle = this.updateWindowTitle.bind(this);
this.updateDocumentEdited = this.updateDocumentEdited.bind(this);
this.didDestroyPaneItem = this.didDestroyPaneItem.bind(this);
this.didChangeActivePaneOnPaneContainer = this.didChangeActivePaneOnPaneContainer.bind(
this
);
this.didChangeActivePaneItemOnPaneContainer = this.didChangeActivePaneItemOnPaneContainer.bind(
this
);
this.didActivatePaneContainer = this.didActivatePaneContainer.bind(this);
this.enablePersistence = params.enablePersistence;
this.packageManager = params.packageManager;
this.config = params.config;
this.project = params.project;
this.notificationManager = params.notificationManager;
this.viewRegistry = params.viewRegistry;
this.grammarRegistry = params.grammarRegistry;
this.applicationDelegate = params.applicationDelegate;
this.assert = params.assert;
this.deserializerManager = params.deserializerManager;
this.textEditorRegistry = params.textEditorRegistry;
this.styleManager = params.styleManager;
this.draggingItem = false;
this.itemLocationStore = new StateStore('AtomPreviousItemLocations', 1);
this.emitter = new Emitter();
this.openers = [];
this.destroyedItemURIs = [];
this.stoppedChangingActivePaneItemTimeout = null;
this.scandalDirectorySearcher = new DefaultDirectorySearcher();
this.ripgrepDirectorySearcher = new RipgrepDirectorySearcher();
this.consumeServices(this.packageManager);
this.paneContainers = {
center: this.createCenter(),
left: this.createDock('left'),
right: this.createDock('right'),
bottom: this.createDock('bottom')
};
this.activePaneContainer = this.paneContainers.center;
this.hasActiveTextEditor = false;
this.panelContainers = {
top: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'top'
}),
left: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'left',
dock: this.paneContainers.left
}),
right: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'right',
dock: this.paneContainers.right
}),
bottom: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'bottom',
dock: this.paneContainers.bottom
}),
header: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'header'
}),
footer: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'footer'
}),
modal: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'modal'
})
};
this.incoming = new Map();
}
get paneContainer() {
Grim.deprecate(
'`atom.workspace.paneContainer` has always been private, but it is now gone. Please use `atom.workspace.getCenter()` instead and consult the workspace API docs for public methods.'
);
return this.paneContainers.center.paneContainer;
}
getElement() {
if (!this.element) {
this.element = new WorkspaceElement().initialize(this, {
config: this.config,
project: this.project,
viewRegistry: this.viewRegistry,
styleManager: this.styleManager
});
}
return this.element;
}
createCenter() {
return new WorkspaceCenter({
config: this.config,
applicationDelegate: this.applicationDelegate,
notificationManager: this.notificationManager,
deserializerManager: this.deserializerManager,
viewRegistry: this.viewRegistry,
didActivate: this.didActivatePaneContainer,
didChangeActivePane: this.didChangeActivePaneOnPaneContainer,
didChangeActivePaneItem: this.didChangeActivePaneItemOnPaneContainer,
didDestroyPaneItem: this.didDestroyPaneItem
});
}
createDock(location) {
return new Dock({
location,
config: this.config,
applicationDelegate: this.applicationDelegate,
deserializerManager: this.deserializerManager,
notificationManager: this.notificationManager,
viewRegistry: this.viewRegistry,
didActivate: this.didActivatePaneContainer,
didChangeActivePane: this.didChangeActivePaneOnPaneContainer,
didChangeActivePaneItem: this.didChangeActivePaneItemOnPaneContainer,
didDestroyPaneItem: this.didDestroyPaneItem
});
}
reset(packageManager) {
this.packageManager = packageManager;
this.emitter.dispose();
this.emitter = new Emitter();
this.paneContainers.center.destroy();
this.paneContainers.left.destroy();
this.paneContainers.right.destroy();
this.paneContainers.bottom.destroy();
_.values(this.panelContainers).forEach(panelContainer => {
panelContainer.destroy();
});
this.paneContainers = {
center: this.createCenter(),
left: this.createDock('left'),
right: this.createDock('right'),
bottom: this.createDock('bottom')
};
this.activePaneContainer = this.paneContainers.center;
this.hasActiveTextEditor = false;
this.panelContainers = {
top: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'top'
}),
left: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'left',
dock: this.paneContainers.left
}),
right: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'right',
dock: this.paneContainers.right
}),
bottom: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'bottom',
dock: this.paneContainers.bottom
}),
header: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'header'
}),
footer: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'footer'
}),
modal: new PanelContainer({
viewRegistry: this.viewRegistry,
location: 'modal'
})
};
this.originalFontSize = null;
this.openers = [];
this.destroyedItemURIs = [];
if (this.element) {
this.element.destroy();
this.element = null;
}
this.consumeServices(this.packageManager);
}
initialize() {
this.originalFontSize = this.config.get('editor.fontSize');
this.project.onDidChangePaths(this.updateWindowTitle);
this.subscribeToAddedItems();
this.subscribeToMovedItems();
this.subscribeToDockToggling();
}
consumeServices({ serviceHub }) {
this.directorySearchers = [];
serviceHub.consume('atom.directory-searcher', '^0.1.0', provider =>
this.directorySearchers.unshift(provider)
);
}
// Called by the Serializable mixin during serialization.
serialize() {
return {
deserializer: 'Workspace',
packagesWithActiveGrammars: this.getPackageNamesWithActiveGrammars(),
destroyedItemURIs: this.destroyedItemURIs.slice(),
// Ensure deserializing 1.17 state with pre 1.17 Atom does not error
// TODO: Remove after 1.17 has been on stable for a while
paneContainer: { version: 2 },
paneContainers: {
center: this.paneContainers.center.serialize(),
left: this.paneContainers.left.serialize(),
right: this.paneContainers.right.serialize(),
bottom: this.paneContainers.bottom.serialize()
}
};
}
deserialize(state, deserializerManager) {
const packagesWithActiveGrammars =
state.packagesWithActiveGrammars != null
? state.packagesWithActiveGrammars
: [];
for (let packageName of packagesWithActiveGrammars) {
const pkg = this.packageManager.getLoadedPackage(packageName);
if (pkg != null) {
pkg.loadGrammarsSync();
}
}
if (state.destroyedItemURIs != null) {
this.destroyedItemURIs = state.destroyedItemURIs;
}
if (state.paneContainers) {
this.paneContainers.center.deserialize(
state.paneContainers.center,
deserializerManager
);
this.paneContainers.left.deserialize(
state.paneContainers.left,
deserializerManager
);
this.paneContainers.right.deserialize(
state.paneContainers.right,
deserializerManager
);
this.paneContainers.bottom.deserialize(
state.paneContainers.bottom,
deserializerManager
);
} else if (state.paneContainer) {
// TODO: Remove this fallback once a lot of time has passed since 1.17 was released
this.paneContainers.center.deserialize(
state.paneContainer,
deserializerManager
);
}
this.hasActiveTextEditor = this.getActiveTextEditor() != null;
this.updateWindowTitle();
}
getPackageNamesWithActiveGrammars() {
const packageNames = [];
const addGrammar = ({ includedGrammarScopes, packageName } = {}) => {
if (!packageName) {
return;
}
// Prevent cycles
if (packageNames.indexOf(packageName) !== -1) {
return;
}
packageNames.push(packageName);
for (let scopeName of includedGrammarScopes != null
? includedGrammarScopes
: []) {
addGrammar(this.grammarRegistry.grammarForScopeName(scopeName));
}
};
const editors = this.getTextEditors();
for (let editor of editors) {
addGrammar(editor.getGrammar());
}
if (editors.length > 0) {
for (let grammar of this.grammarRegistry.getGrammars()) {
if (grammar.injectionSelector) {
addGrammar(grammar);
}
}
}
return _.uniq(packageNames);
}
didActivatePaneContainer(paneContainer) {
if (paneContainer !== this.getActivePaneContainer()) {
this.activePaneContainer = paneContainer;
this.didChangeActivePaneItem(
this.activePaneContainer.getActivePaneItem()
);
this.emitter.emit(
'did-change-active-pane-container',
this.activePaneContainer
);
this.emitter.emit(
'did-change-active-pane',
this.activePaneContainer.getActivePane()
);
this.emitter.emit(
'did-change-active-pane-item',
this.activePaneContainer.getActivePaneItem()
);
}
}
didChangeActivePaneOnPaneContainer(paneContainer, pane) {
if (paneContainer === this.getActivePaneContainer()) {
this.emitter.emit('did-change-active-pane', pane);
}
}
didChangeActivePaneItemOnPaneContainer(paneContainer, item) {
if (paneContainer === this.getActivePaneContainer()) {
this.didChangeActivePaneItem(item);
this.emitter.emit('did-change-active-pane-item', item);
}
if (paneContainer === this.getCenter()) {
const hadActiveTextEditor = this.hasActiveTextEditor;
this.hasActiveTextEditor = item instanceof TextEditor;
if (this.hasActiveTextEditor || hadActiveTextEditor) {
const itemValue = this.hasActiveTextEditor ? item : undefined;
this.emitter.emit('did-change-active-text-editor', itemValue);
}
}
}
didChangeActivePaneItem(item) {
this.updateWindowTitle();
this.updateDocumentEdited();
if (this.activeItemSubscriptions) this.activeItemSubscriptions.dispose();
this.activeItemSubscriptions = new CompositeDisposable();
let modifiedSubscription, titleSubscription;
if (item != null && typeof item.onDidChangeTitle === 'function') {
titleSubscription = item.onDidChangeTitle(this.updateWindowTitle);
} else if (item != null && typeof item.on === 'function') {
titleSubscription = item.on('title-changed', this.updateWindowTitle);
if (
titleSubscription == null ||
typeof titleSubscription.dispose !== 'function'
) {
titleSubscription = new Disposable(() => {
item.off('title-changed', this.updateWindowTitle);
});
}
}
if (item != null && typeof item.onDidChangeModified === 'function') {
modifiedSubscription = item.onDidChangeModified(
this.updateDocumentEdited
);
} else if (item != null && typeof item.on === 'function') {
modifiedSubscription = item.on(
'modified-status-changed',
this.updateDocumentEdited
);
if (
modifiedSubscription == null ||
typeof modifiedSubscription.dispose !== 'function'
) {
modifiedSubscription = new Disposable(() => {
item.off('modified-status-changed', this.updateDocumentEdited);
});
}
}
if (titleSubscription != null) {
this.activeItemSubscriptions.add(titleSubscription);
}
if (modifiedSubscription != null) {
this.activeItemSubscriptions.add(modifiedSubscription);
}
this.cancelStoppedChangingActivePaneItemTimeout();
this.stoppedChangingActivePaneItemTimeout = setTimeout(() => {
this.stoppedChangingActivePaneItemTimeout = null;
this.emitter.emit('did-stop-changing-active-pane-item', item);
}, STOPPED_CHANGING_ACTIVE_PANE_ITEM_DELAY);
}
cancelStoppedChangingActivePaneItemTimeout() {
if (this.stoppedChangingActivePaneItemTimeout != null) {
clearTimeout(this.stoppedChangingActivePaneItemTimeout);
}
}
setDraggingItem(draggingItem) {
_.values(this.paneContainers).forEach(dock => {
dock.setDraggingItem(draggingItem);
});
}
subscribeToAddedItems() {
this.onDidAddPaneItem(({ item, pane, index }) => {
if (item instanceof TextEditor) {
const subscriptions = new CompositeDisposable(
this.textEditorRegistry.add(item),
this.textEditorRegistry.maintainConfig(item)
);
if (!this.project.findBufferForId(item.buffer.id)) {
this.project.addBuffer(item.buffer);
}
item.onDidDestroy(() => {
subscriptions.dispose();
});
this.emitter.emit('did-add-text-editor', {
textEditor: item,
pane,
index
});
// It's important to call handleGrammarUsed after emitting the did-add event:
// if we activate a package between adding the editor to the registry and emitting
// the package may receive the editor twice from `observeTextEditors`.
// (Note that the item can be destroyed by an `observeTextEditors` handler.)
if (!item.isDestroyed()) {
subscriptions.add(
item.observeGrammar(this.handleGrammarUsed.bind(this))
);
}
}
});
}
subscribeToDockToggling() {
const docks = [
this.getLeftDock(),
this.getRightDock(),
this.getBottomDock()
];
docks.forEach(dock => {
dock.onDidChangeVisible(visible => {
if (visible) return;
const { activeElement } = document;
const dockElement = dock.getElement();
if (
dockElement === activeElement ||
dockElement.contains(activeElement)
) {
this.getCenter().activate();
}
});
});
}
subscribeToMovedItems() {
for (const paneContainer of this.getPaneContainers()) {
paneContainer.observePanes(pane => {
pane.onDidAddItem(({ item }) => {
if (typeof item.getURI === 'function' && this.enablePersistence) {
const uri = item.getURI();
if (uri) {
const location = paneContainer.getLocation();
let defaultLocation;
if (typeof item.getDefaultLocation === 'function') {
defaultLocation = item.getDefaultLocation();
}
defaultLocation = defaultLocation || 'center';
if (location === defaultLocation) {
this.itemLocationStore.delete(item.getURI());
} else {
this.itemLocationStore.save(item.getURI(), location);
}
}
}
});
});
}
}
// Updates the application's title and proxy icon based on whichever file is
// open.
updateWindowTitle() {
let itemPath, itemTitle, projectPath, representedPath;
const appName = atom.getAppName();
const left = this.project.getPaths();
const projectPaths = left != null ? left : [];
const item = this.getActivePaneItem();
if (item) {
itemPath =
typeof item.getPath === 'function' ? item.getPath() : undefined;
const longTitle =
typeof item.getLongTitle === 'function'
? item.getLongTitle()
: undefined;
itemTitle =
longTitle == null
? typeof item.getTitle === 'function'
? item.getTitle()
: undefined
: longTitle;
projectPath = _.find(
projectPaths,
projectPath =>
itemPath === projectPath ||
(itemPath != null
? itemPath.startsWith(projectPath + path.sep)
: undefined)
);
}
if (itemTitle == null) {
itemTitle = 'untitled';
}
if (projectPath == null) {
projectPath = itemPath ? path.dirname(itemPath) : projectPaths[0];
}
if (projectPath != null) {
projectPath = fs.tildify(projectPath);
}
const titleParts = [];
if (item != null && projectPath != null) {
titleParts.push(itemTitle, projectPath);
representedPath = itemPath != null ? itemPath : projectPath;
} else if (projectPath != null) {
titleParts.push(projectPath);
representedPath = projectPath;
} else {
titleParts.push(itemTitle);
representedPath = '';
}
if (process.platform !== 'darwin') {
titleParts.push(appName);
}
document.title = titleParts.join(' \u2014 ');
this.applicationDelegate.setRepresentedFilename(representedPath);
this.emitter.emit('did-change-window-title');
}
// On macOS, fades the application window's proxy icon when the current file
// has been modified.
updateDocumentEdited() {
const activePaneItem = this.getActivePaneItem();
const modified =
activePaneItem != null && typeof activePaneItem.isModified === 'function'
? activePaneItem.isModified() || false
: false;
this.applicationDelegate.setWindowDocumentEdited(modified);
}
/*
Section: Event Subscription
*/
onDidChangeActivePaneContainer(callback) {
return this.emitter.on('did-change-active-pane-container', callback);
}
// Essential: Invoke the given callback with all current and future text
// editors in the workspace.
//
// * `callback` {Function} to be called with current and future text editors.
// * `editor` A {TextEditor} that is present in {::getTextEditors} at the time
// of subscription or that is added at some later time.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observeTextEditors(callback) {
for (let textEditor of this.getTextEditors()) {
callback(textEditor);
}
return this.onDidAddTextEditor(({ textEditor }) => callback(textEditor));
}
// Essential: Invoke the given callback with all current and future panes items
// in the workspace.
//
// * `callback` {Function} to be called with current and future pane items.
// * `item` An item that is present in {::getPaneItems} at the time of
// subscription or that is added at some later time.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observePaneItems(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.observePaneItems(callback)
)
);
}
// Essential: Invoke the given callback when the active pane item changes.
//
// Because observers are invoked synchronously, it's important not to perform
// any expensive operations via this method. Consider
// {::onDidStopChangingActivePaneItem} to delay operations until after changes
// stop occurring.
//
// * `callback` {Function} to be called when the active pane item changes.
// * `item` The active pane item.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidChangeActivePaneItem(callback) {
return this.emitter.on('did-change-active-pane-item', callback);
}
// Essential: Invoke the given callback when the active pane item stops
// changing.
//
// Observers are called asynchronously 100ms after the last active pane item
// change. Handling changes here rather than in the synchronous
// {::onDidChangeActivePaneItem} prevents unneeded work if the user is quickly
// changing or closing tabs and ensures critical UI feedback, like changing the
// highlighted tab, gets priority over work that can be done asynchronously.
//
// * `callback` {Function} to be called when the active pane item stops
// changing.
// * `item` The active pane item.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidStopChangingActivePaneItem(callback) {
return this.emitter.on('did-stop-changing-active-pane-item', callback);
}
// Essential: Invoke the given callback when a text editor becomes the active
// text editor and when there is no longer an active text editor.
//
// * `callback` {Function} to be called when the active text editor changes.
// * `editor` The active {TextEditor} or undefined if there is no longer an
// active text editor.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidChangeActiveTextEditor(callback) {
return this.emitter.on('did-change-active-text-editor', callback);
}
// Essential: Invoke the given callback with the current active pane item and
// with all future active pane items in the workspace.
//
// * `callback` {Function} to be called when the active pane item changes.
// * `item` The current active pane item.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observeActivePaneItem(callback) {
callback(this.getActivePaneItem());
return this.onDidChangeActivePaneItem(callback);
}
// Essential: Invoke the given callback with the current active text editor
// (if any), with all future active text editors, and when there is no longer
// an active text editor.
//
// * `callback` {Function} to be called when the active text editor changes.
// * `editor` The active {TextEditor} or undefined if there is not an
// active text editor.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observeActiveTextEditor(callback) {
callback(this.getActiveTextEditor());
return this.onDidChangeActiveTextEditor(callback);
}
// Essential: Invoke the given callback whenever an item is opened. Unlike
// {::onDidAddPaneItem}, observers will be notified for items that are already
// present in the workspace when they are reopened.
//
// * `callback` {Function} to be called whenever an item is opened.
// * `event` {Object} with the following keys:
// * `uri` {String} representing the opened URI. Could be `undefined`.
// * `item` The opened item.
// * `pane` The pane in which the item was opened.
// * `index` The index of the opened item on its pane.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidOpen(callback) {
return this.emitter.on('did-open', callback);
}
// Extended: Invoke the given callback when a pane is added to the workspace.
//
// * `callback` {Function} to be called panes are added.
// * `event` {Object} with the following keys:
// * `pane` The added pane.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidAddPane(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onDidAddPane(callback)
)
);
}
// Extended: Invoke the given callback before a pane is destroyed in the
// workspace.
//
// * `callback` {Function} to be called before panes are destroyed.
// * `event` {Object} with the following keys:
// * `pane` The pane to be destroyed.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onWillDestroyPane(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onWillDestroyPane(callback)
)
);
}
// Extended: Invoke the given callback when a pane is destroyed in the
// workspace.
//
// * `callback` {Function} to be called panes are destroyed.
// * `event` {Object} with the following keys:
// * `pane` The destroyed pane.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidDestroyPane(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onDidDestroyPane(callback)
)
);
}
// Extended: Invoke the given callback with all current and future panes in the
// workspace.
//
// * `callback` {Function} to be called with current and future panes.
// * `pane` A {Pane} that is present in {::getPanes} at the time of
// subscription or that is added at some later time.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observePanes(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.observePanes(callback)
)
);
}
// Extended: Invoke the given callback when the active pane changes.
//
// * `callback` {Function} to be called when the active pane changes.
// * `pane` A {Pane} that is the current return value of {::getActivePane}.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidChangeActivePane(callback) {
return this.emitter.on('did-change-active-pane', callback);
}
// Extended: Invoke the given callback with the current active pane and when
// the active pane changes.
//
// * `callback` {Function} to be called with the current and future active#
// panes.
// * `pane` A {Pane} that is the current return value of {::getActivePane}.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
observeActivePane(callback) {
callback(this.getActivePane());
return this.onDidChangeActivePane(callback);
}
// Extended: Invoke the given callback when a pane item is added to the
// workspace.
//
// * `callback` {Function} to be called when pane items are added.
// * `event` {Object} with the following keys:
// * `item` The added pane item.
// * `pane` {Pane} containing the added item.
// * `index` {Number} indicating the index of the added item in its pane.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidAddPaneItem(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onDidAddPaneItem(callback)
)
);
}
// Extended: Invoke the given callback when a pane item is about to be
// destroyed, before the user is prompted to save it.
//
// * `callback` {Function} to be called before pane items are destroyed. If this function returns
// a {Promise}, then the item will not be destroyed until the promise resolves.
// * `event` {Object} with the following keys:
// * `item` The item to be destroyed.
// * `pane` {Pane} containing the item to be destroyed.
// * `index` {Number} indicating the index of the item to be destroyed in
// its pane.
//
// Returns a {Disposable} on which `.dispose` can be called to unsubscribe.
onWillDestroyPaneItem(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onWillDestroyPaneItem(callback)
)
);
}
// Extended: Invoke the given callback when a pane item is destroyed.
//
// * `callback` {Function} to be called when pane items are destroyed.
// * `event` {Object} with the following keys:
// * `item` The destroyed item.
// * `pane` {Pane} containing the destroyed item.
// * `index` {Number} indicating the index of the destroyed item in its
// pane.
//
// Returns a {Disposable} on which `.dispose` can be called to unsubscribe.
onDidDestroyPaneItem(callback) {
return new CompositeDisposable(
...this.getPaneContainers().map(container =>
container.onDidDestroyPaneItem(callback)
)
);
}
// Extended: Invoke the given callback when a text editor is added to the
// workspace.
//
// * `callback` {Function} to be called panes are added.
// * `event` {Object} with the following keys:
// * `textEditor` {TextEditor} that was added.
// * `pane` {Pane} containing the added text editor.
// * `index` {Number} indicating the index of the added text editor in its
// pane.
//
// Returns a {Disposable} on which `.dispose()` can be called to unsubscribe.
onDidAddTextEditor(callback) {
return this.emitter.on('did-add-text-editor', callback);
}
onDidChangeWindowTitle(callback) {
return this.emitter.on('did-change-window-title', callback);
}
/*
Section: Opening
*/
// Essential: Opens the given URI in Atom asynchronously.
// If the URI is already open, the existing item for that URI will be
// activated. If no URI is given, or no registered opener can open
// the URI, a new empty {TextEditor} will be created.
//
// * `uri` (optional) A {String} containing a URI.
// * `options` (optional) {Object}
// * `initialLine` A {Number} indicating which row to move the cursor to
// initially. Defaults to `0`.
// * `initialColumn` A {Number} indicating which column to move the cursor to
// initially. Defaults to `0`.
// * `split` Either 'left', 'right', 'up' or 'down'.
// If 'left', the item will be opened in leftmost pane of the current active pane's row.
// If 'right', the item will be opened in the rightmost pane of the current active pane's row. If only one pane exists in the row, a new pane will be created.
// If 'up', the item will be opened in topmost pane of the current active pane's column.
// If 'down', the item will be opened in the bottommost pane of the current active pane's column. If only one pane exists in the column, a new pane will be created.
// * `activatePane` A {Boolean} indicating whether to call {Pane::activate} on
// containing pane. Defaults to `true`.
// * `activateItem` A {Boolean} indicating whether to call {Pane::activateItem}
// on containing pane. Defaults to `true`.
// * `pending` A {Boolean} indicating whether or not the item should be opened
// in a pending state. Existing pending items in a pane are replaced with
// new pending items when they are opened.
// * `searchAllPanes` A {Boolean}. If `true`, the workspace will attempt to
// activate an existing item for the given URI on any pane.
// If `false`, only the active pane will be searched for
// an existing item for the same URI. Defaults to `false`.
// * `location` (optional) A {String} containing the name of the location
// in which this item should be opened (one of "left", "right", "bottom",
// or "center"). If omitted, Atom will fall back to the last location in
// which a user has placed an item with the same URI or, if this is a new
// URI, the default location specified by the item. NOTE: This option
// should almost always be omitted to honor user preference.
//
// Returns a {Promise} that resolves to the {TextEditor} for the file URI.
async open(itemOrURI, options = {}) {
let uri, item;
if (typeof itemOrURI === 'string') {
uri = this.project.resolvePath(itemOrURI);
} else if (itemOrURI) {
item = itemOrURI;
if (typeof item.getURI === 'function') uri = item.getURI();
}
let resolveItem = () => {};
if (uri) {
const incomingItem = this.incoming.get(uri);
if (!incomingItem) {
this.incoming.set(
uri,
new Promise(resolve => {
resolveItem = resolve;
})
);
} else {
await incomingItem;
}
}
try {
if (!atom.config.get('core.allowPendingPaneItems')) {
options.pending = false;
}
// Avoid adding URLs as recent documents to work-around this Spotlight crash:
// https://github.com/atom/atom/issues/10071
if (uri && (!url.parse(uri).protocol || process.platform === 'win32')) {
this.applicationDelegate.addRecentDocument(uri);
}
let pane, itemExistsInWorkspace;
// Try to find an existing item in the workspace.
if (item || uri) {
if (options.pane) {
pane = options.pane;
} else if (options.searchAllPanes) {
pane = item ? this.paneForItem(item) : this.paneForURI(uri);
} else {
// If an item with the given URI is already in the workspace, assume
// that item's pane container is the preferred location for that URI.
let container;
if (uri) container = this.paneContainerForURI(uri);
if (!container) container = this.getActivePaneContainer();
// The `split` option affects where we search for the item.
pane = container.getActivePane();
switch (options.split) {
case 'left':
pane = pane.findLeftmostSibling();
break;
case 'right':
pane = pane.findRightmostSibling();
break;
case 'up':
pane = pane.findTopmostSibling();
break;
case 'down':
pane = pane.findBottommostSibling();
break;
}
}
if (pane) {
if (item) {
itemExistsInWorkspace = pane.getItems().includes(item);
} else {
item = pane.itemForURI(uri);
itemExistsInWorkspace = item != null;
}
}
}
// If we already have an item at this stage, we won't need to do an async
// lookup of the URI, so we yield the event loop to ensure this method
// is consistently asynchronous.
if (item) await Promise.resolve();
if (!itemExistsInWorkspace) {
item = item || (await this.createItemForURI(uri, options));
if (!item) return;
if (options.pane) {
pane = options.pane;
} else {
let location = options.location;
if (!location && !options.split && uri && this.enablePersistence) {
location = await this.itemLocationStore.load(uri);
}
if (!location && typeof item.getDefaultLocation === 'function') {
location = item.getDefaultLocation();
}
const allowedLocations =
typeof item.getAllowedLocations === 'function'
? item.getAllowedLocations()
: ALL_LOCATIONS;
location = allowedLocations.includes(location)
? location
: allowedLocations[0];
const container = this.paneContainers[location] || this.getCenter();
pane = container.getActivePane();
switch (options.split) {
case 'left':
pane = pane.findLeftmostSibling();
break;
case 'right':
pane = pane.findOrCreateRightmostSibling();
break;
case 'up':
pane = pane.findTopmostSibling();
break;
case 'down':
pane = pane.findOrCreateBottommostSibling();
break;
}
}
}
if (!options.pending && pane.getPendingItem() === item) {
pane.clearPendingItem();
}
this.itemOpened(item);
if (options.activateItem === false) {
pane.addItem(item, { pending: options.pending });
} else {
pane.activateItem(item, { pending: options.pending });
}
if (options.activatePane !== false) {
pane.activate();
}
let initialColumn = 0;
let initialLine = 0;
if (!Number.isNaN(options.initialLine)) {
initialLine = options.initialLine;
}
if (!Number.isNaN(options.initialColumn)) {
initialColumn = options.initialColumn;
}
if (initialLine >= 0 || initialColumn >= 0) {
if (typeof item.setCursorBufferPosition === 'function') {
item.setCursorBufferPosition([initialLine, initialColumn]);
}
if (typeof item.unfoldBufferRow === 'function') {
item.unfoldBufferRow(initialLine);
}
if (typeof item.scrollToBufferPosition === 'function') {
item.scrollToBufferPosition([initialLine, initialColumn], {
center: true
});
}
}
const index = pane.getActiveItemIndex();
this.emitter.emit('did-open', { uri, pane, item, index });
if (uri) {
this.incoming.delete(uri);
}
} finally {
resolveItem();
}
return item;
}
// Essential: Search the workspace for items matching the given URI and hide them.
//
// * `itemOrURI` The item to hide or a {String} containing the URI
// of the item to hide.
//
// Returns a {Boolean} indicating whether any items were found (and hidden).
hide(itemOrURI) {
let foundItems = false;
// If any visible item has the given URI, hide it
for (const container of this.getPaneContainers()) {
const isCenter = container === this.getCenter();
if (isCenter || container.isVisible()) {
for (const pane of container.getPanes()) {
const activeItem = pane.getActiveItem();
const foundItem =
activeItem != null &&
(activeItem === itemOrURI ||
(typeof activeItem.getURI === 'function' &&
activeItem.getURI() === itemOrURI));
if (foundItem) {
foundItems = true;
// We can't really hide the center so we just destroy the item.
if (isCenter) {
pane.destroyItem(activeItem);
} else {
container.hide();
}
}
}
}
}
return foundItems;
}
// Essential: Search the workspace for items matching the given URI. If any are found, hide them.
// Otherwise, open the URL.
//
// * `itemOrURI` (optional) The item to toggle or a {String} containing the URI
// of the item to toggle.
//
// Returns a Promise that resolves when the item is shown or hidden.
toggle(itemOrURI) {
if (this.hide(itemOrURI)) {
return Promise.resolve();
} else {
return this.open(itemOrURI, { searchAllPanes: true });
}
}
// Open Atom's license in the active pane.
openLicense() {
return this.open(path.join(process.resourcesPath, 'LICENSE.md'));
}
// Synchronously open the given URI in the active pane. **Only use this method
// in specs. Calling this in production code will block the UI thread and
// everyone will be mad at you.**
//
// * `uri` A {String} containing a URI.
// * `options` An optional options {Object}
// * `initialLine` A {Number} indicating which row to move the cursor to
// initially. Defaults to `0`.
// * `initialColumn` A {Number} indicating which column to move the cursor to
// initially. Defaults to `0`.
// * `activatePane` A {Boolean} indicating whether to call {Pane::activate} on
// the containing pane. Defaults to `true`.
// * `activateItem` A {Boolean} indicating whether to call {Pane::activateItem}
// on containing pane. Defaults to `true`.
openSync(uri_ = '', options = {}) {
const { initialLine, initialColumn } = options;
const activatePane =
options.activatePane != null ? options.activatePane : true;
const activateItem =
options.activateItem != null ? options.activateItem : true;
const uri = this.project.resolvePath(uri_);
let item = this.getActivePane().itemForURI(uri);
if (uri && item == null) {
for (const opener of this.getOpeners()) {
item = opener(uri, options);
if (item) break;
}
}
if (item == null) {
item = this.project.openSync(uri, { initialLine, initialColumn });
}
if (activateItem) {
this.getActivePane().activateItem(item);
}
this.itemOpened(item);
if (activatePane) {
this.getActivePane().activate();
}
return item;
}
openURIInPane(uri, pane) {
return this.open(uri, { pane });
}
// Public: Creates a new item that corresponds to the provided URI.
//
// If no URI is given, or no registered opener can open the URI, a new empty
// {TextEditor} will be created.
//
// * `uri` A {String} containing a URI.
//
// Returns a {Promise} that resolves to the {TextEditor} (or other item) for the given URI.
async createItemForURI(uri, options) {
if (uri != null) {
for (const opener of this.getOpeners()) {
const item = opener(uri, options);
if (item != null) return item;
}
}
try {
const item = await this.openTextFile(uri, options);
return item;
} catch (error) {
switch (error.code) {
case 'CANCELLED':
return Promise.resolve();
case 'EACCES':
this.notificationManager.addWarning(
`Permission denied '${error.path}'`
);
return Promise.resolve();
case 'EPERM':
case 'EBUSY':
case 'ENXIO':
case 'EIO':
case 'ENOTCONN':
case 'UNKNOWN':
case 'ECONNRESET':
case 'EINVAL':
case 'EMFILE':
case 'ENOTDIR':
case 'EAGAIN':
this.notificationManager.addWarning(
`Unable to open '${error.path != null ? error.path : uri}'`,
{ detail: error.message }
);
return Promise.resolve();
default:
throw error;
}
}
}
async openTextFile(uri, options) {
const filePath = this.project.resolvePath(uri);
if (filePath != null) {
try {
fs.closeSync(fs.openSync(filePath, 'r'));
} catch (error) {
// allow ENOENT errors to create an editor for paths that dont exist
if (error.code !== 'ENOENT') {
throw error;
}
}
}
const fileSize = fs.getSizeSync(filePath);
if (fileSize >= this.config.get('core.warnOnLargeFileLimit') * 1048576) {
// 40MB by default
await new Promise((resolve, reject) => {
this.applicationDelegate.confirm(
{
message:
'Atom will be unresponsive during the loading of very large files.',
detail: 'Do you still want to load this file?',
buttons: ['Proceed', 'Cancel']
},
response => {
if (response === 1) {
const error = new Error();
error.code = 'CANCELLED';
reject(error);
} else {
resolve();
}
}
);
});
}
const buffer = await this.project.bufferForPath(filePath, options);
return this.textEditorRegistry.build(
Object.assign({ buffer, autoHeight: false }, options)
);
}
handleGrammarUsed(grammar) {
if (grammar == null) {
return;
}
this.packageManager.triggerActivationHook(
`${grammar.scopeName}:root-scope-used`
);
this.packageManager.triggerActivationHook(
`${grammar.packageName}:grammar-used`
);
}
// Public: Returns a {Boolean} that is `true` if `object` is a `TextEditor`.
//
// * `object` An {Object} you want to perform the check against.
isTextEditor(object) {
return object instanceof TextEditor;
}
// Extended: Create a new text editor.
//
// Returns a {TextEditor}.
buildTextEditor(params) {
const editor = this.textEditorRegistry.build(params);
const subscription = this.textEditorRegistry.maintainConfig(editor);
editor.onDidDestroy(() => subscription.dispose());
return editor;
}
// Public: Asynchronously reopens the last-closed item's URI if it hasn't already been
// reopened.
//
// Returns a {Promise} that is resolved when the item is opened
reopenItem() {
const uri = this.destroyedItemURIs.pop();
if (uri) {
return this.open(uri);
} else {
return Promise.resolve();
}
}
// Public: Register an opener for a uri.
//
// When a URI is opened via {Workspace::open}, Atom loops through its registered
// opener functions until one returns a value for the given uri.
// Openers are expected to return an object that inherits from HTMLElement or
// a model which has an associated view in the {ViewRegistry}.
// A {TextEditor} will be used if no opener returns a value.
//
// ## Examples
//
// ```coffee
// atom.workspace.addOpener (uri) ->
// if path.extname(uri) is '.toml'
// return new TomlEditor(uri)
// ```
//
// * `opener` A {Function} to be called when a path is being opened.
//
// Returns a {Disposable} on which `.dispose()` can be called to remove the
// opener.
//
// Note that the opener will be called if and only if the URI is not already open
// in the current pane. The searchAllPanes flag expands the search from the
// current pane to all panes. If you wish to open a view of a different type for
// a file that is already open, consider changing the protocol of the URI. For
// example, perhaps you wish to preview a rendered version of the file `/foo/bar/baz.quux`
// that is already open in a text editor view. You could signal this by calling
// {Workspace::open} on the URI `quux-preview://foo/bar/baz.quux`. Then your opener
// can check the protocol for quux-preview and only handle those URIs that match.
//
// To defer your package's activation until a specific URL is opened, add a
// `workspaceOpeners` field to your `package.json` containing an array of URL
// strings.
addOpener(opener) {
this.openers.push(opener);
return new Disposable(() => {
_.remove(this.openers, opener);
});
}
getOpeners() {
return this.openers;
}
/*
Section: Pane Items
*/
// Essential: Get all pane items in the workspace.
//
// Returns an {Array} of items.
getPaneItems() {
return _.flatten(
this.getPaneContainers().map(container => container.getPaneItems())
);
}
// Essential: Get the active {Pane}'s active item.
//
// Returns an pane item {Object}.
getActivePaneItem() {
return this.getActivePaneContainer().getActivePaneItem();
}
// Essential: Get all text editors in the workspace.
//
// Returns an {Array} of {TextEditor}s.
getTextEditors() {
return this.getPaneItems().filter(item => item instanceof TextEditor);
}
// Essential: Get the workspace center's active item if it is a {TextEditor}.
//
// Returns a {TextEditor} or `undefined` if the workspace center's current
// active item is not a {TextEditor}.
getActiveTextEditor() {
const activeItem = this.getCenter().getActivePaneItem();
if (activeItem instanceof TextEditor) {
return activeItem;
}
}
// Save all pane items.
saveAll() {
this.getPaneContainers().forEach(container => {
container.saveAll();
});
}
confirmClose(options) {
return Promise.all(
this.getPaneContainers().map(container => container.confirmClose(options))
).then(results => !results.includes(false));
}
// Save the active pane item.
//
// If the active pane item currently has a URI according to the item's
// `.getURI` method, calls `.save` on the item. Otherwise
// {::saveActivePaneItemAs} # will be called instead. This method does nothing
// if the active item does not implement a `.save` method.
saveActivePaneItem() {
return this.getCenter()
.getActivePane()
.saveActiveItem();
}
// Prompt the user for a path and save the active pane item to it.
//
// Opens a native dialog where the user selects a path on disk, then calls
// `.saveAs` on the item with the selected path. This method does nothing if
// the active item does not implement a `.saveAs` method.
saveActivePaneItemAs() {
this.getCenter()
.getActivePane()
.saveActiveItemAs();
}
// Destroy (close) the active pane item.
//
// Removes the active pane item and calls the `.destroy` method on it if one is
// defined.
destroyActivePaneItem() {
return this.getActivePane().destroyActiveItem();
}
/*
Section: Panes
*/
// Extended: Get the most recently focused pane container.
//
// Returns a {Dock} or the {WorkspaceCenter}.
getActivePaneContainer() {
return this.activePaneContainer;
}
// Extended: Get all panes in the workspace.
//
// Returns an {Array} of {Pane}s.
getPanes() {
return _.flatten(
this.getPaneContainers().map(container => container.getPanes())
);
}
getVisiblePanes() {
return _.flatten(
this.getVisiblePaneContainers().map(container => container.getPanes())
);
}
// Extended: Get the active {Pane}.
//
// Returns a {Pane}.
getActivePane() {
return this.getActivePaneContainer().getActivePane();
}
// Extended: Make the next pane active.
activateNextPane() {
return this.getActivePaneContainer().activateNextPane();
}
// Extended: Make the previous pane active.
activatePreviousPane() {
return this.getActivePaneContainer().activatePreviousPane();
}
// Extended: Get the first pane container that contains an item with the given
// URI.
//
// * `uri` {String} uri
//
// Returns a {Dock}, the {WorkspaceCenter}, or `undefined` if no item exists
// with the given URI.
paneContainerForURI(uri) {
return this.getPaneContainers().find(container =>
container.paneForURI(uri)
);
}
// Extended: Get the first pane container that contains the given item.
//
// * `item` the Item that the returned pane container must contain.
//
// Returns a {Dock}, the {WorkspaceCenter}, or `undefined` if no item exists
// with the given URI.
paneContainerForItem(uri) {
return this.getPaneContainers().find(container =>
container.paneForItem(uri)
);
}
// Extended: Get the first {Pane} that contains an item with the given URI.
//
// * `uri` {String} uri
//
// Returns a {Pane} or `undefined` if no item exists with the given URI.
paneForURI(uri) {
for (let location of this.getPaneContainers()) {
const pane = location.paneForURI(uri);
if (pane != null) {
return pane;
}
}
}
// Extended: Get the {Pane} containing the given item.
//
// * `item` the Item that the returned pane must contain.
//
// Returns a {Pane} or `undefined` if no pane exists for the given item.
paneForItem(item) {
for (let location of this.getPaneContainers()) {
const pane = location.paneForItem(item);
if (pane != null) {
return pane;
}
}
}
// Destroy (close) the active pane.
destroyActivePane() {
const activePane = this.getActivePane();
if (activePane != null) {
activePane.destroy();
}
}
// Close the active center pane item, or the active center pane if it is
// empty, or the current window if there is only the empty root pane.
closeActivePaneItemOrEmptyPaneOrWindow() {
if (this.getCenter().getActivePaneItem() != null) {
this.getCenter()
.getActivePane()
.destroyActiveItem();
} else if (this.getCenter().getPanes().length > 1) {
this.getCenter().destroyActivePane();
} else if (this.config.get('core.closeEmptyWindows')) {
atom.close();
}
}
// Increase the editor font size by 1px.
increaseFontSize() {
this.config.set('editor.fontSize', this.config.get('editor.fontSize') + 1);
}
// Decrease the editor font size by 1px.
decreaseFontSize() {
const fontSize = this.config.get('editor.fontSize');
if (fontSize > 1) {
this.config.set('editor.fontSize', fontSize - 1);
}
}
// Restore to the window's original editor font size.
resetFontSize() {
if (this.originalFontSize) {
this.config.set('editor.fontSize', this.originalFontSize);
}
}
// Removes the item's uri from the list of potential items to reopen.
itemOpened(item) {
let uri;
if (typeof item.getURI === 'function') {
uri = item.getURI();
} else if (typeof item.getUri === 'function') {
uri = item.getUri();
}
if (uri != null) {
_.remove(this.destroyedItemURIs, uri);
}
}
// Adds the destroyed item's uri to the list of items to reopen.
didDestroyPaneItem({ item }) {
let uri;
if (typeof item.getURI === 'function') {
uri = item.getURI();
} else if (typeof item.getUri === 'function') {
uri = item.getUri();
}
if (uri != null) {
this.destroyedItemURIs.push(uri);
}
}
// Called by Model superclass when destroyed
destroyed() {
this.paneContainers.center.destroy();
this.paneContainers.left.destroy();
this.paneContainers.right.destroy();
this.paneContainers.bottom.destroy();
this.cancelStoppedChangingActivePaneItemTimeout();
if (this.activeItemSubscriptions != null) {
this.activeItemSubscriptions.dispose();
}
if (this.element) this.element.destroy();
}
/*
Section: Pane Locations
*/
// Essential: Get the {WorkspaceCenter} at the center of the editor window.
getCenter() {
return this.paneContainers.center;
}
// Essential: Get the {Dock} to the left of the editor window.
getLeftDock() {
return this.paneContainers.left;
}
// Essential: Get the {Dock} to the right of the editor window.
getRightDock() {
return this.paneContainers.right;
}
// Essential: Get the {Dock} below the editor window.
getBottomDock() {
return this.paneContainers.bottom;
}
getPaneContainers() {
return [
this.paneContainers.center,
this.paneContainers.left,
this.paneContainers.right,
this.paneContainers.bottom
];
}
getVisiblePaneContainers() {
const center = this.getCenter();
return atom.workspace
.getPaneContainers()
.filter(container => container === center || container.isVisible());
}
/*
Section: Panels
Panels are used to display UI related to an editor window. They are placed at one of the four
edges of the window: left, right, top or bottom. If there are multiple panels on the same window
edge they are stacked in order of priority: higher priority is closer to the center, lower
priority towards the edge.
*Note:* If your panel changes its size throughout its lifetime, consider giving it a higher
priority, allowing fixed size panels to be closer to the edge. This allows control targets to
remain more static for easier targeting by users that employ mice or trackpads. (See
[atom/atom#4834](https://github.com/atom/atom/issues/4834) for discussion.)
*/
// Essential: Get an {Array} of all the panel items at the bottom of the editor window.
getBottomPanels() {
return this.getPanels('bottom');
}
// Essential: Adds a panel item to the bottom of the editor window.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addBottomPanel(options) {
return this.addPanel('bottom', options);
}
// Essential: Get an {Array} of all the panel items to the left of the editor window.
getLeftPanels() {
return this.getPanels('left');
}
// Essential: Adds a panel item to the left of the editor window.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addLeftPanel(options) {
return this.addPanel('left', options);
}
// Essential: Get an {Array} of all the panel items to the right of the editor window.
getRightPanels() {
return this.getPanels('right');
}
// Essential: Adds a panel item to the right of the editor window.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addRightPanel(options) {
return this.addPanel('right', options);
}
// Essential: Get an {Array} of all the panel items at the top of the editor window.
getTopPanels() {
return this.getPanels('top');
}
// Essential: Adds a panel item to the top of the editor window above the tabs.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addTopPanel(options) {
return this.addPanel('top', options);
}
// Essential: Get an {Array} of all the panel items in the header.
getHeaderPanels() {
return this.getPanels('header');
}
// Essential: Adds a panel item to the header.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addHeaderPanel(options) {
return this.addPanel('header', options);
}
// Essential: Get an {Array} of all the panel items in the footer.
getFooterPanels() {
return this.getPanels('footer');
}
// Essential: Adds a panel item to the footer.
//
// * `options` {Object}
// * `item` Your panel content. It can be DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// latter. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
//
// Returns a {Panel}
addFooterPanel(options) {
return this.addPanel('footer', options);
}
// Essential: Get an {Array} of all the modal panel items
getModalPanels() {
return this.getPanels('modal');
}
// Essential: Adds a panel item as a modal dialog.
//
// * `options` {Object}
// * `item` Your panel content. It can be a DOM element, a jQuery element, or
// a model with a view registered via {ViewRegistry::addViewProvider}. We recommend the
// model option. See {ViewRegistry::addViewProvider} for more information.
// * `visible` (optional) {Boolean} false if you want the panel to initially be hidden
// (default: true)
// * `priority` (optional) {Number} Determines stacking order. Lower priority items are
// forced closer to the edges of the window. (default: 100)
// * `autoFocus` (optional) {Boolean|Element} true if you want modal focus managed for you by Atom.
// Atom will automatically focus on this element or your modal panel's first tabbable element when the modal
// opens and will restore the previously selected element when the modal closes. Atom will
// also automatically restrict user tab focus within your modal while it is open.
// (default: false)
//
// Returns a {Panel}
addModalPanel(options = {}) {
return this.addPanel('modal', options);
}
// Essential: Returns the {Panel} associated with the given item. Returns
// `null` when the item has no panel.
//
// * `item` Item the panel contains
panelForItem(item) {
for (let location in this.panelContainers) {
const container = this.panelContainers[location];
const panel = container.panelForItem(item);
if (panel != null) {
return panel;
}
}
return null;
}
getPanels(location) {
return this.panelContainers[location].getPanels();
}
addPanel(location, options) {
if (options == null) {
options = {};
}
return this.panelContainers[location].addPanel(
new Panel(options, this.viewRegistry)
);
}
/*
Section: Searching and Replacing
*/
// Public: Performs a search across all files in the workspace.
//
// * `regex` {RegExp} to search with.
// * `options` (optional) {Object}
// * `paths` An {Array} of glob patterns to search within.
// * `onPathsSearched` (optional) {Function} to be periodically called
// with number of paths searched.
// * `leadingContextLineCount` {Number} default `0`; The number of lines
// before the matched line to include in the results object.
// * `trailingContextLineCount` {Number} default `0`; The number of lines
// after the matched line to include in the results object.
// * `iterator` {Function} callback on each file found.
//
// Returns a {Promise} with a `cancel()` method that will cancel all
// of the underlying searches that were started as part of this scan.
scan(regex, options = {}, iterator) {
if (_.isFunction(options)) {
iterator = options;
options = {};
}
// Find a searcher for every Directory in the project. Each searcher that is matched
// will be associated with an Array of Directory objects in the Map.
const directoriesForSearcher = new Map();
for (const directory of this.project.getDirectories()) {
let searcher = options.ripgrep
? this.ripgrepDirectorySearcher
: this.scandalDirectorySearcher;
for (const directorySearcher of this.directorySearchers) {
if (directorySearcher.canSearchDirectory(directory)) {
searcher = directorySearcher;
break;
}
}
let directories = directoriesForSearcher.get(searcher);
if (!directories) {
directories = [];
directoriesForSearcher.set(searcher, directories);
}
directories.push(directory);
}
// Define the onPathsSearched callback.
let onPathsSearched;
if (_.isFunction(options.onPathsSearched)) {
// Maintain a map of directories to the number of search results. When notified of a new count,
// replace the entry in the map and update the total.
const onPathsSearchedOption = options.onPathsSearched;
let totalNumberOfPathsSearched = 0;
const numberOfPathsSearchedForSearcher = new Map();
onPathsSearched = function(searcher, numberOfPathsSearched) {
const oldValue = numberOfPathsSearchedForSearcher.get(searcher);
if (oldValue) {
totalNumberOfPathsSearched -= oldValue;
}
numberOfPathsSearchedForSearcher.set(searcher, numberOfPathsSearched);
totalNumberOfPathsSearched += numberOfPathsSearched;
return onPathsSearchedOption(totalNumberOfPathsSearched);
};
} else {
onPathsSearched = function() {};
}
// Kick off all of the searches and unify them into one Promise.
const allSearches = [];
directoriesForSearcher.forEach((directories, searcher) => {
const searchOptions = {
inclusions: options.paths || [],
includeHidden: true,
excludeVcsIgnores: this.config.get('core.excludeVcsIgnoredPaths'),
exclusions: this.config.get('core.ignoredNames'),
follow: this.config.get('core.followSymlinks'),
leadingContextLineCount: options.leadingContextLineCount || 0,
trailingContextLineCount: options.trailingContextLineCount || 0,
didMatch: result => {
if (!this.project.isPathModified(result.filePath)) {
return iterator(result);
}
},
didError(error) {
return iterator(null, error);
},
didSearchPaths(count) {
return onPathsSearched(searcher, count);
}
};
const directorySearcher = searcher.search(
directories,
regex,
searchOptions
);
allSearches.push(directorySearcher);
});
const searchPromise = Promise.all(allSearches);
for (let buffer of this.project.getBuffers()) {
if (buffer.isModified()) {
const filePath = buffer.getPath();
if (!this.project.contains(filePath)) {
continue;
}
var matches = [];
buffer.scan(regex, match => matches.push(match));
if (matches.length > 0) {
iterator({ filePath, matches });
}
}
}
// Make sure the Promise that is returned to the client is cancelable. To be consistent
// with the existing behavior, instead of cancel() rejecting the promise, it should
// resolve it with the special value 'cancelled'. At least the built-in find-and-replace
// package relies on this behavior.
let isCancelled = false;
const cancellablePromise = new Promise((resolve, reject) => {
const onSuccess = function() {
if (isCancelled) {
resolve('cancelled');
} else {
resolve(null);
}
};
const onFailure = function(error) {
for (let promise of allSearches) {
promise.cancel();
}
reject(error);
};
searchPromise.then(onSuccess, onFailure);
});
cancellablePromise.cancel = () => {
isCancelled = true;
// Note that cancelling all of the members of allSearches will cause all of the searches
// to resolve, which causes searchPromise to resolve, which is ultimately what causes
// cancellablePromise to resolve.
allSearches.map(promise => promise.cancel());
};
return cancellablePromise;
}
// Public: Performs a replace across all the specified files in the project.
//
// * `regex` A {RegExp} to search with.
// * `replacementText` {String} to replace all matches of regex with.
// * `filePaths` An {Array} of file path strings to run the replace on.
// * `iterator` A {Function} callback on each file with replacements:
// * `options` {Object} with keys `filePath` and `replacements`.
//
// Returns a {Promise}.
replace(regex, replacementText, filePaths, iterator) {
return new Promise((resolve, reject) => {
let buffer;
const openPaths = this.project
.getBuffers()
.map(buffer => buffer.getPath());
const outOfProcessPaths = _.difference(filePaths, openPaths);
let inProcessFinished = !openPaths.length;
let outOfProcessFinished = !outOfProcessPaths.length;
const checkFinished = () => {
if (outOfProcessFinished && inProcessFinished) {
resolve();
}
};
if (!outOfProcessFinished.length) {
let flags = 'g';
if (regex.multiline) {
flags += 'm';
}
if (regex.ignoreCase) {
flags += 'i';
}
const task = Task.once(
require.resolve('./replace-handler'),
outOfProcessPaths,
regex.source,
flags,
replacementText,
() => {
outOfProcessFinished = true;
checkFinished();
}
);
task.on('replace:path-replaced', iterator);
task.on('replace:file-error', error => {
iterator(null, error);
});
}
for (buffer of this.project.getBuffers()) {
if (!filePaths.includes(buffer.getPath())) {
continue;
}
const replacements = buffer.replace(regex, replacementText, iterator);
if (replacements) {
iterator({ filePath: buffer.getPath(), replacements });
}
}
inProcessFinished = true;
checkFinished();
});
}
checkoutHeadRevision(editor) {
if (editor.getPath()) {
const checkoutHead = async () => {
const repository = await this.project.repositoryForDirectory(
new Directory(editor.getDirectoryPath())
);
if (repository) repository.checkoutHeadForEditor(editor);
};
if (this.config.get('editor.confirmCheckoutHeadRevision')) {
this.applicationDelegate.confirm(
{
message: 'Confirm Checkout HEAD Revision',
detail: `Are you sure you want to discard all changes to "${editor.getFileName()}" since the last Git commit?`,
buttons: ['OK', 'Cancel']
},
response => {
if (response === 0) checkoutHead();
}
);
} else {
checkoutHead();
}
}
}
};