diff --git a/ui/app/components/global-search/control.js b/ui/app/components/global-search/control.js
new file mode 100644
index 00000000000..1a061ba8943
--- /dev/null
+++ b/ui/app/components/global-search/control.js
@@ -0,0 +1,172 @@
+import Component from '@ember/component';
+import { tagName } from '@ember-decorators/component';
+import { task } from 'ember-concurrency';
+import EmberObject, { action, computed, set } from '@ember/object';
+import { alias } from '@ember/object/computed';
+import { inject as service } from '@ember/service';
+import { run } from '@ember/runloop';
+import Searchable from 'nomad-ui/mixins/searchable';
+import classic from 'ember-classic-decorator';
+
+const SLASH_KEY = 191;
+
+@tagName('')
+export default class GlobalSearchControl extends Component {
+ @service dataCaches;
+ @service router;
+ @service store;
+
+ searchString = null;
+
+ constructor() {
+ super(...arguments);
+
+ this.jobSearch = JobSearch.create({
+ dataSource: this,
+ });
+
+ this.nodeSearch = NodeSearch.create({
+ dataSource: this,
+ });
+ }
+
+ keyDownHandler(e) {
+ const targetElementName = e.target.nodeName.toLowerCase();
+
+ if (targetElementName != 'input' && targetElementName != 'textarea') {
+ if (e.keyCode === SLASH_KEY) {
+ e.preventDefault();
+ this.open();
+ }
+ }
+ }
+
+ didInsertElement() {
+ this.set('_keyDownHandler', this.keyDownHandler.bind(this));
+ document.addEventListener('keydown', this._keyDownHandler);
+ }
+
+ willDestroyElement() {
+ document.removeEventListener('keydown', this._keyDownHandler);
+ }
+
+ @task(function*(string) {
+ try {
+ set(this, 'searchString', string);
+
+ const jobs = yield this.dataCaches.fetch('job');
+ const nodes = yield this.dataCaches.fetch('node');
+
+ set(this, 'jobs', jobs.toArray());
+ set(this, 'nodes', nodes.toArray());
+
+ const jobResults = this.jobSearch.listSearched;
+ const nodeResults = this.nodeSearch.listSearched;
+
+ return [
+ {
+ groupName: `Jobs (${jobResults.length})`,
+ options: jobResults,
+ },
+ {
+ groupName: `Clients (${nodeResults.length})`,
+ options: nodeResults,
+ },
+ ];
+ } catch (e) {
+ // eslint-disable-next-line
+ console.log('exception searching', e);
+ }
+ })
+ search;
+
+ @action
+ open() {
+ if (this.select) {
+ this.select.actions.open();
+ }
+ }
+
+ @action
+ selectOption(model) {
+ const itemModelName = model.constructor.modelName;
+
+ if (itemModelName === 'job') {
+ this.router.transitionTo('jobs.job', model.name, {
+ queryParams: { namespace: model.get('namespace.name') },
+ });
+ } else if (itemModelName === 'node') {
+ this.router.transitionTo('clients.client', model.id);
+ }
+ }
+
+ @action
+ storeSelect(select) {
+ if (select) {
+ this.select = select;
+ }
+ }
+
+ @action
+ openOnClickOrTab(select, { target }) {
+ // Bypass having to press enter to access search after clicking/tabbing
+ const targetClassList = target.classList;
+ const targetIsTrigger = targetClassList.contains('ember-power-select-trigger');
+
+ // Allow tabbing out of search
+ const triggerIsNotActive = !targetClassList.contains('ember-power-select-trigger--active');
+
+ if (targetIsTrigger && triggerIsNotActive) {
+ run.next(() => {
+ select.actions.open();
+ });
+ }
+ }
+
+ calculatePosition(trigger) {
+ const { top, left, width } = trigger.getBoundingClientRect();
+ return {
+ style: {
+ left,
+ width,
+ top,
+ },
+ };
+ }
+}
+
+@classic
+class JobSearch extends EmberObject.extend(Searchable) {
+ @computed
+ get searchProps() {
+ return ['id', 'name'];
+ }
+
+ @computed
+ get fuzzySearchProps() {
+ return ['name'];
+ }
+
+ @alias('dataSource.jobs') listToSearch;
+ @alias('dataSource.searchString') searchTerm;
+
+ fuzzySearchEnabled = true;
+}
+
+@classic
+class NodeSearch extends EmberObject.extend(Searchable) {
+ @computed
+ get searchProps() {
+ return ['id', 'name'];
+ }
+
+ @computed
+ get fuzzySearchProps() {
+ return ['name'];
+ }
+
+ @alias('dataSource.nodes') listToSearch;
+ @alias('dataSource.searchString') searchTerm;
+
+ fuzzySearchEnabled = true;
+}
diff --git a/ui/app/services/data-caches.js b/ui/app/services/data-caches.js
new file mode 100644
index 00000000000..7617c61c583
--- /dev/null
+++ b/ui/app/services/data-caches.js
@@ -0,0 +1,33 @@
+import Service, { inject as service } from '@ember/service';
+
+export const COLLECTION_CACHE_DURATION = 60000; // one minute
+
+export default class DataCachesService extends Service {
+ @service router;
+ @service store;
+ @service system;
+
+ collectionLastFetched = {};
+
+ async fetch(modelName) {
+ const modelNameToRoute = {
+ job: 'jobs',
+ node: 'clients',
+ };
+
+ const route = modelNameToRoute[modelName];
+ const lastFetched = this.collectionLastFetched[modelName];
+ const now = Date.now();
+
+ if (this.router.isActive(route)) {
+ // TODO Incorrect because it’s constantly being fetched by watchers, shouldn’t be marked as last fetched only on search
+ this.collectionLastFetched[modelName] = now;
+ return this.store.peekAll(modelName);
+ } else if (lastFetched && now - lastFetched < COLLECTION_CACHE_DURATION) {
+ return this.store.peekAll(modelName);
+ } else {
+ this.collectionLastFetched[modelName] = now;
+ return this.store.findAll(modelName);
+ }
+ }
+}
diff --git a/ui/app/styles/components.scss b/ui/app/styles/components.scss
index 6f70cceef9c..955c3e73d5f 100644
--- a/ui/app/styles/components.scss
+++ b/ui/app/styles/components.scss
@@ -11,6 +11,8 @@
@import './components/exec-button';
@import './components/exec-window';
@import './components/fs-explorer';
+@import './components/global-search-control';
+@import './components/global-search-dropdown';
@import './components/gutter';
@import './components/gutter-toggle';
@import './components/image-file.scss';
diff --git a/ui/app/styles/components/global-search-control.scss b/ui/app/styles/components/global-search-control.scss
new file mode 100644
index 00000000000..e42243b5021
--- /dev/null
+++ b/ui/app/styles/components/global-search-control.scss
@@ -0,0 +1,37 @@
+.global-search {
+ width: 30em;
+
+ .ember-power-select-trigger {
+ background: $nomad-green-darker;
+
+ .icon {
+ margin-top: 1px;
+ margin-left: 2px;
+
+ fill: white;
+ opacity: 0.7;
+ }
+
+ .placeholder {
+ opacity: 0.7;
+ display: inline-block;
+ padding-left: 2px;
+ transform: translateY(-1px);
+ }
+
+ &.ember-power-select-trigger--active {
+ background: white;
+
+ .icon {
+ fill: black;
+ opacity: 1;
+ }
+ }
+ }
+
+ .ember-basic-dropdown-content-wormhole-origin {
+ position: absolute;
+ top: 0;
+ width: 100%;
+ }
+}
diff --git a/ui/app/styles/components/global-search-dropdown.scss b/ui/app/styles/components/global-search-dropdown.scss
new file mode 100644
index 00000000000..4e9c2f5ef3e
--- /dev/null
+++ b/ui/app/styles/components/global-search-dropdown.scss
@@ -0,0 +1,45 @@
+.global-search-dropdown {
+ background: transparent;
+ border: 0;
+ position: fixed;
+
+ .ember-power-select-search {
+ margin-left: $icon-dimensions;
+ border: 0;
+ }
+
+ input,
+ input:focus {
+ background: transparent;
+ border: 0;
+ outline: 0;
+ }
+
+ .ember-power-select-options {
+ background: white;
+ padding: 0.35rem;
+
+ &[role='listbox'] {
+ border: 1px solid $grey-blue;
+ box-shadow: 0 6px 8px -2px rgba($black, 0.05), 0 8px 4px -4px rgba($black, 0.1);
+ }
+
+ .ember-power-select-option {
+ padding: 0.2rem 0.4rem;
+ border-radius: $radius;
+
+ &[aria-current='true'] {
+ background: transparentize($blue, 0.8);
+ color: $blue;
+ }
+ }
+ }
+
+ .ember-power-select-group-name {
+ text-transform: uppercase;
+ display: inline;
+ color: darken($grey-blue, 20%);
+ font-size: $size-7;
+ font-weight: $weight-semibold;
+ }
+}
diff --git a/ui/app/styles/core/navbar.scss b/ui/app/styles/core/navbar.scss
index 48a2a065d76..9018db8afb8 100644
--- a/ui/app/styles/core/navbar.scss
+++ b/ui/app/styles/core/navbar.scss
@@ -8,6 +8,8 @@
padding-left: 20px;
padding-right: 20px;
overflow: hidden;
+ align-items: center;
+ justify-content: space-between;
.navbar-item {
color: rgba($primary-invert, 0.8);
@@ -35,7 +37,6 @@
display: block;
position: absolute;
left: 0px;
- top: 1.25em;
}
}
}
@@ -44,7 +45,7 @@
display: flex;
align-items: stretch;
justify-content: flex-end;
- margin-left: auto;
+ margin-left: inherit;
}
.navbar-end > a.navbar-item {
@@ -100,7 +101,7 @@
display: flex;
align-items: center;
justify-content: flex-end;
- margin-left: auto;
+ margin-left: inherit;
}
.navbar-end > a.navbar-item {
diff --git a/ui/app/templates/components/global-header.hbs b/ui/app/templates/components/global-header.hbs
index 23c5aa05ccf..e129a42c321 100644
--- a/ui/app/templates/components/global-header.hbs
+++ b/ui/app/templates/components/global-header.hbs
@@ -7,6 +7,9 @@
{{partial "partials/nomad-logo"}}
+ {{#unless (media "isMobile")}}
+