diff --git a/docs/state.md b/docs/state.md
index df748e21d43..e61004eac0b 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 b99ca78f967..4373fc064f8 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
@@ -161,19 +163,88 @@ class GeneralTab extends ImmutableComponent {
}
}
+class SearchEntry extends React.Component {
+ constructor () {
+ super()
+ this.state = {
+ hover: false
+ }
+ }
+
+ onMouseOver () {
+ this.setState({hover: true})
+ }
+
+ onMouseOut () {
+ this.setState({hover: false})
+ }
+
+ onClick (name) {
+ this.props.onChangeSetting(settings.DEFAULT_SEARCH_ENGINE, name)
+ }
+
+ render () {
+ return
+
+
+ {this.props.name}
+
+ }
+}
+
+class SearchSelectEntry extends React.Component {
+ constructor (props) {
+ super(props)
+ this.state = {
+ select: props.name === getSetting(settings.DEFAULT_SEARCH_ENGINE)
+ }
+
+ ipc.on(messages.SETTINGS_UPDATED, (e, settings) => {
+ const settingsMap = Immutable.fromJS(settings || {})
+ if (this.props.name === settingsMap.get('search.default-search-engine')) {
+ this.setState({select: true})
+ } else {
+ this.setState({select: false})
+ }
+ })
+ }
+
+ render () {
+ return
+ {this.state.select ? 'x' : ''}
+
+ }
+}
+
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'
+ }
+ let textStyle = {paddingLeft: '5px'}
+ array.push([,
+ , entry.shortcut])
+ })
+ return array
+ }
render () {
return
-
-
-
-
-
+
+
+
diff --git a/js/components/main.js b/js/components/main.js
index fecb0ad45b1..33c820e6ed0 100644
--- a/js/components/main.js
+++ b/js/components/main.js
@@ -12,7 +12,6 @@ const remote = electron.remote
// 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')
@@ -177,16 +178,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')) || ''
@@ -343,7 +353,7 @@ class Main extends ImmutableComponent {
windowActions.setContextMenuDetail()
})
- this.loadOpenSearch()
+ this.loadSearchProviders()
window.addEventListener('mousemove', (e) => {
self.checkForTitleMode(e.pageY)
diff --git a/js/components/urlBar.js b/js/components/urlBar.js
index 28c66182192..64bafe752b2 100644
--- a/js/components/urlBar.js
+++ b/js/components/urlBar.js
@@ -20,6 +20,7 @@ const settings = require('../constants/settings')
const contextMenus = require('../contextMenus')
const dndData = require('../dndData')
const appStoreRenderer = require('../stores/appStoreRenderer')
+const searchProviders = require('../data/searchProviders')
const { isUrl, isIntermediateAboutPage } = require('../lib/appUrlUtil')
@@ -112,7 +113,19 @@ class UrlBar extends ImmutableComponent {
// load the selected suggestion
this.urlBarSuggestions.clickSelected(e)
} else {
+ let entries = searchProviders.providers
let searchUrl = this.searchDetail.get('searchURL').replace('{searchTerms}', encodeURIComponent(location))
+ if (!isLocationUrl) {
+ entries.forEach((entry) => {
+ const searchRE = new RegExp('^' + entry.shortcut + ' .*', 'g')
+ if (searchRE.test(location)) {
+ const replaceRE = new RegExp('^' + entry.shortcut + ' ', 'g')
+ location = location.replace(replaceRE, '')
+ searchUrl = entry.search.replace('{searchTerms}', encodeURIComponent(location))
+ return false
+ }
+ })
+ }
location = isLocationUrl ? location : searchUrl
// do search.
if (e.altKey) {
diff --git a/js/constants/appConfig.js b/js/constants/appConfig.js
index d82c58799f1..d4c2d309bab 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 182454c8f0f..fcca9442c92 100644
--- a/js/contextMenus.js
+++ b/js/contextMenus.js
@@ -777,7 +777,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..9c0674f594d
--- /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" : "",
+ "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" : "",
+ "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" : "",
+ "search" : "https://duckduckgo.com/?q={searchTerms}&t=brave",
+ "autocomplete" : "https://ac.duckduckgo.com/ac/?q={searchTerms}&type=list",
+ "shortcut" : "d"
+ },
+ {
+ "name" : "Google",
+ "image" : "",
+ "search" : "https://www.google.com/search?q={searchTerms}",
+ "autocomplete" : "https://suggestqueries.google.com/complete/search?client=chrome&q={searchTerms}",
+ "shortcut" : "g"
+ },
+ {
+ "name" : "Twitter",
+ "image" : "data:;base64,AAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAAAQAAAAAAAAAAAAAAAAAAAAAAAD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wz///8f////H////x////8Y////B////wD///8A////AP///wD///8A////AP///wD///8A////A////yj7789g9tV+ofbVfqH21X6h+N+eifz03lf///8o////A////wD///8A////AP///wD///8A////APr37g/PsGCftogQ7r2JAP/bnwD/7qwA/+6sAP/urAD/9Mtftv357z////8D////AP///wD///8A////AP///wD///8A////AP///w/57c5a88VQwu6sAP/urAD/7qwA/+6sAP/xvDDb/fnvP////wD///8A////AP///wD///8A////AP///wDz5L5X7bgv2+6sAP/urAD/7qwA/+6sAP/urAD/7qwA//G8MNv///8Y////AP///wD///8A////AP///wD///8M9NN+n+6wD/PurAD/7qwA/+6sAP/urAD/7qwA/+6sAP/urAD/9Nyegf///wD///8A////AP///wD///8A+O3PVu6wD/PurAD/7qwA/+6sAP/urAD/7qwA/+6sAP/urAD/7qwA//G8MNv///8A////AP///wD///8A////ANasQMXurAD/7qwA/+qpAP/YnAD/1JkA/+6sAP/urAD/7qwA/+6sAP/urAD/////FP///wD///8A////AP///wDz255+7qwA/9icAP+6ixDuz7Bgn9m8b5TurAD/7qwA/+6sAP/urAD/7qwA//jkr3j///8M////AP///wD///8A47Y/ysmRAP/PsGCf+vfuD////wDw584w46QA/+6sAP/urAD/7qwA/9+hAP/orA/z9OrPQv///wD///8A////AMypT7Hhz55g////AP///wD///8A////AMOaL9DMlAD/0JYA/8mRAP/FoEC/vpIf4OPQnmL///8A////AP///wD69+4P////AP///wD///8A////AP///wD69+4P3ceOcNi/foHhz55g////APr37g/69+4P////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A////AP///wD///8A//8AAP//AAD//wAA8P8AAMA/AAD4HwAA8A8AAOAHAADgBwAAwAcAAOAHAADHAwAA3wMAAP/fAAD//wAA//8AAA==",
+ "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" : "%2FAAZGBkAmJiYANjZ2ABXWFcAent6ALm6uQA8OjwAiIiIiIiIiIiIiI4oiL6IiIiIgzuIV4iIiIhndo53KIiIiB%2FWvXoYiIiIfEZfWBSIiIEGi%2FfoqoiIgzuL84i9iIjpGIoMiEHoiMkos3FojmiLlUipYliEWIF%2BiDe0GoRa7D6GPbjcu1yIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA",
+ "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" : "",
+ "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" : "",
+ "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"
+ }
+ ]
+}
+