diff --git a/app/browser/api/topSites.js b/app/browser/api/topSites.js index 48f3c9ccfce..9d2fa46d775 100644 --- a/app/browser/api/topSites.js +++ b/app/browser/api/topSites.js @@ -7,39 +7,36 @@ const Immutable = require('immutable') const appActions = require('../../../js/actions/appActions') const debounce = require('../../../js/lib/debounce') -const siteUtil = require('../../../js/state/siteUtil') +const historyState = require('../../common/state/historyState') +const bookmarkLocationCache = require('../../common/cache/bookmarkLocationCache') +const newTabData = require('../../../js/data/newTabData') const {isSourceAboutUrl} = require('../../../js/lib/appUrlUtil') -const aboutNewTabMaxEntries = 100 +const aboutNewTabMaxEntries = 18 let appStore let minCountOfTopSites let minAccessOfTopSites - -const compareSites = (site1, site2) => { - if (!site1 || !site2) return false - return site1.get('location') === site2.get('location') && - site1.get('partitionNumber') === site2.get('partitionNumber') -} +const staticData = Immutable.fromJS(newTabData.topSites) const pinnedTopSites = (state) => { - return (state.getIn(['about', 'newtab', 'pinnedTopSites']) || Immutable.List()).setSize(18) + return state.getIn(['about', 'newtab', 'pinnedTopSites'], Immutable.List()) } const ignoredTopSites = (state) => { - return state.getIn(['about', 'newtab', 'ignoredTopSites']) || Immutable.List() + return state.getIn(['about', 'newtab', 'ignoredTopSites'], Immutable.List()) } -const isPinned = (state, siteProps) => { - return pinnedTopSites(state).filter((site) => compareSites(site, siteProps)).size > 0 +const isPinned = (state, siteKey) => { + return pinnedTopSites(state).find(site => site.get('key') === siteKey) } -const isIgnored = (state, siteProps) => { - return ignoredTopSites(state).filter((site) => compareSites(site, siteProps)).size > 0 +const isIgnored = (state, siteKey) => { + return ignoredTopSites(state).includes(siteKey) } const sortCountDescending = (left, right) => { - const leftCount = left.get('count') || 0 - const rightCount = right.get('count') || 0 + const leftCount = left.get('count', 0) + const rightCount = right.get('count', 0) if (leftCount < rightCount) { return 1 } @@ -55,54 +52,44 @@ const sortCountDescending = (left, right) => { return 0 } -const removeDuplicateDomains = (list) => { - const siteDomains = new Set() - return list.filter((site) => { - if (!site.get('location')) { - return false - } - try { - const hostname = require('../../common/urlParse')(site.get('location')).hostname - if (!siteDomains.has(hostname)) { - siteDomains.add(hostname) - return true - } - } catch (e) { - console.log('Error parsing hostname: ', e) - } - return false - }) -} - -const calculateTopSites = (clearCache) => { +const calculateTopSites = (clearCache, withoutDebounce = false) => { if (clearCache) { clearTopSiteCacheData() } - startCalculatingTopSiteData() + if (withoutDebounce) { + getTopSiteData() + } else { + debouncedGetTopSiteData() + } } -/** - * TopSites are defined by users for the new tab page. Pinned sites are attached to their positions - * in the grid, and the non pinned indexes are populated with newly accessed sites - */ -const startCalculatingTopSiteData = debounce(() => { +const getTopSiteData = () => { if (!appStore) { appStore = require('../../../js/stores/appStore') } const state = appStore.getState() // remove folders; sort by visit count; enforce a max limit - const sites = (state.get('sites') ? state.get('sites').toList() : new Immutable.List()) - .filter((site) => !siteUtil.isFolder(site) && - !siteUtil.isImportedBookmark(site) && - !isSourceAboutUrl(site.get('location')) && + let sites = historyState.getSites(state) + .filter((site, key) => !isSourceAboutUrl(site.get('location')) && + !isPinned(state, key) && + !isIgnored(state, key) && (minCountOfTopSites === undefined || (site.get('count') || 0) >= minCountOfTopSites) && - (minAccessOfTopSites === undefined || (site.get('lastAccessedTime') || 0) >= minAccessOfTopSites)) + (minAccessOfTopSites === undefined || (site.get('lastAccessedTime') || 0) >= minAccessOfTopSites) + ) .sort(sortCountDescending) .slice(0, aboutNewTabMaxEntries) + .map((site, key) => { + const bookmarkKey = bookmarkLocationCache.getCacheKey(state, site.get('location')) + + site = site.set('bookmarked', !bookmarkKey.isEmpty()) + site = site.set('key', key) + return site + }) + .toList() for (let i = 0; i < sites.size; i++) { - const count = sites.getIn([i, 'count']) || 0 - const access = sites.getIn([i, 'lastAccessedTime']) || 0 + const count = sites.getIn([i, 'count'], 0) + const access = sites.getIn([i, 'lastAccessedTime'], 0) if (minCountOfTopSites === undefined || count < minCountOfTopSites) { minCountOfTopSites = count } @@ -111,33 +98,26 @@ const startCalculatingTopSiteData = debounce(() => { } } - // Filter out pinned and ignored sites - let unpinnedSites = sites.filter((site) => !(isPinned(state, site) || isIgnored(state, site))) - unpinnedSites = removeDuplicateDomains(unpinnedSites) - - // Merge the pinned and unpinned lists together - // Pinned items have priority because the position is important - let gridSites = pinnedTopSites(state).map((pinnedSite) => { - // Fetch latest siteDetail objects from appState.sites using location/partition - if (pinnedSite) { - const matches = sites.filter((site) => compareSites(site, pinnedSite)) - if (matches.size > 0) return matches.first() - } - // Default to unpinned items - const firstSite = unpinnedSites.first() - unpinnedSites = unpinnedSites.shift() - return firstSite - }) - - // Include up to [aboutNewTabMaxEntries] entries so that folks - // can ignore sites and have new items fill those empty spaces - if (unpinnedSites.size > 0) { - gridSites = gridSites.concat(unpinnedSites) + if (sites.size < 18) { + const preDefined = staticData + .filter((site) => { + return !isPinned(state, site.get('key')) && !isIgnored(state, site.get('key')) + }) + .map(site => { + const bookmarkKey = bookmarkLocationCache.getCacheKey(state, site.get('location')) + return site.set('bookmarked', !bookmarkKey.isEmpty()) + }) + sites = sites.concat(preDefined) } - const finalData = gridSites.filter((site) => site != null) - appActions.topSiteDataAvailable(finalData) -}, 5 * 1000) + appActions.topSiteDataAvailable(sites) +} + +/** + * TopSites are defined by users for the new tab page. Pinned sites are attached to their positions + * in the grid, and the non pinned indexes are populated with newly accessed sites + */ +const debouncedGetTopSiteData = debounce(() => getTopSiteData(), 5 * 1000) const clearTopSiteCacheData = () => { minCountOfTopSites = undefined diff --git a/app/browser/bookmarksExporter.js b/app/browser/bookmarksExporter.js index 598501ed3da..cdc5b8de4bf 100644 --- a/app/browser/bookmarksExporter.js +++ b/app/browser/bookmarksExporter.js @@ -11,15 +11,23 @@ const electron = require('electron') const dialog = electron.dialog const app = electron.app const BrowserWindow = electron.BrowserWindow -const getSetting = require('../../js/settings').getSetting + +// State +const bookmarksState = require('../common/state/bookmarksState') + +// Constants const settings = require('../../js/constants/settings') -const siteTags = require('../../js/constants/siteTags') -const siteUtil = require('../../js/state/siteUtil') -const isWindows = process.platform === 'win32' + +// Utils +const {getSetting} = require('../../js/settings') +const platformUtil = require('../common/lib/platformUtil') +const bookmarkFoldersUtil = require('../common/lib/bookmarkFoldersUtil') +const bookmarkUtil = require('../common/lib/bookmarkUtil') + const indentLength = 2 const indentType = ' ' -function showDialog (sites) { +const showDialog = (state) => { const focusedWindow = BrowserWindow.getFocusedWindow() const fileName = moment().format('DD_MM_YYYY') + '.html' const defaultPath = path.join(getSetting(settings.DEFAULT_DOWNLOAD_SAVE_PATH) || app.getPath('downloads'), fileName) @@ -34,17 +42,15 @@ function showDialog (sites) { }] }, (fileName) => { if (fileName) { - personal = createBookmarkArray(sites) - other = createBookmarkArray(sites, -1, false) + personal = createBookmarkArray(state) + other = createBookmarkArray(state, -1, false) fs.writeFileSync(fileName, createBookmarkHTML(personal, other)) } }) } -function createBookmarkArray (sites, parentFolderId, first = true, depth = 1) { - const filteredBookmarks = parentFolderId - ? sites.filter((site) => site.get('parentFolderId') === parentFolderId) - : sites.filter((site) => !site.get('parentFolderId')) +const createBookmarkArray = (state, parentFolderId = 0, first = true, depth = 1) => { + const bookmarks = bookmarksState.getBookmarksWithFolders(state, parentFolderId) let payload = [] let title let indentFirst = indentType.repeat(depth * indentLength) @@ -52,26 +58,23 @@ function createBookmarkArray (sites, parentFolderId, first = true, depth = 1) { if (first) payload.push(`${indentFirst}

`) - filteredBookmarks.forEach((site) => { - if (site.get('tags').includes(siteTags.BOOKMARK) && site.get('location')) { - title = site.get('customTitle') || site.get('title') || site.get('location') + for (let site of bookmarks) { + if (bookmarkUtil.isBookmark(site) && site.get('location')) { + title = site.get('title', site.get('location')) payload.push(`${indentNext}

${title}`) - } else if (siteUtil.isFolder(site)) { - const folderId = site.get('folderId') - - title = site.get('customTitle') || site.get('title') - payload.push(`${indentNext}

${title}

`) - payload = payload.concat(createBookmarkArray(sites, folderId, true, (depth + 1))) + } else if (bookmarkFoldersUtil.isFolder(site)) { + payload.push(`${indentNext}

${site.get('title')}

`) + payload = payload.concat(createBookmarkArray(state, site.get('folderId'), true, (depth + 1))) } - }) + } if (first) payload.push(`${indentFirst}

`) return payload } -function createBookmarkHTML (personal, other) { - const breakTag = (isWindows) ? '\r\n' : '\n' +const createBookmarkHTML = (personal, other) => { + const breakTag = (platformUtil.isWindows()) ? '\r\n' : '\n' const title = 'Bookmarks' return ` diff --git a/app/browser/menu.js b/app/browser/menu.js index a5b550a8619..b83a907d51d 100644 --- a/app/browser/menu.js +++ b/app/browser/menu.js @@ -17,7 +17,6 @@ const appConstants = require('../../js/constants/appConstants') const windowConstants = require('../../js/constants/windowConstants') const messages = require('../../js/constants/messages') const settings = require('../../js/constants/settings') -const siteTags = require('../../js/constants/siteTags') // State const {getByTabId} = require('../common/state/tabState') @@ -35,8 +34,8 @@ const frameStateUtil = require('../../js/state/frameStateUtil') const menuUtil = require('../common/lib/menuUtil') const {getSetting} = require('../../js/settings') const locale = require('../locale') -const {isLocationBookmarked} = require('../../js/state/siteUtil') const platformUtil = require('../common/lib/platformUtil') +const bookmarkUtil = require('../common/lib/bookmarkUtil') const isDarwin = platformUtil.isDarwin() const isLinux = platformUtil.isLinux() const isWindows = platformUtil.isWindows() @@ -376,7 +375,7 @@ const updateRecentlyClosedMenuItems = (state) => { } const isCurrentLocationBookmarked = (state) => { - return isLocationBookmarked(state, currentLocation) + return bookmarkUtil.isLocationBookmarked(state, currentLocation) } const createBookmarksSubmenu = (state) => { @@ -406,7 +405,7 @@ const createBookmarksSubmenu = (state) => { CommonMenu.exportBookmarksMenuItem() ] - const bookmarks = menuUtil.createBookmarkTemplateItems(state.get('sites')) + const bookmarks = menuUtil.createBookmarkTemplateItems(state) if (bookmarks.length > 0) { submenu.push(CommonMenu.separatorMenuItem) submenu = submenu.concat(bookmarks) @@ -693,38 +692,14 @@ const doAction = (state, action) => { } break } - case appConstants.APP_APPLY_SITE_RECORDS: - if (action.records && action.records.find((record) => record.objectData === 'bookmark')) { - createMenu(state) - } - break - case appConstants.APP_ADD_SITE: case appConstants.APP_ADD_BOOKMARK: case appConstants.APP_EDIT_BOOKMARK: - { - if (action.tag === siteTags.BOOKMARK || action.tag === siteTags.BOOKMARK_FOLDER) { - createMenu(state) - } else if (action.siteDetail && action.siteDetail.constructor === Immutable.List && action.tag === undefined) { - let shouldRebuild = false - action.siteDetail.forEach((site) => { - const tag = site.getIn(['tags', 0]) - if (tag === siteTags.BOOKMARK || tag === siteTags.BOOKMARK_FOLDER) { - shouldRebuild = true - } - }) - if (shouldRebuild) { - createMenu(state) - } - } - break - } - case appConstants.APP_REMOVE_SITE: - { - if (action.tag === siteTags.BOOKMARK || action.tag === siteTags.BOOKMARK_FOLDER) { - createMenu(state) - } - break - } + case appConstants.APP_REMOVE_BOOKMARK: + case appConstants.APP_ADD_BOOKMARK_FOLDER: + case appConstants.APP_EDIT_BOOKMARK_FOLDER: + case appConstants.APP_REMOVE_BOOKMARK_FOLDER: + createMenu(state) + break case appConstants.APP_ON_CLEAR_BROWSING_DATA: { const defaults = state.get('clearBrowsingDataDefaults') diff --git a/app/browser/reducers/bookmarkFoldersReducer.js b/app/browser/reducers/bookmarkFoldersReducer.js new file mode 100644 index 00000000000..365bae4ca32 --- /dev/null +++ b/app/browser/reducers/bookmarkFoldersReducer.js @@ -0,0 +1,88 @@ +/* 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/. */ + +const Immutable = require('immutable') + +// State +const bookmarkFoldersState = require('../../common/state/bookmarkFoldersState') + +// Constants +const appConstants = require('../../../js/constants/appConstants') + +// Utils +const {makeImmutable} = require('../../common/state/immutableUtil') +const syncUtil = require('../../../js/state/syncUtil') + +const bookmarkFoldersReducer = (state, action, immutableAction) => { + action = immutableAction || makeImmutable(action) + switch (action.get('actionType')) { + case appConstants.APP_ADD_BOOKMARK_FOLDER: + { + const closestKey = action.get('closestKey') + let folder = action.get('folderDetails') + + if (folder == null) { + break + } + + if (Immutable.List.isList(folder)) { + action.get('folderDetails', Immutable.List()).forEach((folder) => { + state = bookmarkFoldersState.addFolder(state, folder, closestKey) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, folder) + } + }) + } else { + state = bookmarkFoldersState.addFolder(state, folder, closestKey) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, folder) + } + } + break + } + case appConstants.APP_EDIT_BOOKMARK_FOLDER: + { + let folder = action.get('folderDetails') + + if (folder == null) { + break + } + + state = bookmarkFoldersState.editFolder(state, folder, action.get('editKey')) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, folder) + } + + break + } + case appConstants.APP_MOVE_BOOKMARK_FOLDER: + { + state = bookmarkFoldersState.moveFolder( + state, + action.get('folderKey'), + action.get('destinationKey'), + action.get('append'), + action.get('moveIntoParent') + ) + + if (syncUtil.syncEnabled()) { + const destinationDetail = state.getIn(['sites', action.get('destinationKey')]) + state = syncUtil.updateSiteCache(state, destinationDetail) + } + break + } + case appConstants.APP_REMOVE_BOOKMARK_FOLDER: + { + state = bookmarkFoldersState.removeFolder(state, action.get('folderKey')) + break + } + } + + return state +} + +module.exports = bookmarkFoldersReducer diff --git a/app/browser/reducers/bookmarksReducer.js b/app/browser/reducers/bookmarksReducer.js new file mode 100644 index 00000000000..aa202b39d79 --- /dev/null +++ b/app/browser/reducers/bookmarksReducer.js @@ -0,0 +1,103 @@ +/* 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/. */ + +const Immutable = require('immutable') + +// State +const bookmarksState = require('../../common/state/bookmarksState') + +// Constants +const appConstants = require('../../../js/constants/appConstants') + +// Utils +const {makeImmutable} = require('../../common/state/immutableUtil') +const syncUtil = require('../../../js/state/syncUtil') +const bookmarkUtil = require('../../common/lib/bookmarkUtil') +const bookmarkLocationCache = require('../../common/cache/bookmarkLocationCache') + +const bookmarksReducer = (state, action, immutableAction) => { + action = immutableAction || makeImmutable(action) + switch (action.get('actionType')) { + case appConstants.APP_SET_STATE: + state = bookmarkLocationCache.generateCache(state) + break + case appConstants.APP_ADD_BOOKMARK: + { + const closestKey = action.get('closestKey') + let bookmark = action.get('siteDetail') + + if (bookmark == null) { + break + } + + if (Immutable.List.isList(bookmark)) { + action.get('siteDetail', Immutable.List()).forEach((bookmark) => { + state = bookmarksState.addBookmark(state, bookmark, closestKey) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, bookmark) + } + }) + } else { + state = bookmarksState.addBookmark(state, bookmark, closestKey) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, bookmark) + } + } + + state = bookmarkUtil.updateActiveTabBookmarked(state) + break + } + case appConstants.APP_EDIT_BOOKMARK: + { + let bookmark = action.get('siteDetail') + + if (bookmark == null) { + break + } + + state = bookmarksState.editBookmark(state, bookmark, action.get('editKey')) + + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, bookmark) + } + + state = bookmarkUtil.updateActiveTabBookmarked(state) + break + } + case appConstants.APP_MOVE_BOOKMARK: + { + state = bookmarksState.moveBookmark( + state, + action.get('bookmarkKey'), + action.get('destinationKey'), + action.get('append'), + action.get('moveIntoParent') + ) + + if (syncUtil.syncEnabled()) { + const destinationDetail = state.getIn(['sites', action.get('destinationKey')]) + state = syncUtil.updateSiteCache(state, destinationDetail) + } + break + } + case appConstants.APP_REMOVE_BOOKMARK: + { + if (Immutable.List.isList(action.get('bookmarkKey'))) { + action.get('bookmarkKey', Immutable.List()).forEach((key) => { + state = bookmarksState.removeBookmark(state, key) + }) + } else { + state = bookmarksState.removeBookmark(state, action.get('bookmarkKey')) + } + state = bookmarkUtil.updateActiveTabBookmarked(state) + break + } + } + + return state +} + +module.exports = bookmarksReducer diff --git a/app/browser/reducers/historyReducer.js b/app/browser/reducers/historyReducer.js new file mode 100644 index 00000000000..cb3091cad3e --- /dev/null +++ b/app/browser/reducers/historyReducer.js @@ -0,0 +1,89 @@ +/* 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/. */ + +const Immutable = require('immutable') + +// State +const historyState = require('../../common/state/historyState') +const aboutHistoryState = require('../../common/state/aboutHistoryState') + +// Constants +const appConstants = require('../../../js/constants/appConstants') + +// Utils +const {makeImmutable} = require('../../common/state/immutableUtil') +const syncUtil = require('../../../js/state/syncUtil') +const filtering = require('../../filtering') +const {calculateTopSites} = require('../api/topSites') + +const historyReducer = (state, action, immutableAction) => { + action = immutableAction || makeImmutable(action) + switch (action.get('actionType')) { + case appConstants.APP_ON_CLEAR_BROWSING_DATA: + { + const defaults = state.get('clearBrowsingDataDefaults') + const temp = state.get('tempClearBrowsingData', Immutable.Map()) + const clearData = defaults ? defaults.merge(temp) : temp + if (clearData.get('browserHistory')) { + state = historyState.clearSites() + filtering.clearHistory() + } + break + } + case appConstants.APP_ADD_HISTORY_SITE: + { + const isSyncEnabled = syncUtil.syncEnabled() + const detail = action.get('siteDetail') + + if (Immutable.List.isList(detail)) { + detail.forEach((item) => { + state = historyState.addSite(state, item) + if (isSyncEnabled) { + state = syncUtil.updateSiteCache(state, item) + } + }) + } else { + state = historyState.addSite(state, detail) + if (isSyncEnabled) { + state = syncUtil.updateSiteCache(state, detail) + } + } + + calculateTopSites(true) + state = aboutHistoryState.setHistory(state, historyState.getSites(state)) + break + } + + case appConstants.APP_REMOVE_HISTORY_SITE: + { + if (Immutable.List.isList(action.get('historyKey'))) { + action.get('historyKey', Immutable.List()).forEach((key) => { + state = historyState.removeSite(state, key) + }) + } else { + state = historyState.removeSite(state, action.get('historyKey')) + } + + // TODO fix sync + /* + if (syncUtil.syncEnabled()) { + //syncActions.removeSite(historyState.getSite(state, action.get('historyKey'))) + //state = syncUtil.updateSiteCache(state, action.get('siteDetail')) + } + */ + + calculateTopSites(true) + state = aboutHistoryState.setHistory(state, historyState.getSites(state)) + break + } + + case appConstants.APP_POPULATE_HISTORY: + state = aboutHistoryState.setHistory(state, historyState.getSites(state)) + break + } + + return state +} + +module.exports = historyReducer diff --git a/app/browser/reducers/pinnedSitesReducer.js b/app/browser/reducers/pinnedSitesReducer.js new file mode 100644 index 00000000000..f0a3ed6b55e --- /dev/null +++ b/app/browser/reducers/pinnedSitesReducer.js @@ -0,0 +1,72 @@ +/* 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/. */ + +// State +const pinnedSitesState = require('../../common/state/pinnedSitesState') +const tabState = require('../../common/state/tabState') + +// Constants +const appConstants = require('../../../js/constants/appConstants') + +// Utils +const {makeImmutable} = require('../../common/state/immutableUtil') +const syncUtil = require('../../../js/state/syncUtil') +const pinnedSitesUtil = require('../../common/lib/pinnedSitesUtil') + +const pinnedSitesReducer = (state, action, immutableAction) => { + action = immutableAction || makeImmutable(action) + switch (action.get('actionType')) { + case appConstants.APP_TAB_UPDATED: + { + if (action.getIn(['changeInfo', 'pinned']) != null) { + const pinned = action.getIn(['changeInfo', 'pinned']) + const tabId = action.getIn(['tabValue', 'tabId']) + const tab = tabState.getByTabId(state, tabId) + if (!tab) { + console.warn('Trying to pin a tabId which does not exist:', tabId, 'tabs: ', state.get('tabs').toJS()) + break + } + const sites = pinnedSitesState.getSites(state) + const siteDetail = pinnedSitesUtil.getDetailsFromTab(sites, tab) + if (pinned) { + state = pinnedSitesState.addPinnedSite(state, siteDetail) + } else { + state = pinnedSitesState.removePinnedSite(state, siteDetail) + } + if (syncUtil.syncEnabled()) { + state = syncUtil.updateSiteCache(state, siteDetail) + } + } + break + } + case appConstants.APP_CREATE_TAB_REQUESTED: + { + const createProperties = action.get('createProperties') + if (createProperties.get('pinned')) { + state = pinnedSitesState.addPinnedSite(state, pinnedSitesUtil.getDetailFromProperties(createProperties)) + } + break + } + case appConstants.APP_ON_PINNED_TAB_REORDER: + { + state = pinnedSitesState.reOrderSite( + state, + action.siteKey, + action.destinationKey, + action.prepend + ) + + // TODO do we need this for pinned sites? + if (syncUtil.syncEnabled()) { + const newSite = state.getIn(['pinnedSites', action.siteKey]) + state = syncUtil.updateSiteCache(state, newSite) + } + break + } + } + + return state +} + +module.exports = pinnedSitesReducer diff --git a/app/browser/reducers/sitesReducer.js b/app/browser/reducers/sitesReducer.js deleted file mode 100644 index 6f5035404bf..00000000000 --- a/app/browser/reducers/sitesReducer.js +++ /dev/null @@ -1,192 +0,0 @@ -/* 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/. */ - -'use strict' - -const appConstants = require('../../../js/constants/appConstants') -const filtering = require('../../filtering') -const siteCache = require('../../common/state/siteCache') -const siteTags = require('../../../js/constants/siteTags') -const siteUtil = require('../../../js/state/siteUtil') -const syncActions = require('../../../js/actions/syncActions') -const syncUtil = require('../../../js/state/syncUtil') -const Immutable = require('immutable') -const settings = require('../../../js/constants/settings') -const {getSetting} = require('../../../js/settings') -const writeActions = require('../../../js/constants/sync/proto').actions -const tabState = require('../../common/state/tabState') - -const syncEnabled = () => { - return getSetting(settings.SYNC_ENABLED) === true -} - -const updateTabBookmarked = (state, tabValue) => { - if (!tabValue || !tabValue.get('tabId')) { - return state - } - const bookmarked = siteUtil.isLocationBookmarked(state, tabValue.get('url')) - return tabState.updateTabValue(state, tabValue.set('bookmarked', bookmarked)) -} - -const updateActiveTabBookmarked = (state) => { - const tab = tabState.getActiveTab(state) - if (!tab) { - return state - } - return updateTabBookmarked(state, tab) -} - -const sitesReducer = (state, action, immutableAction) => { - switch (action.actionType) { - case appConstants.APP_SET_STATE: - state = siteCache.loadLocationSiteKeysCache(state) - break - case appConstants.APP_ON_CLEAR_BROWSING_DATA: - { - const defaults = state.get('clearBrowsingDataDefaults') - const temp = state.get('tempClearBrowsingData', Immutable.Map()) - const clearData = defaults ? defaults.merge(temp) : temp - if (clearData.get('browserHistory')) { - state = state.set('sites', siteUtil.clearHistory(state.get('sites'))) - filtering.clearHistory() - } - break - } - case appConstants.APP_ADD_SITE: - { - const isSyncEnabled = syncEnabled() - if (Immutable.List.isList(action.siteDetail)) { - action.siteDetail.forEach((s) => { - state = siteUtil.addSite(state, s, action.tag, action.skipSync) - if (isSyncEnabled) { - state = syncUtil.updateSiteCache(state, s) - } - }) - } else { - let sites = state.get('sites') - if (!action.siteDetail.get('folderId') && siteUtil.isFolder(action.siteDetail)) { - action.siteDetail = action.siteDetail.set('folderId', siteUtil.getNextFolderId(sites)) - } - state = siteUtil.addSite(state, action.siteDetail, action.tag, action.skipSync) - if (isSyncEnabled) { - state = syncUtil.updateSiteCache(state, action.siteDetail) - } - } - break - } - case appConstants.APP_ADD_BOOKMARK: - case appConstants.APP_EDIT_BOOKMARK: - { - const isSyncEnabled = syncEnabled() - const sites = state.get('sites') - const closestKey = action.closestKey - let site = action.siteDetail - - if (site == null || action.tag == null) { - break - } - - if (!site.get('folderId') && action.tag === siteTags.BOOKMARK_FOLDER) { - site = site.set('folderId', siteUtil.getNextFolderId(sites)) - } - - state = siteUtil.addSite(state, site, action.tag, action.editKey) - - if (closestKey != null) { - const sourceKey = siteUtil.getSiteKey(site) - state = siteUtil.moveSite(state, sourceKey, closestKey, false, false, true) - } - - if (isSyncEnabled) { - state = syncUtil.updateSiteCache(state, site) - } - - state = updateActiveTabBookmarked(state) - break - } - case appConstants.APP_REMOVE_SITE: - const removeSiteSyncCallback = action.skipSync ? undefined : syncActions.removeSite - state = siteUtil.removeSite(state, action.siteDetail, action.tag, true, removeSiteSyncCallback) - if (syncEnabled()) { - state = syncUtil.updateSiteCache(state, action.siteDetail) - } - state = updateActiveTabBookmarked(state) - break - case appConstants.APP_MOVE_SITE: - state = siteUtil.moveSite(state, - action.sourceKey, action.destinationKey, action.prepend, - action.destinationIsParent, false) - state = state.set('sites', state.get('sites').sort(siteUtil.siteSort)) - if (syncEnabled()) { - const destinationDetail = state.getIn(['sites', action.destinationKey]) - state = syncUtil.updateSiteCache(state, destinationDetail) - } - break - case appConstants.APP_APPLY_SITE_RECORDS: - let nextFolderId = siteUtil.getNextFolderId(state.get('sites')) - // Ensure that all folders are assigned folderIds - action.records.forEach((record, i) => { - if (record.action !== writeActions.DELETE && - record.bookmark && record.bookmark.isFolder && - record.bookmark.site && - typeof record.bookmark.site.folderId !== 'number') { - record.bookmark.site.folderId = nextFolderId - action.records.set(i, record) - nextFolderId = nextFolderId + 1 - } - }) - action.records.forEach((record) => { - if (record.action === writeActions.DELETE) { - state = siteUtil.removeSiteByObjectId(state, record.objectId, record.objectData) - } else { - const siteData = syncUtil.getSiteDataFromRecord(record, state, action.records) - const tag = siteData.tag - let siteDetail = siteData.siteDetail - switch (record.action) { - case writeActions.CREATE: - state = siteUtil.addSite(state, siteDetail, tag, undefined, true) - break - case writeActions.UPDATE: - state = siteUtil.addSite(state, siteDetail, tag, siteData.existingObjectData, true) - break - } - state = syncUtil.updateSiteCache(state, siteDetail) - } - }) - break - case appConstants.APP_TAB_UPDATED: - if (immutableAction.getIn(['changeInfo', 'pinned']) != null) { - const pinned = immutableAction.getIn(['changeInfo', 'pinned']) - const tabId = immutableAction.getIn(['tabValue', 'tabId']) - const tab = state.get('tabs').find((tab) => tab.get('tabId') === tabId) - if (!tab) { - console.warn('Trying to pin a tabId which does not exist:', tabId, 'tabs: ', state.get('tabs').toJS()) - break - } - const sites = state.get('sites') - const siteDetail = siteUtil.getDetailFromTab(tab, siteTags.PINNED, sites) - if (pinned) { - state = siteUtil.addSite(state, siteDetail, siteTags.PINNED) - } else { - state = siteUtil.removeSite(state, siteDetail, siteTags.PINNED) - } - if (syncEnabled()) { - state = syncUtil.updateSiteCache(state, siteDetail) - } - } - state = updateTabBookmarked(state, action.tabValue) - break - case appConstants.APP_CREATE_TAB_REQUESTED: { - const createProperties = immutableAction.get('createProperties') - if (createProperties.get('pinned')) { - state = siteUtil.addSite(state, - siteUtil.getDetailFromCreateProperties(createProperties), siteTags.PINNED) - } - break - } - } - return state -} - -module.exports = sitesReducer diff --git a/app/browser/reducers/urlBarSuggestionsReducer.js b/app/browser/reducers/urlBarSuggestionsReducer.js index 8873091a9ad..b69c594a0ef 100644 --- a/app/browser/reducers/urlBarSuggestionsReducer.js +++ b/app/browser/reducers/urlBarSuggestionsReducer.js @@ -4,16 +4,18 @@ 'use strict' +const Immutable = require('immutable') const appConstants = require('../../../js/constants/appConstants') const {generateNewSuggestionsList, generateNewSearchXHRResults} = require('../../common/lib/suggestion') const {init, add} = require('../../common/lib/siteSuggestions') -const Immutable = require('immutable') const {makeImmutable} = require('../../common/state/immutableUtil') const tabState = require('../../common/state/tabState') +const historyState = require('../../common/state/historyState') +const bookmarksState = require('../../common/state/bookmarksState') const urlBarSuggestionsReducer = (state, action) => { switch (action.actionType) { - case appConstants.APP_ADD_SITE: + case appConstants.APP_ADD_HISTORY_SITE: case appConstants.APP_ADD_BOOKMARK: case appConstants.APP_EDIT_BOOKMARK: if (Immutable.List.isList(action.siteDetail)) { @@ -25,7 +27,9 @@ const urlBarSuggestionsReducer = (state, action) => { } break case appConstants.APP_SET_STATE: - init(Object.values(action.appState.get('sites').toJS())) + const bookmarks = bookmarksState.getBookmarks(action.appState) + const history = historyState.getSites(action.appState) + init(Object.values(bookmarks.concat(history).toJS())) break case appConstants.APP_URL_BAR_TEXT_CHANGED: generateNewSuggestionsList(state, action.windowId, action.tabId, action.input) diff --git a/app/browser/tabs.js b/app/browser/tabs.js index 0e6728fc914..3d105b42da4 100644 --- a/app/browser/tabs.js +++ b/app/browser/tabs.js @@ -22,12 +22,15 @@ const messages = require('../../js/constants/messages') const aboutHistoryState = require('../common/state/aboutHistoryState') const appStore = require('../../js/stores/appStore') const appConfig = require('../../js/constants/appConfig') -const siteTags = require('../../js/constants/siteTags') const {newTabMode} = require('../common/constants/settingsEnums') const {cleanupWebContents, currentWebContents, getWebContents, updateWebContents} = require('./webContentsCache') const {FilterOptions} = require('ad-block') const {isResourceEnabled} = require('../filtering') const autofill = require('../autofill') +const bookmarksState = require('../common/state/bookmarksState') +const bookmarkFoldersState = require('../common/state/bookmarkFoldersState') +const historyState = require('../common/state/historyState') +const bookmarkOrderCache = require('../common/cache/bookmarkOrderCache') let currentPartitionNumber = 0 const incrementPartitionNumber = () => ++currentPartitionNumber @@ -134,21 +137,12 @@ ipcMain.on(messages.ABOUT_COMPONENT_INITIALIZED, (e) => { }) }) -const getBookmarksData = function (state) { - let bookmarkSites = new Immutable.OrderedMap() - let bookmarkFolderSites = new Immutable.OrderedMap() - state.get('sites').forEach((site, siteKey) => { - const tags = site.get('tags') - if (tags.includes(siteTags.BOOKMARK)) { - bookmarkSites = bookmarkSites.set(siteKey, site) - } - if (tags.includes(siteTags.BOOKMARK_FOLDER)) { - bookmarkFolderSites = bookmarkFolderSites.set(siteKey, site) - } - }) - const bookmarks = bookmarkSites.toList().toJS() - const bookmarkFolders = bookmarkFolderSites.toList().toJS() - return {bookmarks, bookmarkFolders} +const getBookmarksData = (state) => { + return { + bookmarks: bookmarksState.getBookmarks(state).toJS(), + bookmarkFolders: bookmarkFoldersState.getFolders(state).toJS(), + bookmarkOrder: bookmarkOrderCache.getOrderCache(state).toJS() + } } const updateAboutDetails = (tab, tabValue) => { @@ -813,7 +807,7 @@ const api = { getHistoryEntries: (state, action) => { const tab = getWebContents(action.get('tabId')) - const sites = state ? state.get('sites') : null + const sites = state ? historyState.getSites(state) : null if (tab && !tab.isDestroyed()) { let history = { @@ -837,7 +831,7 @@ const api = { // TODO: return brave lion (or better: get icon from extension if possible as data URI) } else { if (sites) { - const site = sites.find(function (element) { return element.get('location') === url }) + const site = sites.find((element) => element.get('location') === url) if (site) { entry.icon = site.get('favicon') } diff --git a/app/browser/windows.js b/app/browser/windows.js index 028aee44eed..fec40cd6d14 100644 --- a/app/browser/windows.js +++ b/app/browser/windows.js @@ -9,14 +9,14 @@ const debounce = require('../../js/lib/debounce') const {getSetting} = require('../../js/settings') const locale = require('../locale') const LocalShortcuts = require('../localShortcuts') -const {getPinnedSiteProps} = require('../common/lib/windowsUtil') const {makeImmutable} = require('../common/state/immutableUtil') const {getPinnedTabsByWindowId} = require('../common/state/tabState') const messages = require('../../js/constants/messages') const settings = require('../../js/constants/settings') -const siteTags = require('../../js/constants/siteTags') const windowState = require('../common/state/windowState') const Immutable = require('immutable') +const pinnedSitesState = require('../common/state/pinnedSitesState') +const pinnedSitesUtil = require('../common/lib/pinnedSitesUtil') // TODO(bridiver) - set window uuid let currentWindows = {} @@ -69,8 +69,7 @@ const updatePinnedTabs = (win) => { const appStore = require('../../js/stores/appStore') const state = appStore.getState() const windowId = win.id - const pinnedSites = state.get('sites').toList().filter((site) => - site.get('tags').includes(siteTags.PINNED)).map(site => getPinnedSiteProps(site)) + const pinnedSites = pinnedSitesState.getSites(state).map(site => pinnedSitesUtil.getPinnedSiteProps(site)) const pinnedTabs = getPinnedTabsByWindowId(state, windowId) pinnedSites.filter((site) => diff --git a/app/common/state/siteCache.js b/app/common/cache/bookmarkLocationCache.js similarity index 63% rename from app/common/state/siteCache.js rename to app/common/cache/bookmarkLocationCache.js index beb1a207928..659cb13282c 100644 --- a/app/common/state/siteCache.js +++ b/app/common/cache/bookmarkLocationCache.js @@ -3,22 +3,9 @@ 'use strict' const Immutable = require('immutable') -const siteUtil = require('../../../js/state/siteUtil') const appUrlUtil = require('../../../js/lib/appUrlUtil') const UrlUtil = require('../../../js/lib/urlutil') -const createLocationSiteKeysCache = (state) => { - state = state.set('locationSiteKeysCache', new Immutable.Map()) - state.get('sites').forEach((site, siteKey) => { - const location = siteUtil.getLocationFromSiteKey(siteKey) - if (!location) { - return - } - state = addLocationSiteKey(state, location, siteKey) - }) - return state -} - const normalizeLocation = (location) => { const sourceAboutUrl = appUrlUtil.getSourceAboutUrl(location) if (sourceAboutUrl) { @@ -27,12 +14,40 @@ const normalizeLocation = (location) => { return UrlUtil.getLocationIfPDF(location) } -module.exports.loadLocationSiteKeysCache = (state) => { - const cache = state.get('locationSiteKeysCache') +/** + * Calculate location for siteKey + * + * @param siteKey The site key to to be calculated + * @return {string|null} + */ +const getLocationFromCacheKey = function (siteKey) { + if (!siteKey) { + return null + } + + const splitKey = siteKey.split('|', 2) + if (typeof splitKey[0] === 'string' && typeof splitKey[1] === 'string') { + return splitKey[0] + } + return null +} + +const generateCache = (state) => { + const cache = state.getIn(['cache', 'bookmarkLocation']) if (cache) { return state } - return createLocationSiteKeysCache(state) + + state = state.setIn(['cache', 'bookmarkLocation'], new Immutable.Map()) + const bookmarksState = require('../state/bookmarksState') + bookmarksState.getBookmarks(state).forEach((site, siteKey) => { + const location = getLocationFromCacheKey(siteKey) + if (!location) { + return + } + state = addCacheKey(state, location, siteKey) + }) + return state } /** @@ -42,9 +57,9 @@ module.exports.loadLocationSiteKeysCache = (state) => { * @param location {string} * @return {Immutable.List|null} siteKeys including this location. */ -module.exports.getLocationSiteKeys = (state, location) => { +const getCacheKey = (state, location) => { const normalLocation = normalizeLocation(location) - return state.getIn(['locationSiteKeysCache', normalLocation]) + return state.getIn(['cache', 'bookmarkLocation', normalLocation], Immutable.List()) } /** @@ -54,13 +69,15 @@ module.exports.getLocationSiteKeys = (state, location) => { * @param location {string} * @param siteKey {string} */ -const addLocationSiteKey = (state, location, siteKey) => { +const addCacheKey = (state, location, siteKey) => { if (!siteKey || !location) { return state } + const normalLocation = normalizeLocation(location) - const cacheKey = ['locationSiteKeysCache', normalLocation] + const cacheKey = ['cache', 'bookmarkLocation', normalLocation] const siteKeys = state.getIn(cacheKey) + if (!siteKeys) { return state.setIn(cacheKey, new Immutable.List([siteKey])) } else { @@ -70,7 +87,6 @@ const addLocationSiteKey = (state, location, siteKey) => { return state.setIn(cacheKey, siteKeys.push(siteKey)) } } -module.exports.addLocationSiteKey = addLocationSiteKey /** * Given a location, remove matching appState siteKeys in cache. @@ -79,16 +95,17 @@ module.exports.addLocationSiteKey = addLocationSiteKey * @param location {string} * @param siteKey {string} */ -const removeLocationSiteKey = (state, location, siteKey) => { +const removeCacheKey = (state, location, siteKey) => { if (!siteKey || !location) { return state } const normalLocation = normalizeLocation(location) - const cacheKey = ['locationSiteKeysCache', normalLocation] + const cacheKey = ['cache', 'bookmarkLocation', normalLocation] let siteKeys = state.getIn(cacheKey) if (!siteKeys) { return state } + siteKeys = siteKeys.filter(key => key !== siteKey) if (siteKeys.size > 0) { return state.setIn(cacheKey, siteKeys) @@ -96,4 +113,10 @@ const removeLocationSiteKey = (state, location, siteKey) => { return state.deleteIn(cacheKey) } } -module.exports.removeLocationSiteKey = removeLocationSiteKey + +module.exports = { + generateCache, + getCacheKey, + addCacheKey, + removeCacheKey +} diff --git a/app/common/cache/bookmarkOrderCache.js b/app/common/cache/bookmarkOrderCache.js new file mode 100644 index 00000000000..62dd73eae0e --- /dev/null +++ b/app/common/cache/bookmarkOrderCache.js @@ -0,0 +1,131 @@ +const Immutable = require('immutable') +const siteTags = require('../../../js/constants/siteTags') +const bookmarkFoldersUtil = require('../lib/bookmarkFoldersUtil') + +const setOrder = (cache, key, tag, destinationKey, append = true) => { + let newCache = Immutable.List() + let i = 0 + + for (let item of cache) { + if (item.get('key') === destinationKey) { + if (append) { + newCache = newCache.push(item.set('order', i)) + i++ + newCache = newCache.push(Immutable.fromJS({ + key: key, + order: i, + type: tag + })) + } else { + newCache = newCache.push(Immutable.fromJS({ + key: key, + order: i, + type: tag + })) + i++ + newCache = newCache.push(item.set('order', i)) + } + } else if (item.get('key') === key) { + continue + } else { + newCache = newCache.push(item.set('order', i)) + } + i++ + } + + return newCache +} + +const addCacheItem = (state, parentId = 0, key, destinationKey, tag, append) => { + parentId = parentId.toString() + key = key.toString() + // cache with this parentId doesn't exist yet + if (!state.hasIn(['cache', 'bookmarkOrder', parentId])) { + return state.setIn(['cache', 'bookmarkOrder', parentId], Immutable.fromJS([ + { + key: key, + order: 0, + type: tag + } + ])) + } + + const cache = state.getIn(['cache', 'bookmarkOrder', parentId]) + // destination key is not provided + if (destinationKey == null) { + return state.setIn(['cache', 'bookmarkOrder', parentId], cache.push(Immutable.fromJS( + { + key: key, + order: cache.size, + type: tag + } + ))) + } + + // destination key is given + const newCache = setOrder(cache, key, tag, destinationKey, append) + return state.setIn(['cache', 'bookmarkOrder', parentId], newCache) +} + +const addBookmarkToCache = (state, parentId, key, destinationKey, append) => { + return addCacheItem(state, parentId, key, destinationKey, siteTags.BOOKMARK, append) +} + +const addFolderToCache = (state, parentId, key, destinationKey, append) => { + return addCacheItem(state, parentId, key, destinationKey, siteTags.BOOKMARK_FOLDER, append) +} + +const getFoldersByParentId = (state, parentId) => { + return state.getIn(['cache', 'bookmarkOrder', parentId.toString()], Immutable.List()) + .filter(item => bookmarkFoldersUtil.isFolder(item)) +} + +const getBookmarksByParentId = (state, parentId = 0) => { + return state.getIn(['cache', 'bookmarkOrder', parentId.toString()], Immutable.List()) + .filter(item => bookmarkFoldersUtil.isFolder(item)) +} + +const getBookmarksWithFolders = (state, parentId) => { + return state.getIn(['cache', 'bookmarkOrder', parentId.toString()], Immutable.List()) +} + +const removeCacheKey = (state, parentId, key) => { + parentId = parentId.toString() + key = key.toString() + const cache = state.getIn(['cache', 'bookmarkOrder', parentId]) + + if (cache == null) { + return state + } + + let newCache = Immutable.List() + let i = 0 + + for (let item of cache) { + if (item.get('key') !== key) { + newCache = newCache.push(item.set('order', i)) + i++ + } + } + + return state.setIn(['cache', 'bookmarkOrder', parentId], newCache) +} + +const removeCacheParent = (state, parentId) => { + return state.deleteIn(['cache', 'bookmarkOrder', parentId.toString()]) +} + +const getOrderCache = (state) => { + return state.getIn(['cache', 'bookmarkOrder'], Immutable.Map()) +} + +module.exports = { + addBookmarkToCache, + addFolderToCache, + removeCacheKey, + getFoldersByParentId, + getBookmarksByParentId, + getBookmarksWithFolders, + removeCacheParent, + getOrderCache +} diff --git a/app/common/lib/bookmarkFoldersUtil.js b/app/common/lib/bookmarkFoldersUtil.js new file mode 100644 index 00000000000..9afde39b0cb --- /dev/null +++ b/app/common/lib/bookmarkFoldersUtil.js @@ -0,0 +1,62 @@ +/* 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/. */ + +const siteTags = require('../../../js/constants/siteTags') + +const isFolderNameValid = (title) => { + return title != null && title.trim().length > 0 +} + +const getNextFolderIdItem = (folders) => + folders.max((folderA, folderB) => { + const folderIdA = folderA.get('folderId') + const folderIdB = folderB.get('folderId') + if (folderIdA === folderIdB) { + return 0 + } + if (folderIdA === undefined) { + return false + } + if (folderIdB === undefined) { + return true + } + return folderIdA > folderIdB + }) + +const getNextFolderId = (folders) => { + const defaultFolderId = 0 + if (!folders) { + return defaultFolderId + } + const maxIdItem = getNextFolderIdItem(folders) + return (maxIdItem ? (maxIdItem.get('folderId') || 0) : 0) + 1 +} + +const getNextFolderName = (folders, name) => { + if (!folders) { + return name + } + const site = folders.find((site) => site.get('title') === name) + if (!site) { + return name + } + const filenameFormat = /(.*) \((\d+)\)/ + let result = filenameFormat.exec(name) + if (!result) { + return getNextFolderName(folders, name + ' (1)') + } + + const nextNum = parseInt(result[2]) + 1 + return getNextFolderName(folders, result[1] + ' (' + nextNum + ')') +} + +const isFolder = (folder) => { + return folder.get('type') === siteTags.BOOKMARK_FOLDER +} + +module.exports = { + isFolderNameValid, + getNextFolderId, + getNextFolderName, + isFolder +} diff --git a/app/common/lib/bookmarkUtil.js b/app/common/lib/bookmarkUtil.js index 953f7041946..67a90f1a553 100644 --- a/app/common/lib/bookmarkUtil.js +++ b/app/common/lib/bookmarkUtil.js @@ -4,13 +4,18 @@ const Immutable = require('immutable') +// State +const bookmarksState = require('../state/bookmarksState') +const tabState = require('../state/tabState') + // Constants const dragTypes = require('../../../js/constants/dragTypes') const {bookmarksToolbarMode} = require('../constants/settingsEnums') const settings = require('../../../js/constants/settings') +const siteTags = require('../../../js/constants/siteTags') // Utils -const siteUtil = require('../../../js/state/siteUtil') +const bookmarkLocationCache = require('../cache/bookmarkLocationCache') const {calculateTextWidth} = require('../../../js/lib/textCalculator') const {iconSize} = require('../../../js/constants/config') const {getSetting} = require('../../../js/settings') @@ -18,13 +23,7 @@ const {getSetting} = require('../../../js/settings') // Styles const globalStyles = require('../../renderer/components/styles/global') -function bookmarkHangerHeading (editMode, isFolder, isAdded) { - if (isFolder) { - return editMode - ? 'bookmarkFolderEditing' - : 'bookmarkFolderAdding' - } - +const bookmarkHangerHeading = (editMode, isAdded) => { if (isAdded) { return 'bookmarkAdded' } @@ -34,19 +33,8 @@ function bookmarkHangerHeading (editMode, isFolder, isAdded) { : 'bookmarkCreateNew' } -const displayBookmarkName = (detail) => { - const customTitle = detail.get('customTitle') - if (customTitle !== undefined && customTitle !== '') { - return customTitle || '' - } - return detail.get('title') || '' -} - -const isBookmarkNameValid = (title, location, isFolder, customTitle) => { - const newTitle = title || customTitle - return isFolder - ? (newTitle != null && newTitle !== 0) && newTitle.trim().length > 0 - : location != null && location.trim().length > 0 +const isBookmarkNameValid = (location) => { + return location != null && location.trim().length > 0 } const showOnlyText = () => { @@ -77,24 +65,18 @@ const getDNDBookmarkData = (state, bookmarkKey) => { return data.get('draggingOverKey') === bookmarkKey ? data : Immutable.Map() } -let oldSites +let oldBookmarks let lastValue let lastWidth const getToolbarBookmarks = (state) => { - const sites = state.get('sites', Immutable.List()) + const bookmarks = bookmarksState.getBookmarksWithFolders(state) const windowWidth = window.innerWidth - - if (sites === oldSites && lastWidth === windowWidth && lastValue) { + if (bookmarks === oldBookmarks && lastWidth === windowWidth && lastValue) { return lastValue } - - oldSites = sites + oldBookmarks = bookmarks lastWidth = windowWidth - const noParentItems = siteUtil.getBookmarks(sites) - .filter((bookmark) => !bookmark.get('parentFolderId')) - .sort(siteUtil.siteSort) - let widthAccountedFor = 0 const onlyText = showOnlyText() @@ -123,40 +105,33 @@ const getToolbarBookmarks = (state) => { // Loop through until we fill up the entire bookmark toolbar width let i = 0 - for (let bookmark of noParentItems) { - const current = bookmark[1] - + for (let bookmark of bookmarks) { let iconWidth if (onlyText) { iconWidth = 0 - } else if (textAndFavicon || current.get('folderId')) { + } else if (textAndFavicon || bookmark.get('folderId')) { iconWidth = iconSize + parseInt(globalStyles.spacing.bookmarksItemMargin, 10) } else if (onlyFavicon) { iconWidth = iconSize } - const currentChevronWidth = current.get('folderId') ? chevronWidth : 0 - + const currentChevronWidth = bookmark.get('folderId') ? chevronWidth : 0 + const text = bookmark.get('title') || bookmark.get('location') let extraWidth if (onlyText) { - const text = current.get('customTitle') || current.get('title') || current.get('location') - extraWidth = padding + calculateTextWidth(text, `${fontSize} ${fontFamily}`) - if (current.get('folderId')) { + if (bookmark.get('folderId')) { extraWidth += currentChevronWidth } } else if (textAndFavicon) { - const text = current.get('customTitle') || current.get('title') || current.get('location') - extraWidth = padding + iconWidth + calculateTextWidth(text, `${fontSize} ${fontFamily}`) + currentChevronWidth } else if (onlyFavicon) { extraWidth = padding + iconWidth + currentChevronWidth - if (current.get('folderId')) { - const text = current.get('customTitle') || current.get('title') || current.get('location') + if (bookmark.get('folderId')) { extraWidth += calculateTextWidth(text, `${fontSize} ${fontFamily}`) } } @@ -175,19 +150,100 @@ const getToolbarBookmarks = (state) => { } lastValue = { - visibleBookmarks: noParentItems.take(i).map((item, index) => index).toList(), + visibleBookmarks: bookmarks.take(i).map((item) => item.get('key')).toList(), // Show at most 100 items in the overflow menu - hiddenBookmarks: noParentItems.skip(i).take(100).map((item, index) => index).toList() + hiddenBookmarks: bookmarks.skip(i).take(100).map((item) => item.get('key')).toList() } + return lastValue } +const getDetailFromFrame = (frame) => { + return Immutable.fromJS({ + location: frame.get('location'), + title: frame.get('title'), + partitionNumber: frame.get('partitionNumber'), + favicon: frame.get('icon'), + themeColor: frame.get('themeColor') || frame.get('computedThemeColor') + }) +} + +/** + * Checks if a location is bookmarked. + * + * @param state The application state Immutable map + * @param {string} location + * @return {boolean} + */ +const isLocationBookmarked = (state, location) => { + const bookmarks = bookmarksState.getBookmarks(state) + const siteKeys = bookmarkLocationCache.getCacheKey(state, location) + + if (siteKeys.isEmpty()) { + return false + } + + return siteKeys.some(key => bookmarks.has(key)) +} + +/** + * Converts a siteDetail to createProperties format + * @param {Object} bookmark - A bookmark detail as per app state + * @return {Object} A createProperties plain JS object, not ImmutableJS + */ +const toCreateProperties = (bookmark) => { + return { + url: bookmark.get('location'), + partitionNumber: bookmark.get('partitionNumber') + } +} + +/** + * Filters bookmarks relative to a parent folder + * @param state - The application state + * @param folderKey The folder key to filter to + */ +const getBookmarksByParentId = (state, folderKey) => { + const bookmarks = bookmarksState.getBookmarks(state) + if (!folderKey) { + return bookmarks + } + + return bookmarks.filter((bookmark) => bookmark.get('parentFolderId') === folderKey) +} + +const isBookmark = (bookmark) => { + return bookmark.get('type') === siteTags.BOOKMARK +} + +const updateTabBookmarked = (state, tabValue) => { + if (!tabValue || !tabValue.get('tabId')) { + return state + } + const bookmarked = isLocationBookmarked(state, tabValue.get('url')) + return tabState.updateTabValue(state, tabValue.set('bookmarked', bookmarked)) +} + +const updateActiveTabBookmarked = (state) => { + const tab = tabState.getActiveTab(state) + if (!tab) { + return state + } + return updateTabBookmarked(state, tab) +} + module.exports = { bookmarkHangerHeading, - displayBookmarkName, isBookmarkNameValid, showOnlyFavicon, showFavicon, getDNDBookmarkData, - getToolbarBookmarks + getToolbarBookmarks, + getDetailFromFrame, + isLocationBookmarked, + toCreateProperties, + getBookmarksByParentId, + isBookmark, + updateTabBookmarked, + updateActiveTabBookmarked } diff --git a/app/common/lib/historyUtil.js b/app/common/lib/historyUtil.js index c3cade0a637..15c2f9cf371 100644 --- a/app/common/lib/historyUtil.js +++ b/app/common/lib/historyUtil.js @@ -8,15 +8,13 @@ const {makeImmutable} = require('../state/immutableUtil') const siteUtil = require('../../../js/state/siteUtil') const aboutHistoryMaxEntries = 500 -module.exports.maxEntries = aboutHistoryMaxEntries - const sortTimeDescending = (left, right) => { if (left.get('lastAccessedTime') < right.get('lastAccessedTime')) return 1 if (left.get('lastAccessedTime') > right.get('lastAccessedTime')) return -1 return 0 } -module.exports.getHistory = (sites) => { +const getHistory = (sites) => { sites = makeImmutable(sites) ? makeImmutable(sites).toList() : new Immutable.List() return sites.filter((site) => siteUtil.isHistoryEntry(site)) .sort(sortTimeDescending) @@ -30,7 +28,7 @@ const getDayString = (entry, locale) => { : '' } -module.exports.groupEntriesByDay = (history, locale) => { +const groupEntriesByDay = (history, locale) => { const reduced = history.reduce((previousValue, currentValue, currentIndex, array) => { const result = currentIndex === 1 ? [] : previousValue if (currentIndex === 1) { @@ -60,7 +58,7 @@ module.exports.groupEntriesByDay = (history, locale) => { * Return an array with ALL entries. * Format is expected to be array containing one array per day. */ -module.exports.totalEntries = (entriesByDay) => { +const totalEntries = (entriesByDay) => { entriesByDay = makeImmutable(entriesByDay) || new Immutable.List() let result = new Immutable.List() @@ -69,3 +67,68 @@ module.exports.totalEntries = (entriesByDay) => { }) return result } + +const prepareHistoryEntry = (siteDetail) => { + const time = siteDetail.has('lastAccessedTime') + ? siteDetail.get('lastAccessedTime') + : new Date().getTime() + + return makeImmutable({ + lastAccessedTime: time, + objectId: undefined, + title: siteDetail.get('title'), + location: siteDetail.get('location'), + themeColor: siteDetail.get('themeColor'), + favicon: siteDetail.get('favicon', siteDetail.get('icon')), + count: 1 + }) +} + +const mergeSiteDetails = (oldDetail, newDetail) => { + const objectId = newDetail.has('objectId') ? newDetail.get('objectId') : oldDetail.get('objectId', undefined) + const time = newDetail.has('lastAccessedTime') + ? newDetail.get('lastAccessedTime') + : new Date().getTime() + + let site = makeImmutable({ + title: newDetail.get('title'), + location: newDetail.get('location'), + count: ~~oldDetail.get('count', 0) + 1, + lastAccessedTime: time, + objectId + }) + + const themeColor = newDetail.has('themeColor') ? newDetail.get('themeColor') : oldDetail.get('themeColor') + if (themeColor) { + site = site.set('themeColor', themeColor) + } + + // we need to have a fallback to icon, because frame has icon for it + const favicon = (newDetail.has('favicon') || newDetail.has('icon')) + ? newDetail.get('favicon', newDetail.get('icon')) + : oldDetail.get('favicon') + if (favicon) { + site = site.set('favicon', favicon) + } + + return site +} + +const getDetailFromFrame = (frame) => { + return makeImmutable({ + location: frame.get('location'), + title: frame.get('title'), + partitionNumber: frame.get('partitionNumber'), + favicon: frame.get('icon'), + themeColor: frame.get('themeColor') || frame.get('computedThemeColor') + }) +} + +module.exports = { + getHistory, + groupEntriesByDay, + totalEntries, + prepareHistoryEntry, + mergeSiteDetails, + getDetailFromFrame +} diff --git a/app/common/lib/menuUtil.js b/app/common/lib/menuUtil.js index 65f24d89987..0896e270d94 100644 --- a/app/common/lib/menuUtil.js +++ b/app/common/lib/menuUtil.js @@ -4,16 +4,25 @@ 'use strict' const MenuItem = require('electron').MenuItem + +// Constants +const config = require('../../../js/constants/config') + +// State +const bookmarksState = require('../state/bookmarksState') + +// Actions +const appActions = require('../../../js/actions/appActions') +const windowActions = require('../../../js/actions/windowActions') + +// Utils const {makeImmutable} = require('../../common/state/immutableUtil') const CommonMenu = require('../../common/commonMenu') -const siteTags = require('../../../js/constants/siteTags') const eventUtil = require('../../../js/lib/eventUtil') -const siteUtil = require('../../../js/state/siteUtil') const locale = require('../../locale') -const appActions = require('../../../js/actions/appActions') -const config = require('../../../js/constants/config') const {separatorMenuItem} = require('../../common/commonMenu') -const windowActions = require('../../../js/actions/windowActions') +const bookmarkUtil = require('./bookmarkUtil') +const bookmarkFoldersUtil = require('./bookmarkFoldersUtil') /** * Get the an electron MenuItem object from a Menu based on its label @@ -73,14 +82,12 @@ module.exports.setTemplateItemChecked = (template, label, checked) => { return null } -const createBookmarkTemplateItems = (bookmarks, parentFolderId) => { - const filteredBookmarks = parentFolderId - ? bookmarks.filter((bookmark) => bookmark.get('parentFolderId') === parentFolderId) - : bookmarks.filter((bookmark) => !bookmark.get('parentFolderId')) +const createBookmarkTemplateItems = (state, parentFolderId) => { + const bookmarks = bookmarksState.getBookmarksWithFolders(state, parentFolderId) const payload = [] - filteredBookmarks.forEach((site) => { - if (site.get('tags').includes(siteTags.BOOKMARK) && site.get('location')) { + for (let bookmark of bookmarks) { + if (bookmarkUtil.isBookmark(bookmark)) { payload.push({ // TODO include label made from favicon. It needs to be of type NativeImage // which can be made using a Buffer / DataURL / local image @@ -89,38 +96,37 @@ const createBookmarkTemplateItems = (bookmarks, parentFolderId) => { // and as such there may need to be another mechanism or cache // // see: https://github.com/brave/browser-laptop/issues/3050 - label: site.get('customTitle') || site.get('title') || site.get('location'), + label: bookmark.get('title', bookmark.get('location')), click: (item, focusedWindow, e) => { if (eventUtil.isForSecondaryAction(e)) { appActions.createTabRequested({ - url: site.get('location'), + url: bookmark.get('location'), windowId: focusedWindow.id, active: !!e.shiftKey }) } else { - appActions.loadURLInActiveTabRequested(focusedWindow.id, site.get('location')) + appActions.loadURLInActiveTabRequested(focusedWindow.id, bookmark.get('location')) } } }) - } else if (siteUtil.isFolder(site)) { - const folderId = site.get('folderId') - const submenuItems = bookmarks.filter((bookmark) => bookmark.get('parentFolderId') === folderId) + } else if (bookmarkFoldersUtil.isFolder(bookmark)) { payload.push({ - label: site.get('customTitle') || site.get('title'), - submenu: submenuItems.count() > 0 ? createBookmarkTemplateItems(bookmarks, folderId) : null + label: bookmark.get('title'), + submenu: createBookmarkTemplateItems(state, bookmark.get('folderId')) }) } - }) + } + return payload } /** * Used to create bookmarks and bookmark folder entries for the "Bookmarks" menu * - * @param sites The application state's Immutable sites list + * @param state The application state */ -module.exports.createBookmarkTemplateItems = (sites) => { - return createBookmarkTemplateItems(siteUtil.getBookmarks(sites)) +module.exports.createBookmarkTemplateItems = (state) => { + return createBookmarkTemplateItems(state) } /** diff --git a/app/common/lib/pinnedSitesUtil.js b/app/common/lib/pinnedSitesUtil.js new file mode 100644 index 00000000000..d9662763128 --- /dev/null +++ b/app/common/lib/pinnedSitesUtil.js @@ -0,0 +1,115 @@ +/* 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/. */ + +const Immutable = require('immutable') +const siteUtil = require('../../../js/state/siteUtil') +const {makeImmutable} = require('../state/immutableUtil') + +const getSitesBySubkey = (sites, siteKey) => { + if (!sites || !siteKey) { + return makeImmutable([]) + } + const splitKey = siteKey.split('|', 2) + const partialKey = splitKey.join('|') + const matches = sites.filter((site, key) => { + return key.indexOf(partialKey) > -1 + }) + return matches.toList() +} + +const getDetailsFromTab = (sites, tab) => { + let location = tab.get('url') + const partitionNumber = tab.get('partitionNumber') + let parentFolderId + + // TODO check if needed https://github.com/brave/browser-laptop/pull/8588 + // we need to find which sites should be send in, I am guessing bookmarks + + // if site map is available, look up extra information: + // - original url (if redirected) + // - parent folder id + if (sites) { + // get all sites matching URL and partition (disregarding parentFolderId) + let siteKey = siteUtil.getSiteKey(makeImmutable({location, partitionNumber})) + let results = getSitesBySubkey(sites, siteKey) + + // only check for provisional location if entry is not found + if (results.size === 0) { + // if provisional location is different, grab any results which have that URL + // this may be different if the site was redirected + const provisionalLocation = tab.getIn(['frame', 'provisionalLocation']) + if (provisionalLocation && provisionalLocation !== location) { + siteKey = siteUtil.getSiteKey(makeImmutable({ + location: provisionalLocation, + partitionNumber + })) + results = results.merge(getSitesBySubkey(sites, siteKey)) + } + } + + // update details which get returned below + if (results.size > 0) { + location = results.getIn([0, 'location']) + parentFolderId = results.getIn([0, 'parentFolderId']) + } + } + + const siteDetail = { + location: location, + title: tab.get('title') + } + + // TODO I think that we don't need this one + if (partitionNumber) { + siteDetail.partitionNumber = partitionNumber + } + + if (parentFolderId) { + siteDetail.parentFolderId = parentFolderId + } + + return makeImmutable(siteDetail) +} + +const getDetailFromProperties = (createProperties) => { + const siteDetail = { + location: createProperties.get('url') + } + + if (createProperties.get('partitionNumber') !== undefined) { + siteDetail.partitionNumber = createProperties.get('partitionNumber') + } + return makeImmutable(siteDetail) +} + +const getDetailFromFrame = (frame) => { + const pinnedLocation = frame.get('pinnedLocation') + let location = frame.get('location') + if (pinnedLocation !== 'about:blank') { + location = pinnedLocation + } + + return makeImmutable({ + location, + title: frame.get('title'), + partitionNumber: frame.get('partitionNumber'), + favicon: frame.get('icon'), + themeColor: frame.get('themeColor') || frame.get('computedThemeColor') + }) +} + +const getPinnedSiteProps = site => { + return Immutable.fromJS({ + location: site.get('location'), + order: site.get('order'), + partitionNumber: site.get('partitionNumber', 0) + }) +} + +module.exports = { + getDetailsFromTab, + getDetailFromProperties, + getDetailFromFrame, + getPinnedSiteProps +} diff --git a/app/common/lib/siteSuggestions.js b/app/common/lib/siteSuggestions.js index 229d1bb4841..e2c40e21255 100644 --- a/app/common/lib/siteSuggestions.js +++ b/app/common/lib/siteSuggestions.js @@ -61,11 +61,8 @@ const tokenizeInput = (data) => { return [] } url = data.location - if (data.customTitle) { - parts = getPartsFromNonUrlInput(data.customTitle) - } if (data.title) { - parts = parts.concat(getPartsFromNonUrlInput(data.title)) + parts = getPartsFromNonUrlInput(data.title) } if (data.tags) { parts = parts.concat(data.tags.map(getTagToken)) diff --git a/app/common/lib/windowsUtil.js b/app/common/lib/windowsUtil.js deleted file mode 100644 index 4141a5a060d..00000000000 --- a/app/common/lib/windowsUtil.js +++ /dev/null @@ -1,17 +0,0 @@ -/* 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/. */ - -const Immutable = require('immutable') - -const getPinnedSiteProps = site => { - return Immutable.fromJS({ - location: site.get('location'), - order: site.get('order'), - partitionNumber: site.get('partitionNumber') || 0 - }) -} - -module.exports = { - getPinnedSiteProps -} diff --git a/app/common/state/aboutHistoryState.js b/app/common/state/aboutHistoryState.js index 6f9048993e6..ca79b6ab76c 100644 --- a/app/common/state/aboutHistoryState.js +++ b/app/common/state/aboutHistoryState.js @@ -2,6 +2,7 @@ * 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/. */ +const Immutable = require('immutable') const {makeImmutable} = require('./immutableUtil') const historyUtil = require('../lib/historyUtil') @@ -10,10 +11,16 @@ const aboutHistoryState = { state = makeImmutable(state) return state.getIn(['about', 'history']) }, - setHistory: (state) => { + + setHistory: (state, sites) => { + state = makeImmutable(state) + state = state.setIn(['about', 'history', 'entries'], historyUtil.getHistory(sites)) + return state.setIn(['about', 'history', 'updatedStamp'], new Date().getTime()) + }, + + clearHistory: (state) => { state = makeImmutable(state) - state = state.setIn(['about', 'history', 'entries'], - historyUtil.getHistory(state.get('sites'))) + state = state.setIn(['about', 'history', 'entries'], Immutable.Map()) return state.setIn(['about', 'history', 'updatedStamp'], new Date().getTime()) } } diff --git a/app/common/state/aboutNewTabState.js b/app/common/state/aboutNewTabState.js index ea855edaf04..ac9a551bdd2 100644 --- a/app/common/state/aboutNewTabState.js +++ b/app/common/state/aboutNewTabState.js @@ -31,6 +31,13 @@ const aboutNewTabState = { topSites = makeImmutable(topSites) state = state.setIn(['about', 'newtab', 'sites'], topSites) return state.setIn(['about', 'newtab', 'updatedStamp'], new Date().getTime()) + }, + + clearTopSites: (state) => { + state = makeImmutable(state) + + state = state.setIn(['about', 'newtab', 'sites'], makeImmutable([])) + return state.setIn(['about', 'newtab', 'updatedStamp'], new Date().getTime()) } } diff --git a/app/common/state/bookmarkFoldersState.js b/app/common/state/bookmarkFoldersState.js new file mode 100644 index 00000000000..5c2bae5a335 --- /dev/null +++ b/app/common/state/bookmarkFoldersState.js @@ -0,0 +1,177 @@ +/* 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/. */ + +const assert = require('assert') +const Immutable = require('immutable') + +// Actions +const syncActions = require('../../../js/actions/syncActions') + +// Constants +const settings = require('../../../js/constants/settings') +const siteTags = require('../../../js/constants/siteTags') + +// State +const bookmarkOrderCache = require('../cache/bookmarkOrderCache') + +// Utils +const bookmarkFoldersUtil = require('../lib/bookmarkFoldersUtil') +const siteUtil = require('../../../js/state/siteUtil') +const {makeImmutable, isMap} = require('./immutableUtil') +const {getSetting} = require('../../../js/settings') + +const validateState = function (state) { + state = makeImmutable(state) + assert.ok(isMap(state), 'state must be an Immutable.Map') + assert.ok(isMap(state.get('bookmarkFolders')), 'state must contain an Immutable.Map of bookmarkFolders') + return state +} + +const bookmarkFoldersState = { + getFolders: (state) => { + state = validateState(state) + return state.get('bookmarkFolders', Immutable.Map()) + }, + + getFolder: (state, folderKey) => { + state = validateState(state) + folderKey = folderKey.toString() + return state.getIn(['bookmarkFolders', folderKey], Immutable.Map()) + }, + + getFoldersByParentId: (state, parentFolderId) => { + state = validateState(state) + + const folders = bookmarkOrderCache.getFoldersByParentId(state, parentFolderId) + return folders.map(folder => bookmarkFoldersState.getFolder(state, folder.get('key'))) + }, + + addFolder: (state, folderDetails, destinationKey) => { + state = validateState(state) + folderDetails = makeImmutable(folderDetails) + let folders = bookmarkFoldersState.getFolders(state) + let key = folderDetails.get('folderId') + + if (!folderDetails.has('folderId')) { + key = bookmarkFoldersUtil.getNextFolderId(folders) + } + + const newFolder = makeImmutable({ + title: folderDetails.get('title'), + folderId: ~~key, + key: key.toString(), + parentFolderId: ~~folderDetails.get('parentFolderId', 0), + partitionNumber: ~~folderDetails.get('partitionNumber', 0), + objectId: null, + type: siteTags.BOOKMARK_FOLDER + }) + + state = state.setIn(['bookmarkFolders', key.toString()], newFolder) + state = bookmarkOrderCache.addFolderToCache(state, newFolder.get('parentFolderId'), key, destinationKey) + return state + }, + + editFolder: (state, folderDetails, editKey) => { + state = validateState(state) + const oldFolder = bookmarkFoldersState.getFolder(state, editKey) + + const newFolder = oldFolder.merge(makeImmutable({ + title: folderDetails.get('title'), + parentFolderId: ~~folderDetails.get('parentFolderId', 0) + })) + + if (oldFolder.get('parentFolderId') !== newFolder.get('parentFolderId')) { + state = bookmarkOrderCache.removeCacheKey(state, oldFolder.get('parentFolderId'), editKey) + state = bookmarkOrderCache.addFolderToCache(state, newFolder.get('parentFolderId'), editKey) + } + + state = state.setIn(['bookmarkFolders', editKey.toString()], newFolder) + return state + }, + + removeFolder: (state, folderKey) => { + const bookmarksState = require('./bookmarksState') + const folders = bookmarkFoldersState.getFolders(state) + const folder = bookmarkFoldersState.getFolder(state, folderKey) + + if (folder.isEmpty()) { + return state + } + + if (getSetting(settings.SYNC_ENABLED) === true) { + syncActions.removeSite(folder) + } + + folders.filter(folder => folder.get('parentFolderId') === ~~folderKey) + .map(folder => { + state = bookmarksState.removeBookmarksByParentId(state, folder.get('folderId')) + state = bookmarkFoldersState.removeFolder(state, folder.get('folderId')) + state = bookmarkOrderCache.removeCacheParent(state, folder.get('folderId')) + state = bookmarkOrderCache.removeCacheKey(state, folder.get('parentFolderId'), folderKey) + }) + + state = bookmarksState.removeBookmarksByParentId(state, folderKey) + state = bookmarkOrderCache.removeCacheParent(state, folderKey) + state = bookmarkOrderCache.removeCacheKey(state, folder.get('parentFolderId'), folderKey) + return state.deleteIn(['bookmarkFolders', folderKey.toString()]) + }, + + getFoldersWithoutKey: (state, folderKey, parentFolderId = 0, labelPrefix = '') => { + let folders = [] + const results = bookmarkFoldersState.getFoldersByParentId(state, parentFolderId) + + const resultSize = results.size + for (let i = 0; i < resultSize; i++) { + const folder = results.get(i) + if (folder.get('folderId') === folderKey) { + continue + } + + const label = labelPrefix + folder.get('title') + folders.push({ + folderId: folder.get('folderId'), + label + }) + const subSites = bookmarkFoldersState.getFoldersWithoutKey(state, folderKey, folder.get('folderId'), (label || '') + ' / ') + folders = folders.concat(subSites) + } + + return folders + }, + + moveFolder: (state, folderKey, destinationKey, append, moveIntoParent) => { + const bookmarksState = require('./bookmarksState') + let folder = bookmarkFoldersState.getFolder(state, folderKey) + let destinationItem = bookmarksState.findBookmark(state, destinationKey) + + if (folder.isEmpty()) { + return state + } + + if (moveIntoParent || destinationItem.get('parentFolderId') !== folder.get('parentFolderId')) { + const parentFolderId = destinationItem.get('type') === siteTags.BOOKMARK + ? destinationItem.get('parentFolderId') + : destinationItem.get('folderId') + + state = bookmarkOrderCache.removeCacheKey(state, folder.get('parentFolderId'), folderKey) + folder = folder.set('parentFolderId', ~~parentFolderId) + const newKey = siteUtil.getSiteKey(folder) + state = state.deleteIn(['bookmarkFolders', folderKey]) + state = bookmarkOrderCache.addFolderToCache(state, folder.get('parentFolderId'), newKey) + return state.setIn(['bookmarkFolders', newKey.toString()], folder) + } + + state = bookmarkOrderCache.removeCacheKey(state, folder.get('parentFolderId'), folderKey) + state = bookmarkOrderCache.addFolderToCache( + state, + folder.get('parentFolderId'), + folderKey, + destinationKey, + append + ) + return state + } +} + +module.exports = bookmarkFoldersState diff --git a/app/common/state/bookmarksState.js b/app/common/state/bookmarksState.js new file mode 100644 index 00000000000..212ea31b06c --- /dev/null +++ b/app/common/state/bookmarksState.js @@ -0,0 +1,249 @@ +/* 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/. */ + +const assert = require('assert') +const Immutable = require('immutable') + +// Constants +const settings = require('../../../js/constants/settings') +const siteTags = require('../../../js/constants/siteTags') +const newTabData = require('../../../js/data/newTabData') + +// State +const historyState = require('./historyState') +const bookmarkOrderCache = require('../cache/bookmarkOrderCache') +const bookmarkFoldersState = require('./bookmarkFoldersState') + +// Actions +const syncActions = require('../../../js/actions/syncActions') + +// Utils +const siteUtil = require('../../../js/state/siteUtil') +const UrlUtil = require('../../../js/lib/urlutil') +const bookmarkLocationCache = require('../cache/bookmarkLocationCache') +const {getSetting} = require('../../../js/settings') +const {makeImmutable, isMap} = require('./immutableUtil') + +const validateState = function (state) { + state = makeImmutable(state) + assert.ok(isMap(state), 'state must be an Immutable.Map') + assert.ok(isMap(state.get('bookmarks')), 'state must contain an Immutable.Map of bookmarks') + return state +} + +const bookmarksState = { + getBookmarks: (state) => { + state = validateState(state) + return state.get('bookmarks') + }, + + getBookmark: (state, key) => { + state = validateState(state) + return state.getIn(['bookmarks', key], Immutable.Map()) + }, + + /** + * Use this function if you only have a key and don't know if key is for folder or regular bookmark + * @param state + * @param key + */ + findBookmark: (state, key) => { + state = validateState(state) + let bookmark = bookmarksState.getBookmark(state, key) + if (bookmark.isEmpty()) { + bookmark = bookmarkFoldersState.getFolder(state, key) + } + + return bookmark + }, + + getBookmarksWithFolders: (state, parentFolderId = 0) => { + state = validateState(state) + + const cache = bookmarkOrderCache.getBookmarksWithFolders(state, parentFolderId) + let bookmarks = Immutable.List() + + for (let item of cache) { + if (item.get('type') === siteTags.BOOKMARK) { + bookmarks = bookmarks.push(bookmarksState.getBookmark(state, item.get('key'))) + } else { + bookmarks = bookmarks.push(bookmarkFoldersState.getFolder(state, item.get('key'))) + } + } + + return bookmarks + }, + + addBookmark: (state, bookmarkDetail, destinationKey) => { + state = validateState(state) + + bookmarkDetail = makeImmutable(bookmarkDetail) + let location + if (bookmarkDetail.has('location')) { + location = UrlUtil.getLocationIfPDF(bookmarkDetail.get('location')) + bookmarkDetail = bookmarkDetail.set('location', location) + } + + const key = siteUtil.getSiteKey(bookmarkDetail) + let dataItem = historyState.getSite(state, key) + + if (dataItem.isEmpty()) { + const topSites = Immutable.fromJS(newTabData.topSites.concat(newTabData.pinnedTopSites)) + const topSite = topSites.find(site => site.get('location') === bookmarkDetail.get('location')) || Immutable.Map() + + if (!topSite.isEmpty()) { + dataItem = topSite + } + } + + let bookmark = makeImmutable({ + title: bookmarkDetail.get('title', ' '), + location: bookmarkDetail.get('location'), + parentFolderId: ~~bookmarkDetail.get('parentFolderId', 0), + partitionNumber: ~~dataItem.get('partitionNumber', 0), + objectId: null, + favicon: dataItem.get('favicon'), + themeColor: dataItem.get('themeColor'), + type: siteTags.BOOKMARK, + key: key + }) + + if (key === null) { + return state + } + + state = state.setIn(['bookmarks', key], bookmark) + state = bookmarkLocationCache.addCacheKey(state, location, key) + state = bookmarkOrderCache.addBookmarkToCache(state, bookmark.get('parentFolderId'), key, destinationKey) + return state + }, + + editBookmark: (state, bookmarkDetail, editKey) => { + state = validateState(state) + + const oldBookmark = bookmarksState.getBookmark(state, editKey) + let newBookmark = oldBookmark.merge(bookmarkDetail) + + let location + if (newBookmark.has('location')) { + location = UrlUtil.getLocationIfPDF(newBookmark.get('location')) + newBookmark = newBookmark.set('location', location) + } + + const newKey = siteUtil.getSiteKey(newBookmark) + if (newKey === null) { + return state + } + + if (editKey !== newKey) { + state = state.deleteIn(['bookmarks', editKey]) + state = bookmarkOrderCache.removeCacheKey(state, oldBookmark.get('parentFolderId'), editKey) + state = bookmarkOrderCache.addBookmarkToCache(state, newBookmark.get('parentFolderId'), newKey) + newBookmark = newBookmark.set('key', newKey) + } + + state = state.setIn(['bookmarks', newKey], newBookmark) + state = bookmarkLocationCache.removeCacheKey(state, oldBookmark.get('location'), editKey) + state = bookmarkLocationCache.addCacheKey(state, location, newKey) + return state + }, + + removeBookmark: (state, bookmarkKey) => { + state = validateState(state) + + const bookmark = bookmarksState.getBookmark(state, bookmarkKey) + + if (bookmark.isEmpty()) { + return state + } + + if (getSetting(settings.SYNC_ENABLED) === true) { + syncActions.removeSite(bookmark) + } + + state = bookmarkLocationCache.removeCacheKey(state, bookmark.get('location'), bookmarkKey) + state = bookmarkOrderCache.removeCacheKey(state, bookmark.get('parentFolderId'), bookmarkKey) + + return state.deleteIn(['bookmarks', bookmarkKey]) + }, + + /** + * Removes bookmarks based on the parent ID + * Cache is cleared in the function that is calling this one + * @param state - App state + * @param parentFolderId - parent id of the folder that we are deleting + */ + removeBookmarksByParentId: (state, parentFolderId) => { + state = validateState(state) + + const bookmarks = bookmarksState.getBookmarks(state) + .filter(bookmark => bookmark.get('parentFolderId') !== ~~parentFolderId) + + return state.set('bookmarks', bookmarks) + }, + + /** + * Update the favicon URL for all entries in the state sites + * which match a given location. Currently, there should only be + * one match, but this will handle multiple. + * + * @param state The application state + * @param location URL for the entry needing an update + * @param favicon favicon URL + */ + updateFavicon: (state, location, favicon) => { + state = validateState(state) + + if (UrlUtil.isNotURL(location)) { + return state + } + + const siteKeys = bookmarkLocationCache.getCacheKey(state, location) + if (siteKeys.isEmpty()) { + return state + } + + siteKeys.forEach((siteKey) => { + state = state.setIn(['bookmarks', siteKey, 'favicon'], favicon) + }) + return state + }, + + moveBookmark: (state, bookmarkKey, destinationKey, append, moveIntoParent) => { + let bookmark = bookmarksState.getBookmark(state, bookmarkKey) + let destinationItem = bookmarksState.findBookmark(state, destinationKey) + + if (bookmark.isEmpty()) { + return state + } + + // move bookmark into a new folder + if (moveIntoParent || destinationItem.get('parentFolderId') !== bookmark.get('parentFolderId')) { + const parentFolderId = destinationItem.get('type') === siteTags.BOOKMARK + ? destinationItem.get('parentFolderId') + : destinationItem.get('folderId') + + state = bookmarkOrderCache.removeCacheKey(state, bookmark.get('parentFolderId'), bookmarkKey) + bookmark = bookmark.set('parentFolderId', ~~parentFolderId) + const newKey = siteUtil.getSiteKey(bookmark) + state = state.deleteIn(['bookmarks', bookmarkKey]) + state = bookmarkOrderCache.addBookmarkToCache(state, bookmark.get('parentFolderId'), newKey) + bookmark = bookmark.set('key', newKey) + return state.setIn(['bookmarks', newKey], bookmark) + } + + // move bookmark to another place + state = bookmarkOrderCache.removeCacheKey(state, bookmark.get('parentFolderId'), bookmarkKey) + state = bookmarkOrderCache.addBookmarkToCache( + state, + bookmark.get('parentFolderId'), + bookmarkKey, + destinationKey, + append + ) + return state + } +} + +module.exports = bookmarksState diff --git a/app/common/state/historyState.js b/app/common/state/historyState.js new file mode 100644 index 00000000000..27617d52f67 --- /dev/null +++ b/app/common/state/historyState.js @@ -0,0 +1,81 @@ +/* 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/. */ + +const assert = require('assert') +const Immutable = require('immutable') +const siteUtil = require('../../../js/state/siteUtil') +const historyUtil = require('../lib/historyUtil') +const urlUtil = require('../../../js/lib/urlutil') +const {makeImmutable, isMap} = require('./immutableUtil') + +const validateState = function (state) { + state = makeImmutable(state) + assert.ok(isMap(state), 'state must be an Immutable.Map') + assert.ok(isMap(state.get('historySites')), 'state must contain an Immutable.Map of historySites') + return state +} + +const historyState = { + getSites: (state) => { + state = validateState(state) + return state.get('historySites', Immutable.Map()) + }, + + getSite: (state, key) => { + state = validateState(state) + return state.getIn(['historySites', key], Immutable.Map()) + }, + + addSite: (state, siteDetail) => { + let sites = historyState.getSites(state) + let siteKey = siteUtil.getSiteKey(siteDetail) + siteDetail = makeImmutable(siteDetail) + + const oldSite = sites.get(siteKey) + let site + if (oldSite) { + site = historyUtil.mergeSiteDetails(oldSite, siteDetail) + } else { + let location + if (siteDetail.has('location')) { + location = urlUtil.getLocationIfPDF(siteDetail.get('location')) + siteDetail = siteDetail.set('location', location) + } + + siteKey = siteUtil.getSiteKey(siteDetail) + site = historyUtil.prepareHistoryEntry(siteDetail) + } + + state = state.setIn(['historySites', siteKey], site) + return state + }, + + removeSite: (state, siteKey) => { + // TODO should we remove this only when a tab with this siteKey is not opened + // if not, we are deleting data that is use for bookmarking + return state.deleteIn(['historySites', siteKey]) + }, + + clearSites: (state) => { + return state.set('historySites', Immutable.Map()) + }, + + updateFavicon: (state, siteDetails, favIcon) => { + const historyKey = siteUtil.getSiteKey(siteDetails) + if (historyKey == null) { + return state + } + + let historyItem = historyState.getSite(state, historyKey) + if (historyItem.isEmpty()) { + return state + } + + historyItem = historyItem.set('favicon', favIcon) + + return state.setIn(['historySites', historyKey], historyItem) + } +} + +module.exports = historyState diff --git a/app/common/state/pinnedSitesState.js b/app/common/state/pinnedSitesState.js new file mode 100644 index 00000000000..53ee335416e --- /dev/null +++ b/app/common/state/pinnedSitesState.js @@ -0,0 +1,120 @@ +/* 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/. */ + +const assert = require('assert') +const Immutable = require('immutable') +const siteUtil = require('../../../js/state/siteUtil') +const urlUtil = require('../../../js/lib/urlutil') +const {makeImmutable, isMap} = require('./immutableUtil') + +const validateState = function (state) { + state = makeImmutable(state) + assert.ok(isMap(state), 'state must be an Immutable.Map') + assert.ok(isMap(state.get('pinnedSites')), 'state must contain an Immutable.Map of pinnedSites') + return state +} + +const pinnedSiteState = { + getSites: (state) => { + state = validateState(state) + return state.get('pinnedSites') + }, + + /** + * Adds the specified siteDetail in appState.pinnedSites. + * @param {Immutable.Map} state The application state Immutable map + * @param {Immutable.Map} site The siteDetail that we want to add + */ + addPinnedSite: (state, site) => { + state = validateState(state) + const sites = pinnedSiteState.getSites(state) || Immutable.Map() + let location + if (site.has('location')) { + location = urlUtil.getLocationIfPDF(site.get('location')) + site = site.set('location', location) + } + + site = site.set('order', sites.size) + + const key = siteUtil.getSiteKey(site) + if (key === null) { + return state + } + + state = state.setIn(['pinnedSites', key], site) + return state + }, + + /** + * Removes the given pinned site from the pinnedSites + * + * @param {Immutable.Map} state The application state Immutable map + * @param {Immutable.Map} siteDetail The siteDetail to be removed + * @return {Immutable.Map} The new state Immutable object + */ + removePinnedSite: (state, siteDetail) => { + state = validateState(state) + const key = siteUtil.getSiteKey(siteDetail) + if (!key) { + return state + } + + const stateKey = ['pinnedSites', key] + let site = state.getIn(stateKey) + if (!site) { + return state + } + + // TODO update order, so that is up to date + + return state.deleteIn(stateKey, site) + }, + + /** + * Moves the specified pinned site from one location to another + * + * @param state The application state Immutable map + * @param sourceKey The site key to move + * @param destinationKey The site key to move to + * @param prepend Whether the destination detail should be prepended or not + * @return The new state Immutable object + */ + reOrderSite: (state, sourceKey, destinationKey, prepend) => { + state = validateState(state) + let sites = state.get('pinnedSites') + let sourceSite = sites.get(sourceKey, Immutable.Map()) + const destinationSite = sites.get(destinationKey, Immutable.Map()) + + if (sourceSite.isEmpty()) { + return state + } + + const sourceSiteIndex = sourceSite.get('order') + const destinationSiteIndex = destinationSite.get('order') + let newIndex = destinationSiteIndex + (prepend ? 0 : 1) + if (destinationSiteIndex > sourceSiteIndex) { + --newIndex + } + + state = state.set('pinnedSites', state.get('pinnedSites').map((site, index) => { + const siteOrder = site.get('order') + if (index === sourceKey) { + return site + } + + if (siteOrder >= newIndex && siteOrder < sourceSiteIndex) { + return site.set('order', siteOrder + 1) + } else if (siteOrder <= newIndex && siteOrder > sourceSiteIndex) { + return site.set('order', siteOrder - 1) + } + + return site + })) + + sourceSite = sourceSite.set('order', newIndex) + return state.setIn(['pinnedSites', sourceKey], sourceSite) + } +} + +module.exports = pinnedSiteState diff --git a/app/common/state/siteState.js b/app/common/state/siteState.js deleted file mode 100644 index 0f74f8390a6..00000000000 --- a/app/common/state/siteState.js +++ /dev/null @@ -1,18 +0,0 @@ -const assert = require('assert') -const {makeImmutable, isMap, isList} = require('./immutableUtil') - -const validateState = function (state) { - state = makeImmutable(state) - assert.ok(isMap(state), 'state must be an Immutable.Map') - assert.ok(isList(state.get('sites')), 'state must contain an Immutable.List of sites') - return state -} - -const siteState = { - getSites: (state) => { - state = validateState(state) - return state.get('sites') - } -} - -module.exports = siteState diff --git a/app/common/state/tabState.js b/app/common/state/tabState.js index 4cc5ece7b24..965125eaac2 100644 --- a/app/common/state/tabState.js +++ b/app/common/state/tabState.js @@ -13,7 +13,6 @@ const windowState = require('./windowState') const { makeImmutable, isMap, isList } = require('./immutableUtil') // this file should eventually replace frameStateUtil const frameStateUtil = require('../../../js/state/frameStateUtil') -const {isLocationBookmarked} = require('../../../js/state/siteUtil') const validateId = function (propName, id) { assert.ok(id, `${propName} cannot be null`) @@ -239,6 +238,13 @@ const tabState = { return state.get('tabs').filter((tab) => !!tab.get('pinned')) }, + isTabPinned: (state, tabId) => { + state = validateState(state) + tabId = validateId('tabId', tabId) + const tab = tabState.getByTabId(state, tabId) + return tab != null ? !!tab.get('pinned') : false + }, + getNonPinnedTabs: (state) => { state = validateState(state) return state.get('tabs').filter((tab) => !tab.get('pinned')) @@ -447,8 +453,9 @@ const tabState = { return state } + const bookmarkUtil = require('../lib/bookmarkUtil') const frameLocation = action.getIn(['frame', 'location']) - const frameBookmarked = isLocationBookmarked(state, frameLocation) + const frameBookmarked = bookmarkUtil.isLocationBookmarked(state, frameLocation) const frameValue = action.get('frame').set('bookmarked', frameBookmarked) tabValue = tabValue.set('frame', makeImmutable(frameValue)) return tabState.updateTabValue(state, tabValue) diff --git a/app/importer.js b/app/importer.js index b88c47d7fac..635fc3f6d57 100644 --- a/app/importer.js +++ b/app/importer.js @@ -10,21 +10,32 @@ const importer = electron.importer const dialog = electron.dialog const BrowserWindow = electron.BrowserWindow const session = electron.session -const siteUtil = require('../js/state/siteUtil') -const AppStore = require('../js/stores/appStore') -const siteTags = require('../js/constants/siteTags') -const appActions = require('../js/actions/appActions') + +// Store +const appStore = require('../js/stores/appStore') + +// State +const tabState = require('./common/state/tabState') +const bookmarksState = require('./common/state/bookmarksState') +const bookmarkFoldersState = require('./common/state/bookmarkFoldersState') + +// Constants const messages = require('../js/constants/messages') const settings = require('../js/constants/settings') -const getSetting = require('../js/settings').getSetting + +// Actions +const appActions = require('../js/actions/appActions') + +// Utils +const {getSetting} = require('../js/settings') const locale = require('./locale') const tabMessageBox = require('./browser/tabMessageBox') const {makeImmutable} = require('./common/state/immutableUtil') -const tabState = require('./common/state/tabState') +const bookmarkFoldersUtil = require('./common/lib/bookmarkFoldersUtil') -var isImportingBookmarks = false -var hasBookmarks -var importedSites +let isImportingBookmarks = false +let hasBookmarks +let bookmarkList exports.init = () => { importer.initialize() @@ -33,10 +44,8 @@ exports.init = () => { exports.importData = (selected) => { if (selected.get('favorites')) { isImportingBookmarks = true - const sites = AppStore.getState().get('sites') - hasBookmarks = sites.find( - (site) => siteUtil.isBookmark(site) || siteUtil.isFolder(site) - ) + const state = appStore.getState() + hasBookmarks = bookmarksState.getBookmarks(state).size > 0 || bookmarkFoldersState.getFolders(state).size > 0 } if (selected !== undefined) { importer.importData(selected.toJS()) @@ -45,10 +54,8 @@ exports.importData = (selected) => { exports.importHTML = (selected) => { isImportingBookmarks = true - const sites = AppStore.getState().get('sites') - hasBookmarks = sites.find( - (site) => siteUtil.isBookmark(site) || siteUtil.isFolder(site) - ) + const state = appStore.getState() + hasBookmarks = bookmarksState.getBookmarks(state).size > 0 || bookmarkFoldersState.getFolders(state).size > 0 const files = dialog.showOpenDialog({ properties: ['openFile'], filters: [{ @@ -69,7 +76,7 @@ importer.on('update-supported-browsers', (e, detail) => { } }) -importer.on('add-history-page', (e, history, visitSource) => { +importer.on('add-history-page', (e, history) => { let sites = [] for (let i = 0; i < history.length; ++i) { const site = { @@ -79,80 +86,71 @@ importer.on('add-history-page', (e, history, visitSource) => { } sites.push(site) } - appActions.addSite(makeImmutable(sites)) + appActions.addHistorySite(makeImmutable(sites)) }) importer.on('add-homepage', (e, detail) => { }) -const getParentFolderId = (path, pathMap, sites, topLevelFolderId, nextFolderIdObject) => { +const getParentFolderId = (path, pathMap, folders, topLevelFolderId, nextFolderIdObject) => { const pathLen = path.length if (!pathLen) { return topLevelFolderId } + const parentFolder = path.pop() let parentFolderId = pathMap[parentFolder] if (parentFolderId === undefined) { parentFolderId = nextFolderIdObject.id++ pathMap[parentFolder] = parentFolderId - const folder = { - customTitle: parentFolder, + folders.push({ + title: parentFolder, folderId: parentFolderId, - parentFolderId: getParentFolderId(path, pathMap, sites, topLevelFolderId, nextFolderIdObject), - lastAccessedTime: 0, - creationTime: (new Date()).getTime(), - tags: [siteTags.BOOKMARK_FOLDER] - } - sites.push(folder) + parentFolderId: getParentFolderId(path, pathMap, folders, topLevelFolderId, nextFolderIdObject) + }) } return parentFolderId } -importer.on('add-bookmarks', (e, bookmarks, topLevelFolder) => { - let nextFolderId = siteUtil.getNextFolderId(AppStore.getState().get('sites')) +importer.on('add-bookmarks', (e, importedBookmarks, topLevelFolder) => { + const state = appStore.getState() + const bookmarkFolders = bookmarkFoldersState.getFolders(state) + let nextFolderId = bookmarkFoldersUtil.getNextFolderId(bookmarkFolders) let nextFolderIdObject = { id: nextFolderId } let pathMap = {} - let sites = [] - let topLevelFolderId = 0 - topLevelFolderId = nextFolderIdObject.id++ - sites.push({ - customTitle: siteUtil.getNextFolderName(AppStore.getState().get('sites'), topLevelFolder), + let folders = [] + let bookmarks = [] + let topLevelFolderId = nextFolderIdObject.id++ + + folders.push({ + title: bookmarkFoldersUtil.getNextFolderName(bookmarkFolders, topLevelFolder), folderId: topLevelFolderId, - parentFolderId: 0, - lastAccessedTime: 0, - creationTime: (new Date()).getTime(), - tags: [siteTags.BOOKMARK_FOLDER] + parentFolderId: 0 }) - for (let i = 0; i < bookmarks.length; ++i) { - let path = bookmarks[i].path - let parentFolderId = getParentFolderId(path, pathMap, sites, topLevelFolderId, nextFolderIdObject) - if (bookmarks[i].is_folder) { + + for (let i = 0; i < importedBookmarks.length; ++i) { + let path = importedBookmarks[i].path + let parentFolderId = getParentFolderId(path, pathMap, folders, topLevelFolderId, nextFolderIdObject) + if (importedBookmarks[i].is_folder) { const folderId = nextFolderIdObject.id++ - pathMap[bookmarks[i].title] = folderId - const folder = { - customTitle: bookmarks[i].title, + pathMap[importedBookmarks[i].title] = folderId + folders.push({ + title: importedBookmarks[i].title, folderId: folderId, - parentFolderId: parentFolderId, - lastAccessedTime: 0, - creationTime: bookmarks[i].creation_time * 1000, - tags: [siteTags.BOOKMARK_FOLDER] - } - sites.push(folder) + parentFolderId: parentFolderId + }) } else { - const site = { - title: bookmarks[i].title, - customTitle: bookmarks[i].title, - location: bookmarks[i].url, - parentFolderId: parentFolderId, - lastAccessedTime: 0, - creationTime: bookmarks[i].creation_time * 1000, - tags: [siteTags.BOOKMARK] - } - sites.push(site) + bookmarks.push({ + title: importedBookmarks[i].title, + location: importedBookmarks[i].url, + parentFolderId: parentFolderId + }) } } - importedSites = makeImmutable(sites) - appActions.addSite(makeImmutable(sites)) + + bookmarkList = bookmarks + appActions.addBookmarkFolder(makeImmutable(folders)) + appActions.addBookmark(makeImmutable(bookmarks)) }) importer.on('add-favicons', (e, detail) => { @@ -168,17 +166,24 @@ importer.on('add-favicons', (e, detail) => { } } }) - let sites = importedSites - sites = sites.map((site) => { - if ((site.get('favicon') === undefined && site.get('location') !== undefined && - faviconMap[site.get('location')] !== undefined) || - (site.get('favicon') !== undefined && site.get('favicon').includes('made-up-favicon'))) { + let updatedSites = bookmarkList.map((site) => { + if ( + ( + site.get('favicon') === undefined && + site.get('location') !== undefined && + faviconMap[site.get('location')] !== undefined + ) || + ( + site.get('favicon') !== undefined && + site.get('favicon').includes('made-up-favicon')) + ) { return site.set('favicon', faviconMap[site.get('location')]) } else { return site } }) - appActions.addSite(sites) + // TODO can we call addBookmark only once? we need to create a new functions addFavicons + appActions.addBookmark(updatedSites) }) importer.on('add-keywords', (e, templateUrls, uniqueOnHostAndPath) => { @@ -208,7 +213,7 @@ importer.on('add-cookies', (e, cookies) => { }) const getActiveTabId = () => { - return tabState.getActiveTabId(AppStore.getState()) + return tabState.getActiveTabId(appStore.getState()) } const showImportWarning = function () { diff --git a/app/index.js b/app/index.js index 2ee37e47f22..e8fd4dc0196 100644 --- a/app/index.js +++ b/app/index.js @@ -72,7 +72,6 @@ const contentSettings = require('../js/state/contentSettings') const privacy = require('../js/state/privacy') const settings = require('../js/constants/settings') const BookmarksExporter = require('./browser/bookmarksExporter') -const siteUtil = require('../js/state/siteUtil') app.commandLine.appendSwitch('enable-features', 'BlockSmallPluginContent,PreferHtmlOverPlugins') @@ -164,9 +163,8 @@ app.on('ready', () => { // For tests we always want to load default app state const loadedPerWindowState = initialState.perWindowState delete initialState.perWindowState - // Retore map order after load + // Restore map order after load let state = Immutable.fromJS(initialState) - state = state.set('sites', state.get('sites').sort(siteUtil.siteSort)) appActions.setState(state) setImmediate(() => perWindowStateLoaded(loadedPerWindowState)) }) @@ -301,7 +299,7 @@ app.on('ready', () => { }) ipcMain.on(messages.EXPORT_BOOKMARKS, () => { - BookmarksExporter.showDialog(appStore.getState().get('sites')) + BookmarksExporter.showDialog(appStore.getState()) }) // DO NOT TO THIS LIST - see above @@ -330,7 +328,7 @@ app.on('ready', () => { }) process.on(messages.EXPORT_BOOKMARKS, () => { - BookmarksExporter.showDialog(appStore.getState().get('sites')) + BookmarksExporter.showDialog(appStore.getState()) }) ready = true diff --git a/app/renderer/about/bookmarks/bookmarkFolderItem.js b/app/renderer/about/bookmarks/bookmarkFolderItem.js index f2a31c5d9a4..55bf62d4bd5 100644 --- a/app/renderer/about/bookmarks/bookmarkFolderItem.js +++ b/app/renderer/about/bookmarks/bookmarkFolderItem.js @@ -10,9 +10,11 @@ const ImmutableComponent = require('../../components/immutableComponent') // Actions const aboutActions = require('../../../../js/about/aboutActions') +const appActions = require('../../../../js/actions/appActions') // Constants const dragTypes = require('../../../../js/constants/dragTypes') +const siteTags = require('../../../../js/constants/siteTags') // Utils const dndData = require('../../../../js/dndData') @@ -31,7 +33,7 @@ class BookmarkFolderItem extends ImmutableComponent { e.dataTransfer.effectAllowed = 'all' dndData.setupDataTransferURL(e.dataTransfer, this.props.bookmarkFolder.get('location'), - this.props.bookmarkFolder.get('customTitle') || this.props.bookmarkFolder.get('title')) + this.props.bookmarkFolder.get('title')) dndData.setupDataTransferBraveData(e.dataTransfer, dragTypes.BOOKMARK, this.props.bookmarkFolder) } } @@ -54,11 +56,22 @@ class BookmarkFolderItem extends ImmutableComponent { moveBookmark (e, bookmark) { if (siteUtil.isMoveAllowed(this.props.allBookmarkFolders, bookmark, this.props.bookmarkFolder)) { const bookmarkSiteKey = siteUtil.getSiteKey(bookmark) - const bookmarkFolderSiteKey = siteUtil.getSiteKey(this.props.bookmarkFolder) - aboutActions.moveSite(bookmarkSiteKey, - bookmarkFolderSiteKey, - dndData.shouldPrependVerticalItem(e.target, e.clientY), - true) + + if (bookmark.get('type') === siteTags.BOOKMARK_FOLDER) { + appActions.moveBookmarkFolder( + bookmarkSiteKey, + this.props.bookmarkFolder.get('folderId'), + dndData.shouldPrependVerticalItem(e.target, e.clientY), + true + ) + } else { + appActions.moveBookmark( + bookmarkSiteKey, + this.props.bookmarkFolder.get('folderId'), + dndData.shouldPrependVerticalItem(e.target, e.clientY), + true + ) + } } } clearSelection () { @@ -86,17 +99,27 @@ class BookmarkFolderItem extends ImmutableComponent { this.clearSelection() } } + + get childBookmarkFolders () { + const cached = this.props.bookmarkOrder.get(this.props.bookmarkFolder.get('folderId').toString()) + + if (cached == null) { + return Immutable.Map() + } + return cached + .filter(item => item.get('type') === siteTags.BOOKMARK_FOLDER) + .map(folder => this.props.allBookmarkFolders.get(folder.get('key').toString())) + } + render () { const BookmarkFolderList = require('./bookmarkFolderList') - const childBookmarkFolders = this.props.allBookmarkFolders - .filter((bookmarkFolder) => (bookmarkFolder.get('parentFolderId') || 0) === this.props.bookmarkFolder.get('folderId')) return

- {this.props.bookmarkFolder.get('customTitle') || this.props.bookmarkFolder.get('title')} + {this.props.bookmarkFolder.get('title')}
{ - childBookmarkFolders.size > 0 + !this.childBookmarkFolders.isEmpty() ? : null } diff --git a/app/renderer/about/bookmarks/bookmarkFolderList.js b/app/renderer/about/bookmarks/bookmarkFolderList.js index ee918f34731..4f13f12ae06 100644 --- a/app/renderer/about/bookmarks/bookmarkFolderList.js +++ b/app/renderer/about/bookmarks/bookmarkFolderList.js @@ -35,6 +35,7 @@ class BookmarkFolderList extends ImmutableComponent { allBookmarkFolders={this.props.allBookmarkFolders} selectedFolderId={this.props.selectedFolderId} bookmarkFolder={Immutable.fromJS({folderId: 0, tags: [siteTags.BOOKMARK_FOLDER]})} + bookmarkOrder={this.props.bookmarkOrder} /> : null } @@ -50,6 +51,7 @@ class BookmarkFolderList extends ImmutableComponent { selected={!this.props.search && this.props.selectedFolderId === bookmarkFolder.get('folderId')} selectedFolderId={this.props.selectedFolderId} onChangeSelectedFolder={this.props.onChangeSelectedFolder} + bookmarkOrder={this.props.bookmarkOrder} /> ) } @@ -65,6 +67,7 @@ class BookmarkFolderList extends ImmutableComponent { allBookmarkFolders={this.props.allBookmarkFolders} selectedFolderId={this.props.selectedFolderId} bookmarkFolder={Immutable.fromJS({folderId: -1, tags: [siteTags.BOOKMARK_FOLDER]})} + bookmarkOrder={this.props.bookmarkOrder} /> : null } diff --git a/app/renderer/about/bookmarks/bookmarkTitleCell.js b/app/renderer/about/bookmarks/bookmarkTitleCell.js index f2c99d121e6..ef09e57f44c 100644 --- a/app/renderer/about/bookmarks/bookmarkTitleCell.js +++ b/app/renderer/about/bookmarks/bookmarkTitleCell.js @@ -11,14 +11,14 @@ const ImmutableComponent = require('../../components/immutableComponent') const {iconSize} = require('../../../../js/constants/config') // Utils -const siteUtil = require('../../../../js/state/siteUtil') const cx = require('../../../../js/lib/classSet') +const bookmarkFoldersUtil = require('../../../common/lib/bookmarkFoldersUtil') class BookmarkTitleCell extends ImmutableComponent { render () { let iconStyle const icon = this.props.siteDetail.get('favicon') - if (!siteUtil.isFolder(this.props.siteDetail)) { + if (!bookmarkFoldersUtil.isFolder(this.props.siteDetail)) { if (icon) { iconStyle = { minWidth: iconSize, @@ -30,7 +30,7 @@ class BookmarkTitleCell extends ImmutableComponent { } } - const bookmarkTitle = this.props.siteDetail.get('customTitle') || this.props.siteDetail.get('title') + const bookmarkTitle = this.props.siteDetail.get('title') const bookmarkLocation = this.props.siteDetail.get('location') const defaultIcon = 'fa fa-file-o' diff --git a/app/renderer/about/bookmarks/bookmarkTitleHeader.js b/app/renderer/about/bookmarks/bookmarkTitleHeader.js index 953ca91de7f..9cdad907cba 100644 --- a/app/renderer/about/bookmarks/bookmarkTitleHeader.js +++ b/app/renderer/about/bookmarks/bookmarkTitleHeader.js @@ -11,9 +11,6 @@ const ImmutableComponent = require('../../components/immutableComponent') // Actions const windowActions = require('../../../../js/actions/windowActions') -// Constants -const siteTags = require('../../../../js/constants/siteTags') - class BookmarkTitleHeader extends ImmutableComponent { constructor () { super() @@ -21,8 +18,7 @@ class BookmarkTitleHeader extends ImmutableComponent { } addBookmark () { const newBookmark = Immutable.fromJS({ - parentFolderId: this.props.selectedFolderId, - tags: [siteTags.BOOKMARK] + parentFolderId: this.props.selectedFolderId }) windowActions.addBookmark(newBookmark) } diff --git a/app/renderer/about/bookmarks/bookmarks.js b/app/renderer/about/bookmarks/bookmarks.js index 936af4a748a..3dd745d30d3 100644 --- a/app/renderer/about/bookmarks/bookmarks.js +++ b/app/renderer/about/bookmarks/bookmarks.js @@ -38,6 +38,7 @@ class Bookmarks extends React.Component { this.state = { bookmarks: Immutable.Map(), bookmarkFolders: Immutable.Map(), + bookmarkOrder: Immutable.Map(), selectedFolderId: 0, search: '' } @@ -46,7 +47,8 @@ class Bookmarks extends React.Component { const detail = handle.memory() this.setState({ bookmarks: Immutable.fromJS((detail && detail.bookmarks) || {}), - bookmarkFolders: Immutable.fromJS((detail && detail.bookmarkFolders) || {}) + bookmarkFolders: Immutable.fromJS((detail && detail.bookmarkFolders) || {}), + bookmarkOrder: Immutable.fromJS((detail && detail.bookmarkOrder) || {}) }) }) } @@ -86,13 +88,25 @@ class Bookmarks extends React.Component { searchedBookmarks (searchTerm, bookmarks) { return bookmarks.filter((bookmark) => { - const title = bookmark.get('customTitle') + bookmark.get('title') + bookmark.get('location') + const title = bookmark.get('title') + bookmark.get('location') return title.match(new RegExp(searchTerm, 'gi')) }) } get bookmarksInFolder () { - return this.state.bookmarks.filter((bookmark) => (bookmark.get('parentFolderId') || 0) === this.state.selectedFolderId) + const cached = this.state.bookmarkOrder.get(this.state.selectedFolderId.toString()) + + if (cached == null) { + return Immutable.Map() + } + + return cached + .filter(item => item.get('type') === siteTags.BOOKMARK) + .map(bookmark => this.state.bookmarks.get(bookmark.get('key'))) + } + + get bookmarkFolders () { + return this.state.bookmarkFolders.filter((bookmark) => bookmark.get('parentFolderId') === -1) } importBrowserData () { @@ -104,11 +118,9 @@ class Bookmarks extends React.Component { } addBookmarkFolder () { - const newFolder = Immutable.fromJS({ - parentFolderId: this.state.selectedFolderId, - tags: [siteTags.BOOKMARK_FOLDER] - }) - windowActions.addBookmark(newFolder) + windowActions.addBookmarkFolder(Immutable.fromJS({ + parentFolderId: this.state.selectedFolderId + })) } clearSelection () { @@ -146,11 +158,12 @@ class Bookmarks extends React.Component { bookmark.get('parentFolderId') === -1)} + bookmarkFolders={this.bookmarkFolders} allBookmarkFolders={this.state.bookmarkFolders} isRoot selectedFolderId={this.state.selectedFolderId} - search={this.state.search} /> + search={this.state.search} + bookmarkOrder={this.state.bookmarkOrder} />
folder.get('folderId') === siteDetail.get('parentFolderId')) - : Immutable.fromJS({folderId: 0, tags: [siteTags.BOOKMARK_FOLDER]}) - destinationIsParent = true - } - - if (siteUtil.isMoveAllowed(this.props.allBookmarkFolders, bookmark, siteDetail)) { - const bookmarkSiteKey = siteUtil.getSiteKey(bookmark.toJS()) - const siteKey = siteUtil.getSiteKey(siteDetail.toJS()) + const bookmarkSiteKey = siteUtil.getSiteKey(bookmark) + const siteKey = siteUtil.getSiteKey(siteDetail) - aboutActions.moveSite(bookmarkSiteKey, - siteKey, - dndData.shouldPrependVerticalItem(e.target, e.clientY), - destinationIsParent) - } + appActions.moveBookmark( + bookmarkSiteKey, + siteKey, + dndData.shouldPrependVerticalItem(e.target, e.clientY), + destinationIsParent + ) } /** * Bookmark (one or multiple) or BookmarkFolderItem object was dropped @@ -124,7 +117,7 @@ class BookmarksList extends ImmutableComponent { rows={this.props.bookmarks.map((entry) => [ { cell: , - value: entry.get('customTitle') || entry.get('title') || entry.get('location') + value: entry.get('title', entry.get('location')) }, { html: formatUtil.toLocaleString(entry.get('lastAccessedTime'), ''), diff --git a/app/renderer/components/bookmarks/addEditBookmarkFolder.js b/app/renderer/components/bookmarks/addEditBookmarkFolder.js new file mode 100644 index 00000000000..2bed10a30f0 --- /dev/null +++ b/app/renderer/components/bookmarks/addEditBookmarkFolder.js @@ -0,0 +1,103 @@ +/* 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/. */ + +const React = require('react') +const Immutable = require('immutable') +const {StyleSheet, css} = require('aphrodite/no-important') + +// Components +const ReduxComponent = require('../reduxComponent') +const Dialog = require('../common/dialog') +const AddEditBookmarkFolderForm = require('./addEditBookmarkFolderForm') +const {CommonFormBookmarkHanger} = require('../common/commonForm') + +// State +const bookmarkFoldersState = require('../../../common/state/bookmarkFoldersState') +const bookmarksState = require('../../../common/state/bookmarksState') + +// Actions +const windowActions = require('../../../../js/actions/windowActions') + +// Utils +const cx = require('../../../../js/lib/classSet') +const bookmarkFoldersUtil = require('../../../common/lib/bookmarkFoldersUtil') + +// Styles +const globalStyles = require('../styles/global') + +class AddEditBookmarkFolder extends React.Component { + constructor (props) { + super(props) + this.onClose = this.onClose.bind(this) + this.onClick = this.onClick.bind(this) + } + + onClose () { + windowActions.onBookmarkFolderClose() + } + + onClick (e) { + e.stopPropagation() + } + + mergeProps (state, ownProps) { + const currentWindow = state.get('currentWindow') + const bookmarkDetail = currentWindow.get('bookmarkFolderDetail', Immutable.Map()) + const folderDetails = bookmarkDetail.get('folderDetails') || Immutable.Map() + const editMode = bookmarkDetail.has('editKey') + + const props = {} + // used in renderer + props.heading = editMode + ? 'bookmarkFolderEditing' + : 'bookmarkFolderAdding' + props.parentFolderId = folderDetails.get('parentFolderId') + props.partitionNumber = folderDetails.get('partitionNumber') + props.folderName = folderDetails.get('title') + props.isFolderNameValid = bookmarkFoldersUtil.isFolderNameValid(folderDetails.get('title')) + props.folders = bookmarkFoldersState.getFoldersWithoutKey(state, folderDetails.get('folderId')) // TODO (nejc) improve, primitives only + props.editKey = bookmarkDetail.get('editKey', null) + props.closestKey = bookmarkDetail.get('closestKey', null) + props.hasBookmarks = bookmarksState.getBookmarks(state).size > 0 || bookmarkFoldersState.getFolders(state).size > 0 + + return props + } + + render () { + return + +
+ + +
+ } +} + +const styles = StyleSheet.create({ + // Copied from commonForm.js + commonFormSection: { + // PR #7985 + margin: `${globalStyles.spacing.dialogInsideMargin} 30px` + }, + commonFormTitle: { + color: globalStyles.color.braveOrange, + fontSize: '1.2em' + } +}) + +module.exports = ReduxComponent.connect(AddEditBookmarkFolder) diff --git a/app/renderer/components/bookmarks/addEditBookmarkFolderForm.js b/app/renderer/components/bookmarks/addEditBookmarkFolderForm.js new file mode 100644 index 00000000000..614daa610f3 --- /dev/null +++ b/app/renderer/components/bookmarks/addEditBookmarkFolderForm.js @@ -0,0 +1,235 @@ +/* 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/. */ + +const React = require('react') +const Immutable = require('immutable') +const {StyleSheet, css} = require('aphrodite/no-important') + +// Components +const BrowserButton = require('../common/browserButton') +const { + CommonFormSection, + CommonFormDropdown, + CommonFormButtonWrapper, + commonFormStyles +} = require('../common/commonForm') + +// Actions +const appActions = require('../../../../js/actions/appActions') +const windowActions = require('../../../../js/actions/windowActions') + +// Constants +const KeyCodes = require('../../../common/constants/keyCodes') +const settings = require('../../../../js/constants/settings') + +// Utils +const UrlUtil = require('../../../../js/lib/urlutil') +const {getSetting} = require('../../../../js/settings') +const bookmarkFoldersUtil = require('../../../common/lib/bookmarkFoldersUtil') + +// Styles +const globalStyles = require('../styles/global') +const commonStyles = require('../styles/commonStyles') + +class AddEditBookmarkFolderForm extends React.Component { + constructor (props) { + super(props) + this.onNameChange = this.onNameChange.bind(this) + this.onParentFolderChange = this.onParentFolderChange.bind(this) + this.onKeyDown = this.onKeyDown.bind(this) + this.onClose = this.onClose.bind(this) + this.onSave = this.onSave.bind(this) + this.onFolderRemove = this.onFolderRemove.bind(this) + this.state = { + title: props.folderName, + parentFolderId: props.parentFolderId, + isDisabled: props.isDisabled + } + } + + componentDidMount () { + setImmediate(() => { + this.folderName.select() + }) + } + + onKeyDown (e) { + switch (e.keyCode) { + case KeyCodes.ENTER: + this.onSave() + break + case KeyCodes.ESC: + this.onClose() + break + } + } + + onClose () { + windowActions.onBookmarkFolderClose() + } + + updateButtonStatus (newValue) { + if (newValue !== this.state.isDisabled) { + this.setState({ + isDisabled: newValue + }) + } + } + + onNameChange (e) { + let title = e.target.value + + this.setState({ + title: title + }) + + this.updateButtonStatus(!bookmarkFoldersUtil.isFolderNameValid(title)) + } + + onParentFolderChange (e) { + this.setState({ + parentFolderId: ~~e.target.value + }) + } + + onSave () { + // First check if the title of the bookmarkDetail is set + if (this.state.isDisabled) { + return false + } + + // show bookmark if hidden + if (!this.props.hasBookmarks && !getSetting(settings.SHOW_BOOKMARKS_TOOLBAR)) { + appActions.changeSetting(settings.SHOW_BOOKMARKS_TOOLBAR, true) + } + + let data = Immutable.fromJS({ + parentFolderId: this.state.parentFolderId + }) + + if (this.props.editKey != null) { + data = data.set('folderId', this.props.editKey) + } + + // handle title input + let title = this.state.title + if (typeof title === 'string' && UrlUtil.isURL(title)) { + const punycodeUrl = UrlUtil.getPunycodeUrl(title) + if (punycodeUrl.replace(/\/$/, '') !== title) { + title = punycodeUrl + } + } + data = data.set('title', title) + + if (this.props.editKey != null) { + appActions.editBookmarkFolder(data, this.props.editKey) + } else { + appActions.addBookmarkFolder(data, this.props.closestKey) + } + + this.onClose() + } + + onFolderRemove () { + appActions.removeBookmarkFolder(Immutable.fromJS({ + parentFolderId: this.props.parentFolderId, + partitionNumber: this.props.partitionNumber, + folderId: this.props.editKey + })) + this.onClose() + } + + render () { + return
+ +
+
+
+
+
+
+
+ + { + this.props.editKey != null + ? + : + } + + +
+ } +} + +const styles = StyleSheet.create({ + bookmarkHanger__label: { + display: 'block', + marginBottom: `calc(${globalStyles.spacing.dialogInsideMargin} / 3)` + }, + bookmarkHanger__marginRow: { + marginTop: `calc(${globalStyles.spacing.dialogInsideMargin} / 2)` + }, + + bookmark__sectionWrapper: { + display: 'flex', + flexFlow: 'column nowrap' + } +}) + +module.exports = AddEditBookmarkFolderForm diff --git a/app/renderer/components/bookmarks/addEditBookmarkForm.js b/app/renderer/components/bookmarks/addEditBookmarkForm.js index a744a8d50f4..2dd64b19ea4 100644 --- a/app/renderer/components/bookmarks/addEditBookmarkForm.js +++ b/app/renderer/components/bookmarks/addEditBookmarkForm.js @@ -22,7 +22,6 @@ const windowActions = require('../../../../js/actions/windowActions') // Constants const KeyCodes = require('../../../common/constants/keyCodes') -const siteTags = require('../../../../js/constants/siteTags') const settings = require('../../../../js/constants/settings') // Utils @@ -45,7 +44,7 @@ class AddEditBookmarkForm extends React.Component { this.onSave = this.onSave.bind(this) this.onRemoveBookmark = this.onRemoveBookmark.bind(this) this.state = { - title: props.bookmarkName, + title: props.title, location: props.location, parentFolderId: props.parentFolderId, isDisabled: props.isDisabled @@ -87,8 +86,6 @@ class AddEditBookmarkForm extends React.Component { this.setState({ title: title }) - - this.updateButtonStatus(!isBookmarkNameValid(title, this.state.location, this.props.isFolder)) } onLocationChange (e) { @@ -98,7 +95,7 @@ class AddEditBookmarkForm extends React.Component { location: location }) - this.updateButtonStatus(!isBookmarkNameValid(this.state.title, location, this.props.isFolder)) + this.updateButtonStatus(!isBookmarkNameValid(location)) } onParentFolderChange (e) { @@ -118,16 +115,10 @@ class AddEditBookmarkForm extends React.Component { appActions.changeSetting(settings.SHOW_BOOKMARKS_TOOLBAR, true) } - const tag = this.props.isFolder ? siteTags.BOOKMARK_FOLDER : siteTags.BOOKMARK let data = Immutable.fromJS({ - parentFolderId: this.state.parentFolderId, - title: this.props.currentTitle + parentFolderId: this.state.parentFolderId }) - if (this.props.isFolder && this.props.editKey != null) { - data = data.set('folderId', this.props.editKey) - } - // handle title input let title = this.state.title if (typeof title === 'string' && UrlUtil.isURL(title)) { @@ -136,10 +127,7 @@ class AddEditBookmarkForm extends React.Component { title = punycodeUrl } } - - if (this.props.currentTitle !== title || !title) { - data = data.set('customTitle', title || 0) - } + data = data.set('title', title) // handle location input let location = this.state.location @@ -152,23 +140,16 @@ class AddEditBookmarkForm extends React.Component { data = data.set('location', location) if (this.props.editKey != null) { - appActions.editBookmark(data, this.props.editKey, tag) + appActions.editBookmark(data, this.props.editKey) } else { - appActions.addBookmark(data, tag, this.props.closestKey) + appActions.addBookmark(data, this.props.closestKey) } this.onClose() } onRemoveBookmark () { - const tag = this.props.isFolder ? siteTags.BOOKMARK_FOLDER : siteTags.BOOKMARK - // TODO check if you need to add folderId as prop or you can use editKey - appActions.removeSite(Immutable.fromJS({ - parentFolderId: this.props.parentFolderId, - location: this.props.location, - partitionNumber: this.props.partitionNumber, - folderId: this.props.isFolder ? this.props.editKey : null - }), tag) + appActions.removeBookmark(this.props.editKey) this.onClose() } @@ -203,7 +184,7 @@ class AddEditBookmarkForm extends React.Component {
{ - !this.props.isFolder && !this.props.isAdded + !this.props.isAdded ?
siteUtil.isBookmark(site) || siteUtil.isFolder(site) - ) + props.hasBookmarks = bookmarksState.getBookmarks(state).size > 0 || bookmarkFoldersState.getFolders(state).size > 0 return props } @@ -132,14 +123,12 @@ class AddEditBookmarkHanger extends React.Component { [css(styles.commonFormTitle)]: true })} data-l10n-id={this.props.heading} /> 0) { Array.from(e.dataTransfer.items).forEach((item) => { - item.getAsString((name) => appActions.addSite({ location: item.type, title: name }, siteTags.BOOKMARK)) + item.getAsString((name) => appActions.addBookmark(Immutable.fromJS({ + location: item.type, + title: name + }))) }) return } @@ -92,7 +101,7 @@ class BookmarksToolbar extends React.Component { .map((x) => x.trim()) .filter((x) => !x.startsWith('#') && x.length > 0) .forEach((url) => - appActions.addSite({ location: url }, siteTags.BOOKMARK)) + appActions.addBookmark(Immutable.fromJS({ location: url }))) } onDragEnter (e) { diff --git a/app/renderer/components/frame/frame.js b/app/renderer/components/frame/frame.js index 5ad3ef64eff..c25cd0fc730 100644 --- a/app/renderer/components/frame/frame.js +++ b/app/renderer/components/frame/frame.js @@ -30,7 +30,6 @@ const tabMessageBoxState = require('../../../common/state/tabMessageBoxState') // Utils const frameStateUtil = require('../../../../js/state/frameStateUtil') -const siteUtil = require('../../../../js/state/siteUtil') const UrlUtil = require('../../../../js/lib/urlutil') const cx = require('../../../../js/lib/classSet') const urlParse = require('../../../common/urlParse') @@ -49,6 +48,8 @@ const {isFocused} = require('../../currentWindow') const debounce = require('../../../../js/lib/debounce') const locale = require('../../../../js/l10n') const imageUtil = require('../../../../js/lib/imageUtil') +const historyUtil = require('../../../common/lib/historyUtil') +const siteUtil = require('../../../../js/state/siteUtil') // Constants const settings = require('../../../../js/constants/settings') @@ -624,7 +625,7 @@ class Frame extends React.Component { // with setTitle. We either need to delay this call until the title is // or add a way to update it setTimeout(() => { - appActions.addSite(siteUtil.getDetailFromFrame(this.frame)) + appActions.addHistorySite(historyUtil.getDetailFromFrame(this.frame)) }, 250) } @@ -667,8 +668,9 @@ class Frame extends React.Component { errorCode: e.errorCode, url: e.validatedURL }) + const key = siteUtil.getSiteKey(this.frame) appActions.loadURLRequested(this.props.tabId, 'about:error') - appActions.removeSite(siteUtil.getDetailFromFrame(this.frame)) + appActions.removeHistorySite(key) } else if (isAborted(e.errorCode)) { // just stay put } else if (provisionLoadFailure) { diff --git a/app/renderer/components/main/main.js b/app/renderer/components/main/main.js index 5504295141d..c93ce69aed8 100644 --- a/app/renderer/components/main/main.js +++ b/app/renderer/components/main/main.js @@ -33,6 +33,7 @@ const WidevinePanel = require('./widevinePanel') const AutofillAddressPanel = require('../autofill/autofillAddressPanel') const AutofillCreditCardPanel = require('../autofill/autofillCreditCardPanel') const AddEditBookmarkHanger = require('../bookmarks/addEditBookmarkHanger') +const AddEditBookmarkFolder = require('../bookmarks/addEditBookmarkFolder') const LoginRequired = require('./loginRequired') const ReleaseNotes = require('./releaseNotes') const BookmarksToolbar = require('../bookmarks/bookmarksToolbar') @@ -534,21 +535,22 @@ class Main extends React.Component { props.isFullScreen = activeFrame.get('isFullScreen', false) props.isMaximized = isMaximized() || isFullScreen() props.captionButtonsVisible = isWindows - props.showContextMenu = !!currentWindow.get('contextMenuDetail') - props.showPopupWindow = !!currentWindow.get('popupWindowDetail') + props.showContextMenu = currentWindow.has('contextMenuDetail') + props.showPopupWindow = currentWindow.has('popupWindowDetail') props.showSiteInfo = currentWindow.getIn(['ui', 'siteInfo', 'isVisible']) && !isSourceAboutUrl(activeFrame.get('location')) props.showBravery = shieldState.braveShieldsEnabled(activeFrame) && !!currentWindow.get('braveryPanelDetail') - props.showClearData = !!currentWindow.getIn(['ui', 'isClearBrowsingDataPanelVisible']) - props.showImportData = !!currentWindow.get('importBrowserDataDetail') + props.showClearData = currentWindow.hasIn(['ui', 'isClearBrowsingDataPanelVisible']) + props.showImportData = currentWindow.has('importBrowserDataDetail') props.showWidevine = currentWindow.getIn(['widevinePanelDetail', 'shown']) && !isLinux - props.showAutoFillAddress = !!currentWindow.get('autofillAddressDetail') - props.showAutoFillCC = !!currentWindow.get('autofillCreditCardDetail') + props.showAutoFillAddress = currentWindow.has('autofillAddressDetail') + props.showAutoFillCC = currentWindow.has('autofillCreditCardDetail') props.showLogin = !!loginRequiredDetails - props.showBookmarkHanger = currentWindow.get('bookmarkDetail') && + props.showBookmarkHanger = currentWindow.has('bookmarkDetail') && !currentWindow.getIn(['bookmarkDetail', 'isBookmarkHanger']) - props.showNoScript = currentWindow.getIn(['ui', 'noScriptInfo', 'isVisible']) && + props.showBookmarkFolderDialog = currentWindow.has('bookmarkFolderDetail') + props.showNoScript = currentWindow.hasIn(['ui', 'noScriptInfo', 'isVisible']) && siteUtil.getOrigin(activeFrame.get('location')) props.showReleaseNotes = currentWindow.getIn(['ui', 'releaseNotes', 'isVisible']) props.showCheckDefault = isFocused() && defaultBrowserState.shouldDisplayDialog(state) @@ -657,6 +659,11 @@ class Main extends React.Component { ? : null } + { + this.props.showBookmarkFolderDialog + ? + : null + } { this.props.showNoScript ? diff --git a/app/renderer/components/navigation/navigationBar.js b/app/renderer/components/navigation/navigationBar.js index 59cb062677b..3856548f087 100644 --- a/app/renderer/components/navigation/navigationBar.js +++ b/app/renderer/components/navigation/navigationBar.js @@ -30,10 +30,10 @@ const frameStateUtil = require('../../../../js/state/frameStateUtil') // Utils const cx = require('../../../../js/lib/classSet') const {getBaseUrl} = require('../../../../js/lib/appUrlUtil') -const siteUtil = require('../../../../js/state/siteUtil') const eventUtil = require('../../../../js/lib/eventUtil') const {getSetting} = require('../../../../js/settings') const contextMenus = require('../../../../js/contextMenus') +const bookmarkLocationCache = require('../../../common/cache/bookmarkLocationCache') const {StyleSheet, css} = require('aphrodite/no-important') @@ -47,12 +47,10 @@ class NavigationBar extends React.Component { } onToggleBookmark () { - const editing = this.props.isBookmarked - - if (editing) { + if (this.props.isBookmarked) { windowActions.editBookmark(true, this.props.bookmarkKey) } else { - windowActions.onBookmarkAdded(true, this.props.bookmarkKey) + windowActions.onBookmarkAdded(true) } } @@ -95,6 +93,8 @@ class NavigationBar extends React.Component { const locationId = getBaseUrl(location) const publisherId = state.getIn(['locationInfo', locationId, 'publisher']) const navbar = activeFrame.get('navbar', Immutable.Map()) + const locationCache = bookmarkLocationCache.getCacheKey(state, location) + const bookmarkKey = locationCache.get(0, false) const hasTitle = title && location && title !== location.replace(/^https?:\/\//, '') const titleMode = activeTabShowingMessageBox || @@ -113,8 +113,7 @@ class NavigationBar extends React.Component { // used in renderer props.activeFrameKey = activeFrameKey props.titleMode = titleMode - props.isBookmarked = activeFrameKey !== undefined && - activeTab && activeTab.get('bookmarked') + props.isBookmarked = (bookmarkKey && activeTab) ? activeTab.get('bookmarked') : false props.isWideUrlBarEnabled = getSetting(settings.WIDE_URL_BAR) props.showBookmarkHanger = bookmarkDetail.get('isBookmarkHanger', false) props.isLoading = loading @@ -125,7 +124,7 @@ class NavigationBar extends React.Component { props.isFocused = navbar.getIn(['urlbar', 'focused'], false) props.shouldRenderSuggestions = navbar.getIn(['urlbar', 'suggestions', 'shouldRender']) === true props.activeTabId = activeTabId - props.bookmarkKey = siteUtil.getSiteKey(activeFrame) + props.bookmarkKey = bookmarkKey props.showHomeButton = !props.titleMode && getSetting(settings.SHOW_HOME_BUTTON) return props diff --git a/app/renderer/components/navigation/urlBar.js b/app/renderer/components/navigation/urlBar.js index 8c8c2bc5b68..3249d80eb60 100644 --- a/app/renderer/components/navigation/urlBar.js +++ b/app/renderer/components/navigation/urlBar.js @@ -37,6 +37,7 @@ const {getCurrentWindowId} = require('../../currentWindow') const {normalizeLocation, getNormalizedSuggestion} = require('../../../common/lib/suggestion') const isDarwin = require('../../../common/lib/platformUtil').isDarwin() const publisherUtil = require('../../../common/lib/publisherUtil') +const siteUtil = require('../../../../js/state/siteUtil') // Icons const iconNoScript = require('../../../../img/url-bar-no-script.svg') @@ -167,7 +168,8 @@ class UrlBar extends React.Component { if (e.shiftKey) { const selectedIndex = this.props.urlbarLocationSuffix.length > 0 ? 1 : this.props.selectedIndex if (selectedIndex !== undefined) { - appActions.removeSite({ location: this.props.suggestionLocation }) + const key = siteUtil.getSiteKey(Immutable.fromJS({ location: this.props.suggestionLocation })) + appActions.removeHistorySite(key) } } else { this.hideAutoComplete() diff --git a/app/renderer/components/tabs/content/closeTabIcon.js b/app/renderer/components/tabs/content/closeTabIcon.js index 577350c99cc..34bc48fa5bb 100644 --- a/app/renderer/components/tabs/content/closeTabIcon.js +++ b/app/renderer/components/tabs/content/closeTabIcon.js @@ -50,8 +50,9 @@ class CloseTabIcon extends React.Component { mergeProps (state, ownProps) { const currentWindow = state.get('currentWindow') const frameKey = ownProps.frameKey - const isPinnedTab = frameStateUtil.isPinned(currentWindow, frameKey) const frame = frameStateUtil.getFrameByKey(currentWindow, frameKey) || Immutable.Map() + const tabId = frame.get('tabId', tabState.TAB_ID_NONE) + const isPinnedTab = tabState.isTabPinned(state, tabId) const props = {} // used in renderer @@ -64,7 +65,7 @@ class CloseTabIcon extends React.Component { // used in functions props.frameKey = frameKey props.fixTabWidth = ownProps.fixTabWidth - props.tabId = frame.get('tabId', tabState.TAB_ID_NONE) + props.tabId = tabId props.hasFrame = !frame.isEmpty() return props diff --git a/app/renderer/components/tabs/content/favIcon.js b/app/renderer/components/tabs/content/favIcon.js index 354088de059..181300b4fb7 100644 --- a/app/renderer/components/tabs/content/favIcon.js +++ b/app/renderer/components/tabs/content/favIcon.js @@ -12,6 +12,7 @@ const TabIcon = require('./tabIcon') // State const tabContentState = require('../../../../common/state/tabContentState') +const tabState = require('../../../../common/state/tabState') // Utils const frameStateUtil = require('../../../../../js/state/frameStateUtil') @@ -34,12 +35,13 @@ class Favicon extends React.Component { const frameKey = ownProps.frameKey const frame = frameStateUtil.getFrameByKey(currentWindow, frameKey) || Immutable.Map() const isTabLoading = tabContentState.isTabLoading(currentWindow, frameKey) + const tabId = frame.get('tabId', tabState.TAB_ID_NONE) const props = {} // used in renderer props.isTabLoading = isTabLoading props.favicon = !isTabLoading && frame.get('icon') - props.isPinnedTab = frameStateUtil.isPinned(currentWindow, frameKey) + props.isPinnedTab = tabState.isTabPinned(state, tabId) props.tabIconColor = tabContentState.getTabIconColor(currentWindow, frameKey) props.isNarrowestView = tabContentState.isNarrowestView(currentWindow, frameKey) diff --git a/app/renderer/components/tabs/pinnedTabs.js b/app/renderer/components/tabs/pinnedTabs.js index 5615a69b1b5..4a9cc6b81b4 100644 --- a/app/renderer/components/tabs/pinnedTabs.js +++ b/app/renderer/components/tabs/pinnedTabs.js @@ -18,7 +18,6 @@ const windowActions = require('../../../../js/actions/windowActions') const windowStore = require('../../../../js/stores/windowStore') // Constants -const siteTags = require('../../../../js/constants/siteTags') const dragTypes = require('../../../../js/constants/dragTypes') // Utils @@ -26,6 +25,7 @@ const siteUtil = require('../../../../js/state/siteUtil') const dnd = require('../../../../js/dnd') const dndData = require('../../../../js/dndData') const frameStateUtil = require('../../../../js/state/frameStateUtil') +const pinnedSitesUtil = require('../../../common/lib/pinnedSitesUtil') const {isIntermediateAboutPage} = require('../../../../js/lib/appUrlUtil') class PinnedTabs extends React.Component { @@ -58,12 +58,14 @@ class PinnedTabs extends React.Component { if (!sourceDragData.get('pinnedLocation')) { appActions.tabPinned(sourceDragData.get('tabId'), true) } else { - const sourceDetails = siteUtil.getDetailFromFrame(sourceDragData, siteTags.PINNED) + const sourceDetails = pinnedSitesUtil.getDetailFromFrame(sourceDragData) const droppedOnFrame = this.dropFrame(droppedOnTab.props.frameKey) - const destinationDetails = siteUtil.getDetailFromFrame(droppedOnFrame, siteTags.PINNED) - appActions.moveSite(siteUtil.getSiteKey(sourceDetails), + const destinationDetails = pinnedSitesUtil.getDetailFromFrame(droppedOnFrame) + appActions.onPinnedTabReorder( + siteUtil.getSiteKey(sourceDetails), siteUtil.getSiteKey(destinationDetails), - isLeftSide) + isLeftSide + ) } } }, 0) diff --git a/app/renderer/components/tabs/tab.js b/app/renderer/components/tabs/tab.js index ebac3a874b5..d7a85e975c6 100644 --- a/app/renderer/components/tabs/tab.js +++ b/app/renderer/components/tabs/tab.js @@ -225,6 +225,7 @@ class Tab extends React.Component { const partition = typeof frame.get('partitionNumber') === 'string' ? frame.get('partitionNumber').replace(/^partition-/i, '') : frame.get('partitionNumber') + const tabId = frame.get('tabId', tabState.TAB_ID_NONE) const props = {} // used in renderer @@ -234,7 +235,7 @@ class Tab extends React.Component { props.notificationBarActive = notificationBarActive props.isActive = frameStateUtil.isFrameKeyActive(currentWindow, props.frameKey) props.tabWidth = currentWindow.getIn(['ui', 'tabs', 'fixTabWidth']) - props.isPinnedTab = frameStateUtil.isPinned(currentWindow, props.frameKey) + props.isPinnedTab = tabState.isTabPinned(state, tabId) props.canPlayAudio = tabContentState.canPlayAudio(currentWindow, props.frameKey) props.themeColor = tabContentState.getThemeColor(currentWindow, props.frameKey) props.isNarrowView = tabContentState.isNarrowView(currentWindow, props.frameKey) @@ -256,7 +257,7 @@ class Tab extends React.Component { props.totalTabs = state.get('tabs').size props.dragData = state.getIn(['dragData', 'type']) === dragTypes.TAB && state.get('dragData') props.hasTabInFullScreen = tabContentState.hasTabInFullScreen(currentWindow) - props.tabId = frame.get('tabId', tabState.TAB_ID_NONE) + props.tabId = tabId return props } diff --git a/app/renderer/reducers/contextMenuReducer.js b/app/renderer/reducers/contextMenuReducer.js index 36f3795b6a6..c6056e0c4ab 100644 --- a/app/renderer/reducers/contextMenuReducer.js +++ b/app/renderer/reducers/contextMenuReducer.js @@ -20,6 +20,7 @@ const appStoreRenderer = require('../../../js/stores/appStoreRenderer') // State const contextMenuState = require('../../common/state/contextMenuState') +const bookmarksState = require('../../common/state/bookmarksState') // Actions const appActions = require('../../../js/actions/appActions') @@ -37,6 +38,8 @@ const menuUtil = require('../../common/lib/menuUtil') const urlUtil = require('../../../js/lib/urlutil') const frameStateUtil = require('../../../js/state/frameStateUtil') const dndData = require('../../../js/dndData') +const bookmarkFoldersUtil = require('../../common/lib/bookmarkFoldersUtil') +const historyState = require('../../common/state/historyState') const {makeImmutable, isMap} = require('../../common/state/immutableUtil') const {getCurrentWindow} = require('../../renderer/currentWindow') const {getSetting} = require('../../../js/settings') @@ -237,18 +240,18 @@ const addFolderMenuItem = (closestDestinationDetail, isParent) => { return { label: locale.translation('addFolder'), click: () => { - let siteDetail = Immutable.fromJS({ tags: [siteTags.BOOKMARK_FOLDER] }) let closestKey = null + let folderDetails = Immutable.Map() if (closestDestinationDetail) { closestKey = siteUtil.getSiteKey(closestDestinationDetail) if (isParent) { - siteDetail = siteDetail.set('parentFolderId', (closestDestinationDetail.get('folderId') || closestDestinationDetail.get('parentFolderId'))) + folderDetails = folderDetails.set('parentFolderId', (closestDestinationDetail.get('folderId') || closestDestinationDetail.get('parentFolderId'))) } } - windowActions.addBookmark(siteDetail, closestKey) + windowActions.addBookmarkFolder(folderDetails, closestKey) } } } @@ -273,127 +276,104 @@ const openAllInNewTabsMenuItem = (folderDetail) => { } } -const siteDetailTemplateInit = (state, siteKey) => { - let isHistoryEntry = false - let multipleHistoryEntries = false - let multipleBookmarks = false - let isFolder = false - let isSystemFolder = false - let deleteLabel - let deleteTag - let siteDetail = appStoreRenderer.state.getIn(['sites', siteKey], Immutable.Map()) - const activeFrame = frameStateUtil.getActiveFrame(state) || Immutable.Map() +const getLabel = (siteDetail, activeFrame, type) => { + let label = '' - // TODO(bsclifton): pull this out to a method - if (siteUtil.isBookmark(siteDetail) && activeFrame) { - deleteLabel = 'deleteBookmark' - deleteTag = siteTags.BOOKMARK - } else if (siteUtil.isFolder(siteDetail)) { - isFolder = true - isSystemFolder = siteDetail.get('folderId') === 0 || - siteDetail.get('folderId') === -1 - deleteLabel = 'deleteFolder' - deleteTag = siteTags.BOOKMARK_FOLDER - } else if (siteUtil.isHistoryEntry(siteDetail)) { - isHistoryEntry = true - deleteLabel = 'deleteHistoryEntry' - } else if (Immutable.List.isList(siteDetail)) { - // Multiple bookmarks OR history entries selected - multipleHistoryEntries = true - multipleBookmarks = true - siteDetail.forEach((site) => { - if (!siteUtil.isBookmark(site)) multipleBookmarks = false - if (!siteUtil.isHistoryEntry(site)) multipleHistoryEntries = false - }) - if (multipleBookmarks) { - deleteLabel = 'deleteBookmarks' - deleteTag = siteTags.BOOKMARK - } else if (multipleHistoryEntries) { - deleteLabel = 'deleteHistoryEntries' - } - } else { - deleteLabel = '' + if (type === siteTags.BOOKMARK && activeFrame) { + label = 'deleteBookmark' + } else if (type === siteTags.BOOKMARK_FOLDER) { + label = 'deleteFolder' + } else if (type === siteTags.HISTORY) { + label = 'deleteHistoryEntry' } + return label +} + +const siteDetailTemplateInit = (state, siteKey, type) => { const template = [] + let isFolder = type === siteTags.BOOKMARK_FOLDER + let siteDetail = bookmarksState.findBookmark(appStoreRenderer.state, siteKey) + if (siteDetail.isEmpty()) { + siteDetail = historyState.getSite(state, siteKey) + } - if (!isFolder) { - if (!Immutable.List.isList(siteDetail)) { - const location = siteDetail.get('location') + if (siteDetail.isEmpty()) { + return [] + } - template.push( - openInNewTabMenuItem(location, undefined, siteDetail.get('partitionNumber')), - openInNewPrivateTabMenuItem(location), - openInNewWindowMenuItem(location, undefined, siteDetail.get('partitionNumber')), - openInNewSessionTabMenuItem(location), - copyAddressMenuItem('copyLinkAddress', location), - CommonMenu.separatorMenuItem - ) - } else { - let locations = [] - let partitionNumbers = [] - siteDetail.forEach((site) => { - locations.push(site.get('location')) - partitionNumbers.push(site.get('partitionNumber')) - }) + const activeFrame = frameStateUtil.getActiveFrame(state) || Immutable.Map() + const label = getLabel(siteDetail, activeFrame, type) - template.push( - openInNewTabMenuItem(locations, undefined, partitionNumbers), - openInNewPrivateTabMenuItem(locations), - openInNewSessionTabMenuItem(locations), - CommonMenu.separatorMenuItem - ) - } + if (type !== siteTags.BOOKMARK_FOLDER) { + const location = siteDetail.get('location') + + template.push( + openInNewTabMenuItem(location, undefined, siteDetail.get('partitionNumber')), + openInNewPrivateTabMenuItem(location), + openInNewWindowMenuItem(location, undefined, siteDetail.get('partitionNumber')), + openInNewSessionTabMenuItem(location), + copyAddressMenuItem('copyLinkAddress', location), + CommonMenu.separatorMenuItem + ) } else { template.push(openAllInNewTabsMenuItem(siteDetail), CommonMenu.separatorMenuItem) } - if (!isSystemFolder) { + if (siteDetail.get('folderId') !== 0 && siteDetail.get('folderId') !== -1) { // Picking this menu item pops up the AddEditBookmark modal // - History can be deleted but not edited // - Multiple bookmarks cannot be edited at once // - "Bookmarks Toolbar" and "Other Bookmarks" folders cannot be deleted - if (!isHistoryEntry && !multipleHistoryEntries && !multipleBookmarks) { + if (type !== siteTags.HISTORY) { template.push( { label: locale.translation(isFolder ? 'editFolder' : 'editBookmark'), click: () => { const editKey = siteUtil.getSiteKey(siteDetail) - windowActions.editBookmark(false, editKey) + if (isFolder) { + windowActions.editBookmarkFolder(editKey) + } else { + windowActions.editBookmark(false, editKey) + } } }, - CommonMenu.separatorMenuItem) + CommonMenu.separatorMenuItem + ) } template.push( { - label: locale.translation(deleteLabel), + label: locale.translation(label), click: () => { - if (Immutable.List.isList(siteDetail)) { - siteDetail.forEach((site) => appActions.removeSite(site, deleteTag)) - } else { - appActions.removeSite(siteDetail, deleteTag) + if (type === siteTags.HISTORY) { + appActions.removeHistorySite(siteKey) + } else if (type === siteTags.BOOKMARK) { + appActions.removeBookmark(siteKey) + } else if (type === siteTags.BOOKMARK_FOLDER) { + appActions.removeBookmarkFolder(siteKey) } } }) } - if (!isHistoryEntry && !multipleHistoryEntries) { + if (type !== siteTags.HISTORY) { if (template[template.length - 1] !== CommonMenu.separatorMenuItem) { template.push(CommonMenu.separatorMenuItem) } template.push( - addBookmarkMenuItem('addBookmark', siteUtil.getDetailFromFrame(activeFrame, siteTags.BOOKMARK), siteDetail, true), - addFolderMenuItem(siteDetail, true)) + addBookmarkMenuItem('addBookmark', bookmarkUtil.getDetailFromFrame(activeFrame), siteDetail, true), + addFolderMenuItem(siteDetail, true) + ) } return menuUtil.sanitizeTemplateItems(template) } -const onSiteDetailMenu = (state, siteKey) => { +const onSiteDetailMenu = (state, siteKey, type) => { state = validateState(state) - const template = siteDetailTemplateInit(state, siteKey) + const template = siteDetailTemplateInit(state, siteKey, type) const menu = Menu.buildFromTemplate(template) menu.popup(getCurrentWindow()) @@ -402,9 +382,7 @@ const onSiteDetailMenu = (state, siteKey) => { const showBookmarkFolderInit = (state, parentBookmarkFolderKey) => { const appState = appStoreRenderer.state - const allBookmarkItems = siteUtil.getBookmarks(appState.get('sites', Immutable.List())) - const parentBookmarkFolder = appState.getIn(['sites', parentBookmarkFolderKey]) - const items = siteUtil.filterSitesRelativeTo(allBookmarkItems, parentBookmarkFolder) + const items = bookmarksState.getBookmarksWithFolders(appState, parentBookmarkFolderKey) if (items.size === 0) { return [{ l10nLabelId: 'emptyFolderItem', @@ -417,8 +395,11 @@ const showBookmarkFolderInit = (state, parentBookmarkFolderKey) => { e.preventDefault() const bookmark = dnd.prepareBookmarkDataFromCompatible(e.dataTransfer) if (bookmark) { - const bookmarkSiteKey = siteUtil.getSiteKey(bookmark) - appActions.moveSite(bookmarkSiteKey, parentBookmarkFolderKey, false, true) + if (bookmark.get('type') === siteTags.BOOKMARK_FOLDER) { + appActions.moveBookmarkFolder(bookmark.get('key'), parentBookmarkFolderKey, false, true) + } else { + appActions.moveBookmark(bookmark.get('key'), parentBookmarkFolderKey, false, true) + } } } }] @@ -430,10 +411,9 @@ const bookmarkItemsInit = (state, items) => { const activeFrame = frameStateUtil.getActiveFrame(state) || Immutable.Map() const showFavicon = bookmarkUtil.showFavicon() const template = [] - for (let item of items) { - const site = item[1] - const siteKey = item[0] - const isFolder = siteUtil.isFolder(site) + for (let site of items) { + const siteKey = site.get('key') + const isFolder = bookmarkFoldersUtil.isFolder(site) let faIcon if (showFavicon && !site.get('favicon')) { faIcon = isFolder ? 'fa-folder-o' : 'fa-file-o' @@ -441,11 +421,11 @@ const bookmarkItemsInit = (state, items) => { const templateItem = { bookmark: site, draggable: true, - label: site.get('customTitle', site.get('title', site.get('location'))), + label: site.get('title', site.get('location')), icon: showFavicon ? site.get('favicon') : undefined, faIcon, contextMenu: function () { - windowActions.onSiteDetailMenu(siteKey) + windowActions.onSiteDetailMenu(siteKey, isFolder ? siteTags.BOOKMARK_FOLDER : siteTags.BOOKMARK) }, dragEnd: function () { dnd.onDragEnd() @@ -453,8 +433,9 @@ const bookmarkItemsInit = (state, items) => { dragStart: function (e) { dnd.onDragStart(dragTypes.BOOKMARK, Immutable.fromJS({ location: site.get('location'), - title: site.get('customTitle', site.get('title')), - bookmarkKey: siteKey + title: site.get('title'), + key: siteKey, + type: isFolder ? siteTags.BOOKMARK_FOLDER : siteTags.BOOKMARK }), e) }, dragOver: function (e) { @@ -465,11 +446,23 @@ const bookmarkItemsInit = (state, items) => { e.preventDefault() const bookmarkItem = dnd.prepareBookmarkDataFromCompatible(e.dataTransfer) if (bookmarkItem) { - appActions.moveSite(bookmarkItem.get('bookmarkKey'), siteKey, dndData.shouldPrependVerticalItem(e.target, e.clientY)) + if (bookmarkItem.get('type') === siteTags.BOOKMARK_FOLDER) { + appActions.moveBookmarkFolder( + bookmarkItem.get('key'), + siteKey, + dndData.shouldPrependVerticalItem(e.target, e.clientY) + ) + } else { + appActions.moveBookmark( + bookmarkItem.get('key'), + siteKey, + dndData.shouldPrependVerticalItem(e.target, e.clientY) + ) + } } }, click: function (e) { - bookmarkActions.clickBookmarkItem(siteKey, activeFrame.get('tabId'), e) + bookmarkActions.clickBookmarkItem(siteKey, activeFrame.get('tabId'), isFolder, e) } } if (isFolder) { @@ -485,14 +478,14 @@ const onMoreBookmarksMenu = (state, action) => { action = validateAction(action) state = validateState(state) - const sites = appStoreRenderer.state.get('sites') - let newSites = {} + const appState = appStoreRenderer.state + let newSites = Immutable.OrderedMap() - for (let siteKey of action.get('bookmarks')) { - newSites[siteKey] = sites.get(siteKey) + for (let key of action.get('bookmarks')) { + newSites.set(key, bookmarksState.findBookmark(appState, key)) } - const menuTemplate = bookmarkItemsInit(state, Immutable.fromJS(newSites)) + const menuTemplate = bookmarkItemsInit(state, newSites) menuTemplate.push({ l10nLabelId: 'moreBookmarks', @@ -651,7 +644,7 @@ const contextMenuReducer = (windowState, action) => { windowState = onShowBookmarkFolderMenu(windowState, action) break case windowConstants.WINDOW_ON_SITE_DETAIL_MENU: - windowState = onSiteDetailMenu(windowState, action.bookmarkKey) + windowState = onSiteDetailMenu(windowState, action.bookmarkKey, action.type) break } return windowState diff --git a/app/renderer/reducers/frameReducer.js b/app/renderer/reducers/frameReducer.js index cfe38fcc3d6..a10fac339eb 100644 --- a/app/renderer/reducers/frameReducer.js +++ b/app/renderer/reducers/frameReducer.js @@ -10,7 +10,6 @@ const Immutable = require('immutable') const appConstants = require('../../../js/constants/appConstants') const windowConstants = require('../../../js/constants/windowConstants') const config = require('../../../js/constants/config') -const siteTags = require('../../../js/constants/siteTags') // Actions const appActions = require('../../../js/actions/appActions') @@ -20,7 +19,7 @@ const frameStateUtil = require('../../../js/state/frameStateUtil') const {getCurrentWindowId} = require('../currentWindow') const {getSourceAboutUrl, getSourceMagnetUrl} = require('../../../js/lib/appUrlUtil') const {isURL, isPotentialPhishingUrl, getUrlFromInput} = require('../../../js/lib/urlutil') -const siteUtil = require('../../../js/state/siteUtil') +const bookmarkUtil = require('../../common/lib/bookmarkUtil') const setFullScreen = (state, action) => { const index = frameStateUtil.getIndexByTabId(state, action.tabId) @@ -229,8 +228,8 @@ const frameReducer = (state, action, immutableAction) => { // TODO make this an appAction that gets the bookmark data from tabState const frameProps = frameStateUtil.getFrameByTabId(state, action.tabId) if (frameProps) { - const bookmark = siteUtil.getDetailFromFrame(frameProps, siteTags.BOOKMARK) - appActions.addSite(bookmark, siteTags.BOOKMARK) + const bookmark = bookmarkUtil.getDetailFromFrame(frameProps) + appActions.addBookmark(bookmark) } break } diff --git a/app/sessionStore.js b/app/sessionStore.js index 852fbc3cb33..efab9d7c58d 100644 --- a/app/sessionStore.js +++ b/app/sessionStore.js @@ -17,39 +17,32 @@ const path = require('path') const electron = require('electron') const os = require('os') const app = electron.app -const locale = require('./locale') + +// Constants const UpdateStatus = require('../js/constants/updateStatus') const settings = require('../js/constants/settings') +const siteTags = require('../js/constants/siteTags') const downloadStates = require('../js/constants/downloadStates') + +// State +const tabState = require('./common/state/tabState') +const windowState = require('./common/state/windowState') + +// Utils +const locale = require('./locale') const siteUtil = require('../js/state/siteUtil') -const { topSites, pinnedTopSites } = require('../js/data/newTabData') -const { defaultSiteSettingsList } = require('../js/data/siteSettingsList') -const sessionStorageVersion = 1 +const {pinnedTopSites} = require('../js/data/newTabData') +const {defaultSiteSettingsList} = require('../js/data/siteSettingsList') const filtering = require('./filtering') const autofill = require('./autofill') const {navigatableTypes} = require('../js/lib/appUrlUtil') const Channel = require('./channel') -const { makeImmutable } = require('./common/state/immutableUtil') -const tabState = require('./common/state/tabState') -const windowState = require('./common/state/windowState') - +const {makeImmutable} = require('./common/state/immutableUtil') +const {getSetting} = require('../js/settings') const platformUtil = require('./common/lib/platformUtil') -const getSetting = require('../js/settings').getSetting -const sessionStorageName = `session-store-${sessionStorageVersion}` -const getTopSiteMap = () => { - if (Array.isArray(topSites) && topSites.length) { - let siteMap = {} - let order = 0 - topSites.forEach((site) => { - let key = siteUtil.getSiteKey(Immutable.fromJS(site)) - site.order = order++ - siteMap[key] = site - }) - return siteMap - } - return {} -} +const sessionStorageVersion = 1 +const sessionStorageName = `session-store-${sessionStorageVersion}` const getTempStoragePath = (filename) => { const epochTimestamp = (new Date()).getTime().toString() @@ -329,7 +322,7 @@ module.exports.cleanAppData = (data, isShutdown) => { if (data.sites) { const clearHistory = isShutdown && getSetting(settings.SHUTDOWN_CLEAR_HISTORY) === true if (clearHistory) { - data.sites = siteUtil.clearHistory(Immutable.fromJS(data.sites)).toJS() + data.historySites = {} if (data.about) { delete data.about.history delete data.about.newtab @@ -446,6 +439,30 @@ const setVersionInformation = (data) => { return data } +const sortBookmarkOrder = (bookmarkOrder) => { + const newOrder = {} + for (let key of Object.keys(bookmarkOrder)) { + let i = 0 + const order = bookmarkOrder[key].sort((x, y) => { + if (x.order < y.order) { + return -1 + } else if (x.order > y.order) { + return 1 + } else { + return 0 + } + }).map(item => { + item.order = i + i++ + return item + }) + + newOrder[key] = order + } + + return newOrder +} + module.exports.runPreMigrations = (data) => { // autofill data migration if (data.autofill) { @@ -498,6 +515,162 @@ module.exports.runPreMigrations = (data) => { } } + if (data.sites) { + // pinned sites + if (!data.pinnedSites) { + data.pinnedSites = {} + for (let key of Object.keys(data.sites)) { + const site = data.sites[key] + if (site.tags && site.tags.includes('pinned')) { + delete site.tags + data.pinnedSites[key] = site + } + } + } + + // default sites + let newTab = data.about.newtab + + if (newTab) { + const ignoredSites = [] + const pinnedSites = [] + + if (newTab.ignoredTopSites) { + for (let site of newTab.ignoredTopSites) { + if (site) { + ignoredSites.push(`${site.location}|0|0`) + } + } + data.about.newtab.ignoredTopSites = ignoredSites + } + + if (newTab.pinnedTopSites) { + for (let site of newTab.pinnedTopSites) { + if (site) { + site.key = `${site.location}|0|0` + pinnedSites.push(site) + } + } + data.about.newtab.pinnedTopSites = pinnedSites + } + + data.about.newtab.sites = [] + } + + // bookmark order + let bookmarkOrder = {} + + // bookmark folders + if (!data.bookmarkFolders) { + data.bookmarkFolders = {} + + for (let key of Object.keys(data.sites)) { + const oldFolder = data.sites[key] + if (oldFolder.tags && oldFolder.tags.includes(siteTags.BOOKMARK_FOLDER)) { + let folder = {} + key = key.toString() + + if (oldFolder.customTitle) { + folder.title = oldFolder.customTitle + } else { + folder.title = oldFolder.title + } + + if (oldFolder.parentFolderId == null) { + folder.parentFolderId = 0 + } else { + folder.parentFolderId = oldFolder.parentFolderId + } + + folder.folderId = oldFolder.folderId + folder.partitionNumber = oldFolder.partitionNumber + folder.objectId = oldFolder.objectId + folder.type = siteTags.BOOKMARK_FOLDER + folder.key = key + data.bookmarkFolders[key] = folder + + // bookmark order + const id = folder.parentFolderId.toString() + if (!bookmarkOrder[id]) { + bookmarkOrder[id] = [] + } + + bookmarkOrder[id].push({ + key: key, + order: oldFolder.order, + type: siteTags.BOOKMARK_FOLDER + }) + } + } + } + + // bookmarks + if (!data.bookmarks) { + data.bookmarks = {} + + for (let key of Object.keys(data.sites)) { + const oldBookmark = data.sites[key] + if (oldBookmark.tags && oldBookmark.tags.includes(siteTags.BOOKMARK)) { + let bookmark = {} + + if (oldBookmark.customTitle && oldBookmark.customTitle.length > 0) { + bookmark.title = oldBookmark.customTitle + } else { + bookmark.title = oldBookmark.title + } + + if (oldBookmark.parentFolderId == null) { + bookmark.parentFolderId = 0 + } else { + bookmark.parentFolderId = oldBookmark.parentFolderId + } + + bookmark.location = oldBookmark.location + bookmark.partitionNumber = oldBookmark.partitionNumber + bookmark.objectId = oldBookmark.objectId + bookmark.favicon = oldBookmark.favicon + bookmark.themeColor = oldBookmark.themeColor + bookmark.type = siteTags.BOOKMARK + bookmark.key = key + data.bookmarks[key] = bookmark + + // bookmark order + const id = bookmark.parentFolderId.toString() + if (!bookmarkOrder[id]) { + bookmarkOrder[id] = [] + } + + bookmarkOrder[id].push({ + key: key, + order: oldBookmark.order, + type: siteTags.BOOKMARK + }) + } + } + } + + // Add cache to the state + if (!data.cache) { + data.cache = {} + data.cache.bookmarkLocation = data.locationSiteKeysCache + data.cache.bookmarkOrder = sortBookmarkOrder(bookmarkOrder) + } + + // history + if (!data.historySites) { + data.historySites = {} + + for (let key of Object.keys(data.sites)) { + const site = data.sites[key] + if (site.lastAccessedTime || !site.tags || site.tags.length === 0) { + data.historySites[key] = site + } + } + } + + delete data.sites + } + return data } @@ -633,8 +806,14 @@ module.exports.defaultAppState = () => { pendingRecords: {}, lastConfirmedRecordTimestamp: 0 }, - locationSiteKeysCache: undefined, - sites: getTopSiteMap(), + cache: { + bookmarkLocation: undefined, + bookmarkOrder: {} + }, + pinnedSites: {}, + bookmarks: {}, + bookmarkFolders: {}, + historySites: {}, tabs: [], windows: [], extensions: {}, @@ -658,7 +837,7 @@ module.exports.defaultAppState = () => { about: { newtab: { gridLayoutSize: 'small', - sites: topSites, + sites: [], ignoredTopSites: [], pinnedTopSites: pinnedTopSites }, diff --git a/app/sync.js b/app/sync.js index 05122a0fc51..f5ebcc05c78 100644 --- a/app/sync.js +++ b/app/sync.js @@ -19,13 +19,14 @@ const appActions = require('../js/actions/appActions') const syncConstants = require('../js/constants/syncConstants') const appDispatcher = require('../js/dispatcher/appDispatcher') const AppStore = require('../js/stores/appStore') -const siteUtil = require('../js/state/siteUtil') const syncUtil = require('../js/state/syncUtil') const syncPendState = require('./common/state/syncPendState') const getSetting = require('../js/settings').getSetting const settings = require('../js/constants/settings') const extensions = require('./extensions') const {unescapeJSONPointer} = require('./common/lib/jsonUtil') +const bookmarkFoldersState = require('./common/state/bookmarkFoldersState') +const bookmarksState = require('./common/state/bookmarksState') const CATEGORY_MAP = syncUtil.CATEGORY_MAP const CATEGORY_NAMES = Object.keys(categories) @@ -33,7 +34,7 @@ const SYNC_ACTIONS = Object.values(syncConstants) // Fields that should trigger a sync SEND when changed const SYNC_FIELDS = { - sites: ['location', 'customTitle', 'folderId', 'parentFolderId', 'tags'], + sites: ['location', 'title', 'folderId', 'parentFolderId', 'tags'], siteSettings: Object.keys(syncUtil.siteSettingDefaults) } @@ -260,7 +261,8 @@ module.exports.onSyncReady = (isFirstRun, e) => { sendSyncRecords(e.sender, writeActions.CREATE, [deviceRecord]) deviceIdSent = true } - const sites = appState.get('sites') || new Immutable.List() + const bookmarks = bookmarksState.getBookmarks(appState) + const bookmarkFolders = bookmarkFoldersState.getFolders(appState) const seed = appState.get('seed') || new Immutable.List() /** @@ -273,38 +275,34 @@ module.exports.onSyncReady = (isFirstRun, e) => { */ const folderToObjectId = {} const bookmarksToSync = [] - const shouldSyncBookmark = (site) => { - if (!site) { - return false - } - if (siteUtil.isSiteBookmarked(sites, site) !== true && siteUtil.isFolder(site) !== true) { - return false - } + + const shouldSyncBookmark = (bookmark) => { // originalSeed is set on reset to prevent synced bookmarks on a device // from being re-synced. - const originalSeed = site.get('originalSeed') - if (site.get('objectId') && (!originalSeed || seed.equals(originalSeed))) { + const originalSeed = bookmark.get('originalSeed') + if (bookmark.get('objectId') && (!originalSeed || seed.equals(originalSeed))) { return false } - if (folderToObjectId[site.get('folderId')]) { return false } - return syncUtil.isSyncable('bookmark', site) + + return !folderToObjectId[bookmark.get('folderId')] } - const syncBookmark = (site) => { - const siteJS = site.toJS() - const parentFolderId = site.get('parentFolderId') + const syncBookmark = (bookmark) => { + const bookmarkJS = bookmark.toJS() + + const parentFolderId = bookmark.get('parentFolderId') if (typeof parentFolderId === 'number') { if (!folderToObjectId[parentFolderId]) { - const folderResult = siteUtil.getFolder(sites, parentFolderId) + const folderResult = bookmarkFolders.get(parentFolderId) if (folderResult) { syncBookmark(folderResult[1]) } } - siteJS.parentFolderObjectId = folderToObjectId[parentFolderId] + bookmarkJS.parentFolderObjectId = folderToObjectId[parentFolderId] } - const record = syncUtil.createSiteData(siteJS, appState) - const folderId = site.get('folderId') + const record = syncUtil.createSiteData(bookmarkJS, appState) + const folderId = bookmark.get('folderId') if (typeof folderId === 'number') { folderToObjectId[folderId] = record.objectId } @@ -313,12 +311,16 @@ module.exports.onSyncReady = (isFirstRun, e) => { } // Sync bookmarks that have not been synced yet. - sites.forEach((site) => { - if (shouldSyncBookmark(site) !== true) { + bookmarks.forEach((bookmark) => { + if (shouldSyncBookmark(bookmark) !== true) { return } - syncBookmark(site) + + syncBookmark(bookmark) }) + + // TODO add bookamrkFolder sync as well + sendSyncRecords(e.sender, writeActions.CREATE, bookmarksToSync) // Sync site settings that have not been synced yet @@ -490,7 +492,7 @@ module.exports.init = function (appState) { } if (!bookmarksToolbarShown && isFirstRun) { // syncing for the first time - const bookmarks = siteUtil.getBookmarks(AppStore.getState().get('sites')) + const bookmarks = bookmarksState.getBookmarks(AppStore.getState()) if (!bookmarks.size) { for (const record of records) { if (record && record.objectData === 'bookmark') { diff --git a/docs/state.md b/docs/state.md index cf2a9e3bd44..6ae39d90037 100644 --- a/docs/state.md +++ b/docs/state.md @@ -53,6 +53,30 @@ AppStore timestamp: number } }, + bookmarks: { + [bookmarkKey]: { + favicon: string, // URL of the favicon + key: string, // key is duplication of bookmarkKey + location: string, + objectId: Array., + originalSeed: Array., // bookmarks that have been synced before a sync profile reset + parentFolderId: number, + partitionNumber: number, // optionally specifies a specific session + skipSync: boolean, + title: string + } + }, + bookmarkFolders: { + [folderKey]: { + folderId: number, + key: string, // key is duplication of folderKey + objectId: Array., + originalSeed: Array., // only set for bookmarks that have been synced before a sync profile reset + parentFolderId: number, // set for bookmarks and bookmark folders only + skipSync: boolean, // Set for objects FETCHed by sync + title: string + } + }, clearBrowsingDataDefaults: { allSiteCookies: boolean, autocompleteData: boolean, @@ -118,6 +142,18 @@ AppStore flash: { enabled: boolean // enable flash }, + historySites: { + [siteKey]: { + favicon: string, // URL of the favicon + lastAccessedTime: number, // datetime.getTime() + location: string, + objectId: Array., + partitionNumber: number, // optionally specifies a specific session + skipSync: boolean, // Set for objects FETCHed by sync + title: string, + themeColor: string + } + }, menu: { template: object // used on Windows and by our tests: template object with Menubar control }, @@ -156,6 +192,13 @@ AppStore origin: string, // origin of the form username: string }], + pinnedSites: { + [siteKey]: { + location: string, + title: string, + order: number + } + }, settings: { // See defaults in js/constants/appConfig.js 'adblock.customRules': string, // custom rules in ABP filter syntax @@ -230,24 +273,6 @@ AppStore 'tabs.switch-to-new-tabs': boolean, // true if newly opened tabs should be focused immediately 'tabs.tabs-per-page': number // number of tabs per tab page }, - sites: { - [siteKey]: { - creationTime: number, //creation time of bookmark - customTitle: string, // User provided title for bookmark; overrides title - favicon: string, // URL of the favicon - folderId: number, // set for bookmark folders only - lastAccessedTime: number, // datetime.getTime() - location: string, - objectId: Array., - originalSeed: Array., // only set for bookmarks that have been synced before a sync profile reset - parentFolderId: number, // set for bookmarks and bookmark folders only - partitionNumber: number, // optionally specifies a specific session - skipSync: boolean, // Set for objects FETCHed by sync - tags: [string], // empty, 'bookmark', 'bookmark-folder', 'default', or 'reader' - themeColor: string, // CSS compatible color string - title: string - } // folder: folderId; bookmark/history: location + partitionNumber + parentFolderId - }, locationSiteKeyCache: { [location]: Array. // location -> site keys }, diff --git a/js/about/aboutActions.js b/js/about/aboutActions.js index 1992015338f..f231492ae89 100644 --- a/js/about/aboutActions.js +++ b/js/about/aboutActions.js @@ -192,16 +192,6 @@ const aboutActions = { ipc.sendToHost(messages.CONTEXT_MENU_OPENED, nodeProps, contextMenuType) }, - moveSite: function (sourceKey, destinationKey, prepend, destinationIsParent) { - aboutActions.dispatchAction({ - actionType: appConstants.APP_MOVE_SITE, - sourceKey, - destinationKey, - prepend, - destinationIsParent - }) - }, - downloadRevealed: function (downloadId) { aboutActions.dispatchAction({ actionType: appConstants.APP_DOWNLOAD_REVEALED, diff --git a/js/about/newtab.js b/js/about/newtab.js index e9a2b83010b..3985b00724f 100644 --- a/js/about/newtab.js +++ b/js/about/newtab.js @@ -5,26 +5,34 @@ const path = require('path') const React = require('react') const Immutable = require('immutable') -const messages = require('../constants/messages') -const HTML5Backend = require('react-dnd-html5-backend') const {DragDropContext} = require('react-dnd') +const HTML5Backend = require('react-dnd-html5-backend') + +// Components const Stats = require('./newTabComponents/stats') const Clock = require('./newTabComponents/clock') const Block = require('./newTabComponents/block') const SiteRemovalNotification = require('./newTabComponents/siteRemovalNotification') const FooterInfo = require('./newTabComponents/footerInfo') -const aboutActions = require('./aboutActions') -const siteUtil = require('../state/siteUtil') -const urlutils = require('../lib/urlutil') -const siteTags = require('../constants/siteTags') -const config = require('../constants/config') -const backgrounds = require('../data/backgrounds') -const {random} = require('../../app/common/lib/randomUtil') const NewPrivateTab = require('./newprivatetab') + +// Constants +const messages = require('../constants/messages') +const config = require('../constants/config') + +// Actions +const aboutActions = require('./aboutActions') const windowActions = require('../actions/windowActions') +// Data +const backgrounds = require('../data/backgrounds') + +// Utils +const urlutils = require('../lib/urlutil') +const {random} = require('../../app/common/lib/randomUtil') const ipc = window.chrome.ipcRenderer +// Styles require('../../less/about/newtab.less') require('../../node_modules/font-awesome/css/font-awesome.css') @@ -32,15 +40,16 @@ class NewTabPage extends React.Component { constructor (props) { super(props) this.state = { - showSiteRemovalNotification: false, + showNotification: false, imageLoadFailed: false, updatedStamp: undefined, showEmptyPage: true, showImages: false, backgroundImage: undefined } - ipc.on(messages.NEWTAB_DATA_UPDATED, (e, newTabData) => { - const data = Immutable.fromJS(newTabData || {}) + + ipc.on(messages.NEWTAB_DATA_UPDATED, (e, newData) => { + let data = Immutable.fromJS(newData || {}) const updatedStamp = data.getIn(['newTabDetail', 'updatedStamp']) // Only update if the data has changed. @@ -63,83 +72,73 @@ class NewTabPage extends React.Component { }) }) } + get showImages () { return this.state.showImages && !!this.state.backgroundImage } + get randomBackgroundImage () { const image = Object.assign({}, backgrounds[Math.floor(random() * backgrounds.length)]) image.style = {backgroundImage: 'url(' + image.source + ')'} return image } + get fallbackImage () { const image = Object.assign({}, config.newtab.fallbackImage) const pathToImage = path.join(__dirname, '..', '..', image.source) image.style = {backgroundImage: 'url(' + `${pathToImage}` + ')'} return image } + get topSites () { - return this.state.newTabData.getIn(['newTabDetail', 'sites']) || Immutable.List() + return this.state.newTabData.getIn(['newTabDetail', 'sites']) } + get pinnedTopSites () { - return (this.state.newTabData.getIn(['newTabDetail', 'pinnedTopSites']) || Immutable.List()).setSize(18) + return this.state.newTabData.getIn(['newTabDetail', 'pinnedTopSites'], Immutable.List()) } + get ignoredTopSites () { - return this.state.newTabData.getIn(['newTabDetail', 'ignoredTopSites']) || Immutable.List() + return this.state.newTabData.getIn(['newTabDetail', 'ignoredTopSites'], Immutable.List()) } + get gridLayoutSize () { - return this.state.newTabData.getIn(['newTabDetail', 'gridLayoutSize']) || 'small' - } - isPinned (siteProps) { - return this.pinnedTopSites.filter((site) => { - if (!site || !site.get) return false - return site.get('location') === siteProps.get('location') && - site.get('partitionNumber') === siteProps.get('partitionNumber') - }).size > 0 + return this.state.newTabData.getIn(['newTabDetail', 'gridLayoutSize'], 'small') } - isBookmarked (siteProps) { - return siteUtil.isBookmark(siteProps) + + isPinned (siteKey) { + return this.pinnedTopSites.find(site => site.get('key') === siteKey) } + get gridLayout () { const sizeToCount = {large: 18, medium: 12, small: 6} const count = sizeToCount[this.gridLayoutSize] - return this.topSites.take(count) + + let sites = this.pinnedTopSites.take(count) + + if (sites.size < count) { + sites = sites.concat(this.topSites.take(count - sites.size)) + } + + return sites } - showSiteRemovalNotification () { + + showNotification () { this.setState({ - showSiteRemovalNotification: true + showNotification: true }) } + hideSiteRemovalNotification () { this.setState({ - showSiteRemovalNotification: false + showNotification: false }) } - /** - * save number of rows on store. gridsLayout starts with 3 rows (large). - * Rows are reduced at each click and then reset to three again - */ - onChangeGridLayout () { - const gridLayoutSize = this.gridLayoutSize - const changeGridSizeTo = (size) => aboutActions.setNewTabDetail({gridLayoutSize: size}) - - if (gridLayoutSize === 'large') { - changeGridSizeTo('medium') - } else if (gridLayoutSize === 'medium') { - changeGridSizeTo('small') - } else if (gridLayoutSize === 'small') { - changeGridSizeTo('large') - } else { - changeGridSizeTo('large') - } - - return gridLayoutSize - } - - onDraggedSite (currentId, finalId) { + onDraggedSite (siteKey, destinationKey) { let gridSites = this.topSites - const currentPosition = gridSites.filter((site) => site.get('location') === currentId).get(0) - const finalPosition = gridSites.filter((site) => site.get('location') === finalId).get(0) + const currentPosition = gridSites.find(site => site.get('key') === siteKey) + const finalPosition = gridSites.find(site => site.get('key') === destinationKey) const currentPositionIndex = gridSites.indexOf(currentPosition) const finalPositionIndex = gridSites.indexOf(finalPosition) @@ -154,63 +153,57 @@ class NewTabPage extends React.Component { pinnedTopSites = pinnedTopSites.splice(finalPositionIndex, 0, currentPosition) // If site is pinned, update pinnedTopSites list - const newTabState = {} + let newTabState = Immutable.Map() if (this.isPinned(currentPosition)) { - newTabState.pinnedTopSites = pinnedTopSites + newTabState = newTabState.set('pinnedTopSites', pinnedTopSites) } - newTabState.sites = gridSites + newTabState = newTabState.set('sites', gridSites) // Only update if there was an actual change - const stateDiff = Immutable.fromJS(newTabState) const existingState = this.state.newTabData || Immutable.fromJS({}) - const proposedState = existingState.mergeIn(['newTabDetail'], stateDiff) + const proposedState = existingState.mergeIn(['newTabDetail'], newTabState) if (!proposedState.isSubset(existingState)) { - aboutActions.setNewTabDetail(stateDiff) + aboutActions.setNewTabDetail(newTabState) } } - onToggleBookmark (siteProps) { - const siteDetail = siteUtil.getDetailFromFrame(siteProps, siteTags.BOOKMARK) - const editing = this.isBookmarked(siteProps) - const key = siteUtil.getSiteKey(siteDetail) - - if (editing) { - windowActions.editBookmark(false, key) + onToggleBookmark (site) { + if (site.get('bookmarked')) { + windowActions.editBookmark(false, site.get('key')) } else { - windowActions.onBookmarkAdded(false, key) + windowActions.onBookmarkAdded(false, site) } } - onPinnedTopSite (siteProps) { - const currentPosition = this.topSites.filter((site) => siteProps.get('location') === site.get('location')).get(0) - const currentPositionIndex = this.topSites.indexOf(currentPosition) + onPinnedTopSite (siteKey) { + let pinnedTopSites = this.pinnedTopSites + const siteProps = this.topSites.find(site => site.get('key') === siteKey) - // If pinned, leave it null. Otherwise stores site on ignoredTopSites list, retaining the same position - let pinnedTopSites = this.pinnedTopSites.splice(currentPositionIndex, 1, this.isPinned(siteProps) ? null : siteProps) + if (this.isPinned(siteKey)) { + pinnedTopSites = pinnedTopSites.filter(site => site.get('key') !== siteKey) + } else { + pinnedTopSites = pinnedTopSites.push(siteProps) + } - aboutActions.setNewTabDetail({pinnedTopSites: pinnedTopSites}) + aboutActions.setNewTabDetail({pinnedTopSites: pinnedTopSites}, true) } - onIgnoredTopSite (siteProps) { - this.showSiteRemovalNotification() + onIgnoredTopSite (siteKey) { + this.showNotification(siteKey) // If a pinnedTopSite is ignored, remove it from the pinned list as well const newTabState = {} - if (this.isPinned(siteProps)) { - const gridSites = this.topSites - const currentPosition = gridSites.filter((site) => siteProps.get('location') === site.get('location')).get(0) - const currentPositionIndex = gridSites.indexOf(currentPosition) - const pinnedTopSites = this.pinnedTopSites.splice(currentPositionIndex, 1, null) - newTabState.pinnedTopSites = pinnedTopSites + if (this.isPinned(siteKey)) { + newTabState.pinnedTopSites = this.pinnedTopSites.filter(site => site.get('key') !== siteKey) } - newTabState.ignoredTopSites = this.ignoredTopSites.push(siteProps) + newTabState.ignoredTopSites = this.ignoredTopSites.push(siteKey) aboutActions.setNewTabDetail(newTabState, true) } onUndoIgnoredTopSite () { // Remove last List's entry - const ignoredTopSites = this.ignoredTopSites.splice(-1, 1) + const ignoredTopSites = this.ignoredTopSites.pop() aboutActions.setNewTabDetail({ignoredTopSites: ignoredTopSites}, true) this.hideSiteRemovalNotification() } @@ -236,6 +229,12 @@ class NewTabPage extends React.Component { }) } + getLetterFromUrl (url) { + const hostname = urlutils.getHostname(url.get('location'), true) + const name = url.get('title') || hostname || '?' + return name.charAt(0).toUpperCase() + } + render () { // don't render if user prefers an empty page if (this.state.showEmptyPage && !this.props.isIncognito) { @@ -250,11 +249,6 @@ class NewTabPage extends React.Component { if (!this.state.newTabData) { return null } - const getLetterFromUrl = (url) => { - const hostname = urlutils.getHostname(url.get('location'), true) - const name = url.get('title') || hostname || '?' - return name.charAt(0).toUpperCase() - } const gridLayout = this.gridLayout const backgroundProps = {} let gradientClassName = 'gradient' @@ -278,24 +272,24 @@ class NewTabPage extends React.Component {
{ - this.state.showSiteRemovalNotification + this.state.showNotification ? } records - */ - applySiteRecords: function (records) { - dispatch({ - actionType: appConstants.APP_APPLY_SITE_RECORDS, - records - }) - }, - /** * Dispatch to populate the sync object id -> appState key path mapping cache */ @@ -1472,24 +1433,76 @@ const appActions = { }) }, - addBookmark: function (siteDetail, tag, closestKey) { + addBookmark: function (siteDetail, closestKey) { dispatch({ actionType: appConstants.APP_ADD_BOOKMARK, siteDetail, - tag, closestKey }) }, - editBookmark: function (siteDetail, editKey, tag) { + editBookmark: function (siteDetail, editKey) { dispatch({ actionType: appConstants.APP_EDIT_BOOKMARK, siteDetail, - tag, editKey }) }, + moveBookmark: function (bookmarkKey, destinationKey, append, moveIntoParent) { + dispatch({ + actionType: appConstants.APP_MOVE_BOOKMARK, + bookmarkKey, + destinationKey, + append, + moveIntoParent + }) + }, + + /** + * Removes bookmark + * @param bookmarkKey {string|Immutable.List} - Bookmark key that we want to remove. This could also be list of keys + */ + removeBookmark: function (bookmarkKey) { + dispatch({ + actionType: appConstants.APP_REMOVE_BOOKMARK, + bookmarkKey + }) + }, + + addBookmarkFolder: function (folderDetails, closestKey) { + dispatch({ + actionType: appConstants.APP_ADD_BOOKMARK_FOLDER, + folderDetails, + closestKey + }) + }, + + editBookmarkFolder: function (folderDetails, editKey) { + dispatch({ + actionType: appConstants.APP_EDIT_BOOKMARK_FOLDER, + folderDetails, + editKey + }) + }, + + moveBookmarkFolder: function (folderKey, destinationKey, append, moveIntoParent) { + dispatch({ + actionType: appConstants.APP_MOVE_BOOKMARK_FOLDER, + folderKey, + destinationKey, + append, + moveIntoParent + }) + }, + + removeBookmarkFolder: function (folderKey) { + dispatch({ + actionType: appConstants.APP_REMOVE_BOOKMARK_FOLDER, + folderKey + }) + }, + noReportStateModeClicked: function (windowId) { dispatch({ actionType: appConstants.APP_DEBUG_NO_REPORT_STATE_MODE_CLICKED, @@ -1529,6 +1542,15 @@ const appActions = { name, version }) + }, + + onPinnedTabReorder: function (siteKey, destinationKey, prepend) { + dispatch({ + actionType: appConstants.APP_ON_PINNED_TAB_REORDER, + siteKey, + destinationKey, + prepend + }) } } diff --git a/js/actions/bookmarkActions.js b/js/actions/bookmarkActions.js index 51152a2f38b..583fb158180 100644 --- a/js/actions/bookmarkActions.js +++ b/js/actions/bookmarkActions.js @@ -4,33 +4,31 @@ 'use strict' -const Immutable = require('immutable') -const siteUtil = require('../state/siteUtil') const appActions = require('./appActions') const windowActions = require('./windowActions') const eventUtil = require('../lib/eventUtil') const appStoreRenderer = require('../stores/appStoreRenderer') +const bookmarksUtil = require('../../app/common/lib/bookmarkUtil') +const bookmarksState = require('../../app/common/state/bookmarksState') +const bookmarkFoldersSate = require('../../app/common/state/bookmarkFoldersState') const {SWITCH_TO_NEW_TABS} = require('../constants/settings') -const getSetting = require('../settings').getSetting +const {getSetting} = require('../settings') const bookmarkActions = { openBookmarksInFolder: function (folderDetail) { - const allBookmarkItems = siteUtil.getBookmarks(appStoreRenderer.state.get('sites')) - // We have a middle clicked folder - const bookmarks = allBookmarkItems - .filter((bookmark) => (bookmark.get('parentFolderId') || 0) === (folderDetail.get('folderId') || 0) && siteUtil.isBookmark(bookmark)) + const bookmarks = bookmarksUtil.getBookmarksByParentId(appStoreRenderer.state, folderDetail.get('folderId')) // Only load the first 25 tabs as loaded bookmarks .forEach((bookmark, i) => { if (i <= 25) { appActions.createTabRequested( - Object.assign(siteUtil.toCreateProperties(bookmark), { + Object.assign(bookmarksUtil.toCreateProperties(bookmark), { active: false }), getSetting(SWITCH_TO_NEW_TABS)) } else { appActions.createTabRequested({ - location: bookmark.get('location'), + url: bookmark.get('location'), partitionNumber: bookmark.get('partitionNumber'), discarded: true }, false) @@ -42,26 +40,30 @@ const bookmarkActions = { * Performs an action based on the passed in event to the bookmark item * @return true if an action was performed */ - clickBookmarkItem: function (bookmarkKey, tabId, e) { - const bookmarkItem = appStoreRenderer.state.getIn(['sites', bookmarkKey], Immutable.Map()) - const isFolder = siteUtil.isFolder(bookmarkItem) + clickBookmarkItem: function (key, tabId, isFolder, e) { + const isSecondary = eventUtil.isForSecondaryAction(e) + if (!isFolder) { - if (eventUtil.isForSecondaryAction(e)) { + const bookmarkItem = bookmarksState.getBookmark(appStoreRenderer.state, key) + if (isSecondary) { appActions.createTabRequested({ url: bookmarkItem.get('location'), - partitionNumber: (bookmarkItem && bookmarkItem.get && bookmarkItem.get('partitionNumber')) || undefined, + partitionNumber: bookmarkItem.get('partitionNumber') || undefined, active: !!e.shiftKey || getSetting(SWITCH_TO_NEW_TABS) }) } else { appActions.loadURLRequested(tabId, bookmarkItem.get('location')) } + windowActions.setContextMenuDetail() return true - } else if (eventUtil.isForSecondaryAction(e)) { - this.openBookmarksInFolder(bookmarkItem) + } else if (isSecondary) { + const folderItem = bookmarkFoldersSate.getFolder(appStoreRenderer.state, key) + this.openBookmarksInFolder(folderItem) windowActions.setContextMenuDetail() return true } + return false } } diff --git a/js/actions/windowActions.js b/js/actions/windowActions.js index 9f8c5f73149..0fc58cd6f0b 100644 --- a/js/actions/windowActions.js +++ b/js/actions/windowActions.js @@ -512,11 +512,16 @@ const windowActions = { }) }, - onBookmarkAdded: function (isHanger, editKey, siteDetail) { + /** + * Used for adding bookmark site directly and then allowing to + * edit it right afterwords + * @param isHanger + * @param bookmarkDetail - bookmark data, if empty active frame will be used + */ + onBookmarkAdded: function (isHanger, bookmarkDetail) { dispatch({ actionType: windowConstants.WINDOW_ON_BOOKMARK_ADDED, - siteDetail, - editKey, + bookmarkDetail, isHanger }) }, @@ -530,6 +535,38 @@ const windowActions = { }) }, + /** + * Used for displaying bookmark folder dialog + * when adding bookmark site or folder + */ + addBookmarkFolder: function (folderDetails, closestKey) { + dispatch({ + actionType: windowConstants.WINDOW_ON_ADD_BOOKMARK_FOLDER, + folderDetails, + closestKey + }) + }, + + /** + * Used for displaying bookmark folder dialog + * when editing bookmark site or folder + */ + editBookmarkFolder: function (editKey) { + dispatch({ + actionType: windowConstants.WINDOW_ON_EDIT_BOOKMARK_FOLDER, + editKey + }) + }, + + /** + * Used for closing a bookmark dialog + */ + onBookmarkFolderClose: function () { + dispatch({ + actionType: windowConstants.WINDOW_ON_BOOKMARK_FOLDER_CLOSE + }) + }, + /** * Dispatches a message to set context menu detail. * If set, also indicates that the context menu is shown. @@ -1202,10 +1239,11 @@ const windowActions = { }) }, - onSiteDetailMenu: function (bookmarkKey) { + onSiteDetailMenu: function (bookmarkKey, type) { dispatch({ actionType: windowConstants.WINDOW_ON_SITE_DETAIL_MENU, - bookmarkKey + bookmarkKey, + type }) } } diff --git a/js/constants/appConstants.js b/js/constants/appConstants.js index f8e12793899..f920af8d85d 100644 --- a/js/constants/appConstants.js +++ b/js/constants/appConstants.js @@ -13,11 +13,9 @@ const appConstants = { APP_WINDOW_UPDATED: _, APP_TAB_CLOSED: _, APP_TAB_UPDATED: _, - APP_ADD_SITE: _, + APP_ADD_HISTORY_SITE: _, APP_SET_STATE: _, - APP_REMOVE_SITE: _, - APP_MOVE_SITE: _, - APP_APPLY_SITE_RECORDS: _, + APP_REMOVE_HISTORY_SITE: _, APP_MERGE_DOWNLOAD_DETAIL: _, // Sets an individual download detail APP_CLEAR_COMPLETED_DOWNLOADS: _, // Removes all completed downloads APP_DEFAULT_WINDOW_PARAMS_CHANGED: _, @@ -144,7 +142,14 @@ const appConstants = { APP_SPELLING_SUGGESTED: _, APP_LEARN_SPELLING: _, APP_FORGET_LEARNED_SPELLING: _, - APP_SET_VERSION_INFO: _ + APP_SET_VERSION_INFO: _, + APP_ON_PINNED_TAB_REORDER: _, + APP_ADD_BOOKMARK_FOLDER: _, + APP_EDIT_BOOKMARK_FOLDER: _, + APP_REMOVE_BOOKMARK_FOLDER: _, + APP_REMOVE_BOOKMARK: _, + APP_MOVE_BOOKMARK_FOLDER: _, + APP_MOVE_BOOKMARK: _ } module.exports = mapValuesByKeys(appConstants) diff --git a/js/constants/siteTags.js b/js/constants/siteTags.js index 4a86ba2fa59..e4b1338131d 100644 --- a/js/constants/siteTags.js +++ b/js/constants/siteTags.js @@ -6,11 +6,9 @@ const mapValuesByKeys = require('../lib/functional').mapValuesByKeys const _ = null const siteTags = { - DEFAULT: _, BOOKMARK: _, BOOKMARK_FOLDER: _, - PINNED: _, - READING_LIST: _ + HISTORY: _ } module.exports = mapValuesByKeys(siteTags) diff --git a/js/constants/windowConstants.js b/js/constants/windowConstants.js index 002f927062f..7186fbf9f51 100644 --- a/js/constants/windowConstants.js +++ b/js/constants/windowConstants.js @@ -111,7 +111,10 @@ const windowConstants = { WINDOW_ON_ADD_BOOKMARK: _, WINDOW_ON_EDIT_BOOKMARK: _, WINDOW_ON_BOOKMARK_CLOSE: _, - WINDOW_ON_BOOKMARK_ADDED: _ + WINDOW_ON_BOOKMARK_ADDED: _, + WINDOW_ON_ADD_BOOKMARK_FOLDER: _, + WINDOW_ON_EDIT_BOOKMARK_FOLDER: _, + WINDOW_ON_BOOKMARK_FOLDER_CLOSE: _ } module.exports = mapValuesByKeys(windowConstants) diff --git a/js/contextMenus.js b/js/contextMenus.js index 7095bd4a798..fda15c0077d 100644 --- a/js/contextMenus.js +++ b/js/contextMenus.js @@ -26,17 +26,21 @@ const locale = require('../js/l10n') const {getSetting} = require('./settings') const settings = require('./constants/settings') const textUtils = require('./lib/text') -const {getPartitionFromNumber, getActiveFrame} = require('./state/frameStateUtil') const {isIntermediateAboutPage, isUrl, aboutUrls} = require('./lib/appUrlUtil') const {getBase64FromImageUrl} = require('./lib/imageUtil') const urlParse = require('../app/common/urlParse') const {getCurrentWindow} = require('../app/renderer/currentWindow') const extensionState = require('../app/common/state/extensionState') const extensionActions = require('../app/common/actions/extensionActions') +const bookmarkUtil = require('../app/common/lib/bookmarkUtil') +const bookmarksState = require('../app/common/state/bookmarksState') +const historyState = require('../app/common/state/historyState') +const frameStateUtil = require('./state/frameStateUtil') +const platformUtil = require('../app/common/lib/platformUtil') const {makeImmutable} = require('../app/common/state/immutableUtil') -const isDarwin = process.platform === 'darwin' -const isLinux = process.platform === 'linux' +const isDarwin = platformUtil.isDarwin() +const isLinux = platformUtil.isLinux() /** * Obtains an add bookmark menu item @@ -70,18 +74,18 @@ const addFolderMenuItem = (closestDestinationDetail, isParent) => { return { label: locale.translation('addFolder'), click: () => { - let siteDetail = Immutable.fromJS({ tags: [siteTags.BOOKMARK_FOLDER] }) let closestKey = null + let folderDetails = Immutable.Map() if (closestDestinationDetail) { closestKey = siteUtil.getSiteKey(closestDestinationDetail) if (isParent) { - siteDetail = siteDetail.set('parentFolderId', (closestDestinationDetail.get('folderId') || closestDestinationDetail.get('parentFolderId'))) + folderDetails = folderDetails.set('parentFolderId', (closestDestinationDetail.get('folderId') || closestDestinationDetail.get('parentFolderId'))) } } - windowActions.addBookmark(siteDetail, closestKey) + windowActions.addBookmarkFolder(folderDetails, closestKey) } } } @@ -133,8 +137,7 @@ function tabsToolbarTemplateInit (bookmarkTitle, bookmarkLink, closestDestinatio template.push(addBookmarkMenuItem('addBookmark', { title: bookmarkTitle, - location: bookmarkLink, - tags: [siteTags.BOOKMARK] + location: bookmarkLink }, closestDestinationDetail, isParent)) template.push(addFolderMenuItem(closestDestinationDetail, isParent)) @@ -225,113 +228,158 @@ function downloadsToolbarTemplateInit (downloadId, downloadItem) { return menuUtil.sanitizeTemplateItems(template) } -function siteDetailTemplateInit (siteDetail, activeFrame) { - let isHistoryEntry = false - let multipleHistoryEntries = false - let multipleBookmarks = false - let isFolder = false - let isSystemFolder = false - let deleteLabel - let deleteTag - - // TODO(bsclifton): pull this out to a method - if (siteUtil.isBookmark(siteDetail) && activeFrame) { - deleteLabel = 'deleteBookmark' - deleteTag = siteTags.BOOKMARK - } else if (siteUtil.isFolder(siteDetail)) { - isFolder = true - isSystemFolder = siteDetail.get('folderId') === 0 || - siteDetail.get('folderId') === -1 - deleteLabel = 'deleteFolder' - deleteTag = siteTags.BOOKMARK_FOLDER - } else if (siteUtil.isHistoryEntry(siteDetail)) { - isHistoryEntry = true - deleteLabel = 'deleteHistoryEntry' - } else if (Immutable.List.isList(siteDetail)) { - // Multiple bookmarks OR history entries selected - multipleHistoryEntries = true - multipleBookmarks = true - siteDetail.forEach((site) => { - if (!siteUtil.isBookmark(site)) multipleBookmarks = false - if (!siteUtil.isHistoryEntry(site)) multipleHistoryEntries = false - }) - if (multipleBookmarks) { - deleteLabel = 'deleteBookmarks' - deleteTag = siteTags.BOOKMARK - } else if (multipleHistoryEntries) { - deleteLabel = 'deleteHistoryEntries' +const getLabel = (siteDetail, type, activeFrame) => { + let label = '' + + if (Immutable.List.isList(siteDetail)) { + if (type === siteTags.BOOKMARK) { + label = 'deleteBookmarks' + } else if (type === siteTags.HISTORY) { + label = 'deleteHistoryEntries' } - } else { - deleteLabel = '' + } else if (type === siteTags.BOOKMARK && activeFrame) { + label = 'deleteBookmark' + } else if (type === siteTags.BOOKMARK_FOLDER) { + label = 'deleteFolder' + } else if (type === siteTags.HISTORY) { + label = 'deleteHistoryEntry' } + return label +} + +const siteMultipleDetailTemplate = (data, type, activeFrame) => { const template = [] + const label = getLabel(data, type) + + let locations = [] + let partitionNumbers = [] + let keys = [] + data.forEach((site) => { + locations.push(site.get('location')) + partitionNumbers.push(site.get('partitionNumber')) + keys.push(siteUtil.getSiteKey(site)) + }) - if (!isFolder) { - if (!Immutable.List.isList(siteDetail)) { - const location = siteDetail.get('location') + template.push( + openInNewTabMenuItem(locations, undefined, partitionNumbers), + openInNewPrivateTabMenuItem(locations), + openInNewSessionTabMenuItem(locations), + CommonMenu.separatorMenuItem + ) - template.push(openInNewTabMenuItem(location, undefined, siteDetail.get('partitionNumber')), - openInNewPrivateTabMenuItem(location), - openInNewWindowMenuItem(location, undefined, siteDetail.get('partitionNumber')), - openInNewSessionTabMenuItem(location), - copyAddressMenuItem('copyLinkAddress', location), - CommonMenu.separatorMenuItem) - } else { - let locations = [] - let partitionNumbers = [] - siteDetail.forEach((site) => { - locations.push(site.get('location')) - partitionNumbers.push(site.get('partitionNumber')) - }) + template.push({ + label: locale.translation(label), + click: () => { + if (type === siteTags.BOOKMARK) { + appActions.removeBookmark(keys) + } else if (type === siteTags.HISTORY) { + appActions.removeHistorySite(keys) + } + } + }) - template.push(openInNewTabMenuItem(locations, undefined, partitionNumbers), - openInNewPrivateTabMenuItem(locations), - openInNewSessionTabMenuItem(locations), - CommonMenu.separatorMenuItem) + if (type !== siteTags.HISTORY) { + if (template[template.length - 1] !== CommonMenu.separatorMenuItem) { + template.push(CommonMenu.separatorMenuItem) } + + template.push( + addBookmarkMenuItem('addBookmark', bookmarkUtil.getDetailFromFrame(activeFrame), null, true), + addFolderMenuItem(null, true) + ) + } + + return template +} + +const siteSingleDetailTemplate = (siteKey, type, activeFrame) => { + const template = [] + const state = appStoreRenderer.state + let isFolder = type === siteTags.BOOKMARK_FOLDER + let siteDetail + + if (type === siteTags.HISTORY) { + siteDetail = historyState.getSite(state, siteKey) } else { - template.push(openAllInNewTabsMenuItem(appStoreRenderer.state.get('sites'), siteDetail), - CommonMenu.separatorMenuItem) + siteDetail = bookmarksState.findBookmark(state, siteKey) } - if (!isSystemFolder) { + const label = getLabel(siteDetail, type, activeFrame) + + if (type !== siteTags.BOOKMARK_FOLDER) { + const location = siteDetail.get('location') + + template.push( + openInNewTabMenuItem(location, undefined, siteDetail.get('partitionNumber')), + openInNewPrivateTabMenuItem(location), + openInNewWindowMenuItem(location, undefined, siteDetail.get('partitionNumber')), + openInNewSessionTabMenuItem(location), + copyAddressMenuItem('copyLinkAddress', location), + CommonMenu.separatorMenuItem + ) + } else { + template.push(openAllInNewTabsMenuItem(siteDetail), CommonMenu.separatorMenuItem) + } + + if (siteDetail.get('folderId') !== 0 && siteDetail.get('folderId') !== -1) { // Picking this menu item pops up the AddEditBookmark modal // - History can be deleted but not edited // - Multiple bookmarks cannot be edited at once // - "Bookmarks Toolbar" and "Other Bookmarks" folders cannot be deleted - if (!isHistoryEntry && !multipleHistoryEntries && !multipleBookmarks) { + if (type !== siteTags.HISTORY) { template.push( { label: locale.translation(isFolder ? 'editFolder' : 'editBookmark'), click: () => { const editKey = siteUtil.getSiteKey(siteDetail) - windowActions.editBookmark(false, editKey) + if (isFolder) { + windowActions.editBookmarkFolder(editKey) + } else { + windowActions.editBookmark(false, editKey) + } } }, - CommonMenu.separatorMenuItem) + CommonMenu.separatorMenuItem + ) } - template.push( - { - label: locale.translation(deleteLabel), - click: () => { - if (Immutable.List.isList(siteDetail)) { - siteDetail.forEach((site) => appActions.removeSite(site, deleteTag)) - } else { - appActions.removeSite(siteDetail, deleteTag) - } + template.push({ + label: locale.translation(label), + click: () => { + if (type === siteTags.HISTORY) { + appActions.removeHistorySite(siteKey) + } else if (type === siteTags.BOOKMARK) { + appActions.removeBookmark(siteKey) + } else if (type === siteTags.BOOKMARK_FOLDER) { + appActions.removeBookmarkFolder(siteKey) } - }) + } + }) } - if (!isHistoryEntry && !multipleHistoryEntries) { + if (type !== siteTags.HISTORY) { if (template[template.length - 1] !== CommonMenu.separatorMenuItem) { template.push(CommonMenu.separatorMenuItem) } + template.push( - addBookmarkMenuItem('addBookmark', siteUtil.getDetailFromFrame(activeFrame, siteTags.BOOKMARK), siteDetail, true), - addFolderMenuItem(siteDetail, true)) + addBookmarkMenuItem('addBookmark', bookmarkUtil.getDetailFromFrame(activeFrame), siteDetail, true), + addFolderMenuItem(siteDetail, true) + ) + } + + return template +} + +const siteDetailTemplateInit = (data, type, activeFrame) => { + let multiple = Immutable.List.isList(data) + let template + + if (multiple) { + template = siteMultipleDetailTemplate(data, type, activeFrame) + } else { + template = siteSingleDetailTemplate(data, type, activeFrame) } return menuUtil.sanitizeTemplateItems(template) @@ -724,11 +772,11 @@ const openInNewTabMenuItem = (url, isPrivate, partitionNumber, openerTabId) => { } } -const openAllInNewTabsMenuItem = (allSites, folderDetail) => { +const openAllInNewTabsMenuItem = (folderDetail) => { return { label: locale.translation('openAllInTabs'), click: () => { - bookmarkActions.openBookmarksInFolder(allSites, folderDetail) + bookmarkActions.openBookmarksInFolder(folderDetail) } } } @@ -920,7 +968,7 @@ function mainTemplateInit (nodeProps, frame, tab) { appActions.createTabRequested({ url: nodeProps.srcURL, openerTabId: frame.get('tabId'), - partition: getPartitionFromNumber(frame.get('partitionNumber'), isPrivate), + partition: frameStateUtil.getPartitionFromNumber(frame.get('partitionNumber'), isPrivate), active: active }) } @@ -1009,8 +1057,7 @@ function mainTemplateInit (nodeProps, frame, tab) { ) if (isLink) { template.push(addBookmarkMenuItem('bookmarkLink', { - location: nodeProps.linkURL, - tags: [siteTags.BOOKMARK] + location: nodeProps.linkURL }, false)) } } @@ -1024,8 +1071,7 @@ function mainTemplateInit (nodeProps, frame, tab) { if (!isImage) { if (isLink) { template.push(addBookmarkMenuItem('bookmarkLink', { - location: nodeProps.linkURL, - tags: [siteTags.BOOKMARK] + location: nodeProps.linkURL }, false)) } else { template.push( @@ -1055,7 +1101,7 @@ function mainTemplateInit (nodeProps, frame, tab) { } }, CommonMenu.separatorMenuItem, - addBookmarkMenuItem('bookmarkPage', siteUtil.getDetailFromFrame(frame, siteTags.BOOKMARK), false)) + addBookmarkMenuItem('bookmarkPage', bookmarkUtil.getDetailFromFrame(frame), false)) if (!isAboutPage) { template.push({ @@ -1204,8 +1250,7 @@ function mainTemplateInit (nodeProps, frame, tab) { template.push( CommonMenu.separatorMenuItem, addBookmarkMenuItem('addBookmark', { - location: nodeProps.linkURL, - tags: [siteTags.BOOKMARK] + location: nodeProps.linkURL }), addFolderMenuItem() ) @@ -1226,11 +1271,16 @@ function onHamburgerMenu (location, e) { } function onMainContextMenu (nodeProps, frame, tab, contextMenuType) { - if (contextMenuType === 'bookmark' || contextMenuType === 'bookmark-folder') { + let data = Immutable.fromJS(nodeProps) + if (!Array.isArray(nodeProps)) { + data = siteUtil.getSiteKey(data) + } + + if (contextMenuType === siteTags.BOOKMARK || contextMenuType === siteTags.BOOKMARK_FOLDER) { const activeFrame = Immutable.fromJS({ location: '', title: '', partitionNumber: frame.get('partitionNumber') }) - onSiteDetailContextMenu(Immutable.fromJS(nodeProps), activeFrame) - } else if (contextMenuType === 'history') { - onSiteDetailContextMenu(Immutable.fromJS(nodeProps)) + onSiteDetailContextMenu(data, contextMenuType, activeFrame) + } else if (contextMenuType === siteTags.HISTORY) { + onSiteDetailContextMenu(data, contextMenuType) } else if (contextMenuType === 'synopsis') { onLedgerContextMenu(nodeProps.location, nodeProps.hostPattern) } else if (contextMenuType === 'download') { @@ -1281,7 +1331,7 @@ function onUrlBarContextMenu (e) { e.stopPropagation() const searchDetail = appStoreRenderer.state.get('searchDetail') const windowState = windowStore.getState() - const activeFrame = getActiveFrame(windowState) + const activeFrame = frameStateUtil.getActiveFrame(windowState) const inputMenu = Menu.buildFromTemplate(urlBarTemplateInit(searchDetail, activeFrame, e)) inputMenu.popup(getCurrentWindow()) } @@ -1292,11 +1342,11 @@ function onFindBarContextMenu (e) { findBarMenu.popup(getCurrentWindow()) } -function onSiteDetailContextMenu (siteDetail, activeFrame, e) { +function onSiteDetailContextMenu (data, type, activeFrame, e) { if (e) { e.stopPropagation() } - const menu = Menu.buildFromTemplate(siteDetailTemplateInit(siteDetail, activeFrame)) + const menu = Menu.buildFromTemplate(siteDetailTemplateInit(data, type, activeFrame)) menu.popup(getCurrentWindow()) } diff --git a/js/data/newTabData.js b/js/data/newTabData.js index ccc0b4e286c..94431f18e7e 100644 --- a/js/data/newTabData.js +++ b/js/data/newTabData.js @@ -5,75 +5,64 @@ const {getBraveExtUrl} = require('../lib/appUrlUtil') const iconPath = getBraveExtUrl('img/newtab/defaultTopSitesIcon') -const now = 1 - module.exports.pinnedTopSites = [ - { - "count": 1, - "favicon": `${iconPath}/twitter.png`, - "lastAccessedTime": now, - "location": "https://twitter.com/brave", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgb(255, 255, 255)", - "title": "Brave Software (@brave) | Twitter" + { + 'key': 'https://twitter.com/brave/|0|0', + 'count': 0, + 'favicon': `${iconPath}/twitter.png`, + 'location': 'https://twitter.com/brave/', + 'themeColor': 'rgb(255, 255, 255)', + 'title': 'Brave Software (@brave) | Twitter' } ] module.exports.topSites = [ { - "count": 1, - "favicon": `${iconPath}/twitter.png`, - "lastAccessedTime": now, - "location": "https://twitter.com/brave", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgb(255, 255, 255)", - "title": "Brave Software (@brave) | Twitter" - }, { - "count": 1, - "favicon": `${iconPath}/facebook.png`, - "lastAccessedTime": now, - "location": "https://www.facebook.com/BraveSoftware/", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgb(59, 89, 152)", - "title": "Brave Software | Facebook" - }, { - "count": 1, - "favicon": `${iconPath}/youtube.png`, - "lastAccessedTime": now, - "location": "https://www.youtube.com/bravesoftware", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "#E62117", - "title": "Brave Browser - YouTube" - }, { - "count": 1, - "favicon": `${iconPath}/brave.ico`, - "lastAccessedTime": now, - "location": "https://brave.com/", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgb(255, 255, 255)", - "title": "Brave Software | Building a Better Web" - }, { - "count": 1, - "favicon": `${iconPath}/appstore.png`, - "lastAccessedTime": now, - "location": "https://itunes.apple.com/app/brave-web-browser/id1052879175?mt=8", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgba(255, 255, 255, 1)", - "title": "Brave Web Browser: Fast with built-in adblock on the App Store" - }, { - "count": 1, - "favicon": `${iconPath}/playstore.png`, - "lastAccessedTime": now, - "location": "https://play.google.com/store/apps/details?id=com.brave.browser", - "partitionNumber": 0, - "tags": ['default'], - "themeColor": "rgb(241, 241, 241)", - "title": "Brave Browser: Fast AdBlock – Apps para Android no Google Play" + 'key': 'https://twitter.com/brave/|0|0', + 'count': 0, + 'favicon': `${iconPath}/twitter.png`, + 'location': 'https://twitter.com/brave', + 'themeColor': 'rgb(255, 255, 255)', + 'title': 'Brave Software (@brave) | Twitter' + }, + { + 'key': 'https://www.facebook.com/BraveSoftware/|0|0', + 'count': 0, + 'favicon': `${iconPath}/facebook.png`, + 'location': 'https://www.facebook.com/BraveSoftware/', + 'themeColor': 'rgb(59, 89, 152)', + 'title': 'Brave Software | Facebook' + }, + { + 'key': 'https://www.youtube.com/bravesoftware/|0|0', + 'count': 0, + 'favicon': `${iconPath}/youtube.png`, + 'location': 'https://www.youtube.com/bravesoftware/', + 'themeColor': '#E62117', + 'title': 'Brave Browser - YouTube' + }, + { + 'key': 'https://brave.com/|0|0', + 'count': 0, + 'favicon': `${iconPath}/brave.ico`, + 'location': 'https://brave.com/', + 'themeColor': 'rgb(255, 255, 255)', + 'title': 'Brave Software | Building a Better Web' + }, + { + 'key': 'https://itunes.apple.com/app/brave-web-browser/id1052879175?mt=8|0|0', + 'count': 0, + 'favicon': `${iconPath}/appstore.png`, + 'location': 'https://itunes.apple.com/app/brave-web-browser/id1052879175?mt=8', + 'themeColor': 'rgba(255, 255, 255, 1)', + 'title': 'Brave Web Browser: Fast with built-in adblock on the App Store' + }, + { + 'key': 'https://play.google.com/store/apps/details?id=com.brave.browser|0|0', + 'count': 0, + 'favicon': `${iconPath}/playstore.png`, + 'location': 'https://play.google.com/store/apps/details?id=com.brave.browser', + 'themeColor': 'rgb(241, 241, 241)', + 'title': 'Brave Browser: Fast AdBlock – Apps para Android no Google Play' } ] diff --git a/js/lib/importer.js b/js/lib/importer.js index b0925cff8b2..7d336b86925 100644 --- a/js/lib/importer.js +++ b/js/lib/importer.js @@ -4,15 +4,15 @@ const fs = require('fs') const appActions = require('../actions/appActions') -const siteTags = require('../constants/siteTags') -const siteUtil = require('../state/siteUtil') const Immutable = require('immutable') const appStoreRenderer = require('../stores/appStoreRenderer') +const bookmarFoldersUtil = require('../../app/common/lib/bookmarkFoldersUtil') +const bookmarkFoldersState = require('../../app/common/state/bookmarkFoldersState') /** * Processes a single node from an exported HTML file from Firefox or Chrome * @param {Object} parserState - the current parser state - * @param {Object} node - The current DOM node which is being processed + * @param {Object} domNode - The current DOM node which is being processed */ function processBookmarkNode (parserState, domNode) { switch (domNode.tagName) { @@ -38,12 +38,11 @@ function processBookmarkNode (parserState, domNode) { title: domNode.innerText, folderId: parserState.nextFolderId, parentFolderId: parserState.parentFolderId, - lastAccessedTime: (domNode.getAttribute('LAST_MODIFIED') || domNode.getAttribute('ADD_DATE') || 0) * 1000, - tags: [siteTags.BOOKMARK_FOLDER] + lastAccessedTime: (domNode.getAttribute('LAST_MODIFIED') || domNode.getAttribute('ADD_DATE') || 0) * 1000 } parserState.lastFolderId = parserState.nextFolderId parserState.nextFolderId++ - parserState.sites.push(folder) + parserState.bookmarkFolders.push(folder) } else { parserState.lastFolderId = 0 parserState.foundBookmarksToolbar = true @@ -54,15 +53,13 @@ function processBookmarkNode (parserState, domNode) { if (domNode.href.startsWith('place:')) { break } - const site = { + const bookmarks = { title: domNode.innerText, location: domNode.href, favicon: domNode.getAttribute('ICON'), - parentFolderId: parserState.parentFolderId, - lastAccessedTime: (domNode.getAttribute('LAST_MODIFIED') || domNode.getAttribute('ADD_DATE') || 0) * 1000, - tags: [siteTags.BOOKMARK] + parentFolderId: parserState.parentFolderId } - parserState.sites.push(site) + parserState.bookmarks.push(bookmarks) break } } @@ -91,17 +88,19 @@ module.exports.importFromHTML = (path) => { // Each window's appStoreRenderer holds a copy of the app state, but it's not // mutable, so this is only used for getting the current list of sites. const parserState = { - nextFolderId: siteUtil.getNextFolderId(appStoreRenderer.state.get('sites')), + nextFolderId: bookmarFoldersUtil.getNextFolderId(bookmarkFoldersState.getFolders(appStoreRenderer.state)), lastFolderId: -1, parentFolderId: -1, - sites: [] + bookmarks: [], + bookmarkFolders: [] } // Process each of the nodes starting with the first node which is either DL or DT processBookmarkNode(parserState, doc.querySelector('dl, dt')) // Add the sites to the app store in the main process - appActions.addSite(Immutable.fromJS(parserState.sites)) + appActions.addBookmark(Immutable.fromJS(parserState.bookmarks)) + appActions.addBookmarkFolder(Immutable.fromJS(parserState.bookmarkFolders)) resolve({importCount: parserState.sites.length}) }) }) diff --git a/js/lib/textCalculator.js b/js/lib/textCalculator.js index 8068219c1f4..c543d340b83 100644 --- a/js/lib/textCalculator.js +++ b/js/lib/textCalculator.js @@ -2,8 +2,8 @@ * 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/. */ -const ctx = document.createElement('canvas').getContext('2d') module.exports.calculateTextWidth = (text, font = '11px Arial') => { + const ctx = document.createElement('canvas').getContext('2d') ctx.font = font return ctx.measureText(text).width } diff --git a/js/state/siteUtil.js b/js/state/siteUtil.js index 5320b15a9c7..42b50f0d786 100644 --- a/js/state/siteUtil.js +++ b/js/state/siteUtil.js @@ -2,17 +2,11 @@ * You can obtain one at http://mozilla.org/MPL/2.0/. */ 'use strict' -const Immutable = require('immutable') -const siteCache = require('../../app/common/state/siteCache') const siteTags = require('../constants/siteTags') -const settings = require('../constants/settings') -const getSetting = require('../settings').getSetting const UrlUtil = require('../lib/urlutil') const urlParse = require('../../app/common/urlParse') -const {makeImmutable} = require('../../app/common/state/immutableUtil') - -const defaultTags = new Immutable.List([siteTags.DEFAULT]) +// TODO remove const isBookmark = (tags) => { if (!tags) { return false @@ -20,6 +14,7 @@ const isBookmark = (tags) => { return tags.includes(siteTags.BOOKMARK) } +// TODO remove const isBookmarkFolder = (tags) => { if (!tags) { return false @@ -28,25 +23,6 @@ const isBookmarkFolder = (tags) => { (tags && typeof tags !== 'string' && tags.includes(siteTags.BOOKMARK_FOLDER)) } -const isPinnedTab = (tags) => { - if (!tags) { - return false - } - return tags.includes(siteTags.PINNED) -} -module.exports.isPinnedTab = isPinnedTab - -const reorderSite = (sites, order) => { - sites = sites.map((site) => { - const siteOrder = site.get('order') - if (siteOrder > order) { - return site.set('order', siteOrder - 1) - } - return site - }) - return sites -} - /** * Sort comparator for sort function */ @@ -83,24 +59,6 @@ module.exports.getSiteKey = function (siteDetail) { return null } -/** - * Calculate location for siteKey - * - * @param siteKey The site key to to be calculated - * @return {string|null} - */ -module.exports.getLocationFromSiteKey = function (siteKey) { - if (!siteKey) { - return null - } - - const splitKey = siteKey.split('|', 2) - if (typeof splitKey[0] === 'string' && typeof splitKey[1] === 'string') { - return splitKey[0] - } - return null -} - /** * Checks if a siteDetail has the specified tag. * Depends on siteDeatil siteKey being accurate. @@ -109,227 +67,13 @@ module.exports.getLocationFromSiteKey = function (siteKey) { * @param siteDetail The site to check if it's in the specified tag * @return true if the location is already bookmarked */ +// TODO remove this when sync is updated module.exports.isSiteBookmarked = function (sites, siteDetail) { const siteKey = module.exports.getSiteKey(siteDetail) const siteTags = sites.getIn([siteKey, 'tags']) return isBookmark(siteTags) } -/** - * Checks if a location is bookmarked. - * - * @param state The application state Immutable map - * @param {string} location - * @return {boolean} - */ -module.exports.isLocationBookmarked = function (state, location) { - const sites = state.get('sites') - const siteKeys = siteCache.getLocationSiteKeys(state, location) - if (!siteKeys || siteKeys.length === 0) { - return false - } - return siteKeys.some(key => { - const site = sites.get(key) - if (!site) { - return false - } - return isBookmark(site.get('tags')) - }) -} - -const getNextFolderIdItem = (sites) => - sites.max((siteA, siteB) => { - const folderIdA = siteA.get('folderId') - const folderIdB = siteB.get('folderId') - if (folderIdA === folderIdB) { - return 0 - } - if (folderIdA === undefined) { - return false - } - if (folderIdB === undefined) { - return true - } - return folderIdA > folderIdB - }) - -module.exports.getNextFolderId = (sites) => { - const defaultFolderId = 0 - if (!sites) { - return defaultFolderId - } - const maxIdItem = getNextFolderIdItem(sites) - return (maxIdItem ? (maxIdItem.get('folderId') || 0) : 0) + 1 -} - -module.exports.getNextFolderName = (sites, name) => { - if (!sites) { - return name - } - const site = sites.find((site) => - isBookmarkFolder(site.get('tags')) && - site.get('customTitle') === name - ) - if (!site) { - return name - } - const filenameFormat = /(.*) \((\d+)\)/ - let result = filenameFormat.exec(name) - if (!result) { - return module.exports.getNextFolderName(sites, name + ' (1)') - } - const nextNum = parseInt(result[2]) + 1 - return module.exports.getNextFolderName(sites, result[1] + ' (' + nextNum + ')') -} - -const mergeSiteLastAccessedTime = (oldSiteDetail, newSiteDetail, tag) => { - const newTime = newSiteDetail && newSiteDetail.get('lastAccessedTime') - const oldTime = oldSiteDetail && oldSiteDetail.get('lastAccessedTime') - if (!isBookmark(tag) && !isBookmarkFolder(tag)) { - return newTime || new Date().getTime() - } - if (newTime && newTime !== 0) { - return newTime - } else if (oldTime && oldTime !== 0) { - return oldTime - } else { - return 0 - } -} - -// Some details can be copied from the existing siteDetail if null -// ex: parentFolderId, partitionNumber, and favicon -const mergeSiteDetails = (oldSiteDetail, newSiteDetail, tag, folderId, order) => { - let tags = (oldSiteDetail && oldSiteDetail.get('tags')) || new Immutable.List() - if (tag) { - tags = tags.toSet().add(tag).toList() - } - - const customTitle = typeof newSiteDetail.get('customTitle') === 'string' - ? newSiteDetail.get('customTitle') - : (newSiteDetail.get('customTitle') || (oldSiteDetail && oldSiteDetail.get('customTitle'))) - - const lastAccessedTime = mergeSiteLastAccessedTime(oldSiteDetail, newSiteDetail, tag) - - let site = makeImmutable({ - lastAccessedTime, - tags, - objectId: newSiteDetail.get('objectId') || (oldSiteDetail ? oldSiteDetail.get('objectId') : undefined), - title: newSiteDetail.get('title'), - order - }) - - if (oldSiteDetail && oldSiteDetail.get('order') !== undefined) { - site = site.set('order', oldSiteDetail.get('order')) - } - - if (newSiteDetail.get('location')) { - site = site.set('location', newSiteDetail.get('location')) - } - if (folderId) { - site = site.set('folderId', Number(folderId)) - } - if (typeof customTitle === 'string') { - site = site.set('customTitle', customTitle) - } - if (newSiteDetail.get('parentFolderId') !== undefined || (oldSiteDetail && oldSiteDetail.get('parentFolderId'))) { - let parentFolderId = newSiteDetail.get('parentFolderId') !== undefined - ? newSiteDetail.get('parentFolderId') : oldSiteDetail.get('parentFolderId') - site = site.set('parentFolderId', Number(parentFolderId)) - } - if (newSiteDetail.get('partitionNumber') !== undefined || (oldSiteDetail && oldSiteDetail.get('partitionNumber'))) { - let partitionNumber = newSiteDetail.get('partitionNumber') !== undefined - ? newSiteDetail.get('partitionNumber') : oldSiteDetail.get('partitionNumber') - site = site.set('partitionNumber', Number(partitionNumber)) - } - if (newSiteDetail.get('favicon') || (oldSiteDetail && oldSiteDetail.get('favicon'))) { - site = site.set('favicon', newSiteDetail.get('favicon') || oldSiteDetail.get('favicon')) - } - if (newSiteDetail.get('themeColor') || (oldSiteDetail && oldSiteDetail.get('themeColor'))) { - site = site.set('themeColor', newSiteDetail.get('themeColor') || oldSiteDetail.get('themeColor')) - } - if (site.get('tags').size === 0) { - // Increment the visit count for history items - site = site.set('count', ((oldSiteDetail || site).get('count') || 0) + 1) - } - - return site -} - -/** - * Adds the specified siteDetail in appState.sites. - * - * Examples of updating: - * - editing bookmark in add/edit modal - * - when timestamp is added (history entry) - * - moving bookmark to/from a folder - * - * @param state - The application state - * @param siteDetail - The site details object to add or update - * @param tag - The tag to add for this site - * See siteTags.js for supported types. No tag means just a history item - * @param oldKey - key of a changed site - * @param {boolean} skipSync - True if this site was downloaded by sync and - * does not to be re-uploaded - * @return The new state Immutable object - */ -module.exports.addSite = function (state, siteDetail, tag, oldKey, skipSync) { - let sites = state.get('sites') - // Get tag from siteDetail object if not passed via tag param - if (tag === undefined) { - tag = siteDetail.getIn(['tags', 0]) - } - - if (!oldKey) { - oldKey = module.exports.getSiteKey(siteDetail) - } - - const oldSite = oldKey !== null ? sites.get(oldKey) : null - let folderId = siteDetail.get('folderId') - - if (tag === siteTags.BOOKMARK_FOLDER) { - if (!oldSite && folderId) { - // Remove duplicate folder (needed for import) - const dupFolder = sites.find((site) => isBookmarkFolder(site.get('tags')) && - site.get('parentFolderId') === siteDetail.get('parentFolderId') && - site.get('customTitle') === siteDetail.get('customTitle')) - if (dupFolder) { - state = module.exports.removeSite(state, dupFolder, siteTags.BOOKMARK_FOLDER, true) - } - } else if (!folderId) { - // Assign an id if this is a new folder - folderId = module.exports.getNextFolderId(sites) - } - } - - let site = mergeSiteDetails(oldSite, siteDetail, tag, folderId, sites.size) - if (oldSite) { - sites = sites.delete(oldKey) - } - - let location - if (site.has('location')) { - location = UrlUtil.getLocationIfPDF(site.get('location')) - site = site.set('location', location) - } - - const oldLocation = (oldSite && oldSite.get('location')) || site.get('location') - state = siteCache.removeLocationSiteKey(state, oldLocation, oldKey) - - if (skipSync) { - site = site.set('skipSync', true) - } - - state = state.set('sites', sites) - const key = module.exports.getSiteKey(site) - if (key === null) { - return state - } - state = state.setIn(['sites', key], site) - state = siteCache.addLocationSiteKey(state, location, key) - return state -} - /** * Removes the appropriate tag from a site given the site's sync objectId. * @param {Immutable.Map} state - Application state @@ -347,12 +91,16 @@ module.exports.removeSiteByObjectId = function (state, objectId, objectData) { if (!site) { return state } + + // TODO fix so that correct remove is called + /* let tag = '' if (objectData === 'bookmark') { // determine whether this is a folder tag = site.get('folderId') ? siteTags.BOOKMARK_FOLDER : siteTags.BOOKMARK } state = module.exports.removeSite(state, site, tag) + */ // If the object has been removed completely, purge it from the sync site // cache if (!state.getIn(objectKey)) { @@ -361,67 +109,6 @@ module.exports.removeSiteByObjectId = function (state, objectId, objectData) { return state } -/** - * Removes the specified tag from a siteDetail - * - * @param {Immutable.Map} state The application state Immutable map - * @param {Immutable.Map} siteDetail The siteDetail to remove a tag from - * @param {string} tag - * @param {boolean} reorder whether to reorder sites (default with reorder) - * @param {Function=} syncCallback - * @return {Immutable.Map} The new state Immutable object - */ -module.exports.removeSite = function (state, siteDetail, tag, reorder = true, syncCallback) { - let sites = state.get('sites') - const key = module.exports.getSiteKey(siteDetail) - if (!key) { - return state - } - if (getSetting(settings.SYNC_ENABLED) === true && syncCallback) { - syncCallback(sites.getIn([key])) - } - - const tags = sites.getIn([key, 'tags']) - if (isBookmarkFolder(tags)) { - const folderId = sites.getIn([key, 'folderId']) - const childSites = sites.filter((site) => site.get('parentFolderId') === folderId) - childSites.forEach((site) => { - const tags = site.get('tags') - tags.forEach((tag) => { - state = module.exports.removeSite(state, site, tag, false, syncCallback) - }) - }) - } - - const location = siteDetail.get('location') - state = siteCache.removeLocationSiteKey(state, location, key) - - const stateKey = ['sites', key] - let site = state.getIn(stateKey) - if (!site) { - return state - } - if (isBookmark(tag)) { - if (isPinnedTab(tags)) { - const tags = site.get('tags').filterNot((tag) => tag === siteTags.BOOKMARK) - site = site.set('tags', tags) - return state.setIn(stateKey, site) - } - if (state.get('sites').size && reorder) { - const order = state.getIn(stateKey.concat(['order'])) - state = state.set('sites', reorderSite(state.get('sites'), order)) - } - return state.deleteIn(['sites', key]) - } else if (isPinnedTab(tag)) { - const tags = site.get('tags').filterNot((tag) => tag === siteTags.PINNED) - site = site.set('tags', tags) - return state.setIn(stateKey, site) - } else { - site = site.set('lastAccessedTime', undefined) - return state.setIn(stateKey, site) - } -} - /** * Called by isMoveAllowed * Trace a folder's ancestory, collecting all parent folderIds until reaching Bookmarks Toolbar (folderId=0) @@ -459,223 +146,12 @@ module.exports.isMoveAllowed = (sites, sourceDetail, destinationDetail) => { return true } -/** - * Moves the specified site from one location to another - * - * @param state The application state Immutable map - * @param sourceKey The site key to move - * @param destinationKey The site key to move to - * @param prepend Whether the destination detail should be prepended or not, not used if destinationIsParent is true - * @param destinationIsParent Whether the item should be moved inside of the destinationDetail. - * @param disallowReparent If set to true, parent folder will not be set - * @return The new state Immutable object - */ -module.exports.moveSite = function (state, sourceKey, destinationKey, prepend, - destinationIsParent, disallowReparent) { - let sites = state.get('sites') - let sourceSite = sites.get(sourceKey, Immutable.Map()) - const destinationSite = sites.get(destinationKey, Immutable.Map()) - - if (sourceSite.isEmpty() || !module.exports.isMoveAllowed(sites, sourceSite, destinationSite)) { - return state - } - - const sourceSiteIndex = sourceSite.get('order') - let destinationSiteIndex - if (destinationIsParent) { - // When the destination is the parent we want to put it at the end - destinationSiteIndex = sites.size - 1 - prepend = true - } else { - destinationSiteIndex = destinationSite.get('order') - } - - let newIndex = destinationSiteIndex + (prepend ? 0 : 1) - if (destinationSiteIndex > sourceSiteIndex) { - --newIndex - } - - const location = sourceSite.get('location') - state = siteCache.removeLocationSiteKey(state, location, sourceKey) - state = state.deleteIn(['sites', sourceKey]) - state = state.set('sites', state.get('sites').map((site) => { - const siteOrder = site.get('order') - if (siteOrder >= newIndex && siteOrder < sourceSiteIndex) { - return site.set('order', siteOrder + 1) - } else if (siteOrder <= newIndex && siteOrder > sourceSiteIndex) { - return site.set('order', siteOrder - 1) - } - return site - })) - sourceSite = sourceSite.set('order', newIndex) - - if (!disallowReparent) { - if (destinationIsParent && destinationSite.get('folderId') !== sourceSite.get('folderId')) { - sourceSite = sourceSite.set('parentFolderId', destinationSite.get('folderId')) - } else if (destinationSite.get('parentFolderId') == null) { - sourceSite = sourceSite.set('parentFolderId', 0) - } else if (destinationSite.get('parentFolderId') !== sourceSite.get('parentFolderId')) { - sourceSite = sourceSite.set('parentFolderId', destinationSite.get('parentFolderId')) - } - } - const destinationSiteKey = module.exports.getSiteKey(sourceSite) - state = siteCache.addLocationSiteKey(state, location, destinationSiteKey) - return state.setIn(['sites', destinationSiteKey], sourceSite) -} - -module.exports.getDetailFromFrame = function (frame, tag) { - const pinnedLocation = frame.get('pinnedLocation') - let location = frame.get('location') - if (pinnedLocation && pinnedLocation !== 'about:blank' && tag === siteTags.PINNED) { - location = frame.get('pinnedLocation') - } - - return makeImmutable({ - location, - title: frame.get('title'), - partitionNumber: frame.get('partitionNumber'), - tags: tag ? [tag] : [], - favicon: frame.get('icon'), - themeColor: frame.get('themeColor') || frame.get('computedThemeColor') - }) -} - -const getSitesBySubkey = (sites, siteKey, tag) => { - if (!sites || !siteKey) { - return makeImmutable([]) - } - const splitKey = siteKey.split('|', 2) - const partialKey = splitKey.join('|') - const matches = sites.filter((site, key) => { - if (key.indexOf(partialKey) > -1 && (!tag || (tag && site.get('tags').includes(tag)))) { - return true - } - return false - }) - return matches.toList() -} - -module.exports.getDetailFromTab = function (tab, tag, sites) { - let location = tab.get('url') - const partitionNumber = tab.get('partitionNumber') - let parentFolderId - - // if site map is available, look up extra information: - // - original url (if redirected) - // - parent folder id - if (sites) { - let results = makeImmutable([]) - - // get all sites matching URL and partition (disregarding parentFolderId) - let siteKey = module.exports.getSiteKey(makeImmutable({location, partitionNumber})) - results = results.merge(getSitesBySubkey(sites, siteKey, tag)) - - // only check for provisional location if entry is not found - if (results.size === 0) { - // if provisional location is different, grab any results which have that URL - // this may be different if the site was redirected - const provisionalLocation = tab.getIn(['frame', 'provisionalLocation']) - if (provisionalLocation && provisionalLocation !== location) { - siteKey = module.exports.getSiteKey(makeImmutable({ - location: provisionalLocation, - partitionNumber - })) - results = results.merge(getSitesBySubkey(sites, siteKey, tag)) - } - } - - // update details which get returned below - if (results.size > 0) { - location = results.getIn([0, 'location']) - parentFolderId = results.getIn([0, 'parentFolderId']) - } - } - - const siteDetail = { - location: location, - title: tab.get('title'), - tags: tag ? [tag] : [] - } - if (partitionNumber) { - siteDetail.partitionNumber = partitionNumber - } - if (parentFolderId) { - siteDetail.parentFolderId = parentFolderId - } - return Immutable.fromJS(siteDetail) -} - -module.exports.getDetailFromCreateProperties = function (createProperties, tag) { - const siteDetail = { - location: createProperties.get('url'), - tags: tag ? [tag] : [] - } - if (createProperties.get('partitionNumber') !== undefined) { - siteDetail.partitionNumber = createProperties.get('partitionNumber') - } - return Immutable.fromJS(siteDetail) -} - -/** - * Update the favicon URL for all entries in the state sites - * which match a given location. Currently, there should only be - * one match, but this will handle multiple. - * - * @param state The application state - * @param location URL for the entry needing an update - * @param favicon favicon URL - */ -module.exports.updateSiteFavicon = function (state, location, favicon) { - if (UrlUtil.isNotURL(location)) { - return state - } - const siteKeys = siteCache.getLocationSiteKeys(state, location) - if (!siteKeys || siteKeys.length === 0) { - return state - } - siteKeys.forEach((siteKey) => { - state = state.setIn(['sites', siteKey, 'favicon'], favicon) - }) - return state -} - -/** - * Converts a siteDetail to createProperties format - * @param {Object} siteDetail - A site detail as per app state - * @return {Object} A createProperties plain JS object, not ImmutableJS - */ -module.exports.toCreateProperties = function (siteDetail) { - return { - url: siteDetail.get('location'), - partitionNumber: siteDetail.get('partitionNumber') - } -} - -/** - * Compares 2 site details - * @param siteDetail1 The first site detail to compare. - * @param siteDetail2 The second site detail to compare. - * @return true if the site details should be considered the same. - */ -module.exports.isEquivalent = function (siteDetail1, siteDetail2) { - const isFolder1 = module.exports.isFolder(siteDetail1) - const isFolder2 = module.exports.isFolder(siteDetail2) - if (isFolder1 !== isFolder2) { - return false - } - - // If they are both folders - if (isFolder1) { - return siteDetail1.get('folderId') === siteDetail2.get('folderId') - } - return siteDetail1.get('location') === siteDetail2.get('location') && siteDetail1.get('partitionNumber') === siteDetail2.get('partitionNumber') -} - /** * Determines if the site detail is a bookmark. * @param siteDetail The site detail to check. * @return true if the site detail has a bookmark tag. */ +// TODO remove when sync is refactored module.exports.isBookmark = function (siteDetail) { if (siteDetail) { return isBookmark(siteDetail.get('tags')) @@ -688,6 +164,7 @@ module.exports.isBookmark = function (siteDetail) { * @param siteDetail The site detail to check. * @return true if the site detail is a folder. */ +// TODO remove when sync is refactored module.exports.isFolder = function (siteDetail) { if (siteDetail) { return isBookmarkFolder(siteDetail.get('tags')) && siteDetail.get('folderId') !== undefined @@ -695,25 +172,16 @@ module.exports.isFolder = function (siteDetail) { return false } -/** - * Determines if the site detail is an imported bookmark. - * @param siteDetail The site detail to check. - * @return true if the site detail is a folder. - */ -module.exports.isImportedBookmark = function (siteDetail) { - return siteDetail.get('lastAccessedTime') === 0 -} - /** * Determines if the site detail is a history entry. * @param siteDetail The site detail to check. * @return true if the site detail is a history entry. */ +// TODO remove when sync is refactored module.exports.isHistoryEntry = function (siteDetail) { if (siteDetail && typeof siteDetail.get('location') === 'string') { const tags = siteDetail.get('tags') if (siteDetail.get('location').startsWith('about:') || - module.exports.isDefaultEntry(siteDetail) || isBookmarkFolder(tags)) { return false } @@ -722,122 +190,11 @@ module.exports.isHistoryEntry = function (siteDetail) { return false } -/** - * Determines if the site detail is one of default sites in about:newtab. - * @param {Immutable.Map} siteDetail The site detail to check. - * @returns {boolean} if the site detail is a default newtab entry. - */ -module.exports.isDefaultEntry = function (siteDetail) { - return Immutable.is(siteDetail.get('tags'), defaultTags) && - siteDetail.get('lastAccessedTime') === 1 -} - -/** - * Get a folder by folderId - * @returns {Immutable.List.} sites - * @param {number} folderId - * @returns {Array[, ]|undefined} - */ -module.exports.getFolder = function (sites, folderId) { - const entry = sites.findEntry((site, _path) => { - return module.exports.isFolder(site) && site.get('folderId') === folderId - }) - if (!entry) { return undefined } - return entry -} - -/** - * Obtains an array of folders - */ -module.exports.getFolders = function (sites, folderId, parentId, labelPrefix) { - parentId = parentId || 0 - let folders = [] - const results = sites - .filter(site => { - return (site.get('parentFolderId', 0) === parentId && module.exports.isFolder(site)) - }) - .toList() - .sort(module.exports.siteSort) - - const resultSize = results.size - for (let i = 0; i < resultSize; i++) { - const site = results.get(i) - if (site.get('folderId') === folderId) { - continue - } - - const label = (labelPrefix || '') + (site.get('customTitle') || site.get('title')) - folders.push({ - folderId: site.get('folderId'), - parentFolderId: site.get('parentFolderId'), - label - }) - const subsites = module.exports.getFolders(sites, folderId, site.get('folderId'), (label || '') + ' / ') - folders = folders.concat(subsites) - } - - return folders -} - -/** - * Filters out non recent sites based on the app setting for history size. - * @param sites The application state's Immutable sites list. - */ -module.exports.filterOutNonRecents = function (sites) { - const sitesWithTags = sites - .filter((site) => site.get('tags').size) - const topHistorySites = sites - .filter((site) => site.get('tags').size === 0) - .sort((site1, site2) => (site2.get('lastAccessedTime') || 0) - (site1.get('lastAccessedTime') || 0)) - .take(getSetting(settings.AUTOCOMPLETE_HISTORY_SIZE)) - return sitesWithTags.concat(topHistorySites) -} - -/** - * Filters sites relative to a parent site (folder). - * @param sites The application state's Immutable sites list. - * @param relSite The folder to filter to. - */ -module.exports.filterSitesRelativeTo = function (sites, relSite) { - if (!relSite.get('folderId')) { - return sites - } - return sites.filter((site) => site.get('parentFolderId') === relSite.get('folderId')) -} - -/** - * Clears history by - * - filtering out entries which have no tags - * - setting lastAccessedTime to null for remaining entries (bookmarks) - * @param sites The application state's Immutable sites list. - */ -module.exports.clearHistory = function (sites) { - let bookmarks = sites.filter((site) => site.get('tags') && site.get('tags').size > 0) - bookmarks.forEach((site, index) => { - if (site.get('lastAccessedTime')) { - bookmarks = bookmarks.setIn([index, 'lastAccessedTime'], null) - } - }) - return bookmarks -} - -/** - * Returns all sites that have a bookmark tag. - * @param sites The application state's Immutable sites list. - */ - -module.exports.getBookmarks = function (sites) { - if (sites) { - return sites.filter((site) => isBookmarkFolder(site.get('tags')) || isBookmark(site.get('tags'))) - } - return makeImmutable({}) -} - /** * Gets a site origin (scheme + hostname + port) from a URL or null if not * available. * @param {string} location - * @return {string?} + * @return {string|null} */ module.exports.getOrigin = function (location) { // Returns scheme + hostname + port diff --git a/js/state/syncUtil.js b/js/state/syncUtil.js index 2439eef3d19..63d8b15b4fc 100644 --- a/js/state/syncUtil.js +++ b/js/state/syncUtil.js @@ -9,7 +9,9 @@ const crypto = require('crypto') const writeActions = require('../constants/sync/proto').actions const siteTags = require('../constants/siteTags') const siteUtil = require('./siteUtil') +const {getSetting} = require('../settings') const {isDataUrl} = require('../lib/urlutil') +const settings = require('../constants/settings') const CATEGORY_MAP = { bookmark: { @@ -47,7 +49,19 @@ module.exports.siteSettingDefaults = { // Whitelist of valid browser-laptop site fields. In browser-laptop, site // is used for both bookmarks and history sites. -const SITE_FIELDS = ['objectId', 'location', 'title', 'customTitle', 'tags', 'favicon', 'themeColor', 'lastAccessedTime', 'creationTime', 'partitionNumber', 'folderId', 'parentFolderId'] +const SITE_FIELDS = [ + 'objectId', + 'location', + 'title', + 'tags', + 'favicon', + 'themeColor', + 'lastAccessedTime', + 'creationTime', + 'partitionNumber', + 'folderId', + 'parentFolderId' +] const pickFields = (object, fields) => { return fields.reduce((a, x) => { @@ -101,10 +115,7 @@ module.exports.getSiteDataFromRecord = (record, appState, records) => { record.bookmark && record.bookmark.site, {objectId} ) - if (siteProps.customTitle === '') { - // browser-laptop UI expects the customTitle field to not exist if it is empty - delete siteProps.customTitle - } + delete siteProps.customTitle if (record.objectData === 'bookmark') { const existingFolderId = existingObjectData && existingObjectData.get('folderId') if (existingFolderId) { @@ -151,7 +162,6 @@ const applySiteSettingRecord = (record) => { return value } } - const appActions = require('../actions/appActions') const hostPattern = record.siteSetting.hostPattern if (!hostPattern) { throw new Error('siteSetting.hostPattern is required.') @@ -210,7 +220,7 @@ const applySyncRecord = (record) => { break case 'device': const device = Object.assign({}, record.device, {lastRecordTimestamp: record.syncTimestamp}) - require('../actions/appActions').saveSyncDevices({ + appActions.saveSyncDevices({ [deviceIdString(record.deviceId)]: device }) break @@ -225,19 +235,39 @@ const applySyncRecord = (record) => { */ module.exports.applySyncRecords = (records) => { if (!records || records.length === 0) { return } - const siteRecords = [] + const bookmarkRecords = [] + const bookmarkFoldersRecords = [] + const historyRecords = [] const otherRecords = [] records.forEach((record) => { - if (record && ['bookmark', 'historySite'].includes(record.objectData)) { - siteRecords.push(record) + if (record && ['bookmark'].includes(record.objectData)) { + bookmarkRecords.push(record) + } else if (record && ['historySite'].includes(record.objectData)) { + historyRecords.push(record) + } else if (record && ['bookmark-folder'].includes(record.objectData)) { + bookmarkFoldersRecords.push(record) } else { otherRecords.push(record) } }) applyNonBatchedRecords(otherRecords) - if (siteRecords.length) { + + // TODO we now always add (need to check record.action for what to do) + if (bookmarkRecords.length) { + setImmediate(() => { + appActions.addBookmarks(new Immutable.List(bookmarkRecords)) + }) + } + + if (bookmarkFoldersRecords.length) { setImmediate(() => { - require('../actions/appActions').applySiteRecords(new Immutable.List(siteRecords)) + appActions.addBookmarkFolder(new Immutable.List(bookmarkFoldersRecords)) + }) + } + + if (historyRecords.length) { + setImmediate(() => { + appActions.addHistorySite(new Immutable.List(historyRecords)) }) } } @@ -293,6 +323,7 @@ module.exports.getExistingObject = (categoryName, syncRecord) => { */ module.exports.createSiteCache = (appState) => { const objectsById = new Immutable.Map().withMutations(objectsById => { + // TODO what to do here? appState.get('sites').forEach((site, siteKey) => { const objectId = site.get('objectId') if (!objectId) { return true } @@ -461,7 +492,6 @@ module.exports.createSiteData = (site, appState) => { const siteData = { location: '', title: '', - customTitle: '', favicon: '', lastAccessedTime: 0, creationTime: 0 @@ -582,3 +612,7 @@ module.exports.deepArrayify = deepArrayify module.exports.ipcSafeObject = (object) => { return deepArrayify(object) } + +module.exports.syncEnabled = () => { + return getSetting(settings.SYNC_ENABLED) === true +} diff --git a/js/stores/appStore.js b/js/stores/appStore.js index 04a72501ba5..43ee9cc3a26 100644 --- a/js/stores/appStore.js +++ b/js/stores/appStore.js @@ -45,6 +45,9 @@ const extensionState = require('../../app/common/state/extensionState') const aboutNewTabState = require('../../app/common/state/aboutNewTabState') const aboutHistoryState = require('../../app/common/state/aboutHistoryState') const tabState = require('../../app/common/state/tabState') +const bookmarksState = require('../../app/common/state/bookmarksState') +const bookmarkFoldersState = require('../../app/common/state/bookmarkFoldersState') +const historyState = require('../../app/common/state/historyState') const isDarwin = process.platform === 'darwin' const isWindows = process.platform === 'win32' @@ -343,16 +346,6 @@ function setDefaultWindowSize () { const appStore = new AppStore() const emitChanges = debounce(appStore.emitChanges.bind(appStore), 5) -/** - * Clears out the top X non tagged sites. - * This is debounced to every 1 minute, the cleanup is not particularly intensive - * but there's no point to cleanup frequently. - */ -const filterOutNonRecents = debounce(() => { - appState = appState.set('sites', siteUtil.filterOutNonRecents(appState.get('sites'))) - emitChanges() -}, 60 * 1000) - /** * Useful for updating non-react preferences (electron properties, etc). * Called when any settings are modified (ex: via preferences). @@ -398,7 +391,10 @@ const handleAppAction = (action) => { // until we have a better way to manage dependencies. // tabsReducer must be above dragDropReducer. require('../../app/browser/reducers/tabsReducer'), - require('../../app/browser/reducers/sitesReducer'), + require('../../app/browser/reducers/bookmarksReducer'), + require('../../app/browser/reducers/bookmarkFoldersReducer'), + require('../../app/browser/reducers/historyReducer'), + require('../../app/browser/reducers/pinnedSitesReducer'), require('../../app/browser/reducers/windowsReducer'), require('../../app/browser/reducers/syncReducer'), require('../../app/browser/reducers/clipboardReducer'), @@ -437,6 +433,7 @@ const handleAppAction = (action) => { appState = webtorrent.init(appState, action, appStore) appState = profiles.init(appState, action, appStore) appState = require('../../app/sync').init(appState, action, appStore) + calculateTopSites(true, true) break case appConstants.APP_SHUTTING_DOWN: AppDispatcher.shutdown() @@ -448,31 +445,12 @@ const handleAppAction = (action) => { case appConstants.APP_CHANGE_NEW_TAB_DETAIL: appState = aboutNewTabState.mergeDetails(appState, action) if (action.refresh) { - calculateTopSites(true) + calculateTopSites(true, true) } break - case appConstants.APP_POPULATE_HISTORY: - appState = aboutHistoryState.setHistory(appState, action) - break case appConstants.APP_DATA_URL_COPIED: nativeImage.copyDataURL(action.dataURL, action.html, action.text) break - case appConstants.APP_ADD_SITE: - case appConstants.APP_ADD_BOOKMARK: - case appConstants.APP_EDIT_BOOKMARK: - const oldSiteSize = appState.get('sites').size - calculateTopSites(false) - appState = aboutHistoryState.setHistory(appState, action) - // If there was an item added then clear out the old history entries - if (oldSiteSize !== appState.get('sites').size) { - filterOutNonRecents() - } - break - case appConstants.APP_APPLY_SITE_RECORDS: - case appConstants.APP_REMOVE_SITE: - calculateTopSites(true) - appState = aboutHistoryState.setHistory(appState, action) - break case appConstants.APP_SET_DATA_FILE_ETAG: appState = appState.setIn([action.resourceName, 'etag'], action.etag) break @@ -665,8 +643,8 @@ const handleAppAction = (action) => { const clearData = defaults ? defaults.merge(temp) : temp if (clearData.get('browserHistory')) { - calculateTopSites(true) - appState = aboutHistoryState.setHistory(appState) + appState = aboutNewTabState.clearTopSites(appState) + appState = aboutHistoryState.clearHistory(appState) syncActions.clearHistory() BrowserWindow.getAllWindows().forEach((wnd) => wnd.webContents.send(messages.CLEAR_CLOSED_FRAMES)) } @@ -800,8 +778,9 @@ const handleAppAction = (action) => { appState = appState.set('defaultBrowserCheckComplete', {}) break case windowConstants.WINDOW_SET_FAVICON: - appState = siteUtil.updateSiteFavicon(appState, action.frameProps.get('location'), action.favicon) if (action.frameProps.get('favicon') !== action.favicon) { + appState = bookmarksState.updateFavicon(appState, action.frameProps.get('location'), action.favicon) + appState = historyState.updateFavicon(appState, action.frameProps, action.favicon) calculateTopSites(false) } break @@ -856,13 +835,22 @@ const handleAppAction = (action) => { const syncDefault = Immutable.fromJS(sessionStore.defaultAppState().sync) const originalSeed = appState.getIn(['sync', 'seed']) appState = appState.set('sync', syncDefault) - appState.get('sites').forEach((site, key) => { - if (site.has('objectId') && syncUtil.isSyncable('bookmark', site)) { + bookmarksState.getBookmarks(appState).forEach((site, key) => { + if (site.has('objectId')) { + // Remember which profile this bookmark was originally synced to. + // Since old bookmarks should be synced when a new profile is created, + // we have to keep track of which profile already has these bookmarks + // or else the old profile may have these bookmarks duplicated. #7405 + appState = appState.setIn(['bookmarks', key, 'originalSeed'], originalSeed) + } + }) + bookmarkFoldersState.getFolders(appState).forEach((site, key) => { + if (site.has('objectId')) { // Remember which profile this bookmark was originally synced to. // Since old bookmarks should be synced when a new profile is created, // we have to keep track of which profile already has these bookmarks // or else the old profile may have these bookmarks duplicated. #7405 - appState = appState.setIn(['sites', key, 'originalSeed'], originalSeed) + appState = appState.setIn(['bookmarks', key, 'originalSeed'], originalSeed) } }) appState.setIn(['sync', 'devices'], {}) diff --git a/js/stores/windowStore.js b/js/stores/windowStore.js index 4e09138b53f..2741c6cc2fc 100644 --- a/js/stores/windowStore.js +++ b/js/stores/windowStore.js @@ -23,6 +23,9 @@ const assert = require('assert') const contextMenuState = require('../../app/common/state/contextMenuState') const appStoreRenderer = require('./appStoreRenderer') const windowActions = require('../actions/windowActions') +const siteUtil = require('../state/siteUtil') +const bookmarkFoldersState = require('../../app/common/state/bookmarkFoldersState') +const bookmarksState = require('../../app/common/state/bookmarksState') let windowState = Immutable.fromJS({ activeFrameKey: null, @@ -64,10 +67,6 @@ class WindowStore extends EventEmitter { windowState = newWindowState } - getFrames () { - return frameStateUtil.getFrames(this.state) - } - getFrame (key) { return frameStateUtil.getFrameByKey(windowState, key) } @@ -471,38 +470,55 @@ const doAction = (action) => { windowState = windowState.delete('bookmarkDetail') break case windowConstants.WINDOW_ON_EDIT_BOOKMARK: - const siteDetail = appStoreRenderer.state.getIn(['sites', action.editKey]) + { + const siteDetail = bookmarksState.getBookmark(appStoreRenderer.state, action.editKey) - windowState = windowState.setIn(['bookmarkDetail'], Immutable.fromJS({ - siteDetail: siteDetail, - editKey: action.editKey, - isBookmarkHanger: action.isHanger - })) - break + windowState = windowState.setIn(['bookmarkDetail'], Immutable.fromJS({ + siteDetail: siteDetail, + editKey: action.editKey, + isBookmarkHanger: action.isHanger + })) + break + } case windowConstants.WINDOW_ON_BOOKMARK_ADDED: { - let editKey = action.editKey - const site = appStoreRenderer.state.getIn(['sites', editKey]) - let siteDetail = action.siteDetail + let bookmarkDetail = action.bookmarkDetail - if (site) { - siteDetail = site + if (bookmarkDetail == null) { + bookmarkDetail = frameStateUtil.getActiveFrame(windowState) } - if (siteDetail == null) { - siteDetail = frameStateUtil.getActiveFrame(windowState) - } + bookmarkDetail = bookmarkDetail.set('location', UrlUtil.getLocationIfPDF(bookmarkDetail.get('location'))) - siteDetail = siteDetail.set('location', UrlUtil.getLocationIfPDF(siteDetail.get('location'))) + const editKey = siteUtil.getSiteKey(bookmarkDetail) windowState = windowState.setIn(['bookmarkDetail'], Immutable.fromJS({ - siteDetail: siteDetail, + siteDetail: bookmarkDetail, editKey: editKey, isBookmarkHanger: action.isHanger, isAdded: true })) } break + case windowConstants.WINDOW_ON_ADD_BOOKMARK_FOLDER: + windowState = windowState.setIn(['bookmarkFolderDetail'], Immutable.fromJS({ + folderDetails: action.folderDetails, + closestKey: action.closestKey + })) + break + case windowConstants.WINDOW_ON_EDIT_BOOKMARK_FOLDER: + { + const folderDetails = bookmarkFoldersState.getFolder(appStoreRenderer.state, action.editKey) + + windowState = windowState.setIn(['bookmarkFolderDetail'], Immutable.fromJS({ + folderDetails: folderDetails, + editKey: action.editKey + })) + break + } + case windowConstants.WINDOW_ON_BOOKMARK_FOLDER_CLOSE: + windowState = windowState.delete('bookmarkFolderDetail') + break case windowConstants.WINDOW_AUTOFILL_SELECTION_CLICKED: ipc.send('autofill-selection-clicked', action.tabId, action.value, action.frontEndId, action.index) windowState = windowState.delete('contextMenuDetail') diff --git a/test/about/bookmarksManagerTest.js b/test/about/bookmarksManagerTest.js index 3cecb0dd755..5b4a91ab5d4 100644 --- a/test/about/bookmarksManagerTest.js +++ b/test/about/bookmarksManagerTest.js @@ -10,7 +10,6 @@ const Immutable = require('immutable') describe('about:bookmarks', function () { const folderId = Math.random() const lastVisit = 1476140184441 - const bookmarkTag = [siteTags.BOOKMARK] const browseableSiteUrl = 'page1.html' const browseableSiteTitle = 'Page 1' @@ -30,36 +29,30 @@ describe('about:bookmarks', function () { location: siteWithFavicon, title: 'Page with Favicon', favicon: favicon, - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, { location: siteWithoutFavicon, title: 'Page without Favicon', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit } ]) yield client - .addSiteList(sites) + .addBookmarks(sites) .tabByIndex(0) .loadUrl(aboutBookmarksUrl) } function * addDemoSites (client) { const sites = Immutable.fromJS([ - { - customTitle: 'demo1', - folderId: folderId, - parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, { location: 'https://brave.com', title: 'Brave', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, @@ -67,7 +60,7 @@ describe('about:bookmarks', function () { location: 'https://brave.com/test', title: 'Test', customTitle: 'customTest', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, @@ -75,41 +68,41 @@ describe('about:bookmarks', function () { location: 'https://brave.com/test', title: 'Test', customTitle: 'customTest', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, { location: 'https://www.youtube.com', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, { location: 'https://www.facebook.com', title: 'facebook', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit }, { location: 'https://duckduckgo.com', title: 'duckduckgo', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: folderId, lastAccessedTime: lastVisit }, { location: 'https://google.com', title: 'Google', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: folderId, lastAccessedTime: lastVisit }, { location: 'https://bing.com', title: 'Bing', - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: folderId, lastAccessedTime: lastVisit } @@ -117,7 +110,13 @@ describe('about:bookmarks', function () { yield client .waitForBrowserWindow() - .addSiteList(sites) + .addBookmarkFolder({ + customTitle: 'demo1', + folderId: folderId, + parentFolderId: 0, + type: siteTags.BOOKMARK_FOLDER + }) + .addBookmarks(sites) .tabByIndex(0) .loadUrl(aboutBookmarksUrl) } @@ -126,13 +125,13 @@ describe('about:bookmarks', function () { const site = Brave.server.url(browseableSiteUrl) yield client .waitForBrowserWindow() - .addSite({ + .addBookmark({ location: site, title: browseableSiteTitle, - tags: bookmarkTag, + type: siteTags.BOOKMARK, parentFolderId: 0, lastAccessedTime: lastVisit - }, siteTags.BOOKMARK) + }) .tabByIndex(0) .loadUrl(aboutBookmarksUrl) } diff --git a/test/about/historyTest.js b/test/about/historyTest.js index 8045311ec9e..56ca569e254 100644 --- a/test/about/historyTest.js +++ b/test/about/historyTest.js @@ -19,10 +19,10 @@ describe('about:history', function () { function * addDemoSites (client) { yield client - .addSite({ location: 'https://brave.com', title: 'Brave' }) - .addSite({ location: 'https://brave.com/test', customTitle: 'customTest' }) - .addSite({ location: 'https://www.youtube.com' }) - .addSite({ location: 'https://www.facebook.com' }) + .addHistorySite({ location: 'https://brave.com', title: 'Brave' }) + .addHistorySite({ location: 'https://brave.com/test', customTitle: 'customTest' }) + .addHistorySite({ location: 'https://www.youtube.com' }) + .addHistorySite({ location: 'https://www.facebook.com' }) .waitForExist('[data-test-id="tab"][data-frame-key="1"]') .tabByIndex(0) .url(aboutHistoryUrl) @@ -31,7 +31,7 @@ describe('about:history', function () { function * addBrowseableSite (client) { const site = Brave.server.url(browseableSiteUrl) yield client - .addSite({ + .addHistorySite({ location: site, title: 'Page 1' }) diff --git a/test/about/ledgerPanelTest.js b/test/about/ledgerPanelTest.js index 9788d68db38..0a85f86a712 100644 --- a/test/about/ledgerPanelTest.js +++ b/test/about/ledgerPanelTest.js @@ -154,11 +154,11 @@ describe('Regular payment panel tests', function () { .tabByIndex(0) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -197,11 +197,11 @@ describe('Regular payment panel tests', function () { .tabByIndex(0) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -240,11 +240,11 @@ describe('Regular payment panel tests', function () { .tabByIndex(0) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -296,11 +296,11 @@ describe('Regular payment panel tests', function () { yield this.app.client .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -326,11 +326,11 @@ describe('Regular payment panel tests', function () { .tabByIndex(0) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -354,14 +354,14 @@ describe('Regular payment panel tests', function () { .tabByIndex(0) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .windowByUrl(Brave.browserWindowUrl) .changeSetting(settings.AUTO_SUGGEST_SITES, false) .tabByIndex(0) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) @@ -419,15 +419,15 @@ describe('synopsis', function () { yield this.app.client .loadUrl(site3) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site3, false) + .waitForHistoryEntry(site3, false) .tabByUrl(site3) .loadUrl(site1) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site1) + .waitForHistoryEntry(site1) .tabByUrl(site1) .loadUrl(site2) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site2) + .waitForHistoryEntry(site2) .tabByUrl(site2) .loadUrl(prefsUrl) .waitForVisible(paymentsTab) diff --git a/test/about/ledgerTableTest.js b/test/about/ledgerTableTest.js index 022d7467c01..aa064879123 100644 --- a/test/about/ledgerTableTest.js +++ b/test/about/ledgerTableTest.js @@ -56,7 +56,7 @@ function * before (client, siteList) { .tabByIndex(0) .loadUrl(site) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(site, false) + .waitForHistoryEntry(site, false) .tabByUrl(site) } diff --git a/test/about/newTabTest.js b/test/about/newTabTest.js index 4ec019c0bba..bc7283ed421 100644 --- a/test/about/newTabTest.js +++ b/test/about/newTabTest.js @@ -42,23 +42,23 @@ describe('about:newtab tests', function () { function * addDemoAboutPages (client) { yield client - .addSite({ location: 'about:about' }) - .addSite({ location: 'about:adblock' }) - .addSite({ location: 'about:autofill' }) - .addSite({ location: 'about:blank' }) - .addSite({ location: 'about:bookmarks' }) - .addSite({ location: 'about:brave' }) - .addSite({ location: 'about:certerror' }) - .addSite({ location: 'about:config' }) - .addSite({ location: 'about:downloads' }) - .addSite({ location: 'about:error' }) - .addSite({ location: 'about:extensions' }) - .addSite({ location: 'about:history' }) - .addSite({ location: 'about:newtab' }) - .addSite({ location: 'about:passwords' }) - .addSite({ location: 'about:preferences' }) - .addSite({ location: 'about:safebrowsing' }) - .addSite({ location: 'about:styles' }) + .addHistorySite({ location: 'about:about' }) + .addHistorySite({ location: 'about:adblock' }) + .addHistorySite({ location: 'about:autofill' }) + .addHistorySite({ location: 'about:blank' }) + .addHistorySite({ location: 'about:bookmarks' }) + .addHistorySite({ location: 'about:brave' }) + .addHistorySite({ location: 'about:certerror' }) + .addHistorySite({ location: 'about:config' }) + .addHistorySite({ location: 'about:downloads' }) + .addHistorySite({ location: 'about:error' }) + .addHistorySite({ location: 'about:extensions' }) + .addHistorySite({ location: 'about:history' }) + .addHistorySite({ location: 'about:newtab' }) + .addHistorySite({ location: 'about:passwords' }) + .addHistorySite({ location: 'about:preferences' }) + .addHistorySite({ location: 'about:safebrowsing' }) + .addHistorySite({ location: 'about:styles' }) .waitForExist('[data-test-id="tab"][data-frame-key="1"]') .tabByIndex(0) .url(aboutNewTabUrl) diff --git a/test/app/sessionStoreTest.js b/test/app/sessionStoreTest.js index e3a384529b4..7f5aa9aeee4 100644 --- a/test/app/sessionStoreTest.js +++ b/test/app/sessionStoreTest.js @@ -6,7 +6,7 @@ const siteTags = require('../../js/constants/siteTags') const settings = require('../../js/constants/settings') describe('sessionStore test', function () { - function * setup (client) { + function * setup () { Brave.addCommands() } @@ -16,7 +16,8 @@ describe('sessionStore test', function () { const page1Url = Brave.server.url('page1.html') const site = { location: page1Url, - title: 'some page' + title: 'some page', + type: siteTags.BOOKMARK } yield Brave.startApp() yield setup(Brave.app.client) @@ -28,7 +29,7 @@ describe('sessionStore test', function () { .windowParentByUrl(page1Url) .activateURLMode() .waitForExist(navigatorNotBookmarked) - .addBookmark(site, siteTags.BOOKMARK) + .addBookmark(site) .waitForExist(navigatorBookmarked) yield Brave.stopApp(false) yield Brave.startApp() @@ -62,7 +63,8 @@ describe('sessionStore test', function () { const page1Url = Brave.server.url('page1.html') const site = { location: page1Url, - title: 'some page' + title: 'some page', + type: siteTags.BOOKMARK } yield Brave.startApp() yield setup(Brave.app.client) @@ -74,7 +76,7 @@ describe('sessionStore test', function () { .windowParentByUrl(page1Url) .activateURLMode() .waitForExist(navigatorNotBookmarked) - .addBookmark(site, siteTags.BOOKMARK) + .addBookmark(site) .waitForExist(navigatorBookmarked) yield Brave.stopApp(false) diff --git a/test/bookmark-components/bookmarksTest.js b/test/bookmark-components/bookmarksTest.js index 60859a6aef8..fe75a7ebe48 100644 --- a/test/bookmark-components/bookmarksTest.js +++ b/test/bookmark-components/bookmarksTest.js @@ -1,7 +1,6 @@ /* global describe, it, before, beforeEach */ const Brave = require('../lib/brave') -const Immutable = require('immutable') const {homepageInput, urlInput, navigator, navigatorBookmarked, navigatorNotBookmarked, doneButton, removeButton, bookmarkNameInput, bookmarkLocationInput} = require('../lib/selectors') const siteTags = require('../../js/constants/siteTags') @@ -87,7 +86,7 @@ describe('bookmark tests', function () { .windowByUrl(Brave.browserWindowUrl) .waitUntil(function () { return this.getAppState().then((val) => { - return val.value.sites['https://www.brave.xn--com-8cd/|0|0'].customTitle === 'https://www.brave.xn--com-8cd/' + return val.value.sites['https://www.brave.xn--com-8cd/|0|0'].title === 'https://www.brave.xn--com-8cd/' }) }) }) @@ -303,12 +302,12 @@ describe('bookmark tests', function () { this.page2Url = Brave.server.url('page2.html') yield setup(this.app.client) yield this.app.client - .addSite({ + .addBookmark({ location: this.page1Url, folderId: 1, parentFolderId: 0, - tags: [siteTags.BOOKMARK] - }, siteTags.BOOKMARK) + type: siteTags.BOOKMARK + }) }) it('on new active tabs', function * () { @@ -332,9 +331,18 @@ describe('bookmark tests', function () { }) describe('menu behavior', function () { + // Skip this test if we are not on Windows, + // we only generate menu on windows + const isWindows = process.platform === 'win32' + Brave.beforeAll(this) before(function * () { + if (!isWindows) { + this.skip() + return + } + yield setup(this.app.client) }) @@ -342,12 +350,12 @@ describe('bookmark tests', function () { const folderName = 'bookmark-folder-rebuild-menu-demo' yield this.app.client - .addSite({ + .addBookmarkFolder({ customTitle: folderName, folderId: 1, parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, siteTags.BOOKMARK_FOLDER) + type: siteTags.BOOKMARK_FOLDER + }) .waitUntil(function () { return this.getAppState().then((val) => { const bookmarksMenu = val.value.menu.template.find((item) => { @@ -368,12 +376,12 @@ describe('bookmark tests', function () { const bookmarkTitle = 'bookmark-rebuild-menu-demo' yield this.app.client - .addSite({ + .addBookmark({ lastAccessedTime: 456, - tags: [siteTags.BOOKMARK], + type: siteTags.BOOKMARK, location: 'https://brave.com', title: bookmarkTitle - }, siteTags.BOOKMARK) + }) .waitUntil(function () { return this.getAppState().then((val) => { const bookmarksMenu = val.value.menu.template.find((item) => { @@ -393,22 +401,19 @@ describe('bookmark tests', function () { it('rebuilds the menu when add a list of items', function * () { const bookmarkTitle = 'bookmark-rebuild-menu-demo' const folderName = 'bookmark-folder-rebuild-menu-demo' - const sites = Immutable.fromJS([ - { + yield this.app.client + .addBookmarkFolder({ customTitle: folderName, folderId: 1, parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, - { + type: siteTags.BOOKMARK_FOLDER + }) + .addBookmark({ lastAccessedTime: 123, title: bookmarkTitle, location: 'https://brave.com', - tags: [siteTags.BOOKMARK] - } - ]) - yield this.app.client - .addSiteList(sites) + type: siteTags.BOOKMARK + }) .waitForBrowserWindow() .waitUntil(function () { return this.getAppState().then((val) => { diff --git a/test/bookmark-components/bookmarksToolbarTest.js b/test/bookmark-components/bookmarksToolbarTest.js index f632a03d6bf..759f7a30e6f 100644 --- a/test/bookmark-components/bookmarksToolbarTest.js +++ b/test/bookmark-components/bookmarksToolbarTest.js @@ -57,12 +57,12 @@ describe('bookmarksToolbar', function () { yield this.app.client .changeSetting(settings.SHOW_BOOKMARKS_TOOLBAR, true) .waitForVisible(bookmarksToolbar) - .addSite({ + .addBookmarkFolder({ customTitle: 'demo1', folderId: Math.random(), parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, siteTags.BOOKMARK_FOLDER) + type: siteTags.BOOKMARK_FOLDER + }) .waitForVisible('[data-test-id="bookmarkToolbarButton"][title=demo1]') .click(bookmarksToolbar) .click('[data-test-id="bookmarkToolbarButton"][title=demo1]') @@ -78,23 +78,23 @@ describe('bookmarksToolbar', function () { yield this.app.client .changeSetting(settings.SHOW_BOOKMARKS_TOOLBAR, true) .waitForVisible(bookmarksToolbar) - .addSite({ + .addBookmarkFolder({ customTitle: 'demo1', folderId: folderId1, parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, siteTags.BOOKMARK_FOLDER) + type: siteTags.BOOKMARK_FOLDER + }) .waitUntil(function () { return this.getAppState().then((val) => { return findBookmarkFolder('demo1', val) }) }) - .addSite({ + .addBookmarkFolder({ customTitle: 'demo2', folderId: folderId2, parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, siteTags.BOOKMARK_FOLDER) + type: siteTags.BOOKMARK_FOLDER + }) .waitUntil(function () { return this.getAppState().then((val) => { return findBookmarkFolder('demo2', val) @@ -103,7 +103,7 @@ describe('bookmarksToolbar', function () { .waitForUrl(Brave.newTabUrl) .loadUrl(this.page1Url) .windowParentByUrl(this.page1Url) - .waitForSiteEntry(this.page1Url) + .waitForHistoryEntry(this.page1Url) .waitForVisible(navigator) .activateURLMode() .waitForVisible(navigatorNotBookmarked) @@ -123,12 +123,12 @@ describe('bookmarksToolbar', function () { yield this.app.client .changeSetting(settings.SHOW_BOOKMARKS_TOOLBAR, true) .waitForVisible(bookmarksToolbar) - .addSite({ + .addBookmarkFolder({ customTitle: 'demo1', folderId: Math.random(), parentFolderId: 0, - tags: [siteTags.BOOKMARK_FOLDER] - }, siteTags.BOOKMARK_FOLDER) + type: siteTags.BOOKMARK_FOLDER + }) .waitUntil(function () { return this.getAppState().then((val) => { return findBookmarkFolder('demo1', val) @@ -137,7 +137,7 @@ describe('bookmarksToolbar', function () { .waitForUrl(Brave.newTabUrl) .loadUrl(this.page1Url) .windowParentByUrl(this.page1Url) - .waitForSiteEntry(this.page1Url) + .waitForHistoryEntry(this.page1Url) .waitForVisible(navigator) .activateURLMode() .waitForVisible(navigatorNotBookmarked) @@ -172,7 +172,7 @@ describe('bookmarksToolbar', function () { .waitForUrl(Brave.newTabUrl) .loadUrl(pageWithFavicon) .windowParentByUrl(pageWithFavicon) - .waitForSiteEntry(pageWithFavicon, false) + .waitForHistoryEntry(pageWithFavicon, false) .activateURLMode() .waitForVisible(navigatorNotBookmarked) .click(navigatorNotBookmarked) @@ -197,7 +197,7 @@ describe('bookmarksToolbar', function () { .waitForUrl(Brave.newTabUrl) .loadUrl(pageWithoutFavicon) .windowParentByUrl(pageWithoutFavicon) - .waitForSiteEntry(pageWithoutFavicon, false) + .waitForHistoryEntry(pageWithoutFavicon, false) .activateURLMode() .waitForVisible(navigatorNotBookmarked) .click(navigatorNotBookmarked) diff --git a/test/lib/brave.js b/test/lib/brave.js index f7f7eff2d64..15e981c205d 100644 --- a/test/lib/brave.js +++ b/test/lib/brave.js @@ -452,19 +452,6 @@ var exports = { }, 5000, null, 100) }) - this.app.client.addCommand('waitForSiteEntry', function (location, waitForTitle = true) { - logVerbose('waitForSiteEntry("' + location + '", "' + waitForTitle + '")') - return this.waitUntil(function () { - return this.getAppState().then((val) => { - const ret = val.value && val.value.sites && Array.from(Object.values(val.value.sites)).find( - (site) => site.location === location && - (!waitForTitle || (waitForTitle && site.title))) - logVerbose('waitForSiteEntry("' + location + ', ' + waitForTitle + '") => ' + ret) - return ret - }) - }, 5000, null, 100) - }) - this.app.client.addCommand('waitForAddressEntry', function (location, waitForTitle = true) { logVerbose('waitForAddressEntry("' + location + '", "' + waitForTitle + '")') return this.waitUntil(function () { @@ -473,7 +460,7 @@ var exports = { (site) => site.location === location && (!waitForTitle || (waitForTitle && site.title))) logVerbose('sites:' + JSON.stringify(val.value.sites)) - logVerbose('waitForSiteEntry("' + location + '", ' + waitForTitle + ') => ' + ret) + logVerbose('waitForAddressEntry("' + location + '", ' + waitForTitle + ') => ' + ret) return ret }) }, 5000, null, 100) @@ -672,51 +659,114 @@ var exports = { }) /** - * Adds a site to the sites list, such as a bookmarks. + * Adds a bookmark * * @param {object} siteDetail - Properties for the siteDetail to add - * @param {string} tag - A site tag from js/constants/siteTags.js */ - this.app.client.addCommand('addSite', function (siteDetail, tag) { - logVerbose('addSite("' + siteDetail + '", "' + tag + '")') + this.app.client.addCommand('addBookmark', function (siteDetail) { + logVerbose('addBookmark("' + siteDetail + '")') let waitUrl = siteDetail.location if (isSourceAboutUrl(waitUrl)) { waitUrl = getTargetAboutUrl(waitUrl) } - return this.execute(function (siteDetail, tag) { - return devTools('appActions').addSite(siteDetail, tag) - }, siteDetail, tag).then((response) => response.value) - .waitForSiteEntry(waitUrl, false) + return this.execute(function (siteDetail) { + return devTools('appActions').addBookmark(siteDetail) + }, siteDetail).then((response) => response.value) + .waitForBookmarkEntry(waitUrl, false) + }) + + this.app.client.addCommand('waitForBookmarkEntry', function (location, waitForTitle = true) { + logVerbose('waitForBookmarkEntry("' + location + '", "' + waitForTitle + '")') + return this.waitUntil(function () { + return this.getAppState().then((val) => { + const ret = val.value && val.value.bookmarks && Array.from(Object.values(val.value.bookmarks)).find( + (bookmark) => bookmark.location === location && + (!waitForTitle || (waitForTitle && bookmark.title))) + logVerbose('waitForBookmarkEntry("' + location + ', ' + waitForTitle + '") => ' + ret) + return ret + }) + }, 5000, null, 100) }) /** - * Adds a bookmark to the bookmarks list. + * Adds a history site * * @param {object} siteDetail - Properties for the siteDetail to add - * @param {string} tag - A site tag from js/constants/siteTags.js */ - this.app.client.addCommand('addBookmark', function (siteDetail, tag) { - logVerbose('addBookmark("' + siteDetail + '", "' + tag + '")') + this.app.client.addCommand('addHistorySite', function (siteDetail) { + logVerbose('addHistorySite("' + siteDetail + '")') let waitUrl = siteDetail.location if (isSourceAboutUrl(waitUrl)) { waitUrl = getTargetAboutUrl(waitUrl) } - return this.execute(function (siteDetail, tag) { - return devTools('appActions').addBookmark(siteDetail, tag) - }, siteDetail, tag).then((response) => response.value) - .waitForSiteEntry(waitUrl, false) + return this.execute(function (siteDetail) { + return devTools('appActions').addHistorySite(siteDetail) + }, siteDetail).then((response) => response.value) + .waitForHistoryEntry(waitUrl, false) + }) + + this.app.client.addCommand('waitForHistoryEntry', function (location, waitForTitle = true) { + logVerbose('waitForHistoryEntry("' + location + '", "' + waitForTitle + '")') + return this.waitUntil(function () { + return this.getAppState().then((val) => { + const ret = val.value && val.value.historySites && Array.from(Object.values(val.value.historySites)).find( + (site) => site.location === location && + (!waitForTitle || (waitForTitle && site.title))) + logVerbose('waitForHistoryEntry("' + location + ', ' + waitForTitle + '") => ' + ret) + return ret + }) + }, 5000, null, 100) }) /** - * Adds a list sites to the sites list, including bookmark and foler. + * Adds a bookmark folder * * @param {object} siteDetail - Properties for the siteDetail to add */ - this.app.client.addCommand('addSiteList', function (siteDetail) { - logVerbose('addSiteList("' + siteDetail + '")') + this.app.client.addCommand('addBookmarkFolder', function (siteDetail) { + logVerbose('addBookmarkFolder("' + siteDetail + '")') return this.execute(function (siteDetail) { - return devTools('appActions').addSite(siteDetail) + return devTools('appActions').addBookmarkFolder(siteDetail) }, siteDetail).then((response) => response.value) + .waitForBookmarkFolderEntry(siteDetail.folderId, false) + }) + + this.app.client.addCommand('waitForBookmarkFolderEntry', function (folderId, waitForTitle = true) { + logVerbose('waitForBookmarkFolderEntry("' + folderId + '", "' + waitForTitle + '")') + return this.waitUntil(function () { + return this.getAppState().then((val) => { + const ret = val.value && val.value.bookmarkFolders && Array.from(Object.values(val.value.bookmarkFolders)).find( + (folder) => folder.folderId === folderId && + (!waitForTitle || (waitForTitle && folder.title))) + logVerbose('waitForBookmarkFolderEntry("' + folderId + ', ' + waitForTitle + '") => ' + ret) + return ret + }) + }, 5000, null, 100) + }) + + /** + * Adds a list of bookmarks + * + * @param {object} bookmarkList - List of bookmarks to add + */ + // TODO fix + this.app.client.addCommand('addBookmarks', function (bookmarkList) { + logVerbose('addBookmarks("' + bookmarkList + '")') + return this.execute(function (bookmarkList) { + return devTools('appActions').addBookmark(bookmarkList) + }, bookmarkList).then((response) => response.value) + }) + + /** + * Adds a list of history sites + * + * @param {object} historyList - List of history sites to add + */ + this.app.client.addCommand('addHistorySites', function (historyList) { + logVerbose('addHistorySites("' + historyList + '")') + return this.execute(function (historyList) { + return devTools('appActions').addHistorySites(historyList) + }, historyList).then((response) => response.value) }) /** diff --git a/test/misc-components/syncTest.js b/test/misc-components/syncTest.js index 7f81c8e710b..af375e6e711 100644 --- a/test/misc-components/syncTest.js +++ b/test/misc-components/syncTest.js @@ -673,7 +673,7 @@ describe('Syncing history', function () { .tabByIndex(0) .loadUrl(this.page1Url) .windowParentByUrl(this.page1Url) - .waitForSiteEntry(this.page1Url) + .waitForHistoryEntry(this.page1Url) // For order: Visit page 2 yield Brave.app.client diff --git a/test/navbar-components/navigationBarTest.js b/test/navbar-components/navigationBarTest.js index 48e63e356bc..d54749830f8 100644 --- a/test/navbar-components/navigationBarTest.js +++ b/test/navbar-components/navigationBarTest.js @@ -1181,8 +1181,8 @@ describe('navigationBar tests', function () { .waitForExist(urlInput) .waitForElementFocus(urlInput) .waitForInputText(urlInput, '') - .addSite({ location: 'https://brave.com', title: 'Brave' }) - .waitForSiteEntry('https://brave.com') + .addHistorySite({ location: 'https://brave.com', title: 'Brave' }) + .waitForHistoryEntry('https://brave.com') .keys('br') yield selectsText(this.app.client, 'ave.com') yield blur(this.app.client) @@ -1261,7 +1261,7 @@ describe('navigationBar tests', function () { .url(page1Url) .waitForUrl(page1Url) .windowParentByUrl(page1Url) - .waitForSiteEntry(page1Url) + .waitForHistoryEntry(page1Url) .activateURLMode() .waitForExist(navigatorNotBookmarked) .click(navigatorNotBookmarked) diff --git a/test/navbar-components/urlBarSuggestionsTest.js b/test/navbar-components/urlBarSuggestionsTest.js index 52fa5cee56d..b8cfc3c9e27 100644 --- a/test/navbar-components/urlBarSuggestionsTest.js +++ b/test/navbar-components/urlBarSuggestionsTest.js @@ -24,11 +24,11 @@ describe('urlBarSuggestions', function () { .tabByIndex(0) .loadUrl(this.page1Url) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(this.page1Url) + .waitForHistoryEntry(this.page1Url) .tabByIndex(0) .loadUrl(this.page2Url) .windowByUrl(Brave.browserWindowUrl) - .waitForSiteEntry(this.page2Url) + .waitForHistoryEntry(this.page2Url) .newTab() .waitForUrl(Brave.newTabUrl) .windowByUrl(Brave.browserWindowUrl) diff --git a/test/navbar-components/urlBarTest.js b/test/navbar-components/urlBarTest.js index 6206086dff7..fb840bfe1ec 100644 --- a/test/navbar-components/urlBarTest.js +++ b/test/navbar-components/urlBarTest.js @@ -29,7 +29,7 @@ describe('urlBar tests', function () { yield this.app.client.waitForElementFocus(urlInput) yield this.app.client .onClearBrowsingData('browserHistory', true) - .addSite({ location: 'https://brave.com', title: 'Brave' }) + .addHistorySite({ location: 'https://brave.com', title: 'Brave' }) }) // OMG, Brad would hate this test! @@ -65,10 +65,10 @@ describe('urlBar tests', function () { yield this.app.client .onClearBrowsingData('browserHistory', true) - .addSite({ location: 'https://brave.com', title: 'Brave' }) - .addSite({ location: 'https://brave.com/test' }) - .addSite({ location: 'https://www.youtube.com' }) - .addSite({ location: 'http://uncrate.com' }) + .addHistorySite({ location: 'https://brave.com', title: 'Brave' }) + .addHistorySite({ location: 'https://brave.com/test' }) + .addHistorySite({ location: 'https://www.youtube.com' }) + .addHistorySite({ location: 'http://uncrate.com' }) }) describe('press backspace key', function () { @@ -150,7 +150,7 @@ describe('urlBar tests', function () { describe('typing prefix with characters from "https" prefix', function () { beforeEach(function * () { yield this.app.client - .addSite({ location: 'https://slo-tech.com', title: 'title' }) + .addHistorySite({ location: 'https://slo-tech.com', title: 'title' }) .setInputText(urlInput, '') .keys('s') }) @@ -163,7 +163,7 @@ describe('urlBar tests', function () { describe('typing with characters that do not match prefix should not select first item', function () { beforeEach(function * () { yield this.app.client - .addSite({ location: 'https://slo-tech.com', title: 'title' }) + .addHistorySite({ location: 'https://slo-tech.com', title: 'title' }) .setInputText(urlInput, '') .keys('o') }) @@ -176,7 +176,7 @@ describe('urlBar tests', function () { describe('typing with characters that do not match prefix should not select first item', function () { beforeEach(function * () { yield this.app.client - .addSite({ location: 'https://slo-tech.com', title: 'title' }) + .addHistorySite({ location: 'https://slo-tech.com', title: 'title' }) .setInputText(urlInput, '') .keys('o') }) @@ -378,12 +378,12 @@ describe('urlBar tests', function () { it('typing in the urlbar should override mouse hover for suggestions', function * () { yield this.app.client - .addSite({ location: 'https://brave.com', title: 'Brave' }) - .addSite({ location: 'https://brave.com/test' }) - .addSite({ location: 'https://brave.com/test2' }) - .addSite({ location: 'https://brave.com/test3' }) - .addSite({ location: 'https://brave.com/test4' }) - .addSite({ location: 'https://brianbondy.com/test4' }) + .addHistorySite({ location: 'https://brave.com', title: 'Brave' }) + .addHistorySite({ location: 'https://brave.com/test' }) + .addHistorySite({ location: 'https://brave.com/test2' }) + .addHistorySite({ location: 'https://brave.com/test3' }) + .addHistorySite({ location: 'https://brave.com/test4' }) + .addHistorySite({ location: 'https://brianbondy.com/test4' }) .resizeWindow(500, 300) .setValue(urlInput, 'b') .waitForVisible(urlBarSuggestions) @@ -688,7 +688,7 @@ describe('urlBar tests', function () { yield this.app.client.waitForExist(urlInput) yield this.app.client.waitForElementFocus(urlInput) yield this.app.client - .addSiteList(sites) + .addHistorySites(sites) .waitForInputText(urlInput, '') .windowByUrl(Brave.browserWindowUrl) .newTab() @@ -763,8 +763,8 @@ describe('urlBar tests', function () { yield this.app.client.waitForElementFocus(urlInput) yield this.app.client .onClearBrowsingData('browserHistory', true) - .addSite({ location: 'https://github.com/brave/browser-laptop', title: 'browser-laptop' }) - .addSite({ location: 'https://github.com/brave/ad-block', title: 'Muon' }) + .addHistorySite({ location: 'https://github.com/brave/browser-laptop', title: 'browser-laptop' }) + .addHistorySite({ location: 'https://github.com/brave/ad-block', title: 'Muon' }) }) it('changes only the selection', function * () { diff --git a/test/unit/app/browser/reducers/sitesReducerTest.js b/test/unit/app/browser/reducers/sitesReducerTest.js index 15d5bc560c7..4b400586b01 100644 --- a/test/unit/app/browser/reducers/sitesReducerTest.js +++ b/test/unit/app/browser/reducers/sitesReducerTest.js @@ -9,6 +9,8 @@ const siteTags = require('../../../../../js/constants/siteTags') const { makeImmutable } = require('../../../../../app/common/state/immutableUtil') require('../../../braveUnit') +// TODO reuse what you can + const initState = Immutable.fromJS({ sites: {}, windows: [], diff --git a/test/unit/app/common/state/siteCacheTest.js b/test/unit/app/common/cache/bookmarkLocationCacheTest.js similarity index 97% rename from test/unit/app/common/state/siteCacheTest.js rename to test/unit/app/common/cache/bookmarkLocationCacheTest.js index 8d92a72a32b..ac46ebc4dff 100644 --- a/test/unit/app/common/state/siteCacheTest.js +++ b/test/unit/app/common/cache/bookmarkLocationCacheTest.js @@ -1,12 +1,12 @@ /* global describe, it */ const siteTags = require('../../../../../js/constants/siteTags') -const siteCache = require('../../../../../app/common/state/siteCache') +const siteCache = require('../../../../../app/common/cache/bookmarkLocationCache') const siteUtil = require('../../../../../js/state/siteUtil') const assert = require('assert') const Immutable = require('immutable') -describe('siteCache', function () { +describe('bookmarkLocationCache unit test', function () { const testUrl1 = 'https://brave.com/' const testUrl2 = 'http://example.com/' const bookmark = Immutable.fromJS({ diff --git a/test/unit/app/common/lib/pinnedSitesUtilTest.js b/test/unit/app/common/lib/pinnedSitesUtilTest.js new file mode 100644 index 00000000000..e2ef0ffeedf --- /dev/null +++ b/test/unit/app/common/lib/pinnedSitesUtilTest.js @@ -0,0 +1,39 @@ +/* global describe, it */ +const pinnedSitesUtil = require('../../../../../app/common/lib/pinnedSitesUtil') +const assert = require('assert') +const Immutable = require('immutable') + +require('../../../braveUnit') + +describe('pinnedSitesUtil', () => { + const location = 'https://css-tricks.com/' + const order = 9 + const partitionNumber = 5 + const expectedSiteProps = Immutable.fromJS({ + location, + order, + partitionNumber + }) + + let site = Immutable.fromJS({ + favicon: 'https://css-tricks.com/favicon.ico', + lastAccessedTime: 1493560182224, + location: location, + order: order, + partitionNumber: partitionNumber, + title: 'CSS-Tricks' + }) + + describe('getPinnedSiteProps', () => { + it('returns object with necessary fields', () => { + const result = pinnedSitesUtil.getPinnedSiteProps(site) + assert.deepEqual(expectedSiteProps, result) + }) + + it('set partitionNumber field to 0 in case of missing this field', () => { + const newSite = site.delete('partitionNumber') + const result = pinnedSitesUtil.getPinnedSiteProps(newSite) + assert.equal(0, result.get('partitionNumber')) + }) + }) +}) diff --git a/test/unit/app/common/lib/windowsUtilTest.js b/test/unit/app/common/lib/windowsUtilTest.js deleted file mode 100644 index 60ece75b82b..00000000000 --- a/test/unit/app/common/lib/windowsUtilTest.js +++ /dev/null @@ -1,40 +0,0 @@ -/* global describe, beforeEach, it */ -const windowsUtil = require('../../../../../app/common/lib/windowsUtil') -const assert = require('assert') -const Immutable = require('immutable') - -require('../../../braveUnit') - -describe('windowsUtil', () => { - const location = 'https://css-tricks.com/' - const order = 9 - const partitionNumber = 5 - const expectedSiteProps = Immutable.fromJS({ - location, - order, - partitionNumber - }) - let site - - describe('getPinnedSiteProps', () => { - beforeEach(() => { - site = Immutable.fromJS({ - favicon: 'https://css-tricks.com/favicon.ico', - lastAccessedTime: 1493560182224, - location: location, - order: order, - partitionNumber: partitionNumber, - title: 'CSS-Tricks' - }) - }) - it('returns object with necessary fields', () => { - const result = windowsUtil.getPinnedSiteProps(site) - assert.deepEqual(expectedSiteProps, result) - }) - it('set partitionNumber field to 0 in case of missing this field', () => { - site = site.delete('partitionNumber') - const result = windowsUtil.getPinnedSiteProps(site) - assert.equal(0, result.get('partitionNumber')) - }) - }) -}) diff --git a/test/unit/state/siteUtilTest.js b/test/unit/state/siteUtilTest.js index f45653f3ca0..0114b9feaf8 100644 --- a/test/unit/state/siteUtilTest.js +++ b/test/unit/state/siteUtilTest.js @@ -1335,67 +1335,6 @@ describe('siteUtil', function () { }) }) - describe('isEquivalent', function () { - it('returns true if both siteDetail objects are identical', function () { - const siteDetail1 = Immutable.fromJS({ - location: testUrl1, - partitionNumber: 0, - tags: [siteTags.BOOKMARK] - }) - const siteDetail2 = Immutable.fromJS({ - location: testUrl1, - partitionNumber: 0, - tags: [siteTags.BOOKMARK] - }) - assert.equal(siteUtil.isEquivalent(siteDetail1, siteDetail2), true) - }) - it('returns false if one object is a folder and the other is not', function () { - const siteDetail1 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK] - }) - const siteDetail2 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK_FOLDER], - folderId: 1 - }) - assert.equal(siteUtil.isEquivalent(siteDetail1, siteDetail2), false) - }) - it('returns false if both are folders and have a different folderId', function () { - const siteDetail1 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK_FOLDER], - folderId: 0 - }) - const siteDetail2 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK_FOLDER], - folderId: 1 - }) - assert.equal(siteUtil.isEquivalent(siteDetail1, siteDetail2), false) - }) - it('returns false if both are bookmarks and have a different location', function () { - const siteDetail1 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK], - location: testUrl1 - }) - const siteDetail2 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK], - location: 'http://example.com/' - }) - assert.equal(siteUtil.isEquivalent(siteDetail1, siteDetail2), false) - }) - it('returns false if both are bookmarks and have a different partitionNumber', function () { - const siteDetail1 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK], - location: testUrl1, - partitionNumber: 0 - }) - const siteDetail2 = Immutable.fromJS({ - tags: [siteTags.BOOKMARK], - location: testUrl2, - partitionNumber: 1 - }) - assert.equal(siteUtil.isEquivalent(siteDetail1, siteDetail2), false) - }) - }) - describe('isFolder', function () { it('returns true if the input is a siteDetail and has a `BOOKMARK_FOLDER` tag and a folder ID', function () { const siteDetail = Immutable.fromJS({ @@ -1683,20 +1622,4 @@ describe('siteUtil', function () { assert.strictEqual(siteUtil.getOrigin('http://http/test'), 'http://http') }) }) - describe('isPinnedTab', function () { - it('detects pinned tab site', function () { - assert.strictEqual(siteUtil.isPinnedTab(siteTags.PINNED), true) - assert.strictEqual(siteUtil.isPinnedTab([siteTags.PINNED]), true) - }) - it('detects not pinned for no site tags', function () { - assert.strictEqual(siteUtil.isPinnedTab([]), false) - }) - it('detects not pinned for site tags which are not PINNED', function () { - assert.strictEqual(siteUtil.isPinnedTab(siteTags.BOOKMARK), false) - assert.strictEqual(siteUtil.isPinnedTab([siteTags.BOOKMARK]), false) - }) - it('detects pinned when bookmarked and pinned', function () { - assert.strictEqual(siteUtil.isPinnedTab([siteTags.PINNED, siteTags.BOOKMARK]), true) - }) - }) })