searchWidget.ts 14.0 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.
 *--------------------------------------------------------------------------------------------*/

6 7 8
import * as nls from 'vs/nls';
import * as strings from 'vs/base/common/strings';
import * as dom from 'vs/base/browser/dom';
9 10 11
import { TPromise } from 'vs/base/common/winjs.base';
import { Widget } from 'vs/base/browser/ui/widget';
import { Action } from 'vs/base/common/actions';
S
Sandeep Somavarapu 已提交
12
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
13 14 15 16
import { FindInput, IFindInputOptions } from 'vs/base/browser/ui/findinput/findInput';
import { InputBox } from 'vs/base/browser/ui/inputbox/inputBox';
import { Button } from 'vs/base/browser/ui/button/button';
import { IKeyboardEvent } from 'vs/base/browser/keyboardEvent';
17
import { KeybindingsRegistry } from 'vs/platform/keybinding/common/keybindingsRegistry';
18
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
19
import { ContextKeyExpr, IContextKeyService, IContextKey } from 'vs/platform/contextkey/common/contextkey';
20
import { IContextViewService } from 'vs/platform/contextview/browser/contextView';
21
import { KeyCode, KeyMod } from 'vs/base/common/keyCodes';
22 23 24
import Event, { Emitter } from 'vs/base/common/event';
import { Builder } from 'vs/base/browser/builder';
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
B
Benjamin Pasero 已提交
25
import { IViewletService } from 'vs/workbench/services/viewlet/browser/viewlet';
26
import { isSearchViewletFocussed, appendKeyBindingLabel } from 'vs/workbench/parts/search/browser/searchActions';
A
Alex Dima 已提交
27
import { CONTEXT_FIND_WIDGET_NOT_VISIBLE } from 'vs/editor/contrib/find/common/findController';
28
import { HistoryNavigator } from 'vs/base/common/history';
29
import * as Constants from 'vs/workbench/parts/search/common/constants';
B
Benjamin Pasero 已提交
30
import { attachInputBoxStyler, attachFindInputBoxStyler } from 'vs/platform/theme/common/styler';
B
Benjamin Pasero 已提交
31
import { IThemeService } from 'vs/platform/theme/common/themeService';
32 33

export interface ISearchWidgetOptions {
J
Johannes Rieken 已提交
34 35 36 37
	value?: string;
	isRegex?: boolean;
	isCaseSensitive?: boolean;
	isWholeWords?: boolean;
38 39
}

40 41
class ReplaceAllAction extends Action {

J
Johannes Rieken 已提交
42 43
	private static fgInstance: ReplaceAllAction = null;
	public static ID: string = 'search.action.replaceAll';
44

J
Johannes Rieken 已提交
45
	static get INSTANCE(): ReplaceAllAction {
46
		if (ReplaceAllAction.fgInstance === null) {
J
Johannes Rieken 已提交
47
			ReplaceAllAction.fgInstance = new ReplaceAllAction();
48 49 50 51
		}
		return ReplaceAllAction.fgInstance;
	}

J
Johannes Rieken 已提交
52
	private _searchWidget: SearchWidget = null;
53 54 55 56 57 58

	constructor() {
		super(ReplaceAllAction.ID, '', 'action-replace-all', false);
	}

	set searchWidget(searchWidget: SearchWidget) {
J
Johannes Rieken 已提交
59
		this._searchWidget = searchWidget;
60 61
	}

J
Johannes Rieken 已提交
62
	run(): TPromise<any> {
63 64 65 66 67 68 69
		if (this._searchWidget) {
			return this._searchWidget.triggerReplaceAll();
		}
		return TPromise.as(null);
	}
}

