searchWidget.ts 14.3 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
import Event, { Emitter } from 'vs/base/common/event';
import { Builder } from 'vs/base/browser/builder';
B
Benjamin Pasero 已提交
24
import { IViewletService } from 'vs/workbench/services/viewlet/browser/viewlet';
25
import { isSearchViewletFocussed, appendKeyBindingLabel } from 'vs/workbench/parts/search/browser/searchActions';
A
Alex Dima 已提交
26
import { CONTEXT_FIND_WIDGET_NOT_VISIBLE } from 'vs/editor/contrib/find/common/findController';
27
import { HistoryNavigator } from 'vs/base/common/history';
28
import * as Constants from 'vs/workbench/parts/search/common/constants';
B
Benjamin Pasero 已提交
29
import { attachInputBoxStyler, attachFindInputBoxStyler, attachButtonStyler } from 'vs/platform/theme/common/styler';
B
Benjamin Pasero 已提交
30
import { IThemeService } from 'vs/platform/theme/common/themeService';
31
import { SIDE_BAR_BACKGROUND } from 'vs/workbench/common/theme';
32 33

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

41 42
class ReplaceAllAction extends Action {

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

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

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

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

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

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

71 72
export class SearchWidget extends Widget {

J
Johannes Rieken 已提交
73 74
	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 => {
75
		let kb = keyBindingService2.lookupKeybinding(ReplaceAllAction.ID);
B
Benjamin Pasero 已提交
76
		return appendKeyBindingLabel(nls.localize('search.action.replaceAll.enabled.label', "Replace All"), kb, keyBindingService2);
E
Erich Gamma 已提交
77
	}
S
Sandeep Somavarapu 已提交
78

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

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

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

94 95
	private searchHistory: HistoryNavigator<string>;

96 97 98 99 100 101 102 103 104
	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 已提交
105 106
	private _onReplaceStateChange = this._register(new Emitter<boolean>());
	public onReplaceStateChange: Event<boolean> = this._onReplaceStateChange.event;
107 108 109 110 111 112 113

	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;

114 115 116 117 118 119 120 121
	constructor(
		container: Builder,
		options: ISearchWidgetOptions,
		@IContextViewService private contextViewService: IContextViewService,
		@IThemeService private themeService: IThemeService,
		@IContextKeyService private keyBindingService: IContextKeyService,
		@IKeybindingService private keyBindingService2: IKeybindingService,
	) {
122
		super();
123
		this.searchHistory = new HistoryNavigator<string>(options.history);
124 125
		this.replaceActive = Constants.ReplaceActiveKey.bindTo(this.keyBindingService);
		this.searchInputBoxFocussed = Constants.SearchInputBoxFocussedKey.bindTo(this.keyBindingService);
S
Sandeep Somavarapu 已提交
126
		this.replaceInputBoxFocussed = Constants.ReplaceInputBoxFocussedKey.bindTo(this.keyBindingService);
127 128 129
		this.render(container, options);
	}

J
Johannes Rieken 已提交
130
	public focus(select: boolean = true, focusReplace: boolean = false): void {
131
		if ((!focusReplace && this.searchInput.inputBox.hasFocus())
J
Johannes Rieken 已提交
132
			|| (focusReplace && this.replaceInput.hasFocus())) {
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149
			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) {
150
		this.searchInput.setWidth(width);
J
Johannes Rieken 已提交
151
		this.replaceInput.width = width - 28;
152 153 154 155
	}

	public clear() {
		this.searchInput.clear();
J
Johannes Rieken 已提交
156
		this.replaceInput.value = '';
S
Sandeep Somavarapu 已提交
157
		this.setReplaceAllActionState(false);
158 159 160
	}

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

J
Johannes Rieken 已提交
164
	public getReplaceValue(): string {
S
Sandeep Somavarapu 已提交
165
		return this.replaceInput.value;
166 167
	}

J
Johannes Rieken 已提交
168
	public toggleReplace(show?: boolean): void {
169 170
		if (show === void 0 || show !== this.isReplaceShown()) {
			this.onToggleReplaceButton();
S
Sandeep Somavarapu 已提交
171 172 173
		}
	}

174 175 176 177
	public getHistory(): string[] {
		return this.searchHistory.getHistory();
	}

178 179 180 181 182 183 184 185
	public showNextSearchTerm() {
		let next = this.searchHistory.next();
		if (next) {
			this.searchInput.setValue(next);
		}
	}

	public showPreviousSearchTerm() {
186
		let previous;
187
		if (this.searchInput.getValue().length === 0) {
W
Wim Spaargaren 已提交
188 189
			previous = this.searchHistory.current();
		} else {
190
			this.searchHistory.addIfNotPresent(this.searchInput.getValue());
W
Wim Spaargaren 已提交
191
			previous = this.searchHistory.previous();
192
		}
193 194 195 196 197
		if (previous) {
			this.searchInput.setValue(previous);
		}
	}

198 199 200 201 202 203 204 205
	public searchInputHasFocus(): boolean {
		return this.searchInputBoxFocussed.get();
	}

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

206 207 208 209 210 211 212 213 214
	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 已提交
215
		this.toggleReplaceButton = this._register(new Button(parent));
B
Benjamin Pasero 已提交
216 217 218 219
		attachButtonStyler(this.toggleReplaceButton, this.themeService, {
			buttonBackground: SIDE_BAR_BACKGROUND,
			buttonHoverBackground: SIDE_BAR_BACKGROUND
		});
J
Johannes Rieken 已提交
220
		this.toggleReplaceButton.icon = 'toggle-replace-button collapse';
A
Alex Dima 已提交
221
		this.toggleReplaceButton.addListener('click', () => this.onToggleReplaceButton());
J
Johannes Rieken 已提交
222
		this.toggleReplaceButton.getElement().title = nls.localize('search.replace.toggle.button.title', "Toggle Replace");
223 224 225 226 227 228
	}

	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 已提交
229
			placeholder: nls.localize('search.placeHolder', "Search"),
230 231 232
			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)
233 234
		};

