From 5b147da44cff02be0b5917ce3f405c4db81b856f Mon Sep 17 00:00:00 2001 From: Cezar Augusto Date: Fri, 4 Nov 2016 17:15:03 -0200 Subject: [PATCH] Fix gridLayout pinning action bug Fix #5337 Auditors: @bsclifton @bbondy /cc @luixxiul @srirambv Our topSites grid have actions connected to each other (pinning/removing/reordering), so we must ensure all of them are working properly while doing multiple actions. Test Plan: * Access new tab page; * Access several websites (you can grab this JSON https://github.com/cezaraugusto/brave-json/blob/master/newtab-3-rows.json and copy/paste on session-store-1, it has 18 sites); * Reordering a tile should work on any position; * Pin a site. Pinning a site should add a Pin icon on top-right corner and be kept on its position (not jump); * Keep in mind the website position and reorder it to a new position; * Access another website and get back to new tab. Ensure that your last pinned website kept its position; * Remove that site (pinned or not); * A modal will open. "Undo" removing; * There should be no pinned icon; * Website should be back to its last position; * Pin several websites; * Reorder pinned websites; * Access a new website; * Go back to new tab page: pinned sites should remmember their positions; * Remove a website on removal modal and hit "restore all"; * All sites must be unpinned. --- js/about/newtab.js | 44 ++++++++++++++++++++------------------------ 1 file changed, 20 insertions(+), 24 deletions(-) diff --git a/js/about/newtab.js b/js/about/newtab.js index 8d813c4fbe5..bbdf968fb32 100644 --- a/js/about/newtab.js +++ b/js/about/newtab.js @@ -43,21 +43,10 @@ class NewTabPage extends React.Component { const updatedStamp = sanitizedData.getIn(['newTabDetail', 'updatedStamp']) // Only update if the data has changed. - // console.log(updatedStamp + ']DATA_TS==LAST_TS[' + this.state.updatedStamp) if (updatedStamp === this.state.updatedStamp) { return } - // Ensure top sites only has unique values - const pinnedTopSites = sanitizedData.getIn(['newTabDetail', 'pinnedTopSites']) - if (pinnedTopSites) { - const uniqueValues = pinnedTopSites.filter((element, index, list) => { - if (!element) return false - return index === list.findIndex((site) => site && site.get('location') === element.get('location')) - }) - sanitizedData = sanitizedData.setIn(['newTabDetail', 'pinnedTopSites'], uniqueValues) - } - this.setState({ newTabData: sanitizedData, updatedStamp: updatedStamp @@ -82,7 +71,7 @@ class NewTabPage extends React.Component { } get pinnedTopSites () { - return this.state.newTabData.getIn(['newTabDetail', 'pinnedTopSites']) + return this.state.newTabData.getIn(['newTabDetail', 'pinnedTopSites']).setSize(18) } get ignoredTopSites () { @@ -97,6 +86,10 @@ class NewTabPage extends React.Component { return this.pinnedTopSites.includes(siteProps) } + isIgnored (siteProps) { + return this.ignoredTopSites.includes(siteProps) + } + isBookmarked (siteProps) { return siteUtil.isSiteBookmarked(this.topSites, siteProps) } @@ -106,21 +99,23 @@ class NewTabPage extends React.Component { * in the grid, and the non pinned indexes are populated with newly accessed sites */ get topSites () { - let gridSites = Immutable.List().setSize(18) - let sites = this.sites - const pinnedTopSites = this.pinnedTopSites.setSize(gridSites.size) - const ignoredTopSites = this.ignoredTopSites + const pinnedTopSites = this.pinnedTopSites + const sites = this.sites + let unpinnedTopSites = sites.filter((site) => !this.isPinned(site) ? site : null) + let gridSites + + const getUnpinned = () => { + const firstSite = unpinnedTopSites.first() + unpinnedTopSites = unpinnedTopSites.shift() + return firstSite + } - // We need to know which sites are pinned first, so we can skip them while populating - gridSites = gridSites.push.apply(pinnedTopSites, gridSites) - gridSites = gridSites.map((item, i) => item == null ? sites.get(i) : item) + gridSites = pinnedTopSites.map(pinned => pinned || getUnpinned()) // Remove from grid all ignored sites - gridSites = gridSites.filter((site) => ignoredTopSites.indexOf(site) === -1) - - gridSites = gridSites.filter(site => site != null) + gridSites = gridSites.filter((site) => !this.isIgnored(site)) - return gridSites + return gridSites.filter(site => site != null) } get gridLayout () { @@ -210,7 +205,7 @@ class NewTabPage extends React.Component { const currentPositionIndex = gridSites.indexOf(currentPosition) // If pinned, leave it null. Otherwise stores site on ignoredTopSites list, retaining the same position - let pinnedTopSites = this.pinnedTopSites.setSize(18) + let pinnedTopSites = this.pinnedTopSites pinnedTopSites = pinnedTopSites.splice(currentPositionIndex, 1, this.isPinned(siteProps) ? null : siteProps) aboutActions.setNewTabDetail({pinnedTopSites: pinnedTopSites}) @@ -234,6 +229,7 @@ class NewTabPage extends React.Component { } onUndoIgnoredTopSite () { + // Remove last List's entry const ignoredTopSites = this.ignoredTopSites.splice(-1, 1) aboutActions.setNewTabDetail({ignoredTopSites: ignoredTopSites}) this.hideSiteRemovalNotification()