70 71
export class SearchWidget extends Widget {

J
Johannes Rieken 已提交
72 73
	private static REPLACE_ALL_DISABLED_LABEL = nls.localize('search.action.replaceAll.disabled.label', "Replace All (Submit Search to Enable)");
	private static REPLACE_ALL_ENABLED_LABEL = (keyBindingService2: IKeybindingService): string => {
74
		let kb = keyBindingService2.lookupKeybinding(ReplaceAllAction.ID);
B
Benjamin Pasero 已提交
75
		return appendKeyBindingLabel(nls.localize('search.action.replaceAll.enabled.label', "Replace All"), kb, keyBindingService2);
E
Erich Gamma 已提交
76
	}
S
Sandeep Somavarapu 已提交
77

78 79
	public domNode: HTMLElement;
	public searchInput: FindInput;
80
	private searchInputBoxFocussed: IContextKey<boolean>;
S
Sandeep Somavarapu 已提交
81
	private replaceInputBoxFocussed: IContextKey<boolean>;
82 83
	private replaceInput: InputBox;

84 85 86
	public searchInputFocusTracker: dom.IFocusTracker;
	public replaceInputFocusTracker: dom.IFocusTracker;

S
Sandeep Somavarapu 已提交
87
	private replaceContainer: HTMLElement;
88
	private toggleReplaceButton: Button;
89
	private replaceAllAction: ReplaceAllAction;
A
Alex Dima 已提交
90
	private replaceActive: IContextKey<boolean>;
S
Sandeep Somavarapu 已提交
91
	private replaceActionBar: ActionBar;
92

93 94
	private searchHistory: HistoryNavigator<string>;

95 96 97 98 99 100 101 102 103
	private _onSearchSubmit = this._register(new Emitter<boolean>());
	public onSearchSubmit: Event<boolean> = this._onSearchSubmit.event;

	private _onSearchCancel = this._register(new Emitter<void>());
	public onSearchCancel: Event<void> = this._onSearchCancel.event;

	private _onReplaceToggled = this._register(new Emitter<void>());
	public onReplaceToggled: Event<void> = this._onReplaceToggled.event;

S
Sandeep Somavarapu 已提交
104 105
	private _onReplaceStateChange = this._register(new Emitter<boolean>());
	public onReplaceStateChange: Event<boolean> = this._onReplaceStateChange.event;
106 107 108 109 110 111 112

	private _onReplaceValueChanged = this._register(new Emitter<string>());
	public onReplaceValueChanged: Event<string> = this._onReplaceValueChanged.event;

	private _onReplaceAll = this._register(new Emitter<void>());
	public onReplaceAll: Event<void> = this._onReplaceAll.event;

B
Benjamin Pasero 已提交
113
	constructor(container: Builder, private contextViewService: IContextViewService, private themeService: IThemeService, options: ISearchWidgetOptions = Object.create(null),
J
Johannes Rieken 已提交
114
		private keyBindingService: IContextKeyService, private keyBindingService2: IKeybindingService, private instantiationService: IInstantiationService) {
115
		super();
116
		this.searchHistory = new HistoryNavigator<string>();
117 118
		this.replaceActive = Constants.ReplaceActiveKey.bindTo(this.keyBindingService);
		this.searchInputBoxFocussed = Constants.SearchInputBoxFocussedKey.bindTo(this.keyBindingService);
S
Sandeep Somavarapu 已提交
119
		this.replaceInputBoxFocussed = Constants.ReplaceInputBoxFocussedKey.bindTo(this.keyBindingService);
120 121 122
		this.render(container, options);
	}

J
Johannes Rieken 已提交
123
	public focus(select: boolean = true, focusReplace: boolean = false): void {
124
		if ((!focusReplace && this.searchInput.inputBox.hasFocus())
J
Johannes Rieken 已提交
125
			|| (focusReplace && this.replaceInput.hasFocus())) {
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
			return;
		}

		if (focusReplace && this.isReplaceShown()) {
			this.replaceInput.focus();
			if (select) {
				this.replaceInput.select();
			}
		} else {
			this.searchInput.focus();
			if (select) {
				this.searchInput.select();
			}
		}
	}

