@@ -43,7 +45,9 @@ class SortableTable extends ImmutableComponent {
SortableTable.defaultProps = {
headings: React.PropTypes.array.isRequired,
- rows: React.PropTypes.array.isRequired
+ rows: React.PropTypes.array.isRequired,
+ isHover: React.PropTypes.bool,
+ hoverCallback: React.PropTypes.func
}
module.exports = SortableTable
diff --git a/js/components/urlBar.js b/js/components/urlBar.js
index 117a828b525..c070fd9a2eb 100644
--- a/js/components/urlBar.js
+++ b/js/components/urlBar.js
@@ -21,6 +21,8 @@ const contextMenus = require('../contextMenus')
const dndData = require('../dndData')
const pdfjsExtensionId = require('../constants/config').PDFJSExtensionId
const windowStore = require('../stores/windowStore')
+var searchProviders = require('../data/searchProviders')
+const searchIconSize = 16
const { isUrl, isIntermediateAboutPage } = require('../lib/appUrlUtil')
@@ -35,6 +37,8 @@ class UrlBar extends ImmutableComponent {
this.onChange = this.onChange.bind(this)
this.onClick = this.onClick.bind(this)
this.onContextMenu = this.onContextMenu.bind(this)
+ this.activateSearchEngine = false
+ this.searchSelectEntry = null
}
get activeFrame () {
@@ -121,6 +125,11 @@ class UrlBar extends ImmutableComponent {
this.urlBarSuggestions.clickSelected(e)
} else {
let searchUrl = this.props.searchDetail.get('searchURL').replace('{searchTerms}', encodeURIComponent(location))
+ if (this.activateSearchEngine && this.searchSelectEntry !== null && !isLocationUrl) {
+ const replaceRE = new RegExp('^' + this.searchSelectEntry.shortcut + ' ', 'g')
+ location = location.replace(replaceRE, '')
+ searchUrl = this.searchSelectEntry.search.replace('{searchTerms}', encodeURIComponent(location))
+ }
location = isLocationUrl ? location : searchUrl
// do search.
if (e.altKey) {
@@ -134,6 +143,7 @@ class UrlBar extends ImmutableComponent {
// this can't go through appActions for some reason
// or the whole window will reload on the first page request
this.updateDOMInputFocus(false)
+ this.clearSearchEngine()
}
break
case KeyCodes.UP:
@@ -156,6 +166,7 @@ class UrlBar extends ImmutableComponent {
case KeyCodes.ESC:
e.preventDefault()
ipc.emit(messages.SHORTCUT_ACTIVE_FRAME_STOP)
+ this.clearSearchEngine()
break
case KeyCodes.DELETE:
if (e.shiftKey) {
@@ -193,6 +204,7 @@ class UrlBar extends ImmutableComponent {
// On blur, a user expects the text shown from the last autocomplete suffix
// to be auto entered as the new location.
this.updateLocationToSuggestion()
+ this.clearSearchEngine()
}
updateLocationToSuggestion () {
@@ -201,14 +213,40 @@ class UrlBar extends ImmutableComponent {
}
}
+ detectSearchEngine (input) {
+ let location = input || this.props.urlbar.get('location')
+ if (location !== null && location.length !== 0) {
+ const isLocationUrl = isUrl(location)
+ if (!isLocationUrl &&
+ !(this.searchSelectEntry && location.startsWith(this.searchSelectEntry.shortcut + ' '))) {
+ let entries = searchProviders.providers
+ entries.forEach((entry) => {
+ if (location.startsWith(entry.shortcut + ' ')) {
+ this.activateSearchEngine = true
+ this.searchSelectEntry = entry
+ return false
+ }
+ })
+ }
+ }
+ }
+
+ clearSearchEngine () {
+ this.activateSearchEngine = false
+ this.searchSelectEntry = null
+ }
+
onChange (e) {
windowActions.setUrlBarSelected(false)
windowActions.setUrlBarActive(true)
windowActions.setNavBarUserInput(e.target.value)
+ this.clearSearchEngine()
+ this.detectSearchEngine(e.target.value)
}
onFocus (e) {
windowActions.setUrlBarSelected(true)
+ this.detectSearchEngine()
}
onActiveFrameStop () {
@@ -337,12 +375,24 @@ class UrlBar extends ImmutableComponent {
onClick={this.onSiteInfo}
className={cx({
urlbarIcon: true,
- 'fa': true,
- 'fa-lock': this.isHTTPPage && this.props.isSecure && !this.props.urlbar.get('active'),
- 'fa-unlock-alt': this.isHTTPPage && !this.props.isSecure && !this.props.urlbar.get('active') && !this.props.titleMode,
- 'fa fa-file': this.props.urlbar.get('active') && this.props.loading === false,
+ 'fa': !this.activateSearchEngine,
+ 'fa-lock': !this.activateSearchEngine && this.isHTTPPage && this.isSecure && !this.props.urlbar.get('active'),
+ 'fa-unlock-alt': !this.activateSearchEngine && this.isHTTPPage && !this.isSecure && !this.props.urlbar.get('active') && !this.props.titleMode,
+ 'fa fa-file': !this.activateSearchEngine && this.props.urlbar.get('active') && this.props.loading === false,
extendedValidation: this.extendedValidationSSL
- })} />
+ })}
+ style={
+ this.activateSearchEngine
+ ? {
+ backgroundImage: `url(${this.searchSelectEntry.image})`,
+ minWidth: searchIconSize,
+ width: searchIconSize,
+ backgroundSize: searchIconSize,
+ height: searchIconSize,
+ marginTop: '3px',
+ marginRight: '3px'
+ } : {}
+ } />
{
this.props.titleMode
?
diff --git a/js/constants/appConfig.js b/js/constants/appConfig.js
index 702bfe8e917..69d8a122c87 100644
--- a/js/constants/appConfig.js
+++ b/js/constants/appConfig.js
@@ -86,7 +86,7 @@ module.exports = {
'general.show-home-button': false,
'general.useragent.value': null, // Set at runtime
'general.autohide-menu': true,
- 'search.default-search-engine': 'content/search/google.xml',
+ 'search.default-search-engine': 'Google',
'search.offer-search-suggestions': false, // false by default for privacy reasons
'tabs.switch-to-new-tabs': false,
'tabs.paint-tabs': true,
diff --git a/js/contextMenus.js b/js/contextMenus.js
index b9685bfe562..9d3cf714eb4 100644
--- a/js/contextMenus.js
+++ b/js/contextMenus.js
@@ -788,7 +788,7 @@ function mainTemplateInit (nodeProps, frame) {
},
copyAddressMenuItem('copyImageAddress', nodeProps.srcURL)
)
- if (getSetting(settings.DEFAULT_SEARCH_ENGINE) === 'content/search/google.xml' &&
+ if (getSetting(settings.DEFAULT_SEARCH_ENGINE) === 'Google' &&
nodeProps.srcURL && urlParse(nodeProps.srcURL).protocol !== 'data:') {
template.push(
{
diff --git a/js/data/searchProviders.js b/js/data/searchProviders.js
new file mode 100644
index 00000000000..9c41193617b
--- /dev/null
+++ b/js/data/searchProviders.js
@@ -0,0 +1,65 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this file,
+ * You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+module.exports = { "providers" :
+ [
+ {
+ "name" : "Amazon",
+ "image" : "http://www.amazon.com/favicon.ico",
+ "search" : "http://www.amazon.com/exec/obidos/external-search/?field-keywords={searchTerms}&mode=blended",
+ "autocomplete" : "http://completion.amazon.com/search/complete?method=completion&q={searchTerms}&search-alias=aps&client=amazon-search-ui&mkt=1",
+ "shortcut" : "a"
+ },
+ {
+ "name" : "Bing",
+ "image" : "https://www.bing.com/favicon.ico",
+ "search" : "https://www.bing.com/search?q={searchTerms}",
+ "autocomplete" : "http://api.bing.com/osjson.aspx?query={searchTerms}&language={language}&form=OSDJAS",
+ "shortcut" : "b"
+ },
+ {
+ "name" : "DuckDuckGo",
+ "image" : "https://duckduckgo.com/favicon.ico",
+ "search" : "https://duckduckgo.com/?q={searchTerms}&t=brave",
+ "autocomplete" : "https://ac.duckduckgo.com/ac/?q={searchTerms}&type=list",
+ "shortcut" : "d"
+ },
+ {
+ "name" : "Google",
+ "image" : "https://www.google.com/favicon.ico",
+ "search" : "https://www.google.com/search?q={searchTerms}",
+ "autocomplete" : "https://suggestqueries.google.com/complete/search?client=chrome&q={searchTerms}",
+ "shortcut" : "g"
+ },
+ {
+ "name" : "Twitter",
+ "image" : "https://twitter.com/favicon.ico",
+ "search" : "https://twitter.com/search?q={searchTerms}&source=desktop-search",
+ "autocomplete" : "https://api.twitter.com/1.1/search/tweets.json?q={searchTerms}",
+ "shortcut" : "t"
+ },
+ {
+ "name" : "Wikipedia",
+ "image" : "https://en.wikipedia.org/favicon.ico",
+ "search" : "http://en.wikipedia.org/wiki/Special:Search?search={searchTerms}",
+ "autocomplete": "http://en.wikipedia.org/w/api.php?search={searchTerms}",
+ "shortcut" : "w"
+ },
+ {
+ "name" : "Yahoo",
+ "image" : "https://search.yahoo.com/favicon.ico",
+ "search" : "https://search.yahoo.com/search?p={searchTerms}&fr=opensearch",
+ "autocomplete": "https://search.yahoo.com/sugg/os?command={searchTerms}&output=fxjson&fr=opensearch",
+ "shortcut" : "y"
+ },
+ {
+ "name" : "Youtube",
+ "image" : "https://www.youtube.com/favicon.ico",
+ "search" : "https://www.youtube.com/results?search_type=search_videos&search_query={searchTerms}&search_sort=relevance&search_category=0&page=",
+ "autocomplete": "http://suggestqueries.google.com/complete/search?output=chrome&client=chrome&hl=it&q={searchTerms}&ds=yt",
+ "shortcut" : "yt"
+ }
+ ]
+}
+
diff --git a/less/about/preferences.less b/less/about/preferences.less
index 51eab1616f0..37bbf03b1e7 100644
--- a/less/about/preferences.less
+++ b/less/about/preferences.less
@@ -434,6 +434,12 @@ table.sortableTable {
background: url('') 0 0 / contain no-repeat;
}
+#searchSelectIcon {
+ width: 16px;
+ height: 16px;
+ color: @braveOrange;
+}
+
.prefTabContainer .switchControl {
display: inline-block;
vertical-align: middle;
diff --git a/less/sortableTable.less b/less/sortableTable.less
index 5ee95f1388c..546545e4772 100644
--- a/less/sortableTable.less
+++ b/less/sortableTable.less
@@ -23,10 +23,6 @@ table.sortableTable {
font-weight: 300;
padding: 8px;
box-sizing: border-box;
-
- &:hover {
- text-decoration: underline;
- }
}
td {
@@ -73,3 +69,7 @@ table.sortableTable {
}
}
}
+tr.rowHover:hover {
+ background: #ffcc99;
+ cursor: pointer;
+}
diff --git a/test/components/navigationBarTest.js b/test/components/navigationBarTest.js
index 8a221b0f191..6d2c828d683 100644
--- a/test/components/navigationBarTest.js
+++ b/test/components/navigationBarTest.js
@@ -6,6 +6,7 @@ const {urlInput, activeWebview, activeTabFavicon, activeTab, navigatorLoadTime,
const urlParse = require('url').parse
const assert = require('assert')
const settings = require('../../js/constants/settings')
+const searchProviders = require('../../js/data/searchProviders')
describe('navigationBar', function () {
function * setup (client) {
@@ -459,6 +460,49 @@ describe('navigationBar', function () {
})
})
+ describe('search engine go key', function () {
+ Brave.beforeAll(this)
+ const entries = searchProviders.providers
+
+ before(function * () {
+ yield setup(this.app.client)
+ yield this.app.client.waitForExist(urlInput)
+ yield this.app.client.waitForElementFocus(urlInput)
+ yield this.app.client.waitUntil(function () {
+ return this.getValue(urlInput).then((val) => val === '')
+ })
+ })
+
+ entries.forEach((entry) => {
+ describe('each entry', function () {
+ before(function * () {
+ // escape
+ yield this.app.client.ipcSend('shortcut-active-frame-stop')
+ // type go key
+ yield this.app.client.keys(entry.shortcut + ' ')
+ })
+
+ it('sets the value', function * () {
+ yield this.app.client.waitUntil(function () {
+ return this.getValue(urlInput).then((val) => val === entry.shortcut)
+ })
+ })
+
+ it('has focus', function * () {
+ yield this.app.client.waitForElementFocus(urlInput)
+ })
+
+ it('has the icon', function * () {
+ yield this.app.client
+ .waitForExist(urlbarIcon)
+ .getCssProperty(urlbarIcon, 'background-image').then((backgroundImage) =>
+ backgroundImage.value === `url("${entry.image}")`
+ )
+ })
+ })
+ })
+ })
+
// need to move urlbar state to frame before enabling these
describe('change tabs', function () {
Brave.beforeAll(this)