diff --git a/__tests__/integration/mirador/thumbnail-navigation.test.js b/__tests__/integration/mirador/thumbnail-navigation.test.js
index 35de953ccaa3b13e04ec8ab4fb17bc7a9faf381e..82502cbd53e644f1417557d7fa56bb5f80f704f5 100644
--- a/__tests__/integration/mirador/thumbnail-navigation.test.js
+++ b/__tests__/integration/mirador/thumbnail-navigation.test.js
@@ -6,7 +6,8 @@ describe('Thumbnail navigation', () => {
     await expect(page).toMatchElement('.mirador-window', { polling: 'mutation', timeout: 5000 });
   });
 
-  it('navigates a manifest using thumbnail navigation', async () => {
+  // TODO: pick a new url; this Harvard one is 404
+  xit('navigates a manifest using thumbnail navigation', async () => {
     await expect(page).toMatchElement('.mirador-thumb-navigation');
     let windows = await page.evaluate(() => (
       miradorInstance.store.getState().windows
@@ -20,7 +21,7 @@ describe('Thumbnail navigation', () => {
     ));
     expect(Object.values(windows)[0].canvasId).toBe('https://iiif.harvardartmuseums.org/manifests/object/299843/canvas/canvas-18737483'); // canvas @ index 1
   });
-  it('displays on right side', async () => {
+  xit('displays on right side', async () => {
     await expect(page).toMatchElement('.mirador-thumb-navigation');
     await expect(page).toMatchElement('.mirador-companion-area-far-bottom .mirador-thumb-navigation');
     const windowId = await page.evaluate(() => {
diff --git a/__tests__/src/components/AppProviders.test.js b/__tests__/src/components/AppProviders.test.js
index 82e2f981e309ba39b5afbfc60e4b4304904bfa3d..ae97556638812588c2481172ad0e7bc3943d9a23 100644
--- a/__tests__/src/components/AppProviders.test.js
+++ b/__tests__/src/components/AppProviders.test.js
@@ -1,4 +1,4 @@
-import Button from '@material-ui/core/Button';
+import Button from '@mui/material/Button';
 import { render, screen } from 'test-utils';
 import { useTranslation } from 'react-i18next';
 import { useDrop } from 'react-dnd';
diff --git a/__tests__/src/components/CollapsibleSection.test.js b/__tests__/src/components/CollapsibleSection.test.js
index 9e3ee8757ca6837b22d788e51ecc0ac879fb03e6..aad2dd0f4c16bbce9bfa1a7e826d4cf525baae70 100644
--- a/__tests__/src/components/CollapsibleSection.test.js
+++ b/__tests__/src/components/CollapsibleSection.test.js
@@ -34,9 +34,9 @@ describe('CollapsibleSection', () => {
     expect(screen.getByRole('button')).toHaveAttribute('aria-label', 'expandSection');
   });
 
-  it('renders children based on the open state', async () => {
-    expect(screen.getByTestId('child')).toBeInTheDocument();
+  it('displays children based on the open state', async () => {
+    expect(screen.getByTestId('child')).toBeVisible();
     await userEvent.click(screen.getByRole('button'));
-    expect(screen.queryByTestId('child')).not.toBeInTheDocument();
+    expect(screen.queryByTestId('child')).not.toBeVisible();
   });
 });
diff --git a/__tests__/src/components/CompanionArea.test.js b/__tests__/src/components/CompanionArea.test.js
index 404359b6af87d87a52ae7bd54e2f992a4a8adb9b..53e7bcdb0a5d1e347e664b42f4fe945c78371c2d 100644
--- a/__tests__/src/components/CompanionArea.test.js
+++ b/__tests__/src/components/CompanionArea.test.js
@@ -30,7 +30,7 @@ describe('CompanionArea', () => {
   it('should add the appropriate classes when the companion area fills the full width', () => {
     const { container } = createWrapper({ position: 'bottom' });
 
-    expect(container.querySelector('.mirador-companion-area-bottom')).toHaveClass('horizontal'); // eslint-disable-line testing-library/no-node-access, testing-library/no-container
+    expect(container.querySelector('.mirador-companion-area-bottom')).toBeInTheDocument(); // eslint-disable-line testing-library/no-node-access, testing-library/no-container
   });
 
   it('renders the appropriate <CompanionWindow> components', () => {
diff --git a/__tests__/src/components/CompanionWindow.test.js b/__tests__/src/components/CompanionWindow.test.js
index 60343426fe4b741bc80ad827b67adaf8cc742849..a7fe7b71ef7f4c22b3a3838545c029845a11e84d 100644
--- a/__tests__/src/components/CompanionWindow.test.js
+++ b/__tests__/src/components/CompanionWindow.test.js
@@ -11,7 +11,6 @@ function createWrapper(props) {
       isDisplayed
       direction="ltr"
       windowId="x"
-      classes={{ horizontal: 'horizontal', small: 'small', vertical: 'vertical' }}
       companionWindow={{}}
       position="right"
       {...props}
@@ -96,9 +95,10 @@ describe('CompanionWindow', () => {
         updateCompanionWindow,
       });
 
-      expect(screen.getByRole('complementary')).toHaveClass('vertical');
+      expect(screen.getByRole('complementary')).toHaveClass('mirador-companion-window-right');
 
       await user.click(screen.getByRole('button', { name: 'moveCompanionWindowToBottom' }));
+
       expect(updateCompanionWindow).toHaveBeenCalledWith({ position: 'bottom' });
     });
   });
@@ -113,7 +113,7 @@ describe('CompanionWindow', () => {
         updateCompanionWindow,
       });
 
-      expect(screen.getByRole('complementary')).toHaveClass('horizontal');
+      expect(screen.getByRole('complementary')).toHaveClass('mirador-companion-window-bottom ');
 
       await user.click(screen.getByRole('button', { name: 'moveCompanionWindowToRight' }));
 
@@ -127,12 +127,6 @@ describe('CompanionWindow', () => {
     expect(screen.getByTestId('xyz')).toBeInTheDocument();
   });
 
-  it('adds a small class when the component width is small', () => {
-    const { container } = createWrapper({ size: { width: 369 } });
-
-    expect(container.querySelector('.MuiToolbar-root')).toHaveClass('small'); // eslint-disable-line testing-library/no-node-access, testing-library/no-container
-  });
-
   it('has a resize handler', () => {
     const { container } = createWrapper();
 
diff --git a/__tests__/src/components/GalleryViewThumbnail.test.js b/__tests__/src/components/GalleryViewThumbnail.test.js
index 3ff910b0865188bef23cb3062272b4fcde964a73..bad9e3098edd648ca859b814766c6ddfff075a09 100644
--- a/__tests__/src/components/GalleryViewThumbnail.test.js
+++ b/__tests__/src/components/GalleryViewThumbnail.test.js
@@ -11,7 +11,6 @@ function createWrapper(props) {
   return render(
     <GalleryViewThumbnail
       canvas={Utils.parseManifest(manifestJson).getSequences()[0].getCanvases()[0]}
-      classes={{ selected: 'selected' }}
       focusOnCanvas={() => {}}
       setCanvas={() => {}}
       {...props}
@@ -23,26 +22,6 @@ describe('GalleryView', () => {
   beforeEach(() => {
     window.HTMLElement.prototype.scrollIntoView = jest.fn();
   });
-  it('sets a mirador-current-canvas-grouping class when the canvas is selected', () => {
-    createWrapper({ selected: true });
-    expect(screen.getByRole('button')).toBeInTheDocument();
-    expect(screen.getByRole('button')).toHaveClass('selected');
-  });
-  it('does not set a mirador-current-canvas-grouping class when the canvas is not selected', () => {
-    createWrapper({ selected: false });
-    expect(screen.getByRole('button')).toBeInTheDocument();
-    expect(screen.getByRole('button')).not.toHaveClass('selected');
-  });
-  it('sets a mirador-current-canvas-grouping class when the canvas is selected', () => {
-    createWrapper({ selected: true });
-    expect(screen.getByRole('button')).toBeInTheDocument();
-    expect(screen.getByRole('button')).toHaveClass('selected');
-  });
-  it('does not set a mirador-current-canvas-grouping class when the canvas is not selected', () => {
-    createWrapper({ selected: false });
-    expect(screen.getByRole('button')).toBeInTheDocument();
-    expect(screen.getByRole('button')).not.toHaveClass('selected');
-  });
   it('renders the thumbnail', () => {
     createWrapper({ config: { height: 55 } });
     expect(screen.getByRole('presentation')).toBeInTheDocument();
diff --git a/__tests__/src/components/IIIFAuthentication.test.js b/__tests__/src/components/IIIFAuthentication.test.js
index d9be920ac9d4ae9b4eeb14415b3b03720c271382..d49d5e712d86dfd835d1353a816ed54910da433b 100644
--- a/__tests__/src/components/IIIFAuthentication.test.js
+++ b/__tests__/src/components/IIIFAuthentication.test.js
@@ -33,7 +33,7 @@ describe('IIIFAuthentication', () => {
   describe('without an auth service', () => {
     it('renders nothing', () => {
       createWrapper({ authServiceId: null });
-      expect(screen.queryByText('login', { selector: 'span' })).not.toBeInTheDocument();
+      expect(screen.queryByRole('button', { name: 'login' })).not.toBeInTheDocument();
       expect(screen.queryByRole('button')).not.toBeInTheDocument();
     });
   });
@@ -41,9 +41,7 @@ describe('IIIFAuthentication', () => {
     it('renders a login bar', async () => {
       const handleAuthInteraction = jest.fn();
       createWrapper({ handleAuthInteraction });
-      const confirmBtn = screen.getByText('login', { selector: 'span' });
-      expect(confirmBtn).toBeInTheDocument();
-      await user.click(confirmBtn);
+      await user.click(screen.getByRole('button', { name: 'login' }));
       expect(handleAuthInteraction).toHaveBeenCalledWith('w', 'http://example.com/auth');
     });
     it('renders nothing for a non-interactive login', () => {
@@ -55,7 +53,8 @@ describe('IIIFAuthentication', () => {
     it('renders with an error message', async () => {
       const handleAuthInteraction = jest.fn();
       createWrapper({ handleAuthInteraction, status: 'failed' });
-      const confirmBtn = await screen.findByText('retry', { selector: 'span' });
+      await user.click(screen.getByRole('button', { name: 'continue' }));
+      const confirmBtn = screen.getByRole('button', { name: /retry/ });
       expect(screen.getByText('Login failed')).toBeInTheDocument();
       expect(screen.getByText('cancel')).toBeInTheDocument();
       expect(screen.getByText('... and this is why.')).toBeInTheDocument();
@@ -71,7 +70,7 @@ describe('IIIFAuthentication', () => {
       window.open = mockWindowOpen;
       const resolveCookieMock = jest.fn();
       createWrapper({ resolveAuthenticationRequest: resolveCookieMock, status: 'cookie' });
-      expect(screen.getByText('login', { selector: 'span' })).toBeInTheDocument();
+      expect(screen.getByRole('button', { name: 'login' })).toBeInTheDocument();
       expect(mockWindowOpen).toHaveBeenCalledWith(`http://example.com/auth?origin=${window.origin}`, 'IiifLoginSender', 'centerscreen');
       mockWindow.closed = true;
       jest.runOnlyPendingTimers();
@@ -81,7 +80,7 @@ describe('IIIFAuthentication', () => {
     it('does the IIIF access token behavior', async () => {
       const resolveTokenMock = jest.fn();
       createWrapper({ resolveAccessTokenRequest: resolveTokenMock, status: 'token' });
-      expect(screen.getByText('login', { selector: 'span' })).toBeInTheDocument();
+      expect(screen.getByRole('button', { name: 'login' })).toBeInTheDocument();
       window.dispatchEvent(new MessageEvent('message', {
         data: { messageId: 'http://example.com/token' },
       }));
@@ -100,7 +99,7 @@ describe('IIIFAuthentication', () => {
         resetAuthenticationState,
         status: 'ok',
       });
-      const confirmBtn = await screen.findByText('exit', { selector: 'span' });
+      const confirmBtn = await screen.findByRole('button', { name: 'exit' });
       await user.click(confirmBtn);
       await waitFor(() => expect(resetAuthenticationState).toHaveBeenCalledWith({
         authServiceId: 'http://example.com/auth', tokenServiceId: 'http://example.com/token',
diff --git a/__tests__/src/components/LanguageSettings.test.js b/__tests__/src/components/LanguageSettings.test.js
index 0cddf8b5db29fe186da2a48ddf186f6e5f614ef2..98b0a82f6d7bce81ff7767993531b8377eef9b50 100644
--- a/__tests__/src/components/LanguageSettings.test.js
+++ b/__tests__/src/components/LanguageSettings.test.js
@@ -35,13 +35,6 @@ describe('LanguageSettings', () => {
     expect(screen.getAllByRole('menuitem')).toHaveLength(2);
   });
 
-  it('non-active list items are buttons (and active are not)', () => {
-    createWrapper({ languages });
-
-    expect(screen.getByRole('menuitem', { name: 'Deutsch' })).not.toHaveClass('MuiButtonBase-root');
-    expect(screen.getByRole('menuitem', { name: 'English' })).toHaveClass('MuiButtonBase-root');
-  });
-
   it('renders the check icon when the active prop returns true', () => {
     createWrapper({ languages });
 
diff --git a/__tests__/src/components/LocalePicker.test.js b/__tests__/src/components/LocalePicker.test.js
index a37bdb4bd927a0eb503c6e0b1136d08f99af7e9f..2ff1b2ceefa4c4e1f96a2a06ee1ead0476de8e85 100644
--- a/__tests__/src/components/LocalePicker.test.js
+++ b/__tests__/src/components/LocalePicker.test.js
@@ -3,7 +3,7 @@ import userEvent from '@testing-library/user-event';
 import { LocalePicker } from '../../../src/components/LocalePicker';
 
 /**
- * Helper function to create a shallow wrapper around LanguageSettings
+ * Helper function to create a shallow wrapper around LocalePicker
  */
 function createWrapper(props) {
   return render(
@@ -24,36 +24,42 @@ describe('LocalePicker', () => {
   });
 
   it('renders a select with the current value', () => {
-    createWrapper({ availableLocales: ['en', 'de'], locale: 'en' });
-
-    expect(screen.getByRole('button')).toHaveTextContent('en');
+    createWrapper({ availableLocales: ['en', 'de'], locale: 'de' });
+    // The option to expand the dropdown menu is rendered by a CompanionWindow titleControls prop in WindowSideBarInfoPanel, which is a combobox
+    const dropdownTitle = screen.getByRole('combobox');
+    expect(dropdownTitle).toHaveTextContent('de');
   });
 
-  it('renders a select with a list item for each language passed in props', async () => {
+  it('renders a select with both options and sets the current value', async () => {
     const user = userEvent.setup();
-
-    createWrapper({ availableLocales: ['en', 'de'], locale: 'en' });
-
-    await user.click(screen.getByRole('button'));
-
-    expect(screen.getAllByRole('option')).toHaveLength(2);
+    createWrapper({ availableLocales: ['en', 'de'], locale: 'de' });
+    const dropdownTitle = screen.getByRole('combobox');
+    // Open the menu
+    await user.click(dropdownTitle);
+    // The dropddown menu is not nested within the combobox, it is a sibling in the DOM, an MuiMenu
+    const menu = screen.getByRole('listbox');
+    // Assert that the menu element has 2 children (2 options)
+    expect(menu.children).toHaveLength(2); // eslint-disable-line testing-library/no-node-access
+    // Verify that the select element has the correct value ('de')
+    const deOption = screen.getByRole('option', { name: 'de' });
+    expect(deOption).toHaveAttribute('aria-selected', 'true');
+    // Verify en is also an option
     expect(screen.getByRole('option', { name: 'en' })).toBeInTheDocument();
-    expect(screen.getByRole('option', { name: 'de' })).toBeInTheDocument();
   });
 
   it('triggers setLocale prop when clicking a list item', async () => {
     const user = userEvent.setup();
     const setLocale = jest.fn();
-
     createWrapper({
       availableLocales: ['en', 'de'],
       locale: 'en',
       setLocale,
     });
-
-    await user.click(screen.getByRole('button'));
+    const dropdownTitle = screen.getByRole('combobox');
+    // Open the Select component
+    await user.click(dropdownTitle);
+    // Change the locale to 'de'
     await user.click(screen.getByRole('option', { name: 'de' }));
-
     expect(setLocale).toHaveBeenCalledTimes(1);
     expect(setLocale).toHaveBeenCalledWith('de');
   });
diff --git a/__tests__/src/components/ManifestListItem.test.js b/__tests__/src/components/ManifestListItem.test.js
index ac40ea45865438b9d687ffb8c8d428db64eeb2f6..bd444de9156ad4b77d10e828de451631c19bb105 100644
--- a/__tests__/src/components/ManifestListItem.test.js
+++ b/__tests__/src/components/ManifestListItem.test.js
@@ -23,18 +23,23 @@ describe('ManifestListItem', () => {
     createWrapper({ buttonRef: 'ref' });
 
     expect(screen.getByRole('listitem')).toHaveAttribute('data-manifestid', 'http://example.com');
+    expect(screen.getByRole('listitem')).toHaveClass('MuiListItem-root');
     expect(screen.getByRole('button')).toHaveTextContent('xyz');
   });
   it('adds a class when the item is active', () => {
     createWrapper({ active: true, classes: { active: 'active' } });
 
+    // If this is true, we can assume the proper styling classes are being applied
+    expect(screen.getByRole('listitem')).toHaveAttribute('data-active', 'true');
+
     expect(screen.getByRole('listitem')).toHaveClass('active');
+    expect(screen.getByRole('listitem')).toHaveClass('Mui-selected');
   });
   it('renders a placeholder element until real data is available', () => {
     const { container } = createWrapper({ ready: false });
 
     expect(screen.queryByRole('button')).not.toBeInTheDocument();
-    expect(container.querySelectorAll('.MuiSkeleton-rect').length).toBeGreaterThan(0); // eslint-disable-line testing-library/no-node-access, testing-library/no-container
+    expect(container.querySelectorAll('.MuiSkeleton-rectangular').length).toBeGreaterThan(0); // eslint-disable-line testing-library/no-node-access, testing-library/no-container
   });
   it('renders an error message if fetching the manifest failed', () => {
     createWrapper({ error: 'This is an error message' });
diff --git a/__tests__/src/components/PrimaryWindow.test.js b/__tests__/src/components/PrimaryWindow.test.js
index 52ca6c86b82a9c87bcf08e37c7a09b33c5bc3db8..ad78ebf4b5cf39c5f7433721471429ded5b96237 100644
--- a/__tests__/src/components/PrimaryWindow.test.js
+++ b/__tests__/src/components/PrimaryWindow.test.js
@@ -16,7 +16,7 @@ function createWrapper(props) {
 describe('PrimaryWindow', () => {
   it('should render expected elements', async () => {
     createWrapper({ isFetching: false });
-    await screen.findByRole('region', { accessibleName: 'item' });
+    await screen.findByTestId('test-window');
     expect(document.querySelector('.mirador-primary-window')).toBeInTheDocument(); // eslint-disable-line testing-library/no-node-access
     expect(document.querySelector('.mirador-companion-area-left')).toBeInTheDocument(); // eslint-disable-line testing-library/no-node-access
   });
@@ -30,7 +30,7 @@ describe('PrimaryWindow', () => {
   });
   it('should render <GalleryView> if fetching is complete and view is gallery', async () => {
     createWrapper({ isFetching: false, view: 'gallery' });
-    await screen.findByRole('region', { accessibleName: 'gallery section' });
+    await screen.findByTestId('test-window');
     expect(document.querySelector('#xyz-gallery')).toBeInTheDocument(); // eslint-disable-line testing-library/no-node-access
   });
   it('should render <CollectionDialog> and <SelectCollection> if manifest is collection and isCollectionDialogVisible', async () => {
diff --git a/__tests__/src/components/SanitizedHtml.test.js b/__tests__/src/components/SanitizedHtml.test.js
index 37c09f96c12440c7ab051fe23ca45626754cb3ee..382b90570e5ca8748fab1db2d7da25746e9c1f90 100644
--- a/__tests__/src/components/SanitizedHtml.test.js
+++ b/__tests__/src/components/SanitizedHtml.test.js
@@ -6,7 +6,6 @@ describe('SanitizedHtml', () => {
     render(
       <SanitizedHtml
         data-testid="subject"
-        classes={{ root: 'root' }}
         htmlString="<script>doBadThings()</script><b>Don't worry!</b><a>Some link</a>"
         ruleSet="iiif"
       />,
@@ -18,7 +17,6 @@ describe('SanitizedHtml', () => {
   });
 
   it('should pass correct class name to root element', () => {
-    expect(screen.getByTestId('subject')).toHaveClass('root');
     expect(screen.getByTestId('subject')).toHaveClass('mirador-third-party-html');
   });
 
diff --git a/__tests__/src/components/ScrollIndicatedDialogContent.test.js b/__tests__/src/components/ScrollIndicatedDialogContent.test.js
index 4c76018dca7d1c4614071af837c6e6ec493566ee..ae84979485a464c2d6b98928a42082833702c223 100644
--- a/__tests__/src/components/ScrollIndicatedDialogContent.test.js
+++ b/__tests__/src/components/ScrollIndicatedDialogContent.test.js
@@ -6,7 +6,6 @@ function createWrapper(props) {
   return render(
     <ScrollIndicatedDialogContent
       data-testid="subject"
-      classes={{ shadowScrollDialog: 'shadowScrollDialog' }}
       {...props}
     />,
   );
@@ -19,16 +18,8 @@ describe('ScrollIndicatedDialogContent', () => {
     expect(screen.getByTestId('subject')).toHaveAttribute('randomprop', 'randomPropValue');
   });
 
-  it('provides a className to the DialogContent prop to style it', () => {
-    createWrapper();
-
-    expect(screen.getByTestId('subject')).toHaveClass('shadowScrollDialog');
-  });
-
   it('joins an incoming className prop with our className', () => {
     createWrapper({ className: 'upstreamClassName' });
-
-    expect(screen.getByTestId('subject')).toHaveClass('shadowScrollDialog');
     expect(screen.getByTestId('subject')).toHaveClass('upstreamClassName');
   });
 });
diff --git a/__tests__/src/components/ScrollTo.test.js b/__tests__/src/components/ScrollTo.test.js
index eb2f38c157394c6a7c6c1dd773ecc747fbf56cd4..fd4f34f087645e865eb56725a97705c40b80f7ca 100644
--- a/__tests__/src/components/ScrollTo.test.js
+++ b/__tests__/src/components/ScrollTo.test.js
@@ -1,4 +1,4 @@
-import { render, screen } from 'test-utils';
+import { render } from 'test-utils';
 import { createRef } from 'react';
 import { ScrollTo } from '../../../src/components/ScrollTo';
 
@@ -12,7 +12,7 @@ describe('ScrollTo', () => {
     containerRef = createRef();
     render(<div data-testid="container" ref={containerRef} />);
 
-    containerRef.current.domEl = {
+    containerRef.current = {
       getBoundingClientRect: () => containerBoundingRect,
       getElementsByClassName: () => [{ scrollTo }],
     };
@@ -22,12 +22,6 @@ describe('ScrollTo', () => {
   const scrollToElBelowBoundingRect = { bottom: 601, top: 501 };
   const visibleScrollToElBoundingRect = { bottom: 300, top: 200 };
 
-  it('wraps the given children in a div element', () => {
-    render(<ScrollTo data-testid="subject" scrollTo>Child Prop</ScrollTo>);
-
-    expect(screen.getByTestId('subject')).toHaveTextContent('Child Prop');
-  });
-
   describe('when updating the scrollTo prop', () => {
     beforeEach(() => {
       jest.spyOn(ScrollTo.prototype, 'elementToScrollTo').mockImplementation(() => ({ offsetTop: 450 }));
@@ -38,13 +32,13 @@ describe('ScrollTo', () => {
           ...scrollToElAboveBoundingRect,
         }));
 
-        const { rerender } = render(<ScrollTo scrollTo containerRef={containerRef}>Child</ScrollTo>);
+        const { rerender } = render(<ScrollTo scrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
         // It is called once when initially rendered w/ true
         expect(scrollTo).toHaveBeenCalled();
         scrollTo.mockReset();
 
-        rerender(<ScrollTo containerRef={containerRef}>Child</ScrollTo>);
+        rerender(<ScrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
         // But it is not called on the re-render w/ false
         expect(scrollTo).not.toHaveBeenCalled();
@@ -56,9 +50,9 @@ describe('ScrollTo', () => {
         jest.spyOn(ScrollTo.prototype, 'scrollToBoundingRect').mockImplementation(() => ({
           ...scrollToElAboveBoundingRect,
         }));
-        const { rerender } = render(<ScrollTo containerRef={containerRef}>Child</ScrollTo>);
+        const { rerender } = render(<ScrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
-        rerender(<ScrollTo scrollTo containerRef={containerRef}>Child</ScrollTo>);
+        rerender(<ScrollTo scrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
         expect(scrollTo).toHaveBeenCalledWith(0, 230);
       });
@@ -68,9 +62,9 @@ describe('ScrollTo', () => {
           ...scrollToElBelowBoundingRect,
         }));
 
-        const { rerender } = render(<ScrollTo containerRef={containerRef}>Child</ScrollTo>);
+        const { rerender } = render(<ScrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
-        rerender(<ScrollTo scrollTo containerRef={containerRef}>Child</ScrollTo>);
+        rerender(<ScrollTo scrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
         expect(scrollTo).toHaveBeenCalledWith(0, 230);
       });
@@ -80,9 +74,9 @@ describe('ScrollTo', () => {
           ...visibleScrollToElBoundingRect,
         }));
 
-        const { rerender } = render(<ScrollTo containerRef={containerRef}>Child</ScrollTo>);
+        const { rerender } = render(<ScrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
-        rerender(<ScrollTo scrollTo containerRef={containerRef}>Child</ScrollTo>);
+        rerender(<ScrollTo scrollTo containerRef={containerRef}><div>Child</div></ScrollTo>);
 
         expect(scrollTo).not.toHaveBeenCalled();
       });
diff --git a/__tests__/src/components/SearchHit.test.js b/__tests__/src/components/SearchHit.test.js
index 3db42a226f4f1be2ba8646ee75ab9804a274f318..e999a8b1e8efe2fa83a0294167d76ce2d21d49ae 100644
--- a/__tests__/src/components/SearchHit.test.js
+++ b/__tests__/src/components/SearchHit.test.js
@@ -24,7 +24,6 @@ const Subject = (props) => (
     announcer={() => {}}
     annotation={{ targetId: 'x' }}
     annotationId="foo"
-    classes={{ windowSelected: 'windowSelected' }}
     hit={{
       after: ', and start the chainsaw',
       annotations: ['foo'],
@@ -46,7 +45,6 @@ describe('SearchHit', () => {
     render(<Subject selectAnnotation={selectAnnotation} />);
 
     expect(screen.getByRole('listitem')).toHaveClass('Mui-selected');
-    expect(screen.getByRole('listitem')).toHaveClass('windowSelected');
     expect(screen.getByRole('listitem')).toHaveTextContent('1Light up the moose , and start the chai more');
 
     await user.click(screen.getByRole('button'));
@@ -69,14 +67,13 @@ describe('SearchHit', () => {
     it('renders the annotationLabel if present', () => {
       render(<Subject annotationLabel="The Anno Label" />);
 
-      expect(screen.getAllByRole('heading', { level: 6 })).toHaveLength(2);
-      expect(screen.getByRole('heading', { level: 6, name: 'The Anno Label' })).toHaveClass('MuiTypography-subtitle2');
+      expect(screen.getByRole('heading', { level: 4, name: 'The Anno Label' })).toBeInTheDocument();
     });
 
     it('does not render the typography if no annotation label is present', () => {
       render(<Subject />);
 
-      expect(screen.getByRole('heading', { level: 6 })).toBeInTheDocument();
+      expect(screen.getByRole('heading', { level: 4 })).toBeInTheDocument();
     });
   });
 
diff --git a/__tests__/src/components/SearchPanel.test.js b/__tests__/src/components/SearchPanel.test.js
index 8b4fb3a304ea83af9f521338171e3a1bdea74759..cd7b50ee01958a1ab2aff5109d3edd4e9b7ad274 100644
--- a/__tests__/src/components/SearchPanel.test.js
+++ b/__tests__/src/components/SearchPanel.test.js
@@ -54,7 +54,7 @@ describe('SearchPanel', () => {
   it('has the SearchPanelControls component', () => {
     createWrapper();
 
-    expect(screen.getByRole('textbox', { name: 'searchInputLabel' })).toBeInTheDocument();
+    expect(screen.getByRole('combobox', { name: 'searchInputLabel' })).toBeInTheDocument();
     expect(screen.getByRole('button', { name: 'searchSubmitAria' })).toBeInTheDocument();
   });
 
diff --git a/__tests__/src/components/SearchPanelControls.test.js b/__tests__/src/components/SearchPanelControls.test.js
index e2ab065c2730aafab356b1af114d26ca6e28513d..0b018912404980e4266fbf88a25092ac8f518e09 100644
--- a/__tests__/src/components/SearchPanelControls.test.js
+++ b/__tests__/src/components/SearchPanelControls.test.js
@@ -33,7 +33,7 @@ describe('SearchPanelControls', () => {
       searchService: { id: 'http://example.com/search', options: { resource: { id: 'abc' } } },
     });
 
-    await user.click(screen.getByRole('textbox'));
+    await user.click(screen.getByRole('combobox'));
     await user.keyboard('somestring');
     await user.click(await screen.findByText('somestring 12345'));
     expect(fetchSearch).toHaveBeenCalledWith('window', 'cw', 'http://example.com/search?q=somestring+12345', 'somestring 12345');
@@ -44,7 +44,7 @@ describe('SearchPanelControls', () => {
   it('renders a text input through the renderInput prop', () => {
     createWrapper();
 
-    expect(screen.getByRole('textbox')).toHaveAttribute('id', 'search-cw');
+    expect(screen.getByRole('combobox')).toHaveAttribute('id', 'search-cw');
   });
   it('endAdornment is a SearchIcon (with no CircularProgress indicator)', () => {
     createWrapper();
@@ -67,7 +67,7 @@ describe('SearchPanelControls', () => {
     };
     createWrapper({ fetchSearch, query: 'asdf', searchService });
 
-    await user.clear(screen.getByRole('textbox'));
+    await user.clear(screen.getByRole('combobox'));
     await user.keyboard('yolo');
     await user.click(screen.getByRole('button'));
 
@@ -84,7 +84,7 @@ describe('SearchPanelControls', () => {
 
     createWrapper({ fetchSearch, query: '', searchService });
 
-    await user.clear(screen.getByRole('textbox'));
+    await user.clear(screen.getByRole('combobox'));
     await user.click(screen.getByRole('button', { name: 'searchSubmitAria' }));
     expect(fetchSearch).not.toHaveBeenCalled();
   });
@@ -92,12 +92,12 @@ describe('SearchPanelControls', () => {
   describe('input', () => {
     it('has the query prop has the input value on intial render', () => {
       createWrapper({ query: 'Wolpertinger' });
-      expect(screen.getByRole('textbox')).toHaveValue('Wolpertinger');
+      expect(screen.getByRole('combobox')).toHaveValue('Wolpertinger');
     });
 
     it('clears the local search state/input when the incoming query prop has been cleared', () => {
       const wrapper = createWrapper({ query: 'Wolpertinger' });
-      expect(screen.getByRole('textbox')).toHaveValue('Wolpertinger');
+      expect(screen.getByRole('combobox')).toHaveValue('Wolpertinger');
 
       wrapper.rerender((
         <SearchPanelControls
@@ -107,7 +107,7 @@ describe('SearchPanelControls', () => {
         />
       ));
 
-      expect(screen.getByRole('textbox')).toHaveValue('');
+      expect(screen.getByRole('combobox')).toHaveValue('');
     });
   });
 });
diff --git a/__tests__/src/components/SearchResults.test.js b/__tests__/src/components/SearchResults.test.js
index 14a906cb6bf052c50fafea29a677a7f372039e35..3579a9789afeea5b5de2df963e75b981cfb92fb4 100644
--- a/__tests__/src/components/SearchResults.test.js
+++ b/__tests__/src/components/SearchResults.test.js
@@ -95,8 +95,8 @@ describe('SearchResults', () => {
         searchHits: [],
       });
 
-      expect(screen.getByRole('heading', { level: 6, name: 'The Anno Label' })).toBeInTheDocument();
-      expect(screen.getByRole('heading', { level: 6, name: 'Annother Anno Label' })).toBeInTheDocument();
+      expect(screen.getByRole('heading', { level: 4, name: 'The Anno Label' })).toBeInTheDocument();
+      expect(screen.getByRole('heading', { level: 4, name: 'Annother Anno Label' })).toBeInTheDocument();
     });
   });
 
diff --git a/__tests__/src/components/ThumbnailCanvasGrouping.test.js b/__tests__/src/components/ThumbnailCanvasGrouping.test.js
index 96895ba47f9f91d210fbb08f3a061669089405f9..8bfff1d6d06dc63f5b0ff8a348b4eb52c744e72d 100644
--- a/__tests__/src/components/ThumbnailCanvasGrouping.test.js
+++ b/__tests__/src/components/ThumbnailCanvasGrouping.test.js
@@ -36,12 +36,13 @@ describe('ThumbnailCanvasGrouping', () => {
     setCanvas = jest.fn();
     wrapper = createWrapper({ data, setCanvas });
   });
+  const spyCurrentCanvasClass = jest.spyOn(ThumbnailCanvasGrouping.prototype, 'currentCanvasClass');
+  afterEach(() => {
+    spyCurrentCanvasClass.mockClear();
+  });
   it('renders', () => {
     expect(screen.getByRole('gridcell')).toBeInTheDocument();
   });
-  it('sets a mirador-current-canvas-grouping class on current canvas', () => {
-    expect(screen.getByRole('button')).toHaveClass('mirador-current-canvas-grouping');
-  });
   it('renders a CaptionedIIIFThumbnail', () => {
     expect(screen.getByText('Image 1')).toBeInTheDocument();
   });
@@ -49,9 +50,9 @@ describe('ThumbnailCanvasGrouping', () => {
     wrapper.unmount();
     const user = userEvent.setup();
     wrapper = createWrapper({ data, index: 0, setCanvas });
-
     await user.click(wrapper.container.querySelector('.mirador-thumbnail-nav-canvas-0')); // eslint-disable-line testing-library/no-node-access
-
+    expect(spyCurrentCanvasClass).toHaveBeenCalledWith([0]);
+    expect(spyCurrentCanvasClass).toHaveReturnedWith('current-canvas-grouping');
     expect(setCanvas).toHaveBeenCalledWith('http://iiif.io/api/presentation/2.0/example/fixtures/canvas/24/c1.json');
   });
   describe('attributes based off far-bottom position', () => {
diff --git a/__tests__/src/components/WindowAuthenticationBar.test.js b/__tests__/src/components/WindowAuthenticationBar.test.js
index b47771b1d06e105ceb32e335cae3ce96d8038a57..c4368b81132bfd1c1202473d4fc296444eb73404 100644
--- a/__tests__/src/components/WindowAuthenticationBar.test.js
+++ b/__tests__/src/components/WindowAuthenticationBar.test.js
@@ -37,38 +37,36 @@ describe('AuthenticationControl', () => {
   it('renders a non-collapsing version if there is no description', () => {
     createWrapper({ description: undefined, header: undefined });
     expect(screen.getByText('Log in to see more', { selector: 'span' })).toBeInTheDocument();
-    expect(screen.getByText('Login', { selector: 'span' })).toBeInTheDocument();
+    expect(screen.getByRole('button')).toHaveTextContent('Login');
   });
 
   it('renders a collapsable version if there is a description', async () => {
     createWrapper({ description: 'long description', header: 'Login to Example Institution' });
     const continueBtn = document.querySelectorAll('.MuiButtonBase-root')[0];
-    const cancelBtn = screen.getByText('cancel', { selector: 'span' }).closest('button');
-    const loginBtn = screen.getByText('Login', { selector: 'span' });
     const collapseEl = document.querySelector('.MuiCollapse-hidden');
 
     // disable transition animations for easier testing of the Mui Collapse open/close state
     config.disabled = true;
     // initial collapsed state: Presence of continue button text. Hidden cancelBtn, loginBtn, and description
     expect(screen.getByText('continue')).toBeInTheDocument();
-    expect(cancelBtn).not.toBeVisible();
-    expect(loginBtn).not.toBeVisible();
+    expect(screen.queryByRole('button', { name: 'cancel' })).not.toBeInTheDocument();
+    expect(screen.queryByRole('button', { name: 'Login' })).not.toBeInTheDocument();
     expect(within(collapseEl).getByText('long description')).not.toBeVisible();
     // click to expand
     await user.click(continueBtn);
     // expanded state: Removal of continue button text from DOM. Visible cancelBtn, loginBtn, and description
     expect(screen.queryByText('continue')).not.toBeInTheDocument();
-    expect(cancelBtn).toBeVisible();
-    expect(loginBtn).toBeVisible();
+    expect(screen.getByRole('button', { name: 'cancel' })).toBeVisible();
+    expect(screen.getByRole('button', { name: 'Login' })).toBeVisible();
     expect(within(collapseEl).getByText('long description')).toBeVisible();
     expect(collapseEl).toHaveClass('MuiCollapse-entered');
 
     // click the cancel button to collapse
-    await user.click(cancelBtn);
+    await user.click(screen.getByRole('button', { name: 'cancel' }));
     // collapsed state: Presence of continue button text. Hidden cancelBtn, loginBtn, and description
     expect(screen.getByText('continue')).toBeInTheDocument();
-    expect(cancelBtn).not.toBeVisible();
-    expect(loginBtn).not.toBeVisible();
+    expect(screen.queryByRole('button', { name: 'cancel' })).not.toBeInTheDocument();
+    expect(screen.queryByRole('button', { name: 'Login' })).not.toBeInTheDocument();
     expect(within(collapseEl).getByText('long description')).not.toBeVisible();
     // re-enable transition animation
     config.disabled = false;
@@ -77,7 +75,7 @@ describe('AuthenticationControl', () => {
   it('triggers an action when the confirm button is clicked', async () => {
     const onConfirm = jest.fn();
     createWrapper({ onConfirm });
-    await user.click(screen.getByText('Login', { selector: 'span' }).closest('button'));
+    await user.click(screen.getByRole('button', { name: 'Login' }));
     expect(onConfirm).toHaveBeenCalled();
   });
 
diff --git a/__tests__/src/components/WindowCanvasNavigationControls.test.js b/__tests__/src/components/WindowCanvasNavigationControls.test.js
index d089de29157852f4bd77cb0d56e582cac1145b7f..de08147cc359a800fcaae3b5f26e54b996c67af1 100644
--- a/__tests__/src/components/WindowCanvasNavigationControls.test.js
+++ b/__tests__/src/components/WindowCanvasNavigationControls.test.js
@@ -28,7 +28,7 @@ describe('WindowCanvasNavigationControls', () => {
 
   it('renders only a screen-reader accessibile version when visible=false', () => {
     const { container } = render(<Subject visible={false} />);
-    expect(container.firstChild.classList[1]).toMatch(/srOnly/); // eslint-disable-line testing-library/no-node-access
+    expect(container.firstChild).toHaveStyle({ height: '1px', margin: '-1px', width: '1px' }); // eslint-disable-line testing-library/no-node-access
   });
 
   it('stacks the nav controls on small width screens', () => {
@@ -37,10 +37,7 @@ describe('WindowCanvasNavigationControls', () => {
   });
 
   it('shows the zoom control component when specified', () => {
-    render(
-      <Subject />,
-      { preloadedState: { workspace: { showZoomControls: true } } },
-    );
+    render(<Subject showZoomControls />);
     expect(screen.getByRole('button', { name: 'zoomIn' })).toBeInTheDocument();
     expect(screen.getByRole('button', { name: 'zoomOut' })).toBeInTheDocument();
     expect(screen.getByRole('button', { name: 'zoomReset' })).toBeInTheDocument();
diff --git a/__tests__/src/components/WindowSideBar.test.js b/__tests__/src/components/WindowSideBar.test.js
index 895e20fb3516246cd1e681630d9733af96dc8133..0fa75243cf9bb7be907346cdfc7bdd06db3ffd20 100644
--- a/__tests__/src/components/WindowSideBar.test.js
+++ b/__tests__/src/components/WindowSideBar.test.js
@@ -5,7 +5,6 @@ import { WindowSideBar } from '../../../src/components/WindowSideBar';
 function createWrapper({ ...props }) {
   return render(
     <WindowSideBar
-      classes={{ drawer: 'test-drawer' }}
       t={k => k}
       windowId="xyz"
       {...props}
@@ -26,7 +25,6 @@ function createWrapper({ ...props }) {
 describe('WindowSideBar when closed', () => {
   it('renders without an error', () => {
     createWrapper({});
-    expect(document.querySelector('.test-drawer')).toBeInTheDocument(); // eslint-disable-line testing-library/no-node-access
     expect(screen.queryByRole('navigation', { accessibleName: 'sidebarPanelsNavigation' })).not.toBeInTheDocument();
   });
 });
diff --git a/__tests__/src/components/WindowSideBarCanvasPanel.test.js b/__tests__/src/components/WindowSideBarCanvasPanel.test.js
index 9ad3ce6cca69634ab1cc5015e595680a894fb1bf..9c3c48ca0ac8ebef6279fba2c6ba317018967054 100644
--- a/__tests__/src/components/WindowSideBarCanvasPanel.test.js
+++ b/__tests__/src/components/WindowSideBarCanvasPanel.test.js
@@ -65,7 +65,7 @@ describe('WindowSideBarCanvasPanel', () => {
     createWrapper({ multipleSequences: true, updateSequence });
 
     expect(screen.getByTestId('sequence-select')).toHaveTextContent('a');
-    await user.click(within(screen.getByTestId('sequence-select')).getByRole('button'));
+    await user.click(within(screen.getByTestId('sequence-select')).getByRole('combobox'));
 
     const listbox = within(screen.getByRole('listbox'));
     expect(listbox.getAllByRole('option')).toHaveLength(2);
diff --git a/__tests__/src/components/WindowTopMenuButton.test.js b/__tests__/src/components/WindowTopMenuButton.test.js
index 86ef9584d47c462e9db7ba0f522c36f3e8ea7013..15d4b65f42d325a951427a68d261e999aba4f1dd 100644
--- a/__tests__/src/components/WindowTopMenuButton.test.js
+++ b/__tests__/src/components/WindowTopMenuButton.test.js
@@ -38,9 +38,16 @@ describe('WindowTopMenuButton', () => {
     expect(screen.queryByRole('menu')).not.toBeInTheDocument();
   });
 
-  it('the button has a class indicating that it is "selected" once it is clicked', async () => {
+  it('the open attribute of the button is null without being clicked', async () => {
+    render(<Subject />);
+    // without a click, the button is not open and therefore doesn't have aria-owns attr
+    expect(screen.getByLabelText('windowMenu')).not.toHaveAttribute('aria-owns'); // eslint-disable-line testing-library/no-node-access
+  });
+
+  it('the open attribute of the button is applied once it is clicked', async () => {
     render(<Subject />);
     await user.click(screen.getByLabelText('windowMenu'));
-    expect(screen.getByLabelText('windowMenu')).toHaveClass('ctrlBtnSelected'); // eslint-disable-line testing-library/no-node-access
+    // when 'open' is true, aria-owns is set to the id of the window
+    expect(screen.getByLabelText('windowMenu')).toHaveAttribute('aria-owns', 'window-menu_xyz'); // eslint-disable-line testing-library/no-node-access
   });
 });
diff --git a/__tests__/src/components/Workspace.test.js b/__tests__/src/components/Workspace.test.js
index 3f1fd853e79f7aec16a9fad1ab6b05b789675b1b..493598f751109c49936e69dd95271179c79dbbdb 100644
--- a/__tests__/src/components/Workspace.test.js
+++ b/__tests__/src/components/Workspace.test.js
@@ -83,22 +83,6 @@ describe('Workspace', () => {
     });
   });
 
-  describe('when the workspace control panel is displayed', () => {
-    it('has the *-with-control-panel class applied', () => {
-      const { container } = createWrapper();
-
-      expect(container.querySelector('.mirador-workspace-with-control-panel')).toBeInTheDocument();
-    });
-  });
-
-  describe('when the workspace control panel is not displayed', () => {
-    it('does not have the *-with-control-panel class applied', () => {
-      const { container } = createWrapper({ isWorkspaceControlPanelVisible: false });
-
-      expect(container.querySelector('.mirador-workspace-with-control-panel')).not.toBeInTheDocument();
-    });
-  });
-
   describe('drag and drop', () => {
     it('adds a new catalog entry from a manifest', async () => {
       const manifestJson = '{ "data": "123" }';
@@ -106,7 +90,7 @@ describe('Workspace', () => {
       const addWindow = jest.fn();
 
       const { container } = createWrapper({ addWindow });
-      const dropTarget = container.querySelector('.mirador-workspace-with-control-panel');
+      const dropTarget = container.querySelector('.mirador-workspace-viewport');
 
       const file = new File([manifestJson], 'manifest.json', { type: 'application/json' });
       const dataTransfer = {
@@ -129,7 +113,7 @@ describe('Workspace', () => {
 
       const { container } = createWrapper({ addWindow, allowNewWindows: false });
 
-      const dropTarget = container.querySelector('.mirador-workspace-with-control-panel');
+      const dropTarget = container.querySelector('.mirador-workspace-viewport');
 
       const file = new File([manifestJson], 'manifest.json', { type: 'application/json' });
       const dataTransfer = {
diff --git a/__tests__/src/components/WorkspaceElasticWindow.test.js b/__tests__/src/components/WorkspaceElasticWindow.test.js
index 5d8982ea6b553f3bafc7fa8ae0fb103ff307c8d5..03f565afd16baebc97bd1774e00dc388eb0273df 100644
--- a/__tests__/src/components/WorkspaceElasticWindow.test.js
+++ b/__tests__/src/components/WorkspaceElasticWindow.test.js
@@ -46,18 +46,13 @@ describe('WorkspaceElasticWindow', () => {
     expect(el).toHaveStyle({ height: '200px', transform: 'translate(5040px,5040px)', width: '200px' });
   });
   describe('focuses the window', () => {
-    it('adds class to focused window', () => {
-      const { container } = createWrapper({ classes: { focused: 'focused-window' }, focused: true, layout });
-      const el = container.firstChild;
-
-      expect(el).toHaveClass('focused-window');
-    });
     it('calls focusWindow when clicked', async () => {
       const user = userEvent.setup();
       const mockFocusWindow = jest.fn();
       const { container } = createWrapper({ focusWindow: mockFocusWindow, layout });
-      const el = container.querySelector('.mirador-window-top-bar');
-      await user.click(el);
+      const topBar = container.querySelector('.mirador-window-top-bar');
+      await user.click(topBar);
+
       expect(mockFocusWindow).toHaveBeenCalled();
     });
   });
diff --git a/__tests__/src/components/ZoomControls.test.js b/__tests__/src/components/ZoomControls.test.js
index 6f7be5e3ad62133eec3c26a4b1ddadb9eda7dfbe..41a562513eb8b5d87d6765b003ebbc9771046786 100644
--- a/__tests__/src/components/ZoomControls.test.js
+++ b/__tests__/src/components/ZoomControls.test.js
@@ -6,74 +6,47 @@ import { ZoomControls } from '../../../src/components/ZoomControls';
 function createWrapper(props) {
   return render(
     <ZoomControls
-      classes={{ divider: 'divider', zoom_controls: 'zoom_controls' }}
       windowId="xyz"
       zoomToWorld={() => {}}
       {...props}
-
     />,
   );
 }
 
 describe('ZoomControls', () => {
   const viewer = { x: 100, y: 100, zoom: 1 };
-  const showZoomControls = false;
   let updateViewport;
 
-  describe('with showZoomControls=false', () => {
-    it('renders nothing unless asked', () => {
-      const { container } = createWrapper({ showZoomControls, updateViewport, viewer });
-      expect(container).toBeEmptyDOMElement();
+  const zoomToWorld = jest.fn();
+  let user;
+  beforeEach(() => {
+    user = userEvent.setup();
+    updateViewport = jest.fn();
+    createWrapper({
+      updateViewport, viewer, zoomToWorld,
     });
   });
 
-  describe('with showZoomControls=true', () => {
-    const zoomToWorld = jest.fn();
-    let user;
-    beforeEach(() => {
-      user = userEvent.setup();
-      updateViewport = jest.fn();
-      createWrapper({
-        showZoomControls: true, updateViewport, viewer, zoomToWorld,
-      });
-    });
-
-    it('renders a couple buttons', () => {
-      expect(screen.getByRole('button', { name: 'zoomIn' })).toBeInTheDocument();
-      expect(screen.getByRole('button', { name: 'zoomOut' })).toBeInTheDocument();
-      expect(screen.getByRole('button', { name: 'zoomReset' })).toBeInTheDocument();
-    });
-
-    it('has a zoom-in button', async () => {
-      await user.click(screen.getByRole('button', { name: 'zoomIn' }));
-
-      expect(updateViewport).toHaveBeenCalledWith('xyz', { zoom: 2 });
-    });
-
-    it('has a zoom-out button', async () => {
-      await user.click(screen.getByRole('button', { name: 'zoomOut' }));
-      expect(updateViewport).toHaveBeenCalledWith('xyz', { zoom: 0.5 });
-    });
+  it('renders a couple buttons', () => {
+    expect(screen.getByRole('button', { name: 'zoomIn' })).toBeInTheDocument();
+    expect(screen.getByRole('button', { name: 'zoomOut' })).toBeInTheDocument();
+    expect(screen.getByRole('button', { name: 'zoomReset' })).toBeInTheDocument();
+  });
 
-    it('has a zoom reset button', async () => {
-      await user.click(screen.getByRole('button', { name: 'zoomReset' }));
+  it('has a zoom-in button', async () => {
+    await user.click(screen.getByRole('button', { name: 'zoomIn' }));
 
-      expect(zoomToWorld).toHaveBeenCalledWith(false);
-    });
+    expect(updateViewport).toHaveBeenCalledWith('xyz', { zoom: 2 });
   });
 
-  /* eslint-disable testing-library/no-container, testing-library/no-node-access */
-  describe('responsive divider', () => {
-    it('is present when the displayDivider prop is true (default)', () => {
-      const { container } = createWrapper({ showZoomControls: true, viewer });
-
-      expect(container.querySelector('.divider')).toBeInTheDocument();
-    });
+  it('has a zoom-out button', async () => {
+    await user.click(screen.getByRole('button', { name: 'zoomOut' }));
+    expect(updateViewport).toHaveBeenCalledWith('xyz', { zoom: 0.5 });
+  });
 
-    it('is not present when the displayDivider prop is false', () => {
-      const { container } = createWrapper({ displayDivider: false, showZoomControls: true, viewer });
+  it('has a zoom reset button', async () => {
+    await user.click(screen.getByRole('button', { name: 'zoomReset' }));
 
-      expect(container.querySelector('.divider')).not.toBeInTheDocument();
-    });
+    expect(zoomToWorld).toHaveBeenCalledWith(false);
   });
 });
diff --git a/__tests__/utils/test-utils.js b/__tests__/utils/test-utils.js
index 7bc1adf23faae99a0d9146a63145f7947dea6088..7104dec6b038df0a91a8744b5378e22aeab848e9 100644
--- a/__tests__/utils/test-utils.js
+++ b/__tests__/utils/test-utils.js
@@ -3,9 +3,12 @@ import { render } from '@testing-library/react';
 import PropTypes from 'prop-types';
 import { createStore, applyMiddleware } from 'redux';
 import thunkMiddleware from 'redux-thunk';
+import { createTheme, ThemeProvider } from '@mui/material/styles';
 import createRootReducer from '../../src/state/reducers/rootReducer';
+import settings from '../../src/config/settings';
 
 const rootReducer = createRootReducer();
+const theme = createTheme(settings.theme);
 
 /**
  * Hook up our rendered object to redux
@@ -21,7 +24,7 @@ function renderWithProviders(
 ) {
   /** :nodoc: */
   function Wrapper({ children }) {
-    return <Provider store={store}>{children}</Provider>;
+    return <ThemeProvider theme={theme}><Provider store={store}>{children}</Provider></ThemeProvider>;
   }
 
   Wrapper.propTypes = {
diff --git a/jest-puppeteer.config.js b/jest-puppeteer.config.js
index b7f3d29270c58ad2de562ff6eceb7c3ffa280179..31bf265ad41b8b662bbac4fa8cb0062e7652777a 100644
--- a/jest-puppeteer.config.js
+++ b/jest-puppeteer.config.js
@@ -1,6 +1,6 @@
 module.exports = {
   launch: {
-    headless: process.env.HEADLESS !== 'false',
+    headless: process.env.HEADLESS !== 'false' ? 'new' : false,
   },
   server: [{
     command: 'npm run server -- -p 4488',
diff --git a/jest.json b/jest.json
index b5713a9dc77afc15459a2b531716ba0d63c852f3..947d95cd44f04a7c4501be996ca01b82001e7832 100644
--- a/jest.json
+++ b/jest.json
@@ -33,7 +33,7 @@
       "<rootDir>/__tests__/utils"
     ],
     "preset": "jest-puppeteer",
-    "setupFilesAfterEnv": ["@testing-library/jest-dom/extend-expect"],
+    "setupFilesAfterEnv": ["@testing-library/jest-dom/jest-globals"],
     "transformIgnorePatterns": [
       "<rootDir>/node_modules/(?!@react-dnd|react-dnd|dnd-core|react-dnd-html5-backend|dnd-multi-backend|rdndmb-html5-to-touch)"
     ],
@@ -58,7 +58,7 @@
     "preset": "jest-puppeteer",
     "setupFilesAfterEnv": [
       "<rootDir>/setupJestIntegration.js",
-      "@testing-library/jest-dom/extend-expect"
+      "@testing-library/jest-dom/jest-globals"
     ],
     "transformIgnorePatterns": [
       "<rootDir>/node_modules/(?!@react-dnd|react-dnd|dnd-core|react-dnd-html5-backend|dnd-multi-backend|rdndmb-html5-to-touch)"
diff --git a/package.json b/package.json
index 567470b7f75eb603e8af7c4f6618dd03b1a7608e..d77b1f238acfc073e66039b66cadfc3bbf861a69 100644
--- a/package.json
+++ b/package.json
@@ -33,18 +33,21 @@
   ],
   "repository": "https://github.com/ProjectMirador/mirador",
   "dependencies": {
+    "@emotion/cache": "^11.11.0",
+    "@emotion/react": "^11.10.6",
+    "@emotion/styled": "^11.10.6",
     "@hello-pangea/dnd": "^16.0.1",
-    "@material-ui/core": "^4.12.3",
-    "@material-ui/icons": "^4.9.1",
-    "@material-ui/lab": "^4.0.0-alpha.53",
+    "@mui/icons-material": "^5.11.16",
+    "@mui/lab": "^5.0.0-alpha.134",
+    "@mui/material": "^5.13.5",
+    "@mui/utils": "^5.13.1",
+    "@mui/x-tree-view": "^6.17.0",
     "@react-aria/live-announcer": "^3.1.2",
+    "@redux-devtools/extension": "^3.3.0",
     "classnames": "^2.2.6",
-    "clsx": "^1.0.4",
     "deepmerge": "^4.2.2",
     "dompurify": "^3.0.0",
-    "i18next": "^21.0.0 || ^22.0.0",
-    "jss": "^10.3.0",
-    "jss-rtl": "^0.3.0",
+    "i18next": "^21.0.0 || ^22.0.0 || ^23.0.0",
     "lodash": "^4.17.11",
     "manifesto.js": "^4.2.0",
     "normalize-url": "^4.5.0",
@@ -58,7 +61,7 @@
     "react-dnd-multi-backend": "^8.0.0",
     "react-dnd-touch-backend": "^16.0.0",
     "react-full-screen": "^1.1.1",
-    "react-i18next": "^11.7.0 || ^12.0.0",
+    "react-i18next": "^11.7.0 || ^12.0.0 || ^13.0.0",
     "react-image": "^4.0.1",
     "react-intersection-observer": "^9.0.0",
     "react-mosaic-component": "^6.0.0",
@@ -73,6 +76,8 @@
     "redux-saga": "^1.1.3",
     "redux-thunk": "^2.3.0",
     "reselect": "^4.0.0",
+    "stylis": "^4.3.0",
+    "stylis-plugin-rtl": "^2.1.1",
     "url": "^0.11.0",
     "uuid": "^8.1.0 || ^9.0.0"
   },
@@ -88,7 +93,7 @@
     "@babel/preset-react": "^7.16.7",
     "@pmmmwh/react-refresh-webpack-plugin": "^0.5.4",
     "@testing-library/dom": "^9.2.0",
-    "@testing-library/jest-dom": "^5.16.5",
+    "@testing-library/jest-dom": "^6.1.5",
     "@testing-library/react": "^12.1.5",
     "@testing-library/user-event": "^14.4.3",
     "@typescript-eslint/eslint-plugin": "^5.15.0",
@@ -108,19 +113,19 @@
     "eslint-plugin-flowtype": "^8.0.3",
     "eslint-plugin-import": "^2.25.4",
     "eslint-plugin-jest": "^27.1.5",
-    "eslint-plugin-jest-dom": "^4.0.3",
+    "eslint-plugin-jest-dom": "^5.1.0",
     "eslint-plugin-jsx-a11y": "^6.4.1",
     "eslint-plugin-react": "^7.29.4",
     "eslint-plugin-react-hooks": "^4.6.0",
     "eslint-plugin-testing-library": "^5.10.2",
-    "glob": "^9.3.0",
+    "glob": "^10.3.0",
     "http-server": "^14.1.0",
     "jest": "^29.3.1",
     "jest-environment-jsdom": "^29.4.3",
     "jest-fetch-mock": "^3.0.0",
-    "jest-puppeteer": "^8.0.0",
-    "jsdom": "^21.0.0",
-    "puppeteer": "^19.0.0",
+    "jest-puppeteer": "^9.0.2",
+    "jsdom": "^23.0.0",
+    "puppeteer": "^21.0.0",
     "react": "^17.0.0",
     "react-dnd-test-backend": "^16.0.1",
     "react-dom": "^17.0.0",
diff --git a/src/components/AnnotationSettings.js b/src/components/AnnotationSettings.js
index dd91bdc754f4ef9f4494d62a1f43ce64593af6dd..f6fec27ad711fa018e2ce39177c5d302c31c2e5a 100644
--- a/src/components/AnnotationSettings.js
+++ b/src/components/AnnotationSettings.js
@@ -1,7 +1,7 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import VisibilityIcon from '@material-ui/icons/VisibilitySharp';
-import VisibilityOffIcon from '@material-ui/icons/VisibilityOffSharp';
+import VisibilityIcon from '@mui/icons-material/VisibilitySharp';
+import VisibilityOffIcon from '@mui/icons-material/VisibilityOffSharp';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 
 /**
diff --git a/src/components/AppProviders.js b/src/components/AppProviders.js
index e4bcddeac7af38bf14d8c537da32fcbc9717119f..f39aae60984e174a60d8a569b591bc78b6275753 100644
--- a/src/components/AppProviders.js
+++ b/src/components/AppProviders.js
@@ -3,13 +3,15 @@ import PropTypes from 'prop-types';
 import { FullScreen, useFullScreenHandle } from 'react-full-screen';
 import { I18nextProvider } from 'react-i18next';
 import {
-  ThemeProvider, StylesProvider, createTheme, jssPreset, createGenerateClassName,
-} from '@material-ui/core/styles';
+  ThemeProvider, StyledEngineProvider, createTheme,
+} from '@mui/material/styles';
 import { DndContext, DndProvider } from 'react-dnd';
 import { MultiBackend } from 'react-dnd-multi-backend';
 import { HTML5toTouch } from 'rdndmb-html5-to-touch';
-import { create } from 'jss';
-import rtl from 'jss-rtl';
+import rtlPlugin from 'stylis-plugin-rtl';
+import { prefixer } from 'stylis';
+import { CacheProvider } from '@emotion/react';
+import createCache from '@emotion/cache';
 import createI18nInstance from '../i18n';
 import FullScreenContext from '../contexts/FullScreenContext';
 
@@ -92,12 +94,25 @@ export class AppProviders extends Component {
   /** */
   render() {
     const {
-      children, createGenerateClassNameOptions,
+      children,
       theme, translations,
       dndManager,
     } = this.props;
 
-    const generateClassName = createGenerateClassName(createGenerateClassNameOptions);
+    /**
+     * Create rtl emotion cache
+     */
+    const cacheRtl = createCache({
+      key: 'muirtl',
+      stylisPlugins: [prefixer, rtlPlugin],
+    });
+
+    /**
+     * Create default emotion cache
+     */
+    const cacheDefault = createCache({
+      key: 'mui',
+    });
 
     Object.keys(translations).forEach((lng) => {
       this.i18n.addResourceBundle(lng, 'translation', translations[lng], true, true);
@@ -106,18 +121,15 @@ export class AppProviders extends Component {
     return (
       <FullScreenShim>
         <I18nextProvider i18n={this.i18n}>
-          <ThemeProvider
-            theme={createTheme(theme)}
-          >
-            <StylesProvider
-              jss={create({ plugins: [...jssPreset().plugins, rtl()] })}
-              generateClassName={generateClassName}
-            >
-              <MaybeDndProvider dndManager={dndManager}>
-                {children}
-              </MaybeDndProvider>
-            </StylesProvider>
-          </ThemeProvider>
+          <StyledEngineProvider injectFirst>
+            <CacheProvider value={theme.direction === 'rtl' ? cacheRtl : cacheDefault}>
+              <ThemeProvider theme={createTheme((theme))}>
+                <MaybeDndProvider dndManager={dndManager}>
+                  {children}
+                </MaybeDndProvider>
+              </ThemeProvider>
+            </CacheProvider>
+          </StyledEngineProvider>
         </I18nextProvider>
       </FullScreenShim>
     );
@@ -126,7 +138,6 @@ export class AppProviders extends Component {
 
 AppProviders.propTypes = {
   children: PropTypes.node,
-  createGenerateClassNameOptions: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   dndManager: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   language: PropTypes.string.isRequired,
   theme: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
@@ -135,6 +146,5 @@ AppProviders.propTypes = {
 
 AppProviders.defaultProps = {
   children: null,
-  createGenerateClassNameOptions: {},
   dndManager: undefined,
 };
diff --git a/src/components/AttributionPanel.js b/src/components/AttributionPanel.js
index c2d8cc6979e1332c0b9ef46a4c0369209d645342..b1d9c6ebec7acf6af795a26c83c8e205d0df8e1d 100644
--- a/src/components/AttributionPanel.js
+++ b/src/components/AttributionPanel.js
@@ -1,14 +1,24 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import Link from '@material-ui/core/Link';
-import Skeleton from '@material-ui/lab/Skeleton';
+import { styled } from '@mui/material/styles';
+import Typography from '@mui/material/Typography';
+import Link from '@mui/material/Link';
+import Skeleton from '@mui/material/Skeleton';
 import { Img } from 'react-image';
 import CompanionWindow from '../containers/CompanionWindow';
+import { CompanionWindowSection } from './CompanionWindowSection';
 import { LabelValueMetadata } from './LabelValueMetadata';
 import ns from '../config/css-ns';
 import { PluginHook } from './PluginHook';
 
+const StyledLogo = styled(Img)(() => ({
+  maxWidth: '100%',
+}));
+
+const StyledPlaceholder = styled(Skeleton)(({ theme }) => ({
+  backgroundColor: theme.palette.grey[300],
+}));
+
 /**
  * WindowSideBarInfoPanel
  */
@@ -24,7 +34,6 @@ export class AttributionPanel extends Component {
       rights,
       windowId,
       id,
-      classes,
       t,
     } = this.props;
 
@@ -35,7 +44,7 @@ export class AttributionPanel extends Component {
         windowId={windowId}
         id={id}
       >
-        <div className={classes.section}>
+        <CompanionWindowSection>
           { requiredStatement && (
             <LabelValueMetadata labelValuePairs={requiredStatement} defaultLabel={t('attribution')} />
           )}
@@ -53,20 +62,19 @@ export class AttributionPanel extends Component {
               </dl>
             )
           }
-        </div>
+        </CompanionWindowSection>
 
         { manifestLogo && (
-          <div className={classes.section}>
-            <Img
+          <CompanionWindowSection>
+            <StyledLogo
               src={[manifestLogo]}
               alt=""
               role="presentation"
-              className={classes.logo}
               unloader={
-                <Skeleton className={classes.placeholder} variant="rect" height={60} width={60} />
+                <StyledPlaceholder variant="rectangular" height={60} width={60} />
               }
             />
-          </div>
+          </CompanionWindowSection>
         )}
 
         <PluginHook {...this.props} />
@@ -76,7 +84,6 @@ export class AttributionPanel extends Component {
 }
 
 AttributionPanel.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   id: PropTypes.string.isRequired,
   manifestLogo: PropTypes.string,
   requiredStatement: PropTypes.arrayOf(PropTypes.shape({
@@ -89,7 +96,6 @@ AttributionPanel.propTypes = {
 };
 
 AttributionPanel.defaultProps = {
-  classes: {},
   manifestLogo: null,
   requiredStatement: null,
   rights: null,
diff --git a/src/components/AudioViewer.js b/src/components/AudioViewer.js
index ab7b9d6883b7186b6b990744707d5be5923eab32..9ceb07d78826542af7e9133df4c66f4ea42232a4 100644
--- a/src/components/AudioViewer.js
+++ b/src/components/AudioViewer.js
@@ -1,40 +1,48 @@
-import { Component, Fragment } from 'react';
+import { Fragment } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
+
+const StyledContainer = styled('div')({
+  alignItems: 'center',
+  display: 'flex',
+  width: '100%',
+});
+
+const StyledAudio = styled('audio')({
+  width: '100%',
+});
 
 /** */
-export class AudioViewer extends Component {
+export function AudioViewer(props) {
   /* eslint-disable jsx-a11y/media-has-caption */
   /** */
-  render() {
-    const {
-      captions, classes, audioOptions, audioResources,
-    } = this.props;
+  const {
+    captions, audioOptions, audioResources,
+  } = props;
 
-    return (
-      <div className={classes.container}>
-        <audio className={classes.audio} {...audioOptions}>
-          {audioResources.map(audio => (
-            <Fragment key={audio.id}>
-              <source src={audio.id} type={audio.getFormat()} />
-            </Fragment>
-          ))}
-          {captions.map(caption => (
-            <Fragment key={caption.id}>
-              <track src={caption.id} label={caption.getDefaultLabel()} srcLang={caption.getProperty('language')} />
-            </Fragment>
-          ))}
-        </audio>
-      </div>
-    );
-  }
-  /* eslint-enable jsx-a11y/media-has-caption */
+  return (
+    <StyledContainer>
+      <StyledAudio {...audioOptions}>
+        {audioResources.map((audio) => (
+          <Fragment key={audio.id}>
+            <source src={audio.id} type={audio.getFormat()} />
+          </Fragment>
+        ))}
+        {captions.map((caption) => (
+          <Fragment key={caption.id}>
+            <track src={caption.id} label={caption.getDefaultLabel()} srcLang={caption.getProperty('language')} />
+          </Fragment>
+        ))}
+      </StyledAudio>
+    </StyledContainer>
+  );
 }
+/* eslint-enable jsx-a11y/media-has-caption */
 
 AudioViewer.propTypes = {
   audioOptions: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   audioResources: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
   captions: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
 };
 
 AudioViewer.defaultProps = {
diff --git a/src/components/Branding.js b/src/components/Branding.js
index d2f4d4f9e1a42cd605136baa9da81cba0526586e..9b3af649ccbb0520a5842e95778b45091a97205b 100644
--- a/src/components/Branding.js
+++ b/src/components/Branding.js
@@ -1,7 +1,8 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import IconButton from '@material-ui/core/IconButton';
-import Typography from '@material-ui/core/Typography';
+import IconButton from '@mui/material/IconButton';
+import Typography from '@mui/material/Typography';
+import Stack from '@mui/material/Stack';
 import MiradorIcon from './icons/MiradorIcon';
 
 /**
@@ -13,7 +14,7 @@ export class Branding extends Component {
     const { t, variant, ...ContainerProps } = this.props;
 
     return (
-      <div {...ContainerProps}>
+      <Stack alignItems="center" {...ContainerProps}>
         { variant === 'wide' && (
         <div>
           <Typography align="center" component="p" variant="h3">{t('mirador')}</Typography>
@@ -25,11 +26,12 @@ export class Branding extends Component {
             href="https://projectmirador.org"
             target="_blank"
             rel="noopener"
+            size="large"
           >
             <MiradorIcon aria-label={t('aboutMirador')} titleAccess={t('aboutMirador')} fontSize="large" />
           </IconButton>
         </Typography>
-      </div>
+      </Stack>
     );
   }
 }
diff --git a/src/components/CanvasAnnotations.js b/src/components/CanvasAnnotations.js
index 544e135d78469b3d2364dec08008cfdeeea14846..8b4993c3e28145bf4ed63fcd53e918cdc71b64b3 100644
--- a/src/components/CanvasAnnotations.js
+++ b/src/components/CanvasAnnotations.js
@@ -1,11 +1,10 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import clsx from 'clsx';
-import Chip from '@material-ui/core/Chip';
-import MenuList from '@material-ui/core/MenuList';
-import MenuItem from '@material-ui/core/MenuItem';
-import ListItemText from '@material-ui/core/ListItemText';
-import Typography from '@material-ui/core/Typography';
+import Chip from '@mui/material/Chip';
+import MenuList from '@mui/material/MenuList';
+import MenuItem from '@mui/material/MenuItem';
+import ListItemText from '@mui/material/ListItemText';
+import Typography from '@mui/material/Typography';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import { ScrollTo } from './ScrollTo';
 
@@ -58,7 +57,7 @@ export class CanvasAnnotations extends Component {
   */
   render() {
     const {
-      annotations, classes, index, label, selectedAnnotationId, t, totalSize,
+      annotations, index, label, selectedAnnotationId, t, totalSize,
       listContainerComponent, htmlSanitizationRuleSet, hoveredAnnotationIds,
       containerRef,
     } = this.props;
@@ -66,53 +65,50 @@ export class CanvasAnnotations extends Component {
 
     return (
       <>
-        <Typography className={classes.sectionHeading} variant="overline">
+        <Typography sx={{ paddingLeft: 2, paddingRight: 1, paddingTop: 2 }} variant="overline">
           {t('annotationCanvasLabel', { context: `${index + 1}/${totalSize}`, label })}
         </Typography>
         <MenuList autoFocusItem variant="selectedMenu">
-          {
-            annotations.map(annotation => (
+          {annotations.map((annotation) => (
+            <ScrollTo
+              containerRef={containerRef}
+              key={`${annotation.id}-scroll`}
+              offsetTop={96} // offset for the height of the form above
+              scrollTo={selectedAnnotationId === annotation.id}
+              selected={selectedAnnotationId === annotation.id}
+            >
               <MenuItem
-                button
                 component={listContainerComponent}
-                className={clsx(
-                  classes.annotationListItem,
-                  {
-                    [classes.hovered]: hoveredAnnotationIds.includes(annotation.id),
+                variant="multiline"
+                divider
+                sx={{
+                  '&:hover,&:focus': {
+                    backgroundColor: 'action.hover',
                   },
-                )}
+                  backgroundColor: hoveredAnnotationIds.includes(annotation.id) ? 'action.hover' : '',
+                }}
                 key={annotation.id}
                 annotationid={annotation.id}
-                selected={selectedAnnotationId === annotation.id}
-                onClick={e => this.handleClick(e, annotation)}
+                onClick={(e) => this.handleClick(e, annotation)}
                 onFocus={() => this.handleAnnotationHover(annotation)}
                 onBlur={this.handleAnnotationBlur}
                 onMouseEnter={() => this.handleAnnotationHover(annotation)}
                 onMouseLeave={this.handleAnnotationBlur}
               >
-                <ScrollTo
-                  containerRef={containerRef}
-                  key={`${annotation.id}-scroll`}
-                  offsetTop={96} // offset for the height of the form above
-                  scrollTo={selectedAnnotationId === annotation.id}
-                >
-                  <ListItemText primaryTypographyProps={{ variant: 'body2' }}>
-                    <SanitizedHtml
-                      ruleSet={htmlSanitizationRuleSet}
-                      htmlString={annotation.content}
-                    />
-                    <div>
-                      {
-                        annotation.tags.map(tag => (
-                          <Chip size="small" variant="outlined" label={tag} id={tag} className={classes.chip} key={tag.toString()} />
-                        ))
-                      }
-                    </div>
-                  </ListItemText>
-                </ScrollTo>
+                <ListItemText
+                  primaryTypographyProps={{ variant: 'body2' }}
+                  primary={
+                    <SanitizedHtml ruleSet={htmlSanitizationRuleSet} htmlString={annotation.content} />
+                  }
+                  secondary={
+                    annotation.tags.map((tag) => (
+                      <Chip component="span" size="small" variant="outlined" label={tag} id={tag} key={tag.toString()} />
+                    ))
+                  }
+                />
               </MenuItem>
-            ))
-          }
+            </ScrollTo>
+          ))}
         </MenuList>
       </>
     );
@@ -126,7 +122,6 @@ CanvasAnnotations.propTypes = {
       id: PropTypes.string.isRequired,
     }),
   ),
-  classes: PropTypes.objectOf(PropTypes.string),
   containerRef: PropTypes.oneOfType([
     PropTypes.func,
     PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
@@ -146,7 +141,6 @@ CanvasAnnotations.propTypes = {
 };
 CanvasAnnotations.defaultProps = {
   annotations: [],
-  classes: {},
   containerRef: undefined,
   hoveredAnnotationIds: [],
   htmlSanitizationRuleSet: 'iiif',
diff --git a/src/components/CanvasInfo.js b/src/components/CanvasInfo.js
index 93f4b31243d6b9e5b33632085d6b53c4b3e3207c..39ec1a03fd29aeb92d23f60e2be2d672427bac5b 100644
--- a/src/components/CanvasInfo.js
+++ b/src/components/CanvasInfo.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
+import Typography from '@mui/material/Typography';
 import CollapsibleSection from '../containers/CollapsibleSection';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import { LabelValueMetadata } from './LabelValueMetadata';
diff --git a/src/components/CanvasLayers.js b/src/components/CanvasLayers.js
index 99e400ca2573b8237ef3137fa27712764c9b6d81..87f78490043e3700c25a0572f61f996372b18e7e 100644
--- a/src/components/CanvasLayers.js
+++ b/src/components/CanvasLayers.js
@@ -1,23 +1,36 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import clsx from 'clsx';
+import { styled } from '@mui/material/styles';
 import { v4 as uuid } from 'uuid';
-import Input from '@material-ui/core/Input';
-import InputAdornment from '@material-ui/core/InputAdornment';
-import List from '@material-ui/core/List';
-import ListItem from '@material-ui/core/ListItem';
-import Slider from '@material-ui/core/Slider';
-import Tooltip from '@material-ui/core/Tooltip';
-import DragHandleIcon from '@material-ui/icons/DragHandleSharp';
-import MoveToTopIcon from '@material-ui/icons/VerticalAlignTopSharp';
-import VisibilityIcon from '@material-ui/icons/VisibilitySharp';
-import VisibilityOffIcon from '@material-ui/icons/VisibilityOffSharp';
-import OpacityIcon from '@material-ui/icons/OpacitySharp';
-import Typography from '@material-ui/core/Typography';
+import Input from '@mui/material/Input';
+import InputAdornment from '@mui/material/InputAdornment';
+import List from '@mui/material/List';
+import ListItem from '@mui/material/ListItem';
+import Slider from '@mui/material/Slider';
+import Tooltip from '@mui/material/Tooltip';
+import DragHandleIcon from '@mui/icons-material/DragHandleSharp';
+import MoveToTopIcon from '@mui/icons-material/VerticalAlignTopSharp';
+import VisibilityIcon from '@mui/icons-material/VisibilitySharp';
+import VisibilityOffIcon from '@mui/icons-material/VisibilityOffSharp';
+import OpacityIcon from '@mui/icons-material/OpacitySharp';
+import Typography from '@mui/material/Typography';
 import { DragDropContext, Droppable, Draggable } from '@hello-pangea/dnd';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import IIIFThumbnail from '../containers/IIIFThumbnail';
 
+const StyledDragHandle = styled('div')(({ theme }) => ({
+  alignItems: 'center',
+  borderRight: `0.5px solid ${theme.palette.divider}`,
+  display: 'flex',
+  flex: 1,
+  flexDirection: 'row',
+  marginBottom: theme.spacing(-2),
+  marginRight: theme.spacing(1),
+  marginTop: theme.spacing(-2),
+  maxWidth: theme.spacing(3),
+  width: theme.spacing(3),
+}));
+
 /** */
 const reorder = (list, startIndex, endIndex) => {
   const result = Array.from(list);
@@ -116,7 +129,6 @@ export class CanvasLayers extends Component {
   /** @private */
   renderLayer(resource, index) {
     const {
-      classes,
       layerMetadata,
       t,
     } = this.props;
@@ -136,10 +148,12 @@ export class CanvasLayers extends Component {
             maxHeight={height}
             maxWidth={width}
             resource={resource}
-            classes={{ image: classes.image, root: classes.thumbnail }}
+            border
           />
           <Typography
-            className={classes.label}
+            sx={{
+              paddingLeft: 1,
+            }}
             component="div"
             variant="body1"
           >
@@ -158,10 +172,21 @@ export class CanvasLayers extends Component {
         </div>
         <div style={{ alignItems: 'center', display: 'flex' }}>
           <Tooltip title={t('layer_opacity')}>
-            <OpacityIcon className={classes.opacityIcon} color={layer.visibility ? 'inherit' : 'disabled'} fontSize="small" />
+            <OpacityIcon sx={{ marginRight: 0.5 }} color={layer.visibility ? 'inherit' : 'disabled'} fontSize="small" />
           </Tooltip>
           <Input
-            classes={{ input: classes.opacityInput }}
+            sx={{
+              'MuiInput-input': {
+                '&::-webkit-outer-spin-button,&::-webkit-inner-spin-button': {
+                  margin: 0,
+                  WebkitAppearance: 'none',
+                },
+                MozAppearance: 'textfield',
+                textAlign: 'right',
+                typography: 'caption',
+                width: '3ch',
+              },
+            }}
             disabled={!layer.visibility}
             value={Math.round(layer.opacity * 100)}
             type="number"
@@ -174,7 +199,11 @@ export class CanvasLayers extends Component {
             }}
           />
           <Slider
-            className={classes.slider}
+            sx={{
+              marginLeft: 2,
+              marginRight: 2,
+              maxWidth: 150,
+            }}
             disabled={!layer.visibility}
             value={layer.opacity * 100}
             onChange={(e, value) => this.handleOpacityChange(resource.id, value)}
@@ -187,7 +216,6 @@ export class CanvasLayers extends Component {
   /** @private */
   renderDraggableLayer(resource, index) {
     const {
-      classes,
       t,
     } = this.props;
 
@@ -198,20 +226,33 @@ export class CanvasLayers extends Component {
             ref={provided.innerRef}
             {...provided.draggableProps}
             component="li"
-            className={clsx(
-              classes.listItem,
-              {
-                [classes.dragging]: snapshot.isDragging,
-              },
-            )}
+            divider
+            sx={{
+              alignItems: 'stretch',
+              cursor: 'pointer',
+              paddingBottom: 2,
+              paddingRight: 2,
+              paddingTop: 2,
+              ...(snapshot.isDragging && {
+                backgroundColor: 'action.hover',
+              }),
+            }}
             disableGutters
             key={resource.id}
           >
-            <div {...provided.dragHandleProps} className={classes.dragHandle}>
+            <StyledDragHandle
+              {...provided.dragHandleProps}
+              sx={{
+                '&:hover': {
+                  backgroundColor: snapshot.isDragging ? 'action.selected' : 'action.hover',
+                },
+                backgroundColor: snapshot.isDragging ? 'action.selected' : 'shades.light',
+              }}
+            >
               <Tooltip title={t('layer_move')}>
                 <DragHandleIcon />
               </Tooltip>
-            </div>
+            </StyledDragHandle>
             {this.renderLayer(resource, index)}
           </ListItem>
         )}
@@ -222,7 +263,6 @@ export class CanvasLayers extends Component {
   /** */
   render() {
     const {
-      classes,
       index,
       label,
       layers,
@@ -233,7 +273,14 @@ export class CanvasLayers extends Component {
     return (
       <>
         { totalSize > 1 && (
-          <Typography className={classes.sectionHeading} variant="overline">
+          <Typography
+            sx={{
+              paddingLeft: 1,
+              paddingRight: 1,
+              paddingTop: 2,
+            }}
+            variant="overline"
+          >
             {t('annotationCanvasLabel', { context: `${index + 1}/${totalSize}`, label })}
           </Typography>
         )}
@@ -241,7 +288,9 @@ export class CanvasLayers extends Component {
           <Droppable droppableId={this.droppableId}>
             {(provided, snapshot) => (
               <List
-                className={classes.list}
+                sx={{
+                  paddingTop: 0,
+                }}
                 {...provided.droppableProps}
                 ref={provided.innerRef}
               >
@@ -262,7 +311,6 @@ export class CanvasLayers extends Component {
 
 CanvasLayers.propTypes = {
   canvasId: PropTypes.string.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string),
   index: PropTypes.number.isRequired,
   label: PropTypes.string.isRequired,
   layerMetadata: PropTypes.objectOf(PropTypes.shape({
@@ -277,6 +325,5 @@ CanvasLayers.propTypes = {
 };
 
 CanvasLayers.defaultProps = {
-  classes: {},
   layerMetadata: undefined,
 };
diff --git a/src/components/ChangeThemeDialog.js b/src/components/ChangeThemeDialog.js
index 146071b855d6dc492722e5ebec2c9d8dc9da2d7a..17bd83171aab087aa3eafd88173a167674f1a3fa 100644
--- a/src/components/ChangeThemeDialog.js
+++ b/src/components/ChangeThemeDialog.js
@@ -1,16 +1,20 @@
 import { Component } from 'react';
 import {
-  Dialog,
   DialogTitle,
   ListItemIcon,
   ListItemText,
   MenuList,
   MenuItem,
-  Typography,
   DialogContent,
-} from '@material-ui/core';
-import PaletteIcon from '@material-ui/icons/PaletteSharp';
+} from '@mui/material';
+import PaletteIcon from '@mui/icons-material/PaletteSharp';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
+import { WorkspaceDialog } from './WorkspaceDialog';
+
+const ThemeIcon = styled(PaletteIcon, { name: 'ThemeIcon', slot: 'icon' })(({ theme }) => ({
+  color: '#BDBDBD',
+}));
 
 /**
  * a simple dialog providing the possibility to switch the theme
@@ -36,7 +40,6 @@ export class ChangeThemeDialog extends Component {
   /** */
   render() {
     const {
-      classes,
       handleClose,
       open,
       selectedTheme,
@@ -44,42 +47,34 @@ export class ChangeThemeDialog extends Component {
       themeIds,
     } = this.props;
     return (
-      <Dialog
-        onClose={handleClose}
-        open={open}
-      >
-        <DialogTitle disableTypography>
-          <Typography variant="h2">
-            {t('changeTheme')}
-          </Typography>
+      <WorkspaceDialog onClose={handleClose} open={open} variant="menu">
+        <DialogTitle>
+          {t('changeTheme')}
         </DialogTitle>
-        <DialogContent className={classes.dialogContent}>
+        <DialogContent>
           <MenuList autoFocusItem>
-            {
-              themeIds.map(value => (
-                <MenuItem
-                  key={value}
-                  className={classes.listitem}
-                  onClick={() => { this.handleThemeChange(value); }}
-                  selected={value === selectedTheme}
-                  value={value}
-                >
-                  <ListItemIcon>
-                    <PaletteIcon className={classes[value]} />
-                  </ListItemIcon>
-                  <ListItemText>{t(value)}</ListItemText>
-                </MenuItem>
-              ))
-            }
+            {themeIds.map((value) => (
+              <MenuItem
+                key={value}
+                className="listitem"
+                onClick={() => this.handleThemeChange(value)}
+                selected={value === selectedTheme}
+                value={value}
+              >
+                <ListItemIcon>
+                  <ThemeIcon ownerState={{ value }} />
+                </ListItemIcon>
+                <ListItemText>{t(value)}</ListItemText>
+              </MenuItem>
+            ))}
           </MenuList>
         </DialogContent>
-      </Dialog>
+      </WorkspaceDialog>
     );
   }
 }
 
 ChangeThemeDialog.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   handleClose: PropTypes.func.isRequired,
   open: PropTypes.bool,
   selectedTheme: PropTypes.string.isRequired,
diff --git a/src/components/CollapsibleSection.js b/src/components/CollapsibleSection.js
index 8773c9b6d951d9c14ccbb6ec3000db74bd032811..04c5f7d5137bd28367c52808d6663c1f98b60d05 100644
--- a/src/components/CollapsibleSection.js
+++ b/src/components/CollapsibleSection.js
@@ -1,9 +1,10 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import KeyboardArrowDown from '@material-ui/icons/KeyboardArrowDownSharp';
-import KeyboardArrowUp from '@material-ui/icons/KeyboardArrowUpSharp';
-import MiradorMenuButton from '../containers/MiradorMenuButton';
+import Typography from '@mui/material/Typography';
+import Accordion from '@mui/material/Accordion';
+import AccordionDetails from '@mui/material/AccordionDetails';
+import AccordionSummary from '@mui/material/AccordionSummary';
+import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
 
 /**
  * CollapsableSection ~
@@ -14,14 +15,12 @@ export class CollapsibleSection extends Component {
     super(props);
 
     this.state = { open: true };
-    this.toggleSection = this.toggleSection.bind(this);
+    this.handleChange = this.handleChange.bind(this);
   }
 
-  /** */
-  toggleSection() {
-    const { open } = this.state;
-
-    this.setState({ open: !open });
+  /** Control the accordion state so we can provide aria labeling */
+  handleChange(event, isExpanded) {
+    this.setState({ open: isExpanded });
   }
 
   /**
@@ -29,45 +28,28 @@ export class CollapsibleSection extends Component {
   */
   render() {
     const {
-      children, classes, id, label, t,
+      children, id, label, t,
     } = this.props;
     const { open } = this.state;
 
     return (
-      <>
-        <div className={classes.container}>
-          <Typography
-            className={classes.heading}
-            id={id}
-            onClick={this.toggleSection}
-            variant="overline"
-            component="h4"
-          >
+      <Accordion id={id} elevation={0} expanded={open} onChange={this.handleChange} disableGutters square variant="compact">
+        <AccordionSummary id={`${id}-header`} aria-controls={`${id}-content`} aria-label={t(open ? 'collapseSection' : 'expandSection', { section: label })} expandIcon={<ExpandMoreIcon />}>
+          <Typography variant="overline" component="h4">
             {label}
           </Typography>
-          <MiradorMenuButton
-            aria-label={
-              t(
-                open ? 'collapseSection' : 'expandSection',
-                { section: label },
-              )
-            }
-            aria-expanded={open}
-            className={classes.button}
-            onClick={this.toggleSection}
-          >
-            {open ? <KeyboardArrowUp /> : <KeyboardArrowDown />}
-          </MiradorMenuButton>
-        </div>
-        {open && children}
-      </>
+        </AccordionSummary>
+        <AccordionDetails>
+          {children}
+        </AccordionDetails>
+      </Accordion>
     );
   }
 }
 
 CollapsibleSection.propTypes = {
   children: PropTypes.node.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
+
   id: PropTypes.string.isRequired,
   label: PropTypes.string.isRequired,
   t: PropTypes.func.isRequired,
diff --git a/src/components/CollectionDialog.js b/src/components/CollectionDialog.js
index 3dc335ba9b3bfb13ef74ff4a66fc7348b59c129d..75fa641f14cc78bced024ae8cb14736effba434c 100644
--- a/src/components/CollectionDialog.js
+++ b/src/components/CollectionDialog.js
@@ -10,15 +10,32 @@ import {
   MenuList,
   MenuItem,
   Typography,
-} from '@material-ui/core';
-import ArrowBackIcon from '@material-ui/icons/ArrowBackSharp';
-import Skeleton from '@material-ui/lab/Skeleton';
+} from '@mui/material';
+import ArrowBackIcon from '@mui/icons-material/ArrowBackSharp';
+import Skeleton from '@mui/material/Skeleton';
+import { styled } from '@mui/material/styles';
 import asArray from '../lib/asArray';
 import { LabelValueMetadata } from './LabelValueMetadata';
 import CollapsibleSection from '../containers/CollapsibleSection';
 import ScrollIndicatedDialogContent from '../containers/ScrollIndicatedDialogContent';
 import ManifestInfo from '../containers/ManifestInfo';
 
+const StyledScrollIndicatedDialogContent = styled(ScrollIndicatedDialogContent)(() => ({
+  padding: (theme) => theme.spacing(1),
+}));
+
+const StyledCollectionMetadata = styled('div')(() => ({
+  '& .MuiPaper-root': {
+    background: 'transparent',
+  },
+  padding: (theme) => theme.spacing(2),
+}));
+
+const StyledCollectionFilter = styled('div')(() => ({
+  padding: (theme) => theme.spacing(2),
+  paddingTop: 0,
+}));
+
 /**
  * a dialog providing the possibility to select the collection
  */
@@ -108,37 +125,27 @@ export class CollectionDialog extends Component {
 
   /** */
   placeholder() {
-    const { classes } = this.props;
-
     return (
       <Dialog
-        className={classes.dialog}
+        variant="contained"
         onClose={this.hideDialog}
         open
         container={this.dialogContainer()}
-        BackdropProps={this.backdropProps()}
       >
-        <DialogTitle id="select-collection" disableTypography>
-          <Skeleton className={classes.placeholder} variant="text" />
+        <DialogTitle id="select-collection">
+          <Skeleton variant="text" />
         </DialogTitle>
         <ScrollIndicatedDialogContent>
-          <Skeleton className={classes.placeholder} variant="text" />
-          <Skeleton className={classes.placeholder} variant="text" />
+          <Skeleton variant="text" />
+          <Skeleton variant="text" />
         </ScrollIndicatedDialogContent>
       </Dialog>
     );
   }
 
-  /** */
-  backdropProps() {
-    const { classes } = this.props;
-    return { classes: { root: classes.dialog } };
-  }
-
   /** */
   render() {
     const {
-      classes,
       collection,
       error,
       isMultipart,
@@ -173,21 +180,20 @@ export class CollectionDialog extends Component {
 
     return (
       <Dialog
-        className={classes.dialog}
+        variant="contained"
         onClose={this.hideDialog}
         container={this.dialogContainer()}
-        BackdropProps={this.backdropProps()}
         open
       >
-        <DialogTitle id="select-collection" disableTypography>
+        <DialogTitle id="select-collection">
           <Typography component="div" variant="overline">
             { t(isMultipart ? 'multipartCollection' : 'collection') }
           </Typography>
-          <Typography variant="h3">
+          <Typography component="div" variant="h3">
             {CollectionDialog.getUseableLabel(manifest)}
           </Typography>
         </DialogTitle>
-        <ScrollIndicatedDialogContent className={classes.dialogContent}>
+        <StyledScrollIndicatedDialogContent>
           { collection && (
             <Button
               startIcon={<ArrowBackIcon />}
@@ -197,7 +203,7 @@ export class CollectionDialog extends Component {
             </Button>
           )}
 
-          <div className={classes.collectionMetadata}>
+          <StyledCollectionMetadata>
             <ManifestInfo manifestId={manifest.id} />
             <CollapsibleSection
               id="select-collection-rights"
@@ -221,15 +227,15 @@ export class CollectionDialog extends Component {
                 )
               }
             </CollapsibleSection>
-          </div>
-          <div className={classes.collectionFilter}>
+          </StyledCollectionMetadata>
+          <StyledCollectionFilter>
             {manifest.getTotalCollections() > 0 && (
               <Chip clickable color={currentFilter === 'collections' ? 'primary' : 'default'} onClick={() => this.setFilter('collections')} label={t('totalCollections', { count: manifest.getTotalCollections() })} />
             )}
             {manifest.getTotalManifests() > 0 && (
               <Chip clickable color={currentFilter === 'manifests' ? 'primary' : 'default'} onClick={() => this.setFilter('manifests')} label={t('totalManifests', { count: manifest.getTotalManifests() })} />
             )}
-          </div>
+          </StyledCollectionFilter>
           { currentFilter === 'collections' && (
             <MenuList>
               {
@@ -237,7 +243,7 @@ export class CollectionDialog extends Component {
                   <MenuItem
                     key={c.id}
                     onClick={() => { this.selectCollection(c); }}
-                    className={classes.collectionItem}
+                    variant="multiline"
                   >
                     {CollectionDialog.getUseableLabel(c)}
                   </MenuItem>
@@ -252,7 +258,7 @@ export class CollectionDialog extends Component {
                   <MenuItem
                     key={m.id}
                     onClick={() => { this.selectManifest(m); }}
-                    className={classes.collectionItem}
+                    variant="multiline"
                   >
                     {CollectionDialog.getUseableLabel(m)}
                   </MenuItem>
@@ -260,7 +266,7 @@ export class CollectionDialog extends Component {
               }
             </MenuList>
           )}
-        </ScrollIndicatedDialogContent>
+        </StyledScrollIndicatedDialogContent>
         <DialogActions>
           <Button onClick={this.hideDialog}>
             {t('close')}
@@ -273,7 +279,6 @@ export class CollectionDialog extends Component {
 
 CollectionDialog.propTypes = {
   addWindow: PropTypes.func.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   collection: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   collectionPath: PropTypes.arrayOf(PropTypes.string),
   container: PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
diff --git a/src/components/CollectionInfo.js b/src/components/CollectionInfo.js
index 4cd3d3961e5bf804d24687e703199d8b0e6b4d61..ead13a799bbd0b346fa4f1e5b0a4124c395e2700 100644
--- a/src/components/CollectionInfo.js
+++ b/src/components/CollectionInfo.js
@@ -1,8 +1,8 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import Typography from '@material-ui/core/Typography';
-import ViewListIcon from '@material-ui/icons/ViewListSharp';
+import Button from '@mui/material/Button';
+import Typography from '@mui/material/Typography';
+import ViewListIcon from '@mui/icons-material/ViewListSharp';
 import CollapsibleSection from '../containers/CollapsibleSection';
 
 /**
diff --git a/src/components/CompanionArea.js b/src/components/CompanionArea.js
index bdf7610b6e5e06f26505d86cce8b4622818ed98b..b73350d644204d6e36febe1491f0be5ff69431a9 100644
--- a/src/components/CompanionArea.js
+++ b/src/components/CompanionArea.js
@@ -1,19 +1,58 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Slide from '@material-ui/core/Slide';
-import ArrowLeftIcon from '@material-ui/icons/ArrowLeftSharp';
-import ArrowRightIcon from '@material-ui/icons/ArrowRightSharp';
+import { styled } from '@mui/material/styles';
+import Slide from '@mui/material/Slide';
+import ArrowLeftIcon from '@mui/icons-material/ArrowLeftSharp';
+import ArrowRightIcon from '@mui/icons-material/ArrowRightSharp';
+import classNames from 'classnames';
 import CompanionWindowFactory from '../containers/CompanionWindowFactory';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import ns from '../config/css-ns';
 
+const Root = styled('div', { name: 'CompanionArea', slot: 'root' })(({ ownerState, theme }) => ({
+  display: 'flex',
+  minHeight: 0,
+  position: 'relative',
+  zIndex: theme.zIndex.appBar - 2,
+  ...((ownerState.position === 'bottom' || ownerState.position === 'far-bottom') && {
+    flexDirection: 'column',
+    width: '100%',
+  }),
+}));
+
+const Container = styled('div', { name: 'CompanionArea', slot: 'container' })(({ ownerState }) => ({
+  display: ownerState?.companionAreaOpen ? 'flex' : 'none',
+  ...((ownerState?.position === 'bottom' || ownerState?.position === 'far-bottom') && {
+    flexDirection: 'column',
+    width: '100%',
+  }),
+  ...((ownerState?.position === 'left' && (ownerState?.companionWindowIds && ownerState.companionWindowIds.length > 0)) && {
+    minWidth: '235px',
+  }),
+}));
+
+const StyledToggle = styled('div', { name: 'CompanionArea', slot: 'toggle' })(({ theme }) => ({
+  alignItems: 'center',
+  backgroundColor: theme.palette.background.paper,
+  border: `1px solid ${theme.palette.mode === 'dark' ? theme.palette.divider : theme.palette.shades?.dark}`,
+  borderInlineStart: 0,
+  borderRadius: 0,
+  display: 'inline-flex',
+  height: '48px',
+  left: '100%',
+  marginTop: '1rem',
+  overflow: 'hidden',
+  padding: 2,
+  position: 'absolute',
+  width: '23px',
+  zIndex: theme.zIndex.drawer,
+}));
+
 /** */
 export class CompanionArea extends Component {
   /** */
   areaLayoutClass() {
-    const {
-      classes, position,
-    } = this.props;
+    const { classes, position } = this.props;
 
     return (position === 'bottom' || position === 'far-bottom') ? classes.horizontal : null;
   }
@@ -51,45 +90,44 @@ export class CompanionArea extends Component {
   /** */
   render() {
     const {
-      classes, companionWindowIds, companionAreaOpen, setCompanionAreaOpen,
+      className,
+      companionWindowIds, companionAreaOpen, setCompanionAreaOpen,
       position, sideBarOpen, t, windowId,
     } = this.props;
-
+    const classes = classNames(this.areaLayoutClass(), ns(`companion-area-${position}`), className);
     return (
-      <div className={[classes.root, this.areaLayoutClass(), ns(`companion-area-${position}`)].join(' ')}>
+      <Root ownerState={this.props} className={classes}>
         <Slide in={companionAreaOpen} direction={this.slideDirection()}>
-          <div className={[ns('companion-windows'), companionWindowIds.length > 0 && classes[position], this.areaLayoutClass()].join(' ')} style={{ display: companionAreaOpen ? 'flex' : 'none' }}>
-            {
-              companionWindowIds.map(id => (
-                <CompanionWindowFactory id={id} key={id} windowId={windowId} />
-              ))
-            }
-          </div>
+          <Container
+            ownerState={this.props}
+            className={`${ns('companion-windows')}`}
+          >
+            {companionWindowIds.map((id) => (
+              <CompanionWindowFactory id={id} key={id} windowId={windowId} />
+            ))}
+          </Container>
         </Slide>
-        {
-          setCompanionAreaOpen && position === 'left' && sideBarOpen && companionWindowIds.length > 0
-          && (
-            <div className={classes.toggle}>
-              <MiradorMenuButton
-                aria-expanded={companionAreaOpen}
-                aria-label={companionAreaOpen ? t('collapseSidePanel') : t('expandSidePanel')}
-                className={classes.toggleButton}
-                key={companionAreaOpen ? 'collapse' : 'expand'}
-                onClick={() => { setCompanionAreaOpen(windowId, !companionAreaOpen); }}
-                TooltipProps={{ placement: 'right' }}
-              >
-                {this.collapseIcon()}
-              </MiradorMenuButton>
-            </div>
-          )
-        }
-      </div>
+        {setCompanionAreaOpen && position === 'left' && sideBarOpen && companionWindowIds.length > 0 && (
+        <StyledToggle>
+          <MiradorMenuButton
+            aria-expanded={companionAreaOpen}
+            aria-label={companionAreaOpen ? t('collapseSidePanel') : t('expandSidePanel')}
+            edge="start"
+            onClick={() => { setCompanionAreaOpen(windowId, !companionAreaOpen); }}
+            TooltipProps={{ placement: 'right' }}
+          >
+            {this.collapseIcon()}
+          </MiradorMenuButton>
+        </StyledToggle>
+        )}
+      </Root>
     );
   }
 }
 
 CompanionArea.propTypes = {
   classes: PropTypes.objectOf(PropTypes.string),
+  className: PropTypes.string,
   companionAreaOpen: PropTypes.bool.isRequired,
   companionWindowIds: PropTypes.arrayOf(PropTypes.string).isRequired,
   direction: PropTypes.string.isRequired,
@@ -102,6 +140,7 @@ CompanionArea.propTypes = {
 
 CompanionArea.defaultProps = {
   classes: {},
+  className: undefined,
   setCompanionAreaOpen: () => {},
   sideBarOpen: false,
 };
diff --git a/src/components/CompanionWindow.js b/src/components/CompanionWindow.js
index 9cf4fe29a4e55f7576b7b3a6af724b0a87a4297a..8a31ce2fb1b8a4b1127bcdec3e1d9566b7911241 100644
--- a/src/components/CompanionWindow.js
+++ b/src/components/CompanionWindow.js
@@ -1,15 +1,25 @@
 import { Children, cloneElement, Component } from 'react';
 import PropTypes from 'prop-types';
-import CloseIcon from '@material-ui/icons/CloseSharp';
-import OpenInNewIcon from '@material-ui/icons/OpenInNewSharp';
-import MoveIcon from '@material-ui/icons/DragIndicatorSharp';
-import Paper from '@material-ui/core/Paper';
-import Typography from '@material-ui/core/Typography';
-import Toolbar from '@material-ui/core/Toolbar';
+import { styled } from '@mui/material/styles';
+import CloseIcon from '@mui/icons-material/CloseSharp';
+import OpenInNewIcon from '@mui/icons-material/OpenInNewSharp';
+import MoveIcon from '@mui/icons-material/DragIndicatorSharp';
+import Paper from '@mui/material/Paper';
+import Typography from '@mui/material/Typography';
+import Toolbar from '@mui/material/Toolbar';
 import { Rnd } from 'react-rnd';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import ns from '../config/css-ns';
 
+const Root = styled(Paper, { name: 'CompanionWindow', slot: 'root' })({});
+const StyledToolbar = styled(Toolbar, { name: 'CompanionWindow', slot: 'toolbar' })({});
+const StyledTitle = styled(Typography, { name: 'CompanionWindow', slot: 'title' })({});
+const StyledTitleControls = styled('div', { name: 'CompanionWindow', slot: 'controls' })({});
+const Contents = styled(Paper, { name: 'CompanionWindow', slot: 'contents' })({});
+const StyledRnd = styled(Rnd, { name: 'CompanionWindow', slot: 'resize' })({});
+const StyledPositionButton = styled(MiradorMenuButton, { name: 'CompanionWindow', slot: 'positionButton' })({});
+const StyledCloseButton = styled(MiradorMenuButton, { name: 'CompanionWindow', slot: 'closeButton' })({});
+
 /**
  * CompanionWindow
  */
@@ -69,7 +79,7 @@ export class CompanionWindow extends Component {
     const {
       ariaLabel, classes, paperClassName, onCloseClick, updateCompanionWindow, isDisplayed,
       position, t, title, children, titleControls, size,
-      defaultSidebarPanelWidth, defaultSidebarPanelHeight,
+      defaultSidebarPanelWidth, defaultSidebarPanelHeight, innerRef,
     } = this.props;
 
     const isBottom = (position === 'bottom' || position === 'far-bottom');
@@ -87,19 +97,21 @@ export class CompanionWindow extends Component {
     });
 
     return (
-      <Paper
-        className={[classes.root, position === 'bottom' ? classes.horizontal : classes.vertical, classes[`companionWindow-${position}`], ns(`companion-window-${position}`), paperClassName].join(' ')}
+      <Root
+        ownerState={this.props}
+        ref={innerRef}
         style={{
           display: isDisplayed ? null : 'none',
           order: position === 'left' ? -1 : null,
         }}
+        className={[ns(`companion-window-${position}`), paperClassName, position === 'bottom' ? classes.horizontal : classes.vertical].join(' ')}
         square
         component="aside"
         aria-label={ariaLabel || title}
       >
-        <Rnd
-          className={[classes.rnd]}
-          style={{ display: 'flex', position: 'relative' }}
+        <StyledRnd
+          style={{ display: 'inherit', position: 'inherit' }}
+          ownerState={this.props}
           default={{
             height: isBottom ? defaultSidebarPanelHeight : '100%',
             width: isBottom ? 'auto' : defaultSidebarPanelWidth,
@@ -110,18 +122,12 @@ export class CompanionWindow extends Component {
           minWidth={position === 'left' ? 235 : 100}
         >
 
-          <Toolbar
-            className={[
-              classes.toolbar,
-              classes.companionWindowHeader,
-              size.width < 370 ? classes.small : null,
-              ns('companion-window-header'),
-            ].join(' ')}
+          <StyledToolbar
+            variant="dense"
+            className={[ns('companion-window-header'), size.width < 370 ? classes.small : null].join(' ')}
             disableGutters
           >
-            <Typography variant="h3" className={classes.windowSideBarTitle}>
-              {title}
-            </Typography>
+            <StyledTitle variant="h3">{title}</StyledTitle>
             {
               position === 'left'
                 ? updateCompanionWindow
@@ -137,49 +143,50 @@ export class CompanionWindow extends Component {
                   <>
                     {
                       updateCompanionWindow && (
-                        <MiradorMenuButton
+                        <StyledPositionButton
                           aria-label={position === 'bottom' ? t('moveCompanionWindowToRight') : t('moveCompanionWindowToBottom')}
-                          className={classes.positionButton}
                           onClick={() => { updateCompanionWindow({ position: position === 'bottom' ? 'right' : 'bottom' }); }}
                         >
                           <MoveIcon />
-                        </MiradorMenuButton>
+                        </StyledPositionButton>
                       )
                     }
-                    <MiradorMenuButton
+                    <StyledCloseButton
+                      sx={{
+                        ...(size.width < 370 && {
+                          order: 'unset',
+                        }),
+                      }}
                       aria-label={t('closeCompanionWindow')}
-                      className={classes.closeButton}
                       onClick={onCloseClick}
                     >
                       <CloseIcon />
-                    </MiradorMenuButton>
+                    </StyledCloseButton>
                   </>
                 )
             }
             {
               titleControls && (
-                <div
-                  className={[
-                    classes.titleControls,
-                    isBottom
-                      ? classes.companionWindowTitleControlsBottom
-                      : classes.companionWindowTitleControls,
-                    ns('companion-window-title-controls'),
-                  ].join(' ')}
+                <StyledTitleControls
+                  ownerState={{ position }}
+                  sx={{
+                    order: isBottom || size.width < 370 ? 'unset' : 1000,
+                  }}
+                  className={ns('companion-window-title-controls')}
                 >
                   {titleControls}
-                </div>
+                </StyledTitleControls>
               )
             }
-          </Toolbar>
-          <Paper
-            className={[classes.content, ns('scrollto-scrollable')].join(' ')}
+          </StyledToolbar>
+          <Contents
+            className={ns('scrollto-scrollable')}
             elevation={0}
           >
             {childrenWithAdditionalProps}
-          </Paper>
-        </Rnd>
-      </Paper>
+          </Contents>
+        </StyledRnd>
+      </Root>
     );
   }
 }
@@ -187,10 +194,14 @@ export class CompanionWindow extends Component {
 CompanionWindow.propTypes = {
   ariaLabel: PropTypes.string,
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
+  classes: PropTypes.objectOf(PropTypes.string),
   defaultSidebarPanelHeight: PropTypes.number,
   defaultSidebarPanelWidth: PropTypes.number,
   direction: PropTypes.string.isRequired,
+  innerRef: PropTypes.oneOfType([
+    PropTypes.func,
+    PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
+  ]),
   isDisplayed: PropTypes.bool,
   onCloseClick: PropTypes.func,
   paperClassName: PropTypes.string,
@@ -208,8 +219,10 @@ CompanionWindow.propTypes = {
 CompanionWindow.defaultProps = {
   ariaLabel: undefined,
   children: undefined,
+  classes: {},
   defaultSidebarPanelHeight: 201,
   defaultSidebarPanelWidth: 235,
+  innerRef: undefined,
   isDisplayed: false,
   onCloseClick: () => {},
   paperClassName: '',
diff --git a/src/components/CompanionWindowSection.js b/src/components/CompanionWindowSection.js
new file mode 100644
index 0000000000000000000000000000000000000000..312164e85b190960ee6e0adab8b030d3146104fd
--- /dev/null
+++ b/src/components/CompanionWindowSection.js
@@ -0,0 +1,8 @@
+import { styled } from '@mui/material/styles';
+
+export const CompanionWindowSection = styled('div', { name: 'CompanionWindowSection', slot: 'root' })(({ theme }) => ({
+  paddingBlockEnd: theme.spacing(1),
+  paddingBlockStart: theme.spacing(2),
+  paddingInlineEnd: theme.spacing(1),
+  paddingInlineStart: theme.spacing(2),
+}));
diff --git a/src/components/ErrorContent.js b/src/components/ErrorContent.js
index d6e996ece45f003bfe8bb20b09483454a870409d..b179f769530332094f51158ba1dd664ee310f3c1 100644
--- a/src/components/ErrorContent.js
+++ b/src/components/ErrorContent.js
@@ -1,19 +1,34 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Accordion from '@material-ui/core/Accordion';
-import AccordionSummary from '@material-ui/core/AccordionSummary';
-import AccordionDetails from '@material-ui/core/AccordionDetails';
-import Typography from '@material-ui/core/Typography';
-import Alert from '@material-ui/lab/Alert';
-import ExpandMoreIcon from '@material-ui/icons/ExpandMore';
+import { styled } from '@mui/material/styles';
+import Accordion from '@mui/material/Accordion';
+import AccordionSummary from '@mui/material/AccordionSummary';
+import AccordionDetails from '@mui/material/AccordionDetails';
+import Stack from '@mui/material/Stack';
+import Alert from '@mui/material/Alert';
+import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
 import { PluginHook } from './PluginHook';
 
+const ErrorStackTrace = styled('pre', { name: 'ErrorContent', slot: 'stacktrace' })({
+  overflowY: 'scroll',
+});
+
+const ErrorMetadata = styled('pre', { name: 'ErrorContent', slot: 'metadata' })({
+  height: '100px',
+  overflowY: 'scroll',
+});
+
+const InlineAccordion = styled(Accordion, { name: 'ErrorContent', slot: 'accordion' })({
+  backgroundColor: 'inherit',
+  color: 'inherit',
+  margin: 0,
+});
+
 /** */
 export class ErrorContent extends Component {
   /** */
   render() {
     const {
-      classes,
       error,
       metadata,
       showJsError,
@@ -23,34 +38,28 @@ export class ErrorContent extends Component {
     if (!showJsError) return null;
 
     return (
-      <>
-        <Alert elevation={6} variant="filled" severity="error">
-          {t('errorDialogTitle')}
-        </Alert>
-
+      <Alert elevation={6} variant="filled" severity="error">
+        {t('errorDialogTitle')}
         {showJsError && (
-          <Accordion square className={classes.alert}>
-            <AccordionSummary
-              expandIcon={<ExpandMoreIcon />}
-            >
-              <Typography>{t('jsError', { message: error.message, name: error.name })}</Typography>
+          <InlineAccordion elevation={2} square>
+            <AccordionSummary sx={{ marginInlineStart: '-1rem' }} expandIcon={<ExpandMoreIcon sx={{ color: '#fff' }} />}>
+              {t('jsError', { message: error.message, name: error.name })}
             </AccordionSummary>
-            <AccordionDetails className={classes.details}>
-              <pre>{ t('jsStack', { stack: error.stack }) }</pre>
-              { metadata && (
-                <pre>{JSON.stringify(metadata, null, 2)}</pre>
-              )}
+            <AccordionDetails>
+              <Stack>
+                <ErrorStackTrace>{t('jsStack', { stack: error.stack })}</ErrorStackTrace>
+                {metadata && <ErrorMetadata>{JSON.stringify(metadata, null, 2)}</ErrorMetadata>}
+              </Stack>
             </AccordionDetails>
-          </Accordion>
+          </InlineAccordion>
         )}
         <PluginHook {...this.props} />
-      </>
+      </Alert>
     );
   }
 }
 
 ErrorContent.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   error: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
   metadata: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   showJsError: PropTypes.bool,
diff --git a/src/components/ErrorDialog.js b/src/components/ErrorDialog.js
index e9f5b342ae0a07d856a68df8d7a9927eced27d70..5828a6b98b4f21249f9a0e808945c418f82ce055 100644
--- a/src/components/ErrorDialog.js
+++ b/src/components/ErrorDialog.js
@@ -1,10 +1,10 @@
 import { Component } from 'react';
-import Dialog from '@material-ui/core/Dialog';
-import DialogContent from '@material-ui/core/DialogContent';
-import DialogTitle from '@material-ui/core/DialogTitle';
+import Dialog from '@mui/material/Dialog';
+import DialogContent from '@mui/material/DialogContent';
+import DialogTitle from '@mui/material/DialogTitle';
 import PropTypes from 'prop-types';
-import { DialogActions, DialogContentText, Typography } from '@material-ui/core';
-import Button from '@material-ui/core/Button';
+import { DialogActions, DialogContentText } from '@mui/material';
+import Button from '@mui/material/Button';
 import { isUndefined } from 'lodash';
 
 /**
@@ -28,8 +28,8 @@ export class ErrorDialog extends Component {
         onClose={() => removeError(error.id)}
         open={hasError}
       >
-        <DialogTitle id="error-dialog-title" disableTypography>
-          <Typography variant="h2">{t('errorDialogTitle')}</Typography>
+        <DialogTitle id="error-dialog-title">
+          {t('errorDialogTitle')}
         </DialogTitle>
         <DialogContent>
           <DialogContentText variant="body2" noWrap color="inherit">
diff --git a/src/components/FullScreenButton.js b/src/components/FullScreenButton.js
index 2038fa9bc09f1675a5b0c9d31ea565f308419eee..0aabc6722ef9a9192bfd431c1464a9ea05f7e380 100644
--- a/src/components/FullScreenButton.js
+++ b/src/components/FullScreenButton.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
-import FullscreenIcon from '@material-ui/icons/FullscreenSharp';
-import FullscreenExitIcon from '@material-ui/icons/FullscreenExitSharp';
+import FullscreenIcon from '@mui/icons-material/FullscreenSharp';
+import FullscreenExitIcon from '@mui/icons-material/FullscreenExitSharp';
 import PropTypes from 'prop-types';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import FullScreenContext from '../contexts/FullScreenContext';
diff --git a/src/components/GalleryView.js b/src/components/GalleryView.js
index 66d0348db419d1cf8e1ef5925c7cdede472768ed..1738f56baae0c98c0373d9d84d5e9272ecebccfe 100644
--- a/src/components/GalleryView.js
+++ b/src/components/GalleryView.js
@@ -1,8 +1,20 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Paper from '@material-ui/core/Paper';
+import { styled } from '@mui/material/styles';
+import Paper from '@mui/material/Paper';
 import GalleryViewThumbnail from '../containers/GalleryViewThumbnail';
 
+const Root = styled(Paper, { name: 'GalleryView', slot: 'root' })(({ theme }) => ({
+  alignItems: 'flex-start',
+  display: 'flex',
+  flexDirection: 'row',
+  flexWrap: 'wrap',
+  overflowX: 'hidden',
+  overflowY: 'scroll',
+  padding: '50px 0 50px 20px',
+  width: '100%',
+}));
+
 /**
  * Renders a GalleryView overview of the manifest.
  */
@@ -12,17 +24,16 @@ export class GalleryView extends Component {
    */
   render() {
     const {
-      canvases, classes, viewingDirection, windowId,
+      canvases, viewingDirection, windowId,
     } = this.props;
     const htmlDir = viewingDirection === 'right-to-left' ? 'rtl' : 'ltr';
     return (
-      <Paper
+      <Root
         component="section"
         aria-label="gallery section"
         dir={htmlDir}
         square
         elevation={0}
-        className={classes.galleryContainer}
         id={`${windowId}-gallery`}
       >
         {
@@ -34,19 +45,17 @@ export class GalleryView extends Component {
             />
           ))
         }
-      </Paper>
+      </Root>
     );
   }
 }
 
 GalleryView.propTypes = {
   canvases: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string),
   viewingDirection: PropTypes.string,
   windowId: PropTypes.string.isRequired,
 };
 
 GalleryView.defaultProps = {
-  classes: {},
   viewingDirection: '',
 };
diff --git a/src/components/GalleryViewThumbnail.js b/src/components/GalleryViewThumbnail.js
index e475c98518f7307d6e3641fc56a54b8460ea48bc..efa36629d24ccc1240441ad26f2017e9a55b6948 100644
--- a/src/components/GalleryViewThumbnail.js
+++ b/src/components/GalleryViewThumbnail.js
@@ -1,14 +1,52 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
-import Avatar from '@material-ui/core/Avatar';
-import Chip from '@material-ui/core/Chip';
-import AnnotationIcon from '@material-ui/icons/CommentSharp';
-import SearchIcon from '@material-ui/icons/SearchSharp';
-import classNames from 'classnames';
+import { styled } from '@mui/material/styles';
+import Chip from '@mui/material/Chip';
+import AnnotationIcon from '@mui/icons-material/CommentSharp';
+import SearchIcon from '@mui/icons-material/SearchSharp';
 import { InView } from 'react-intersection-observer';
-import MiradorCanvas from '../lib/MiradorCanvas';
 import IIIFThumbnail from '../containers/IIIFThumbnail';
 
+const Root = styled('div', { name: 'GalleryView', slot: 'thumbnail' })(({ ownerState, theme }) => ({
+  '&:focus': {
+    outline: 'none',
+  },
+  '&:hover': {
+    backgroundColor: theme.palette.action.hover,
+  },
+  border: '2px solid transparent',
+  ...(ownerState.selected && {
+    borderColor: theme.palette.primary.main,
+  }),
+  ...(!ownerState.selected && ownerState.searchAnnotationsCount > 0 && {
+    borderColor: theme.palette.action.selected,
+  }),
+  cursor: 'pointer',
+  display: 'inline-block',
+  margin: theme.spacing(1, 0.5),
+  maxHeight: ownerState.config.height + 45,
+  minWidth: '60px',
+  overflow: 'hidden',
+  padding: theme.spacing(0.5),
+  position: 'relative',
+  width: 'min-content',
+}));
+
+const StyledChipsContainer = styled('div', { name: 'GalleryView', slot: 'chipArea' })(({ theme }) => ({
+  display: 'flex',
+  flexDirection: 'column',
+  gap: theme.spacing(0.25),
+  position: 'absolute',
+  right: 0,
+  top: 0,
+}));
+
+const AnnotationChip = styled(Chip, { name: 'GalleryView', slot: 'chip' })(({ theme }) => ({
+  backgroundColor: theme.palette.annotations.chipBackground,
+  opacity: 0.875,
+  textAlign: 'right',
+}));
+
 /**
  * Represents a WindowViewer in the mirador workspace. Responsible for mounting
  * OSD and Navigation
@@ -102,22 +140,15 @@ export class GalleryViewThumbnail extends Component {
   render() {
     const {
       annotationsCount, searchAnnotationsCount,
-      canvas, classes, config, selected,
+      canvas, config, selected,
     } = this.props;
 
-    const miradorCanvas = new MiradorCanvas(canvas);
-
     return (
       <InView onChange={this.handleIntersection}>
-        <div
-          key={canvas.index}
-          className={
-            classNames(
-              classes.galleryViewItem,
-              selected ? classes.selected : '',
-              searchAnnotationsCount > 0 ? classes.hasAnnotations : '',
-            )
-          }
+        <Root
+          ownerState={this.props}
+          key={canvas.id || canvas.index}
+          className={selected ? 'selected' : ''}
           onClick={this.handleSelect}
           onKeyUp={this.handleKey}
           ref={this.myRef}
@@ -128,45 +159,27 @@ export class GalleryViewThumbnail extends Component {
             resource={canvas}
             labelled
             variant="outside"
-            maxWidth={config.width}
             maxHeight={config.height}
-            style={{
-              margin: '0 auto',
-              maxWidth: `${Math.ceil(config.height * miradorCanvas.aspectRatio)}px`,
-            }}
+            maxWidth={config.width}
           >
-            <div className={classes.chips}>
-              { searchAnnotationsCount > 0 && (
-                <Chip
-                  avatar={(
-                    <Avatar className={classes.avatar} classes={{ circle: classes.avatarIcon }}>
-                      <SearchIcon fontSize="small" />
-                    </Avatar>
-                  )}
+            <StyledChipsContainer>
+              {searchAnnotationsCount > 0 && (
+                <AnnotationChip
+                  icon={<SearchIcon fontSize="small" />}
                   label={searchAnnotationsCount}
-                  className={classNames(classes.searchChip)}
                   size="small"
                 />
               )}
-              { (annotationsCount || 0) > 0 && (
-                <Chip
-                  avatar={(
-                    <Avatar className={classes.avatar} classes={{ circle: classes.avatarIcon }}>
-                      <AnnotationIcon className={classes.annotationIcon} />
-                    </Avatar>
-                  )}
+              {annotationsCount > 0 && (
+                <AnnotationChip
+                  icon={<AnnotationIcon fontSize="small" />}
                   label={annotationsCount}
-                  className={
-                    classNames(
-                      classes.annotationsChip,
-                    )
-                  }
                   size="small"
                 />
               )}
-            </div>
+            </StyledChipsContainer>
           </IIIFThumbnail>
-        </div>
+        </Root>
       </InView>
     );
   }
@@ -175,7 +188,6 @@ export class GalleryViewThumbnail extends Component {
 GalleryViewThumbnail.propTypes = {
   annotationsCount: PropTypes.number,
   canvas: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   config: PropTypes.shape({
     height: PropTypes.number,
     width: PropTypes.number,
diff --git a/src/components/IIIFDropTarget.js b/src/components/IIIFDropTarget.js
index 8b48beefc91d35ca4d149023e1f3846f58c4012f..6a15f7123055df530e6f5576d93d48176f3c8697 100644
--- a/src/components/IIIFDropTarget.js
+++ b/src/components/IIIFDropTarget.js
@@ -1,7 +1,7 @@
 import PropTypes from 'prop-types';
-import Backdrop from '@material-ui/core/Backdrop';
-import InsertDriveFileSharpIcon from '@material-ui/icons/InsertDriveFileSharp';
-import { grey } from '@material-ui/core/colors';
+import Backdrop from '@mui/material/Backdrop';
+import InsertDriveFileSharpIcon from '@mui/icons-material/InsertDriveFileSharp';
+import { grey } from '@mui/material/colors';
 import { v4 as uuid } from 'uuid';
 import { NativeTypes } from 'react-dnd-html5-backend';
 import { useDrop } from 'react-dnd';
diff --git a/src/components/IIIFThumbnail.js b/src/components/IIIFThumbnail.js
index ade25534a6495af499aa63f2707a98dcf10f8756..1fd61e66fe9a976cee7a569d17c75eb5f580ccf5 100644
--- a/src/components/IIIFThumbnail.js
+++ b/src/components/IIIFThumbnail.js
@@ -2,17 +2,27 @@ import {
   Component, useMemo, useEffect, useState,
 } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
+import { styled } from '@mui/material/styles';
 import { useInView } from 'react-intersection-observer';
-import classNames from 'classnames';
 import getThumbnail from '../lib/ThumbnailFactory';
 
+const Root = styled('div', { name: 'IIIFThumbnail', slot: 'root' })({});
+
+const Label = styled('span', { name: 'IIIFThumbnail', slot: 'label' })(({ theme }) => ({
+  ...theme.typography.caption,
+}));
+
+const Image = styled('img', { name: 'IIIFThumbnail', slot: 'image' })(() => ({
+  height: 'auto',
+  width: 'auto',
+}));
+
 /**
  * A lazy-loaded image that uses IntersectionObserver to determine when to
  * try to load the image (or even calculate that the image url/height/width are)
  */
 const LazyLoadedImage = ({
-  placeholder, style = {}, thumbnail, resource, maxHeight, maxWidth, thumbnailsConfig, ...props
+  border, placeholder, style = {}, thumbnail, resource, maxHeight, maxWidth, thumbnailsConfig, ...props
 }) => {
   const { ref, inView } = useInView();
   const [loaded, setLoaded] = useState(false);
@@ -38,9 +48,14 @@ const LazyLoadedImage = ({
   }, [resource, thumbnail, maxWidth, maxHeight, thumbnailsConfig]);
 
   const imageStyles = useMemo(() => {
-    const styleProps = { height: 'auto', width: 'auto' };
+    const styleProps = {
+      height: undefined,
+      maxHeight: undefined,
+      maxWidth: undefined,
+      width: undefined,
+    };
 
-    if (!image) return { ...style, height: maxHeight || 'auto', width: maxWidth || 'auto' };
+    if (!image) return { ...style, height: maxHeight, width: maxWidth };
 
     const { height: thumbHeight, width: thumbWidth } = image;
     if (thumbHeight && thumbWidth) {
@@ -87,7 +102,8 @@ const LazyLoadedImage = ({
   const { url: src = placeholder } = (loaded && (thumbnail || image)) || {};
 
   return (
-    <img
+    <Image
+      ownerState={{ border }}
       ref={ref}
       alt=""
       role="presentation"
@@ -99,6 +115,7 @@ const LazyLoadedImage = ({
 };
 
 LazyLoadedImage.propTypes = {
+  border: PropTypes.bool,
   maxHeight: PropTypes.number.isRequired,
   maxWidth: PropTypes.number.isRequired,
   placeholder: PropTypes.string.isRequired,
@@ -113,6 +130,7 @@ LazyLoadedImage.propTypes = {
 };
 
 LazyLoadedImage.defaultProps = {
+  border: false,
   style: {},
   thumbnail: null,
   thumbnailsConfig: {},
@@ -142,8 +160,8 @@ export class IIIFThumbnail extends Component {
    */
   render() {
     const {
+      border,
       children,
-      classes,
       imagePlaceholder,
       labelled,
       maxHeight,
@@ -152,11 +170,10 @@ export class IIIFThumbnail extends Component {
       style,
       thumbnail,
       thumbnailsConfig,
-      variant,
     } = this.props;
 
     return (
-      <div className={classNames(classes.root, { [classes[`${variant}Root`]]: variant })}>
+      <Root ownerState={this.props}>
         <LazyLoadedImage
           placeholder={imagePlaceholder}
           thumbnail={thumbnail}
@@ -165,25 +182,23 @@ export class IIIFThumbnail extends Component {
           maxWidth={maxWidth}
           thumbnailsConfig={thumbnailsConfig}
           style={style}
-          className={classes.image}
+          border={border}
         />
 
         { labelled && (
-          <div className={classNames(classes.label, { [classes[`${variant}Label`]]: variant })}>
-            <Typography variant="caption" classes={{ root: classNames(classes.caption, { [classes[`${variant}Caption`]]: variant }) }}>
-              {this.label()}
-            </Typography>
-          </div>
+          <Label ownerState={this.props}>
+            {this.label()}
+          </Label>
         )}
         {children}
-      </div>
+      </Root>
     );
   }
 }
 
 IIIFThumbnail.propTypes = {
+  border: PropTypes.bool,
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string),
   imagePlaceholder: PropTypes.string,
   label: PropTypes.string,
   labelled: PropTypes.bool,
@@ -197,12 +212,12 @@ IIIFThumbnail.propTypes = {
     width: PropTypes.number,
   }),
   thumbnailsConfig: PropTypes.object, // eslint-disable-line react/forbid-prop-types
-  variant: PropTypes.oneOf(['inside', 'outside']),
+  variant: PropTypes.oneOf(['inside', 'outside']), // eslint-disable-line react/no-unused-prop-types
 };
 
 IIIFThumbnail.defaultProps = {
+  border: false,
   children: null,
-  classes: {},
   // Transparent "gray"
   imagePlaceholder: '',
   label: undefined,
diff --git a/src/components/LabelValueMetadata.js b/src/components/LabelValueMetadata.js
index 9cb73df2718649d39f1ae71470b0841d926167d2..b1fe6c9a29d5d11e9f76a56394d30ace140f5fa1 100644
--- a/src/components/LabelValueMetadata.js
+++ b/src/components/LabelValueMetadata.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
+import Typography from '@mui/material/Typography';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import ns from '../config/css-ns';
 
diff --git a/src/components/LanguageSettings.js b/src/components/LanguageSettings.js
index 7901c4e440ada6bd9e5f251b71998158a850ed6a..f0048c4788abbc9b9ef50d49818dbd40cf4f7456 100644
--- a/src/components/LanguageSettings.js
+++ b/src/components/LanguageSettings.js
@@ -1,8 +1,8 @@
 import { Component } from 'react';
-import ListItemIcon from '@material-ui/core/ListItemIcon';
-import ListItemText from '@material-ui/core/ListItemText';
-import MenuItem from '@material-ui/core/MenuItem';
-import CheckIcon from '@material-ui/icons/CheckSharp';
+import ListItemIcon from '@mui/material/ListItemIcon';
+import ListItemText from '@mui/material/ListItemText';
+import MenuItem from '@mui/material/MenuItem';
+import CheckIcon from '@mui/icons-material/CheckSharp';
 import PropTypes from 'prop-types';
 
 /**
@@ -23,7 +23,6 @@ export class LanguageSettings extends Component {
         {
           languages.map(language => (
             <MenuItem
-              button={!language.current}
               key={language.locale}
               onClick={() => { handleClick(language.locale); }}
             >
diff --git a/src/components/LocalePicker.js b/src/components/LocalePicker.js
index beb212324eb40595e03bde77284bd7a2e3303636..a1f9d8593dea7ff5e7751ea8e583951859c4f22a 100644
--- a/src/components/LocalePicker.js
+++ b/src/components/LocalePicker.js
@@ -1,9 +1,9 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import MenuItem from '@material-ui/core/MenuItem';
-import FormControl from '@material-ui/core/FormControl';
-import Select from '@material-ui/core/Select';
-import Typography from '@material-ui/core/Typography';
+import MenuItem from '@mui/material/MenuItem';
+import FormControl from '@mui/material/FormControl';
+import Select from '@mui/material/Select';
+import Typography from '@mui/material/Typography';
 
 /**
  * Provide a locale picker
@@ -16,7 +16,6 @@ export class LocalePicker extends Component {
   render() {
     const {
       availableLocales,
-      classes,
       locale,
       setLocale,
     } = this.props;
@@ -30,14 +29,11 @@ export class LocalePicker extends Component {
               horizontal: 'left',
               vertical: 'bottom',
             },
-            getContentAnchorEl: null,
           }}
           displayEmpty
           value={locale}
           onChange={(e) => { setLocale(e.target.value); }}
           name="locale"
-          classes={{ select: classes.select }}
-          className={classes.selectEmpty}
         >
           {
             availableLocales.map(l => (
@@ -52,14 +48,12 @@ export class LocalePicker extends Component {
 
 LocalePicker.propTypes = {
   availableLocales: PropTypes.arrayOf(PropTypes.string),
-  classes: PropTypes.objectOf(PropTypes.string),
   locale: PropTypes.string,
   setLocale: PropTypes.func,
 };
 
 LocalePicker.defaultProps = {
   availableLocales: [],
-  classes: {},
   locale: '',
   setLocale: undefined,
 };
diff --git a/src/components/ManifestForm.js b/src/components/ManifestForm.js
index 18d304cd6dca0f1301e1dd735d812d9135641eb8..2266db77c9df80ab685463034e7e9872117c2868 100644
--- a/src/components/ManifestForm.js
+++ b/src/components/ManifestForm.js
@@ -1,8 +1,8 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import Grid from '@material-ui/core/Grid';
-import TextField from '@material-ui/core/TextField';
+import Button from '@mui/material/Button';
+import Grid from '@mui/material/Grid';
+import TextField from '@mui/material/TextField';
 
 /**
  * Provides a form for user input of a manifest url
@@ -68,7 +68,6 @@ export class ManifestForm extends Component {
     const { formValue } = this.state;
     const {
       addResourcesOpen,
-      classes,
       onCancel,
       t,
     } = this.props;
@@ -92,11 +91,19 @@ export class ManifestForm extends Component {
                 shrink: true,
               }}
               InputProps={{
-                className: classes.input,
+                style: { typography: 'body1' },
               }}
             />
           </Grid>
-          <Grid item xs={12} sm={4} md={3} className={classes.buttons}>
+          <Grid
+            item
+            xs={12}
+            sm={4}
+            md={3}
+            sx={{
+              textAlign: { sm: 'inherit', xs: 'right' },
+            }}
+          >
             { onCancel && (
               <Button onClick={this.handleCancel}>
                 {t('cancel')}
@@ -115,14 +122,12 @@ export class ManifestForm extends Component {
 ManifestForm.propTypes = {
   addResource: PropTypes.func.isRequired,
   addResourcesOpen: PropTypes.bool.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string),
   onCancel: PropTypes.func,
   onSubmit: PropTypes.func,
   t: PropTypes.func,
 };
 
 ManifestForm.defaultProps = {
-  classes: {},
   onCancel: null,
   onSubmit: () => {},
   t: key => key,
diff --git a/src/components/ManifestInfo.js b/src/components/ManifestInfo.js
index 4d9b2339257fd314eeae95916a35b6bf71ce453c..e465a054da023abe4245147b934ff4759aef2918 100644
--- a/src/components/ManifestInfo.js
+++ b/src/components/ManifestInfo.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
+import Typography from '@mui/material/Typography';
 import CollapsibleSection from '../containers/CollapsibleSection';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import { LabelValueMetadata } from './LabelValueMetadata';
diff --git a/src/components/ManifestListItem.js b/src/components/ManifestListItem.js
index accd5ede56d5dadcfb6ab8d791359aaeaf838b13..18e26fbc921ecb673c5afc94870e2e7e214ea555 100644
--- a/src/components/ManifestListItem.js
+++ b/src/components/ManifestListItem.js
@@ -1,14 +1,42 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import ListItem from '@material-ui/core/ListItem';
-import ButtonBase from '@material-ui/core/ButtonBase';
-import Grid from '@material-ui/core/Grid';
-import Typography from '@material-ui/core/Typography';
-import Skeleton from '@material-ui/lab/Skeleton';
+import { styled } from '@mui/material/styles';
+import ListItem from '@mui/material/ListItem';
+import ButtonBase from '@mui/material/ButtonBase';
+import Grid from '@mui/material/Grid';
+import Typography from '@mui/material/Typography';
+import Skeleton from '@mui/material/Skeleton';
 import { Img } from 'react-image';
 import ManifestListItemError from '../containers/ManifestListItemError';
 import ns from '../config/css-ns';
 
+const Root = styled(ListItem, { name: 'ManifestListItem', slot: 'root' })(({ ownerState, theme }) => ({
+  '&:hover,&:focus-within': {
+    backgroundColor: theme.palette.action.hover,
+    borderLeftColor: ownerState?.active ? theme.palette.primary.main : theme.palette.action.hover,
+  },
+  borderLeft: '4px solid',
+  borderLeftColor: ownerState?.active ? theme.palette.primary.main : 'transparent',
+  paddingLeft: theme.spacing(2),
+  paddingRight: theme.spacing(2),
+  [theme.breakpoints.up('sm')]: {
+    paddingLeft: theme.spacing(3),
+    paddingRight: theme.spacing(3),
+  },
+}));
+
+const StyledThumbnail = styled(Img, { name: 'ManifestListItem', slot: 'thumbnail' })(({ theme }) => ({
+  maxWidth: '100%',
+  objectFit: 'contain',
+}));
+
+const StyledLogo = styled(Img, { name: 'ManifestListItem', slot: 'logo' })(({ theme }) => ({
+  height: '2.5rem',
+  maxWidth: '100%',
+  objectFit: 'contain',
+  paddingRight: 1,
+}));
+
 /**
  * Represents an item in a list of currently-loaded or loading manifests
  * @param {object} props
@@ -57,7 +85,6 @@ export class ManifestListItem extends Component {
       thumbnail,
       manifestLogo,
       size,
-      classes,
       provider,
       t,
       error,
@@ -68,61 +95,81 @@ export class ManifestListItem extends Component {
     const placeholder = (
       <Grid container className={ns('manifest-list-item')} spacing={2}>
         <Grid item xs={3} sm={2}>
-          <Skeleton className={classes.placeholder} variant="rect" height={80} width={120} />
+          <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="rectangular" height={80} width={120} />
         </Grid>
         <Grid item xs={9} sm={6}>
-          <Skeleton className={classes.placeholder} variant="text" />
+          <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="text" />
         </Grid>
         <Grid item xs={8} sm={2}>
-          <Skeleton className={classes.placeholder} variant="text" />
-          <Skeleton className={classes.placeholder} variant="text" />
+          <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="text" />
+          <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="text" />
         </Grid>
         <Grid item xs={4} sm={2}>
-          <Skeleton className={classes.placeholder} variant="rect" height={60} width={60} />
+          <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="rectangular" height={60} width={60} />
         </Grid>
       </Grid>
     );
 
     if (error) {
       return (
-        <ListItem divider className={classes.root} data-manifestid={manifestId}>
+        <Root
+          ownerState={this.props}
+          divider
+          selected={active}
+          className={active ? 'active' : ''}
+          data-manifestid={manifestId}
+        >
           <ManifestListItemError manifestId={manifestId} />
-        </ListItem>
+        </Root>
       );
     }
 
     return (
-      <ListItem divider className={[classes.root, active ? classes.active : ''].join(' ')} data-manifestid={manifestId}>
+      <Root
+        divider
+        selected={active}
+        className={active ? 'active' : ''}
+        data-manifestid={manifestId}
+        data-active={active}
+      >
         {ready ? (
           <Grid container className={ns('manifest-list-item')} spacing={2}>
-            <Grid item xs={12} sm={6} className={classes.buttonGrid}>
+            <Grid item xs={12} sm={6}>
               <ButtonBase
                 ref={buttonRef}
                 className={ns('manifest-list-item-title')}
                 style={{ width: '100%' }}
                 onClick={this.handleOpenButtonClick}
               >
-                <Grid container spacing={2} className={classes.label} component="span">
+                <Grid
+                  container
+                  spacing={2}
+                  sx={{
+                    textAlign: 'left',
+                    textTransform: 'initial',
+                  }}
+                  component="span"
+                >
                   <Grid item xs={4} sm={3} component="span">
                     { thumbnail
                       ? (
-                        <Img
-                          className={[classes.thumbnail, ns('manifest-list-item-thumb')].join(' ')}
+                        <StyledThumbnail
+                          className={[ns('manifest-list-item-thumb')]}
                           src={[thumbnail]}
                           alt=""
                           height="80"
                           unloader={(
                             <Skeleton
-                              variant="rect"
+                              variant="rectangular"
                               animation={false}
-                              className={classes.placeholder}
+                              sx={{ bgcolor: 'grey[300]' }}
                               height={80}
                               width={120}
                             />
                           )}
                         />
                       )
-                      : <Skeleton className={classes.placeholder} variant="rect" height={80} width={120} />}
+                      : <Skeleton sx={{ bgcolor: 'grey[300]' }} variant="rectangular" height={80} width={120} />}
                   </Grid>
                   <Grid item xs={8} sm={9} component="span">
                     { isCollection && (
@@ -145,16 +192,15 @@ export class ManifestListItem extends Component {
             <Grid item xs={4} sm={2}>
               { manifestLogo
                 && (
-                <Img
+                <StyledLogo
                   src={[manifestLogo]}
                   alt=""
                   role="presentation"
-                  className={classes.logo}
                   unloader={(
                     <Skeleton
-                      variant="rect"
+                      variant="rectangular"
                       animation={false}
-                      className={classes.placeholder}
+                      sx={{ bgcolor: 'grey[300]' }}
                       height={60}
                       width={60}
                     />
@@ -166,7 +212,7 @@ export class ManifestListItem extends Component {
         ) : (
           placeholder
         )}
-      </ListItem>
+      </Root>
     );
   }
 }
@@ -175,7 +221,6 @@ ManifestListItem.propTypes = {
   active: PropTypes.bool,
   addWindow: PropTypes.func.isRequired,
   buttonRef: PropTypes.elementType,
-  classes: PropTypes.objectOf(PropTypes.string),
   error: PropTypes.string,
   fetchManifest: PropTypes.func.isRequired,
   handleClose: PropTypes.func,
@@ -195,7 +240,6 @@ ManifestListItem.propTypes = {
 ManifestListItem.defaultProps = {
   active: false,
   buttonRef: undefined,
-  classes: {},
   error: null,
   handleClose: () => {},
   isCollection: false,
diff --git a/src/components/ManifestListItemError.js b/src/components/ManifestListItemError.js
index 3d63e48afd3445ff3df92814908d91c9ca19a1d2..b8d5176aa5aaa6a967b432f2af601f48fcd67268 100644
--- a/src/components/ManifestListItemError.js
+++ b/src/components/ManifestListItemError.js
@@ -1,9 +1,9 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import ErrorIcon from '@material-ui/icons/ErrorOutlineSharp';
-import Grid from '@material-ui/core/Grid';
-import Typography from '@material-ui/core/Typography';
+import Button from '@mui/material/Button';
+import ErrorIcon from '@mui/icons-material/ErrorOutlineSharp';
+import Grid from '@mui/material/Grid';
+import Typography from '@mui/material/Typography';
 
 /**
  * ManifestListItemError renders a component displaying a
@@ -15,7 +15,7 @@ export class ManifestListItemError extends Component {
   */
   render() {
     const {
-      classes, manifestId, onDismissClick, onTryAgainClick, t,
+      manifestId, onDismissClick, onTryAgainClick, t,
     } = this.props;
 
     return (
@@ -24,12 +24,17 @@ export class ManifestListItemError extends Component {
           <Grid container item xs={12} sm={6}>
             <Grid item xs={4} sm={3}>
               <Grid container justifyContent="center">
-                <ErrorIcon className={classes.errorIcon} />
+                <ErrorIcon sx={{
+                  color: 'error.main',
+                  height: '2rem',
+                  width: '2rem',
+                }}
+                />
               </Grid>
             </Grid>
             <Grid item xs={8} sm={9}>
               <Typography>{t('manifestError')}</Typography>
-              <Typography className={classes.manifestIdText}>{manifestId}</Typography>
+              <Typography sx={{ wordBreak: 'break-all' }}>{manifestId}</Typography>
             </Grid>
           </Grid>
         </Grid>
@@ -52,7 +57,6 @@ export class ManifestListItemError extends Component {
 }
 
 ManifestListItemError.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   manifestId: PropTypes.string.isRequired,
   onDismissClick: PropTypes.func.isRequired,
   onTryAgainClick: PropTypes.func.isRequired,
diff --git a/src/components/ManifestRelatedLinks.js b/src/components/ManifestRelatedLinks.js
index 62436ea79a4ba2009ff7ae9cee421a83e1995d40..cbba6aefd1defc8c2a02ab6f68a6b459a27d2ec6 100644
--- a/src/components/ManifestRelatedLinks.js
+++ b/src/components/ManifestRelatedLinks.js
@@ -1,12 +1,19 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import Link from '@material-ui/core/Link';
+import { styled } from '@mui/material/styles';
+import Typography from '@mui/material/Typography';
+import Link from '@mui/material/Link';
 import classNames from 'classnames';
 import CollapsibleSection from '../containers/CollapsibleSection';
 import ns from '../config/css-ns';
 import { PluginHook } from './PluginHook';
 
+const StyledDl = styled('dl')(({ theme }) => ({
+  '& dd': {
+    marginBottom: '.5em',
+    marginLeft: '0',
+  },
+}));
 /**
  * ManifestRelatedLinks
  */
@@ -17,7 +24,6 @@ export class ManifestRelatedLinks extends Component {
    */
   render() {
     const {
-      classes,
       homepage,
       manifestUrl,
       related,
@@ -40,7 +46,7 @@ export class ManifestRelatedLinks extends Component {
         >
           {t('links')}
         </Typography>
-        <dl className={classNames(ns('label-value-metadata'), classes.labelValueMetadata)}>
+        <StyledDl className={classNames(ns('label-value-metadata'))}>
           { homepage && (
             <>
               <Typography variant="subtitle2" component="dt">{t('iiif_homepage')}</Typography>
@@ -113,7 +119,7 @@ export class ManifestRelatedLinks extends Component {
               </Typography>
             </>
           )}
-        </dl>
+        </StyledDl>
         <PluginHook {...this.props} />
       </CollapsibleSection>
     );
@@ -121,7 +127,6 @@ export class ManifestRelatedLinks extends Component {
 }
 
 ManifestRelatedLinks.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   homepage: PropTypes.arrayOf(PropTypes.shape({
     label: PropTypes.string,
     value: PropTypes.string,
diff --git a/src/components/MinimalWindow.js b/src/components/MinimalWindow.js
index f19b35385f339bf38761b7d6390e421101833330..b04661692437238242a47ea242933c80c82c0bfe 100644
--- a/src/components/MinimalWindow.js
+++ b/src/components/MinimalWindow.js
@@ -1,15 +1,19 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import MenuIcon from '@material-ui/icons/MenuSharp';
+import { styled } from '@mui/material/styles';
+import MenuIcon from '@mui/icons-material/MenuSharp';
 import cn from 'classnames';
-import Paper from '@material-ui/core/Paper';
-import AppBar from '@material-ui/core/AppBar';
-import Toolbar from '@material-ui/core/Toolbar';
-import Typography from '@material-ui/core/Typography';
-import CloseIcon from '@material-ui/icons/CloseSharp';
+import Paper from '@mui/material/Paper';
+import AppBar from '@mui/material/AppBar';
+import Toolbar from '@mui/material/Toolbar';
+import Typography from '@mui/material/Typography';
+import CloseIcon from '@mui/icons-material/CloseSharp';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import ns from '../config/css-ns';
 
+const StyledMiradorMenuButton = styled(MiradorMenuButton)(() => ({
+  marginLeft: 'auto',
+}));
 /** */
 export class MinimalWindow extends Component {
   /** */
@@ -19,7 +23,6 @@ export class MinimalWindow extends Component {
       allowWindowSideBar,
       ariaLabel,
       children,
-      classes,
       label,
       removeWindow,
       t,
@@ -32,17 +35,31 @@ export class MinimalWindow extends Component {
         elevation={1}
         id={windowId}
         className={
-          cn(classes.window, ns('placeholder-window'))
+          cn(ns('placeholder-window'))
         }
+        sx={{
+          backgroundColor: 'shades.dark',
+          borderRadius: 0,
+          display: 'flex',
+          flexDirection: 'column',
+          height: '100%',
+          minHeight: 0,
+          overflow: 'hidden',
+          width: '100%',
+        }}
         aria-label={label && ariaLabel ? t('window', { label }) : null}
       >
-        <AppBar position="relative" color="default">
+        <AppBar position="relative" color="default" enableColorOnDark>
           <Toolbar
             disableGutters
-            className={cn(
-              classes.windowTopBarStyle,
-              ns('window-top-bar'),
-            )}
+            className={cn(ns('window-top-bar'))}
+            sx={{
+              backgroundColor: 'shades.main',
+              borderTop: '2px solid transparent',
+              minHeight: 32,
+              paddingLeft: 0.5,
+              paddingRight: 0.5,
+            }}
             variant="dense"
           >
             {allowWindowSideBar && (
@@ -53,20 +70,29 @@ export class MinimalWindow extends Component {
                 <MenuIcon />
               </MiradorMenuButton>
             )}
-            <Typography variant="h2" noWrap color="inherit" className={classes.title}>
+            <Typography
+              variant="h2"
+              noWrap
+              color="inherit"
+              sx={{
+                flexGrow: 1,
+                paddingLeft: 0.5,
+                typography: 'h6',
+              }}
+            >
               {label}
             </Typography>
             {allowClose && removeWindow && (
-              <MiradorMenuButton
+              <StyledMiradorMenuButton
                 aria-label={t('closeWindow')}
-                className={cn(classes.button, ns('window-close'))}
+                className={cn(ns('window-close'))}
                 onClick={removeWindow}
                 TooltipProps={{
-                  tabIndex: ariaLabel ? '0' : '-1',
+                  tabIndex: ariaLabel ? 0 : -1,
                 }}
               >
                 <CloseIcon />
-              </MiradorMenuButton>
+              </StyledMiradorMenuButton>
             )}
           </Toolbar>
         </AppBar>
@@ -81,7 +107,6 @@ MinimalWindow.propTypes = {
   allowWindowSideBar: PropTypes.bool,
   ariaLabel: PropTypes.bool,
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string),
   label: PropTypes.string,
   removeWindow: PropTypes.func,
   t: PropTypes.func,
@@ -93,7 +118,6 @@ MinimalWindow.defaultProps = {
   allowWindowSideBar: true,
   ariaLabel: true,
   children: null,
-  classes: {},
   label: '',
   removeWindow: () => {},
   t: key => key,
diff --git a/src/components/MiradorMenuButton.js b/src/components/MiradorMenuButton.js
index 869c1c0a024b9c54df2960d9698da1be2e47d535..e1def5a7b87b9283121bacf3e8a85e04da4e6b2f 100644
--- a/src/components/MiradorMenuButton.js
+++ b/src/components/MiradorMenuButton.js
@@ -1,7 +1,15 @@
 import PropTypes from 'prop-types';
-import Badge from '@material-ui/core/Badge';
-import IconButton from '@material-ui/core/IconButton';
-import Tooltip from '@material-ui/core/Tooltip';
+import { styled } from '@mui/material/styles';
+import Badge from '@mui/material/Badge';
+import IconButton from '@mui/material/IconButton';
+import Tooltip from '@mui/material/Tooltip';
+
+const Root = styled(IconButton, { name: 'MiradorMenuButton', slot: 'root' })(({ ownerState, theme }) => ({
+  fill: 'currentcolor',
+  ...(ownerState.selected && {
+    backgroundColor: theme.palette.action.selected,
+  }),
+}));
 
 /**
  * MiradorMenuButton ~ Wrap the given icon prop in an IconButton and a Tooltip.
@@ -17,15 +25,28 @@ export function MiradorMenuButton(props) {
     dispatch,
     BadgeProps,
     TooltipProps,
+    sx,
     ...iconButtonProps
   } = props;
 
   const button = (
-    <IconButton {...iconButtonProps}>
+    <Root
+      ownerState={props}
+      {...iconButtonProps}
+      sx={sx}
+      size="large"
+    >
       {badge
-        ? <Badge overlap="rectangular" {...BadgeProps}>{children}</Badge>
+        ? (
+          <Badge
+            overlap="rectangular"
+            {...BadgeProps}
+          >
+            {children}
+          </Badge>
+        )
         : children}
-    </IconButton>
+    </Root>
   );
 
   if (iconButtonProps.disabled) return button;
@@ -50,6 +71,8 @@ MiradorMenuButton.propTypes = {
   children: PropTypes.element.isRequired,
   container: PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
   dispatch: PropTypes.func,
+  selected: PropTypes.bool,
+  sx: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   TooltipProps: PropTypes.object, // eslint-disable-line react/forbid-prop-types
 };
 
@@ -58,5 +81,7 @@ MiradorMenuButton.defaultProps = {
   BadgeProps: {},
   container: null,
   dispatch: () => {},
+  selected: false,
+  sx: {},
   TooltipProps: {},
 };
diff --git a/src/components/NestedMenu.js b/src/components/NestedMenu.js
index 65fcfeec57a1d7681510747b3dc3bb120358d57f..b61107786453d1b98aff1749f704089d9e686e10 100644
--- a/src/components/NestedMenu.js
+++ b/src/components/NestedMenu.js
@@ -1,10 +1,10 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import ListItemIcon from '@material-ui/core/ListItemIcon';
-import ListItemText from '@material-ui/core/ListItemText';
-import MenuItem from '@material-ui/core/MenuItem';
-import ExpandLess from '@material-ui/icons/ExpandLessSharp';
-import ExpandMore from '@material-ui/icons/ExpandMoreSharp';
+import ListItemIcon from '@mui/material/ListItemIcon';
+import ListItemText from '@mui/material/ListItemText';
+import MenuItem from '@mui/material/MenuItem';
+import ExpandLess from '@mui/icons-material/ExpandLessSharp';
+import ExpandMore from '@mui/icons-material/ExpandMoreSharp';
 
 /**
  * NestedMenu ~ A presentation component to render a menu item and have
diff --git a/src/components/OpenSeadragonViewer.js b/src/components/OpenSeadragonViewer.js
index f2ce17d670a9b5b5d7c0586fc8bda242fdc92b67..120aba5151fee7995e5985902cf3e8cf8bcdcc84 100644
--- a/src/components/OpenSeadragonViewer.js
+++ b/src/components/OpenSeadragonViewer.js
@@ -2,6 +2,7 @@ import {
   createRef, Children, cloneElement, Component,
 } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import debounce from 'lodash/debounce';
 import isEqual from 'lodash/isEqual';
 import OpenSeadragon from 'openseadragon';
@@ -12,6 +13,11 @@ import CanvasWorld from '../lib/CanvasWorld';
 import { PluginHook } from './PluginHook';
 import { OSDReferences } from '../plugins/OSDReferences';
 
+const StyledSection = styled('section')({
+  cursor: 'grab',
+  flex: 1,
+  position: 'relative',
+});
 /**
  * Represents a OpenSeadragonViewer in the mirador workspace. Responsible for mounting
  * and rendering OSD.
@@ -340,7 +346,7 @@ export class OpenSeadragonViewer extends Component {
    */
   render() {
     const {
-      children, classes, label, t, windowId,
+      children, label, t, windowId,
       drawAnnotations,
     } = this.props;
     const { viewer, grabbing } = this.state;
@@ -355,8 +361,8 @@ export class OpenSeadragonViewer extends Component {
     ));
 
     return (
-      <section
-        className={classNames(ns('osd-container'), classes.osdContainer)}
+      <StyledSection
+        className={classNames(ns('osd-container'))}
         style={{ cursor: grabbing ? 'grabbing' : undefined }}
         id={`${windowId}-osd`}
         ref={this.ref}
@@ -367,7 +373,7 @@ export class OpenSeadragonViewer extends Component {
             && <AnnotationsOverlay viewer={viewer} windowId={windowId} /> }
         { enhancedChildren }
         <PluginHook viewer={viewer} {...{ ...this.props, children: null }} />
-      </section>
+      </StyledSection>
     );
   }
 }
@@ -385,7 +391,6 @@ OpenSeadragonViewer.defaultProps = {
 OpenSeadragonViewer.propTypes = {
   canvasWorld: PropTypes.instanceOf(CanvasWorld).isRequired,
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   drawAnnotations: PropTypes.bool,
   infoResponses: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
   label: PropTypes.string,
diff --git a/src/components/PrimaryWindow.js b/src/components/PrimaryWindow.js
index 988b6784aea40a2167782f35ea06cc8964e10851..30c4d302eb4e9794fc9c9b73e318c02a24e78982 100644
--- a/src/components/PrimaryWindow.js
+++ b/src/components/PrimaryWindow.js
@@ -1,5 +1,6 @@
 import { Component, lazy, Suspense } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
 import WindowSideBar from '../containers/WindowSideBar';
 import CompanionArea from '../containers/CompanionArea';
@@ -16,6 +17,12 @@ GalleryView.displayName = 'GalleryView';
 SelectCollection.displayName = 'SelectCollection';
 WindowViewer.displayName = 'WindowViewer';
 
+const Root = styled('div', { name: 'PrimaryWindow', slot: 'root' })(() => ({
+  display: 'flex',
+  flex: 1,
+  position: 'relative',
+}));
+
 /**
  * PrimaryWindow - component that renders the primary content of a Mirador
  * window. Right now this differentiates between a Image, Video, or Audio viewer.
@@ -74,17 +81,18 @@ export class PrimaryWindow extends Component {
    */
   render() {
     const {
-      isCollectionDialogVisible, windowId, classes, children,
+      isCollectionDialogVisible, windowId, children, className,
     } = this.props;
+
     return (
-      <div className={classNames(ns('primary-window'), classes.primaryWindow)}>
+      <Root data-testid="test-window" className={classNames(ns('primary-window'), className)}>
         <WindowSideBar windowId={windowId} />
         <CompanionArea windowId={windowId} position="left" />
         { isCollectionDialogVisible && <CollectionDialog windowId={windowId} /> }
         <Suspense fallback={<div />}>
           {children || this.renderViewer()}
         </Suspense>
-      </div>
+      </Root>
     );
   }
 }
@@ -92,7 +100,7 @@ export class PrimaryWindow extends Component {
 PrimaryWindow.propTypes = {
   audioResources: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
+  className: PropTypes.string,
   isCollection: PropTypes.bool,
   isCollectionDialogVisible: PropTypes.bool,
   isFetching: PropTypes.bool,
@@ -104,6 +112,7 @@ PrimaryWindow.propTypes = {
 PrimaryWindow.defaultProps = {
   audioResources: [],
   children: undefined,
+  className: undefined,
   isCollection: false,
   isCollectionDialogVisible: false,
   isFetching: false,
diff --git a/src/components/SanitizedHtml.js b/src/components/SanitizedHtml.js
index 1cd1f4b24f8e66ba51301bcf4776b116327d2a44..dcbab42e3b20ca80f030dd607c9f5639d4658cb3 100644
--- a/src/components/SanitizedHtml.js
+++ b/src/components/SanitizedHtml.js
@@ -1,9 +1,12 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import DOMPurify from 'dompurify';
 import ns from '../config/css-ns';
 import htmlRules from '../lib/htmlRules';
 
+const Root = styled('span', { name: 'IIIFHtmlContent', slot: 'root' })({});
+
 /**
 */
 export class SanitizedHtml extends Component {
@@ -25,8 +28,8 @@ export class SanitizedHtml extends Component {
     });
 
     return (
-      <span
-        className={[classes.root, ns('third-party-html')].join(' ')}
+      <Root
+        className={[ns('third-party-html'), classes.root].join(' ')}
         dangerouslySetInnerHTML={{ // eslint-disable-line react/no-danger
           __html: DOMPurify.sanitize(htmlString, htmlRules[ruleSet]),
         }}
diff --git a/src/components/ScrollIndicatedDialogContent.js b/src/components/ScrollIndicatedDialogContent.js
index a8de5229d8d8e09e70aa881432246ccb568c5fc3..01e10ca255355b32e255b4f3380d0bfa6e814fb5 100644
--- a/src/components/ScrollIndicatedDialogContent.js
+++ b/src/components/ScrollIndicatedDialogContent.js
@@ -1,5 +1,49 @@
 import PropTypes from 'prop-types';
-import DialogContent from '@material-ui/core/DialogContent';
+import DialogContent from '@mui/material/DialogContent';
+import { alpha, styled } from '@mui/material/styles';
+
+/**
+ * From https://github.com/mui/material-ui/blob/v5.15.0/packages/mui-material/src/styles/getOverlayAlpha.ts
+ */
+const getOverlayAlpha = (elevation) => {
+  let alphaValue;
+  if (elevation < 1) {
+    alphaValue = 5.11916 * elevation ** 2;
+  } else {
+    alphaValue = 4.5 * Math.log(elevation + 1) + 2;
+  }
+  return (alphaValue / 100).toFixed(2);
+};
+
+const Root = styled(DialogContent, { name: 'ScrollIndicatedDialogContent', slot: 'root' })(({ ownerState, theme }) => {
+  // In dark mode, paper has a elevation-dependent background color:
+  // https://github.com/mui/material-ui/blob/v5.15.0/packages/mui-material/src/Paper/Paper.js#L55-L60
+  const bgcolor = theme.palette.mode === 'dark' ? {
+    backgroundImage: `linear-gradient(${alpha(
+      '#fff',
+      getOverlayAlpha(ownerState?.elevation || 24),
+    )}, ${alpha('#fff', getOverlayAlpha(ownerState?.elevation || 24))})`,
+  } : theme.palette.background.paper;
+  return {
+    /* Shadow covers */
+    background: `linear-gradient(${bgcolor} 30%, rgba(255, 255, 255, 0)), `
+      + `linear-gradient(rgba(255, 255, 255, 0), ${bgcolor} 70%) 0 100%, `
+      // Shaddows
+      + 'radial-gradient(50% 0, farthest-side, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)), '
+      + 'radial-gradient(50% 100%, farthest-side, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)) 0 100%,',
+    /* Shadow covers */
+    background: `linear-gradient(${bgcolor} 30%, rgba(255, 255, 255, 0)), ` // eslint-disable-line no-dupe-keys
+      + `linear-gradient(rgba(255, 255, 255, 0), ${bgcolor} 70%) 0 100%, `
+      // Shaddows
+      + 'radial-gradient(farthest-side at 50% 0, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)), '
+      + 'radial-gradient(farthest-side at 50% 100%, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)) 0 100%;',
+
+    backgroundAttachment: 'local, local, scroll, scroll',
+    backgroundRepeat: 'no-repeat',
+    backgroundSize: '100% 40px, 100% 40px, 100% 14px, 100% 14px',
+    overflowY: 'auto',
+  };
+});
 
 /**
  * ScrollIndicatedDialogContent ~ Inject a style into the DialogContent component
@@ -10,18 +54,21 @@ export function ScrollIndicatedDialogContent(props) {
   const ourClassName = [className, classes.shadowScrollDialog].join(' ');
 
   return (
-    <DialogContent className={ourClassName} {...otherProps} />
+    <Root
+      className={ourClassName}
+      {...otherProps}
+    />
   );
 }
 
 ScrollIndicatedDialogContent.propTypes = {
   classes: PropTypes.shape({
     shadowScrollDialog: PropTypes.string,
-  }).isRequired,
-
+  }),
   className: PropTypes.string,
 };
 
 ScrollIndicatedDialogContent.defaultProps = {
+  classes: {},
   className: '',
 };
diff --git a/src/components/ScrollTo.js b/src/components/ScrollTo.js
index e73d602cf998bf9d9adcdc17d6b730c4b91406f5..4e11892d7267729ba6619f090c407910785d3612 100644
--- a/src/components/ScrollTo.js
+++ b/src/components/ScrollTo.js
@@ -1,5 +1,6 @@
-import { createRef, Component } from 'react';
+import { cloneElement, createRef, Component } from 'react';
 import PropTypes from 'prop-types';
+import isEmpty from 'lodash/isEmpty';
 
 /**
  * ScrollTo ~
@@ -36,9 +37,9 @@ export class ScrollTo extends Component {
   containerBoundingRect() {
     const { containerRef } = this.props;
 
-    if (!containerRef || !containerRef.current || !containerRef.current.domEl) return {};
+    if (!containerRef || !containerRef.current) return {};
 
-    return containerRef.current.domEl.getBoundingClientRect();
+    return containerRef.current.getBoundingClientRect();
   }
 
   /**
@@ -59,14 +60,14 @@ export class ScrollTo extends Component {
   }
 
   /**
-   * The container provided in the containersRef dome structure in which scrolling
+   * The container provided in the containersRef dom structure in which scrolling
    * should happen.
   */
   scrollableContainer() {
     const { containerRef } = this.props;
 
-    if (!containerRef || !containerRef.current || !containerRef.current.domEl) return null;
-    return containerRef.current.domEl.getElementsByClassName('mirador-scrollto-scrollable')[0];
+    if (!containerRef || !containerRef.current) return null;
+    return containerRef.current.getElementsByClassName('mirador-scrollto-scrollable')[0];
   }
 
   /**
@@ -104,16 +105,12 @@ export class ScrollTo extends Component {
   */
   render() {
     const {
-      children, containerRef, offsetTop, scrollTo, ...otherProps
+      children, containerRef, offsetTop, scrollTo, nodeId, ...otherProps
     } = this.props;
 
-    if (!scrollTo) return children;
+    if (!scrollTo && isEmpty(otherProps)) return children;
 
-    return (
-      <div ref={this.scrollToRef} {...otherProps}>
-        {children}
-      </div>
-    );
+    return cloneElement(children, { ref: this.scrollToRef, ...otherProps });
   }
 }
 
@@ -123,6 +120,7 @@ ScrollTo.propTypes = {
     PropTypes.func,
     PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
   ]).isRequired,
+  nodeId: PropTypes.string.isRequired,
   offsetTop: PropTypes.number,
   scrollTo: PropTypes.bool.isRequired,
 };
diff --git a/src/components/SearchHit.js b/src/components/SearchHit.js
index 6314f2f6add3f3ad81cb003bbb345d91b18f1116..41a8be606d62b03431e3b9c2a283a8591a8a9b1c 100644
--- a/src/components/SearchHit.js
+++ b/src/components/SearchHit.js
@@ -1,15 +1,49 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import clsx from 'clsx';
-import Button from '@material-ui/core/Button';
-import ListItem from '@material-ui/core/ListItem';
-import ListItemText from '@material-ui/core/ListItemText';
-import Typography from '@material-ui/core/Typography';
-import Chip from '@material-ui/core/Chip';
+import Button from '@mui/material/Button';
+import ListItem from '@mui/material/ListItem';
+import ListItemText from '@mui/material/ListItemText';
+import Typography from '@mui/material/Typography';
+import Chip from '@mui/material/Chip';
+import { styled } from '@mui/material/styles';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import TruncatedHit from '../lib/TruncatedHit';
 import { ScrollTo } from './ScrollTo';
 
+const Root = styled(ListItem, { name: 'SearchHit', slot: 'root' })(({ ownerState, theme }) => ({
+  '&.Mui-focused': {
+    '&:hover': {
+      ...(ownerState.windowSelected && {
+        backgroundColor: 'inherit',
+      }),
+    },
+    ...(ownerState.windowSelected && {
+      backgroundColor: 'inherit',
+    }),
+  },
+  paddingRight: theme.spacing(1),
+}));
+
+const CanvasLabel = styled('h4', { name: 'SearchHit', slot: 'canvasLabel' })(({ theme }) => ({
+  display: 'inline',
+  marginBottom: theme.spacing(1.5),
+}));
+
+const Counter = styled(Chip, { name: 'SearchHit', slot: 'counter' })(({ ownerState, theme }) => ({
+  // eslint-disable-next-line no-nested-ternary
+  backgroundColor: theme.palette.hitCounter.default,
+  ...(ownerState.windowSelected && {
+    backgroundColor: theme.palette.highlights.primary,
+  }),
+  ...(ownerState.adjacent && !ownerState.windowSelected && {
+    backgroundColor: theme.palette.highlights.secondary,
+  }),
+  height: 30,
+  marginRight: theme.spacing(1),
+  typography: 'subtitle2',
+  verticalAlign: 'inherit',
+}));
+
 /** */
 export class SearchHit extends Component {
   /** */
@@ -78,7 +112,6 @@ export class SearchHit extends Component {
       annotation,
       annotationLabel,
       canvasLabel,
-      classes,
       companionWindowId,
       containerRef,
       hit,
@@ -95,6 +128,25 @@ export class SearchHit extends Component {
     const truncatedHit = focused ? hit : hit && new TruncatedHit(hit, annotation);
     const truncated = hit && (truncatedHit.before !== hit.before || truncatedHit.after !== hit.after);
     const canvasLabelHtmlId = `${companionWindowId}-${index}`;
+    const ownerState = {
+      adjacent, focused, selected, windowSelected,
+    };
+
+    const header = (
+      <>
+        <Counter
+          component="span"
+          ownerState={ownerState}
+          label={index + 1}
+        />
+        <CanvasLabel id={canvasLabelHtmlId}>
+          {canvasLabel}
+          {annotationLabel && (
+            <Typography component="span" sx={{ display: 'block', marginTop: 1 }}>{annotationLabel}</Typography>
+          )}
+        </CanvasLabel>
+      </>
+    );
 
     return (
       <ScrollTo
@@ -102,51 +154,56 @@ export class SearchHit extends Component {
         offsetTop={96} // offset for the height of the form above
         scrollTo={windowSelected && !focused}
       >
-        <ListItem
-          className={clsx(
-            classes.listItem,
-            {
-              [classes.adjacent]: adjacent,
-              [classes.selected]: selected,
-              [classes.focused]: focused,
-              [classes.windowSelected]: windowSelected,
-            },
-          )}
+        <Root
+          ownerState={ownerState}
+          className={windowSelected ? 'windowSelected' : ''}
+          divider
           button={!selected}
           component="li"
           onClick={this.handleClick}
           selected={selected}
         >
-          <ListItemText primaryTypographyProps={{ variant: 'body1' }}>
-            <Typography variant="subtitle2" className={classes.subtitle}>
-              <Chip component="span" label={index + 1} className={classes.hitCounter} />
-              <span id={canvasLabelHtmlId}>
-                {canvasLabel}
-              </span>
-            </Typography>
-            {annotationLabel && (
-              <Typography variant="subtitle2">{annotationLabel}</Typography>
-            )}
-            {hit && (
+          <ListItemText
+            primary={header}
+            primaryTypographyProps={{ component: 'div', sx: { marginBottom: 1 }, variant: 'subtitle2' }}
+            secondaryTypographyProps={{ variant: 'body1' }}
+            secondary={(
               <>
-                <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.before} />
-                {' '}
-                <strong>
-                  <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.match} />
-                </strong>
-                {' '}
-                <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.after} />
-                {' '}
-                { truncated && !focused && (
-                  <Button className={classes.inlineButton} onClick={showDetails} color="secondary" size="small" aria-describedby={canvasLabelHtmlId}>
-                    {t('more')}
-                  </Button>
+                {hit && (
+                  <>
+                    <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.before} />
+                    {' '}
+                    <strong>
+                      <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.match} />
+                    </strong>
+                    {' '}
+                    <SanitizedHtml ruleSet="iiif" htmlString={truncatedHit.after} />
+                    {' '}
+                    {truncated && !focused && (
+                      <Button
+                        sx={{
+                          '& span': {
+                            lineHeight: '1.5em',
+                          },
+                          margin: 0,
+                          padding: 0,
+                          textTransform: 'none',
+                        }}
+                        onClick={showDetails}
+                        color="secondary"
+                        size="small"
+                        aria-describedby={canvasLabelHtmlId}
+                      >
+                        {t('more')}
+                      </Button>
+                    )}
+                  </>
                 )}
+                {!hit && annotation && <SanitizedHtml ruleSet="iiif" htmlString={annotation.chars} />}
               </>
             )}
-            {!hit && annotation && <SanitizedHtml ruleSet="iiif" htmlString={annotation.chars} />}
-          </ListItemText>
-        </ListItem>
+          />
+        </Root>
       </ScrollTo>
     );
   }
@@ -162,7 +219,6 @@ SearchHit.propTypes = {
   annotationLabel: PropTypes.string,
   announcer: PropTypes.func,
   canvasLabel: PropTypes.string,
-  classes: PropTypes.objectOf(PropTypes.string),
   companionWindowId: PropTypes.string,
   containerRef: PropTypes.oneOfType([
     PropTypes.func,
@@ -191,7 +247,6 @@ SearchHit.defaultProps = {
   annotationLabel: undefined,
   announcer: undefined,
   canvasLabel: undefined,
-  classes: {},
   companionWindowId: undefined,
   containerRef: undefined,
   focused: false,
diff --git a/src/components/SearchPanel.js b/src/components/SearchPanel.js
index 68063764646d16ba6669f11b6464736063b91d2f..42550dddf38e39fe4681302927f4ad07b09a379b 100644
--- a/src/components/SearchPanel.js
+++ b/src/components/SearchPanel.js
@@ -1,8 +1,8 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import Chip from '@material-ui/core/Chip';
-import Typography from '@material-ui/core/Typography';
+import Button from '@mui/material/Button';
+import Chip from '@mui/material/Chip';
+import Typography from '@mui/material/Typography';
 import CompanionWindow from '../containers/CompanionWindow';
 import SearchPanelControls from '../containers/SearchPanelControls';
 import SearchResults from '../containers/SearchResults';
@@ -19,7 +19,6 @@ export class SearchPanel extends Component {
   /** */
   render() {
     const {
-      classes,
       fetchSearch,
       windowId,
       id,
@@ -40,7 +39,7 @@ export class SearchPanel extends Component {
               query && query !== '' && (
                 <Chip
                   role="button"
-                  className={classes.clearChip}
+                  sx={{ marginLeft: 1 }}
                   color="secondary"
                   label={t('clearSearch')}
                   onClick={removeSearch}
@@ -65,8 +64,12 @@ export class SearchPanel extends Component {
         />
         {
           fetchSearch && suggestedSearches && query === '' && suggestedSearches.map(search => (
-            <Typography component="p" key={search} variant="body1">
-              <Button className={classes.inlineButton} color="secondary" onClick={() => fetchSearch(`${searchService.id}?q=${search}`, search)}>
+            <Typography component="p" key={search} variant="body1" sx={{ margin: 2 }}>
+              <Button
+                variant="inlineText"
+                color="secondary"
+                onClick={() => fetchSearch(`${searchService.id}?q=${search}`, search)}
+              >
                 {t('suggestSearch', { query: search })}
               </Button>
             </Typography>
@@ -78,10 +81,6 @@ export class SearchPanel extends Component {
 }
 
 SearchPanel.propTypes = {
-  classes: PropTypes.shape({
-    clearChip: PropTypes.string,
-    inlineButton: PropTypes.string,
-  }),
   fetchSearch: PropTypes.func,
   id: PropTypes.string.isRequired,
   query: PropTypes.string,
@@ -95,7 +94,6 @@ SearchPanel.propTypes = {
 };
 
 SearchPanel.defaultProps = {
-  classes: {},
   fetchSearch: undefined,
   query: '',
   suggestedSearches: [],
diff --git a/src/components/SearchPanelControls.js b/src/components/SearchPanelControls.js
index f16939757a6831f843b1dfd7bcb2d470fb63d4ef..0ebdc7185a243bf533c1263525b3e5dd39996f07 100644
--- a/src/components/SearchPanelControls.js
+++ b/src/components/SearchPanelControls.js
@@ -1,15 +1,23 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import deburr from 'lodash/deburr';
 import debounce from 'lodash/debounce';
 import isObject from 'lodash/isObject';
-import Autocomplete from '@material-ui/lab/Autocomplete';
-import CircularProgress from '@material-ui/core/CircularProgress';
-import TextField from '@material-ui/core/TextField';
-import SearchIcon from '@material-ui/icons/SearchSharp';
+import Autocomplete from '@mui/material/Autocomplete';
+import CircularProgress from '@mui/material/CircularProgress';
+import TextField from '@mui/material/TextField';
+import InputAdornment from '@mui/material/InputAdornment';
+import SearchIcon from '@mui/icons-material/SearchSharp';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import SearchPanelNavigation from '../containers/SearchPanelNavigation';
 
+const StyledForm = styled('form', { name: 'SearchPanelControls', slot: 'form' })(({ theme }) => ({
+  paddingBottom: theme.spacing(1),
+  paddingRight: theme.spacing(1.5),
+  width: '100%',
+}));
+
 /** Sometimes an autocomplete match can be a simple string, other times an object
     with a `match` property, this function abstracts that away */
 const getMatch = (option) => (isObject(option) ? option.match : option);
@@ -117,48 +125,62 @@ export class SearchPanelControls extends Component {
   /** */
   render() {
     const {
-      classes, companionWindowId, searchIsFetching, t, windowId,
+      companionWindowId, searchIsFetching, t, windowId,
     } = this.props;
 
     const { search, suggestions } = this.state;
     const id = `search-${companionWindowId}`;
     return (
       <>
-        <form aria-label={t('searchTitle')} onSubmit={this.submitSearch} className={classes.form}>
+        <StyledForm
+          aria-label={t('searchTitle')}
+          onSubmit={this.submitSearch}
+        >
           <Autocomplete
             id={id}
             inputValue={search}
             options={suggestions}
             getOptionLabel={getMatch}
-            getOptionSelected={(option, value) => (
+            isOptionEqualToValue={(option, value) => (
               deburr(getMatch(option).trim()).toLowerCase()
-                === deburr(getMatch(value).trim()).toLowerCase()
+              === deburr(getMatch(value).trim()).toLowerCase()
             )}
             noOptionsText=""
             onChange={this.selectItem}
             onInputChange={this.handleChange}
             freeSolo
+            disableClearable
             renderInput={params => (
               <TextField
                 {...params}
                 label={t('searchInputLabel')}
+                variant="standard"
                 InputProps={{
                   ...params.InputProps,
                   endAdornment: (
-                    <div className={classes.endAdornment}>
+                    <InputAdornment sx={{ position: 'relative' }} position="end">
                       <MiradorMenuButton aria-label={t('searchSubmitAria')} type="submit">
                         <SearchIcon />
                       </MiradorMenuButton>
                       {Boolean(searchIsFetching) && (
-                        <CircularProgress className={classes.searchProgress} size={50} />
+                      <CircularProgress
+                        sx={{
+                          left: '50%',
+                          marginLeft: '-25px',
+                          marginTop: '-25px',
+                          position: 'absolute',
+                          top: '50%',
+                        }}
+                        size={50}
+                      />
                       )}
-                    </div>
+                    </InputAdornment>
                   ),
                 }}
               />
             )}
           />
-        </form>
+        </StyledForm>
         <SearchPanelNavigation windowId={windowId} companionWindowId={companionWindowId} />
       </>
     );
@@ -169,7 +191,6 @@ SearchPanelControls.propTypes = {
   autocompleteService: PropTypes.shape({
     id: PropTypes.string,
   }),
-  classes: PropTypes.objectOf(PropTypes.string),
   companionWindowId: PropTypes.string.isRequired,
   fetchSearch: PropTypes.func.isRequired,
   query: PropTypes.string,
@@ -183,7 +204,6 @@ SearchPanelControls.propTypes = {
 
 SearchPanelControls.defaultProps = {
   autocompleteService: undefined,
-  classes: {},
   query: '',
   t: key => key,
 };
diff --git a/src/components/SearchPanelNavigation.js b/src/components/SearchPanelNavigation.js
index 72eb84e1fe181a6756d8a32644721fe28e4f7b40..3ddf2c49b5dc8631e09c7002aef30ef731c3f9e9 100644
--- a/src/components/SearchPanelNavigation.js
+++ b/src/components/SearchPanelNavigation.js
@@ -1,8 +1,8 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import ChevronLeftIcon from '@material-ui/icons/ChevronLeftSharp';
-import ChevronRightIcon from '@material-ui/icons/ChevronRightSharp';
-import Typography from '@material-ui/core/Typography';
+import ChevronLeftIcon from '@mui/icons-material/ChevronLeftSharp';
+import ChevronRightIcon from '@mui/icons-material/ChevronRightSharp';
+import Typography from '@mui/material/Typography';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 
 /**
@@ -42,7 +42,7 @@ export class SearchPanelNavigation extends Component {
   */
   render() {
     const {
-      numTotal, searchHits, selectedContentSearchAnnotation, classes, t, direction,
+      numTotal, searchHits, selectedContentSearchAnnotation, t, direction,
     } = this.props;
 
     const iconStyle = direction === 'rtl' ? { transform: 'rotate(180deg)' } : {};
@@ -57,7 +57,7 @@ export class SearchPanelNavigation extends Component {
     if (searchHits.length === 0) return null;
 
     return (
-      <Typography variant="body2" align="center" classes={classes}>
+      <Typography variant="body2" align="center">
         <MiradorMenuButton
           aria-label={t('searchPreviousResult')}
           disabled={!this.hasPreviousResult(currentHitIndex)}
@@ -80,7 +80,6 @@ export class SearchPanelNavigation extends Component {
   }
 }
 SearchPanelNavigation.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   direction: PropTypes.string.isRequired,
   numTotal: PropTypes.number,
   searchHits: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
@@ -93,7 +92,6 @@ SearchPanelNavigation.propTypes = {
   windowId: PropTypes.string.isRequired, // eslint-disable-line react/no-unused-prop-types
 };
 SearchPanelNavigation.defaultProps = {
-  classes: {},
   numTotal: undefined,
   searchHits: [],
   t: key => key,
diff --git a/src/components/SearchResults.js b/src/components/SearchResults.js
index c38e5c082a2f858ccce60237ddbd1a50c0384a2e..15263db3badcdef2e994cbd17f78073eb9c2c673 100644
--- a/src/components/SearchResults.js
+++ b/src/components/SearchResults.js
@@ -1,9 +1,9 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import List from '@material-ui/core/List';
-import Typography from '@material-ui/core/Typography';
-import BackIcon from '@material-ui/icons/ArrowBackSharp';
+import Button from '@mui/material/Button';
+import List from '@mui/material/List';
+import Typography from '@mui/material/Typography';
+import BackIcon from '@mui/icons-material/ArrowBackSharp';
 import { announce } from '@react-aria/live-announcer';
 import SearchHit from '../containers/SearchHit';
 import { ScrollTo } from './ScrollTo';
@@ -80,7 +80,6 @@ export class SearchResults extends Component {
   /** */
   render() {
     const {
-      classes,
       companionWindowId,
       containerRef,
       isFetching,
@@ -106,14 +105,18 @@ export class SearchResults extends Component {
       <>
         { focused && (
           <ScrollTo containerRef={containerRef} offsetTop={96} scrollTo>
-            <Button onClick={this.toggleFocus} className={classes.navigation} size="small">
+            <Button onClick={this.toggleFocus} sx={{ textTransform: 'none' }} size="small">
               <BackIcon />
               {t('backToResults')}
             </Button>
           </ScrollTo>
         )}
         {noResultsState && (
-          <Typography className={classes.noResults}>
+          <Typography sx={{
+            padding: 2,
+            typography: 'h6',
+          }}
+          >
             {t('searchNoResults')}
           </Typography>
         )}
@@ -122,7 +125,7 @@ export class SearchResults extends Component {
         </List>
         { nextSearch && (
           <Button
-            className={classes.moreButton}
+            sx={{ width: '100%' }}
             color="secondary"
             onClick={() => fetchSearch(windowId, companionWindowId, nextSearch, query)}
           >
@@ -137,7 +140,6 @@ export class SearchResults extends Component {
 }
 
 SearchResults.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   companionWindowId: PropTypes.string.isRequired,
   containerRef: PropTypes.oneOfType([
     PropTypes.func,
@@ -155,7 +157,6 @@ SearchResults.propTypes = {
 };
 
 SearchResults.defaultProps = {
-  classes: {},
   containerRef: undefined,
   isFetching: false,
   nextSearch: undefined,
diff --git a/src/components/SelectCollection.js b/src/components/SelectCollection.js
index cc7531393381731a4d8ff87ffc2684e4701d49f3..c51af8f531c25e41baf455a1eacb260a42eae8c2 100644
--- a/src/components/SelectCollection.js
+++ b/src/components/SelectCollection.js
@@ -1,9 +1,9 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import Grid from '@material-ui/core/Grid';
-import Typography from '@material-ui/core/Typography';
-import ListSharpIcon from '@material-ui/icons/ListSharp';
+import Button from '@mui/material/Button';
+import Grid from '@mui/material/Grid';
+import Typography from '@mui/material/Typography';
+import ListSharpIcon from '@mui/icons-material/ListSharp';
 
 /**
  *
diff --git a/src/components/SidebarIndexItem.js b/src/components/SidebarIndexItem.js
index d6b1a96668006e072062bd0cd0b1b0d0d1f9dc4e..11bb4af9c08171d10ee6a9a8be4c059a3a7711b1 100644
--- a/src/components/SidebarIndexItem.js
+++ b/src/components/SidebarIndexItem.js
@@ -1,19 +1,17 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import classNames from 'classnames';
+import Typography from '@mui/material/Typography';
 
 /** */
 export class SidebarIndexItem extends Component {
   /** */
   render() {
     const {
-      classes, label,
+      label,
     } = this.props;
 
     return (
       <Typography
-        className={classNames(classes.label)}
         variant="body1"
       >
         {label}
@@ -23,6 +21,5 @@ export class SidebarIndexItem extends Component {
 }
 
 SidebarIndexItem.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   label: PropTypes.string.isRequired,
 };
diff --git a/src/components/SidebarIndexList.js b/src/components/SidebarIndexList.js
index 5cb8f6d860df95a419309e2ac5b91e704a71adc3..d19394ef97ec83f0718e161c4171f37446829a0d 100644
--- a/src/components/SidebarIndexList.js
+++ b/src/components/SidebarIndexList.js
@@ -1,12 +1,21 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import MenuList from '@material-ui/core/MenuList';
-import MenuItem from '@material-ui/core/MenuItem';
+import MenuList from '@mui/material/MenuList';
+import MenuItem from '@mui/material/MenuItem';
+import { styled } from '@mui/material/styles';
 import { ScrollTo } from './ScrollTo';
 import MiradorCanvas from '../lib/MiradorCanvas';
 import SidebarIndexItem from '../containers/SidebarIndexItem';
 import SidebarIndexThumbnail from '../containers/SidebarIndexThumbnail';
 
+const StyledItem = styled(MenuItem, { name: 'SidebarIndexList', slot: 'item' })(({ theme }) => ({
+  alignItems: 'flex-start',
+  paddingLeft: theme.spacing(2),
+  paddingRight: theme.spacing(1),
+  position: 'initial',
+  whiteSpace: 'normal',
+}));
+
 /** */
 export class SidebarIndexList extends Component {
   /** @private */
@@ -23,7 +32,6 @@ export class SidebarIndexList extends Component {
   render() {
     const {
       canvases,
-      classes,
       containerRef,
       selectedCanvasIds,
       setCanvas,
@@ -49,24 +57,22 @@ export class SidebarIndexList extends Component {
             const onClick = () => { setCanvas(windowId, canvas.id); }; // eslint-disable-line require-jsdoc, max-len
 
             return (
-              <MenuItem
-                key={canvas.id}
-                className={classes.listItem}
-                alignItems="flex-start"
-                onClick={onClick}
-                button
-                component="li"
+              <ScrollTo
+                containerRef={containerRef}
+                key={`${canvas.id}-${variant}`}
+                offsetTop={96} // offset for the height of the form above
                 selected={selectedCanvasIds.includes(canvas.id)}
+                scrollTo={selectedCanvasIds.includes(canvas.id)}
               >
-                <ScrollTo
-                  containerRef={containerRef}
-                  key={`${canvas.id}-${variant}`}
-                  offsetTop={96} // offset for the height of the form above
-                  scrollTo={selectedCanvasIds.includes(canvas.id)}
+                <StyledItem
+                  key={canvas.id}
+                  divider
+                  onClick={onClick}
+                  component="li"
                 >
                   <Item label={canvas.label} canvas={canvases[canvasIndex]} />
-                </ScrollTo>
-              </MenuItem>
+                </StyledItem>
+              </ScrollTo>
             );
           })
         }
@@ -77,7 +83,6 @@ export class SidebarIndexList extends Component {
 
 SidebarIndexList.propTypes = {
   canvases: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   containerRef: PropTypes.oneOf([PropTypes.func, PropTypes.object]).isRequired,
   selectedCanvasIds: PropTypes.arrayOf(PropTypes.string),
   setCanvas: PropTypes.func.isRequired,
diff --git a/src/components/SidebarIndexTableOfContents.js b/src/components/SidebarIndexTableOfContents.js
index e897a4e411a38738639f0ed6559945182229ee1f..296555e8c4ef73db40f56f37dea7211c62fa5492 100644
--- a/src/components/SidebarIndexTableOfContents.js
+++ b/src/components/SidebarIndexTableOfContents.js
@@ -1,12 +1,15 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import TreeView from '@material-ui/lab/TreeView';
-import ExpandMoreIcon from '@material-ui/icons/ExpandMore';
-import ChevronRightIcon from '@material-ui/icons/ChevronRight';
-import TreeItem from '@material-ui/lab/TreeItem';
-import clsx from 'clsx';
+import { alpha, styled } from '@mui/material/styles';
+import { TreeView } from '@mui/x-tree-view/TreeView';
+import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
+import ChevronRightIcon from '@mui/icons-material/ChevronRight';
+import { TreeItem } from '@mui/x-tree-view/TreeItem';
 import { ScrollTo } from './ScrollTo';
 
+const StyledVisibleNode = styled('div')(() => ({
+
+}));
 /** */
 function getStartCanvasId(node) {
   const jsonld = node.data.__jsonld; // eslint-disable-line no-underscore-dangle
@@ -65,20 +68,13 @@ export class SidebarIndexTableOfContents extends Component {
   }
 
   /** */
-  handleKeyPressed(event, nodeId) {
+  handleNodeSelect(event, nodeId) {
     const { toggleNode } = this.props;
 
-    if (event.key === 'Enter') {
-      this.selectTreeItem(nodeId);
-    }
-
     if (event.key === ' ' || event.key === 'Spacebar') {
       toggleNode(nodeId);
     }
-  }
 
-  /** */
-  handleNodeSelect(_event, nodeId) {
     this.selectTreeItem(nodeId);
   }
 
@@ -109,7 +105,7 @@ export class SidebarIndexTableOfContents extends Component {
   /** */
   render() {
     const {
-      classes, treeStructure, visibleNodeIds, expandedNodeIds, containerRef, nodeIdToScrollTo,
+      treeStructure, visibleNodeIds, expandedNodeIds, containerRef, nodeIdToScrollTo,
     } = this.props;
 
     if (!treeStructure) {
@@ -127,22 +123,45 @@ export class SidebarIndexTableOfContents extends Component {
       >
         <TreeItem
           nodeId={node.id}
-          classes={{
-            content: classes.content,
-            group: classes.group,
-            label: classes.label,
-            root: classes.treeItemRoot,
-            selected: classes.selected,
+          sx={{
+            '& .MuiTreeItem-content': {
+              alignItems: 'flex-start',
+              borderLeft: '1px solid transparent',
+              padding: '8px 16px 8px 0',
+              width: 'auto',
+            },
+            '& .MuiTreeItem-group': {
+              borderLeft: '1px solid',
+              borderLeftColor: 'grey.300',
+            },
+            '& .MuiTreeItem-iconContainer': {
+              paddingBlockStart: 0.5,
+            },
+            '& .MuiTreeItem-label': {
+              paddingLeft: 0,
+            },
+            '& .MuiTreeItem-root': {
+              '&:focus > .MuiTreeItem-content': {
+                backgroundColor: 'action.selected',
+              },
+              '&:hover > .MuiTreeItem-content': {
+                backgroundColor: 'action.hover',
+              },
+              '&:hover > .MuiTreeItem-content .MuiTreeItem-label, &:focus > .MuiTreeItem-content .MuiTreeItem-label, &.MuiTreeItem-selected > .MuiTreeItem-content .MuiTreeItem-label, &.MuiTreeItem-selected > .MuiTreeItem-content .MuiTreeItem-label:hover, &.MuiTreeItem-selected:focus > .MuiTreeItem-content .MuiTreeItem-label': {
+                backgroundColor: 'transparent',
+              },
+            },
           }}
-          onKeyDown={e => this.handleKeyPressed(e, node.id)}
           label={(
-            <div
-              className={clsx({
-                [classes.visibleNode]: visibleNodeIds.indexOf(node.id) !== -1,
+            <StyledVisibleNode
+              sx={theme => ({
+                backgroundColor: visibleNodeIds.indexOf(node.id) !== -1
+                && alpha(theme.palette.highlights?.primary || theme.palette.action.selected, 0.35),
+                display: visibleNodeIds.indexOf(node.id) !== -1 && 'inline',
               })}
             >
               {node.label}
-            </div>
+            </StyledVisibleNode>
         )}
         >
           {Array.isArray(node.nodes) ? node.nodes.map((n) => renderTree(n)) : null}
@@ -152,7 +171,7 @@ export class SidebarIndexTableOfContents extends Component {
 
     return (
       <TreeView
-        className={classes.root}
+        sx={{ flexGrow: 1 }}
         defaultCollapseIcon={<ExpandMoreIcon color="action" />}
         defaultExpandIcon={<ChevronRightIcon color="action" />}
         defaultEndIcon={null}
@@ -167,7 +186,6 @@ export class SidebarIndexTableOfContents extends Component {
 }
 
 SidebarIndexTableOfContents.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   containerRef: PropTypes.oneOfType([
     PropTypes.func,
     PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
diff --git a/src/components/SidebarIndexThumbnail.js b/src/components/SidebarIndexThumbnail.js
index 7abd1973d6051d0e71e4c18bbaf40bd3a4031669..05126d8de02dc6d6d184bea3d94bb838607e9335 100644
--- a/src/components/SidebarIndexThumbnail.js
+++ b/src/components/SidebarIndexThumbnail.js
@@ -1,7 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import classNames from 'classnames';
+import Typography from '@mui/material/Typography';
 import IIIFThumbnail from '../containers/IIIFThumbnail';
 
 /** */
@@ -9,7 +8,7 @@ export class SidebarIndexThumbnail extends Component {
   /** */
   render() {
     const {
-      classes, canvas, height, label, width,
+      canvas, height, label, width,
     } = this.props;
 
     return (
@@ -18,15 +17,11 @@ export class SidebarIndexThumbnail extends Component {
           <IIIFThumbnail
             label={label}
             resource={canvas}
-            className={classNames(classes.clickable)}
             maxHeight={height}
             maxWidth={width}
           />
         </div>
-        <Typography
-          className={classNames(classes.label)}
-          variant="body1"
-        >
+        <Typography>
           {label}
         </Typography>
       </>
@@ -36,7 +31,6 @@ export class SidebarIndexThumbnail extends Component {
 
 SidebarIndexThumbnail.propTypes = {
   canvas: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   height: PropTypes.number,
   label: PropTypes.string.isRequired,
   width: PropTypes.number,
diff --git a/src/components/ThumbnailCanvasGrouping.js b/src/components/ThumbnailCanvasGrouping.js
index 6e8902339c361816aad67a35f270418eee0eed4b..f01d399260c4e032e300683e73c133cec53cc130 100644
--- a/src/components/ThumbnailCanvasGrouping.js
+++ b/src/components/ThumbnailCanvasGrouping.js
@@ -1,9 +1,17 @@
 import { PureComponent } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
 import IIIFThumbnail from '../containers/IIIFThumbnail';
 import ns from '../config/css-ns';
 
+const StyledCanvas = styled('div')(({ theme }) => ({
+  boxSizing: 'border-box',
+  color: theme.palette.common.white,
+  cursor: 'pointer',
+  display: 'inline-block',
+  whiteSpace: 'nowrap',
+}));
 /** */
 export class ThumbnailCanvasGrouping extends PureComponent {
   /** */
@@ -31,7 +39,7 @@ export class ThumbnailCanvasGrouping extends PureComponent {
   /** */
   render() {
     const {
-      index, style, data, classes, currentCanvasId,
+      index, style, data, currentCanvasId,
     } = this.props;
     const {
       canvasGroupings, position, height,
@@ -52,24 +60,27 @@ export class ThumbnailCanvasGrouping extends PureComponent {
         role="gridcell"
         aria-colindex={index + 1}
       >
-        <div
+        <StyledCanvas
           role="button"
           data-canvas-id={currentGroupings[0].id}
           data-canvas-index={currentGroupings[0].index}
           onKeyUp={this.setCanvas}
           onClick={this.setCanvas}
           tabIndex={-1}
-          style={{
+          sx={theme => ({
+            '&:hover': {
+              outline: `9px solid ${theme.palette.action.hover}`,
+              outlineOffset: '-2px',
+            },
             height: (position === 'far-right') ? 'auto' : `${height - SPACING}px`,
+            outline: currentGroupings.map(canvas => canvas.id).includes(currentCanvasId) ? `2px solid ${theme.palette.primary.main}` : 0,
+            ...(currentGroupings.map(canvas => canvas.id).includes(currentCanvasId) && {
+              outlineOffset: '3px',
+            }),
             width: (position === 'far-bottom') ? 'auto' : `${style.width}px`,
-          }}
+          })}
           className={classNames(
             ns(['thumbnail-nav-canvas', `thumbnail-nav-canvas-${index}`, this.currentCanvasClass(currentGroupings.map(canvas => canvas.index))]),
-            classes.canvas,
-            {
-              [classes.currentCanvas]: currentGroupings
-                .map(canvas => canvas.id).includes(currentCanvasId),
-            },
           )}
         >
           {currentGroupings.map((canvas, i) => (
@@ -81,14 +92,13 @@ export class ThumbnailCanvasGrouping extends PureComponent {
               variant="inside"
             />
           ))}
-        </div>
+        </StyledCanvas>
       </div>
     );
   }
 }
 
 ThumbnailCanvasGrouping.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   currentCanvasId: PropTypes.string.isRequired,
   data: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
   index: PropTypes.number.isRequired,
diff --git a/src/components/ThumbnailNavigation.js b/src/components/ThumbnailNavigation.js
index 59219d4446528d1b66c3cd300f5e137e1b1c20b3..1a4ce20cf69e8e7c9daf22151cc1c8aa4cadab01 100644
--- a/src/components/ThumbnailNavigation.js
+++ b/src/components/ThumbnailNavigation.js
@@ -1,6 +1,6 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
-import Paper from '@material-ui/core/Paper';
+import Paper from '@mui/material/Paper';
 import AutoSizer from 'react-virtualized-auto-sizer';
 import { VariableSizeList as List } from 'react-window';
 import classNames from 'classnames';
@@ -176,7 +176,6 @@ export class ThumbnailNavigation extends Component {
     const {
       t,
       canvasGroupings,
-      classes,
       position,
       thumbnailNavigation,
       viewingDirection,
@@ -196,8 +195,13 @@ export class ThumbnailNavigation extends Component {
       <Paper
         className={classNames(
           ns('thumb-navigation'),
-          classes.thumbNavigation,
         )}
+        sx={{
+          '&:focus': {
+            boxShadow: 0,
+            outline: 0,
+          },
+        }}
         aria-label={t('thumbnailNavigation')}
         square
         elevation={0}
@@ -235,7 +239,6 @@ export class ThumbnailNavigation extends Component {
 ThumbnailNavigation.propTypes = {
   canvasGroupings: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types
   canvasIndex: PropTypes.number.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   hasNextCanvas: PropTypes.bool,
   hasPreviousCanvas: PropTypes.bool,
   position: PropTypes.string.isRequired,
diff --git a/src/components/VideoViewer.js b/src/components/VideoViewer.js
index e8a3d617df6b3f516fbacb7abcfd7b05c2894a78..a133aceed3b06116bd430d162afcbd70d6d8fa3d 100644
--- a/src/components/VideoViewer.js
+++ b/src/components/VideoViewer.js
@@ -1,5 +1,17 @@
 import { Component, Fragment } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
+
+const StyledContainer = styled('div')(() => ({
+  alignItems: 'center',
+  display: 'flex',
+  width: '100%',
+}));
+
+const StyledVideo = styled('video')(() => ({
+  maxHeight: '100%',
+  width: '100%',
+}));
 
 /** */
 export class VideoViewer extends Component {
@@ -7,11 +19,11 @@ export class VideoViewer extends Component {
   /** */
   render() {
     const {
-      captions, classes, videoOptions, videoResources,
+      captions, videoOptions, videoResources,
     } = this.props;
     return (
-      <div className={classes.container}>
-        <video className={classes.video} {...videoOptions}>
+      <StyledContainer>
+        <StyledVideo {...videoOptions}>
           {videoResources.map(video => (
             <Fragment key={video.id}>
               <source src={video.id} type={video.getFormat()} />
@@ -22,8 +34,8 @@ export class VideoViewer extends Component {
               <track src={caption.id} label={caption.getDefaultLabel()} srcLang={caption.getProperty('language')} />
             </Fragment>
           ))}
-        </video>
-      </div>
+        </StyledVideo>
+      </StyledContainer>
     );
   }
   /* eslint-enable jsx-a11y/media-has-caption */
@@ -31,7 +43,6 @@ export class VideoViewer extends Component {
 
 VideoViewer.propTypes = {
   captions: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   videoOptions: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   videoResources: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
 };
diff --git a/src/components/ViewerInfo.js b/src/components/ViewerInfo.js
index 595ed37d0f7c94bbef408772c3e39c419a14a481..1958bea5e714189630e6ecc586dcaa0f7b5b646b 100644
--- a/src/components/ViewerInfo.js
+++ b/src/components/ViewerInfo.js
@@ -1,9 +1,19 @@
 import { Component } from 'react';
-import Typography from '@material-ui/core/Typography';
+import Typography from '@mui/material/Typography';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
 import ns from '../config/css-ns';
 
+const StyledOsdInfo = styled('div')(() => ({
+  overflow: 'hidden',
+  paddingBottom: 0.5,
+  textOverflow: 'ellipsis',
+  unicodeBidi: 'plaintext',
+  whiteSpace: 'nowrap',
+  width: '100%',
+}));
+
 /**
  *
  */
@@ -14,19 +24,18 @@ export class ViewerInfo extends Component {
       canvasCount,
       canvasIndex,
       canvasLabel,
-      classes,
       t,
     } = this.props;
 
     return (
-      <div className={classNames(ns('osd-info'), classes.osdInfo)}>
+      <StyledOsdInfo className={classNames(ns('osd-info'))}>
         <Typography display="inline" variant="caption" className={ns('canvas-count')}>
           { t('pagination', { current: canvasIndex + 1, total: canvasCount }) }
         </Typography>
         <Typography display="inline" variant="caption" className={ns('canvas-label')}>
           {canvasLabel && ` • ${canvasLabel}`}
         </Typography>
-      </div>
+      </StyledOsdInfo>
     );
   }
 }
@@ -40,6 +49,5 @@ ViewerInfo.propTypes = {
   canvasCount: PropTypes.number.isRequired,
   canvasIndex: PropTypes.number.isRequired,
   canvasLabel: PropTypes.string,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   t: PropTypes.func,
 };
diff --git a/src/components/ViewerNavigation.js b/src/components/ViewerNavigation.js
index 0253cefa5278df9db46b91480be5dac6c12362e2..e22b4ce4fac1b116b72654e80acd030042850e2e 100644
--- a/src/components/ViewerNavigation.js
+++ b/src/components/ViewerNavigation.js
@@ -1,5 +1,5 @@
 import { Component } from 'react';
-import NavigationIcon from '@material-ui/icons/PlayCircleOutlineSharp';
+import NavigationIcon from '@mui/icons-material/PlayCircleOutlineSharp';
 import PropTypes from 'prop-types';
 import classNames from 'classnames';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
@@ -14,7 +14,7 @@ export class ViewerNavigation extends Component {
   render() {
     const {
       hasNextCanvas, hasPreviousCanvas, setNextCanvas, setPreviousCanvas, t,
-      classes, viewingDirection,
+      viewingDirection,
     } = this.props;
 
     let htmlDir = 'ltr';
@@ -41,7 +41,7 @@ export class ViewerNavigation extends Component {
 
     return (
       <div
-        className={classNames(ns('osd-navigation'), classes.osdNavigation)}
+        className={classNames(ns('osd-navigation'))}
         dir={htmlDir}
       >
         <MiradorMenuButton
@@ -66,7 +66,6 @@ export class ViewerNavigation extends Component {
 }
 
 ViewerNavigation.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   hasNextCanvas: PropTypes.bool,
   hasPreviousCanvas: PropTypes.bool,
   setNextCanvas: PropTypes.func,
diff --git a/src/components/Window.js b/src/components/Window.js
index 6033c76919ba2e2364984d351fa2267eb4d93447..4610fb6cf2d4116f9edd1901bf6988a88b9c5b05 100644
--- a/src/components/Window.js
+++ b/src/components/Window.js
@@ -1,7 +1,7 @@
-import { Component } from 'react';
+import { Component, useContext } from 'react';
 import PropTypes from 'prop-types';
-import cn from 'classnames';
-import Paper from '@material-ui/core/Paper';
+import { styled } from '@mui/material/styles';
+import Paper from '@mui/material/Paper';
 import { MosaicWindowContext } from 'react-mosaic-component/lib/contextTypes';
 import ns from '../config/css-ns';
 import WindowTopBar from '../containers/WindowTopBar';
@@ -12,6 +12,68 @@ import ErrorContent from '../containers/ErrorContent';
 import IIIFAuthentication from '../containers/IIIFAuthentication';
 import { PluginHook } from './PluginHook';
 
+const rowMixin = {
+  display: 'flex',
+  flex: '1',
+  flexDirection: 'row',
+  minHeight: 0,
+};
+
+const columnMixin = {
+  display: 'flex',
+  flex: '1',
+  flexDirection: 'column',
+  minHeight: 0,
+};
+
+const Root = styled(Paper, { name: 'Window', slot: 'root' })(({ ownerState, theme }) => ({
+  ...columnMixin,
+  backgroundColor: theme.palette.shades?.dark,
+  borderRadius: 0,
+  height: '100%',
+  overflow: 'hidden',
+  width: '100%',
+  ...(ownerState?.maximized && {
+    left: 0,
+    position: 'absolute',
+    top: 0,
+    zIndex: theme.zIndex.modal - 1,
+  }),
+}));
+
+const ContentRow = styled('div', { name: 'Window', slot: 'row' })(() => ({
+  ...rowMixin,
+}));
+
+const ContentColumn = styled('div', { name: 'Window', slot: 'column' })(() => ({
+  ...columnMixin,
+}));
+
+const StyledPrimaryWindow = styled(PrimaryWindow, { name: 'Window', slot: 'primary' })(() => ({
+  ...rowMixin,
+  height: '300px',
+  position: 'relative',
+}));
+
+const StyledCompanionAreaBottom = styled(CompanionArea, { name: 'Window', slot: 'bottom' })(() => ({
+  ...rowMixin,
+  flex: '0',
+  flexBasis: 'auto',
+}));
+
+const StyledCompanionAreaRight = styled('div', { name: 'Window', slot: 'right' })(() => ({
+  ...rowMixin,
+  flex: '0 1 auto',
+}));
+
+/** Window title bar wrapper for drag controls in the mosaic view */
+const DraggableNavBar = ({ children, ...props }) => {
+  const { mosaicWindowActions } = useContext(MosaicWindowContext);
+  return mosaicWindowActions.connectDragSource(
+    <nav {...props}>{children}</nav>,
+  );
+};
+
 /**
  * Represents a Window in the mirador workspace
  * @param {object} window
@@ -29,40 +91,13 @@ export class Window extends Component {
     return { error, hasError: true };
   }
 
-  /**
-   * wrappedTopBar - will conditionally wrap a WindowTopBar for needed
-   * additional functionality based on workspace type
-   */
-  wrappedTopBar() {
-    const {
-      windowId, workspaceType, windowDraggable,
-    } = this.props;
-
-    const topBar = (
-      <div>
-        <WindowTopBar
-          windowId={windowId}
-          windowDraggable={windowDraggable}
-        />
-        <IIIFAuthentication windowId={windowId} />
-      </div>
-    );
-    if (workspaceType === 'mosaic' && windowDraggable) {
-      const { mosaicWindowActions } = this.context;
-      return mosaicWindowActions.connectDragSource(
-        topBar,
-      );
-    }
-    return topBar;
-  }
-
   /**
    * Renders things
    */
   render() {
     const {
-      focusWindow, label, isFetching, maximized, sideBarOpen,
-      view, windowId, classes, t,
+      focusWindow, label, isFetching, sideBarOpen,
+      view, windowDraggable, windowId, workspaceType, t,
       manifestError,
     } = this.props;
 
@@ -77,44 +112,40 @@ export class Window extends Component {
     }
 
     return (
-      <Paper
+      <Root
         onFocus={focusWindow}
+        ownerState={this.props}
         component="section"
         elevation={1}
         id={windowId}
-        className={
-          cn(
-            classes.window,
-            ns('window'),
-            maximized ? classes.maximized : null,
-          )
-}
+        className={ns('window')}
         aria-label={t('window', { label })}
       >
-        {this.wrappedTopBar()}
+        <WindowTopBar
+          component={workspaceType === 'mosaic' && windowDraggable ? DraggableNavBar : undefined}
+          windowId={windowId}
+          windowDraggable={windowDraggable}
+        />
+        <IIIFAuthentication windowId={windowId} />
         { manifestError && <ErrorContent error={{ stack: manifestError }} windowId={windowId} /> }
-        <div className={classes.middle}>
-          <div className={classes.middleLeft}>
-            <div className={classes.primaryWindow}>
-              <PrimaryWindow
-                view={view}
-                windowId={windowId}
-                isFetching={isFetching}
-                sideBarOpen={sideBarOpen}
-              />
-            </div>
-            <div className={classes.companionAreaBottom}>
-              <CompanionArea windowId={windowId} position="bottom" />
-            </div>
-          </div>
-          <div className={classes.companionAreaRight}>
+        <ContentRow>
+          <ContentColumn>
+            <StyledPrimaryWindow
+              view={view}
+              windowId={windowId}
+              isFetching={isFetching}
+              sideBarOpen={sideBarOpen}
+            />
+            <StyledCompanionAreaBottom windowId={windowId} position="bottom" />
+          </ContentColumn>
+          <StyledCompanionAreaRight>
             <CompanionArea windowId={windowId} position="right" />
             <CompanionArea windowId={windowId} position="far-right" />
-          </div>
-        </div>
+          </StyledCompanionAreaRight>
+        </ContentRow>
         <CompanionArea windowId={windowId} position="far-bottom" />
         <PluginHook {...this.props} />
-      </Paper>
+      </Root>
     );
   }
 }
@@ -122,7 +153,6 @@ export class Window extends Component {
 Window.contextType = MosaicWindowContext;
 
 Window.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   focusWindow: PropTypes.func,
   isFetching: PropTypes.bool,
   label: PropTypes.string,
@@ -137,7 +167,6 @@ Window.propTypes = {
 };
 
 Window.defaultProps = {
-  classes: {},
   focusWindow: () => {},
   isFetching: false,
   label: null,
diff --git a/src/components/WindowAuthenticationBar.js b/src/components/WindowAuthenticationBar.js
index bbfbd5a5388cc7889297364298b394c23edcdb96..05d2a365f69dd75d1c3bd331d9c8d8a9079e376f 100644
--- a/src/components/WindowAuthenticationBar.js
+++ b/src/components/WindowAuthenticationBar.js
@@ -1,14 +1,26 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Button from '@material-ui/core/Button';
-import Paper from '@material-ui/core/Paper';
-import Collapse from '@material-ui/core/Collapse';
-import DialogActions from '@material-ui/core/DialogActions';
-import Typography from '@material-ui/core/Typography';
-import LockIcon from '@material-ui/icons/LockSharp';
+import { alpha, styled } from '@mui/material/styles';
+import Button from '@mui/material/Button';
+import Paper from '@mui/material/Paper';
+import Collapse from '@mui/material/Collapse';
+import DialogActions from '@mui/material/DialogActions';
+import Typography from '@mui/material/Typography';
+import LockIcon from '@mui/icons-material/LockSharp';
 import SanitizedHtml from '../containers/SanitizedHtml';
 import { PluginHook } from './PluginHook';
 
+const StyledTopBar = styled('div')(({ theme }) => ({
+  '&:hover': {
+    backgroundColor: theme.palette.secondary.main,
+  },
+  alignItems: 'center',
+  display: 'flex',
+}));
+
+const StyledFauxButton = styled('span')(({ theme }) => ({
+  marginLeft: theme.spacing(2.5),
+}));
 /** */
 export class WindowAuthenticationBar extends Component {
   /** */
@@ -35,7 +47,7 @@ export class WindowAuthenticationBar extends Component {
   /** */
   render() {
     const {
-      classes, confirmButton, continueLabel,
+      confirmButton, continueLabel,
       header, description, icon, label, t,
       ruleSet, hasLogoutService, status, ConfirmProps,
     } = this.props;
@@ -45,47 +57,90 @@ export class WindowAuthenticationBar extends Component {
     const { open } = this.state;
 
     const button = (
-      <Button onClick={this.onSubmit} className={classes.buttonInvert} color="secondary" {...ConfirmProps}>
+      <Button
+        onClick={this.onSubmit}
+        color="secondary"
+        sx={(theme) => ({
+          '&:hover': {
+            backgroundColor: alpha(theme.palette.secondary.contrastText, 1 - theme.palette.action.hoverOpacity),
+          },
+          backgroundColor: theme.palette.secondary.contrastText,
+        })}
+        {...ConfirmProps}
+      >
         {confirmButton || t('login')}
       </Button>
     );
 
     if (!description && !header) {
       return (
-        <Paper square elevation={4} color="secondary" classes={{ root: classes.paper }}>
-          <div className={classes.topBar}>
-            { icon || <LockIcon className={classes.icon} /> }
-            <Typography className={classes.label} component="h3" variant="body1" color="inherit">
+        <Paper
+          square
+          elevation={4}
+          color="secondary"
+        >
+          <StyledTopBar>
+            { icon || (
+              <LockIcon sx={{ marginInlineEnd: 1.5 }} />
+            ) }
+            <Typography component="h3" variant="body1" color="inherit">
               { ruleSet ? <SanitizedHtml htmlString={label} ruleSet={ruleSet} /> : label }
             </Typography>
             <PluginHook {...this.props} />
             { button }
-          </div>
+          </StyledTopBar>
         </Paper>
       );
     }
 
     return (
-      <Paper square elevation={4} color="secondary" classes={{ root: classes.paper }}>
-        <Button fullWidth className={classes.topBar} onClick={() => this.setOpen(true)} component="div" color="inherit">
-          { icon || <LockIcon className={classes.icon} /> }
-          <Typography className={classes.label} component="h3" variant="body1" color="inherit">
+      <Paper
+        square
+        elevation={4}
+        color="secondary"
+      >
+        <Button
+          fullWidth
+          onClick={() => this.setOpen(true)}
+          component="div"
+          color="inherit"
+          sx={(theme) => ({
+            '&:hover': {
+              backgroundColor: theme.palette.secondary.main,
+            },
+            backgroundColor: theme.palette.secondary.main,
+            borderRadius: 0,
+            color: theme.palette.secondary.contrastText,
+            justifyContent: 'start',
+            textTransform: 'none',
+          })}
+        >
+          { icon || (
+          <LockIcon sx={{ marginInlineEnd: 1.5 }} />
+          ) }
+          <Typography sx={{ paddingBlockEnd: 1, paddingBlockStart: 1 }} component="h3" variant="body1" color="inherit">
             { ruleSet ? <SanitizedHtml htmlString={label} ruleSet={ruleSet} /> : label }
           </Typography>
           <PluginHook {...this.props} />
-          <span className={classes.fauxButton}>
+          <StyledFauxButton>
             { !open && (
               <Typography variant="button" color="inherit">
                 { continueLabel || t('continue') }
               </Typography>
             )}
-          </span>
+          </StyledFauxButton>
         </Button>
         <Collapse
+          sx={(theme) => ({
+            backgroundColor: theme.palette.secondary.main,
+            color: theme.palette.secondary.contrastText,
+            paddingInlineEnd: theme.spacing(1),
+            paddingInlineStart: theme.spacing(1),
+          })}
           in={open}
           onClose={() => this.setOpen(false)}
         >
-          <Typography variant="body1" color="inherit" className={classes.expanded}>
+          <Typography variant="body1" color="inherit">
             { ruleSet ? <SanitizedHtml htmlString={header} ruleSet={ruleSet} /> : header }
             { header && description ? ': ' : '' }
             { ruleSet ? <SanitizedHtml htmlString={description} ruleSet={ruleSet} /> : description }
@@ -104,7 +159,6 @@ export class WindowAuthenticationBar extends Component {
 }
 
 WindowAuthenticationBar.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   confirmButton: PropTypes.string,
   ConfirmProps: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   continueLabel: PropTypes.string,
diff --git a/src/components/WindowCanvasNavigationControls.js b/src/components/WindowCanvasNavigationControls.js
index eaee4dc152d82477faf268ccb1f636010fda3647..df9e3ae73788c18d6993a394211704d329d8edda 100644
--- a/src/components/WindowCanvasNavigationControls.js
+++ b/src/components/WindowCanvasNavigationControls.js
@@ -1,14 +1,33 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { alpha, styled } from '@mui/material/styles';
 import classNames from 'classnames';
-import Paper from '@material-ui/core/Paper';
-import Typography from '@material-ui/core/Typography';
+import Paper from '@mui/material/Paper';
+import Stack from '@mui/material/Stack';
+import Divider from '@mui/material/Divider';
+import Typography from '@mui/material/Typography';
+import { visuallyHidden } from '@mui/utils';
 import ZoomControls from '../containers/ZoomControls';
 import ViewerInfo from '../containers/ViewerInfo';
 import ViewerNavigation from '../containers/ViewerNavigation';
 import ns from '../config/css-ns';
 import { PluginHook } from './PluginHook';
 
+const Root = styled(Paper, { name: 'WindowCanvasNavigationControls', slot: 'root' })(({ theme }) => ({
+  alignItems: 'center',
+  backgroundColor: alpha(theme.palette.background.paper, 0.5),
+  bottom: 0,
+  cursor: 'default',
+  display: 'flex',
+  flexDirection: 'column',
+  flexWrap: 'wrap',
+  justifyContent: 'center',
+  position: 'absolute',
+  textAlign: 'center',
+  width: '100%',
+  zIndex: 50,
+}));
+
 /**
  * Represents the viewer controls in the mirador workspace.
  */
@@ -25,41 +44,40 @@ export class WindowCanvasNavigationControls extends Component {
   /** */
   render() {
     const {
-      classes, visible, windowId, zoomToWorld,
+      showZoomControls, visible, windowId, zoomToWorld,
     } = this.props;
 
-    if (!visible) return (<Typography variant="srOnly" component="div"><ViewerInfo windowId={windowId} /></Typography>);
+    if (!visible) return (<Typography style={visuallyHidden} component="div"><ViewerInfo windowId={windowId} /></Typography>);
 
     return (
-      <Paper
+      <Root
         square
         className={
           classNames(
-            classes.controls,
             ns('canvas-nav'),
-            classes.canvasNav,
             this.canvasNavControlsAreStacked() ? ns('canvas-nav-stacked') : null,
-            this.canvasNavControlsAreStacked() ? classes.canvasNavStacked : null,
           )
-}
+        }
         elevation={0}
       >
-        <ZoomControls
-          displayDivider={!this.canvasNavControlsAreStacked()}
-          windowId={windowId}
-          zoomToWorld={zoomToWorld}
-        />
-        <ViewerNavigation windowId={windowId} />
+        <Stack
+          direction={this.canvasNavControlsAreStacked() ? 'column' : 'row'}
+          divider={<Divider orientation={this.canvasNavControlsAreStacked() ? 'horizontal' : 'vertical'} variant="middle" flexItem />}
+          spacing={0}
+        >
+          { showZoomControls && <ZoomControls windowId={windowId} zoomToWorld={zoomToWorld} /> }
+          <ViewerNavigation windowId={windowId} />
+        </Stack>
         <ViewerInfo windowId={windowId} />
 
         <PluginHook {...this.props} />
-      </Paper>
+      </Root>
     );
   }
 }
 
 WindowCanvasNavigationControls.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
+  showZoomControls: PropTypes.bool,
   size: PropTypes.shape({ width: PropTypes.number }).isRequired,
   visible: PropTypes.bool,
   windowId: PropTypes.string.isRequired,
@@ -67,6 +85,6 @@ WindowCanvasNavigationControls.propTypes = {
 };
 
 WindowCanvasNavigationControls.defaultProps = {
-  classes: {},
+  showZoomControls: false,
   visible: true,
 };
diff --git a/src/components/WindowList.js b/src/components/WindowList.js
index 8ed83f402ca09c3a7c495b3ae1094229e9ff85ec..2db96ba42b70f6672147013dfadac62309f3ad2f 100644
--- a/src/components/WindowList.js
+++ b/src/components/WindowList.js
@@ -1,8 +1,8 @@
 import { Component } from 'react';
-import Menu from '@material-ui/core/Menu';
-import MenuItem from '@material-ui/core/MenuItem';
-import ListItemText from '@material-ui/core/ListItemText';
-import ListSubheader from '@material-ui/core/ListSubheader';
+import Menu from '@mui/material/Menu';
+import MenuItem from '@mui/material/MenuItem';
+import ListItemText from '@mui/material/ListItemText';
+import ListSubheader from '@mui/material/ListSubheader';
 import PropTypes from 'prop-types';
 
 /**
diff --git a/src/components/WindowListButton.js b/src/components/WindowListButton.js
index 68a9bd2ac974089ddfb88609ab29b27bb4cf5954..758e227ac71202a71e2da63f7474852d608de962 100644
--- a/src/components/WindowListButton.js
+++ b/src/components/WindowListButton.js
@@ -1,7 +1,6 @@
 import { Component } from 'react';
-import BookmarksIcon from '@material-ui/icons/BookmarksSharp';
+import BookmarksIcon from '@mui/icons-material/BookmarksSharp';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
 import WindowList from '../containers/WindowList';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 
@@ -33,7 +32,7 @@ export class WindowListButton extends Component {
   */
   render() {
     const {
-      classes, disabled, t, windowCount,
+      disabled, t, windowCount,
     } = this.props;
     const { windowListAnchor } = this.state;
 
@@ -43,24 +42,29 @@ export class WindowListButton extends Component {
           aria-haspopup="true"
           aria-label={t('listAllOpenWindows')}
           aria-owns={windowListAnchor ? 'window-list' : null}
-          className={
-            classNames(classes.ctrlBtn, (windowListAnchor ? classes.ctrlBtnSelected : null))
-          }
+          selected={Boolean(windowListAnchor)}
           disabled={disabled}
           badge
-          BadgeProps={{ badgeContent: windowCount, classes: { badge: classes.badge } }}
-          onClick={e => this.handleOpen(e)}
+          BadgeProps={{
+            badgeContent: windowCount,
+            sx: {
+              '.MuiBadge-badge': {
+                paddingLeft: 1.5,
+              },
+            },
+          }}
+          onClick={(e) => this.handleOpen(e)}
         >
           <BookmarksIcon />
         </MiradorMenuButton>
 
         {Boolean(windowListAnchor) && (
-          <WindowList
-            anchorEl={windowListAnchor}
-            id="window-list"
-            open={Boolean(windowListAnchor)}
-            handleClose={this.handleClose}
-          />
+        <WindowList
+          anchorEl={windowListAnchor}
+          id="window-list"
+          open={Boolean(windowListAnchor)}
+          handleClose={this.handleClose}
+        />
         )}
       </>
     );
@@ -68,12 +72,10 @@ export class WindowListButton extends Component {
 }
 
 WindowListButton.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   disabled: PropTypes.bool,
   t: PropTypes.func.isRequired,
   windowCount: PropTypes.number.isRequired,
 };
 WindowListButton.defaultProps = {
-  classes: {},
   disabled: false,
 };
diff --git a/src/components/WindowSideBar.js b/src/components/WindowSideBar.js
index c6fbc6ef3f75ecd493fc2a36b9df35fe234539f6..7750eeb294d9e3e9e42651fbd7870da8a999e37b 100644
--- a/src/components/WindowSideBar.js
+++ b/src/components/WindowSideBar.js
@@ -1,9 +1,20 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
-import Drawer from '@material-ui/core/Drawer';
+import { styled } from '@mui/material/styles';
+import Drawer from '@mui/material/Drawer';
 import WindowSideBarButtons from '../containers/WindowSideBarButtons';
 
+const Root = styled(Drawer, { name: 'WindowSideBar', slot: 'root' })(({ theme }) => ({
+  flexShrink: 0,
+  order: -1000,
+  zIndex: theme.zIndex.appBar - 1,
+}));
+
+const Nav = styled('nav', { name: 'WindowSideBar', slot: 'nav' })({
+  position: 'relative !important',
+  width: 48,
+});
+
 /**
  * WindowSideBar
  */
@@ -18,27 +29,26 @@ export class WindowSideBar extends Component {
     } = this.props;
 
     return (
-      <Drawer
+      <Root
         variant="persistent"
-        className={classNames(classes.drawer)}
-        classes={{ paper: classNames(classes.paper) }}
+        className={classes.drawer}
         anchor={direction === 'rtl' ? 'right' : 'left'}
         PaperProps={{
           'aria-label': t('sidebarPanelsNavigation'),
-          component: 'nav',
-          style: { height: '100%', position: 'relative' },
+          component: Nav,
+          variant: 'outlined',
         }}
         SlideProps={{ direction: direction === 'rtl' ? 'left' : 'right', mountOnEnter: true, unmountOnExit: true }}
         open={sideBarOpen}
       >
         <WindowSideBarButtons windowId={windowId} />
-      </Drawer>
+      </Root>
     );
   }
 }
 
 WindowSideBar.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
+  classes: PropTypes.objectOf(PropTypes.string),
   direction: PropTypes.string.isRequired,
   sideBarOpen: PropTypes.bool,
   t: PropTypes.func.isRequired,
@@ -46,5 +56,6 @@ WindowSideBar.propTypes = {
 };
 
 WindowSideBar.defaultProps = {
+  classes: {},
   sideBarOpen: false,
 };
diff --git a/src/components/WindowSideBarAnnotationsPanel.js b/src/components/WindowSideBarAnnotationsPanel.js
index f5182e6443dcfc8430aba05f808200e6ac83f25b..d2e6f781d0c3b160395df19809fa5f2ea39dc846 100644
--- a/src/components/WindowSideBarAnnotationsPanel.js
+++ b/src/components/WindowSideBarAnnotationsPanel.js
@@ -1,9 +1,10 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
+import Typography from '@mui/material/Typography';
 import AnnotationSettings from '../containers/AnnotationSettings';
 import CanvasAnnotations from '../containers/CanvasAnnotations';
 import CompanionWindow from '../containers/CompanionWindow';
+import { CompanionWindowSection } from './CompanionWindowSection';
 import ns from '../config/css-ns';
 
 /**
@@ -22,7 +23,7 @@ export class WindowSideBarAnnotationsPanel extends Component {
   */
   render() {
     const {
-      annotationCount, classes, canvasIds, t, windowId, id,
+      annotationCount, canvasIds, t, windowId, id,
     } = this.props;
     return (
       <CompanionWindow
@@ -31,12 +32,11 @@ export class WindowSideBarAnnotationsPanel extends Component {
         windowId={windowId}
         id={id}
         ref={this.containerRef}
-        otherRef={this.containerRef}
         titleControls={<AnnotationSettings windowId={windowId} />}
       >
-        <div className={classes.section}>
+        <CompanionWindowSection>
           <Typography component="p" variant="subtitle2">{t('showingNumAnnotations', { count: annotationCount, number: annotationCount })}</Typography>
-        </div>
+        </CompanionWindowSection>
 
         {canvasIds.map((canvasId, index) => (
           <CanvasAnnotations
@@ -56,7 +56,6 @@ export class WindowSideBarAnnotationsPanel extends Component {
 WindowSideBarAnnotationsPanel.propTypes = {
   annotationCount: PropTypes.number.isRequired,
   canvasIds: PropTypes.arrayOf(PropTypes.string),
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   id: PropTypes.string.isRequired,
   t: PropTypes.func,
   windowId: PropTypes.string.isRequired,
diff --git a/src/components/WindowSideBarButtons.js b/src/components/WindowSideBarButtons.js
index e4b0a2f6cc7cd6d1128d40e392b7d3d59a49eff9..bba780d3d62a01e7da62d25287bc628aa4b7af42 100644
--- a/src/components/WindowSideBarButtons.js
+++ b/src/components/WindowSideBarButtons.js
@@ -1,21 +1,62 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Badge from '@material-ui/core/Badge';
-import Tabs from '@material-ui/core/Tabs';
-import Tab from '@material-ui/core/Tab';
-import Tooltip from '@material-ui/core/Tooltip';
-import InfoIcon from '@material-ui/icons/InfoSharp';
-import AnnotationIcon from '@material-ui/icons/CommentSharp';
-import AttributionIcon from '@material-ui/icons/CopyrightSharp';
-import LayersIcon from '@material-ui/icons/LayersSharp';
-import SearchIcon from '@material-ui/icons/SearchSharp';
+import { styled } from '@mui/material/styles';
+import Badge from '@mui/material/Badge';
+import Tabs from '@mui/material/Tabs';
+import Tab from '@mui/material/Tab';
+import Tooltip from '@mui/material/Tooltip';
+import InfoIcon from '@mui/icons-material/InfoSharp';
+import AnnotationIcon from '@mui/icons-material/CommentSharp';
+import AttributionIcon from '@mui/icons-material/CopyrightSharp';
+import LayersIcon from '@mui/icons-material/LayersSharp';
+import SearchIcon from '@mui/icons-material/SearchSharp';
 import CanvasIndexIcon from './icons/CanvasIndexIcon';
 
+const Root = styled(Tabs, { name: 'WindowSideBarButtons', slot: 'root' })({
+  '& .MuiTabs-flexContainer': {
+    flexDirection: 'column',
+  },
+  '&.MuiTabs-indicator': {
+    display: 'none',
+  },
+});
+
+const StyledTabButton = styled(Tab, { name: 'WindowSideBarButtons', slot: 'button' })(({ theme }) => ({
+  '&.Mui-selected': {
+    borderRight: '2px solid',
+    borderRightColor: theme.palette.primary.main,
+  },
+  '&.MuiTab-root': {
+    '&:active': {
+      backgroundColor: theme.palette.action.active,
+    },
+    '&:focus': {
+      '@media (hover: none)': {
+        backgroundColor: 'transparent',
+      },
+      backgroundColor: theme.palette.action.hover,
+      textDecoration: 'none',
+      // Reset on touch devices, it doesn't add specificity
+    },
+    '&:hover': {
+      '@media (hover: none)': {
+        backgroundColor: 'transparent',
+      },
+      backgroundColor: theme.palette.action.hover,
+      textDecoration: 'none',
+      // Reset on touch devices, it doesn't add specificity
+    },
+    borderRight: '2px solid transparent',
+    minWidth: 'auto',
+  },
+  fill: 'currentcolor',
+}));
+
 /** */
 function TabButton({ t, value, ...tabProps }) {
   return (
     <Tooltip title={t('openCompanionWindow', { context: value })}>
-      <Tab
+      <StyledTabButton
         {...tabProps}
         value={value}
         aria-label={
@@ -59,7 +100,6 @@ export class WindowSideBarButtons extends Component {
    */
   render() {
     const {
-      classes,
       hasAnnotations,
       hasAnyAnnotations,
       hasAnyLayers,
@@ -73,8 +113,7 @@ export class WindowSideBarButtons extends Component {
     } = this.props;
 
     return (
-      <Tabs
-        classes={{ flexContainer: classes.tabsFlexContainer, indicator: classes.tabsIndicator }}
+      <Root
         value={sideBarPanel === 'closed' ? false : sideBarPanel}
         onChange={this.handleChange}
         variant="fullWidth"
@@ -88,7 +127,6 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="info"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(<InfoIcon />)}
           />
@@ -97,7 +135,6 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="attribution"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(<AttributionIcon />)}
           />
@@ -106,7 +143,6 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="canvas"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(<CanvasIndexIcon />)}
           />
@@ -115,10 +151,9 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="annotations"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(
-              <Badge overlap="rectangular" classes={{ badge: classes.badge }} invisible={!hasAnnotations} variant="dot">
+              <Badge overlap="rectangular" color="notification" invisible={!hasAnnotations} variant="dot">
                 <AnnotationIcon />
               </Badge>
             )}
@@ -128,10 +163,9 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="search"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(
-              <Badge overlap="rectangular" classes={{ badge: classes.badge }} invisible={!hasSearchResults} variant="dot">
+              <Badge overlap="rectangular" color="notification" invisible={!hasSearchResults} variant="dot">
                 <SearchIcon />
               </Badge>
             )}
@@ -141,10 +175,9 @@ export class WindowSideBarButtons extends Component {
           <TabButton
             value="layers"
             onKeyUp={this.handleKeyUp}
-            classes={{ root: classes.tab, selected: classes.tabSelected }}
             t={t}
             icon={(
-              <Badge overlap="rectangular" classes={{ badge: classes.badge }} invisible={!hasCurrentLayers} variant="dot">
+              <Badge overlap="rectangular" color="notification" invisible={!hasCurrentLayers} variant="dot">
                 <LayersIcon />
               </Badge>
             )}
@@ -154,21 +187,19 @@ export class WindowSideBarButtons extends Component {
           && PluginComponents.map(PluginComponent => (
             <TabButton
               onKeyUp={this.handleKeyUp}
-              classes={{ root: classes.tab, selected: classes.tabSelected }}
               t={t}
               key={PluginComponent.value}
               value={PluginComponent.value}
               icon={<PluginComponent />}
             />
           ))}
-      </Tabs>
+      </Root>
     );
   }
 }
 
 WindowSideBarButtons.propTypes = {
   addCompanionWindow: PropTypes.func.isRequired,
-  classes: PropTypes.objectOf(PropTypes.string),
   hasAnnotations: PropTypes.bool,
   hasAnyAnnotations: PropTypes.bool,
   hasAnyLayers: PropTypes.bool,
@@ -182,7 +213,6 @@ WindowSideBarButtons.propTypes = {
 };
 
 WindowSideBarButtons.defaultProps = {
-  classes: {},
   hasAnnotations: false,
   hasAnyAnnotations: false,
   hasAnyLayers: false,
diff --git a/src/components/WindowSideBarCanvasPanel.js b/src/components/WindowSideBarCanvasPanel.js
index d785b5abe23635693be6a1d8616fd8eaaf9d3779..768f8c65d5a4e5d062479d8ca9c57719e13bc5b2 100644
--- a/src/components/WindowSideBarCanvasPanel.js
+++ b/src/components/WindowSideBarCanvasPanel.js
@@ -1,21 +1,26 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
-import Tabs from '@material-ui/core/Tabs';
-import Tab from '@material-ui/core/Tab';
-import Tooltip from '@material-ui/core/Tooltip';
-import Button from '@material-ui/core/Button';
-import ItemListIcon from '@material-ui/icons/ReorderSharp';
-import TocIcon from '@material-ui/icons/SortSharp';
-import ThumbnailListIcon from '@material-ui/icons/ViewListSharp';
-import Typography from '@material-ui/core/Typography';
-import ArrowForwardIcon from '@material-ui/icons/ArrowForwardSharp';
-import FormControl from '@material-ui/core/FormControl';
-import Select from '@material-ui/core/Select';
-import MenuItem from '@material-ui/core/MenuItem';
+import { styled } from '@mui/material/styles';
+import Tabs from '@mui/material/Tabs';
+import Tab from '@mui/material/Tab';
+import Tooltip from '@mui/material/Tooltip';
+import Button from '@mui/material/Button';
+import ItemListIcon from '@mui/icons-material/ReorderSharp';
+import TocIcon from '@mui/icons-material/SortSharp';
+import ThumbnailListIcon from '@mui/icons-material/ViewListSharp';
+import Typography from '@mui/material/Typography';
+import ArrowForwardIcon from '@mui/icons-material/ArrowForwardSharp';
+import FormControl from '@mui/material/FormControl';
+import Select from '@mui/material/Select';
+import MenuItem from '@mui/material/MenuItem';
 import CompanionWindow from '../containers/CompanionWindow';
 import SidebarIndexList from '../containers/SidebarIndexList';
 import SidebarIndexTableOfContents from '../containers/SidebarIndexTableOfContents';
 
+const StyledBreak = styled('div')(() => ({
+  flexBasis: '100%',
+  height: 0,
+}));
 /**
  * a panel showing the canvases for a given manifest
  */
@@ -57,7 +62,6 @@ export class WindowSideBarCanvasPanel extends Component {
    */
   render() {
     const {
-      classes,
       collection,
       id,
       showMultipart,
@@ -95,7 +99,6 @@ export class WindowSideBarCanvasPanel extends Component {
         id={id}
         windowId={windowId}
         ref={this.containerRef}
-        otherRef={this.containerRef}
         titleControls={(
           <>
             {
@@ -107,14 +110,19 @@ export class WindowSideBarCanvasPanel extends Component {
                         horizontal: 'left',
                         vertical: 'bottom',
                       },
-                      getContentAnchorEl: null,
                     }}
                     displayEmpty
                     value={sequenceId}
                     onChange={this.handleSequenceChange}
                     name="sequenceId"
-                    classes={{ select: classes.select }}
-                    className={classes.selectEmpty}
+                    sx={{
+                      '&.MuiSelect-select': {
+                        '&:focus': {
+                          backgroundColor: 'background.paper',
+                        },
+                      },
+                      backgroundColor: 'background.paper',
+                    }}
                     data-testid="sequence-select"
                   >
                     { sequences.map((s, i) => <MenuItem value={s.id} key={s.id}><Typography variant="body2">{ WindowSideBarCanvasPanel.getUseableLabel(s, i) }</Typography></MenuItem>) }
@@ -122,7 +130,7 @@ export class WindowSideBarCanvasPanel extends Component {
                 </FormControl>
               )
             }
-            <div className={classes.break} />
+            <StyledBreak />
             <Tabs
               value={variant}
               onChange={this.handleVariantChange}
@@ -131,10 +139,10 @@ export class WindowSideBarCanvasPanel extends Component {
               textColor="primary"
             >
               {showToc && (
-                <Tooltip title={t('tableOfContentsList')} value="tableOfContents"><Tab className={classes.variantTab} value="tableOfContents" aria-label={t('tableOfContentsList')} aria-controls={`tab-panel-${id}`} icon={<TocIcon style={{ transform: 'scale(-1, 1)' }} />} /></Tooltip>
+                <Tooltip title={t('tableOfContentsList')} value="tableOfContents"><Tab sx={{ minWidth: 'auto' }} value="tableOfContents" aria-label={t('tableOfContentsList')} aria-controls={`tab-panel-${id}`} icon={<TocIcon style={{ transform: 'scale(-1, 1)' }} />} /></Tooltip>
               )}
-              <Tooltip title={t('itemList')} value="item"><Tab className={classes.variantTab} value="item" aria-label={t('itemList')} aria-controls={`tab-panel-${id}`} icon={<ItemListIcon />} /></Tooltip>
-              <Tooltip title={t('thumbnailList')} value="thumbnail"><Tab className={classes.variantTab} value="thumbnail" aria-label={t('thumbnailList')} aria-controls={`tab-panel-${id}`} icon={<ThumbnailListIcon />} /></Tooltip>
+              <Tooltip title={t('itemList')} value="item"><Tab sx={{ minWidth: 'auto' }} value="item" aria-label={t('itemList')} aria-controls={`tab-panel-${id}`} icon={<ItemListIcon />} /></Tooltip>
+              <Tooltip title={t('thumbnailList')} value="thumbnail"><Tab sx={{ minWidth: 'auto' }} value="thumbnail" aria-label={t('thumbnailList')} aria-controls={`tab-panel-${id}`} icon={<ThumbnailListIcon />} /></Tooltip>
             </Tabs>
           </>
         )}
@@ -146,7 +154,7 @@ export class WindowSideBarCanvasPanel extends Component {
               onClick={showMultipart}
               endIcon={<ArrowForwardIcon />}
             >
-              <Typography className={classes.collectionNavigationButton}>
+              <Typography sx={{ textTransform: 'none' }}>
                 {WindowSideBarCanvasPanel.getUseableLabel(collection)}
               </Typography>
             </Button>
@@ -159,7 +167,6 @@ export class WindowSideBarCanvasPanel extends Component {
 }
 
 WindowSideBarCanvasPanel.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   collection: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   id: PropTypes.string.isRequired,
   sequenceId: PropTypes.string,
diff --git a/src/components/WindowSideBarCollectionPanel.js b/src/components/WindowSideBarCollectionPanel.js
index a6964ca77705542d3092fabc33881113f91abad8..f55a2dbdef728fbc29d65c9c04cb965644182325 100644
--- a/src/components/WindowSideBarCollectionPanel.js
+++ b/src/components/WindowSideBarCollectionPanel.js
@@ -1,14 +1,14 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import List from '@material-ui/core/List';
-import ListItem from '@material-ui/core/ListItem';
-import ListItemIcon from '@material-ui/core/ListItemIcon';
-import ListItemText from '@material-ui/core/ListItemText';
-import MenuList from '@material-ui/core/MenuList';
-import MenuItem from '@material-ui/core/MenuItem';
-import Typography from '@material-ui/core/Typography';
-import Skeleton from '@material-ui/lab/Skeleton';
-import ArrowUpwardIcon from '@material-ui/icons/ArrowUpwardSharp';
+import List from '@mui/material/List';
+import ListItem from '@mui/material/ListItem';
+import ListItemIcon from '@mui/material/ListItemIcon';
+import ListItemText from '@mui/material/ListItemText';
+import MenuList from '@mui/material/MenuList';
+import MenuItem from '@mui/material/MenuItem';
+import Typography from '@mui/material/Typography';
+import Skeleton from '@mui/material/Skeleton';
+import ArrowUpwardIcon from '@mui/icons-material/ArrowUpwardSharp';
 import CompanionWindow from '../containers/CompanionWindow';
 import IIIFThumbnail from '../containers/IIIFThumbnail';
 
@@ -20,7 +20,12 @@ function Item({
     <MenuItem
       alignItems="flex-start"
       button
+      divider
       component="li"
+      variant="multiline"
+      sx={{
+        paddingRight: 1,
+      }}
       {...otherProps}
     >
       { variant === 'thumbnail' && (
@@ -74,7 +79,6 @@ export class WindowSideBarCollectionPanel extends Component {
   render() {
     const {
       canvasNavigation,
-      classes,
       collectionPath,
       collection,
       id,
@@ -114,7 +118,7 @@ export class WindowSideBarCollectionPanel extends Component {
             )}
             <Typography variant="h6">
               { collection && WindowSideBarCollectionPanel.getUseableLabel(collection)}
-              { isFetching && <Skeleton className={classes.placeholder} variant="text" />}
+              { isFetching && <Skeleton variant="text" />}
             </Typography>
           </>
         )}
@@ -123,9 +127,9 @@ export class WindowSideBarCollectionPanel extends Component {
           { isFetching && (
             <MenuItem>
               <ListItemText>
-                <Skeleton className={classes.placeholder} variant="text" />
-                <Skeleton className={classes.placeholder} variant="text" />
-                <Skeleton className={classes.placeholder} variant="text" />
+                <Skeleton variant="text" />
+                <Skeleton variant="text" />
+                <Skeleton variant="text" />
               </ListItemText>
             </MenuItem>
           )}
@@ -144,7 +148,6 @@ export class WindowSideBarCollectionPanel extends Component {
                   canvasNavigation={canvasNavigation}
                   manifest={manifest}
                   variant={variant}
-                  className={classes.menuItem}
                   selected={manifestId === manifest.id}
                 />
               );
@@ -167,7 +170,6 @@ export class WindowSideBarCollectionPanel extends Component {
                   canvasNavigation={canvasNavigation}
                   manifest={manifest}
                   variant={variant}
-                  className={classes.menuItem}
                   selected={manifestId === manifest.id}
                 />
               );
@@ -184,7 +186,6 @@ WindowSideBarCollectionPanel.propTypes = {
     height: PropTypes.number,
     width: PropTypes.number,
   }).isRequired,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   collection: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   collectionPath: PropTypes.arrayOf(PropTypes.string),
   id: PropTypes.string.isRequired,
diff --git a/src/components/WindowSideBarInfoPanel.js b/src/components/WindowSideBarInfoPanel.js
index e2931f70e34426beac5cb635019dff157a79f99e..3261679503f289fa29caab0b4fb986e8bb963069 100644
--- a/src/components/WindowSideBarInfoPanel.js
+++ b/src/components/WindowSideBarInfoPanel.js
@@ -1,6 +1,7 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
 import CompanionWindow from '../containers/CompanionWindow';
+import { CompanionWindowSection } from './CompanionWindowSection';
 import CanvasInfo from '../containers/CanvasInfo';
 import LocalePicker from '../containers/LocalePicker';
 import ManifestInfo from '../containers/ManifestInfo';
@@ -21,7 +22,6 @@ export class WindowSideBarInfoPanel extends Component {
       windowId,
       id,
       canvasIds,
-      classes,
       collectionPath,
       t,
       locale,
@@ -49,7 +49,9 @@ export class WindowSideBarInfoPanel extends Component {
       >
         {
           canvasIds.map((canvasId, index) => (
-            <div key={canvasId} className={classes.section}>
+            <CompanionWindowSection
+              key={canvasId}
+            >
               <CanvasInfo
                 id={id}
                 canvasId={canvasId}
@@ -57,22 +59,22 @@ export class WindowSideBarInfoPanel extends Component {
                 totalSize={canvasIds.length}
                 windowId={windowId}
               />
-            </div>
+            </CompanionWindowSection>
           ))
         }
         { collectionPath.length > 0 && (
-          <div className={classes.section}>
+          <CompanionWindowSection>
             <CollectionInfo id={id} windowId={windowId} />
-          </div>
+          </CompanionWindowSection>
         )}
 
-        <div className={classes.section}>
+        <CompanionWindowSection>
           <ManifestInfo id={id} windowId={windowId} />
-        </div>
+        </CompanionWindowSection>
 
-        <div className={classes.section}>
+        <CompanionWindowSection>
           <ManifestRelatedLinks id={id} windowId={windowId} />
-        </div>
+        </CompanionWindowSection>
       </CompanionWindow>
     );
   }
@@ -81,7 +83,6 @@ export class WindowSideBarInfoPanel extends Component {
 WindowSideBarInfoPanel.propTypes = {
   availableLocales: PropTypes.arrayOf(PropTypes.string),
   canvasIds: PropTypes.arrayOf(PropTypes.string),
-  classes: PropTypes.objectOf(PropTypes.string),
   collectionPath: PropTypes.arrayOf(PropTypes.string),
   id: PropTypes.string.isRequired,
   locale: PropTypes.string,
@@ -94,7 +95,6 @@ WindowSideBarInfoPanel.propTypes = {
 WindowSideBarInfoPanel.defaultProps = {
   availableLocales: [],
   canvasIds: [],
-  classes: {},
   collectionPath: [],
   locale: '',
   setLocale: undefined,
diff --git a/src/components/WindowThumbnailSettings.js b/src/components/WindowThumbnailSettings.js
index 22b7eb57db316222e7a93a9e18b8301d138e7fb3..042355e4d01054086ec6cd21195670e166a06ffd 100644
--- a/src/components/WindowThumbnailSettings.js
+++ b/src/components/WindowThumbnailSettings.js
@@ -1,11 +1,25 @@
 import { Component } from 'react';
-import FormControlLabel from '@material-ui/core/FormControlLabel';
-import ListSubheader from '@material-ui/core/ListSubheader';
-import MenuItem from '@material-ui/core/MenuItem';
-import ThumbnailsOffIcon from '@material-ui/icons/CropDinSharp';
+import { styled } from '@mui/material/styles';
+import FormControlLabel from '@mui/material/FormControlLabel';
+import ListSubheader from '@mui/material/ListSubheader';
+import MenuItem from '@mui/material/MenuItem';
+import ThumbnailsOffIcon from '@mui/icons-material/CropDinSharp';
 import PropTypes from 'prop-types';
 import ThumbnailNavigationBottomIcon from './icons/ThumbnailNavigationBottomIcon';
 import ThumbnailNavigationRightIcon from './icons/ThumbnailNavigationRightIcon';
+
+const ThumbnailOption = styled(MenuItem, { name: 'WindowThumbnailSettings', slot: 'option' })(({ selected, theme }) => ({
+  '& .MuiFormControlLabel-label': {
+    borderBottom: '2px solid transparent',
+    ...(selected && {
+      borderBottomColor: theme.palette.secondary.main,
+    }),
+  },
+  backgroundColor: 'transparent !important',
+  color: selected ? theme.palette.secondary.main : undefined,
+  display: 'inline-block',
+}));
+
 /**
  *
  */
@@ -34,56 +48,53 @@ export class WindowThumbnailSettings extends Component {
    */
   render() {
     const {
-      classes, handleClose, t, thumbnailNavigationPosition, direction,
+      handleClose, t, thumbnailNavigationPosition, direction,
     } = this.props;
 
     return (
       <>
-        <ListSubheader role="presentation" disableSticky tabIndex="-1">{t('thumbnails')}</ListSubheader>
+        <ListSubheader role="presentation" disableSticky tabIndex={-1}>{t('thumbnails')}</ListSubheader>
 
-        <MenuItem className={classes.MenuItem} onClick={() => { this.handleChange('off'); handleClose(); }}>
+        <ThumbnailOption selected={thumbnailNavigationPosition === 'off'} onClick={() => { this.handleChange('off'); handleClose(); }}>
           <FormControlLabel
             value="off"
-            classes={{ label: thumbnailNavigationPosition === 'off' ? classes.selectedLabel : classes.label }}
             control={
-              <ThumbnailsOffIcon color={thumbnailNavigationPosition === 'off' ? 'secondary' : undefined} />
+              <ThumbnailsOffIcon color={thumbnailNavigationPosition === 'off' ? 'secondary' : undefined} fill="currentcolor" />
             }
             label={t('off')}
             labelPlacement="bottom"
           />
-        </MenuItem>
-        <MenuItem className={classes.MenuItem} onClick={() => { this.handleChange('far-bottom'); handleClose(); }}>
+        </ThumbnailOption>
+        <ThumbnailOption selected={thumbnailNavigationPosition === 'far-bottom'} onClick={() => { this.handleChange('far-bottom'); handleClose(); }}>
           <FormControlLabel
             value="far-bottom"
-            classes={{ label: thumbnailNavigationPosition === 'far-bottom' ? classes.selectedLabel : classes.label }}
             control={
-              <ThumbnailNavigationBottomIcon color={thumbnailNavigationPosition === 'far-bottom' ? 'secondary' : undefined} />
+              <ThumbnailNavigationBottomIcon color={thumbnailNavigationPosition === 'far-bottom' ? 'secondary' : undefined} fill="currentcolor" />
             }
             label={t('bottom')}
             labelPlacement="bottom"
           />
-        </MenuItem>
-        <MenuItem className={classes.MenuItem} onClick={() => { this.handleChange('far-right'); handleClose(); }}>
+        </ThumbnailOption>
+        <ThumbnailOption selected={thumbnailNavigationPosition === 'far-right'} onClick={() => { this.handleChange('far-right'); handleClose(); }}>
           <FormControlLabel
             value="far-right"
-            classes={{ label: thumbnailNavigationPosition === 'far-right' ? classes.selectedLabel : classes.label }}
             control={(
               <ThumbnailNavigationRightIcon
                 color={thumbnailNavigationPosition === 'far-right' ? 'secondary' : undefined}
+                fill="currentcolor"
                 style={direction === 'rtl' ? { transform: 'rotate(180deg)' } : {}}
               />
             )}
             label={t('right')}
             labelPlacement="bottom"
           />
-        </MenuItem>
+        </ThumbnailOption>
       </>
     );
   }
 }
 
 WindowThumbnailSettings.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   direction: PropTypes.string.isRequired,
   handleClose: PropTypes.func,
   setWindowThumbnailPosition: PropTypes.func.isRequired,
diff --git a/src/components/WindowTopBar.js b/src/components/WindowTopBar.js
index cf204b2f88c260c8f4a40c08ca594a209deb1de2..5e32f8cd0787303091ddba23f3fb1843cf6d40d6 100644
--- a/src/components/WindowTopBar.js
+++ b/src/components/WindowTopBar.js
@@ -1,9 +1,10 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import MenuIcon from '@material-ui/icons/MenuSharp';
-import CloseIcon from '@material-ui/icons/CloseSharp';
-import Toolbar from '@material-ui/core/Toolbar';
-import AppBar from '@material-ui/core/AppBar';
+import { styled } from '@mui/material/styles';
+import MenuIcon from '@mui/icons-material/MenuSharp';
+import CloseIcon from '@mui/icons-material/CloseSharp';
+import Toolbar from '@mui/material/Toolbar';
+import AppBar from '@mui/material/AppBar';
 import classNames from 'classnames';
 import WindowTopMenuButton from '../containers/WindowTopMenuButton';
 import WindowTopBarPluginArea from '../containers/WindowTopBarPluginArea';
@@ -15,6 +16,22 @@ import WindowMaxIcon from './icons/WindowMaxIcon';
 import WindowMinIcon from './icons/WindowMinIcon';
 import ns from '../config/css-ns';
 
+const Root = styled(AppBar, { name: 'WindowTopBar', slot: 'root' })(() => ({
+  zIndex: 1100,
+}));
+
+const StyledToolbar = styled(Toolbar, { name: 'WindowTopBar', slot: 'toolbar' })(({ ownerState, theme }) => ({
+  backgroundColor: theme.palette.shades?.main,
+  borderTop: '2px solid',
+  borderTopColor: ownerState?.focused ? theme.palette.primary.main : 'transparent',
+  minHeight: 32,
+  paddingLeft: theme.spacing(0.5),
+  paddingRight: theme.spacing(0.5),
+  ...(ownerState?.windowDraggable && {
+    cursor: 'move',
+  }),
+}));
+
 /**
  * WindowTopBar
  */
@@ -25,66 +42,61 @@ export class WindowTopBar extends Component {
    */
   render() {
     const {
-      removeWindow, windowId, classes, toggleWindowSideBar, t, windowDraggable,
-      maximizeWindow, maximized, minimizeWindow, focused, allowClose, allowMaximize,
+      removeWindow, windowId, toggleWindowSideBar, t,
+      maximizeWindow, maximized, minimizeWindow, allowClose, allowMaximize,
       focusWindow, allowFullscreen, allowTopMenuButton, allowWindowSideBar,
+      component,
     } = this.props;
 
     return (
-      <AppBar position="relative" color="default">
-        <nav aria-label={t('windowNavigation')}>
-          <Toolbar
-            disableGutters
-            onMouseDown={focusWindow}
-            className={classNames(
-              classes.windowTopBarStyle,
-              windowDraggable ? classes.windowTopBarStyleDraggable : null,
-              focused ? classes.focused : null,
-              ns('window-top-bar'),
-            )}
-            variant="dense"
-          >
-            {allowWindowSideBar && (
-              <MiradorMenuButton
-                aria-label={t('toggleWindowSideBar')}
-                onClick={toggleWindowSideBar}
-                className={ns('window-menu-btn')}
-              >
-                <MenuIcon />
-              </MiradorMenuButton>
-            )}
-            <WindowTopBarTitle
-              windowId={windowId}
-            />
-            {allowTopMenuButton && (
-              <WindowTopMenuButton windowId={windowId} className={ns('window-menu-btn')} />
-            )}
-            <WindowTopBarPluginArea windowId={windowId} />
-            <WindowTopBarPluginMenu windowId={windowId} />
-            {allowMaximize && (
-              <MiradorMenuButton
-                aria-label={(maximized ? t('minimizeWindow') : t('maximizeWindow'))}
-                className={classNames(ns('window-maximize'), ns('window-menu-btn'))}
-                onClick={(maximized ? minimizeWindow : maximizeWindow)}
-              >
-                {(maximized ? <WindowMinIcon /> : <WindowMaxIcon />)}
-              </MiradorMenuButton>
-            )}
-            {allowFullscreen && (
-              <FullScreenButton className={ns('window-menu-btn')} />
-            )}
-            {allowClose && (
-              <MiradorMenuButton
-                aria-label={t('closeWindow')}
-                className={classNames(ns('window-close'), ns('window-menu-btn'))}
-                onClick={removeWindow}
-              >
-                <CloseIcon />
-              </MiradorMenuButton>
-            )}
-          </Toolbar>
-        </nav>
-      </AppBar>
+      <Root component={component} aria-label={t('windowNavigation')} position="relative" color="default" enableColorOnDark>
+        <StyledToolbar
+          disableGutters
+          onMouseDown={focusWindow}
+          ownerState={this.props}
+          className={classNames(ns('window-top-bar'))}
+          variant="dense"
+        >
+          {allowWindowSideBar && (
+            <MiradorMenuButton
+              aria-label={t('toggleWindowSideBar')}
+              onClick={toggleWindowSideBar}
+              className={ns('window-menu-btn')}
+            >
+              <MenuIcon />
+            </MiradorMenuButton>
+          )}
+          <WindowTopBarTitle
+            windowId={windowId}
+          />
+          {allowTopMenuButton && (
+            <WindowTopMenuButton windowId={windowId} className={ns('window-menu-btn')} />
+          )}
+          <WindowTopBarPluginArea windowId={windowId} />
+          <WindowTopBarPluginMenu windowId={windowId} />
+          {allowMaximize && (
+            <MiradorMenuButton
+              aria-label={(maximized ? t('minimizeWindow') : t('maximizeWindow'))}
+              className={classNames(ns('window-maximize'), ns('window-menu-btn'))}
+              onClick={(maximized ? minimizeWindow : maximizeWindow)}
+            >
+              {(maximized ? <WindowMinIcon /> : <WindowMaxIcon />)}
+            </MiradorMenuButton>
+          )}
+          {allowFullscreen && (
+            <FullScreenButton className={ns('window-menu-btn')} />
+          )}
+          {allowClose && (
+            <MiradorMenuButton
+              aria-label={t('closeWindow')}
+              className={classNames(ns('window-close'), ns('window-menu-btn'))}
+              onClick={removeWindow}
+            >
+              <CloseIcon />
+            </MiradorMenuButton>
+          )}
+        </StyledToolbar>
+      </Root>
     );
   }
 }
@@ -95,8 +107,8 @@ WindowTopBar.propTypes = {
   allowMaximize: PropTypes.bool,
   allowTopMenuButton: PropTypes.bool,
   allowWindowSideBar: PropTypes.bool,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
-  focused: PropTypes.bool,
+  component: PropTypes.elementType,
+  focused: PropTypes.bool, // eslint-disable-line react/no-unused-prop-types
   focusWindow: PropTypes.func,
   maximized: PropTypes.bool,
   maximizeWindow: PropTypes.func,
@@ -104,7 +116,7 @@ WindowTopBar.propTypes = {
   removeWindow: PropTypes.func.isRequired,
   t: PropTypes.func,
   toggleWindowSideBar: PropTypes.func.isRequired,
-  windowDraggable: PropTypes.bool,
+  windowDraggable: PropTypes.bool, // eslint-disable-line react/no-unused-prop-types
   windowId: PropTypes.string.isRequired,
 };
 
@@ -114,6 +126,7 @@ WindowTopBar.defaultProps = {
   allowMaximize: true,
   allowTopMenuButton: true,
   allowWindowSideBar: true,
+  component: 'nav',
   focused: false,
   focusWindow: () => {},
   maximized: false,
diff --git a/src/components/WindowTopBarPluginMenu.js b/src/components/WindowTopBarPluginMenu.js
index 78f95a73ec5566fab2d4565c34d86b3ce2a0d732..972cbf544b6c32b1aa87f2d8fcfff1e770ef0b54 100644
--- a/src/components/WindowTopBarPluginMenu.js
+++ b/src/components/WindowTopBarPluginMenu.js
@@ -1,7 +1,7 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import MoreVertIcon from '@material-ui/icons/MoreVertSharp';
-import Menu from '@material-ui/core/Menu';
+import MoreVertIcon from '@mui/icons-material/MoreVertSharp';
+import Menu from '@mui/material/Menu';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import { PluginHook } from './PluginHook';
 
@@ -47,7 +47,7 @@ export class WindowTopBarPluginMenu extends Component {
    */
   render() {
     const {
-      classes, container, PluginComponents, t, windowId, menuIcon,
+      container, PluginComponents, t, windowId, menuIcon,
     } = this.props;
     const { anchorEl, open } = this.state;
     const windowPluginMenuId = `window-plugin-menu_${windowId}`;
@@ -59,7 +59,7 @@ export class WindowTopBarPluginMenu extends Component {
           aria-haspopup="true"
           aria-label={t('windowPluginMenu')}
           aria-owns={open ? windowPluginMenuId : undefined}
-          className={open ? classes.ctrlBtnSelected : null}
+          selected={open}
           onClick={this.handleMenuClick}
         >
           {menuIcon}
@@ -77,7 +77,6 @@ export class WindowTopBarPluginMenu extends Component {
             horizontal: 'right',
             vertical: 'top',
           }}
-          getContentAnchorEl={null}
           open={open}
           onClose={() => this.handleMenuClose()}
         >
@@ -90,9 +89,6 @@ export class WindowTopBarPluginMenu extends Component {
 
 WindowTopBarPluginMenu.propTypes = {
   anchorEl: PropTypes.object, // eslint-disable-line react/forbid-prop-types
-  classes: PropTypes.shape({
-    ctrlBtnSelected: PropTypes.string,
-  }),
   container: PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
   menuIcon: PropTypes.element,
   open: PropTypes.bool,
@@ -105,7 +101,6 @@ WindowTopBarPluginMenu.propTypes = {
 
 WindowTopBarPluginMenu.defaultProps = {
   anchorEl: null,
-  classes: {},
   container: null,
   menuIcon: <MoreVertIcon />,
   open: false,
diff --git a/src/components/WindowTopBarTitle.js b/src/components/WindowTopBarTitle.js
index b47f00c2c2e03433cf198724553503e38fecb5ec..5c91250e12861c7e2a4426e214fdc5499e663bba 100644
--- a/src/components/WindowTopBarTitle.js
+++ b/src/components/WindowTopBarTitle.js
@@ -1,9 +1,21 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Typography from '@material-ui/core/Typography';
-import Skeleton from '@material-ui/lab/Skeleton';
-import ErrorIcon from '@material-ui/icons/ErrorOutlineSharp';
+import { styled } from '@mui/material/styles';
+import Typography from '@mui/material/Typography';
+import Skeleton from '@mui/material/Skeleton';
+import ErrorIcon from '@mui/icons-material/ErrorOutlineSharp';
 
+const StyledTitleTypography = styled(TitleTypography)(({ theme }) => ({
+  ...theme.typography.h6,
+  flexGrow: 1,
+  paddingLeft: theme.spacing(0.5),
+}));
+
+const StyledTitle = styled('div')(({ theme }) => ({
+  ...theme.typography.h6,
+  flexGrow: 1,
+  paddingLeft: theme.spacing(0.5),
+}));
 /** */
 function TitleTypography({ children, ...props }) {
   return (
@@ -27,32 +39,32 @@ export class WindowTopBarTitle extends Component {
    */
   render() {
     const {
-      classes, error, hideWindowTitle, isFetching, manifestTitle,
+      error, hideWindowTitle, isFetching, manifestTitle,
     } = this.props;
 
     let title = null;
     if (isFetching) {
       title = (
-        <TitleTypography className={classes.title}>
+        <StyledTitleTypography>
           <Skeleton variant="text" />
-        </TitleTypography>
+        </StyledTitleTypography>
       );
     } else if (error) {
       title = (
         <>
           <ErrorIcon color="error" />
-          <TitleTypography color="textSecondary" className={classes.title}>
+          <StyledTitleTypography color="textSecondary">
             {error}
-          </TitleTypography>
+          </StyledTitleTypography>
         </>
       );
     } else if (hideWindowTitle) {
-      title = (<div className={classes.title} />);
+      title = (<StyledTitle />);
     } else {
       title = (
-        <TitleTypography className={classes.title}>
+        <StyledTitleTypography>
           {manifestTitle}
-        </TitleTypography>
+        </StyledTitleTypography>
       );
     }
     return title;
@@ -60,7 +72,6 @@ export class WindowTopBarTitle extends Component {
 }
 
 WindowTopBarTitle.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   error: PropTypes.string,
   hideWindowTitle: PropTypes.bool,
   isFetching: PropTypes.bool,
diff --git a/src/components/WindowTopMenu.js b/src/components/WindowTopMenu.js
index d978876a33debb3f9ea428b398720c1dc21094f4..feb45377bd37aafa51e484b40216faf69c2d2340 100644
--- a/src/components/WindowTopMenu.js
+++ b/src/components/WindowTopMenu.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
-import Menu from '@material-ui/core//Menu';
-import ListSubheader from '@material-ui/core/ListSubheader';
+import Menu from '@mui/material//Menu';
+import ListSubheader from '@mui/material/ListSubheader';
 import PropTypes from 'prop-types';
 import WindowThumbnailSettings from '../containers/WindowThumbnailSettings';
 import WindowViewSettings from '../containers/WindowViewSettings';
@@ -11,7 +11,7 @@ function PluginHookWithHeader(props) {
   const { PluginComponents, t } = props; // eslint-disable-line react/prop-types
   return PluginComponents ? (
     <>
-      <ListSubheader role="presentation" disableSticky tabIndex="-1">{t('windowPluginButtons')}</ListSubheader>
+      <ListSubheader role="presentation" disableSticky tabIndex={-1}>{t('windowPluginButtons')}</ListSubheader>
       <PluginHook {...props} />
     </>
   ) : null;
@@ -47,7 +47,6 @@ export class WindowTopMenu extends Component {
           onExit: toggleDraggingEnabled,
         }}
         orientation="horizontal"
-        getContentAnchorEl={null}
         anchorEl={anchorEl}
         open={open}
       >
diff --git a/src/components/WindowTopMenuButton.js b/src/components/WindowTopMenuButton.js
index ca9ab8db0109377d74a17f229cbf40a25bd40ffb..e48bd84d2c3c5aa0d6fbbab209a1d1f28be262d8 100644
--- a/src/components/WindowTopMenuButton.js
+++ b/src/components/WindowTopMenuButton.js
@@ -1,6 +1,5 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
 import WindowTopMenu from '../containers/WindowTopMenu';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import WindowOptionsIcon from './icons/WindowOptionsIcon';
@@ -47,7 +46,7 @@ export class WindowTopMenuButton extends Component {
    */
   render() {
     const {
-      classes, className, t, windowId,
+      classes, t, windowId,
     } = this.props;
     const { open, anchorEl } = this.state;
     const menuId = `window-menu_${windowId}`;
@@ -57,7 +56,8 @@ export class WindowTopMenuButton extends Component {
           aria-haspopup="true"
           aria-label={t('windowMenu')}
           aria-owns={open ? menuId : undefined}
-          className={classNames(className, open ? classes.ctrlBtnSelected : null)}
+          className={open ? classes.ctrlBtnSelected : undefined}
+          selected={open}
           onClick={this.handleMenuClick}
         >
           <WindowOptionsIcon />
@@ -75,13 +75,12 @@ export class WindowTopMenuButton extends Component {
 }
 
 WindowTopMenuButton.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
-  className: PropTypes.string,
+  classes: PropTypes.objectOf(PropTypes.string),
   t: PropTypes.func,
   windowId: PropTypes.string.isRequired,
 };
 
 WindowTopMenuButton.defaultProps = {
-  className: '',
+  classes: {},
   t: key => key,
 };
diff --git a/src/components/WindowViewSettings.js b/src/components/WindowViewSettings.js
index 90c56b823269cf733b54c88132b4cb732467883e..e693b2ffcdc62caab53200c3178d6597257169ea 100644
--- a/src/components/WindowViewSettings.js
+++ b/src/components/WindowViewSettings.js
@@ -1,13 +1,26 @@
 import { Component } from 'react';
-import FormControlLabel from '@material-ui/core/FormControlLabel';
-import MenuItem from '@material-ui/core/MenuItem';
-import ListSubheader from '@material-ui/core/ListSubheader';
-import SingleIcon from '@material-ui/icons/CropOriginalSharp';
-import ScrollViewIcon from '@material-ui/icons/ViewColumn';
+import { styled } from '@mui/material/styles';
+import FormControlLabel from '@mui/material/FormControlLabel';
+import MenuItem from '@mui/material/MenuItem';
+import ListSubheader from '@mui/material/ListSubheader';
+import SingleIcon from '@mui/icons-material/CropOriginalSharp';
+import ScrollViewIcon from '@mui/icons-material/ViewColumn';
 import PropTypes from 'prop-types';
 import BookViewIcon from './icons/BookViewIcon';
 import GalleryViewIcon from './icons/GalleryViewIcon';
 
+const ViewOption = styled(MenuItem, { name: 'WindowViewSettings', slot: 'option' })(({ selected, theme }) => ({
+  '& .MuiFormControlLabel-label': {
+    borderBottom: '2px solid transparent',
+    ...(selected && {
+      borderBottomColor: theme.palette.secondary.main,
+    }),
+  },
+  backgroundColor: 'transparent !important',
+  color: selected ? theme.palette.secondary.main : undefined,
+  display: 'inline-block',
+}));
+
 /**
  *
  */
@@ -36,7 +49,7 @@ export class WindowViewSettings extends Component {
    */
   render() {
     const {
-      classes, handleClose, t, windowViewType, viewTypes,
+      handleClose, t, windowViewType, viewTypes,
     } = this.props;
 
     const iconMap = {
@@ -49,26 +62,25 @@ export class WindowViewSettings extends Component {
     /** Suspiciously similar to a component, yet if it is invoked through JSX
         none of the click handlers work? */
     const menuItem = ({ value, Icon }) => (
-      <MenuItem
+      <ViewOption
+        selected={windowViewType === value}
         key={value}
-        className={classes.MenuItem}
         autoFocus={windowViewType === value}
         onClick={() => { this.handleChange(value); handleClose(); }}
       >
         <FormControlLabel
           value={value}
-          classes={{ label: windowViewType === value ? classes.selectedLabel : classes.label }}
-          control={<Icon color={windowViewType === value ? 'secondary' : undefined} />}
+          control={<Icon fill="currentcolor" color={windowViewType === value ? 'secondary' : undefined} />}
           label={t(value)}
           labelPlacement="bottom"
         />
-      </MenuItem>
+      </ViewOption>
     );
 
     if (viewTypes.length === 0) return null;
     return (
       <>
-        <ListSubheader role="presentation" disableSticky tabIndex="-1">{t('view')}</ListSubheader>
+        <ListSubheader role="presentation" disableSticky tabIndex={-1}>{t('view')}</ListSubheader>
         { viewTypes.map(value => menuItem({ Icon: iconMap[value], value })) }
       </>
     );
@@ -76,7 +88,6 @@ export class WindowViewSettings extends Component {
 }
 
 WindowViewSettings.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   handleClose: PropTypes.func,
   setWindowViewType: PropTypes.func.isRequired,
   t: PropTypes.func,
diff --git a/src/components/Workspace.js b/src/components/Workspace.js
index f99f61d4e4351078b811d874dba828aca7d7ad6f..c94f8a30b3fc1eb1550ff0fe3a90211bedc14c5b 100644
--- a/src/components/Workspace.js
+++ b/src/components/Workspace.js
@@ -1,14 +1,22 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
-import Grid from '@material-ui/core/Grid';
-import Typography from '@material-ui/core/Typography';
+import Grid from '@mui/material/Grid';
+import Typography from '@mui/material/Typography';
+import { visuallyHidden } from '@mui/utils';
 import Window from '../containers/Window';
 import WorkspaceMosaic from '../containers/WorkspaceMosaic';
 import WorkspaceElastic from '../containers/WorkspaceElastic';
 import ns from '../config/css-ns';
 import { IIIFDropTarget } from './IIIFDropTarget';
 
+const Root = styled('div', { name: 'Workspace', slot: 'root' })(() => ({
+  height: '100%',
+  position: 'relative',
+  width: '100%',
+}));
+
 /**
  * Represents a work area that contains any number of windows
  * @memberof Workspace
@@ -62,26 +70,28 @@ export class Workspace extends Component {
     const { t } = this.props;
 
     return (
-      <Grid
-        alignItems="center"
-        container
-        style={{
-          height: '100%',
-        }}
-      >
+      <Root>
         <Grid
-          xs={12}
-          item
+          alignItems="center"
+          container
+          style={{
+            height: '100%',
+          }}
         >
-          <Typography
-            variant="h1"
-            component="div"
-            align="center"
+          <Grid
+            xs={12}
+            item
           >
-            {t('welcome')}
-          </Typography>
+            <Typography
+              variant="h1"
+              component="div"
+              align="center"
+            >
+              {t('welcome')}
+            </Typography>
+          </Grid>
         </Grid>
-      </Grid>
+      </Root>
     );
   }
 
@@ -107,23 +117,21 @@ export class Workspace extends Component {
    * render
    */
   render() {
-    const { classes, isWorkspaceControlPanelVisible, t } = this.props;
+    const { t } = this.props;
 
     return (
       <IIIFDropTarget onDrop={this.handleDrop}>
-        <div
+        <Root
+          ownerState={this.props}
           className={
             classNames(
               ns('workspace-viewport'),
-              (isWorkspaceControlPanelVisible && ns('workspace-with-control-panel')),
-              (isWorkspaceControlPanelVisible && classes.workspaceWithControlPanel),
-              classes.workspaceViewport,
             )
           }
         >
-          <Typography variant="srOnly" component="h1">{t('miradorViewer')}</Typography>
+          <Typography style={visuallyHidden} component="h1">{t('miradorViewer')}</Typography>
           {this.workspaceByType()}
-        </div>
+        </Root>
       </IIIFDropTarget>
     );
   }
@@ -132,8 +140,6 @@ export class Workspace extends Component {
 Workspace.propTypes = {
   addWindow: PropTypes.func,
   allowNewWindows: PropTypes.bool,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
-  isWorkspaceControlPanelVisible: PropTypes.bool.isRequired,
   maximizedWindowIds: PropTypes.arrayOf(PropTypes.string),
   t: PropTypes.func.isRequired,
   windowIds: PropTypes.arrayOf(PropTypes.string),
diff --git a/src/components/WorkspaceAdd.js b/src/components/WorkspaceAdd.js
index 172c8d3474970c769e39084dbde0bad3b43c846f..6e744878c40c1503314274002e1fe6cca2c6de97 100644
--- a/src/components/WorkspaceAdd.js
+++ b/src/components/WorkspaceAdd.js
@@ -1,16 +1,18 @@
 import { createRef, Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
-import AddIcon from '@material-ui/icons/AddSharp';
-import ExpandMoreIcon from '@material-ui/icons/ExpandMoreSharp';
-import AppBar from '@material-ui/core/AppBar';
-import Drawer from '@material-ui/core/Drawer';
-import Grid from '@material-ui/core/Grid';
-import Fab from '@material-ui/core/Fab';
-import List from '@material-ui/core/List';
-import Paper from '@material-ui/core/Paper';
-import Toolbar from '@material-ui/core/Toolbar';
-import Typography from '@material-ui/core/Typography';
+import AddIcon from '@mui/icons-material/AddSharp';
+import ExpandMoreIcon from '@mui/icons-material/ExpandMoreSharp';
+import AppBar from '@mui/material/AppBar';
+import Drawer from '@mui/material/Drawer';
+import Grid from '@mui/material/Grid';
+import Fab from '@mui/material/Fab';
+import List from '@mui/material/List';
+import Paper from '@mui/material/Paper';
+import Toolbar from '@mui/material/Toolbar';
+import Typography from '@mui/material/Typography';
+import { visuallyHidden } from '@mui/utils';
 import ns from '../config/css-ns';
 import ManifestForm from '../containers/ManifestForm';
 import ManifestListItem from '../containers/ManifestListItem';
@@ -18,6 +20,17 @@ import MiradorMenuButton from '../containers/MiradorMenuButton';
 import { IIIFDropTarget } from './IIIFDropTarget';
 import { PluginHook } from './PluginHook';
 
+const StyledWorkspaceAdd = styled('div')(() => ({
+  boxSizing: 'border-box',
+  height: '100%',
+  overflowX: 'hidden',
+  overflowY: 'auto',
+}));
+
+const StyledMiradorMenuButton = styled(MiradorMenuButton)(() => ({
+  marginLeft: -12,
+  marginRight: 20,
+}));
 /**
  * An area for managing manifests and adding them to workspace
  * @memberof Workspace
@@ -75,7 +88,7 @@ export class WorkspaceAdd extends Component {
    */
   render() {
     const {
-      catalog, setWorkspaceAddVisibility, t, classes,
+      catalog, setWorkspaceAddVisibility, t,
     } = this.props;
     const { addResourcesOpen } = this.state;
 
@@ -91,7 +104,7 @@ export class WorkspaceAdd extends Component {
 
     return (
       <IIIFDropTarget onDrop={this.handleDrop}>
-        <div ref={this.ref} className={classNames(ns('workspace-add'), classes.workspaceAdd)}>
+        <StyledWorkspaceAdd ref={this.ref} className={classNames(ns('workspace-add'))}>
           {catalog.length < 1 ? (
             <Grid
               alignItems="center"
@@ -114,8 +127,8 @@ export class WorkspaceAdd extends Component {
               </Grid>
             </Grid>
           ) : (
-            <Paper className={classes.list}>
-              <Typography variant="srOnly" component="h1">{t('miradorResources')}</Typography>
+            <Paper sx={{ margin: 2 }}>
+              <Typography style={visuallyHidden} component="h1">{t('miradorResources')}</Typography>
               <PluginHook {...this.props} />
               <List disablePadding>
                 {manifestList}
@@ -125,7 +138,12 @@ export class WorkspaceAdd extends Component {
           <Fab
             variant="extended"
             disabled={addResourcesOpen}
-            className={classNames(classes.fab, ns('add-resource-button'))}
+            sx={(theme) => ({
+              bottom: theme.spacing(2),
+              position: 'absolute',
+              right: theme.spacing(2),
+            })}
+            className={classNames(ns('add-resource-button'))}
             color="primary"
             onClick={() => (this.setAddResourcesVisibility(true))}
           >
@@ -134,10 +152,18 @@ export class WorkspaceAdd extends Component {
           </Fab>
 
           <Drawer
-            className={classNames({
-              [classes.displayNone]: !addResourcesOpen,
+            sx={theme => ({
+              '.MuiDrawer-paper': {
+                borderTop: '0',
+                left: '0',
+                [theme.breakpoints.up('sm')]: {
+                  left: '65px',
+                },
+              },
+              ...(!addResourcesOpen && {
+                display: 'none',
+              }),
             })}
-            classes={{ paper: classes.paper }}
             variant="persistent"
             anchor="bottom"
             open={addResourcesOpen}
@@ -148,18 +174,25 @@ export class WorkspaceAdd extends Component {
             }}
           >
             <Paper
-              className={classes.form}
+              sx={{
+                left: '0',
+                marginTop: 6,
+                paddingBottom: 2,
+                paddingLeft: { sm: 3, xs: 2 },
+                paddingRight: { sm: 3, xs: 2 },
+                paddingTop: 2,
+                right: '0',
+              }}
             >
-              <AppBar position="absolute" color="primary" onClick={() => (this.setAddResourcesVisibility(false))}>
+              <AppBar position="absolute" color="primary" enableColorOnDark onClick={() => (this.setAddResourcesVisibility(false))}>
                 <Toolbar variant="dense">
-                  <MiradorMenuButton
+                  <StyledMiradorMenuButton
                     aria-label={t('closeAddResourceForm')}
-                    className={classes.menuButton}
                     color="inherit"
                   >
                     <ExpandMoreIcon />
-                  </MiradorMenuButton>
-                  <Typography variant="h2" noWrap color="inherit" className={classes.typographyBody}>
+                  </StyledMiradorMenuButton>
+                  <Typography variant="h2" noWrap color="inherit" sx={{ flexGrow: 1 }}>
                     {t('addResource')}
                   </Typography>
                 </Toolbar>
@@ -171,7 +204,7 @@ export class WorkspaceAdd extends Component {
               />
             </Paper>
           </Drawer>
-        </div>
+        </StyledWorkspaceAdd>
       </IIIFDropTarget>
     );
   }
@@ -183,7 +216,6 @@ WorkspaceAdd.propTypes = {
     manifestId: PropTypes.string.isRequired,
     provider: PropTypes.string,
   })),
-  classes: PropTypes.objectOf(PropTypes.string),
   setWorkspaceAddVisibility: PropTypes.func.isRequired,
   t: PropTypes.func,
 };
@@ -191,6 +223,5 @@ WorkspaceAdd.propTypes = {
 WorkspaceAdd.defaultProps = {
   addResource: () => {},
   catalog: [],
-  classes: {},
   t: key => key,
 };
diff --git a/src/components/WorkspaceAddButton.js b/src/components/WorkspaceAddButton.js
index b80ec8136911b0e8a34a16b74ab6c748de3ce6f2..b50679226e4082364041ce4eb77c73b56d1579ca 100644
--- a/src/components/WorkspaceAddButton.js
+++ b/src/components/WorkspaceAddButton.js
@@ -1,9 +1,14 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import Fab from '@material-ui/core/Fab';
-import Tooltip from '@material-ui/core/Tooltip';
-import AddIcon from '@material-ui/icons/AddSharp';
-import CloseIcon from '@material-ui/icons/CloseSharp';
+import Fab from '@mui/material/Fab';
+import Tooltip from '@mui/material/Tooltip';
+import AddIcon from '@mui/icons-material/AddSharp';
+import CloseIcon from '@mui/icons-material/CloseSharp';
+import { styled } from '@mui/material/styles';
+
+const Root = styled(Fab, { name: 'WorkspaceAddButton', slot: 'root' })(({ theme }) => ({
+  marginBottom: theme.spacing(1),
+}));
 
 /**
  */
@@ -14,22 +19,19 @@ export class WorkspaceAddButton extends Component {
    */
   render() {
     const {
-      classes, t, setWorkspaceAddVisibility, isWorkspaceAddVisible, useExtendedFab,
+      t, setWorkspaceAddVisibility, isWorkspaceAddVisible, useExtendedFab,
     } = this.props;
     return (
       <Tooltip title={isWorkspaceAddVisible ? t('closeAddResourceMenu') : t('addResource')}>
-        <Fab
+        <Root
           size="medium"
           color="primary"
           id="addBtn"
-          disableRipple
           aria-label={
             isWorkspaceAddVisible
               ? t('closeAddResourceMenu')
               : ((useExtendedFab && t('startHere')) || t('addResource'))
           }
-          className={classes.fab}
-          classes={{ primary: classes.fabPrimary, secondary: classes.fabSecondary }}
           variant={useExtendedFab ? 'extended' : 'circular'}
           onClick={() => { setWorkspaceAddVisibility(!isWorkspaceAddVisible); }}
         >
@@ -39,14 +41,13 @@ export class WorkspaceAddButton extends Component {
               : <AddIcon />
           }
           { useExtendedFab && t('startHere') }
-        </Fab>
+        </Root>
       </Tooltip>
     );
   }
 }
 
 WorkspaceAddButton.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   isWorkspaceAddVisible: PropTypes.bool,
   setWorkspaceAddVisibility: PropTypes.func.isRequired,
   t: PropTypes.func,
diff --git a/src/components/WorkspaceArea.js b/src/components/WorkspaceArea.js
index b6e2d10ba71a2fff9a5744cb89acb484e76acdb5..16200f8d817f18c107737aa30cdf3f7703f082fc 100644
--- a/src/components/WorkspaceArea.js
+++ b/src/components/WorkspaceArea.js
@@ -1,6 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
+import { styled, lighten, darken } from '@mui/material/styles';
 import ErrorDialog from '../containers/ErrorDialog';
 import WorkspaceControlPanel from '../containers/WorkspaceControlPanel';
 import Workspace from '../containers/Workspace';
@@ -8,6 +8,29 @@ import WorkspaceAdd from '../containers/WorkspaceAdd';
 import BackgroundPluginArea from '../containers/BackgroundPluginArea';
 import ns from '../config/css-ns';
 
+const Root = styled('div', { name: 'WorkspaceArea', slot: 'root' })(({ theme }) => {
+  const getBackgroundColor = theme.palette.mode === 'light' ? darken : lighten;
+
+  return {
+    background: getBackgroundColor(theme.palette.shades.light, 0.1),
+    bottom: 0,
+    display: 'flex',
+    flexDirection: 'column',
+    left: 0,
+    position: 'absolute',
+    right: 0,
+    top: 0,
+    [theme.breakpoints.up('sm')]: {
+      flexDirection: 'row',
+    },
+  };
+});
+
+const ViewerArea = styled('main', { name: 'WorkspaceArea', slot: 'viewer' })(() => ({
+  flexGrow: 1,
+  position: 'relative',
+}));
+
 /**
  * This is the top level Mirador component.
  * @prop {Object} manifests
@@ -20,7 +43,6 @@ export class WorkspaceArea extends Component {
   render() {
     const {
       areaRef,
-      classes,
       controlPanelVariant,
       isWorkspaceAddVisible,
       isWorkspaceControlPanelVisible,
@@ -29,13 +51,13 @@ export class WorkspaceArea extends Component {
     } = this.props;
 
     return (
-      <>
+      <Root ownerState={this.props}>
         {
           isWorkspaceControlPanelVisible
             && <WorkspaceControlPanel variant={controlPanelVariant} />
         }
-        <main
-          className={classNames(classes.viewer, ns('viewer'))}
+        <ViewerArea
+          className={ns('viewer')}
           lang={lang}
           aria-label={t('workspace')}
           {...(areaRef ? { ref: areaRef } : {})}
@@ -47,15 +69,14 @@ export class WorkspaceArea extends Component {
           }
           <ErrorDialog />
           <BackgroundPluginArea />
-        </main>
-      </>
+        </ViewerArea>
+      </Root>
     );
   }
 }
 
 WorkspaceArea.propTypes = {
   areaRef: PropTypes.shape({ current: PropTypes.instanceOf(Element) }),
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   controlPanelVariant: PropTypes.string,
   isWorkspaceAddVisible: PropTypes.bool,
   isWorkspaceControlPanelVisible: PropTypes.bool.isRequired,
diff --git a/src/components/WorkspaceControlPanel.js b/src/components/WorkspaceControlPanel.js
index 376558285d3ac7154425a555690f1a08220f5fc0..1331341330cc8ff29e1199eb1f21f98ad6b2b325 100644
--- a/src/components/WorkspaceControlPanel.js
+++ b/src/components/WorkspaceControlPanel.js
@@ -1,13 +1,59 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import classNames from 'classnames';
-import AppBar from '@material-ui/core/AppBar';
-import Toolbar from '@material-ui/core/Toolbar';
+import AppBar from '@mui/material/AppBar';
+import Toolbar from '@mui/material/Toolbar';
 import WorkspaceAddButton from '../containers/WorkspaceAddButton';
 import WorkspaceControlPanelButtons from '../containers/WorkspaceControlPanelButtons';
 import Branding from '../containers/Branding';
 import ns from '../config/css-ns';
 
+const Root = styled(AppBar, { name: 'WorkspaceControlPanel', slot: 'root' })(({ ownerState, theme }) => ({
+  display: 'flex',
+  height: 64,
+  padding: theme.spacing(1),
+  paddingBottom: 0,
+  position: 'relative',
+  [theme.breakpoints.up('sm')]: {
+    height: 'auto',
+    width: ownerState.variant === 'wide' ? 'auto' : 64,
+  },
+  ...(ownerState.variant === 'wide' && {
+    width: 'auto',
+  }),
+}));
+
+const StyledToolbar = styled(Toolbar, { name: 'WorkspaceControlPanel', slot: 'toolbar' })(({ theme }) => ({
+  display: 'flex',
+  flexGrow: 1,
+  justifyContent: 'space-between',
+  [theme.breakpoints.up('sm')]: {
+    flexDirection: 'column',
+    justifyContent: 'flex-start',
+    minHeight: 0,
+  },
+}));
+
+const StyledWorkspaceButtons = styled('div', { name: 'WorkspaceControlPanel', slot: 'buttonArea' })(({ theme }) => ({
+  [theme.breakpoints.up('sm')]: {
+    display: 'flex',
+    flexDirection: 'column',
+    gap: theme.spacing(2),
+    marginBottom: theme.spacing(1),
+    marginTop: theme.spacing(1),
+  },
+}));
+
+const StyledBranding = styled(Branding, { name: 'WorkspaceControlPanel', slot: 'branding' })(({ theme }) => ({
+  [theme.breakpoints.up('xs')]: {
+    display: 'none',
+  },
+  [theme.breakpoints.up('sm')]: {
+    display: 'flex',
+  },
+}));
+
 /**
  * Provides the panel responsible for controlling the entire workspace
  */
@@ -17,34 +63,36 @@ export class WorkspaceControlPanel extends Component {
    * @return {String} - HTML markup for the component
    */
   render() {
-    const { classes, t, variant } = this.props;
+    const { t, variant } = this.props;
     return (
-      <AppBar
-        className={classNames(classes.root, ns('workspace-control-panel'), variant === 'wide' ? classes.wide : null)}
+      <Root
+        ownerState={this.props}
+        className={classNames(ns('workspace-control-panel'))}
         color="default"
+        enableColorOnDark
         position="absolute"
         component="nav"
         aria-label={t('workspaceNavigation')}
       >
-        <Toolbar disableGutters className={classes.toolbar}>
+        <StyledToolbar
+          disableGutters
+        >
           <WorkspaceAddButton />
-          <div className={classes.workspaceButtons}>
+          <StyledWorkspaceButtons>
             <WorkspaceControlPanelButtons />
-          </div>
-        </Toolbar>
-        <Branding className={classes.branding} t={t} variant={variant} />
-      </AppBar>
+          </StyledWorkspaceButtons>
+        </StyledToolbar>
+        <StyledBranding t={t} variant={variant} />
+      </Root>
     );
   }
 }
 
 WorkspaceControlPanel.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
   t: PropTypes.func.isRequired,
   variant: PropTypes.oneOf(['default', 'wide']),
 };
 
 WorkspaceControlPanel.defaultProps = {
-  classes: {},
   variant: 'default',
 };
diff --git a/src/components/WorkspaceControlPanelButtons.js b/src/components/WorkspaceControlPanelButtons.js
index b07adfc54298955b08cbfc1ffd60299bbad125f9..34cbf840ea9a1810a90e13bbbae9776b684cfea5 100644
--- a/src/components/WorkspaceControlPanelButtons.js
+++ b/src/components/WorkspaceControlPanelButtons.js
@@ -1,5 +1,4 @@
 import { Component } from 'react';
-import PropTypes from 'prop-types';
 import FullScreenButton from '../containers/FullScreenButton';
 import WorkspaceMenuButton from '../containers/WorkspaceMenuButton';
 import WorkspaceOptionsButton from '../containers/WorkspaceOptionsButton';
@@ -16,23 +15,14 @@ export class WorkspaceControlPanelButtons extends Component {
    * @return {type}  description
    */
   render() {
-    const { classes } = this.props;
     return (
       <>
         <WindowListButton />
         <WorkspaceMenuButton />
         <WorkspaceOptionsButton />
-        <FullScreenButton className={classes.ctrlBtn} />
+        <FullScreenButton />
         <PluginHook {...this.props} />
       </>
     );
   }
 }
-
-WorkspaceControlPanelButtons.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string),
-};
-
-WorkspaceControlPanelButtons.defaultProps = {
-  classes: {},
-};
diff --git a/src/components/WorkspaceDialog.js b/src/components/WorkspaceDialog.js
new file mode 100644
index 0000000000000000000000000000000000000000..0928db6502f527924127ff0032422de6fa8aed1b
--- /dev/null
+++ b/src/components/WorkspaceDialog.js
@@ -0,0 +1,13 @@
+import {
+  Dialog,
+} from '@mui/material';
+import { styled } from '@mui/material/styles';
+
+export const WorkspaceDialog = styled(Dialog, { name: 'WorkspaceDialog', slot: 'root' })(({ theme, variant }) => ({
+  '& .MuiDialogTitle-root': theme.unstable_sx({ typography: 'h2' }),
+  ...(variant === 'menu' && {
+    '& .MuiDialogContent-root': {
+      padding: 0,
+    },
+  }),
+}));
diff --git a/src/components/WorkspaceElastic.js b/src/components/WorkspaceElastic.js
index 01c2e79d2beba8f64d1d76b0607a3a2d69569a5f..f871ab9983bac48fd8445862cfda64de337265b3 100644
--- a/src/components/WorkspaceElastic.js
+++ b/src/components/WorkspaceElastic.js
@@ -1,11 +1,29 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import { Rnd } from 'react-rnd';
 import ResizeObserver from 'react-resize-observer';
-import classNames from 'classnames';
 import WorkspaceElasticWindow from '../containers/WorkspaceElasticWindow';
 import ns from '../config/css-ns';
 
+const Root = styled('div', { name: 'WorkspaceElastic', slot: 'root' })({
+  height: '100%',
+  position: 'relative',
+  width: '100%',
+});
+
+const StyledRnd = styled(Rnd)({
+  boxSizing: 'border-box',
+  margin: 0,
+  position: 'absolute',
+  transitionDuration: '.7s',
+  // order matters
+  // eslint-disable-next-line sort-keys
+  '&.react-draggable-dragging': {
+    transitionDuration: 'unset',
+  },
+});
+
 /**
  * Represents a work area that contains any number of windows
  * @memberof Workspace
@@ -16,7 +34,6 @@ class WorkspaceElastic extends Component {
    */
   render() {
     const {
-      classes,
       workspace,
       elasticLayout,
       setWorkspaceViewportDimensions,
@@ -28,13 +45,13 @@ class WorkspaceElastic extends Component {
     const offsetY = workspace.height / 2;
 
     return (
-      <div style={{ height: '100%', position: 'relative', width: '100%' }}>
+      <Root>
         <ResizeObserver
           onReflow={() => {}}
           onResize={(rect) => { setWorkspaceViewportDimensions(rect); }}
         />
 
-        <Rnd
+        <StyledRnd
           size={{
             height: workspace.height,
             width: workspace.width,
@@ -56,7 +73,7 @@ class WorkspaceElastic extends Component {
             setWorkspaceViewportPosition({ x: -1 * d.x - offsetX, y: -1 * d.y - offsetY });
           }}
           cancel={`.${ns('window')}`}
-          className={classNames(classes.workspace, ns('workspace'))}
+          className={ns('workspace')}
           disableDragging={!workspace.draggingEnabled}
         >
           {
@@ -67,14 +84,13 @@ class WorkspaceElastic extends Component {
               />
             ))
           }
-        </Rnd>
-      </div>
+        </StyledRnd>
+      </Root>
     );
   }
 }
 
 WorkspaceElastic.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   elasticLayout: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
   setWorkspaceViewportDimensions: PropTypes.func.isRequired,
   setWorkspaceViewportPosition: PropTypes.func.isRequired,
diff --git a/src/components/WorkspaceElasticWindow.js b/src/components/WorkspaceElasticWindow.js
index 54224913c4c213bf3f2f8058f9f8e05e520b6113..dea8c87b6ad8318300aae46219ba30f763da09af 100644
--- a/src/components/WorkspaceElasticWindow.js
+++ b/src/components/WorkspaceElasticWindow.js
@@ -1,9 +1,14 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
 import { Rnd } from 'react-rnd';
 import Window from '../containers/Window';
 import ns from '../config/css-ns';
 
+const StyledRnd = styled(Rnd)(({ focused, theme }) => ({
+  zIndex: focused ? theme.zIndex.modal - 1 : 'auto',
+}));
+
 /**
  * Represents a work area that contains any number of windows
  * @memberof Workspace
@@ -27,7 +32,9 @@ class WorkspaceElasticWindow extends Component {
     const offsetY = workspace.height / 2;
 
     return (
-      <Rnd
+      <StyledRnd
+        focused={focused}
+        className={focused ? classes.focused : undefined}
         key={`${layout.windowId}-${workspace.id}`}
         size={{
           height: layout.height + companionWindowDimensions.height,
@@ -52,14 +59,11 @@ class WorkspaceElasticWindow extends Component {
         }}
         dragHandleClassName={ns('window-top-bar')}
         cancel={`.${ns('window-menu-btn')}`}
-        className={
-          focused ? classes.focused : null
-        }
       >
         <Window
           windowId={layout.windowId}
         />
-      </Rnd>
+      </StyledRnd>
     );
   }
 }
diff --git a/src/components/WorkspaceExport.js b/src/components/WorkspaceExport.js
index 6e2a1b18a6fe1740125174c21cfd96a907c83154..d6a7962bd3f262c099ad38869a0a1f859bf536ed 100644
--- a/src/components/WorkspaceExport.js
+++ b/src/components/WorkspaceExport.js
@@ -1,19 +1,19 @@
 import { Component } from 'react';
-import Button from '@material-ui/core/Button';
-import Dialog from '@material-ui/core/Dialog';
-import DialogActions from '@material-ui/core/DialogActions';
-import DialogTitle from '@material-ui/core/DialogTitle';
-import DialogContent from '@material-ui/core/DialogContent';
-import Typography from '@material-ui/core/Typography';
-import Snackbar from '@material-ui/core/Snackbar';
-import IconButton from '@material-ui/core/IconButton';
-import ExpandMoreIcon from '@material-ui/icons/ExpandMore';
-import CloseIcon from '@material-ui/icons/Close';
-import Accordion from '@material-ui/core/Accordion';
-import AccordionSummary from '@material-ui/core/AccordionSummary';
-import AccordionDetails from '@material-ui/core/AccordionDetails';
+import Button from '@mui/material/Button';
+import DialogActions from '@mui/material/DialogActions';
+import DialogTitle from '@mui/material/DialogTitle';
+import DialogContent from '@mui/material/DialogContent';
+import Typography from '@mui/material/Typography';
+import Snackbar from '@mui/material/Snackbar';
+import IconButton from '@mui/material/IconButton';
+import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
+import CloseIcon from '@mui/icons-material/Close';
+import Accordion from '@mui/material/Accordion';
+import AccordionSummary from '@mui/material/AccordionSummary';
+import AccordionDetails from '@mui/material/AccordionDetails';
 import PropTypes from 'prop-types';
 import { CopyToClipboard } from 'react-copy-to-clipboard';
+import { WorkspaceDialog } from './WorkspaceDialog';
 
 /**
  */
@@ -54,7 +54,7 @@ export class WorkspaceExport extends Component {
    */
   render() {
     const {
-      children, classes, container, open, t,
+      children, container, open, t,
     } = this.props;
     const { copied } = this.state;
 
@@ -79,7 +79,7 @@ export class WorkspaceExport extends Component {
     }
 
     return (
-      <Dialog
+      <WorkspaceDialog
         id="workspace-export"
         container={container}
         open={open}
@@ -88,19 +88,18 @@ export class WorkspaceExport extends Component {
         fullWidth
         maxWidth="sm"
       >
-        <DialogTitle id="form-dialog-title" disableTypography>
-          <Typography variant="h2">{t('downloadExport')}</Typography>
+        <DialogTitle id="form-dialog-title">
+          {t('downloadExport')}
         </DialogTitle>
 
         <DialogContent>
-          <Accordion elevation={0}>
+          <Accordion elevation={2}>
             <AccordionSummary
-              classes={{ root: classes.accordionTitle }}
               expandIcon={<ExpandMoreIcon />}
             >
               <Typography variant="h4">{t('viewWorkspaceConfiguration')}</Typography>
             </AccordionSummary>
-            <AccordionDetails>
+            <AccordionDetails sx={{ overflow: 'scroll' }}>
               {children}
               <pre>
                 {this.exportedState()}
@@ -118,14 +117,13 @@ export class WorkspaceExport extends Component {
             <Button variant="contained" color="primary">{t('copy')}</Button>
           </CopyToClipboard>
         </DialogActions>
-      </Dialog>
+      </WorkspaceDialog>
     );
   }
 }
 
 WorkspaceExport.propTypes = {
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string),
   container: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   exportableState: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
   handleClose: PropTypes.func.isRequired,
@@ -135,7 +133,6 @@ WorkspaceExport.propTypes = {
 
 WorkspaceExport.defaultProps = {
   children: null,
-  classes: {},
   container: null,
   open: false,
   t: key => key,
diff --git a/src/components/WorkspaceImport.js b/src/components/WorkspaceImport.js
index b0f2c347274ce179f6ddb02e6e9982896e238bad..3a50c482871fa14b6be568a083e83edea289c799 100644
--- a/src/components/WorkspaceImport.js
+++ b/src/components/WorkspaceImport.js
@@ -1,13 +1,12 @@
 import { Component } from 'react';
-import Dialog from '@material-ui/core/Dialog';
-import DialogTitle from '@material-ui/core/DialogTitle';
+import DialogTitle from '@mui/material/DialogTitle';
 import PropTypes from 'prop-types';
 import {
   DialogActions,
   TextField,
-  Typography,
-} from '@material-ui/core';
-import Button from '@material-ui/core/Button';
+} from '@mui/material';
+import Button from '@mui/material/Button';
+import { WorkspaceDialog } from './WorkspaceDialog';
 import ScrollIndicatedDialogContent from '../containers/ScrollIndicatedDialogContent';
 
 /**
@@ -65,7 +64,7 @@ export class WorkspaceImport extends Component {
     } = this.props;
 
     return (
-      <Dialog
+      <WorkspaceDialog
         aria-labelledby="workspace-import-title"
         id="workspace-import"
         onClose={handleClose}
@@ -73,8 +72,8 @@ export class WorkspaceImport extends Component {
         fullWidth
         maxWidth="sm"
       >
-        <DialogTitle id="workspace-import-title" disableTypography>
-          <Typography variant="h2">{t('importWorkspace')}</Typography>
+        <DialogTitle id="workspace-import-title">
+          {t('importWorkspace')}
         </DialogTitle>
         <ScrollIndicatedDialogContent>
           <TextField
@@ -84,19 +83,23 @@ export class WorkspaceImport extends Component {
             onChange={this.handleChange}
             minRows={15}
             variant="filled"
-            inputProps={{ autoFocus: 'autofocus', className: classes.textInput }}
+            sx={{
+              '& .MuiInputBase-input': { fontFamily: 'monospace' },
+              width: '100%',
+            }}
+            inputProps={{ autoFocus: 'autofocus' }}
             helperText={t('importWorkspaceHint')}
           />
         </ScrollIndicatedDialogContent>
         <DialogActions>
-          <Button className={classes.cancelBtn} onClick={handleClose}>
+          <Button onClick={handleClose}>
             {t('cancel')}
           </Button>
           <Button color="primary" onClick={this.handleImportConfig} variant="contained">
             {t('import')}
           </Button>
         </DialogActions>
-      </Dialog>
+      </WorkspaceDialog>
     );
   }
 }
diff --git a/src/components/WorkspaceMenu.js b/src/components/WorkspaceMenu.js
index 65e01ba2ed6dca3fff716f0c250a32cb0e9ea8d4..c6e9a191a60a864ef82114775e7d57a413c3e394 100644
--- a/src/components/WorkspaceMenu.js
+++ b/src/components/WorkspaceMenu.js
@@ -1,7 +1,7 @@
 import { Component } from 'react';
-import Menu from '@material-ui/core/Menu';
-import MenuItem from '@material-ui/core/MenuItem';
-import Typography from '@material-ui/core/Typography';
+import Menu from '@mui/material/Menu';
+import MenuItem from '@mui/material/MenuItem';
+import Typography from '@mui/material/Typography';
 import PropTypes from 'prop-types';
 import LanguageSettings from '../containers/LanguageSettings';
 import { NestedMenu } from './NestedMenu';
@@ -71,8 +71,10 @@ export class WorkspaceMenu extends Component {
       t,
       showZoomControls,
       toggleZoomControls,
-      ...menuProps
+      ...rest
     } = this.props;
+    const menuProps = { ...rest };
+    delete menuProps.tReady;
 
     const {
       changeTheme,
diff --git a/src/components/WorkspaceMenuButton.js b/src/components/WorkspaceMenuButton.js
index 71d8db245f5f7eeb25313dd953e8230be9058d6b..1ff00786511d9794a21b77b5230d4682a110fbe5 100644
--- a/src/components/WorkspaceMenuButton.js
+++ b/src/components/WorkspaceMenuButton.js
@@ -1,7 +1,6 @@
 import { Component } from 'react';
-import SettingsIcon from '@material-ui/icons/SettingsSharp';
+import SettingsIcon from '@mui/icons-material/SettingsSharp';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
 import WorkspaceMenu from '../containers/WorkspaceMenu';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 
@@ -46,7 +45,7 @@ export class WorkspaceMenuButton extends Component {
    * @return
    */
   render() {
-    const { classes, t } = this.props;
+    const { t } = this.props;
     const { anchorEl, open } = this.state;
 
     return (
@@ -55,7 +54,7 @@ export class WorkspaceMenuButton extends Component {
           aria-haspopup="true"
           aria-label={t('workspaceMenu')}
           aria-owns={open ? 'workspace-menu' : undefined}
-          className={classNames(classes.ctrlBtn, (open ? classes.ctrlBtnSelected : null))}
+          selected={open}
           id="menuBtn"
           onClick={this.handleMenuClick}
         >
@@ -73,7 +72,6 @@ export class WorkspaceMenuButton extends Component {
 }
 
 WorkspaceMenuButton.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   t: PropTypes.func,
 };
 
diff --git a/src/components/WorkspaceMosaic.js b/src/components/WorkspaceMosaic.js
index aad63fe5014fa3d760b79f87000ee56a7bb87702..0ac4faa5725b56d46a6691b3f1ba2c853b1d2191 100644
--- a/src/components/WorkspaceMosaic.js
+++ b/src/components/WorkspaceMosaic.js
@@ -1,5 +1,7 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
+import GlobalStyles from '@mui/material/GlobalStyles';
 import { DndContext } from 'react-dnd';
 import {
   Mosaic, MosaicWindow, getLeaves, createBalancedTreeFromLeaves,
@@ -10,6 +12,22 @@ import classNames from 'classnames';
 import MosaicRenderPreview from '../containers/MosaicRenderPreview';
 import Window from '../containers/Window';
 import MosaicLayout from '../lib/MosaicLayout';
+import globalReactMosaicStyles from '../styles/react-mosaic-component';
+
+const StyledMosaic = styled(Mosaic)({
+  '& .mosaic-preview': {
+    boxShadow: 'none',
+  },
+  '& .mosaic-tile': {
+    boxShadow: '0 1px 3px 0 rgba(0, 0, 0, .2), 0 1px 1px 0 rgba(0, 0, 0, .2), 0 2px 1px -1px rgba(0, 0, 0, .2)',
+  },
+  '& .mosaic-window': {
+    boxShadow: 'none',
+  },
+  '& .mosaic-window-toolbar': {
+    display: 'none !important',
+  },
+});
 
 /**
  * Represents a work area that contains any number of windows
@@ -143,18 +161,21 @@ export class WorkspaceMosaic extends Component {
   /**
    */
   render() {
-    const { layout, classes } = this.props;
+    const { layout } = this.props;
     return (
       <DndContext.Consumer>
         {(ctx) => (
-          <Mosaic
-            dragAndDropManager={ctx.dragDropManager}
-            renderTile={this.tileRenderer}
-            initialValue={layout || this.determineWorkspaceLayout()}
-            onChange={this.mosaicChange}
-            className={classNames('mirador-mosaic', classes.root)}
-            zeroStateView={this.zeroStateView}
-          />
+          <>
+            <GlobalStyles styles={{ ...globalReactMosaicStyles }} />
+            <StyledMosaic
+              dragAndDropManager={ctx.dragDropManager}
+              renderTile={this.tileRenderer}
+              initialValue={layout || this.determineWorkspaceLayout()}
+              onChange={this.mosaicChange}
+              className={classNames('mirador-mosaic')}
+              zeroStateView={this.zeroStateView}
+            />
+          </>
         )}
       </DndContext.Consumer>
     );
@@ -162,7 +183,6 @@ export class WorkspaceMosaic extends Component {
 }
 
 WorkspaceMosaic.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   layout: PropTypes.oneOfType(
     [PropTypes.object, PropTypes.string],
   ), // eslint-disable-line react/forbid-prop-types
diff --git a/src/components/WorkspaceOptionsButton.js b/src/components/WorkspaceOptionsButton.js
index bfde64bfb78daff64e4517f63240ff5261c21cd3..ad549551f2fe45cc99475b120ed0f6f57c317274 100644
--- a/src/components/WorkspaceOptionsButton.js
+++ b/src/components/WorkspaceOptionsButton.js
@@ -1,7 +1,6 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import classNames from 'classnames';
-import MoreHorizontalIcon from '@material-ui/icons/MoreHorizSharp';
+import MoreHorizontalIcon from '@mui/icons-material/MoreHorizSharp';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 import WorkspaceOptionsMenu from '../containers/WorkspaceOptionsMenu';
 
@@ -46,21 +45,18 @@ export class WorkspaceOptionsButton extends Component {
    * Returns the rendered component
   */
   render() {
-    const { classes, t } = this.props;
+    const { t } = this.props;
     const { anchorEl, open } = this.state;
 
     return (
       <>
         <MiradorMenuButton
           aria-label={t('workspaceOptions')}
-          className={
-            classNames(classes.ctrlBtn, (open ? classes.ctrlBtnSelected : null))
-          }
           onClick={this.handleMenuClick}
+          selected={open}
         >
           <MoreHorizontalIcon />
         </MiradorMenuButton>
-
         <WorkspaceOptionsMenu
           anchorEl={anchorEl}
           handleClose={this.handleMenuClose}
@@ -72,6 +68,5 @@ export class WorkspaceOptionsButton extends Component {
 }
 
 WorkspaceOptionsButton.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   t: PropTypes.func.isRequired,
 };
diff --git a/src/components/WorkspaceOptionsMenu.js b/src/components/WorkspaceOptionsMenu.js
index 6da23860f7748820f2ab7b2dd08b0028877f17bb..75aac27a94e95a337614cd8126e09cd823524530 100644
--- a/src/components/WorkspaceOptionsMenu.js
+++ b/src/components/WorkspaceOptionsMenu.js
@@ -1,11 +1,11 @@
 import { Component } from 'react';
 import PropTypes from 'prop-types';
-import ImportIcon from '@material-ui/icons/Input';
-import SaveAltIcon from '@material-ui/icons/SaveAltSharp';
-import ListItemIcon from '@material-ui/core/ListItemIcon';
-import Menu from '@material-ui/core/Menu';
-import MenuItem from '@material-ui/core/MenuItem';
-import Typography from '@material-ui/core/Typography';
+import ImportIcon from '@mui/icons-material/Input';
+import SaveAltIcon from '@mui/icons-material/SaveAltSharp';
+import ListItemIcon from '@mui/material/ListItemIcon';
+import Menu from '@mui/material/Menu';
+import MenuItem from '@mui/material/MenuItem';
+import Typography from '@mui/material/Typography';
 import WorkspaceExport from '../containers/WorkspaceExport';
 import WorkspaceImport from '../containers/WorkspaceImport';
 import { PluginHook } from './PluginHook';
diff --git a/src/components/WorkspaceSelectionDialog.js b/src/components/WorkspaceSelectionDialog.js
index f1efbf48d942d886ce160885882dfb157478282b..e606ef872fe8974f1ceba61c6e5c3f53afdd7922 100644
--- a/src/components/WorkspaceSelectionDialog.js
+++ b/src/components/WorkspaceSelectionDialog.js
@@ -1,18 +1,23 @@
 import { Component } from 'react';
-import Dialog from '@material-ui/core/Dialog';
-import DialogTitle from '@material-ui/core/DialogTitle';
+import DialogTitle from '@mui/material/DialogTitle';
 import {
   Card,
   CardContent,
   MenuList,
   MenuItem,
   Typography,
-} from '@material-ui/core';
+} from '@mui/material';
 import PropTypes from 'prop-types';
+import { styled } from '@mui/material/styles';
+import { WorkspaceDialog } from './WorkspaceDialog';
 import WorkspaceTypeElasticIcon from './icons/WorkspaceTypeElasticIcon';
 import WorkspaceTypeMosaicIcon from './icons/WorkspaceTypeMosaicIcon';
 import ScrollIndicatedDialogContent from '../containers/ScrollIndicatedDialogContent';
 
+const StyledDetails = styled('div')(() => ({
+  display: 'flex',
+  flexDirection: 'column',
+}));
 /**
  */
 export class WorkspaceSelectionDialog extends Component {
@@ -42,80 +47,133 @@ export class WorkspaceSelectionDialog extends Component {
    */
   render() {
     const {
-      classes, container, handleClose, open, children, t, workspaceType,
+      container, handleClose, open, children, t, workspaceType,
     } = this.props;
     return (
-      <Dialog
+      <WorkspaceDialog
         aria-labelledby="workspace-selection-dialog-title"
         container={container}
         id="workspace-selection-dialog"
         onClose={handleClose}
         open={open}
       >
-        <DialogTitle id="workspace-selection-dialog-title" disableTypography>
-          <Typography variant="h2">{t('workspaceSelectionTitle')}</Typography>
+        <DialogTitle id="workspace-selection-dialog-title">
+          {t('workspaceSelectionTitle')}
         </DialogTitle>
         <ScrollIndicatedDialogContent>
           {children}
           <MenuList
-            classes={{ root: classes.list }}
+            sx={{
+              '&active': {
+                outline: 'none',
+              },
+              '&focus': {
+                outline: 'none',
+              },
+              outline: 'none',
+            }}
             selected={workspaceType}
             autoFocusItem
           >
             <MenuItem
-              className={classes.menuItem}
+              sx={{
+                height: 'auto',
+                overflow: 'auto',
+                whiteSpace: 'inherit',
+              }}
               onClick={() => this.handleWorkspaceTypeChange('elastic')}
               selected={workspaceType === 'elastic'}
               value="elastic"
             >
-              <Card className={classes.card}>
+              <Card sx={{
+                backgroundColor: 'transparent',
+                borderRadius: '0',
+                boxShadow: '0 0 transparent',
+                display: 'flex',
+              }}
+              >
                 <WorkspaceTypeElasticIcon
-                  className={classes.svgIcon}
+                  sx={{
+                    flexShrink: 0,
+                    height: '90px',
+                    width: '120px',
+                  }}
                   viewBox="0 0 120 90"
                 />
-                <div className={classes.details}>
+                <StyledDetails>
                   <CardContent
-                    classes={{ root: classes.root }}
-                    className={classes.content}
+                    sx={{
+                      '&.MuiCardContent-root': {
+                        '&:last-child': {
+                          paddingBottom: '12px',
+                        },
+                        paddingBottom: 0,
+                        paddingTop: 0,
+                        textAlign: 'left',
+                      },
+                      flex: '1 0 auto',
+                    }}
                   >
-                    <Typography className={classes.headline} component="p" variant="h3">{t('elastic')}</Typography>
+                    <Typography sx={{ paddingBottom: '6px' }} component="p" variant="h3">{t('elastic')}</Typography>
                     <Typography variant="body1">{t('elasticDescription')}</Typography>
                   </CardContent>
-                </div>
+                </StyledDetails>
               </Card>
             </MenuItem>
             <MenuItem
-              className={classes.menuItem}
+              sx={{
+                height: 'auto',
+                overflow: 'auto',
+                whiteSpace: 'inherit',
+              }}
               onClick={() => this.handleWorkspaceTypeChange('mosaic')}
               selected={workspaceType === 'mosaic'}
               value="mosaic"
             >
-              <Card className={classes.card}>
+              <Card sx={{
+                backgroundColor: 'transparent',
+                borderRadius: '0',
+                boxShadow: '0 0 transparent',
+                display: 'flex',
+              }}
+              >
                 <WorkspaceTypeMosaicIcon
-                  className={classes.svgIcon}
+                  sx={{
+                    flexShrink: 0,
+                    height: '90px',
+                    width: '120px',
+                  }}
                   viewBox="0 0 120 90"
                 />
-                <div className={classes.details}>
+                <StyledDetails>
                   <CardContent
-                    className={classes.content}
-                    classes={{ root: classes.root }}
+                    sx={{
+                      '&.MuiCardContent-root': {
+                        '&:last-child': {
+                          paddingBottom: '12px',
+                        },
+                        paddingBottom: 0,
+                        paddingTop: 0,
+                        textAlign: 'left',
+                      },
+                      flex: '1 0 auto',
+                    }}
                   >
-                    <Typography className={classes.headline} component="p" variant="h3">{t('mosaic')}</Typography>
+                    <Typography sx={{ paddingBottom: '6px' }} component="p" variant="h3">{t('mosaic')}</Typography>
                     <Typography variant="body1">{t('mosaicDescription')}</Typography>
                   </CardContent>
-                </div>
+                </StyledDetails>
               </Card>
             </MenuItem>
           </MenuList>
         </ScrollIndicatedDialogContent>
-      </Dialog>
+      </WorkspaceDialog>
     );
   }
 }
 
 WorkspaceSelectionDialog.propTypes = {
   children: PropTypes.node,
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
   container: PropTypes.object, // eslint-disable-line react/forbid-prop-types
   handleClose: PropTypes.func.isRequired,
   open: PropTypes.bool,
diff --git a/src/components/ZoomControls.js b/src/components/ZoomControls.js
index 133c9950a8d8facc4598ac95c74a034d4e1d1e7a..19dc0002291173fe8c4de3467cd11a91b4519b77 100644
--- a/src/components/ZoomControls.js
+++ b/src/components/ZoomControls.js
@@ -1,10 +1,17 @@
 import { Component } from 'react';
-import AddCircleIcon from '@material-ui/icons/AddCircleOutlineSharp';
-import RemoveCircleIcon from '@material-ui/icons/RemoveCircleOutlineSharp';
+import AddCircleIcon from '@mui/icons-material/AddCircleOutlineSharp';
+import RemoveCircleIcon from '@mui/icons-material/RemoveCircleOutlineSharp';
+import { styled } from '@mui/material/styles';
 import PropTypes from 'prop-types';
 import RestoreZoomIcon from './icons/RestoreZoomIcon';
 import MiradorMenuButton from '../containers/MiradorMenuButton';
 
+const StyledZoomControlsWrapper = styled('div')({
+  display: 'flex',
+  flexDirection: 'row',
+  justifyContent: 'center',
+});
+
 /**
  */
 export class ZoomControls extends Component {
@@ -46,17 +53,11 @@ export class ZoomControls extends Component {
    */
   render() {
     const {
-      displayDivider, showZoomControls, classes, t, zoomToWorld,
+      t, zoomToWorld,
     } = this.props;
 
-    if (!showZoomControls) {
-      return (
-        <>
-        </>
-      );
-    }
     return (
-      <div className={classes.zoom_controls}>
+      <StyledZoomControlsWrapper>
         <MiradorMenuButton aria-label={t('zoomIn')} onClick={this.handleZoomInClick}>
           <AddCircleIcon />
         </MiradorMenuButton>
@@ -66,16 +67,12 @@ export class ZoomControls extends Component {
         <MiradorMenuButton aria-label={t('zoomReset')} onClick={() => zoomToWorld(false)}>
           <RestoreZoomIcon />
         </MiradorMenuButton>
-        {displayDivider && <span className={classes.divider} />}
-      </div>
+      </StyledZoomControlsWrapper>
     );
   }
 }
 
 ZoomControls.propTypes = {
-  classes: PropTypes.objectOf(PropTypes.string).isRequired,
-  displayDivider: PropTypes.bool,
-  showZoomControls: PropTypes.bool,
   t: PropTypes.func,
   updateViewport: PropTypes.func,
   viewer: PropTypes.shape({
@@ -88,8 +85,6 @@ ZoomControls.propTypes = {
 };
 
 ZoomControls.defaultProps = {
-  displayDivider: true,
-  showZoomControls: false,
   t: key => key,
   updateViewport: () => {},
   viewer: {},
diff --git a/src/components/icons/BookViewIcon.js b/src/components/icons/BookViewIcon.js
index a6668b1f30dd3f2334232b6a78defa451bf7e982..d26b9fe3ccb9071786167e894586d2f44d406e24 100644
--- a/src/components/icons/BookViewIcon.js
+++ b/src/components/icons/BookViewIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * BookViewIcon ~
diff --git a/src/components/icons/CanvasIndexIcon.js b/src/components/icons/CanvasIndexIcon.js
index 8d5af98b10ffc6d5a8b644a75b88260ccf3ef5af..2bac4808d03e9a90c20f4dd4531c8a5aa99cfb90 100644
--- a/src/components/icons/CanvasIndexIcon.js
+++ b/src/components/icons/CanvasIndexIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * Render the canvas index svg
diff --git a/src/components/icons/GalleryViewIcon.js b/src/components/icons/GalleryViewIcon.js
index 9a410033404f1fb35a6260eae3892bb9f91f8d11..cccd9d2c711c176af48926783c844e35d6a35600 100644
--- a/src/components/icons/GalleryViewIcon.js
+++ b/src/components/icons/GalleryViewIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * GalleryViewIcon ~
diff --git a/src/components/icons/MiradorIcon.js b/src/components/icons/MiradorIcon.js
index 920256dd9bf3c856729973c4343eff41265b0d60..43301fe4cb9c8bdf808da08666d037a97db9ee0b 100644
--- a/src/components/icons/MiradorIcon.js
+++ b/src/components/icons/MiradorIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * ThumbnailNavigationRightIcon ~
diff --git a/src/components/icons/RestoreZoomIcon.js b/src/components/icons/RestoreZoomIcon.js
index 2005e6e89c5b61f851bcf398014e26581008eb7e..f1cf1e2d1e3eafc22b8eb6777115b7ea6ee0b898 100644
--- a/src/components/icons/RestoreZoomIcon.js
+++ b/src/components/icons/RestoreZoomIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * RestoreZoomIcon ~
diff --git a/src/components/icons/ThumbnailNavigationBottomIcon.js b/src/components/icons/ThumbnailNavigationBottomIcon.js
index 5c7080a787b68f4431e3f5b31b73d9996a859eb3..4ef2b95448cac052e7cd245fe94e6fcf4fcd7e36 100644
--- a/src/components/icons/ThumbnailNavigationBottomIcon.js
+++ b/src/components/icons/ThumbnailNavigationBottomIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * ThumbnailNavigationBottomIcon ~
diff --git a/src/components/icons/ThumbnailNavigationRightIcon.js b/src/components/icons/ThumbnailNavigationRightIcon.js
index 8f22b31d00ac81fffb763d4fc82e49c2eedc1ae6..6690b01d280f0e7b0362ca138f9cda8d4fbaa875 100644
--- a/src/components/icons/ThumbnailNavigationRightIcon.js
+++ b/src/components/icons/ThumbnailNavigationRightIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * ThumbnailNavigationRightIcon ~
diff --git a/src/components/icons/WindowMaxIcon.js b/src/components/icons/WindowMaxIcon.js
index 9590bbb180d5ac1bbb1b322dfadc7960604efb70..3de34b6bf5e3d9efffcf7eefe179c5b229157546 100644
--- a/src/components/icons/WindowMaxIcon.js
+++ b/src/components/icons/WindowMaxIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * WindowMaxIcon ~
diff --git a/src/components/icons/WindowMinIcon.js b/src/components/icons/WindowMinIcon.js
index 7a2ef3be9ca1ec7ffd926f0b5edad8f8b66b247c..3118b915f7e0fcd8f0a22fc1269e8bd9fa62f3ca 100644
--- a/src/components/icons/WindowMinIcon.js
+++ b/src/components/icons/WindowMinIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * WindowMinIcon ~
diff --git a/src/components/icons/WindowOptionsIcon.js b/src/components/icons/WindowOptionsIcon.js
index 3e365c29307ee2d02b5970e4a4253f84caa5f657..e87dd282539643d1c8049c9b2a44dd4ef37913c0 100644
--- a/src/components/icons/WindowOptionsIcon.js
+++ b/src/components/icons/WindowOptionsIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * WindowMinIcon ~
diff --git a/src/components/icons/WorkspaceTypeElasticIcon.js b/src/components/icons/WorkspaceTypeElasticIcon.js
index c817330897909d6a7daafa8fbb88d29074187a48..e1ee53918ad5afed6ff58cad42078605514b34e9 100644
--- a/src/components/icons/WorkspaceTypeElasticIcon.js
+++ b/src/components/icons/WorkspaceTypeElasticIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * ElasticWorkspaceIcon ~
diff --git a/src/components/icons/WorkspaceTypeMosaicIcon.js b/src/components/icons/WorkspaceTypeMosaicIcon.js
index 5cdb0c34a10bed5204c03410330c7439bf0894ee..c1724ee7e88ed1b7dbf86f150f5935dcd13cf648 100644
--- a/src/components/icons/WorkspaceTypeMosaicIcon.js
+++ b/src/components/icons/WorkspaceTypeMosaicIcon.js
@@ -1,4 +1,4 @@
-import SvgIcon from '@material-ui/core/SvgIcon';
+import SvgIcon from '@mui/material/SvgIcon';
 
 /**
  * WorkspaceTypeMosaicIcon ~
diff --git a/src/config/settings.js b/src/config/settings.js
index 5aa08c8f99e97afc8a1d755cbf2a2677def553f9..caaebe2e15e340a62508363a3947e9618805e314 100644
--- a/src/config/settings.js
+++ b/src/config/settings.js
@@ -12,7 +12,7 @@ export default {
   themes: {
     dark: {
       palette: {
-        type: 'dark',
+        mode: 'dark',
         primary: {
           main: '#4db6ac',
         },
@@ -28,13 +28,13 @@ export default {
     },
     light: {
       palette: {
-        type: 'light',
+        mode: 'light',
       }
     }
   },
   theme: { // Sets up a MaterialUI theme. See https://material-ui.com/customization/default-theme/
     palette: {
-      type: 'light',
+      mode: 'light',
       primary: {
         main: '#1967d2', // Controls the color of the Add button and current window indicator
       },
@@ -61,6 +61,7 @@ export default {
       },
       section_divider: 'rgba(0, 0, 0, 0.25)',
       annotations: {
+        chipBackground: '#e0e0e0',
         hidden: { globalAlpha: 0 },
         default: { strokeStyle: '#00BFFF', globalAlpha: 1 },
         hovered: { strokeStyle: '#BF00FF', globalAlpha: 1 },
@@ -167,51 +168,248 @@ export default {
       },
       useNextVariants: true // set so that console deprecation warning is removed
     },
-    overrides: {
-      MuiListSubheader: {
-        root: {
-          '&[role="presentation"]:focus': {
-            outline: 0,
+    components: {
+      MuiMenuItem: {
+        variants: [
+          {
+            props: { variant: 'multiline' },
+            style: { whiteSpace: 'normal' }
           },
-        },
+        ]
       },
-      MuiTooltip: { // Overridden from https://github.com/mui-org/material-ui/blob/master/packages/material-ui/src/Tooltip/Tooltip.js#L40-L70
-        tooltipPlacementLeft: {
-          ['@media (min-width:600px)']: {
-            margin: 0,
+      CompanionWindow: {
+        styleOverrides: {
+          closeButton: {
+            order: 4,
           },
-        },
-        tooltipPlacementRight: {
-          ['@media (min-width:600px)']: {
-            margin: 0,
+          contents: {
+            overflowY: 'auto',
+            wordBreak: 'break-word',
           },
-        },
-        tooltipPlacementTop: {
-          ['@media (min-width:600px)']: {
-            margin: 0,
+          controls: ({ ownerState }) => ({
+            alignItems: 'center',
+            display: 'flex',
+            flexFlow: 'row wrap',
+            flexGrow: 1,
+            justifyContent: (ownerState?.position === 'bottom' || ownerState?.position === 'far-bottom') ? 'flex-end' : 'flex-start',
+            minHeight: 48,
+            order: 3
+          }),
+          positionButton: {
+            marginLeft: -16,
+            order: -100,
+            width: 24,
           },
+          resize: ({ ownerState }) => ({
+            display: 'flex',
+            flexDirection: 'column',
+            minHeight: 50,
+            minWidth: (ownerState?.position === 'left') ? 235 : 100,
+            position: 'relative',
+          }),
+          root: ({ ownerState }) => ({
+            boxShadow: 'none',
+            boxSizing: 'border-box',
+            display: 'flex',
+            flexDirection: 'column',
+            minHeight: 0,
+            ...(ownerState?.position === 'right' && {
+              borderLeft: '0.5px solid rgba(0, 0, 0, 0.125)'
+            }),
+            ...(ownerState?.position === 'left' && {
+              borderRight: '0.5px solid rgba(0, 0, 0, 0.125)'
+            }),
+            ...(ownerState?.position === 'bottom' && {
+              borderTop: '0.5px solid rgba(0, 0, 0, 0.125)'
+            }),
+          }),
+          title: ({ theme }) => ({
+            ...theme.typography.subtitle1,
+            alignSelf: 'center',
+            flexGrow: 1,
+            width: 160
+          }),
+          toolbar: ({ theme }) => ({
+            alignItems: 'flex-start',
+            backgroundColor: theme.palette.shades.light,
+            flexWrap: 'wrap',
+            justifyContent: 'space-between',
+            minHeight: 'max-content',
+            paddingInlineStart: '1rem',
+          }),
         },
-        tooltipPlacementBottom: {
-          ['@media (min-width:600px)']: {
-            margin: 0,
+      },
+      CompanionWindowSection: {
+        styleOverrides: {
+          root: {
+            borderBlockEnd: '.5px solid rgba(0, 0, 0, 0.25)'
           },
         },
       },
-      MuiTouchRipple: {
-        childPulsate: {
-          animation: 'none',
+      IIIFHtmlContent: {
+        styleOverrides: {
+          root: ({ theme }) => ({
+            '& a': {
+              color: theme.palette.primary.main,
+              textDecoration: 'underline',
+            },
+          }),
         },
-        rippleVisible: {
-          animation: 'none',
+      },
+      IIIFThumbnail: {
+        styleOverrides: {
+          root: ({ ownerState }) => ({
+            ...(ownerState?.variant === 'inside' && {
+              display: 'inline-block',
+              height: 'inherit',
+              position: 'relative',
+            }),
+          }),
+          label: ({ ownerState }) => ({
+            overflow: 'hidden',
+            textOverflow: 'ellipsis',
+            lineHeight: '1.5em',
+            wordBreak: 'break-word',
+            ...(ownerState?.variant === 'inside' && {
+              color: '#ffffff',
+              WebkitLineClamp: 1,
+              whiteSpace: 'nowrap',
+            }),
+            ...(ownerState?.variant === 'outside' && {
+              display: '-webkit-box',
+              maxHeight: '3em',
+              MozBoxOrient: 'vertical',
+              WebkitLineClamp: 2,
+            }),
+            ...(ownerState?.variant === 'inside' && {
+              background: 'linear-gradient(to top, rgba(0,0,0,0.7) 0%, rgba(0,0,0,0.3) 70%, rgba(0,0,0,0) 100%)',
+              bottom: '5px',
+              boxSizing: 'border-box',
+              left: '0px',
+              padding: '4px',
+              position: 'absolute',
+              width: '100%',
+            }),
+          }),
+          image: ({ ownerState }) => ({
+            ...(ownerState?.border && {
+              border: '1px solid rgba(0, 0, 0, 0.125)',
+            }),
+          })
+        }
+      },
+      ThemeIcon: {
+        styleOverrides: {
+          icon: ({ ownerState }) => ({
+            color: (ownerState?.value === 'dark' ? '#000000' : undefined)
+          }),
         },
       },
-    },
-    props: {
+      MuiAccordion: {
+        variants: [
+          {
+            props: { variant: 'compact' },
+            style: {
+              '& .MuiAccordionSummary-root': {
+                minHeight: 'unset',
+                padding: 0,
+              },
+              '& .MuiAccordionSummary-content': {
+                margin: 0,
+              },
+              '& .MuiAccordionDetails-root': {
+                padding: 0,
+              },
+            },
+          },
+        ],
+      },
+      MuiButton: {
+        styleOverrides: {
+          inlineText: {
+            lineHeight: '1.5em',
+            padding: 0,
+            textAlign: 'inherit',
+            textTransform: 'none',
+
+          },
+          inlineTextSecondary: ({ theme }) => ({
+            color: theme.palette.secondary.main,
+          }),
+        }
+      },
       MuiButtonBase: {
-        disableTouchRipple: true,
+        defaultProps: {
+          disableTouchRipple: true,
+        },
+      },
+      MuiDialog: {
+        variants: [
+          {
+            props: { variant: 'contained' },
+            style: {
+              position: 'absolute',
+              '& .MuiBackdrop-root': {
+                position: 'absolute'
+              }
+            },
+          }
+        ]
+      },
+      MuiFab: {
+        styleOverrides: {
+          root: {
+            transition: 'none',
+          }
+        },
       },
       MuiLink: {
-        underline: 'always'
+        defaultProps: {
+          underline: 'always'
+        },
+      },
+      MuiListSubheader: {
+        styleOverrides: {
+          root: {
+            '&[role="presentation"]:focus': {
+              outline: 0,
+            },
+          },
+        },
+      },
+      MuiTooltip: { // Overridden from https://github.com/mui-org/material-ui/blob/master/packages/material-ui/src/Tooltip/Tooltip.js#L40-L70
+        styleOverrides: {
+          tooltipPlacementLeft: {
+            ['@media (min-width:600px)']: {
+              margin: '0 !important',
+            },
+          },
+          tooltipPlacementRight: {
+            ['@media (min-width:600px)']: {
+              margin: '0 !important',
+            },
+          },
+          tooltipPlacementTop: {
+            ['@media (min-width:600px)']: {
+              margin: '0 !important',
+            },
+          },
+          tooltipPlacementBottom: {
+            ['@media (min-width:600px)']: {
+              margin: '0 !important',
+            },
+          },
+        },
+      },
+      MuiTouchRipple: {
+        styleOverrides: {
+          childPulsate: {
+            animation: 'none',
+          },
+          rippleVisible: {
+            animation: 'none',
+          },
+        },
       },
     },
   },
diff --git a/src/containers/AppProviders.js b/src/containers/AppProviders.js
index 22e8a3a06d4381880ecb305bea240a6df748ac1b..1ad1c6784dde9f7e5b583f13b2e39851bc397f40 100644
--- a/src/containers/AppProviders.js
+++ b/src/containers/AppProviders.js
@@ -11,7 +11,6 @@ import { AppProviders } from '../components/AppProviders';
  */
 const mapStateToProps = state => (
   {
-    createGenerateClassNameOptions: getConfig(state).createGenerateClassNameOptions,
     language: getConfig(state).language,
     theme: getTheme(state),
     translations: getConfig(state).translations,
diff --git a/src/containers/AttributionPanel.js b/src/containers/AttributionPanel.js
index 8e0801444dd60e1786f4924b52bc0c3b3316fe7d..94ae930e0f78abee992b3e42f660b539f9a3696d 100644
--- a/src/containers/AttributionPanel.js
+++ b/src/containers/AttributionPanel.js
@@ -1,13 +1,12 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
-import { withPlugins } from '../extend/withPlugins';
 import {
   getManifestLogo,
   getRequiredStatement,
   getRights,
 } from '../state/selectors';
+import { withPlugins } from '../extend/withPlugins';
 import { AttributionPanel } from '../components/AttributionPanel';
 
 /**
@@ -21,29 +20,7 @@ const mapStateToProps = (state, { id, windowId }) => ({
   rights: getRights(state, { windowId }),
 });
 
-/**
- *
- * @param theme
- * @returns {label: {paddingLeft: number}}}
- */
-const styles = theme => ({
-  logo: {
-    maxWidth: '100%',
-  },
-  placeholder: {
-    backgroundColor: theme.palette.grey[300],
-  },
-  section: {
-    borderBottom: `.5px solid ${theme.palette.section_divider}`,
-    paddingBottom: theme.spacing(1),
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps),
   withPlugins('AttributionPanel'),
diff --git a/src/containers/AudioViewer.js b/src/containers/AudioViewer.js
index e422c6d6b6e00458773de3c365080307590b9fd9..2bcd274096fcff1ff41ce76fd584853c964734ed 100644
--- a/src/containers/AudioViewer.js
+++ b/src/containers/AudioViewer.js
@@ -1,7 +1,6 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { AudioViewer } from '../components/AudioViewer';
 import { getConfig, getVisibleCanvasAudioResources, getVisibleCanvasCaptions } from '../state/selectors';
@@ -15,21 +14,8 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/** */
-const styles = () => ({
-  audio: {
-    width: '100%',
-  },
-  container: {
-    alignItems: 'center',
-    display: 'flex',
-    width: '100%',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('AudioViewer'),
 );
diff --git a/src/containers/CanvasAnnotations.js b/src/containers/CanvasAnnotations.js
index 5bebeae7885399b8685ac82bab9462b38020add4..5d420e745df44d7853cea0faacdf14aea44c0606 100644
--- a/src/containers/CanvasAnnotations.js
+++ b/src/containers/CanvasAnnotations.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import {
@@ -49,35 +48,8 @@ const mapDispatchToProps = {
   selectAnnotation: actions.selectAnnotation,
 };
 
-/** For withStlyes */
-const styles = theme => ({
-  annotationListItem: {
-    '&$hovered': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    '&:hover,&:focus': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    borderBottom: `0.5px solid ${theme.palette.divider}`,
-    cursor: 'pointer',
-    whiteSpace: 'normal',
-  },
-  chip: {
-    backgroundColor: theme.palette.background.paper,
-    marginRight: theme.spacing(0.5),
-    marginTop: theme.spacing(1),
-  },
-  hovered: {},
-  sectionHeading: {
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('CanvasAnnotations'),
 );
diff --git a/src/containers/CanvasLayers.js b/src/containers/CanvasLayers.js
index 87041bed9628dd6faf56f67ebe2711078f0e60b3..1b5baf2ffb1580d4fc5cf233bdec4e11bcb9b263 100644
--- a/src/containers/CanvasLayers.js
+++ b/src/containers/CanvasLayers.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import * as actions from '../state/actions';
 import {
   getCanvasLabel,
@@ -26,81 +25,8 @@ const mapDispatchToProps = {
   updateLayers: actions.updateLayers,
 };
 
-/** For withStlyes */
-const styles = theme => ({
-  dragging: {},
-  dragHandle: {
-    alignItems: 'center',
-    borderRight: `0.5px solid ${theme.palette.divider}`,
-    display: 'flex',
-    flex: 1,
-    flexDirection: 'row',
-    marginBottom: -1 * theme.spacing(2) + 0.5,
-    marginRight: theme.spacing(1),
-    marginTop: -1 * theme.spacing(2),
-    maxWidth: theme.spacing(3),
-    width: theme.spacing(3),
-  },
-  image: {
-    borderBottom: `1px solid ${theme.palette.divider}`,
-  },
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-  list: {
-    paddingTop: 0,
-  },
-  listItem: {
-    '& $dragHandle': {
-      '&:hover': {
-        backgroundColor: theme.palette.action.hover,
-      },
-      backgroundColor: theme.palette.shades?.light,
-    },
-    '&$dragging': {
-      '& $dragHandle, & $dragHandle:hover': {
-        backgroundColor: theme.palette.action.selected,
-      },
-      backgroundColor: theme.palette.action.hover,
-    },
-    alignItems: 'stretch',
-    borderBottom: `0.5px solid ${theme.palette.divider}`,
-    cursor: 'pointer',
-    paddingBottom: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-  opacityIcon: {
-    marginRight: theme.spacing(0.5),
-  },
-  opacityInput: {
-    ...theme.typography.caption,
-    '&::-webkit-outer-spin-button,&::-webkit-inner-spin-button': {
-      '-webkit-appearance': 'none',
-      margin: 0,
-    },
-    '-moz-appearance': 'textfield',
-    textAlign: 'right',
-    width: '3ch',
-  },
-  sectionHeading: {
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-  slider: {
-    marginLeft: theme.spacing(2),
-    marginRight: theme.spacing(2),
-    maxWidth: 150,
-  },
-  thumbnail: {
-    minWidth: 50,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
 );
 
diff --git a/src/containers/ChangeThemeDialog.js b/src/containers/ChangeThemeDialog.js
index ade8c4eeebe49ebeb76b720f099b1ec4be4863c6..a473f22f1b8346f7d7c7fee65d637eee463b79cb 100644
--- a/src/containers/ChangeThemeDialog.js
+++ b/src/containers/ChangeThemeDialog.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
@@ -26,31 +25,8 @@ const mapStateToProps = state => ({
   themeIds: getThemeIds(state),
 });
 
-/** */
-const styles = theme => ({
-  dark: {
-    color: '#000000',
-  },
-  dialogContent: {
-    padding: 0,
-  },
-  light: {
-    color: '#BDBDBD',
-  },
-  listitem: {
-    '&:focus': {
-      backgroundColor: theme.palette.action.focus,
-    },
-    '&:hover': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    cursor: 'pointer',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ChangeThemeDialog'),
 );
diff --git a/src/containers/CollapsibleSection.js b/src/containers/CollapsibleSection.js
index 3daa9258b1ae29be903287fca527ea8ba41a459d..9e32b49c242eff19469c3f98207e15b163544b80 100644
--- a/src/containers/CollapsibleSection.js
+++ b/src/containers/CollapsibleSection.js
@@ -1,24 +1,9 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { CollapsibleSection } from '../components/CollapsibleSection';
 
-const styles = {
-  button: {
-    padding: 0,
-  },
-  container: {
-    display: 'flex',
-    justifyContent: 'space-between',
-  },
-  heading: {
-    cursor: 'pointer',
-  },
-};
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
 );
 
 export default enhance(CollapsibleSection);
diff --git a/src/containers/CollectionDialog.js b/src/containers/CollectionDialog.js
index 1920dee3cd50bdb1303849969f2887d823aca7bd..02297bccbcae46e1233f62c569e812bbad4790b1 100644
--- a/src/containers/CollectionDialog.js
+++ b/src/containers/CollectionDialog.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
@@ -48,44 +47,8 @@ const mapStateToProps = (state, { windowId }) => {
   };
 };
 
-/** */
-const styles = theme => ({
-  collectionFilter: {
-    padding: '16px',
-    paddingTop: 0,
-  },
-  collectionItem: {
-    whiteSpace: 'normal',
-  },
-  collectionMetadata: {
-    padding: '16px',
-  },
-  dark: {
-    color: '#000000',
-  },
-  dialog: {
-    position: 'absolute !important',
-  },
-  dialogContent: {
-    padding: theme.spacing(1),
-  },
-  light: {
-    color: theme.palette.grey[400],
-  },
-  listitem: {
-    '&:focus': {
-      backgroundColor: theme.palette.action.focus,
-    },
-    '&:hover': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    cursor: 'pointer',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withWorkspaceContext,
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('CollectionDialog'),
diff --git a/src/containers/CompanionArea.js b/src/containers/CompanionArea.js
index 8ff8b9cb628481f47bbb5728f4b15dec52c463fb..f403a90849fa55e4ae43f23f98385d885eb3c8bc 100644
--- a/src/containers/CompanionArea.js
+++ b/src/containers/CompanionArea.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import {
@@ -21,43 +20,8 @@ const mapDispatchToProps = ({
   setCompanionAreaOpen: actions.setCompanionAreaOpen,
 });
 
-/** */
-const styles = theme => ({
-  horizontal: {
-    flexDirection: 'column',
-    width: '100%',
-  },
-  left: {
-    minWidth: 235,
-  },
-  root: {
-    display: 'flex',
-    minHeight: 0,
-    position: 'relative',
-    zIndex: theme.zIndex.appBar - 2,
-  },
-  toggle: {
-    backgroundColor: theme.palette.background.paper,
-    border: `1px solid ${theme.palette.shades?.dark}`,
-    borderRadius: 0,
-    height: '48px',
-    left: '100%',
-    marginTop: '1rem',
-    padding: 2,
-    position: 'absolute',
-    width: '23px',
-    zIndex: theme.zIndex.drawer,
-  },
-  toggleButton: {
-    marginBottom: 12,
-    marginTop: 12,
-    padding: 0,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('CompanionArea'),
 );
diff --git a/src/containers/CompanionWindow.js b/src/containers/CompanionWindow.js
index 9def7a3c3c0e468c0f643c9a302b42655cacb6a3..64a1806440f48be886d013f8cb4878be2b1f409d 100644
--- a/src/containers/CompanionWindow.js
+++ b/src/containers/CompanionWindow.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withSize } from 'react-sizeme';
 import { withPlugins } from '../extend/withPlugins';
 import { withRef } from '../extend/withRef';
@@ -45,95 +44,9 @@ const mapDispatchToProps = (dispatch, { windowId, id }) => ({
   ),
 });
 
-/**
- *
- * @param theme
- * @returns {{closeButton: {top: number, position: string, right: number},
- * root: {overflowY: string, width: string}}}
- */
-const styles = theme => ({
-  closeButton: {
-    order: 4,
-  },
-  'companionWindow-bottom': {
-    borderTop: `0.5px solid ${theme.palette.divider}`,
-  },
-  'companionWindow-left': {
-    borderRight: `0.5px solid ${theme.palette.divider}`,
-  },
-  'companionWindow-right': {
-    borderLeft: `0.5px solid ${theme.palette.divider}`,
-  },
-  companionWindowHeader: {
-    flexWrap: 'wrap',
-  },
-  companionWindowTitleControls: {
-    flexGrow: 1,
-    order: 1000,
-  },
-  companionWindowTitleControlsBottom: {
-    order: 'unset',
-  },
-  content: {
-    overflowY: 'auto',
-    wordBreak: 'break-word',
-  },
-  horizontal: {
-  },
-  positionButton: {
-    marginLeft: -16,
-    order: -100,
-    width: 24,
-  },
-  rnd: {
-    display: 'flex',
-    flexDirection: 'column',
-    minHeight: 0,
-  },
-  root: {
-    boxShadow: 'none',
-    boxSizing: 'border-box',
-    display: 'flex',
-    flexDirection: 'column',
-    minHeight: 0,
-  },
-  small: {},
-  titleControls: {
-    alignItems: 'center',
-    display: 'flex',
-    flexFlow: 'row wrap',
-    minHeight: 48,
-    order: 3,
-  },
-  toolbar: {
-    '&$small': {
-      '& $closeButton': {
-        order: 'unset',
-      },
-      '& $titleControls': {
-        order: 'unset',
-      },
-    },
-    alignItems: 'flex-start',
-    background: theme.palette.shades?.light,
-    justifyContent: 'space-between',
-    minHeight: 'max-content',
-    paddingLeft: theme.spacing(2),
-  },
-  vertical: {
-  },
-  windowSideBarTitle: {
-    ...theme.typography.subtitle1,
-    alignSelf: 'center',
-    flexGrow: 1,
-    width: 160,
-  },
-});
-
 const enhance = compose(
   withRef(),
   withTranslation(),
-  withStyles(styles),
   withSize(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('CompanionWindow'),
diff --git a/src/containers/CustomPanel.js b/src/containers/CustomPanel.js
index 493ad146335bc1e55955512e721ad1bd83244ed4..3f83547018ed6f2b604d0bdb4d506556662133b8 100644
--- a/src/containers/CustomPanel.js
+++ b/src/containers/CustomPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { CustomPanel } from '../components/CustomPanel';
 
@@ -11,17 +10,8 @@ import { CustomPanel } from '../components/CustomPanel';
 const mapStateToProps = (state, { id, windowId }) => ({
 });
 
-/**
- *
- * @param theme
- * @returns {label: {paddingLeft: number}}}
- */
-const styles = theme => ({
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps),
   withPlugins('CustomPanel'),
 );
diff --git a/src/containers/ErrorContent.js b/src/containers/ErrorContent.js
index ce3089f9d0f01b14b4d642e5194e2cf517ecd9e0..85443562090cbcdbf4501e01a7d765d561c44ef8 100644
--- a/src/containers/ErrorContent.js
+++ b/src/containers/ErrorContent.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { ErrorContent } from '../components/ErrorContent';
 import {
@@ -23,29 +22,8 @@ const mapStateToProps = (state, { companionWindowId, windowId }) => ({
   showJsError: getConfig(state).window.showJsError,
 });
 
-/**
- * @param theme
- * @returns {{typographyBody: {flexGrow: number, fontSize: number|string},
- * windowTopBarStyle: {minHeight: number, paddingLeft: number, backgroundColor: string}}}
- */
-const styles = theme => ({
-  alert: {
-    backgroundColor: theme.palette.error.main,
-    color: '#fff',
-    fontWeight: theme.typography.fontWeightMedium,
-  },
-  details: {
-    '& pre': {
-      height: '100px',
-      overflowY: 'scroll',
-    },
-    flexDirection: 'column',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps),
   withPlugins('ErrorContent'),
 );
diff --git a/src/containers/GalleryView.js b/src/containers/GalleryView.js
index 6b916d755abb443155a84da0efcac4ccd453b082..9c54be5a4b3971d371f187fdafbb58490e461945 100644
--- a/src/containers/GalleryView.js
+++ b/src/containers/GalleryView.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { GalleryView } from '../components/GalleryView';
 import { getCanvases, getSequenceViewingDirection } from '../state/selectors';
@@ -17,24 +16,7 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/**
- * Styles to be passed to the withStyles HOC
- */
-const styles = theme => ({
-  galleryContainer: {
-    alignItems: 'flex-start',
-    display: 'flex',
-    flexDirection: 'row',
-    flexWrap: 'wrap',
-    overflowX: 'hidden',
-    overflowY: 'scroll',
-    padding: '50px 0 50px 20px',
-    width: '100%',
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   connect(mapStateToProps),
   withPlugins('GalleryView'),
   // further HOC go here
diff --git a/src/containers/GalleryViewThumbnail.js b/src/containers/GalleryViewThumbnail.js
index 66d6a94a625c7f12db73a32d8e8d6fa2f29e16ca..c0306777ce9d9aecd58dfa13b733764dc95c5b51 100644
--- a/src/containers/GalleryViewThumbnail.js
+++ b/src/containers/GalleryViewThumbnail.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import flatten from 'lodash/flatten';
-import { withStyles } from '@material-ui/core/styles';
 import * as actions from '../state/actions';
 import { GalleryViewThumbnail } from '../components/GalleryViewThumbnail';
 import {
@@ -12,62 +11,6 @@ import {
   getCompanionWindowsForContent,
 } from '../state/selectors';
 
-/**
- * Styles to be passed to the withStyles HOC
- */
-const styles = theme => ({
-  annotationIcon: {
-    height: '1rem',
-    width: '1rem',
-  },
-  annotationsChip: {
-    ...theme.typography.caption,
-  },
-  avatar: {
-    backgroundColor: 'transparent',
-  },
-  chips: {
-    opacity: 0.875,
-    position: 'absolute',
-    right: 0,
-    textAlign: 'right',
-    top: 0,
-  },
-  galleryViewItem: {
-    '&$hasAnnotations': {
-      border: `2px solid ${theme.palette.action.selected}`,
-    },
-    '&$selected,&$selected$hasAnnotations': {
-      border: `2px solid ${theme.palette.primary.main}`,
-    },
-    '&:focus': {
-      outline: 'none',
-    },
-    '&:hover': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    border: '2px solid transparent',
-    cursor: 'pointer',
-    display: 'inline-block',
-    margin: `${theme.spacing(1)}px ${theme.spacing(0.5)}px`,
-    maxHeight: props => props.config.height + 45,
-    minWidth: '60px',
-    overflow: 'hidden',
-    padding: theme.spacing(0.5),
-    position: 'relative',
-    width: 'min-content',
-  },
-  hasAnnotations: {},
-  searchChip: {
-    ...theme.typography.caption,
-    '&$selected $avatar': {
-      backgroundColor: theme.palette.highlights?.primary,
-    },
-    marginTop: 2,
-  },
-  selected: {},
-});
-
 /** */
 const mapStateToProps = (state, { canvas, windowId }) => {
   const currentCanvas = getCurrentCanvas(state, { windowId });
@@ -115,7 +58,6 @@ const mapDispatchToProps = (dispatch, { canvas, id, windowId }) => ({
 
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   // further HOC go here
 );
 
diff --git a/src/containers/IIIFThumbnail.js b/src/containers/IIIFThumbnail.js
index c596b508cd92fee25b69b5e09e198011182363e7..6631a8deb3d5e296210213072e71a68cc30439c4 100644
--- a/src/containers/IIIFThumbnail.js
+++ b/src/containers/IIIFThumbnail.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import {
   getConfig,
@@ -16,51 +15,7 @@ const mapStateToProps = (state) => ({
   thumbnailsConfig: getConfig(state).thumbnails,
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  caption: {
-    lineHeight: '1.5em',
-    wordBreak: 'break-word',
-  },
-  image: {},
-  insideCaption: {
-    color: '#ffffff',
-    lineClamp: '1',
-    whiteSpace: 'nowrap',
-  },
-  insideLabel: {
-    background: 'linear-gradient(to top, rgba(0,0,0,0.7) 0%, rgba(0,0,0,0.3) 70%, rgba(0,0,0,0) 100%)',
-    bottom: '5px',
-    boxSizing: 'border-box',
-    left: '0px',
-    padding: '4px',
-    position: 'absolute',
-    width: '100%',
-  },
-  insideRoot: {
-    display: 'inline-block',
-    height: 'inherit',
-    position: 'relative',
-  },
-  label: {
-    overflow: 'hidden',
-    textOverflow: 'ellipsis',
-  },
-  outsideCaption: {
-    boxOrient: 'vertical',
-    display: '-webkit-box',
-    lineClamp: '2',
-    maxHeight: '3em',
-  },
-  outsideLabel: {},
-  outsideRoot: {},
-  root: {},
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps),
   withPlugins('IIIFThumbnail'),
diff --git a/src/containers/LayersPanel.js b/src/containers/LayersPanel.js
index 16a1a9b01b5cec0d1005c357debb1fd72dcf801d..2d7352f4eb2ce54a4927221a3e2aef820a258e23 100644
--- a/src/containers/LayersPanel.js
+++ b/src/containers/LayersPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { LayersPanel } from '../components/LayersPanel';
 import {
@@ -15,17 +14,8 @@ const mapStateToProps = (state, { id, windowId }) => ({
   canvasIds: getVisibleCanvasIds(state, { windowId }),
 });
 
-/**
- *
- * @param theme
- * @returns {label: {paddingLeft: number}}}
- */
-const styles = theme => ({
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps),
   withPlugins('LayersPanel'),
 );
diff --git a/src/containers/LocalePicker.js b/src/containers/LocalePicker.js
index 509fef4e3648f8a7b4e4baaf5b0052e12dc76607..2edb53b784605e79a283efcb2a08838131297894 100644
--- a/src/containers/LocalePicker.js
+++ b/src/containers/LocalePicker.js
@@ -1,27 +1,9 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { LocalePicker } from '../components/LocalePicker';
 
-/**
- *
- * @param theme
- * @returns {label: {paddingLeft: number}}}
- */
-const styles = theme => ({
-  select: {
-    '&:focus': {
-      backgroundColor: theme.palette.background.paper,
-    },
-  },
-  selectEmpty: {
-    backgroundColor: theme.palette.background.paper,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
 );
 
 export default enhance(LocalePicker);
diff --git a/src/containers/ManifestForm.js b/src/containers/ManifestForm.js
index e64b68d68c74d31e07ae6c7c3627b761b83a22a7..02a32621445f22f08e4137054d6b2895bd0e69fb 100644
--- a/src/containers/ManifestForm.js
+++ b/src/containers/ManifestForm.js
@@ -1,7 +1,6 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { ManifestForm } from '../components/ManifestForm';
@@ -12,24 +11,8 @@ import { ManifestForm } from '../components/ManifestForm';
  * @private
  */
 const mapDispatchToProps = { addResource: actions.addResource };
-/**
- *
- * @param theme
- */
-const styles = theme => ({
-  buttons: {
-    textAlign: 'right',
-    [theme.breakpoints.up('sm')]: {
-      textAlign: 'inherit',
-    },
-  },
-  input: {
-    ...theme.typography.body1,
-  },
-});
 
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(null, mapDispatchToProps),
   withPlugins('ManifestForm'),
diff --git a/src/containers/ManifestListItem.js b/src/containers/ManifestListItem.js
index 6553a9cfa24a9e74766a40e28b1d66b3c7d87a15..55f5c43e90be8bbb35d0a048eeaef46d74a7ae43 100644
--- a/src/containers/ManifestListItem.js
+++ b/src/containers/ManifestListItem.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import {
   getManifest,
@@ -50,56 +49,8 @@ const mapDispatchToProps = {
   fetchManifest: actions.fetchManifest,
 };
 
-/**
- *
- * @param theme
- * @returns {{root: {}, label: {textAlign: string, textTransform: string}}}
- */
-const styles = theme => ({
-  active: {},
-  buttonGrid: {
-  },
-  label: {
-    textAlign: 'left',
-    textTransform: 'initial',
-  },
-  logo: {
-    height: '2.5rem',
-    maxWidth: '100%',
-    objectFit: 'contain',
-    paddingRight: 8,
-  },
-  placeholder: {
-    backgroundColor: theme.palette.grey[300],
-  },
-  root: {
-    '&$active': {
-      borderLeft: `4px solid ${theme.palette.primary.main}`,
-    },
-    '&:hover,&:focus-within': {
-      '&$active': {
-        borderLeft: `4px solid ${theme.palette.primary.main}`,
-      },
-      backgroundColor: theme.palette.action.hover,
-      borderLeft: `4px solid ${theme.palette.action.hover}`,
-    },
-    borderLeft: '4px solid transparent',
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(2),
-    [theme.breakpoints.up('sm')]: {
-      paddingLeft: theme.spacing(3),
-      paddingRight: theme.spacing(3),
-    },
-  },
-  thumbnail: {
-    maxWidth: '100%',
-    objectFit: 'contain',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ManifestListItem'),
 );
diff --git a/src/containers/ManifestListItemError.js b/src/containers/ManifestListItemError.js
index 001d98750b73b2916dded1f97fb205e82df2f433..40d9f5f14dd188eb03b0b691d6eae96cffee308f 100644
--- a/src/containers/ManifestListItemError.js
+++ b/src/containers/ManifestListItemError.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { fetchManifest, removeResource } from '../state/actions';
 import { ManifestListItemError } from '../components/ManifestListItemError';
@@ -12,26 +11,8 @@ const mapDispatchToProps = {
   onTryAgainClick: fetchManifest,
 };
 
-/**
- *
- * @param theme
- * @returns {{manifestIdText: {wordBreak: string},
- * errorIcon: {color: string, width: string, height: string}}}
- */
-const styles = theme => ({
-  errorIcon: {
-    color: theme.palette.error.main,
-    height: '2rem',
-    width: '2rem',
-  },
-  manifestIdText: {
-    wordBreak: 'break-all',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(null, mapDispatchToProps),
   withPlugins('ManifestListItemError'),
 );
diff --git a/src/containers/ManifestRelatedLinks.js b/src/containers/ManifestRelatedLinks.js
index 6bcc356778bf8d8f6388d25c2e2698462ea1a119..a3f1edf9e0a702b8b94fe9f28b8e1317493a0cb3 100644
--- a/src/containers/ManifestRelatedLinks.js
+++ b/src/containers/ManifestRelatedLinks.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import {
   getManifestHomepage,
@@ -25,17 +24,7 @@ const mapStateToProps = (state, { id, windowId }) => ({
   seeAlso: getManifestSeeAlso(state, { windowId }),
 });
 
-const styles = {
-  labelValueMetadata: {
-    '& dd': {
-      marginBottom: '.5em',
-      marginLeft: '0',
-    },
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps),
   withPlugins('ManifestRelatedLinks'),
diff --git a/src/containers/MinimalWindow.js b/src/containers/MinimalWindow.js
index 5ed8aec777d6abe9e2abaa2eedf6c55ecca01abf..d4e57cb75b41a5db524aaef50172dada9a6563d4 100644
--- a/src/containers/MinimalWindow.js
+++ b/src/containers/MinimalWindow.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { MinimalWindow } from '../components/MinimalWindow';
@@ -22,42 +21,8 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   removeWindow: () => dispatch(actions.removeWindow(windowId)),
 });
 
-/**
- * @param theme
- * @returns {{typographyBody: {flexGrow: number, fontSize: number|string},
- * windowTopBarStyle: {minHeight: number, paddingLeft: number, backgroundColor: string}}}
- */
-const styles = theme => ({
-  button: {
-    marginLeft: 'auto',
-  },
-  title: {
-    ...theme.typography.h6,
-    flexGrow: 1,
-    paddingLeft: theme.spacing(0.5),
-  },
-  window: {
-    backgroundColor: theme.palette.shades?.dark,
-    borderRadius: 0,
-    display: 'flex',
-    flexDirection: 'column',
-    height: '100%',
-    minHeight: 0,
-    overflow: 'hidden',
-    width: '100%',
-  },
-  windowTopBarStyle: {
-    backgroundColor: theme.palette.shades?.main,
-    borderTop: '2px solid transparent',
-    minHeight: 32,
-    paddingLeft: theme.spacing(0.5),
-    paddingRight: theme.spacing(0.5),
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('MinimalWindow'),
 );
diff --git a/src/containers/MosaicRenderPreview.js b/src/containers/MosaicRenderPreview.js
index 0103bab9c5beeba6658c2573ade4196ef5f2243b..149861f89f6eabdde49a83dd85592a80228a62e2 100644
--- a/src/containers/MosaicRenderPreview.js
+++ b/src/containers/MosaicRenderPreview.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import { getManifestTitle } from '../state/selectors';
@@ -13,18 +12,7 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/**
- *
- * @param theme
- */
-const styles = theme => ({
-  preview: {
-    ...theme.typography.h4,
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, null),
   withPlugins('MosaicRenderPreview'),
diff --git a/src/containers/OpenSeadragonViewer.js b/src/containers/OpenSeadragonViewer.js
index b5c59dff19b3cccda6d45e2430ffcf60487d4e80..adc01cf90c054e3d7eb0a28f26158bef22b89f5e 100644
--- a/src/containers/OpenSeadragonViewer.js
+++ b/src/containers/OpenSeadragonViewer.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import flatten from 'lodash/flatten';
 import { withPlugins } from '../extend/withPlugins';
 import { OpenSeadragonViewer } from '../components/OpenSeadragonViewer';
@@ -55,16 +54,7 @@ const mapDispatchToProps = {
   updateViewport: actions.updateViewport,
 };
 
-const styles = {
-  osdContainer: {
-    cursor: 'grab',
-    flex: 1,
-    position: 'relative',
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('OpenSeadragonViewer'),
diff --git a/src/containers/PrimaryWindow.js b/src/containers/PrimaryWindow.js
index 336894235f31a71867a2e77da22347d9099c7909..1895e8053964c70d57bdb34f8316dc7704832f63 100644
--- a/src/containers/PrimaryWindow.js
+++ b/src/containers/PrimaryWindow.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import {
   getManifestoInstance, getVisibleCanvasAudioResources, getVisibleCanvasVideoResources, getWindow,
@@ -18,16 +17,7 @@ const mapStateToProps = (state, { windowId }) => {
   };
 };
 
-const styles = {
-  primaryWindow: {
-    display: 'flex',
-    flex: 1,
-    position: 'relative',
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('PrimaryWindow'),
 );
diff --git a/src/containers/SanitizedHtml.js b/src/containers/SanitizedHtml.js
index 932bfa0c0a650823e8c4d473d230fd749581b251..b6771d0d9e5585f2d35e7fd2ef7b7caaf468e334 100644
--- a/src/containers/SanitizedHtml.js
+++ b/src/containers/SanitizedHtml.js
@@ -1,16 +1,3 @@
-import { withStyles } from '@material-ui/core/styles';
 import { SanitizedHtml } from '../components/SanitizedHtml';
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  root: {
-    '& a': {
-      color: theme.palette.primary.main,
-      textDecoration: 'underline',
-    },
-  },
-});
-
-export default withStyles(styles)(SanitizedHtml);
+export default SanitizedHtml;
diff --git a/src/containers/ScrollIndicatedDialogContent.js b/src/containers/ScrollIndicatedDialogContent.js
index 66ec29e1abc18cec67c6603ae5dbc848bec301fe..aad0df3d57d586b115f3664b4001dfe7f58ed383 100644
--- a/src/containers/ScrollIndicatedDialogContent.js
+++ b/src/containers/ScrollIndicatedDialogContent.js
@@ -1,29 +1,3 @@
-import { withStyles } from '@material-ui/core/styles';
 import { ScrollIndicatedDialogContent } from '../components/ScrollIndicatedDialogContent';
 
-/**
- * Styles for the withStyles HOC
- */
-const styles = theme => ({
-  shadowScrollDialog: {
-    /* Shadow covers */
-    background: `linear-gradient(${theme.palette.background.paper} 30%, rgba(255, 255, 255, 0)), `
-                + `linear-gradient(rgba(255, 255, 255, 0), ${theme.palette.background.paper} 70%) 0 100%, `
-                // Shaddows
-                + 'radial-gradient(50% 0, farthest-side, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)), '
-                + 'radial-gradient(50% 100%, farthest-side, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)) 0 100%,',
-    /* Shadow covers */
-    background: `linear-gradient(${theme.palette.background.paper} 30%, rgba(255, 255, 255, 0)), ` // eslint-disable-line no-dupe-keys
-                + `linear-gradient(rgba(255, 255, 255, 0), ${theme.palette.background.paper} 70%) 0 100%, `
-                // Shaddows
-                + 'radial-gradient(farthest-side at 50% 0, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)), '
-                + 'radial-gradient(farthest-side at 50% 100%, rgba(0, 0, 0, .2), rgba(0, 0, 0, 0)) 0 100%;',
-
-    backgroundAttachment: 'local, local, scroll, scroll',
-    backgroundRepeat: 'no-repeat',
-    backgroundSize: '100% 40px, 100% 40px, 100% 14px, 100% 14px',
-    overflowY: 'auto',
-  },
-});
-
-export default withStyles(styles)(ScrollIndicatedDialogContent);
+export default ScrollIndicatedDialogContent;
diff --git a/src/containers/SearchHit.js b/src/containers/SearchHit.js
index 009f44903e6e8f5f058248b27f98b1d9460911e1..7e67df152f62bde81582c712da441141f1590ecf 100644
--- a/src/containers/SearchHit.js
+++ b/src/containers/SearchHit.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SearchHit } from '../components/SearchHit';
 import * as actions from '../state/actions';
@@ -70,60 +69,8 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   ),
 });
 
-/** */
-const styles = theme => ({
-  adjacent: {},
-  focused: {},
-  hitCounter: {
-    ...theme.typography.subtitle2,
-    backgroundColor: theme.palette.hitCounter?.default,
-    height: 30,
-    marginRight: theme.spacing(1),
-    verticalAlign: 'inherit',
-  },
-  inlineButton: {
-    '& span': {
-      lineHeight: '1.5em',
-    },
-    margin: 0,
-    padding: 0,
-    textTransform: 'none',
-  },
-  listItem: {
-    '&$adjacent': {
-      '& $hitCounter': {
-        backgroundColor: theme.palette.highlights?.secondary,
-      },
-      '&$windowSelected': {
-        '& $hitCounter': {
-          backgroundColor: theme.palette.highlights?.primary,
-        },
-      },
-    },
-    '&$windowSelected': {
-      '& $hitCounter': {
-        backgroundColor: theme.palette.highlights?.primary,
-      },
-      '&$focused': {
-        '&:hover': {
-          backgroundColor: 'inherit',
-        },
-        backgroundColor: 'inherit',
-      },
-    },
-    borderBottom: `0.5px solid ${theme.palette.divider}`,
-    paddingRight: 8,
-  },
-  selected: {},
-  subtitle: {
-    marginBottom: theme.spacing(1.5),
-  },
-  windowSelected: {},
-});
-
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   withTranslation(),
   withPlugins('SearchHit'),
 );
diff --git a/src/containers/SearchPanel.js b/src/containers/SearchPanel.js
index 64461b48c8f2ae69f343890eeaeae1d9ce34e208..5a5f4aa4088eb554c56345381e82ec0d54583c2a 100644
--- a/src/containers/SearchPanel.js
+++ b/src/containers/SearchPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import * as actions from '../state/actions';
 import { withPlugins } from '../extend/withPlugins';
 import { SearchPanel } from '../components/SearchPanel';
@@ -22,27 +21,8 @@ const mapDispatchToProps = (dispatch, props) => ({
   removeSearch: () => dispatch(actions.removeSearch(props.windowId, props.id)),
 });
 
-/**
-* Styles for withStyles HOC
-*/
-const styles = theme => ({
-  clearChip: {
-    marginLeft: theme.spacing(1),
-  },
-  inlineButton: {
-    '& span': {
-      lineHeight: '1.5em',
-    },
-    margin: theme.spacing(2),
-    padding: 0,
-    textAlign: 'inherit',
-    textTransform: 'none',
-  },
-});
-
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   withTranslation(),
   withPlugins('SearchPanel'),
 );
diff --git a/src/containers/SearchPanelControls.js b/src/containers/SearchPanelControls.js
index e81d46a4f3b48dedb66f93f4e2bc3034aeb77f12..e2f718ae07e613b059bc5d4ad2e206586be32b3c 100644
--- a/src/containers/SearchPanelControls.js
+++ b/src/containers/SearchPanelControls.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SearchPanelControls } from '../components/SearchPanelControls';
 import * as actions from '../state/actions';
@@ -33,26 +32,8 @@ const mapDispatchToProps = {
   fetchSearch: actions.fetchSearch,
 };
 
-/** */
-const styles = theme => ({
-  endAdornment: {
-    position: 'absolute',
-    right: 0,
-  },
-  form: {
-    paddingBottom: theme.spacing(1),
-    paddingRight: theme.spacing(1.5),
-    width: '100%',
-  },
-  searchProgress: {
-    position: 'absolute',
-    right: 0,
-  },
-});
-
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   withTranslation(),
   withPlugins('SearchPanelControls'),
 );
diff --git a/src/containers/SearchPanelNavigation.js b/src/containers/SearchPanelNavigation.js
index 614b2419209114973c256fc4029944724185b715..83b421ac3bcca2e24ca0e593390ecfca4eb56869 100644
--- a/src/containers/SearchPanelNavigation.js
+++ b/src/containers/SearchPanelNavigation.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SearchPanelNavigation } from '../components/SearchPanelNavigation';
 import * as actions from '../state/actions';
@@ -37,17 +36,8 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   ),
 });
 
-/** */
-const styles = theme => ({
-  body2: {
-    marginLeft: '-16px',
-    width: '100%',
-  },
-});
-
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   withTranslation(),
   withPlugins('SearchPanelNavigation'),
 );
diff --git a/src/containers/SearchResults.js b/src/containers/SearchResults.js
index 15e660d99a297bfdb9a13a57d4f19e48697c8097..0684e36b8ef5ea04d37945200b31e4484639c9fe 100644
--- a/src/containers/SearchResults.js
+++ b/src/containers/SearchResults.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SearchResults } from '../components/SearchResults';
 import * as actions from '../state/actions';
@@ -33,26 +32,8 @@ const mapDispatchToProps = {
   fetchSearch: actions.fetchSearch,
 };
 
-/** */
-const styles = theme => ({
-  moreButton: {
-    width: '100%',
-  },
-  navigation: {
-    textTransform: 'none',
-  },
-  noResults: {
-    ...theme.typography.h6,
-    padding: theme.spacing(2),
-  },
-  toggleFocus: {
-    ...theme.typography.subtitle1,
-  },
-});
-
 const enhance = compose(
   connect(mapStateToProps, mapDispatchToProps),
-  withStyles(styles),
   withTranslation(),
   withPlugins('SearchResults'),
 );
diff --git a/src/containers/SelectCollection.js b/src/containers/SelectCollection.js
index 446c32711f233e68c9ca4fe904e643c4e71d0ab2..e711f53708a42c3f8c395178f2c0dbca35b88e8a 100644
--- a/src/containers/SelectCollection.js
+++ b/src/containers/SelectCollection.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import * as actions from '../state/actions';
 import { withPlugins } from '../extend/withPlugins';
 import {
@@ -22,13 +21,9 @@ const mapStateToProps = (state, { windowId }) => {
 const mapDispatchToProps = {
   showCollectionDialog: actions.showCollectionDialog,
 };
-/** */
-const styles = (theme) => ({
-});
 
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('SelectCollection'),
 );
diff --git a/src/containers/SidebarIndexItem.js b/src/containers/SidebarIndexItem.js
index e58afff81c4fdd32cd41d211f51aff84590fed28..bc8a19001ea4d2ac41e743ecb4f338d9873506f7 100644
--- a/src/containers/SidebarIndexItem.js
+++ b/src/containers/SidebarIndexItem.js
@@ -1,21 +1,10 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SidebarIndexItem } from '../components/SidebarIndexItem';
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(null, null),
   withPlugins('SidebarIndexItem'),
diff --git a/src/containers/SidebarIndexList.js b/src/containers/SidebarIndexList.js
index 866e7d61d7229e239182378c8cd3e075a5a85cfa..de46a8893914f51188a78541e0899a866524b0f5 100644
--- a/src/containers/SidebarIndexList.js
+++ b/src/containers/SidebarIndexList.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import {
@@ -29,21 +28,7 @@ const mapDispatchToProps = (dispatch, { id, windowId }) => ({
   setCanvas: (...args) => dispatch(actions.setCanvas(...args)),
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-  listItem: {
-    borderBottom: `0.5px solid ${theme.palette.divider}`,
-    paddingRight: theme.spacing(1),
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('SidebarIndexList'),
diff --git a/src/containers/SidebarIndexTableOfContents.js b/src/containers/SidebarIndexTableOfContents.js
index a626981f346b31775b7511a3c85d4065c0874c27..870f0c4087c0b5e0ae1ba55e59b5e048a3cc58cc 100644
--- a/src/containers/SidebarIndexTableOfContents.js
+++ b/src/containers/SidebarIndexTableOfContents.js
@@ -1,8 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
-import { alpha } from '@material-ui/core/styles/colorManipulator';
 import { withPlugins } from '../extend/withPlugins';
 import { SidebarIndexTableOfContents } from '../components/SidebarIndexTableOfContents';
 import {
@@ -34,45 +32,7 @@ const mapDispatchToProps = (dispatch, { id, windowId }) => ({
   toggleNode: nodeId => dispatch(actions.toggleNode(windowId, id, nodeId)),
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  content: {
-    alignItems: 'flex-start',
-    borderLeft: '1px solid transparent',
-    padding: '8px 16px 8px 0',
-    width: 'auto',
-  },
-  group: {
-    borderLeft: `1px solid ${theme.palette.grey[300]}`,
-  },
-  label: {
-    paddingLeft: 0,
-  }, // needed for pseudo $label class
-  root: {
-    flexGrow: 1,
-  },
-  selected: {}, // needed for pseudo $selected class
-  treeItemRoot: {
-    '&:focus > $content': {
-      backgroundColor: theme.palette.action.selected,
-    },
-    '&:hover > $content': {
-      backgroundColor: theme.palette.action.hover,
-    },
-    '&:hover >$content $label, &:focus > $content $label, &$selected > $content $label, &$selected > $content $label:hover, &$selected:focus > $content $label': {
-      backgroundColor: 'transparent',
-    },
-  },
-  visibleNode: {
-    backgroundColor: alpha(theme.palette.highlights?.primary || theme.palette.action.selected, 0.35),
-    display: 'inline',
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('SidebarIndexTableOfContents'),
diff --git a/src/containers/SidebarIndexThumbnail.js b/src/containers/SidebarIndexThumbnail.js
index 9f582dc4647693cfb9d78d574648bd4169f066d1..74c831814fefcc36ec6e69a06643bce378d7ec89 100644
--- a/src/containers/SidebarIndexThumbnail.js
+++ b/src/containers/SidebarIndexThumbnail.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { SidebarIndexThumbnail } from '../components/SidebarIndexThumbnail';
 import { getConfig } from '../state/selectors';
@@ -15,17 +14,7 @@ const mapStateToProps = (state, { data }) => ({
   ...(getConfig(state).canvasNavigation || {}),
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, null),
   withPlugins('SidebarIndexThumbnail'),
diff --git a/src/containers/ThumbnailCanvasGrouping.js b/src/containers/ThumbnailCanvasGrouping.js
index 0804cee2d51e20fbf78c28f64f4d73cef0b9aa93..548c941ba941931b77e7915ca1549a43269c5211 100644
--- a/src/containers/ThumbnailCanvasGrouping.js
+++ b/src/containers/ThumbnailCanvasGrouping.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { getCurrentCanvas } from '../state/selectors';
@@ -25,33 +24,8 @@ const mapStateToProps = (state, { data }) => ({
   currentCanvasId: (getCurrentCanvas(state, { windowId: data.windowId }) || {}).id,
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  canvas: {
-    '&$currentCanvas': {
-      outline: `2px solid ${theme.palette.primary.main}`,
-      outlineOffset: '3px',
-    },
-    '&:hover': {
-      outline: `9px solid ${theme.palette.action.hover}`,
-      outlineOffset: '-2px',
-    },
-    boxSizing: 'border-box',
-    color: theme.palette.common.white,
-    cursor: 'pointer',
-    display: 'inline-block',
-    outline: 0,
-    whiteSpace: 'nowrap',
-  },
-  currentCanvas: {
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ThumbnailCanvasGrouping'),
 );
diff --git a/src/containers/ThumbnailNavigation.js b/src/containers/ThumbnailNavigation.js
index e3be9a8f3363f1068771976c0d71929445d9e771..bf7c3490aaa6db191facdaad429aa0c14c5087b4 100644
--- a/src/containers/ThumbnailNavigation.js
+++ b/src/containers/ThumbnailNavigation.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { ThumbnailNavigation } from '../components/ThumbnailNavigation';
@@ -40,20 +39,7 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   setPreviousCanvas: (...args) => dispatch(actions.setPreviousCanvas(windowId)),
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  thumbNavigation: {
-    '&:focus': {
-      boxShadow: 0,
-      outline: 0,
-    },
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ThumbnailNavigation'),
diff --git a/src/containers/VideoViewer.js b/src/containers/VideoViewer.js
index 70e8ebc0832ba985c42d185bb3ffc98e074b8993..3c868940a211a4b048006184fb0928f8416eab99 100644
--- a/src/containers/VideoViewer.js
+++ b/src/containers/VideoViewer.js
@@ -1,7 +1,6 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { VideoViewer } from '../components/VideoViewer';
 import { getConfig, getVisibleCanvasCaptions, getVisibleCanvasVideoResources } from '../state/selectors';
@@ -15,22 +14,8 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/** */
-const styles = () => ({
-  container: {
-    alignItems: 'center',
-    display: 'flex',
-    width: '100%',
-  },
-  video: {
-    maxHeight: '100%',
-    width: '100%',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('VideoViewer'),
 );
diff --git a/src/containers/ViewerInfo.js b/src/containers/ViewerInfo.js
index 7e2714d9b7880b574dea33859bb272bd98445394..6757102f98ea7cf276f753ea852eb133e2938148 100644
--- a/src/containers/ViewerInfo.js
+++ b/src/containers/ViewerInfo.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { ViewerInfo } from '../components/ViewerInfo';
 import {
@@ -29,20 +28,7 @@ const mapStateToProps = (state, props) => {
   };
 };
 
-const styles = {
-  osdInfo: {
-    order: 2,
-    overflow: 'hidden',
-    paddingBottom: 3,
-    textOverflow: 'ellipsis',
-    unicodeBidi: 'plaintext',
-    whiteSpace: 'nowrap',
-    width: '100%',
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, null),
   withPlugins('ViewerInfo'),
diff --git a/src/containers/ViewerNavigation.js b/src/containers/ViewerNavigation.js
index 38fb8d4602b0e24e8095486f1c13aa6edaac7a33..3f142de4315e01ad9ad214fc58e0bfdcfc58111f 100644
--- a/src/containers/ViewerNavigation.js
+++ b/src/containers/ViewerNavigation.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import {
@@ -28,14 +27,7 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   setPreviousCanvas: (...args) => dispatch(actions.setPreviousCanvas(windowId)),
 });
 
-const styles = {
-  osdNavigation: {
-    order: 1,
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ViewerNavigation'),
diff --git a/src/containers/Window.js b/src/containers/Window.js
index a06801e2a445d6a9530b9ec6de882fc994c79984..36cd4e14e1a5a87a6fd43cf8ec78a0f16e15c1f5 100644
--- a/src/containers/Window.js
+++ b/src/containers/Window.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
@@ -36,70 +35,8 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   focusWindow: () => dispatch(actions.focusWindow(windowId)),
 });
 
-/**
- * @param theme
- */
-const styles = theme => ({
-  companionAreaBottom: {
-    display: 'flex',
-    flex: '0',
-    flexBasis: 'auto',
-    minHeight: 0,
-  },
-  companionAreaRight: {
-    display: 'flex',
-    flex: '0 1 auto',
-    minHeight: 0,
-  },
-  maximized: {},
-  middle: {
-    display: 'flex',
-    flex: '1',
-    flexDirection: 'row',
-    minHeight: 0,
-  },
-  middleLeft: {
-    display: 'flex',
-    flex: '1',
-    flexDirection: 'column',
-    minHeight: 0,
-  },
-  primaryWindow: {
-    display: 'flex',
-    flex: '1',
-    height: '300px',
-    minHeight: 0,
-    position: 'relative',
-  },
-  thumbnailArea: {
-    backgroundColor: theme.palette.shades?.dark,
-  },
-  thumbnailAreaBottom: {
-  },
-  thumbnailAreaRight: {
-    minWidth: 100,
-  },
-  window: {
-    '&$maximized': {
-      left: 0,
-      position: 'absolute',
-      top: 0,
-      zIndex: theme.zIndex.modal - 1,
-    },
-    backgroundColor: theme.palette.shades?.dark,
-    borderRadius: 0,
-    display: 'flex',
-    flexDirection: 'column',
-    height: '100%',
-    minHeight: 0,
-    overflow: 'hidden',
-    width: '100%',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('Window'),
 );
diff --git a/src/containers/WindowAuthenticationBar.js b/src/containers/WindowAuthenticationBar.js
index f3b7f8733367a8b7f2adf4c1e35f74d49b3353f1..b38a244aa80c6a2c31568c07caff1133affde8ab 100644
--- a/src/containers/WindowAuthenticationBar.js
+++ b/src/containers/WindowAuthenticationBar.js
@@ -1,62 +1,10 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
-import { alpha } from '@material-ui/core/styles/colorManipulator';
 import { withPlugins } from '../extend/withPlugins';
 import { WindowAuthenticationBar } from '../components/WindowAuthenticationBar';
 
-/**
- * @param theme
- * @returns {{typographyBody: {flexGrow: number, fontSize: number|string},
- * windowTopBarStyle: {minHeight: number, paddingLeft: number, backgroundColor: string}}}
- */
-const styles = theme => ({
-  buttonInvert: {
-    '&:hover': {
-      backgroundColor: alpha(theme.palette.secondary.contrastText, 1 - theme.palette.action.hoverOpacity),
-    },
-    backgroundColor: theme.palette.secondary.contrastText,
-    marginLeft: theme.spacing(5),
-    paddingBottom: 0,
-    paddingTop: 0,
-  },
-  expanded: {
-    paddingLeft: theme.spacing(),
-    paddingRight: theme.spacing(),
-  },
-  failure: {
-    backgroundColor: theme.palette.error.dark,
-  },
-  fauxButton: {
-    marginLeft: theme.spacing(2.5),
-  },
-  icon: {
-    marginRight: theme.spacing(1.5),
-    verticalAlign: 'text-bottom',
-  },
-  label: {
-    lineHeight: 2.25,
-  },
-  paper: {
-    backgroundColor: theme.palette.secondary.main,
-    color: theme.palette.secondary.contrastText,
-    cursor: 'pointer',
-  },
-  topBar: {
-    '&:hover': {
-      backgroundColor: theme.palette.secondary.main,
-    },
-    alignItems: 'center',
-    display: 'flex',
-    justifyContent: 'inherit',
-    padding: theme.spacing(1),
-    textTransform: 'none',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withPlugins('WindowAuthenticationBar'),
 );
 
diff --git a/src/containers/WindowCanvasNavigationControls.js b/src/containers/WindowCanvasNavigationControls.js
index f1f17053f8af45c4d492d70f51b93e49771e38d2..ca5088f34b6e32c34ef511435f01dd01143add70 100644
--- a/src/containers/WindowCanvasNavigationControls.js
+++ b/src/containers/WindowCanvasNavigationControls.js
@@ -1,45 +1,18 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
 import { withSize } from 'react-sizeme';
-import { withStyles } from '@material-ui/core';
-import { alpha } from '@material-ui/core/styles/colorManipulator';
 import { withPlugins } from '../extend/withPlugins';
-import { getWorkspace } from '../state/selectors';
+import { getShowZoomControlsConfig, getWorkspace } from '../state/selectors';
 import { WindowCanvasNavigationControls } from '../components/WindowCanvasNavigationControls';
 
 /** */
 const mapStateToProps = (state, { windowId }) => ({
+  showZoomControls: getShowZoomControlsConfig(state),
   visible: getWorkspace(state).focusedWindowId === windowId,
 });
 
-/**
- *
- * @param theme
- */
-const styles = theme => ({
-  canvasNav: {
-    cursor: 'default',
-    display: 'flex',
-    flexDirection: 'row',
-    flexWrap: 'wrap',
-    justifyContent: 'center',
-    textAlign: 'center',
-  },
-  canvasNavStacked: {
-    flexDirection: 'column',
-  },
-  controls: {
-    backgroundColor: alpha(theme.palette.background.paper, 0.5),
-    bottom: 0,
-    position: 'absolute',
-    width: '100%',
-    zIndex: 50,
-  },
-});
-
 const enhance = compose(
   connect(mapStateToProps),
-  withStyles(styles),
   withSize(),
   withPlugins('WindowCanvasNavigationControls'),
 );
diff --git a/src/containers/WindowListButton.js b/src/containers/WindowListButton.js
index 74506d9dbaec7e421ae6288d6eb93e64210aeaec..68e984e497f7addbed5b701fcc07ec7499cae9bf 100644
--- a/src/containers/WindowListButton.js
+++ b/src/containers/WindowListButton.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { getWindowIds, getWorkspace } from '../state/selectors';
 import { WindowListButton } from '../components/WindowListButton';
@@ -12,26 +11,8 @@ const mapStateToProps = (state) => ({
   windowCount: getWindowIds(state).length,
 });
 
-/**
- *
- * @param theme
- * @returns {{background: {background: string}}}
- */
-const styles = theme => ({
-  badge: {
-    paddingLeft: 12,
-  },
-  ctrlBtn: {
-    margin: theme.spacing(1),
-  },
-  ctrlBtnSelected: {
-    backgroundColor: theme.palette.action.selected,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('WindowListButton'),
 );
diff --git a/src/containers/WindowSideBar.js b/src/containers/WindowSideBar.js
index 012b2fd5d5cf273865948294150c6bb72c6d8043..022e34f310dbb64f9f64c5492c165314e84c7430 100644
--- a/src/containers/WindowSideBar.js
+++ b/src/containers/WindowSideBar.js
@@ -1,6 +1,5 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import { WindowSideBar } from '../components/WindowSideBar';
@@ -19,32 +18,7 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/**
- *
- * @param theme
- * @returns {{toolbar: CSSProperties | toolbar | {minHeight}, grow: {flexGrow: number},
- * drawer: {overflowX: string, left: number, flexShrink: number, width: number, height: string}}}
- */
-const styles = theme => ({
-  drawer: {
-    flexShrink: 0,
-    height: '100%',
-    order: -1000,
-    zIndex: theme.zIndex.appBar - 1,
-  },
-  grow: {
-    flexGrow: 1,
-  },
-  paper: {
-    borderInlineEnd: `1px solid ${theme.palette.divider}`,
-    overflowX: 'hidden',
-    width: 48,
-  },
-  toolbar: theme.mixins.toolbar,
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, null),
   withPlugins('WindowSideBar'),
diff --git a/src/containers/WindowSideBarAnnotationsPanel.js b/src/containers/WindowSideBarAnnotationsPanel.js
index 904cea2f93053bcbca8d2c0903931f934ccc6e9d..8370a0c375692dbbf729443fe0607ebab4de0621 100644
--- a/src/containers/WindowSideBarAnnotationsPanel.js
+++ b/src/containers/WindowSideBarAnnotationsPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import {
   getVisibleCanvasIds,
@@ -22,20 +21,8 @@ const mapStateToProps = (state, { windowId }) => ({
   canvasIds: getVisibleCanvasIds(state, { windowId }),
 });
 
-/** */
-const styles = theme => ({
-  section: {
-    borderBottom: `.5px solid ${theme.palette.section_divider}`,
-    paddingBottom: theme.spacing(1),
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('WindowSideBarAnnotationsPanel'),
   // further HOC
diff --git a/src/containers/WindowSideBarButtons.js b/src/containers/WindowSideBarButtons.js
index 8449e9b6a28c6a0c2067618a8933da7fa888aa68..3b0511ae0fa9c694439f68416106ef173881b1d4 100644
--- a/src/containers/WindowSideBarButtons.js
+++ b/src/containers/WindowSideBarButtons.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
@@ -65,49 +64,8 @@ const mapStateToProps = (state, { windowId }) => ({
   sideBarPanel: ((getCompanionWindowsForPosition(state, { position: 'left', windowId }))[0] || {}).content,
 });
 
-/** */
-const style = theme => ({
-  badge: {
-    backgroundColor: theme.palette.notification?.main,
-  },
-  tab: {
-    '&:active': {
-      backgroundColor: theme.palette.action?.active,
-    },
-    '&:focus': {
-      '@media (hover: none)': {
-        backgroundColor: 'transparent',
-      },
-      backgroundColor: theme.palette.action?.hover,
-      textDecoration: 'none',
-      // Reset on touch devices, it doesn't add specificity
-    },
-    '&:hover': {
-      '@media (hover: none)': {
-        backgroundColor: 'transparent',
-      },
-      backgroundColor: theme.palette.action?.hover,
-      textDecoration: 'none',
-      // Reset on touch devices, it doesn't add specificity
-    },
-
-    borderRight: '2px solid transparent',
-    minWidth: 'auto',
-  },
-  tabSelected: {
-    borderRight: `2px solid ${theme.palette.primary?.main}`,
-  },
-  tabsFlexContainer: {
-    flexDirection: 'column',
-  },
-  tabsIndicator: {
-    display: 'none',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(style),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WindowSideBarButtons'),
 );
diff --git a/src/containers/WindowSideBarCanvasPanel.js b/src/containers/WindowSideBarCanvasPanel.js
index 24324d253865b996f88b608e36c67245c11a0f36..e6b4befc734f459a96201631e3d1be18cb0b94cb 100644
--- a/src/containers/WindowSideBarCanvasPanel.js
+++ b/src/containers/WindowSideBarCanvasPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { WindowSideBarCanvasPanel } from '../components/WindowSideBarCanvasPanel';
@@ -53,37 +52,8 @@ const mapDispatchToProps = (dispatch, { id, windowId }) => ({
   ),
 });
 
-/**
- *
- * @param theme
- */
-const styles = theme => ({
-  break: {
-    flexBasis: '100%',
-    height: 0,
-  },
-  collectionNavigationButton: {
-    textTransform: 'none',
-  },
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-  select: {
-    '&:focus': {
-      backgroundColor: theme.palette.background.paper,
-    },
-  },
-  selectEmpty: {
-    backgroundColor: theme.palette.background.paper,
-  },
-  variantTab: {
-    minWidth: 'auto',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WindowSideBarCanvasPanel'),
 );
diff --git a/src/containers/WindowSideBarCollectionPanel.js b/src/containers/WindowSideBarCollectionPanel.js
index 32b31a77b283f90756a568cffe44be7282f31d5b..738c11125a4d23e87226f674a85ee2423d71603c 100644
--- a/src/containers/WindowSideBarCollectionPanel.js
+++ b/src/containers/WindowSideBarCollectionPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import {
@@ -56,22 +55,7 @@ const mapDispatchToProps = (dispatch, { id, windowId }) => ({
   updateWindow: (...args) => dispatch(actions.updateWindow(windowId, ...args)),
 });
 
-/**
- * Styles for withStyles HOC
- */
-const styles = theme => ({
-  label: {
-    paddingLeft: theme.spacing(1),
-  },
-  menuItem: {
-    borderBottom: `0.5px solid ${theme.palette.divider}`,
-    paddingRight: theme.spacing(1),
-    whiteSpace: 'normal',
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WindowSideBarCollectionPanel'),
diff --git a/src/containers/WindowSideBarInfoPanel.js b/src/containers/WindowSideBarInfoPanel.js
index 9fc498ce117521c64e176a2559bebd8c8b40692d..19de86776b88a5277762ce3e4b8c74eb95cf9e86 100644
--- a/src/containers/WindowSideBarInfoPanel.js
+++ b/src/containers/WindowSideBarInfoPanel.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import {
@@ -33,24 +32,8 @@ const mapDispatchToProps = (dispatch, { windowId, id }) => ({
   setLocale: locale => dispatch(actions.updateCompanionWindow(windowId, id, { locale })),
 });
 
-/**
- *
- * @param theme
- * @returns {label: {paddingLeft: number}}}
- */
-const styles = theme => ({
-  section: {
-    borderBottom: `.5px solid ${theme.palette.section_divider}`,
-    paddingBottom: theme.spacing(1),
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(1),
-    paddingTop: theme.spacing(2),
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WindowSideBarInfoPanel'),
 );
diff --git a/src/containers/WindowThumbnailSettings.js b/src/containers/WindowThumbnailSettings.js
index 24878457a780b19e3e280e0e289179636d0fb28b..d773247feef8d32d57fcaa1835dba73c3eb9dff3 100644
--- a/src/containers/WindowThumbnailSettings.js
+++ b/src/containers/WindowThumbnailSettings.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { getThumbnailNavigationPosition, getThemeDirection } from '../state/selectors';
@@ -26,22 +25,7 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/** */
-const styles = theme => ({
-  label: {
-    borderBottom: '2px solid transparent',
-  },
-  MenuItem: {
-    display: 'inline-block',
-  },
-  selectedLabel: {
-    borderBottom: `2px solid ${theme.palette.secondary.main}`,
-    color: theme.palette.secondary.main,
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(null, { withRef: true }),
   connect(mapStateToProps, mapDispatchToProps, null, { forwardRef: true }),
   withPlugins('WindowThumbnailSettings'),
diff --git a/src/containers/WindowTopBar.js b/src/containers/WindowTopBar.js
index 27dd9b83248c59267293ebb9b86512ef3b8a13e1..ae9fc34b90eef656931403890f734f980993ac6b 100644
--- a/src/containers/WindowTopBar.js
+++ b/src/containers/WindowTopBar.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { getWindowConfig, isFocused } from '../state/selectors';
@@ -35,31 +34,8 @@ const mapDispatchToProps = (dispatch, { windowId }) => ({
   toggleWindowSideBar: () => dispatch(actions.toggleWindowSideBar(windowId)),
 });
 
-/**
- * @param theme
- * @returns {{typographyBody: {flexGrow: number, fontSize: number|string},
- * windowTopBarStyle: {minHeight: number, paddingLeft: number, backgroundColor: string}}}
- */
-const styles = theme => ({
-  focused: {},
-  windowTopBarStyle: {
-    '&$focused': {
-      borderTop: `2px solid ${theme.palette.primary.main}`,
-    },
-    backgroundColor: theme.palette.shades?.main,
-    borderTop: '2px solid transparent',
-    minHeight: 32,
-    paddingLeft: theme.spacing(0.5),
-    paddingRight: theme.spacing(0.5),
-  },
-  windowTopBarStyleDraggable: {
-    cursor: 'move',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WindowTopBar'),
 );
diff --git a/src/containers/WindowTopBarPluginArea.js b/src/containers/WindowTopBarPluginArea.js
index 4bcdc88f5ae95da56b721b52d856012ded348c0a..cf05b64af3e923718c58de9d28f952c2e8147d50 100644
--- a/src/containers/WindowTopBarPluginArea.js
+++ b/src/containers/WindowTopBarPluginArea.js
@@ -1,17 +1,11 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WindowTopBarPluginArea } from '../components/WindowTopBarPluginArea';
 
-/**
- */
-const styles = {};
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(null, null),
   withPlugins('WindowTopBarPluginArea'),
 );
diff --git a/src/containers/WindowTopBarPluginMenu.js b/src/containers/WindowTopBarPluginMenu.js
index 510d7a35de8418cb7c831e145e9d5178cc0092e2..4900528e9f5cd7ca363f978ced229ed8c46bff41 100644
--- a/src/containers/WindowTopBarPluginMenu.js
+++ b/src/containers/WindowTopBarPluginMenu.js
@@ -1,25 +1,12 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WindowTopBarPluginMenu } from '../components/WindowTopBarPluginMenu';
 import { withWorkspaceContext } from '../contexts/WorkspaceContext';
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)}}}
- */
-const styles = theme => ({
-  ctrlBtnSelected: {
-    backgroundColor: theme.palette.action.selected,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
   withWorkspaceContext,
-  withStyles(styles),
   withPlugins('WindowTopBarPluginMenu'),
 );
 
diff --git a/src/containers/WindowTopBarTitle.js b/src/containers/WindowTopBarTitle.js
index 5cb51b8274b7ffc7b1bca026c5b5fe5cfea0207f..fbdf31288be412e6bddec49981aa5a6a0a4838d3 100644
--- a/src/containers/WindowTopBarTitle.js
+++ b/src/containers/WindowTopBarTitle.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { getManifestStatus, getManifestTitle, getWindowConfig } from '../state/selectors';
 import { WindowTopBarTitle } from '../components/WindowTopBarTitle';
@@ -14,20 +13,8 @@ const mapStateToProps = (state, { windowId }) => ({
   manifestTitle: getManifestTitle(state, { windowId }),
 });
 
-/**
- * @param theme
- */
-const styles = theme => ({
-  title: {
-    ...theme.typography.h6,
-    flexGrow: 1,
-    paddingLeft: theme.spacing(0.5),
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, null),
   withPlugins('WindowTopBarTitle'),
 );
diff --git a/src/containers/WindowTopMenuButton.js b/src/containers/WindowTopMenuButton.js
index 54d863e2b688ca032ac5ed7af9e093e176b7fc18..8a114a8dea2d3d6f7c2b455ab27f820b8f97653d 100644
--- a/src/containers/WindowTopMenuButton.js
+++ b/src/containers/WindowTopMenuButton.js
@@ -1,23 +1,10 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WindowTopMenuButton } from '../components/WindowTopMenuButton';
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)}}}
- */
-const styles = theme => ({
-  ctrlBtnSelected: {
-    backgroundColor: theme.palette.action.selected,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withPlugins('WindowTopMenuButton'),
 );
 
diff --git a/src/containers/WindowViewSettings.js b/src/containers/WindowViewSettings.js
index 6fc417f284b1f5b585c85ec369ddbd31946ece8a..3e34e5949b1ede7323c50648a8699cd3f111dfe7 100644
--- a/src/containers/WindowViewSettings.js
+++ b/src/containers/WindowViewSettings.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { getAllowedWindowViewTypes, getWindowViewType } from '../state/selectors';
@@ -26,22 +25,7 @@ const mapStateToProps = (state, { windowId }) => (
   }
 );
 
-/** */
-const styles = theme => ({
-  label: {
-    borderBottom: '2px solid transparent',
-  },
-  MenuItem: {
-    display: 'inline-block',
-  },
-  selectedLabel: {
-    borderBottom: `2px solid ${theme.palette.secondary.main}`,
-    color: theme.palette.secondary.main,
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(null, { withRef: true }),
   connect(mapStateToProps, mapDispatchToProps, null, { forwardRef: true }),
   withPlugins('WindowViewSettings'),
diff --git a/src/containers/Workspace.js b/src/containers/Workspace.js
index 6eac59013b39931e67ce89e965cb58bfa3678c7b..f9288e8d94c750a2e8bcafb881abbbea7779bf55 100644
--- a/src/containers/Workspace.js
+++ b/src/containers/Workspace.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { Workspace } from '../components/Workspace';
 import {
@@ -18,7 +17,6 @@ import * as actions from '../state/actions';
 const mapStateToProps = state => (
   {
     allowNewWindows: getConfig(state).workspace.allowNewWindows,
-    isWorkspaceControlPanelVisible: getConfig(state).workspaceControlPanel.enabled,
     maximizedWindowIds: getMaximizedWindowsIds(state),
     windowIds: getWindowIds(state),
     workspaceId: getWorkspace(state).id,
@@ -35,35 +33,8 @@ const mapDispatchToProps = {
   addWindow: actions.addWindow,
 };
 
-/**
- * @param theme
- */
-const styles = theme => ({
-  workspaceViewport: {
-    bottom: 0,
-    left: 0,
-    margin: 0,
-    overflow: 'hidden',
-    position: 'absolute',
-    right: 0,
-    top: 0,
-  },
-  workspaceWithControlPanel: {
-    paddingTop: 74,
-  },
-  // injection order matters here
-  // eslint-disable-next-line sort-keys
-  '@media (min-width: 600px)': {
-    workspaceWithControlPanel: {
-      paddingLeft: 68,
-      paddingTop: 0,
-    },
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('Workspace'),
   // further HOC go here
diff --git a/src/containers/WorkspaceAdd.js b/src/containers/WorkspaceAdd.js
index 25f8903234c26197b3b0746936db93a93ef8eda6..0c0883945d1b04b1e8ea07aa1e822f5486ac15b9 100644
--- a/src/containers/WorkspaceAdd.js
+++ b/src/containers/WorkspaceAdd.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { WorkspaceAdd } from '../components/WorkspaceAdd';
@@ -24,73 +23,8 @@ const mapDispatchToProps = {
   setWorkspaceAddVisibility: actions.setWorkspaceAddVisibility,
 };
 
-/**
- *
- * @param theme
- * @returns {{typographyBody: {flexGrow: number, fontSize: string},
- * form: {paddingBottom: number, paddingTop: number, marginTop: number},
- * fab: {bottom: number, position: string, right: number},
- * menuButton: {marginRight: number, marginLeft: number}}}
- */
-const styles = theme => ({
-  displayNone: {
-    display: 'none',
-  },
-  fab: {
-    bottom: theme.spacing(2),
-    position: 'absolute',
-    right: theme.spacing(2),
-  },
-  form: {
-    left: '0',
-    marginTop: 48,
-    paddingBottom: theme.spacing(2),
-    paddingLeft: theme.spacing(2),
-    paddingRight: theme.spacing(2),
-    [theme.breakpoints.up('sm')]: {
-      paddingLeft: theme.spacing(3),
-      paddingRight: theme.spacing(3),
-    },
-    paddingTop: theme.spacing(2),
-    right: '0',
-  },
-  list: {
-    margin: '16px',
-  },
-  menuButton: {
-    marginLeft: -12,
-    marginRight: 20,
-  },
-  paper: {
-    borderTop: '0',
-    left: '0',
-    [theme.breakpoints.up('sm')]: {
-      left: '65px',
-    },
-  },
-  typographyBody: {
-    flexGrow: 1,
-  },
-  workspaceAdd: {
-    boxSizing: 'border-box',
-    height: '100%',
-    overflowX: 'hidden',
-    overflowY: 'auto',
-    paddingTop: 68,
-  },
-  // injection order matters
-  // eslint-disable-next-line sort-keys
-  '@media (min-width: 600px)': {
-    workspaceAdd: {
-      paddingLeft: 68,
-      paddingTop: 0,
-    },
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WorkspaceAdd'),
 );
diff --git a/src/containers/WorkspaceAddButton.js b/src/containers/WorkspaceAddButton.js
index 427676b481fa75442a6f9ae9eba09eba199e7151..cec63fcb4a2e3b12701ff83936e7a44349af6e4f 100644
--- a/src/containers/WorkspaceAddButton.js
+++ b/src/containers/WorkspaceAddButton.js
@@ -1,13 +1,37 @@
 import { connect } from 'react-redux';
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
-import withWidth from '@material-ui/core/withWidth';
+import { useTheme } from '@mui/material/styles';
+import useMediaQuery from '@mui/material/useMediaQuery';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import { getWindowIds, getWorkspace } from '../state/selectors';
 import { WorkspaceAddButton } from '../components/WorkspaceAddButton';
 
+/**
+ * Be careful using this hook. It only works because the number of
+ * breakpoints in theme is static. It will break once you change the number of
+ * breakpoints. See https://legacy.reactjs.org/docs/hooks-rules.html#only-call-hooks-at-the-top-level
+ */
+function useWidth() {
+  const theme = useTheme();
+  const keys = [...theme.breakpoints.keys].reverse();
+  return (
+    keys.reduce((output, key) => {
+      // eslint-disable-next-line react-hooks/rules-of-hooks
+      const matches = useMediaQuery(theme.breakpoints.up(key));
+      return !output && matches ? key : output;
+    }, null) || 'xs'
+  );
+}
+
+/**
+ * withWidth
+ * @memberof WorkspaceControlPanel
+ * @private
+ */
+const withWidth = () => (WrappedComponent) => (props) => <WrappedComponent {...props} width={useWidth()} />;
+
 /**
  * mapStateToProps - to hook up connect
  * @memberof WorkspaceControlPanel
@@ -32,30 +56,8 @@ const mapStateToProps = (state, { width }) => {
  */
 const mapDispatchToProps = { setWorkspaceAddVisibility: actions.setWorkspaceAddVisibility };
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)}}}
- */
-const styles = theme => ({
-  fab: {
-    margin: theme.spacing(1),
-  },
-  fabPrimary: {
-    '&:focus': {
-      backgroundColor: theme.palette.primary.dark,
-    },
-  },
-  fabSecondary: {
-    '&:focus': {
-      backgroundColor: theme.palette.secondary.dark,
-    },
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withWidth({ initialWidth: 'xs' }),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WorkspaceAddButton'),
diff --git a/src/containers/WorkspaceArea.js b/src/containers/WorkspaceArea.js
index c2655978c98e9a934c485eccbf5984192b2aa31a..f58fa21317a454ecbeba92513635bb76dc634b74 100644
--- a/src/containers/WorkspaceArea.js
+++ b/src/containers/WorkspaceArea.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles, lighten, darken } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceArea } from '../components/WorkspaceArea';
 import { getConfig, getWindowIds, getWorkspace } from '../state/selectors';
@@ -20,30 +19,8 @@ const mapStateToProps = state => (
   }
 );
 
-/**
- *
- * @param theme
- * @returns {{background: {background: string}}}
- */
-const styles = (theme) => {
-  const getBackgroundColor = theme.palette.type === 'light' ? darken : lighten;
-
-  return {
-    viewer: {
-      background: getBackgroundColor(theme.palette.shades.light, 0.1),
-      bottom: 0,
-      left: 0,
-      overflow: 'hidden',
-      position: 'absolute',
-      right: 0,
-      top: 0,
-    },
-  };
-};
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps),
   withPlugins('WorkspaceArea'),
 );
diff --git a/src/containers/WorkspaceControlPanel.js b/src/containers/WorkspaceControlPanel.js
index 88f482e91d3a8cb92d78770c855eae6e7bda8907..e23bb417a950d24c6d7cb7bd6ec5bd690758e92b 100644
--- a/src/containers/WorkspaceControlPanel.js
+++ b/src/containers/WorkspaceControlPanel.js
@@ -1,69 +1,10 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceControlPanel } from '../components/WorkspaceControlPanel';
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)},
- * drawer: {overflowX: string, height: string}}}
- */
-const styles = theme => ({
-  branding: {
-    display: 'flex',
-    position: 'absolute',
-    [theme.breakpoints.up('xs')]: {
-      display: 'none',
-    },
-    [theme.breakpoints.up('sm')]: {
-      bottom: 0,
-      display: 'block',
-      float: 'none',
-      right: 'auto',
-      width: '100%',
-    },
-    right: 0,
-  },
-  ctrlBtn: {
-    margin: theme.spacing(1),
-  },
-  drawer: {
-    overflowX: 'hidden',
-  },
-  root: {
-    height: 64,
-    [theme.breakpoints.up('sm')]: {
-      height: '100%',
-      left: 0,
-      right: 'auto',
-      width: 64,
-    },
-  },
-  toolbar: {
-    display: 'flex',
-    justifyContent: 'space-between',
-    [theme.breakpoints.up('sm')]: {
-      flexDirection: 'column',
-      justifyContent: 'flex-start',
-      minHeight: 0,
-    },
-  },
-  wide: {
-    width: 'auto',
-  },
-  workspaceButtons: {
-    [theme.breakpoints.up('sm')]: {
-      display: 'flex',
-      flexDirection: 'column',
-    },
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withPlugins('WorkspaceControlPanel'),
   // further HOC go here
 );
diff --git a/src/containers/WorkspaceControlPanelButtons.js b/src/containers/WorkspaceControlPanelButtons.js
index eaef08bd775196782632a0bb6022759aa41b3d57..9d500357b4190a08d0a7b3a1a278fc067c06beb2 100644
--- a/src/containers/WorkspaceControlPanelButtons.js
+++ b/src/containers/WorkspaceControlPanelButtons.js
@@ -1,21 +1,8 @@
 import { compose } from 'redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceControlPanelButtons } from '../components/WorkspaceControlPanelButtons';
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)}}}
- */
-const styles = theme => ({
-  ctrlBtn: {
-    margin: theme.spacing(1),
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withPlugins('WorkspaceControlPanelButtons'),
 );
 
diff --git a/src/containers/WorkspaceElastic.js b/src/containers/WorkspaceElastic.js
index 9badc4614bbc259c42c3f64d6ca1654b37343fcc..5cc72f575697d02c3a32bdb5ddb7394ad1ffd505 100644
--- a/src/containers/WorkspaceElastic.js
+++ b/src/containers/WorkspaceElastic.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
 import WorkspaceElastic from '../components/WorkspaceElastic';
@@ -44,22 +43,7 @@ const mapDispatchToProps = (dispatch, props) => ({
   },
 });
 
-const styles = {
-  workspace: {
-    boxSizing: 'border-box',
-    margin: 0,
-    position: 'absolute',
-    transitionDuration: '.7s',
-    // order matters
-    // eslint-disable-next-line sort-keys
-    '& .react-draggable-dragging': {
-      transitionDuration: 'unset',
-    },
-  },
-};
-
 const enhance = compose(
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WorkspaceElastic'),
   // further HOC go here
diff --git a/src/containers/WorkspaceElasticWindow.js b/src/containers/WorkspaceElasticWindow.js
index 1c071396f719be6aafabf396f8611b0bbab612c2..7e6b5ccac2ff2756781fd1a66208580554442e78 100644
--- a/src/containers/WorkspaceElasticWindow.js
+++ b/src/containers/WorkspaceElasticWindow.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core';
 import * as actions from '../state/actions';
 import WorkspaceElasticWindow from '../components/WorkspaceElasticWindow';
 import {
@@ -36,17 +35,7 @@ const mapDispatchToProps = (dispatch, props) => ({
   },
 });
 
-/**
- * @param theme
- */
-const styles = theme => ({
-  focused: {
-    zIndex: theme.zIndex.modal - 1,
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   // further HOC go here
 );
diff --git a/src/containers/WorkspaceExport.js b/src/containers/WorkspaceExport.js
index e1ac982c10aa03c0cff9451dd1c3fbb63bac98ab..069a52542815f19231609bdfe7e0f81997efbff7 100644
--- a/src/containers/WorkspaceExport.js
+++ b/src/containers/WorkspaceExport.js
@@ -1,6 +1,5 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withTranslation } from 'react-i18next';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceExport } from '../components/WorkspaceExport';
@@ -17,18 +16,8 @@ const mapStateToProps = state => ({
   exportableState: getExportableState(state),
 });
 
-/**
- * Styles for the withStyles HOC
- */
-const styles = theme => ({
-  accordionTitle: {
-    padding: 0,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, {}),
   withPlugins('WorkspaceExport'),
 );
diff --git a/src/containers/WorkspaceImport.js b/src/containers/WorkspaceImport.js
index 01c22d154776e0dff666ef3a28af1df9f5daaf1a..c529bca89a68af3d2f688ed4f4617670cf141076 100644
--- a/src/containers/WorkspaceImport.js
+++ b/src/containers/WorkspaceImport.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceImport } from '../components/WorkspaceImport';
 import * as actions from '../state/actions';
@@ -16,22 +15,8 @@ const mapDispatchToProps = {
   importConfig: actions.importMiradorState,
 };
 
-/** */
-const styles = theme => ({
-  cancelBtn: {
-    color: theme.palette.text.primary,
-  },
-  textField: {
-    width: '100%',
-  },
-  textInput: {
-    fontFamily: 'monospace',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(null, mapDispatchToProps),
   withPlugins('WorkspaceImport'),
 );
diff --git a/src/containers/WorkspaceMenuButton.js b/src/containers/WorkspaceMenuButton.js
index 16e10d569baeeb4c63666db50e14fddf26cbeca5..b5f2518eb255db4cf81d47c6ab1b68d9ed602931 100644
--- a/src/containers/WorkspaceMenuButton.js
+++ b/src/containers/WorkspaceMenuButton.js
@@ -1,26 +1,10 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceMenuButton } from '../components/WorkspaceMenuButton';
 
-/**
- *
- * @param theme
- * @returns {{ctrlBtn: {margin: (number|string)}}}
- */
-const styles = theme => ({
-  ctrlBtn: {
-    margin: theme.spacing(1),
-  },
-  ctrlBtnSelected: {
-    backgroundColor: theme.palette.action.selected,
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   withPlugins('WorkspaceMenuButton'),
   // further HOC
 );
diff --git a/src/containers/WorkspaceMosaic.js b/src/containers/WorkspaceMosaic.js
index b66e2f4aa67fd4741712f7d9e131b052e693b9f6..4ab6529d059266edca949428ed7c0ff7a6c2b509 100644
--- a/src/containers/WorkspaceMosaic.js
+++ b/src/containers/WorkspaceMosaic.js
@@ -1,11 +1,9 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { getWorkspace } from '../state/selectors';
 import * as actions from '../state/actions';
 import { WorkspaceMosaic } from '../components/WorkspaceMosaic';
-import globalReactMosaicStyles from '../styles/react-mosaic-component';
 
 /**
  * mapStateToProps - to hook up connect
@@ -27,26 +25,7 @@ const mapStateToProps = state => (
  */
 const mapDispatchToProps = { updateWorkspaceMosaicLayout: actions.updateWorkspaceMosaicLayout };
 
-const styles = {
-  root: {
-    '& .mosaic-preview': {
-      boxShadow: 'none',
-    },
-    '& .mosaic-tile': {
-      boxShadow: '0 1px 3px 0 rgba(0, 0, 0, .2), 0 1px 1px 0 rgba(0, 0, 0, .2), 0 2px 1px -1px rgba(0, 0, 0, .2)',
-    },
-    '& .mosaic-window': {
-      boxShadow: 'none',
-    },
-    '& .mosaic-window-toolbar': {
-      display: 'none !important',
-    },
-  },
-  ...globalReactMosaicStyles,
-};
-
 const enhance = compose(
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WorkspaceMosaic'),
   // further HOC go here
diff --git a/src/containers/WorkspaceOptionsButton.js b/src/containers/WorkspaceOptionsButton.js
index f17b51310f56018393b23b6593ff87da902f2857..5768c4cd2aae646aef2e586106f357cd3856fd59 100644
--- a/src/containers/WorkspaceOptionsButton.js
+++ b/src/containers/WorkspaceOptionsButton.js
@@ -1,24 +1,9 @@
 import { compose } from 'redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core/styles';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceOptionsButton } from '../components/WorkspaceOptionsButton';
 
-/**
- *
- * @param theme
- */
-const styles = theme => ({
-  ctrlBtn: {
-    margin: theme.spacing(1),
-  },
-  ctrlBtnSelected: {
-    backgroundColor: theme.palette.action.selected,
-  },
-});
-
 const enhance = compose(
-  withStyles(styles),
   withTranslation(),
   withPlugins('WorkspaceOptionsButton'),
 );
diff --git a/src/containers/WorkspaceSelectionDialog.js b/src/containers/WorkspaceSelectionDialog.js
index d14881af77bf8f0a0a894d50cf82b821910c037f..3305e5321672755ccd59391bc0d61d330e5b17a5 100644
--- a/src/containers/WorkspaceSelectionDialog.js
+++ b/src/containers/WorkspaceSelectionDialog.js
@@ -1,7 +1,6 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import { WorkspaceSelectionDialog } from '../components/WorkspaceSelectionDialog';
 import * as actions from '../state/actions';
@@ -23,60 +22,8 @@ const mapDispatchToProps = {
  */
 const mapStateToProps = state => ({ workspaceType: getWorkspaceType(state) });
 
-/** */
-const styles = theme => ({
-  card: {
-    backgroundColor: 'transparent',
-    borderRadius: '0',
-    boxShadow: '0 0 transparent',
-    display: 'flex',
-  },
-  content: {
-    flex: '1 0 auto',
-  },
-  details: {
-    display: 'flex',
-    flexDirection: 'column',
-  },
-  headline: {
-    paddingBottom: '6px',
-  },
-  list: {
-    '&active': {
-      outline: 'none',
-    },
-    '&focus': {
-      outline: 'none',
-    },
-    outline: 'none',
-  },
-  media: {
-    flex: '0 0 120px',
-    height: '90px',
-  },
-  menuItem: {
-    height: 'auto',
-    overflow: 'auto',
-    whiteSpace: 'inherit',
-  },
-  root: {
-    '&:last-child': {
-      paddingBottom: '12px',
-    },
-    paddingBottom: 0,
-    paddingTop: 0,
-    textAlign: 'left',
-  },
-  svgIcon: {
-    flexShrink: 0,
-    height: '90px',
-    width: '120px',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('WorkspaceSelectionDialog'),
 );
diff --git a/src/containers/ZoomControls.js b/src/containers/ZoomControls.js
index 512ce860cbef74cb3534084c3d8ba9d0e06db0bc..1cc85d545509720db14d9af2c07a189a5e10b7a6 100644
--- a/src/containers/ZoomControls.js
+++ b/src/containers/ZoomControls.js
@@ -1,10 +1,9 @@
 import { compose } from 'redux';
 import { connect } from 'react-redux';
 import { withTranslation } from 'react-i18next';
-import { withStyles } from '@material-ui/core';
 import { withPlugins } from '../extend/withPlugins';
 import * as actions from '../state/actions';
-import { getShowZoomControlsConfig, getViewer } from '../state/selectors';
+import { getViewer } from '../state/selectors';
 import { ZoomControls } from '../components/ZoomControls';
 
 /**
@@ -14,7 +13,6 @@ import { ZoomControls } from '../components/ZoomControls';
  */
 const mapStateToProps = (state, { windowId }) => (
   {
-    showZoomControls: getShowZoomControlsConfig(state),
     viewer: getViewer(state, { windowId }),
   }
 );
@@ -26,33 +24,8 @@ const mapStateToProps = (state, { windowId }) => (
  */
 const mapDispatchToProps = { updateViewport: actions.updateViewport };
 
-/**
- *
- * @param theme
- * @returns {{zoom_controls: {position: string, right: number},
- * ListItem: {paddingBottom: number, paddingTop: number}}}
- */
-const styles = theme => ({
-  divider: {
-    borderRight: '1px solid #808080',
-    display: 'inline-block',
-    height: '24px',
-    margin: '12px 6px',
-  },
-  ListItem: {
-    paddingBottom: 0,
-    paddingTop: 0,
-  },
-  zoom_controls: {
-    display: 'flex',
-    flexDirection: 'row',
-    justifyContent: 'center',
-  },
-});
-
 const enhance = compose(
   withTranslation(),
-  withStyles(styles),
   connect(mapStateToProps, mapDispatchToProps),
   withPlugins('ZoomControls'),
 );
diff --git a/src/extend/pluginMapping.js b/src/extend/pluginMapping.js
index efc449252abff501d89a04062516592fe7c2dd9b..7ddeefd6bb9c6e88f9cb1134abb0f66dffef05c6 100644
--- a/src/extend/pluginMapping.js
+++ b/src/extend/pluginMapping.js
@@ -18,7 +18,7 @@ import CompanionWindowRegistry from '../lib/CompanionWindowRegistry';
  */
 export function createTargetToPluginMapping(plugins) {
   return plugins.reduce((map, plugin) => (
-    update(map, [plugin.target, plugin.mode], x => [...x || [], plugin])
+    update(map, [plugin.target, plugin.mode], x => [...(x || []), plugin])
   ), {});
 }
 
diff --git a/src/state/createStore.js b/src/state/createStore.js
index 9467f322c5535aab6d73abcd48d0aaed5b3dc9a1..c115859260c6db7cd5398b22b308bf9572cf5b32 100644
--- a/src/state/createStore.js
+++ b/src/state/createStore.js
@@ -6,7 +6,7 @@
 import thunkMiddleware from 'redux-thunk';
 import createSagaMiddleware from 'redux-saga';
 import { combineReducers, createStore, applyMiddleware } from 'redux';
-import { composeWithDevTools } from 'redux-devtools-extension';
+import { composeWithDevTools } from '@redux-devtools/extension';
 import createRootReducer from './reducers/rootReducer';
 import getRootSaga from './sagas';
 import settings from '../config/settings';
diff --git a/src/styles/react-mosaic-component.js b/src/styles/react-mosaic-component.js
index 23d46a4fb2c67f5531ab4c4e4a2fc01c3c464110..2f6885ea1208517f1943c303366875083a3d4bfd 100644
--- a/src/styles/react-mosaic-component.js
+++ b/src/styles/react-mosaic-component.js
@@ -1,285 +1,283 @@
 // Generated using jss convert node_modules/react-mosaic-component/react-mosaic-component.css > src/styles/react-mosaic-component.js
 // .mosaic-blueprint-theme styles have also been removed
 const globalReactMosaicStyles = {
-  '@global': {
-    '.mosaic': {
-      height: '100%',
-      width: '100%',
-    },
-    '.mosaic, .mosaic > *': {
-      boxSizing: 'border-box',
-    },
-    '.mosaic .mosaic-zero-state': {
-      position: 'absolute',
-      top: 6,
-      right: 6,
-      bottom: 6,
-      left: 6,
-      width: 'auto',
-      height: 'auto',
-      zIndex: '1',
-    },
-    '.mosaic-root': {
-      position: 'absolute',
-      top: 3,
-      right: 3,
-      bottom: 3,
-      left: 3,
-    },
-    '.mosaic-split': {
-      position: 'absolute',
-      zIndex: '1',
-      touchAction: 'none',
-    },
-    '.mosaic-split:hover': {
-      background: 'black',
-    },
-    '.mosaic-split .mosaic-split-line': {
-      position: 'absolute',
-    },
-    '.mosaic-split.-row': {
-      marginLeft: -3,
-      width: 6,
-      cursor: 'ew-resize',
-    },
-    '.mosaic-split.-row .mosaic-split-line': {
-      top: '0',
-      bottom: '0',
-      left: 3,
-      right: 3,
-    },
-    '.mosaic-split.-column': {
-      marginTop: -3,
-      height: 6,
-      cursor: 'ns-resize',
-    },
-    '.mosaic-split.-column .mosaic-split-line': {
-      top: 3,
-      bottom: 3,
-      left: '0',
-      right: '0',
-    },
-    '.mosaic-tile': {
-      position: 'absolute',
-      margin: 3,
-    },
-    '.mosaic-tile > *': {
-      height: '100%',
-      width: '100%',
-    },
-    '.mosaic-drop-target': {
-      position: 'relative',
-    },
-    '.mosaic-drop-target.drop-target-hover .drop-target-container': {
-      display: 'block',
-    },
-    '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.left': {
-      right: 'calc(100% -  10px )',
-    },
-    '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.right': {
-      left: 'calc(100% -  10px )',
-    },
-    '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.bottom': {
-      top: 'calc(100% -  10px )',
-    },
-    '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.top': {
-      bottom: 'calc(100% -  10px )',
-    },
-    '.mosaic-drop-target .drop-target-container': {
-      position: 'absolute',
-      top: '0',
-      right: '0',
-      bottom: '0',
-      left: '0',
-      display: 'none',
-    },
-    '.mosaic-drop-target .drop-target-container.-dragging': {
-      display: 'block',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target': {
-      position: 'absolute',
-      top: '0',
-      right: '0',
-      bottom: '0',
-      left: '0',
-      background: 'rgba(0, 0, 0, 0.2)',
-      border: '2px solid black',
-      opacity: '0',
-      zIndex: '5',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.left': {
-      right: 'calc(100% -  30% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.right': {
-      left: 'calc(100% -  30% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.bottom': {
-      top: 'calc(100% -  30% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.top': {
-      bottom: 'calc(100% -  30% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover': {
-      opacity: '1',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.left': {
-      right: 'calc(100% -  50% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.right': {
-      left: 'calc(100% -  50% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.bottom': {
-      top: 'calc(100% -  50% )',
-    },
-    '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.top': {
-      bottom: 'calc(100% -  50% )',
-    },
-    '.mosaic-window, .mosaic-preview': {
-      position: 'relative',
-      display: 'flex',
-      fallbacks: [
-        {
-          display: '-webkit-box',
-        },
-      ],
-      webkitBoxOrient: 'vertical',
-      webkitBoxDirection: 'normal',
-      flexDirection: 'column',
-      overflow: 'hidden',
-      boxShadow: '0 0 1px rgba(0, 0, 0, 0.2)',
-    },
-    '.mosaic-window .mosaic-window-toolbar, .mosaic-preview .mosaic-window-toolbar': {
-      zIndex: '4',
-      display: 'flex',
-      fallbacks: [
-        {
-          display: '-webkit-box',
-        },
-      ],
-      webkitBoxPack: 'justify',
-      justifyContent: 'space-between',
-      webkitBoxAlign: 'center',
-      alignItems: 'center',
-      flexShrink: '0',
-      height: 30,
-      background: 'white',
-      boxShadow: '0 1px 1px rgba(0, 0, 0, 0.2)',
-    },
-    '.mosaic-window .mosaic-window-toolbar.draggable, .mosaic-preview .mosaic-window-toolbar.draggable': {
-      cursor: 'move',
-    },
-    '.mosaic-window .mosaic-window-title, .mosaic-preview .mosaic-window-title': {
-      paddingLeft: 15,
-      webkitBoxFlex: '1',
-      flex: '1',
-      textOverflow: 'ellipsis',
-      whiteSpace: 'nowrap',
-      overflow: 'hidden',
-      minHeight: 18,
-    },
-    '.mosaic-window .mosaic-window-controls, .mosaic-preview .mosaic-window-controls': {
-      display: 'flex',
-      fallbacks: [
-        {
-          display: '-webkit-box',
-        },
-      ],
-      height: '100%',
-    },
-    '.mosaic-window .mosaic-window-controls .separator, .mosaic-preview .mosaic-window-controls .separator': {
-      height: 20,
-      borderLeft: '1px solid black',
-      margin: '5px 4px',
-    },
-    '.mosaic-window .mosaic-window-body, .mosaic-preview .mosaic-window-body': {
-      position: 'relative',
-      webkitBoxFlex: '1',
-      flex: '1',
-      height: '0',
-      background: 'white',
-      zIndex: '1',
-      overflow: 'hidden',
-    },
-    '.mosaic-window .mosaic-window-additional-actions-bar, .mosaic-preview .mosaic-window-additional-actions-bar': {
-      position: 'absolute',
-      top: 30,
-      right: '0',
-      bottom: 'initial',
-      left: '0',
-      height: '0',
-      overflow: 'hidden',
-      background: 'white',
-      webkitBoxPack: 'end',
-      justifyContent: 'flex-end',
-      display: 'flex',
-      fallbacks: [
-        {
-          display: '-webkit-box',
-        },
-      ],
-      zIndex: '3',
-    },
-    '.mosaic-window .mosaic-window-additional-actions-bar .bp3-button, .mosaic-preview .mosaic-window-additional-actions-bar .bp3-button': {
-      margin: '0',
-    },
-    '.mosaic-window .mosaic-window-additional-actions-bar .bp3-button:after, .mosaic-preview .mosaic-window-additional-actions-bar .bp3-button:after': {
-      display: 'none',
-    },
-    '.mosaic-window .mosaic-window-body-overlay, .mosaic-preview .mosaic-window-body-overlay': {
-      position: 'absolute',
-      top: '0',
-      right: '0',
-      bottom: '0',
-      left: '0',
-      opacity: '0',
-      background: 'white',
-      display: 'none',
-      zIndex: '2',
-    },
-    '.mosaic-window.additional-controls-open .mosaic-window-additional-actions-bar, .mosaic-preview.additional-controls-open .mosaic-window-additional-actions-bar': {
-      height: 30,
-    },
-    '.mosaic-window.additional-controls-open .mosaic-window-body-overlay, .mosaic-preview.additional-controls-open .mosaic-window-body-overlay': {
-      display: 'block',
-    },
-    '.mosaic-window .mosaic-preview, .mosaic-preview .mosaic-preview': {
-      height: '100%',
-      width: '100%',
-      position: 'absolute',
-      zIndex: '0',
-      border: '1px solid black',
-      maxHeight: 400,
-    },
-    '.mosaic-window .mosaic-preview .mosaic-window-body, .mosaic-preview .mosaic-preview .mosaic-window-body': {
-      display: 'flex',
-      fallbacks: [
-        {
-          display: '-webkit-box',
-        },
-      ],
-      webkitBoxOrient: 'vertical',
-      webkitBoxDirection: 'normal',
-      flexDirection: 'column',
-      webkitBoxAlign: 'center',
-      alignItems: 'center',
-      webkitBoxPack: 'center',
-      justifyContent: 'center',
-    },
-    '.mosaic-window .mosaic-preview h4, .mosaic-preview .mosaic-preview h4': {
-      marginBottom: 10,
-    },
-    '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.close-button:before': {
-      content: '\'Close\'',
-    },
-    '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.split-button:before': {
-      content: '\'Split\'',
-    },
-    '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.replace-button:before': {
-      content: '\'Replace\'',
-    },
-    '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.expand-button:before': {
-      content: '\'Expand\'',
-    },
+  '.mosaic': {
+    height: '100%',
+    width: '100%',
+  },
+  '.mosaic, .mosaic > *': {
+    boxSizing: 'border-box',
+  },
+  '.mosaic .mosaic-zero-state': {
+    position: 'absolute',
+    top: 6,
+    right: 6,
+    bottom: 6,
+    left: 6,
+    width: 'auto',
+    height: 'auto',
+    zIndex: '1',
+  },
+  '.mosaic-root': {
+    position: 'absolute',
+    top: 3,
+    right: 3,
+    bottom: 3,
+    left: 3,
+  },
+  '.mosaic-split': {
+    position: 'absolute',
+    zIndex: '1',
+    touchAction: 'none',
+  },
+  '.mosaic-split:hover': {
+    background: 'black',
+  },
+  '.mosaic-split .mosaic-split-line': {
+    position: 'absolute',
+  },
+  '.mosaic-split.-row': {
+    marginLeft: -3,
+    width: 6,
+    cursor: 'ew-resize',
+  },
+  '.mosaic-split.-row .mosaic-split-line': {
+    top: '0',
+    bottom: '0',
+    left: 3,
+    right: 3,
+  },
+  '.mosaic-split.-column': {
+    marginTop: -3,
+    height: 6,
+    cursor: 'ns-resize',
+  },
+  '.mosaic-split.-column .mosaic-split-line': {
+    top: 3,
+    bottom: 3,
+    left: '0',
+    right: '0',
+  },
+  '.mosaic-tile': {
+    position: 'absolute',
+    margin: 3,
+  },
+  '.mosaic-tile > *': {
+    height: '100%',
+    width: '100%',
+  },
+  '.mosaic-drop-target': {
+    position: 'relative',
+  },
+  '.mosaic-drop-target.drop-target-hover .drop-target-container': {
+    display: 'block',
+  },
+  '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.left': {
+    right: 'calc(100% -  10px )',
+  },
+  '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.right': {
+    left: 'calc(100% -  10px )',
+  },
+  '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.bottom': {
+    top: 'calc(100% -  10px )',
+  },
+  '.mosaic-drop-target.mosaic > .drop-target-container .drop-target.top': {
+    bottom: 'calc(100% -  10px )',
+  },
+  '.mosaic-drop-target .drop-target-container': {
+    position: 'absolute',
+    top: '0',
+    right: '0',
+    bottom: '0',
+    left: '0',
+    display: 'none',
+  },
+  '.mosaic-drop-target .drop-target-container.-dragging': {
+    display: 'block',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target': {
+    position: 'absolute',
+    top: '0',
+    right: '0',
+    bottom: '0',
+    left: '0',
+    background: 'rgba(0, 0, 0, 0.2)',
+    border: '2px solid black',
+    opacity: '0',
+    zIndex: '5',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.left': {
+    right: 'calc(100% -  30% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.right': {
+    left: 'calc(100% -  30% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.bottom': {
+    top: 'calc(100% -  30% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.top': {
+    bottom: 'calc(100% -  30% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover': {
+    opacity: '1',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.left': {
+    right: 'calc(100% -  50% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.right': {
+    left: 'calc(100% -  50% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.bottom': {
+    top: 'calc(100% -  50% )',
+  },
+  '.mosaic-drop-target .drop-target-container .drop-target.drop-target-hover.top': {
+    bottom: 'calc(100% -  50% )',
+  },
+  '.mosaic-window, .mosaic-preview': {
+    position: 'relative',
+    display: 'flex',
+    fallbacks: [
+      {
+        display: '-webkit-box',
+      },
+    ],
+    webkitBoxOrient: 'vertical',
+    webkitBoxDirection: 'normal',
+    flexDirection: 'column',
+    overflow: 'hidden',
+    boxShadow: '0 0 1px rgba(0, 0, 0, 0.2)',
+  },
+  '.mosaic-window .mosaic-window-toolbar, .mosaic-preview .mosaic-window-toolbar': {
+    zIndex: '4',
+    display: 'flex',
+    fallbacks: [
+      {
+        display: '-webkit-box',
+      },
+    ],
+    webkitBoxPack: 'justify',
+    justifyContent: 'space-between',
+    webkitBoxAlign: 'center',
+    alignItems: 'center',
+    flexShrink: '0',
+    height: 30,
+    background: 'white',
+    boxShadow: '0 1px 1px rgba(0, 0, 0, 0.2)',
+  },
+  '.mosaic-window .mosaic-window-toolbar.draggable, .mosaic-preview .mosaic-window-toolbar.draggable': {
+    cursor: 'move',
+  },
+  '.mosaic-window .mosaic-window-title, .mosaic-preview .mosaic-window-title': {
+    paddingLeft: 15,
+    webkitBoxFlex: '1',
+    flex: '1',
+    textOverflow: 'ellipsis',
+    whiteSpace: 'nowrap',
+    overflow: 'hidden',
+    minHeight: 18,
+  },
+  '.mosaic-window .mosaic-window-controls, .mosaic-preview .mosaic-window-controls': {
+    display: 'flex',
+    fallbacks: [
+      {
+        display: '-webkit-box',
+      },
+    ],
+    height: '100%',
+  },
+  '.mosaic-window .mosaic-window-controls .separator, .mosaic-preview .mosaic-window-controls .separator': {
+    height: 20,
+    borderLeft: '1px solid black',
+    margin: '5px 4px',
+  },
+  '.mosaic-window .mosaic-window-body, .mosaic-preview .mosaic-window-body': {
+    position: 'relative',
+    webkitBoxFlex: '1',
+    flex: '1',
+    height: '0',
+    background: 'white',
+    zIndex: '1',
+    overflow: 'hidden',
+  },
+  '.mosaic-window .mosaic-window-additional-actions-bar, .mosaic-preview .mosaic-window-additional-actions-bar': {
+    position: 'absolute',
+    top: 30,
+    right: '0',
+    bottom: 'initial',
+    left: '0',
+    height: '0',
+    overflow: 'hidden',
+    background: 'white',
+    webkitBoxPack: 'end',
+    justifyContent: 'flex-end',
+    display: 'flex',
+    fallbacks: [
+      {
+        display: '-webkit-box',
+      },
+    ],
+    zIndex: '3',
+  },
+  '.mosaic-window .mosaic-window-additional-actions-bar .bp3-button, .mosaic-preview .mosaic-window-additional-actions-bar .bp3-button': {
+    margin: '0',
+  },
+  '.mosaic-window .mosaic-window-additional-actions-bar .bp3-button:after, .mosaic-preview .mosaic-window-additional-actions-bar .bp3-button:after': {
+    display: 'none',
+  },
+  '.mosaic-window .mosaic-window-body-overlay, .mosaic-preview .mosaic-window-body-overlay': {
+    position: 'absolute',
+    top: '0',
+    right: '0',
+    bottom: '0',
+    left: '0',
+    opacity: '0',
+    background: 'white',
+    display: 'none',
+    zIndex: '2',
+  },
+  '.mosaic-window.additional-controls-open .mosaic-window-additional-actions-bar, .mosaic-preview.additional-controls-open .mosaic-window-additional-actions-bar': {
+    height: 30,
+  },
+  '.mosaic-window.additional-controls-open .mosaic-window-body-overlay, .mosaic-preview.additional-controls-open .mosaic-window-body-overlay': {
+    display: 'block',
+  },
+  '.mosaic-window .mosaic-preview, .mosaic-preview .mosaic-preview': {
+    height: '100%',
+    width: '100%',
+    position: 'absolute',
+    zIndex: '0',
+    border: '1px solid black',
+    maxHeight: 400,
+  },
+  '.mosaic-window .mosaic-preview .mosaic-window-body, .mosaic-preview .mosaic-preview .mosaic-window-body': {
+    display: 'flex',
+    fallbacks: [
+      {
+        display: '-webkit-box',
+      },
+    ],
+    webkitBoxOrient: 'vertical',
+    webkitBoxDirection: 'normal',
+    flexDirection: 'column',
+    webkitBoxAlign: 'center',
+    alignItems: 'center',
+    webkitBoxPack: 'center',
+    justifyContent: 'center',
+  },
+  '.mosaic-window .mosaic-preview h4, .mosaic-preview .mosaic-preview h4': {
+    marginBottom: 10,
+  },
+  '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.close-button:before': {
+    content: '\'Close\'',
+  },
+  '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.split-button:before': {
+    content: '\'Split\'',
+  },
+  '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.replace-button:before': {
+    content: '\'Replace\'',
+  },
+  '.mosaic:not(.mosaic-blueprint-theme) .mosaic-default-control.expand-button:before': {
+    content: '\'Expand\'',
   },
 };