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

import 'vs/css!./simpleFindWidget';
import * as nls from 'vs/nls';
A
Alex Dima 已提交
8
import * as dom from 'vs/base/browser/dom';
9
import { FindInput, IFindInputStyles } from 'vs/base/browser/ui/findinput/findInput';
10
import { Widget } from 'vs/base/browser/ui/widget';
11
import { Delayer } from 'vs/base/common/async';
12
import { KeyCode, KeyMod } from 'vs/base/common/keyCodes';
A
Alex Dima 已提交
13
import { FindReplaceState } from 'vs/editor/contrib/find/findState';
M
mgquan@myseneca.ca 已提交
14
import { IMessage as InputBoxMessage } from 'vs/base/browser/ui/inputbox/inputBox';
A
Alex Dima 已提交
15 16
import { SimpleButton } from 'vs/editor/contrib/find/findWidget';
import { IContextKeyService } from 'vs/platform/contextkey/common/contextkey';
17
import { IContextViewService } from 'vs/platform/contextview/browser/contextView';
A
Alex Dima 已提交
18 19
import { editorWidgetBackground, inputActiveOptionBorder, inputBackground, inputBorder, inputForeground, inputValidationErrorBackground, inputValidationErrorBorder, inputValidationErrorForeground, inputValidationInfoBackground, inputValidationInfoBorder, inputValidationInfoForeground, inputValidationWarningBackground, inputValidationWarningBorder, inputValidationWarningForeground, widgetShadow } from 'vs/platform/theme/common/colorRegistry';
import { ITheme, registerThemingParticipant } from 'vs/platform/theme/common/themeService';
20
import { ContextScopedFindInput } from 'vs/platform/browser/contextScopedHistoryWidget';
21 22 23 24 25 26 27 28

const NLS_FIND_INPUT_LABEL = nls.localize('label.find', "Find");
const NLS_FIND_INPUT_PLACEHOLDER = nls.localize('placeholder.find', "Find");
const NLS_PREVIOUS_MATCH_BTN_LABEL = nls.localize('label.previousMatchButton', "Previous match");
const NLS_NEXT_MATCH_BTN_LABEL = nls.localize('label.nextMatchButton', "Next match");
const NLS_CLOSE_BTN_LABEL = nls.localize('label.closeButton', "Close");