	public setWidth(width: number) {
		this.searchInput.setWidth(width - 2);
J
Johannes Rieken 已提交
144
		this.replaceInput.width = width - 28;
145 146 147 148
	}

	public clear() {
		this.searchInput.clear();
J
Johannes Rieken 已提交
149
		this.replaceInput.value = '';
S
Sandeep Somavarapu 已提交
150
		this.setReplaceAllActionState(false);
151 152 153
	}

	public isReplaceShown(): boolean {
S
Sandeep Somavarapu 已提交
154
		return !dom.hasClass(this.replaceContainer, 'disabled');
155 156
	}

J
Johannes Rieken 已提交
157
	public getReplaceValue(): string {
S
Sandeep Somavarapu 已提交
158
		return this.replaceInput.value;
159 160
	}

J
Johannes Rieken 已提交
161
	public toggleReplace(show?: boolean): void {
162 163
		if (show === void 0 || show !== this.isReplaceShown()) {
			this.onToggleReplaceButton();
S
Sandeep Somavarapu 已提交
164 165 166
		}
	}

167 168 169 170 171 172 173 174
	public showNextSearchTerm() {
		let next = this.searchHistory.next();
		if (next) {
			this.searchInput.setValue(next);
		}
	}

	public showPreviousSearchTerm() {
175
		let previous;
176
		if (this.searchInput.getValue().length === 0) {
W
Wim Spaargaren 已提交
177 178
			previous = this.searchHistory.current();
		} else {
179
			this.searchHistory.addIfNotPresent(this.searchInput.getValue());
W
Wim Spaargaren 已提交
180
			previous = this.searchHistory.previous();
181
		}
182 183 184 185 186
		if (previous) {
			this.searchInput.setValue(previous);
		}
	}

187 188 189 190 191 192 193 194
	public searchInputHasFocus(): boolean {
		return this.searchInputBoxFocussed.get();
	}

	public replaceInputHasFocus(): boolean {
		return this.replaceInput.hasFocus();
	}

195 196 197 198 199 200 201 202 203
	private render(container: Builder, options: ISearchWidgetOptions): void {
		this.domNode = container.div({ 'class': 'search-widget' }).style({ position: 'relative' }).getHTMLElement();
		this.renderToggleReplaceButton(this.domNode);

		this.renderSearchInput(this.domNode, options);
		this.renderReplaceInput(this.domNode);
	}

	private renderToggleReplaceButton(parent: HTMLElement): void {
J
Johannes Rieken 已提交
204 205
		this.toggleReplaceButton = this._register(new Button(parent));
		this.toggleReplaceButton.icon = 'toggle-replace-button collapse';
A
Alex Dima 已提交
206
		this.toggleReplaceButton.addListener('click', () => this.onToggleReplaceButton());
J
Johannes Rieken 已提交
207
		this.toggleReplaceButton.getElement().title = nls.localize('search.replace.toggle.button.title', "Toggle Replace");
208 209 210 211 212 213
	}

