diff --git a/app/extensions.js b/app/extensions.js index d964b030334..b11427b801b 100644 --- a/app/extensions.js +++ b/app/extensions.js @@ -111,7 +111,7 @@ let generateBraveManifest = () => { 'form-action': '\'none\'', 'referrer': 'no-referrer', 'style-src': '\'self\' \'unsafe-inline\'', - 'img-src': '\'self\' data:' + 'img-src': '* data:' } if (process.env.NODE_ENV === 'development') { diff --git a/app/extensions/brave/locales/en-US/preferences.properties b/app/extensions/brave/locales/en-US/preferences.properties index 6e40a9b4654..46ed631e8e6 100644 --- a/app/extensions/brave/locales/en-US/preferences.properties +++ b/app/extensions/brave/locales/en-US/preferences.properties @@ -43,7 +43,9 @@ startsWithOptionLastTime=my windows / tabs from last time startsWithOptionHomePage=my home page startsWithOptionNewTabPage=the new tab page myHomepage=My homepage is -defaultSearchEngine=Default search engine: +default=Default +searchEngine=Search Engine +engineGoKey=Engine Go Key (Type First) switchToNewTabs=Switch to new tabs immediately paintTabs=Show tabs in page theme color tabsPerTabPage=Number of tabs per tab set: diff --git a/app/sessionStore.js b/app/sessionStore.js index 198f50c4d31..7898deffbc8 100644 --- a/app/sessionStore.js +++ b/app/sessionStore.js @@ -301,6 +301,13 @@ module.exports.loadAppState = () => { try { data = Object.assign(module.exports.defaultAppState(), JSON.parse(data)) + // xml migration + if (data.settings['search.default-search-engine'] === 'content/search/google.xml') { + data.settings['search.default-search-engine'] = 'Google' + } + if (data.settings['search.default-search-engine'] === 'contnt/search/duckduckgo.xml') { + data.settings['search.default-search-engine'] = 'DuckDuckGo' + } } catch (e) { // TODO: Session state is corrupted, maybe we should backup this // corrupted value for people to report into support. diff --git a/docs/state.md b/docs/state.md index 04181771a24..5860f47bde9 100644 --- a/docs/state.md +++ b/docs/state.md @@ -134,7 +134,7 @@ AppStore 'general.downloads.default-save-path': string, // default path for saving files 'general.autohide-menu': boolean, // true if the Windows menu should be autohidden 'general.disable-title-mode': boolean, // true if title mode should always be disabled - 'search.default-search-engine': string, // path to the open search XML + 'search.default-search-engine': string, // name of search engine, from js/data/searchProviders.js 'search.offer-search-suggestions': boolean, // true if suggestions should be offered from the default search engine when available. 'tabs.switch-to-new-tabs': boolean, // true if newly opened tabs should be focused immediately 'tabs.paint-tabs': boolean, // true if the page theme color and favicon color should be used for tabs diff --git a/js/about/preferences.js b/js/about/preferences.js index bfd3449ebe5..0350d94d3cb 100644 --- a/js/about/preferences.js +++ b/js/about/preferences.js @@ -16,6 +16,8 @@ const messages = require('../constants/messages') const settings = require('../constants/settings') const aboutActions = require('./aboutActions') const getSetting = require('../settings').getSetting +const SortableTable = require('../components/sortableTable') +const searchProviders = require('../data/searchProviders') const adblock = appConfig.resourceNames.ADBLOCK const cookieblock = appConfig.resourceNames.COOKIEBLOCK @@ -173,19 +175,68 @@ class GeneralTab extends ImmutableComponent { } } +class SearchSelectEntry extends ImmutableComponent { + shouldComponentUpdate (nextProps, nextState) { + return this.props.settings.get('search.default-search-engine') !== nextProps.settings.get('search.default-search-engine') + } + render () { + return
+ {this.props.settings.get('search.default-search-engine') === this.props.name + ? : null} +
+ } +} + +class SearchEntry extends ImmutableComponent { + render () { + return
+ + + {this.props.name} +
+ } +} + +class SearchShortcutEntry extends ImmutableComponent { + render () { + return
+ {this.props.shortcut} +
+ } +} + class SearchTab extends ImmutableComponent { + get searchProviders () { + let entries = searchProviders.providers + let array = [] + const iconSize = 16 + entries.forEach((entry) => { + let iconStyle = { + backgroundImage: `url(${entry.image})`, + minWidth: iconSize, + width: iconSize, + backgroundSize: iconSize, + height: iconSize, + display: 'inline-block', + verticalAlign: 'middle' + } + array.push([, + , + ]) + }) + return array + } + + hoverCallback (rows) { + this.props.onChangeSetting(settings.DEFAULT_SEARCH_ENGINE, rows[1].props.children.props.name) + } + render () { return
- - - - - +
diff --git a/js/components/main.js b/js/components/main.js index 158844121e9..ed8cc28ffc2 100644 --- a/js/components/main.js +++ b/js/components/main.js @@ -11,7 +11,6 @@ const ipc = electron.ipcRenderer // Actions const windowActions = require('../actions/windowActions') const webviewActions = require('../actions/webviewActions') -const loadOpenSearch = require('../lib/openSearch').loadOpenSearch const contextMenus = require('../contextMenus') const getSetting = require('../settings').getSetting const getOrigin = require('../state/siteUtil').getOrigin @@ -48,6 +47,8 @@ const keyCodes = require('../constants/keyCodes') // State handling const FrameStateUtil = require('../state/frameStateUtil') +const searchProviders = require('../data/searchProviders') + // Util const cx = require('../lib/classSet.js') const eventUtil = require('../lib/eventUtil') @@ -201,16 +202,25 @@ class Main extends ImmutableComponent { ipc.on(messages.LEAVE_FULL_SCREEN, this.exitFullScreen.bind(this)) } - loadOpenSearch () { + loadSearchProviders () { + let entries = searchProviders.providers let engine = getSetting(settings.DEFAULT_SEARCH_ENGINE) - if (this.lastLoadedOpenSearch === undefined || engine !== this.lastLoadedOpenSearch) { - loadOpenSearch(engine).then((searchDetail) => windowActions.setSearchDetail(searchDetail)) - this.lastLoadedOpenSearch = engine + if (this.lastLoadedSearchProviders === undefined || engine !== this.lastLoadedSearchProviders) { + entries.forEach((entry) => { + if (entry.name === engine) { + windowActions.setSearchDetail(Immutable.fromJS({ + searchURL: entry.search, + autocompleteURL: entry.autocomplete + })) + this.lastLoadedSearchProviders = engine + return false + } + }) } } componentDidUpdate (prevProps) { - this.loadOpenSearch() + this.loadSearchProviders() const activeFrame = FrameStateUtil.getActiveFrame(this.props.windowState) const activeFramePrev = FrameStateUtil.getActiveFrame(prevProps.windowState) const activeFrameTitle = activeFrame && (activeFrame.get('title') || activeFrame.get('location')) || '' @@ -369,7 +379,7 @@ class Main extends ImmutableComponent { windowActions.setContextMenuDetail() }) - this.loadOpenSearch() + this.loadSearchProviders() window.addEventListener('mousemove', (e) => { if (e.pageY !== this.pageY) { diff --git a/js/components/sortableTable.js b/js/components/sortableTable.js index 2018186f0e3..3bdaf93ae7c 100644 --- a/js/components/sortableTable.js +++ b/js/components/sortableTable.js @@ -14,6 +14,7 @@ class SortableTable extends ImmutableComponent { componentDidMount (event) { return tableSort(document.getElementsByClassName('sortableTable')[0]) } + render () { var headings = [] var rows = [] @@ -26,7 +27,8 @@ class SortableTable extends ImmutableComponent { headings[j] = headings[j] || rows[i][j] = {this.props.rows[i][j] === true ? '✕' : this.props.rows[i][j]} } - rows[i] = {rows[i]} + rows[i] = {rows[i]} } return @@ -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)