// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'chrome://personalization/strings.m.js';
import {beginLoadSelectedImageAction, beginSelectImageAction, cancelPreviewWallpaper, DailyRefreshType, DefaultImageSymbol, DisplayableImage, endSelectImageAction, fetchCollections, fetchGooglePhotosAlbum, fetchGooglePhotosAlbums, fetchGooglePhotosEnabled, fetchGooglePhotosPhotos, fetchLocalData, FullscreenPreviewState, getDefaultImageThumbnail, GooglePhotosEnablementState, GooglePhotosPhoto, initializeBackdropData, isDefaultImage, isGooglePhotosPhoto, isWallpaperImage, kDefaultImageSymbol, selectGooglePhotosAlbum, selectWallpaper, setAttributionAction, setDailyRefreshCollectionId, setFullscreenStateAction, setSelectedImageAction, updateDailyRefreshWallpaper, WallpaperLayout, WallpaperObserver, WallpaperType} from 'chrome://personalization/js/personalization_app.js';
import {isNonEmptyFilePath} from 'chrome://resources/ash/common/sea_pen/sea_pen_utils.js';
import {assertNotReached} from 'chrome://resources/js/assert.js';
import {loadTimeData} from 'chrome://resources/js/load_time_data.js';
import {FilePath} from 'chrome://resources/mojo/mojo/public/mojom/base/file_path.mojom-webui.js';
import {assertDeepEquals, assertEquals, assertFalse, assertTrue} from 'chrome://webui-test/chai_assert.js';
import {baseSetup, filterAndFlattenState} from './personalization_app_test_utils.js';
import {TestPersonalizationStore} from './test_personalization_store.js';
import {TestWallpaperProvider} from './test_wallpaper_interface_provider.js';
function getImageKey(image: DisplayableImage): string|undefined {
if (isDefaultImage(image)) {
return undefined;
}
if (isGooglePhotosPhoto(image)) {
return image.dedupKey ? image.dedupKey : image.id;
}
if (isWallpaperImage(image)) {
return image.unitId.toString();
}
if (isNonEmptyFilePath(image)) {
return image.path;
}
assertNotReached('unknown wallpaper type');
}
suite('Personalization app controller', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
});
test('initializes Google Photos data in store', async () => {
loadTimeData.overrideValues({isGooglePhotosIntegrationEnabled: true});
await fetchGooglePhotosEnabled(wallpaperProvider, personalizationStore);
const expectedEnabled = GooglePhotosEnablementState.kEnabled;
assertDeepEquals(
[
{
name: 'begin_load_google_photos_enabled',
},
{
name: 'set_google_photos_enabled',
enabled: expectedEnabled,
},
],
personalizationStore.actions);
assertDeepEquals(
[
// BEGIN_LOAD_GOOGLE_PHOTOS_ENABLED.
{
'wallpaper.loading.googlePhotos': {
enabled: true,
albums: false,
albumsShared: false,
photos: false,
photosByAlbumId: {},
},
'wallpaper.googlePhotos': {
enabled: undefined,
albums: undefined,
albumsShared: undefined,
photos: undefined,
photosByAlbumId: {},
resumeTokens: {
albums: null,
albumsShared: null,
photos: null,
photosByAlbumId: {},
},
},
},
// SET_GOOGLE_PHOTOS_ENABLED.
{
'wallpaper.loading.googlePhotos': {
enabled: false,
albums: false,
albumsShared: false,
photos: false,
photosByAlbumId: {},
},
'wallpaper.googlePhotos': {
enabled: expectedEnabled,
albums: undefined,
albumsShared: undefined,
photos: undefined,
photosByAlbumId: {},
resumeTokens: {
albums: null,
albumsShared: null,
photos: null,
photosByAlbumId: {},
},
},
},
],
personalizationStore.states.map(filterAndFlattenState(
['wallpaper.googlePhotos', 'wallpaper.loading.googlePhotos'])));
});
test('sets Google Photos album in store', async () => {
loadTimeData.overrideValues({isGooglePhotosIntegrationEnabled: true});
const album = {
id: '9bd1d7a3-f995-4445-be47-53c5b58ce1cb',
title: '',
photoCount: 0,
isShared: false,
preview: {url: 'bar.com'},
timestamp: {internalValue: BigInt(0)},
};
const photos: GooglePhotosPhoto[] = [{
id: '9bd1d7a3-f995-4445-be47-53c5b58ce1cb',
dedupKey: '2d0d1595-14af-4471-b2db-b9c8eae3a491',
name: 'foo',
date: {data: []},
url: {url: 'foo.com'},
location: 'home',
}];
wallpaperProvider.setGooglePhotosAlbums([album]);
wallpaperProvider.setGooglePhotosPhotosByAlbumId(album.id, photos);
// Attempts to `fetchGooglePhotosAlbum()` will fail unless the entire list
// of Google Photos albums has already been fetched and saved to the store.
await fetchGooglePhotosEnabled(wallpaperProvider, personalizationStore);
await fetchGooglePhotosAlbums(wallpaperProvider, personalizationStore);
personalizationStore.reset(personalizationStore.data);
await fetchGooglePhotosAlbum(
wallpaperProvider, personalizationStore, album.id);
// The wallpaper controller is expected to impose max resolution.
album.preview.url += '=s512';
photos.forEach(photo => photo.url.url += '=s512');
assertDeepEquals(
[
{
name: 'begin_load_google_photos_album',
albumId: album.id,
},
{
name: 'append_google_photos_album',
albumId: album.id,
photos: photos,
resumeToken: null,
},
],
personalizationStore.actions);
assertDeepEquals(
[
// BEGIN_LOAD_GOOGLE_PHOTOS_ALBUM
{
'wallpaper.loading.googlePhotos': {
enabled: false,
albums: false,
albumsShared: false,
photos: false,
photosByAlbumId: {
[album.id]: true,
},
},
'wallpaper.googlePhotos': {
enabled: GooglePhotosEnablementState.kEnabled,
albums: [
{
id: album.id,
title: album.title,
preview: album.preview,
photoCount: album.photoCount,
isShared: album.isShared,
timestamp: album.timestamp,
},
],
albumsShared: undefined,
photos: undefined,
photosByAlbumId: {},
resumeTokens: {
albums: null,
albumsShared: null,
photos: null,
photosByAlbumId: {},
},
},
},
// APPEND_GOOGLE_PHOTOS_ALBUM
{
'wallpaper.loading.googlePhotos': {
enabled: false,
albums: false,
albumsShared: false,
photos: false,
photosByAlbumId: {
[album.id]: false,
},
},
'wallpaper.googlePhotos': {
enabled: GooglePhotosEnablementState.kEnabled,
albums: [
{
id: album.id,
title: album.title,
photoCount: album.photoCount,
isShared: album.isShared,
preview: album.preview,
timestamp: album.timestamp,
},
],
albumsShared: undefined,
photos: undefined,
photosByAlbumId: {
[album.id]: photos,
},
resumeTokens: {
albums: null,
albumsShared: null,
photos: null,
photosByAlbumId: {[album.id]: null},
},
},
},
],
personalizationStore.states.map(filterAndFlattenState(
['wallpaper.googlePhotos', 'wallpaper.loading.googlePhotos'])));
});
test('sets local images in store', async () => {
await fetchLocalData(wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{name: 'begin_load_local_images'},
{
name: 'set_local_images',
images: [
{path: 'LocalImage0.png'},
{path: 'LocalImage1.png'},
],
},
{name: 'begin_load_local_image_data', id: 'LocalImage0.png'},
{name: 'begin_load_local_image_data', id: 'LocalImage1.png'},
{
name: 'set_local_image_data',
id: 'LocalImage0.png',
data: {url: ''},
},
{
name: 'set_local_image_data',
id: 'LocalImage1.png',
data: {url: ''},
},
],
personalizationStore.actions);
assertEquals(
JSON.stringify([
// Begin loading local image list.
{
'wallpaper.local': {images: null, data: {}},
'wallpaper.loading.local': {images: true, data: {}},
},
// Done loading local image data.
{
'wallpaper.local': {
images: [
{path: 'LocalImage0.png'},
{path: 'LocalImage1.png'},
],
data: {},
},
'wallpaper.loading.local': {data: {}, images: false},
},
// Mark image 0 as loading.
{
'wallpaper.local': {
images: [{path: 'LocalImage0.png'}, {path: 'LocalImage1.png'}],
data: {},
},
'wallpaper.loading.local': {
data: {'LocalImage0.png': true},
images: false,
},
},
// Mark image 1 as loading.
{
'wallpaper.local': {
images: [
{path: 'LocalImage0.png'},
{path: 'LocalImage1.png'},
],
data: {},
},
'wallpaper.loading.local': {
data: {'LocalImage0.png': true, 'LocalImage1.png': true},
images: false,
},
},
// Finish loading image 0.
{
'wallpaper.local': {
images: [
{path: 'LocalImage0.png'},
{path: 'LocalImage1.png'},
],
data: {
'LocalImage0.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
data: {'LocalImage0.png': false, 'LocalImage1.png': true},
images: false,
},
},
// Finish loading image 1.
{
'wallpaper.local': {
images: [
{path: 'LocalImage0.png'},
{path: 'LocalImage1.png'},
],
data: {
'LocalImage0.png': {
url: '',
},
'LocalImage1.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
data: {'LocalImage0.png': false, 'LocalImage1.png': false},
images: false,
},
},
]),
JSON.stringify(personalizationStore.states.map(filterAndFlattenState(
['wallpaper.local', 'wallpaper.loading.local']))));
});
test('subtracts an image from state when it disappears', async () => {
await fetchLocalData(wallpaperProvider, personalizationStore);
// Keep the current state but reset the history of actions and states.
personalizationStore.reset(personalizationStore.data);
// Only keep the first image.
wallpaperProvider.localImages = [wallpaperProvider.localImages![0]!];
await fetchLocalData(wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{name: 'begin_load_local_images'},
{
name: 'set_local_images',
images: [{path: 'LocalImage0.png'}],
},
],
personalizationStore.actions,
);
assertEquals(
JSON.stringify([
// Begin loading new image list.
{
'wallpaper.local': {
images: [{path: 'LocalImage0.png'}, {path: 'LocalImage1.png'}],
data: {
'LocalImage0.png': {
url: '',
},
'LocalImage1.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
data: {'LocalImage0.png': false, 'LocalImage1.png': false},
images: true,
},
},
// Load new image list with only image 0. Deletes image 1 information
// from loading state and thumbnail state.
{
'wallpaper.local': {
images: [{path: 'LocalImage0.png'}],
data: {
'LocalImage0.png': {
url: '',
},
},
},
'wallpaper.loading.local':
{data: {'LocalImage0.png': false}, images: false},
},
]),
JSON.stringify(personalizationStore.states.map(filterAndFlattenState(
['wallpaper.local', 'wallpaper.loading.local']))));
});
test('fetches new images that are added', async () => {
await fetchLocalData(wallpaperProvider, personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
// Subtract image 1 and add NewPath.png.
wallpaperProvider.localImages = [
wallpaperProvider.localImages![0]!,
{path: 'NewPath.png'},
];
wallpaperProvider.localImageData = {
...wallpaperProvider.localImageData,
'NewPath.png': {url: ''},
};
await fetchLocalData(wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{
name: 'begin_load_local_images',
},
{
name: 'set_local_images',
images: [{path: 'LocalImage0.png'}, {path: 'NewPath.png'}],
},
// Only loads data for new image.
{
name: 'begin_load_local_image_data',
id: 'NewPath.png',
},
// Sets data for new image.
{
name: 'set_local_image_data',
id: 'NewPath.png',
data: {url: ''},
},
],
personalizationStore.actions,
JSON.stringify(personalizationStore.actions));
assertEquals(
JSON.stringify([
// Begin loading image list.
{
'wallpaper.local': {
'images': [
{'path': 'LocalImage0.png'},
{'path': 'LocalImage1.png'},
],
'data': {
'LocalImage0.png': {
url: '',
},
'LocalImage1.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
'data': {'LocalImage0.png': false, 'LocalImage1.png': false},
'images': true,
},
},
// Done loading image list.
{
'wallpaper.local': {
'images': [{'path': 'LocalImage0.png'}, {'path': 'NewPath.png'}],
'data': {
'LocalImage0.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
'data': {'LocalImage0.png': false},
'images': false,
},
},
// Begin loading NewPath.png data.
{
'wallpaper.local': {
'images': [{'path': 'LocalImage0.png'}, {'path': 'NewPath.png'}],
'data': {
'LocalImage0.png': {
url: '',
},
},
},
'wallpaper.loading.local': {
'data': {'LocalImage0.png': false, 'NewPath.png': true},
'images': false,
},
},
// Done loading NewPath.png data.
{
'wallpaper.local': {
'images': [{'path': 'LocalImage0.png'}, {'path': 'NewPath.png'}],
'data': {
'LocalImage0.png': {
url: '',
},
'NewPath.png': {url: ''},
},
},
'wallpaper.loading.local': {
'data': {'LocalImage0.png': false, 'NewPath.png': false},
'images': false,
},
},
]),
JSON.stringify(personalizationStore.states.map(filterAndFlattenState(
['wallpaper.local', 'wallpaper.loading.local']))));
});
test('clears local images when fetching new image list fails', async () => {
// No default image on this device.
wallpaperProvider.defaultImageThumbnail = {url: ''};
await getDefaultImageThumbnail(wallpaperProvider, personalizationStore);
await fetchLocalData(wallpaperProvider, personalizationStore);
wallpaperProvider.localImages = null;
await fetchLocalData(wallpaperProvider, personalizationStore);
assertEquals(
null, personalizationStore.data.wallpaper.local.images,
'local images set to null');
assertEquals(
JSON.stringify({}),
JSON.stringify(personalizationStore.data.wallpaper.local.data));
assertDeepEquals(
{url: ''},
personalizationStore.data.wallpaper.local.data[kDefaultImageSymbol],
'default image still present but set to empty url');
assertEquals(
JSON.stringify({}),
JSON.stringify(personalizationStore.data.wallpaper.loading.local.data),
'local images not loading');
assertFalse(
personalizationStore.data.wallpaper.loading.local
.data[kDefaultImageSymbol],
'default image is not loading');
});
test('fails to enable Google Photos daily refresh', async () => {
wallpaperProvider.selectGooglePhotosAlbumResponse = false;
wallpaperProvider.albumId = 'albumId';
wallpaperProvider.collectionId = '';
const selectGooglePhotosAlbumPromise = selectGooglePhotosAlbum(
'albumId', wallpaperProvider, personalizationStore);
await Promise.all([
wallpaperProvider.getDailyRefreshCollectionId(),
wallpaperProvider.getGooglePhotosDailyRefreshAlbumId(),
]);
await selectGooglePhotosAlbumPromise;
assertDeepEquals(
[
{
name: 'begin_update_daily_refresh_image',
},
// Set error action when daily refresh failed.
{
name: 'set_error',
error: {message: loadTimeData.getString('googlePhotosError')},
},
// Set daily refresh enabled for the selected Google Photos album.
{
name: 'set_google_photos_daily_refresh_album_id',
albumId: 'albumId',
},
],
personalizationStore.actions,
JSON.stringify(personalizationStore.actions));
});
test(
'fails to refresh a new wallpaper in a Google Photos album', async () => {
personalizationStore.data.wallpaper.dailyRefresh = {
id: 'abumId',
type: DailyRefreshType.GOOGLE_PHOTOS,
};
wallpaperProvider.updateDailyRefreshWallpaperResponse = false;
await updateDailyRefreshWallpaper(
wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{
name: 'begin_update_daily_refresh_image',
},
{
name: 'begin_load_selected_image',
},
{
name: 'set_updated_daily_refreshed_image',
},
{
name: 'set_attribution',
attribution: personalizationStore.data.wallpaper.attribution,
},
{
name: 'set_selected_image',
image: personalizationStore.data.wallpaper.currentSelected,
},
{
name: 'set_error',
error: {message: loadTimeData.getString('googlePhotosError')},
},
],
personalizationStore.actions);
});
});
suite('full screen mode', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
});
test('enters full screen mode when in tablet', async () => {
await initializeBackdropData(wallpaperProvider, personalizationStore);
assertEquals(
FullscreenPreviewState.OFF,
personalizationStore.data.wallpaper.fullscreen);
wallpaperProvider.isInTabletModeResponse = true;
assertEquals(0, wallpaperProvider.getCallCount('makeTransparent'));
assertEquals(0, wallpaperProvider.getCallCount('makeOpaque'));
const selectWallpaperPromise = selectWallpaper(
wallpaperProvider.images![0]!, wallpaperProvider, personalizationStore);
const [unitId, previewMode] =
await wallpaperProvider.whenCalled('selectWallpaper');
assertTrue(previewMode);
assertEquals(wallpaperProvider.images![0]!.unitId, unitId);
await selectWallpaperPromise;
assertEquals(
1, wallpaperProvider.getCallCount('makeTransparent'),
'makeTransparent is called while calling selectWallpaper');
assertEquals(
FullscreenPreviewState.LOADING,
personalizationStore.data.wallpaper.fullscreen);
// Simulate `wallpaper_observer.ts` signaling that the preview wallpaper has
// been set.
personalizationStore.dispatch(
setFullscreenStateAction(FullscreenPreviewState.VISIBLE));
await cancelPreviewWallpaper(wallpaperProvider);
assertEquals(
1, wallpaperProvider.getCallCount('makeOpaque'),
'makeOpaque is called while calling cancelPreviewWallpaper');
});
});
suite('observes pendingState during wallpaper selection', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
});
test(
'sets pendingState to selected image for successful operation',
async () => {
await selectWallpaper(
wallpaperProvider.images![0]!, wallpaperProvider,
personalizationStore);
assertDeepEquals(
[
beginSelectImageAction(wallpaperProvider.images![0]!),
beginLoadSelectedImageAction(),
setFullscreenStateAction(FullscreenPreviewState.LOADING),
endSelectImageAction(wallpaperProvider.images![0]!, true),
],
personalizationStore.actions, 'expected actions are sent');
assertDeepEquals(
[
// Begin selecting image.
{
'wallpaper.pendingSelected': wallpaperProvider.images![0],
},
// Begin loading image.
{
'wallpaper.pendingSelected': wallpaperProvider.images![0],
},
// End selecting image.
{
'wallpaper.pendingSelected': wallpaperProvider.images![0],
},
// Set fullscreen enabled
{
'wallpaper.pendingSelected': wallpaperProvider.images![0],
},
],
personalizationStore.states.map(
filterAndFlattenState(['wallpaper.pendingSelected'])),
'expected states are observed');
});
test(
'clears pendingState when error occured and only one request',
async () => {
await selectWallpaper(
wallpaperProvider.localImages![0]!, wallpaperProvider,
personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
loadTimeData.overrideValues({['setWallpaperError']: 'someError'});
// sets selected image without file path to force fail the operation.
wallpaperProvider.localImages = [{path: ''}];
await selectWallpaper(
wallpaperProvider.localImages[0]!, wallpaperProvider,
personalizationStore);
assertDeepEquals(
[
beginSelectImageAction(wallpaperProvider.localImages[0]!),
beginLoadSelectedImageAction(),
setFullscreenStateAction(FullscreenPreviewState.LOADING),
endSelectImageAction(wallpaperProvider.localImages[0]!, false),
setFullscreenStateAction(FullscreenPreviewState.OFF),
setAttributionAction(
personalizationStore.data.wallpaper.attribution),
setSelectedImageAction(
personalizationStore.data.wallpaper.currentSelected),
],
personalizationStore.actions, 'sends expected actions');
assertDeepEquals(
[
// Begin selecting image
{
'wallpaper.pendingSelected': wallpaperProvider.localImages[0],
},
// Begin loading image
{
'wallpaper.pendingSelected': wallpaperProvider.localImages[0],
},
// Start full screen
{
'wallpaper.pendingSelected': wallpaperProvider.localImages[0],
},
// End selecting image, pendingState is cleared
{
'wallpaper.pendingSelected': null,
},
// Clear full screen
{
'wallpaper.pendingSelected': null,
},
// Set attribution
{
'wallpaper.pendingSelected': null,
},
// Set selected image
{
'wallpaper.pendingSelected': null,
},
],
personalizationStore.states.map(
filterAndFlattenState(['wallpaper.pendingSelected'])),
'sets expected states');
});
});
suite('local images available but no internet connection', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
});
test('error displays when fetch collections failed', async () => {
// Set collections to null to simulate collections failure.
wallpaperProvider.setCollectionsToFail();
// Assume that collections are loaded before local images.
const collectionsPromise =
fetchCollections(wallpaperProvider, personalizationStore);
await collectionsPromise;
assertFalse(personalizationStore.data.wallpaper.loading.collections);
assertEquals(
null, personalizationStore.data.wallpaper.backdrop.collections);
assertDeepEquals(
[
{
name: 'set_collections',
collections: null,
},
],
personalizationStore.actions,
);
assertDeepEquals(
[
// Set collections.
// Collections are completed loading with null value. Error displays.
{
'error': {message: loadTimeData.getString('wallpaperNetworkError')},
},
],
personalizationStore.states.map(filterAndFlattenState(['error'])));
});
});
suite('does not respond to re-selecting the current wallpaper', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
wallpaperProvider.isInTabletModeResponse = false;
});
function getImageType(image: DisplayableImage): WallpaperType {
if (isDefaultImage(image)) {
return WallpaperType.kDefault;
}
if (isGooglePhotosPhoto(image)) {
return WallpaperType.kOnceGooglePhotos;
}
if (isWallpaperImage(image)) {
return WallpaperType.kOnline;
}
if (isNonEmptyFilePath(image)) {
return WallpaperType.kCustomized;
}
assertNotReached('unknown wallpaper type');
}
// Selects `image` as the wallpaper twice and verifies that the second attempt
// quits early because there is no work to do.
async function testReselectWallpaper(image: DisplayableImage) {
const selectWallpaperActions = [
{
name: 'begin_select_image',
image: image,
},
{
name: 'begin_load_selected_image',
},
{
name: 'end_select_image',
image: image,
success: true,
},
];
// Select a wallpaper and verify that the correct actions are taken.
await selectWallpaper(image, wallpaperProvider, personalizationStore);
assertDeepEquals(personalizationStore.actions, selectWallpaperActions);
// Complete the pending selection as would happen in production code.
const pendingSelected = personalizationStore.data.wallpaper.pendingSelected;
assertEquals(pendingSelected, image);
personalizationStore.data.wallpaper.attribution = {
attribution: [],
key: getImageKey(image)!,
};
personalizationStore.data.wallpaper.currentSelected = {
descriptionContent: '',
descriptionTitle: '',
key: getImageKey(image)!,
layout: WallpaperLayout.kCenterCropped,
type: getImageType(image),
};
personalizationStore.data.wallpaper.pendingSelected = null;
// Select the same wallpaper and verify that no further actions are taken.
await selectWallpaper(image, wallpaperProvider, personalizationStore);
assertDeepEquals(personalizationStore.actions, selectWallpaperActions);
}
test('re-selects online wallpaper', async () => {
await initializeBackdropData(wallpaperProvider, personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
const onlineImages = wallpaperProvider.images;
assertTrue(!!onlineImages && onlineImages.length > 0);
const image = onlineImages[0]!;
await testReselectWallpaper(image);
});
test('re-selects local wallpaper', async () => {
await fetchLocalData(wallpaperProvider, personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
const localImages = personalizationStore.data.wallpaper.local.images;
assertTrue(!!localImages && localImages.length > 0);
const image = localImages[0]!;
await testReselectWallpaper(image);
});
// Check with both |dedupKey| absent and present for backwards compatibility
// with older clients that do not support the latter.
[undefined, '2d0d1595-14af-4471-b2db-b9c8eae3a491'].forEach(
dedupKey => test('re-selects Google Photos wallpaper', async () => {
const image: GooglePhotosPhoto = {
id: '9bd1d7a3-f995-4445-be47-53c5b58ce1cb',
dedupKey: dedupKey === undefined ? null : dedupKey,
name: 'foo',
date: {data: []},
url: {url: 'foo.com'},
location: 'home',
};
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
await testReselectWallpaper(image);
}));
test('re-selects default image', async () => {
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
await testReselectWallpaper(kDefaultImageSymbol);
});
});
suite('updates default image', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(() => {
wallpaperProvider = new TestWallpaperProvider();
personalizationStore = new TestPersonalizationStore({});
personalizationStore.setReducersEnabled(true);
wallpaperProvider.isInTabletModeResponse = false;
});
test('get default image thumbnail', async () => {
// Initialize some local image data.
await fetchLocalData(wallpaperProvider, personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
assertTrue(
Array.isArray(personalizationStore.data.wallpaper.local.images),
'wallpaper.local.images is not array');
assertTrue(
personalizationStore.data.wallpaper.local.images.every(
(image: FilePath|DefaultImageSymbol) => isNonEmptyFilePath(image) &&
!!personalizationStore.data.wallpaper.local.data[image.path]),
'every image is file path with data');
await getDefaultImageThumbnail(wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{name: 'begin_load_default_image'},
{
thumbnail: {url: '_image_thumbnail'},
name: 'set_default_image',
},
],
personalizationStore.actions,
'load default image thumbnail actions',
);
assertDeepEquals(
[true, false],
personalizationStore.states.map(
state => state.wallpaper.loading.local.data[kDefaultImageSymbol]),
'expected loading state while fetching default thumbnail',
);
assertDeepEquals(
wallpaperProvider.defaultImageThumbnail,
personalizationStore.data.wallpaper.local.data[kDefaultImageSymbol],
'default image thumbnail is set');
});
test('refresh local image list keeps default thumbnail', async () => {
// Initialize some local image data.
await fetchLocalData(wallpaperProvider, personalizationStore);
await getDefaultImageThumbnail(wallpaperProvider, personalizationStore);
// Reset the history of actions and prior states, but keep the current
// state.
personalizationStore.reset(personalizationStore.data);
assertDeepEquals(
wallpaperProvider.defaultImageThumbnail,
personalizationStore.data.wallpaper.local.data[kDefaultImageSymbol],
'default image thumbnail is set');
assertDeepEquals(
[kDefaultImageSymbol, ...wallpaperProvider.localImages!],
personalizationStore.data.wallpaper.local.images,
'local images include default thumbnail',
);
// Simulate user deleting a local image from Downloads directory. Keep the
// first image only.
wallpaperProvider.localImages = wallpaperProvider.localImages!.slice(0, 1);
await fetchLocalData(wallpaperProvider, personalizationStore);
// Default image symbol does not show up in Object.keys.
assertDeepEquals(
[wallpaperProvider.localImages[0]!.path],
Object.keys(personalizationStore.data.wallpaper.local.data),
'local image data deleted for missing image');
assertEquals(
wallpaperProvider.defaultImageThumbnail,
personalizationStore.data.wallpaper.local.data[kDefaultImageSymbol],
'default image thumbnail is still set');
});
});
suite('daily refresh loading', () => {
let wallpaperProvider: TestWallpaperProvider;
let personalizationStore: TestPersonalizationStore;
setup(async () => {
const mocks = baseSetup();
wallpaperProvider = mocks.wallpaperProvider;
personalizationStore = mocks.personalizationStore;
personalizationStore.setReducersEnabled(true);
wallpaperProvider.isInTabletModeResponse = false;
WallpaperObserver.initWallpaperObserverIfNeeded();
await wallpaperProvider.whenCalled('getDailyRefreshCollectionId');
});
teardown(() => {
WallpaperObserver.shutdown();
});
suite('google photos', () => {
const mockAlbum = {
id: 'google-photos-album-0',
title: '',
photoCount: 0,
isShared: false,
preview: {url: 'bar.com'},
timestamp: {internalValue: BigInt(0)},
};
const mockPhotos: GooglePhotosPhoto[] = [
{
id: 'google-photos-photo-0',
dedupKey: 'google-photos-photo-0',
name: 'photo 0',
date: {data: []},
url: {url: ''},
location: 'home',
},
{
id: 'google-photos-photo-1',
dedupKey: 'google-photos-photo-1',
name: 'photo 1',
date: {data: []},
url: {url: ''},
location: 'home',
},
];
setup(async () => {
wallpaperProvider.setGooglePhotosPhotos(mockPhotos);
wallpaperProvider.setGooglePhotosAlbums([mockAlbum]);
wallpaperProvider.setGooglePhotosPhotosByAlbumId(
mockAlbum.id, mockPhotos);
await fetchGooglePhotosEnabled(wallpaperProvider, personalizationStore);
await fetchGooglePhotosPhotos(wallpaperProvider, personalizationStore);
await fetchGooglePhotosAlbums(wallpaperProvider, personalizationStore);
await fetchGooglePhotosAlbum(
wallpaperProvider, personalizationStore, mockAlbum.id);
personalizationStore.reset(personalizationStore.data);
});
test('triggers loading if select new album', async () => {
// Set new daily refresh state to return after
// `selectGooglePhotosAlbum`.
wallpaperProvider.collectionId = '';
wallpaperProvider.albumId = mockAlbum.id;
assertFalse(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh not loading');
await selectGooglePhotosAlbum(
mockAlbum.id, wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{name: 'begin_update_daily_refresh_image'},
{
name: 'set_google_photos_daily_refresh_album_id',
albumId: 'google-photos-album-0',
},
],
personalizationStore.actions,
'begin update daily refresh action sent');
assertTrue(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh loading should be set');
wallpaperProvider.resetResolver('getGooglePhotosDailyRefreshAlbumId');
wallpaperProvider.wallpaperObserverRemote!.onWallpaperChanged({
descriptionContent: '',
descriptionTitle: '',
key: getImageKey(mockPhotos[0]!)!,
layout: WallpaperLayout.kCenterCropped,
type: WallpaperType.kDailyGooglePhotos,
});
wallpaperProvider.wallpaperObserverRemote!.onAttributionChanged({
attribution: [],
key: getImageKey(mockPhotos[0]!)!,
});
// Wait for observer to handle the above event.
await wallpaperProvider.whenCalled('getGooglePhotosDailyRefreshAlbumId');
assertFalse(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh no longer loading after receiving new wallpaper');
});
test('no loading for already selected album', async () => {
// Set new daily refresh state to return after `selectGooglePhotosAlbum`.
wallpaperProvider.collectionId = '';
wallpaperProvider.albumId = mockAlbum.id;
assertFalse(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh not loading');
personalizationStore.data.wallpaper.attribution = {
attribution: [],
key: mockPhotos[0]!.dedupKey!,
};
personalizationStore.data.wallpaper.currentSelected = {
descriptionContent: '',
descriptionTitle: '',
key: mockPhotos[0]!.dedupKey!,
layout: WallpaperLayout.kCenter,
type: WallpaperType.kOnceGooglePhotos,
};
await selectGooglePhotosAlbum(
mockAlbum.id, wallpaperProvider, personalizationStore);
assertDeepEquals(
[
{
name: 'set_google_photos_daily_refresh_album_id',
albumId: 'google-photos-album-0',
},
],
personalizationStore.actions, 'no begin update daily refresh action');
assertFalse(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh still not loading');
});
});
suite('backdrop', () => {
setup(async () => {
await initializeBackdropData(wallpaperProvider, personalizationStore);
personalizationStore.reset(personalizationStore.data);
});
test('sets loading state for new collectionId', async () => {
// Set daily refresh state response.
wallpaperProvider.setDailyRefreshCollectionIdResponse = {success: true};
wallpaperProvider.albumId = '';
wallpaperProvider.collectionId = wallpaperProvider.collections![0]!.id;
// Reset any wallpaper that is selected.
personalizationStore.data.wallpaper.currentSelected = null;
await setDailyRefreshCollectionId(
wallpaperProvider.collectionId, wallpaperProvider,
personalizationStore);
assertDeepEquals(
[
{
name: 'begin_update_daily_refresh_image',
},
{
name: 'set_daily_refresh_collection_id',
collectionId: 'id_0',
},
],
personalizationStore.actions,
'sends begin update daily refresh action' +
JSON.stringify(personalizationStore.actions));
assertTrue(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh still loading');
wallpaperProvider.resetResolver('getDailyRefreshCollectionId');
wallpaperProvider.wallpaperObserverRemote!.onWallpaperChanged({
descriptionContent: '',
descriptionTitle: '',
key: getImageKey(wallpaperProvider.images![0]!)!,
layout: WallpaperLayout.kCenterCropped,
type: WallpaperType.kDailyGooglePhotos,
});
wallpaperProvider.wallpaperObserverRemote!.onAttributionChanged({
attribution: [],
key: getImageKey(wallpaperProvider.images![0]!)!,
});
// Wait for observer to handle the above event.
await wallpaperProvider.whenCalled('getDailyRefreshCollectionId');
assertFalse(
personalizationStore.data.wallpaper.loading.refreshWallpaper,
'daily refresh no longer loading after receiving new wallpaper');
});
test('no loading state for already selected collection', async () => {
wallpaperProvider.setDailyRefreshCollectionIdResponse = {success: true};
wallpaperProvider.albumId = '';
wallpaperProvider.collectionId = wallpaperProvider.collections![0]!.id;
personalizationStore.data.wallpaper.attribution = {
attribution: [],
key: getImageKey(wallpaperProvider.images![0]!)!,
};
personalizationStore.data.wallpaper.currentSelected = {
layout: WallpaperLayout.kCenter,
type: WallpaperType.kOnline,
key: getImageKey(wallpaperProvider.images![0]!)!,
descriptionContent: '',
descriptionTitle: '',
};
await setDailyRefreshCollectionId(
wallpaperProvider.collectionId, wallpaperProvider,
personalizationStore);
assertDeepEquals(
[
{
name: 'set_daily_refresh_collection_id',
collectionId: 'id_0',
},
],
personalizationStore.actions,
'no begin update daily refresh action' +
JSON.stringify(personalizationStore.actions));
});
});
});