	private renderSearchInput(parent: HTMLElement, options: ISearchWidgetOptions): void {
		let inputOptions: IFindInputOptions = {
			label: nls.localize('label.Search', 'Search: Type Search Term and press Enter to search or Escape to cancel'),
			validation: (value: string) => this.validatSearchInput(value),
S
Sandeep Somavarapu 已提交
214
			placeholder: nls.localize('search.placeHolder', "Search"),
215 216 217
			appendCaseSensitiveLabel: appendKeyBindingLabel('', this.keyBindingService2.lookupKeybinding(Constants.ToggleCaseSensitiveActionId), this.keyBindingService2),
			appendWholeWordsLabel: appendKeyBindingLabel('', this.keyBindingService2.lookupKeybinding(Constants.ToggleWholeWordActionId), this.keyBindingService2),
			appendRegexLabel: appendKeyBindingLabel('', this.keyBindingService2.lookupKeybinding(Constants.ToggleRegexActionId), this.keyBindingService2)
218 219
		};

J
Johannes Rieken 已提交
220
		let searchInputContainer = dom.append(parent, dom.$('.search-container.input-box'));
221
		this.searchInput = this._register(new FindInput(searchInputContainer, this.contextViewService, inputOptions));
B
Benjamin Pasero 已提交
222
		this._register(attachFindInputBoxStyler(this.searchInput, this.themeService));
223 224 225 226 227
		this.searchInput.onKeyUp((keyboardEvent: IKeyboardEvent) => this.onSearchInputKeyUp(keyboardEvent));
		this.searchInput.setValue(options.value || '');
		this.searchInput.setRegex(!!options.isRegex);
		this.searchInput.setCaseSensitive(!!options.isCaseSensitive);
		this.searchInput.setWholeWords(!!options.isWholeWords);
228 229 230
		this._register(this.onSearchSubmit(() => {
			this.searchHistory.add(this.searchInput.getValue());
		}));
231

S
Sandeep Somavarapu 已提交
232
		this.searchInputFocusTracker = this._register(dom.trackFocus(this.searchInput.inputBox.inputElement));
233 234 235 236 237 238
		this._register(this.searchInputFocusTracker.addFocusListener(() => {
			this.searchInputBoxFocussed.set(true);
		}));
		this._register(this.searchInputFocusTracker.addBlurListener(() => {
			this.searchInputBoxFocussed.set(false);
		}));
239 240 241
	}

	private renderReplaceInput(parent: HTMLElement): void {
J
Joao Moreno 已提交
242
		this.replaceContainer = dom.append(parent, dom.$('.replace-container.disabled'));
J
Johannes Rieken 已提交
243
		let replaceBox = dom.append(this.replaceContainer, dom.$('.input-box'));
S
Sandeep Somavarapu 已提交
244
		this.replaceInput = this._register(new InputBox(replaceBox, this.contextViewService, {
S
Sandeep Somavarapu 已提交
245
			ariaLabel: nls.localize('label.Replace', 'Replace: Type replace term and press Enter to preview or Escape to cancel'),
S
Sandeep Somavarapu 已提交
246
			placeholder: nls.localize('search.replace.placeHolder', "Replace")
247
		}));
B
Benjamin Pasero 已提交
248
		this._register(attachInputBoxStyler(this.replaceInput, this.themeService));
249 250
		this.onkeyup(this.replaceInput.inputElement, (keyboardEvent) => this.onReplaceInputKeyUp(keyboardEvent));
		this.replaceInput.onDidChange(() => this._onReplaceValueChanged.fire());
S
Sandeep Somavarapu 已提交
251
		this.searchInput.inputBox.onDidChange(() => this.onSearchInputChanged());
S
Sandeep Somavarapu 已提交
252 253

		this.replaceAllAction = ReplaceAllAction.INSTANCE;
J
Johannes Rieken 已提交
254
		this.replaceAllAction.searchWidget = this;
S
Sandeep Somavarapu 已提交
255 256 257
		this.replaceAllAction.label = SearchWidget.REPLACE_ALL_DISABLED_LABEL;
		this.replaceActionBar = this._register(new ActionBar(this.replaceContainer));
		this.replaceActionBar.push([this.replaceAllAction], { icon: true, label: false });
258

S
Sandeep Somavarapu 已提交
259 260 261 262 263 264 265
		this.replaceInputFocusTracker = this._register(dom.trackFocus(this.replaceInput.inputElement));
		this._register(this.replaceInputFocusTracker.addFocusListener(() => {
			this.replaceInputBoxFocussed.set(true);
		}));
		this._register(this.replaceInputFocusTracker.addBlurListener(() => {
			this.replaceInputBoxFocussed.set(false);
		}));
266 267
	}

268 269 270 271 272
	triggerReplaceAll(): TPromise<any> {
		this._onReplaceAll.fire();
		return TPromise.as(null);
	}

J
Johannes Rieken 已提交
273
	private onToggleReplaceButton(): void {
S
Sandeep Somavarapu 已提交
274
		dom.toggleClass(this.replaceContainer, 'disabled');
275 276
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'collapse');
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'expand');
277
		this.updateReplaceActiveState();
