searchWidget.ts 14.2 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 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;

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

J
Johannes Rieken 已提交
129
	public focus(select: boolean = true, focusReplace: boolean = false): void {
130
		if ((!focusReplace && this.searchInput.inputBox.hasFocus())
J
Johannes Rieken 已提交
131
			|| (focusReplace && this.replaceInput.hasFocus())) {
132 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) {
		this.searchInput.setWidth(width - 2);
J
Johannes Rieken 已提交
150
		this.replaceInput.width = width - 28;
151 152 153 154
	}

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

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

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

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

173 174 175 176 177 178 179 180
	public showNextSearchTerm() {
		let next = this.searchHistory.next();
		if (next) {
			this.searchInput.setValue(next);
		}
	}

	public showPreviousSearchTerm() {
181
		let previous;
182
		if (this.searchInput.getValue().length === 0) {
W
Wim Spaargaren 已提交
183 184
			previous = this.searchHistory.current();
		} else {
185
			this.searchHistory.addIfNotPresent(this.searchInput.getValue());
W
Wim Spaargaren 已提交
186
			previous = this.searchHistory.previous();
187
		}
188 189 190 191 192
		if (previous) {
			this.searchInput.setValue(previous);
		}
	}

193 194 195 196 197 198 199 200
	public searchInputHasFocus(): boolean {
		return this.searchInputBoxFocussed.get();
	}

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

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

	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 已提交
224
			placeholder: nls.localize('search.placeHolder', "Search"),
225 226 227
			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)
228 229
		};

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

S
Sandeep Somavarapu 已提交
242
		this.searchInputFocusTracker = this._register(dom.trackFocus(this.searchInput.inputBox.inputElement));
243 244 245 246 247 248
		this._register(this.searchInputFocusTracker.addFocusListener(() => {
			this.searchInputBoxFocussed.set(true);
		}));
		this._register(this.searchInputFocusTracker.addBlurListener(() => {
			this.searchInputBoxFocussed.set(false);
		}));
249 250 251
	}

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

		this.replaceAllAction = ReplaceAllAction.INSTANCE;
J
Johannes Rieken 已提交
264
		this.replaceAllAction.searchWidget = this;
S
Sandeep Somavarapu 已提交
265 266 267
		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 });
268

S
Sandeep Somavarapu 已提交
269 270 271 272 273 274 275
		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);
		}));
276 277
	}

278 279 280 281 282
	triggerReplaceAll(): TPromise<any> {
		this._onReplaceAll.fire();
		return TPromise.as(null);
	}

J
Johannes Rieken 已提交
283
	private onToggleReplaceButton(): void {
S
Sandeep Somavarapu 已提交
284
		dom.toggleClass(this.replaceContainer, 'disabled');
285 286
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'collapse');
		dom.toggleClass(this.toggleReplaceButton.getElement(), 'expand');
287
		this.updateReplaceActiveState();
288
		this._onReplaceToggled.fire();
289 290
	}

J
Johannes Rieken 已提交
291
	public setReplaceAllActionState(enabled: boolean): void {
292
		if (this.replaceAllAction.enabled !== enabled) {
J
Johannes Rieken 已提交
293 294
			this.replaceAllAction.enabled = enabled;
			this.replaceAllAction.label = enabled ? SearchWidget.REPLACE_ALL_ENABLED_LABEL(this.keyBindingService2) : SearchWidget.REPLACE_ALL_DISABLED_LABEL;
295 296 297 298 299
			this.updateReplaceActiveState();
		}
	}

	private isReplaceActive(): boolean {
A
Alex Dima 已提交
300
		return this.replaceActive.get();
301 302 303
	}

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

	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 已提交
330
	private onSearchInputChanged(): void {
S
Sandeep Somavarapu 已提交
331
		this.setReplaceAllActionState(false);
S
Sandeep Somavarapu 已提交
332 333
	}

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

	public dispose(): void {
364
		this.setReplaceAllActionState(false);
J
Johannes Rieken 已提交
365
		this.replaceAllAction.searchWidget = null;
S
Sandeep Somavarapu 已提交
366
		this.replaceActionBar = null;
367 368
		super.dispose();
	}
369 370 371
}

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