export abstract class SimpleFindWidget extends Widget {
29 30 31
	private readonly _findInput: FindInput;
	private readonly _domNode: HTMLElement;
	private readonly _innerDomNode: HTMLElement;
32
	private _isVisible: boolean = false;
33 34 35
	private readonly _focusTracker: dom.IFocusTracker;
	private readonly _findInputFocusTracker: dom.IFocusTracker;
	private readonly _updateHistoryDelayer: Delayer<void>;
36

37
	constructor(
S
#50583  
Sandeep Somavarapu 已提交
38
		@IContextViewService private readonly _contextViewService: IContextViewService,
39 40 41
		@IContextKeyService contextKeyService: IContextKeyService,
		private readonly _state: FindReplaceState = new FindReplaceState(),
		showOptionButtons?: boolean
42 43
	) {
		super();
M
Matt Bierner 已提交
44

S
#50583  
Sandeep Somavarapu 已提交
45
		this._findInput = this._register(new ContextScopedFindInput(null, this._contextViewService, {
46 47
			label: NLS_FIND_INPUT_LABEL,
			placeholder: NLS_FIND_INPUT_PLACEHOLDER,
M
mgquan@myseneca.ca 已提交
48
			validation: (value: string): InputBoxMessage | null => {
M
updates  
mgquan@myseneca.ca 已提交
49
				if (value.length === 0 || !this._findInput.getRegex()) {
M
mgquan@myseneca.ca 已提交
50 51 52
					return null;
				}
				try {
M
updates  
mgquan@myseneca.ca 已提交
53
					/* tslint:disable-next-line:no-unused-expression */
M
mgquan@myseneca.ca 已提交
54 55 56 57 58 59
					new RegExp(value);
					return null;
				} catch (e) {
					return { content: e.message };
				}
			}
60
		}, contextKeyService, showOptionButtons));
61

62 63 64
		// Find History with update delayer
		this._updateHistoryDelayer = new Delayer<void>(500);

65 66
		this.oninput(this._findInput.domNode, (e) => {
			this.onInputChanged();
67
			this._delayedUpdateHistory();
68 69
		});

70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
		this._findInput.setRegex(!!this._state.isRegex);
		this._findInput.setCaseSensitive(!!this._state.matchCase);
		this._findInput.setWholeWords(!!this._state.wholeWord);

		this._register(this._findInput.onDidOptionChange(() => {
			this._state.change({
				isRegex: this._findInput.getRegex(),
				wholeWord: this._findInput.getWholeWords(),
				matchCase: this._findInput.getCaseSensitive()
			}, true);
		}));

		this._register(this._state.onFindReplaceStateChange(() => {
			this._findInput.setRegex(this._state.isRegex);
			this._findInput.setWholeWords(this._state.wholeWord);
			this._findInput.setCaseSensitive(this._state.matchCase);
		}));

88 89 90 91 92 93 94 95 96 97 98 99 100 101
		this._register(this._findInput.onKeyDown((e) => {
			if (e.equals(KeyCode.Enter)) {
				this.find(false);
				e.preventDefault();
				return;
			}

			if (e.equals(KeyMod.Shift | KeyCode.Enter)) {
				this.find(true);
				e.preventDefault();
				return;
			}
		}));

102
		const prevBtn = new SimpleButton({
103 104 105 106
			label: NLS_PREVIOUS_MATCH_BTN_LABEL,
			className: 'previous',
			onTrigger: () => {
				this.find(true);
M
Matt Bierner 已提交
107
			}
108 109
		});

110
		const nextBtn = new SimpleButton({
111 112 113 114
			label: NLS_NEXT_MATCH_BTN_LABEL,
			className: 'next',
			onTrigger: () => {
				this.find(false);
M
Matt Bierner 已提交
115
			}
116 117
		});

118
		const closeBtn = new SimpleButton({
119 120 121 122
			label: NLS_CLOSE_BTN_LABEL,
			className: 'close-fw',
			onTrigger: () => {
				this.hide();
M
Matt Bierner 已提交
123
			}
124 125
		});

126 127 128 129 130 131 132 133
		this._innerDomNode = document.createElement('div');
		this._innerDomNode.classList.add('simple-find-part');
		this._innerDomNode.appendChild(this._findInput.domNode);
		this._innerDomNode.appendChild(prevBtn.domNode);
		this._innerDomNode.appendChild(nextBtn.domNode);
		this._innerDomNode.appendChild(closeBtn.domNode);

		// _domNode wraps _innerDomNode, ensuring that
134
		this._domNode = document.createElement('div');
135 136
		this._domNode.classList.add('simple-find-part-wrapper');
		this._domNode.appendChild(this._innerDomNode);
137

138
		this.onkeyup(this._innerDomNode, e => {
139 140 141 142 143 144 145
			if (e.equals(KeyCode.Escape)) {
				this.hide();
				e.preventDefault();
				return;
			}
		});

146
		this._focusTracker = this._register(dom.trackFocus(this._innerDomNode));
147 148
		this._register(this._focusTracker.onDidFocus(this.onFocusTrackerFocus.bind(this)));
		this._register(this._focusTracker.onDidBlur(this.onFocusTrackerBlur.bind(this)));
149

150 151 152 153
		this._findInputFocusTracker = this._register(dom.trackFocus(this._findInput.domNode));
		this._register(this._findInputFocusTracker.onDidFocus(this.onFindInputFocusTrackerFocus.bind(this)));
		this._register(this._findInputFocusTracker.onDidBlur(this.onFindInputFocusTrackerBlur.bind(this)));

154
		this._register(dom.addDisposableListener(this._innerDomNode, 'click', (event) => {
155 156 157 158
			event.stopPropagation();
		}));
	}

159 160 161 162 163 164
	protected abstract onInputChanged(): void;
	protected abstract find(previous: boolean): void;
	protected abstract onFocusTrackerFocus(): void;
	protected abstract onFocusTrackerBlur(): void;
	protected abstract onFindInputFocusTrackerFocus(): void;
	protected abstract onFindInputFocusTrackerBlur(): void;
C
cleidigh 已提交
165

166 167 168 169
	protected get inputValue() {
		return this._findInput.getValue();
	}

170
	public get focusTracker(): dom.IFocusTracker {
171
		return this._focusTracker;
172 173
	}

M
Matt Bierner 已提交
174
	public updateTheme(theme: ITheme): void {
175
		const inputStyles: IFindInputStyles = {
176 177 178 179 180 181 182 183 184 185 186 187 188
			inputActiveOptionBorder: theme.getColor(inputActiveOptionBorder),
			inputBackground: theme.getColor(inputBackground),
			inputForeground: theme.getColor(inputForeground),
			inputBorder: theme.getColor(inputBorder),
			inputValidationInfoBackground: theme.getColor(inputValidationInfoBackground),
			inputValidationInfoForeground: theme.getColor(inputValidationInfoForeground),
			inputValidationInfoBorder: theme.getColor(inputValidationInfoBorder),
			inputValidationWarningBackground: theme.getColor(inputValidationWarningBackground),
			inputValidationWarningForeground: theme.getColor(inputValidationWarningForeground),
			inputValidationWarningBorder: theme.getColor(inputValidationWarningBorder),
			inputValidationErrorBackground: theme.getColor(inputValidationErrorBackground),
			inputValidationErrorForeground: theme.getColor(inputValidationErrorForeground),
			inputValidationErrorBorder: theme.getColor(inputValidationErrorBorder)
189 190 191 192
		};
		this._findInput.style(inputStyles);
	}

M
Matt Bierner 已提交
193
	dispose() {
194 195 196 197 198 199 200
		super.dispose();

		if (this._domNode && this._domNode.parentElement) {
			this._domNode.parentElement.removeChild(this._domNode);
		}
	}

M
Matt Bierner 已提交
201
	public getDomNode() {
202 203 204
		return this._domNode;
	}

205
	public reveal(initialInput?: string): void {
206 207 208 209
		if (initialInput) {
			this._findInput.setValue(initialInput);
		}

210
		if (this._isVisible) {
211 212 213 214 215 216 217
			this._findInput.select();
			return;
		}

		this._isVisible = true;

		setTimeout(() => {
218
			dom.addClass(this._innerDomNode, 'visible');
219
			dom.addClass(this._innerDomNode, 'visible-transition');
220
			this._innerDomNode.setAttribute('aria-hidden', 'false');
221
			this._findInput.select();
222 223 224
		}, 0);
	}

225 226 227 228 229 230 231 232 233
	public show(initialInput?: string): void {
		if (initialInput && !this._isVisible) {
			this._findInput.setValue(initialInput);
		}

		this._isVisible = true;

		setTimeout(() => {
			dom.addClass(this._innerDomNode, 'visible');
234
			dom.addClass(this._innerDomNode, 'visible-transition');
235 236 237 238
			this._innerDomNode.setAttribute('aria-hidden', 'false');
		}, 0);
	}

239 240
	public hide(): void {
		if (this._isVisible) {
241
			dom.removeClass(this._innerDomNode, 'visible-transition');
242
			this._innerDomNode.setAttribute('aria-hidden', 'true');
243 244
			// Need to delay toggling visibility until after Transition, then visibility hidden - removes from tabIndex list
			setTimeout(() => {
245
				this._isVisible = false;
246 247
				dom.removeClass(this._innerDomNode, 'visible');
			}, 200);
248 249
		}
	}
250 251 252 253 254 255

	protected _delayedUpdateHistory() {
		this._updateHistoryDelayer.trigger(this._updateHistory.bind(this));
	}

	protected _updateHistory() {
S
Sandeep Somavarapu 已提交
256
		this._findInput.inputBox.addToHistory();
257
	}
258 259 260 261 262 263 264 265 266 267 268 269

	protected _getRegexValue(): boolean {
		return this._findInput.getRegex();
	}

	protected _getWholeWordValue(): boolean {
		return this._findInput.getWholeWords();
	}

	protected _getCaseSensitiveValue(): boolean {
		return this._findInput.getCaseSensitive();
	}
270 271 272 273 274 275 276 277 278
}

// theming
registerThemingParticipant((theme, collector) => {
	const findWidgetBGColor = theme.getColor(editorWidgetBackground);
	if (findWidgetBGColor) {
		collector.addRule(`.monaco-workbench .simple-find-part { background-color: ${findWidgetBGColor} !important; }`);
	}

M
Matt Bierner 已提交
279
	const widgetShadowColor = theme.getColor(widgetShadow);
280 281 282 283
	if (widgetShadowColor) {
		collector.addRule(`.monaco-workbench .simple-find-part { box-shadow: 0 2px 8px ${widgetShadowColor}; }`);
	}
});