278
		this._onReplaceToggled.fire();
279 280
	}

J
Johannes Rieken 已提交
281
	public setReplaceAllActionState(enabled: boolean): void {
282
		if (this.replaceAllAction.enabled !== enabled) {
J
Johannes Rieken 已提交
283 284
			this.replaceAllAction.enabled = enabled;
			this.replaceAllAction.label = enabled ? SearchWidget.REPLACE_ALL_ENABLED_LABEL(this.keyBindingService2) : SearchWidget.REPLACE_ALL_DISABLED_LABEL;
285 286 287 288 289
			this.updateReplaceActiveState();
		}
	}

	private isReplaceActive(): boolean {
A
Alex Dima 已提交
290
		return this.replaceActive.get();
291 292 293
	}

	private updateReplaceActiveState(): void {
J
Johannes Rieken 已提交
294 295
		let currentState = this.isReplaceActive();
		let newState = this.isReplaceShown() && this.replaceAllAction.enabled;
296
		if (currentState !== newState) {
S
Sandeep Somavarapu 已提交
297 298
			this.replaceActive.set(newState);
			this._onReplaceStateChange.fire(newState);
299
		}
300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319
	}

	private validatSearchInput(value: string): any {
		if (value.length === 0) {
			return null;
		}
		if (!this.searchInput.getRegex()) {
			return null;
		}
		let regExp: RegExp;
		try {
			regExp = new RegExp(value);
		} catch (e) {
			return { content: e.message };
		}
		if (strings.regExpLeadsToEndlessLoop(regExp)) {
			return { content: nls.localize('regexp.validationFailure', "Expression matches everything") };
		}
	}

S
Sandeep Somavarapu 已提交
320
	private onSearchInputChanged(): void {
S
Sandeep Somavarapu 已提交
321
		this.setReplaceAllActionState(false);
S
Sandeep Somavarapu 已提交
322 323
	}

324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346
	private onSearchInputKeyUp(keyboardEvent: IKeyboardEvent) {
		switch (keyboardEvent.keyCode) {
			case KeyCode.Enter:
				this.submitSearch();
				return;
			case KeyCode.Escape:
				this._onSearchCancel.fire();
				return;
			default:
				return;
		}
	}

	private onReplaceInputKeyUp(keyboardEvent: IKeyboardEvent) {
		switch (keyboardEvent.keyCode) {
			case KeyCode.Enter:
				this.submitSearch();
				return;
			default:
				return;
		}
	}

J
Johannes Rieken 已提交
347
	private submitSearch(refresh: boolean = true): void {
348 349 350 351 352 353
		if (this.searchInput.getValue()) {
			this._onSearchSubmit.fire(refresh);
		}
	}

	public dispose(): void {
354
		this.setReplaceAllActionState(false);
J
Johannes Rieken 已提交
355
		this.replaceAllAction.searchWidget = null;
S
Sandeep Somavarapu 已提交
356
		this.replaceActionBar = null;
357 358
		super.dispose();
	}
359 360 361
}

export function registerContributions() {
J
Johannes Rieken 已提交
362 363
	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: ReplaceAllAction.ID,
364
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
S
Sandeep Somavarapu 已提交
365
		when: ContextKeyExpr.and(Constants.SearchViewletVisibleKey, Constants.ReplaceActiveKey, CONTEXT_FIND_WIDGET_NOT_VISIBLE),
366 367 368 369 370 371 372
		primary: KeyMod.Alt | KeyMod.CtrlCmd | KeyCode.Enter,
		handler: accessor => {
			if (isSearchViewletFocussed(accessor.get(IViewletService))) {
				ReplaceAllAction.INSTANCE.run();
			}
		}
	});
A
Alex Dima 已提交
373
}