J
Johannes Rieken 已提交
235
		let searchInputContainer = dom.append(parent, dom.$('.search-container.input-box'));
236
		this.searchInput = this._register(new FindInput(searchInputContainer, this.contextViewService, inputOptions));
B
Benjamin Pasero 已提交
237
		this._register(attachFindInputBoxStyler(this.searchInput, this.themeService));
238 239 240 241 242
		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);
243 244 245
		this._register(this.onSearchSubmit(() => {
			this.searchHistory.add(this.searchInput.getValue());
		}));
246

S
Sandeep Somavarapu 已提交
247
		this.searchInputFocusTracker = this._register(dom.trackFocus(this.searchInput.inputBox.inputElement));
248 249 250 251 252 253
		this._register(this.searchInputFocusTracker.addFocusListener(() => {
			this.searchInputBoxFocussed.set(true);
		}));
		this._register(this.searchInputFocusTracker.addBlurListener(() => {
			this.searchInputBoxFocussed.set(false);
		}));
254 255 256
	}

	private renderReplaceInput(parent: HTMLElement): void {
J
Joao Moreno 已提交
257
		this.replaceContainer = dom.append(parent, dom.$('.replace-container.disabled'));
J
Johannes Rieken 已提交
258
		let replaceBox = dom.append(this.replaceContainer, dom.$('.input-box'));
S
Sandeep Somavarapu 已提交
259
		this.replaceInput = this._register(new InputBox(replaceBox, this.contextViewService, {
S
Sandeep Somavarapu 已提交
260
			ariaLabel: nls.localize('label.Replace', 'Replace: Type replace term and press Enter to preview or Escape to cancel'),
S
Sandeep Somavarapu 已提交
261
			placeholder: nls.localize('search.replace.placeHolder', "Replace")
262
		}));
B
Benjamin Pasero 已提交
263
		this._register(attachInputBoxStyler(this.replaceInput, this.themeService));
264 265
		this.onkeyup(this.replaceInput.inputElement, (keyboardEvent) => this.onReplaceInputKeyUp(keyboardEvent));
		this.replaceInput.onDidChange(() => this._onReplaceValueChanged.fire());
S
Sandeep Somavarapu 已提交
266
		this.searchInput.inputBox.onDidChange(() => this.onSearchInputChanged());
S
Sandeep Somavarapu 已提交
267 268

		this.replaceAllAction = ReplaceAllAction.INSTANCE;
J
Johannes Rieken 已提交
269
		this.replaceAllAction.searchWidget = this;
S
Sandeep Somavarapu 已提交
270 271 272
		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 });
