chromium/chrome/test/data/webui/chromeos/personalization_app/personalization_app_controller_test.ts

// 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));
    });
  });
});