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" + } + ] +} +