273

S
Sandeep Somavarapu 已提交
274 275 276 277 278 279 280
		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);
		}));
281 282
	}

283 284 285 286 287
	triggerReplaceAll(): TPromise<any> {
		this._onReplaceAll.fire();
		return TPromise.as(null);
	}

J
Johannes Rieken 已提交
288
	private onToggleReplaceButton(): void {
S
Sandeep Somavarapu 已提交
289
		dom.toggleClass(this.replaceContainer, 'disabled');
290 291
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'collapse');
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'expand');
292
		this.updateReplaceActiveState();
293
		this._onReplaceToggled.fire();
294 295
	}

J
Johannes Rieken 已提交
296
	public setReplaceAllActionState(enabled: boolean): void {
297
		if (this.replaceAllAction.enabled !== enabled) {
J
Johannes Rieken 已提交
298 299
			this.replaceAllAction.enabled = enabled;
			this.replaceAllAction.label = enabled ? SearchWidget.REPLACE_ALL_ENABLED_LABEL(this.keyBindingService2) : SearchWidget.REPLACE_ALL_DISABLED_LABEL;
300 301 302 303 304
			this.updateReplaceActiveState();
		}
	}

	private isReplaceActive(): boolean {
A
Alex Dima 已提交
305
		return this.replaceActive.get();
306 307 308
	}

	private updateReplaceActiveState(): void {
J
Johannes Rieken 已提交
309 310
		let currentState = this.isReplaceActive();
		let newState = this.isReplaceShown() && this.replaceAllAction.enabled;
311
		if (currentState !== newState) {
S
Sandeep Somavarapu 已提交
312 313
			this.replaceActive.set(newState);
			this._onReplaceStateChange.fire(newState);
314
		}
315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
	}

	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 已提交
335
	private onSearchInputChanged(): void {
S
Sandeep Somavarapu 已提交
336
		this.setReplaceAllActionState(false);
S
Sandeep Somavarapu 已提交
337 338
	}

339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361
	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 已提交
362
	private submitSearch(refresh: boolean = true): void {
363 364 365 366 367 368
		if (this.searchInput.getValue()) {
			this._onSearchSubmit.fire(refresh);
		}
	}

	public dispose(): void {
369
		this.setReplaceAllActionState(false);
J
Johannes Rieken 已提交
370
		this.replaceAllAction.searchWidget = null;
S
Sandeep Somavarapu 已提交
371
		this.replaceActionBar = null;
372 373
		super.dispose();
	}
374 375 376
}

export function registerContributions() {
J
Johannes Rieken 已提交
377 378
	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: ReplaceAllAction.ID,
379
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
S
Sandeep Somavarapu 已提交
380
		when: ContextKeyExpr.and(Constants.SearchViewletVisibleKey, Constants.ReplaceActiveKey, CONTEXT_FIND_WIDGET_NOT_VISIBLE),
381 382 383 384 385 386 387
		primary: KeyMod.Alt | KeyMod.CtrlCmd | KeyCode.Enter,
		handler: accessor => {
			if (isSearchViewletFocussed(accessor.get(IViewletService))) {
				ReplaceAllAction.INSTANCE.run();
			}
		}
	});
A
Alex Dima 已提交
388
}