|  | @@ -6,10 +6,14 @@ import React, {
 | 
	
		
			
				|  |  |    useState,
 | 
	
		
			
				|  |  |  } from "react";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -import { MIME_TYPES, arrayToMap } from "@excalidraw/common";
 | 
	
		
			
				|  |  | +import { MIME_TYPES, arrayToMap, nextAnimationFrame } from "@excalidraw/common";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  import { duplicateElements } from "@excalidraw/element";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +import clsx from "clsx";
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import { deburr } from "../deburr";
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  import { useLibraryCache } from "../hooks/useLibraryItemSvg";
 | 
	
		
			
				|  |  |  import { useScrollPosition } from "../hooks/useScrollPosition";
 | 
	
		
			
				|  |  |  import { t } from "../i18n";
 | 
	
	
		
			
				|  | @@ -26,6 +30,10 @@ import Stack from "./Stack";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  import "./LibraryMenuItems.scss";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +import { TextField } from "./TextField";
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import { useDevice } from "./App";
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  import type { ExcalidrawLibraryIds } from "../data/types";
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  import type {
 | 
	
	
		
			
				|  | @@ -65,6 +73,7 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |    selectedItems: LibraryItem["id"][];
 | 
	
		
			
				|  |  |    onSelectItems: (id: LibraryItem["id"][]) => void;
 | 
	
		
			
				|  |  |  }) {
 | 
	
		
			
				|  |  | +  const device = useDevice();
 | 
	
		
			
				|  |  |    const libraryContainerRef = useRef<HTMLDivElement>(null);
 | 
	
		
			
				|  |  |    const scrollPosition = useScrollPosition<HTMLDivElement>(libraryContainerRef);
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -76,6 +85,30 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |    }, []); // eslint-disable-line react-hooks/exhaustive-deps
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    const { svgCache } = useLibraryCache();
 | 
	
		
			
				|  |  | +  const [lastSelectedItem, setLastSelectedItem] = useState<
 | 
	
		
			
				|  |  | +    LibraryItem["id"] | null
 | 
	
		
			
				|  |  | +  >(null);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const [searchInputValue, setSearchInputValue] = useState("");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const IS_LIBRARY_EMPTY = !libraryItems.length && !pendingElements.length;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const IS_SEARCHING = !IS_LIBRARY_EMPTY && !!searchInputValue.trim();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const filteredItems = useMemo(() => {
 | 
	
		
			
				|  |  | +    const searchQuery = deburr(searchInputValue.trim().toLowerCase());
 | 
	
		
			
				|  |  | +    if (!searchQuery) {
 | 
	
		
			
				|  |  | +      return [];
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    return libraryItems.filter((item) => {
 | 
	
		
			
				|  |  | +      const itemName = item.name || "";
 | 
	
		
			
				|  |  | +      return (
 | 
	
		
			
				|  |  | +        itemName.trim() && deburr(itemName.toLowerCase()).includes(searchQuery)
 | 
	
		
			
				|  |  | +      );
 | 
	
		
			
				|  |  | +    });
 | 
	
		
			
				|  |  | +  }, [libraryItems, searchInputValue]);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    const unpublishedItems = useMemo(
 | 
	
		
			
				|  |  |      () => libraryItems.filter((item) => item.status !== "published"),
 | 
	
		
			
				|  |  |      [libraryItems],
 | 
	
	
		
			
				|  | @@ -86,23 +119,10 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |      [libraryItems],
 | 
	
		
			
				|  |  |    );
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  const showBtn = !libraryItems.length && !pendingElements.length;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  const isLibraryEmpty =
 | 
	
		
			
				|  |  | -    !pendingElements.length &&
 | 
	
		
			
				|  |  | -    !unpublishedItems.length &&
 | 
	
		
			
				|  |  | -    !publishedItems.length;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  const [lastSelectedItem, setLastSelectedItem] = useState<
 | 
	
		
			
				|  |  | -    LibraryItem["id"] | null
 | 
	
		
			
				|  |  | -  >(null);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |    const onItemSelectToggle = useCallback(
 | 
	
		
			
				|  |  |      (id: LibraryItem["id"], event: React.MouseEvent) => {
 | 
	
		
			
				|  |  |        const shouldSelect = !selectedItems.includes(id);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |        const orderedItems = [...unpublishedItems, ...publishedItems];
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |        if (shouldSelect) {
 | 
	
		
			
				|  |  |          if (event.shiftKey && lastSelectedItem) {
 | 
	
		
			
				|  |  |            const rangeStart = orderedItems.findIndex(
 | 
	
	
		
			
				|  | @@ -128,7 +148,6 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              [],
 | 
	
		
			
				|  |  |            );
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |            onSelectItems(nextSelectedIds);
 | 
	
		
			
				|  |  |          } else {
 | 
	
		
			
				|  |  |            onSelectItems([...selectedItems, id]);
 | 
	
	
		
			
				|  | @@ -194,7 +213,6 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |        if (!id) {
 | 
	
		
			
				|  |  |          return false;
 | 
	
		
			
				|  |  |        }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |        return selectedItems.includes(id);
 | 
	
		
			
				|  |  |      },
 | 
	
		
			
				|  |  |      [selectedItems],
 | 
	
	
		
			
				|  | @@ -214,10 +232,120 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |    );
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    const itemsRenderedPerBatch =
 | 
	
		
			
				|  |  | -    svgCache.size >= libraryItems.length
 | 
	
		
			
				|  |  | +    svgCache.size >=
 | 
	
		
			
				|  |  | +    (filteredItems.length ? filteredItems : libraryItems).length
 | 
	
		
			
				|  |  |        ? CACHED_ITEMS_RENDERED_PER_BATCH
 | 
	
		
			
				|  |  |        : ITEMS_RENDERED_PER_BATCH;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  const searchInputRef = useRef<HTMLInputElement>(null);
 | 
	
		
			
				|  |  | +  useEffect(() => {
 | 
	
		
			
				|  |  | +    // focus could be stolen by tab trigger button
 | 
	
		
			
				|  |  | +    nextAnimationFrame(() => {
 | 
	
		
			
				|  |  | +      searchInputRef.current?.focus();
 | 
	
		
			
				|  |  | +    });
 | 
	
		
			
				|  |  | +  }, []);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const JSX_whenNotSearching = !IS_SEARCHING && (
 | 
	
		
			
				|  |  | +    <>
 | 
	
		
			
				|  |  | +      {!IS_LIBRARY_EMPTY && (
 | 
	
		
			
				|  |  | +        <div className="library-menu-items-container__header">
 | 
	
		
			
				|  |  | +          {t("labels.personalLib")}
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +      )}
 | 
	
		
			
				|  |  | +      {!pendingElements.length && !unpublishedItems.length ? (
 | 
	
		
			
				|  |  | +        <div className="library-menu-items__no-items">
 | 
	
		
			
				|  |  | +          {!publishedItems.length && (
 | 
	
		
			
				|  |  | +            <div className="library-menu-items__no-items__label">
 | 
	
		
			
				|  |  | +              {t("library.noItems")}
 | 
	
		
			
				|  |  | +            </div>
 | 
	
		
			
				|  |  | +          )}
 | 
	
		
			
				|  |  | +          <div className="library-menu-items__no-items__hint">
 | 
	
		
			
				|  |  | +            {publishedItems.length > 0
 | 
	
		
			
				|  |  | +              ? t("library.hint_emptyPrivateLibrary")
 | 
	
		
			
				|  |  | +              : t("library.hint_emptyLibrary")}
 | 
	
		
			
				|  |  | +          </div>
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +      ) : (
 | 
	
		
			
				|  |  | +        <LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +          {pendingElements.length > 0 && (
 | 
	
		
			
				|  |  | +            <LibraryMenuSection
 | 
	
		
			
				|  |  | +              itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | +              items={[{ id: null, elements: pendingElements }]}
 | 
	
		
			
				|  |  | +              onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | +              onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | +              onClick={onAddToLibraryClick}
 | 
	
		
			
				|  |  | +              isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | +              svgCache={svgCache}
 | 
	
		
			
				|  |  | +            />
 | 
	
		
			
				|  |  | +          )}
 | 
	
		
			
				|  |  | +          <LibraryMenuSection
 | 
	
		
			
				|  |  | +            itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | +            items={unpublishedItems}
 | 
	
		
			
				|  |  | +            onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | +            onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | +            onClick={onItemClick}
 | 
	
		
			
				|  |  | +            isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | +            svgCache={svgCache}
 | 
	
		
			
				|  |  | +          />
 | 
	
		
			
				|  |  | +        </LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +      )}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      {publishedItems.length > 0 && (
 | 
	
		
			
				|  |  | +        <div
 | 
	
		
			
				|  |  | +          className="library-menu-items-container__header"
 | 
	
		
			
				|  |  | +          style={{ marginTop: "0.75rem" }}
 | 
	
		
			
				|  |  | +        >
 | 
	
		
			
				|  |  | +          {t("labels.excalidrawLib")}
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +      )}
 | 
	
		
			
				|  |  | +      {publishedItems.length > 0 && (
 | 
	
		
			
				|  |  | +        <LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +          <LibraryMenuSection
 | 
	
		
			
				|  |  | +            itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | +            items={publishedItems}
 | 
	
		
			
				|  |  | +            onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | +            onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | +            onClick={onItemClick}
 | 
	
		
			
				|  |  | +            isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | +            svgCache={svgCache}
 | 
	
		
			
				|  |  | +          />
 | 
	
		
			
				|  |  | +        </LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +      )}
 | 
	
		
			
				|  |  | +    </>
 | 
	
		
			
				|  |  | +  );
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const JSX_whenSearching = IS_SEARCHING && (
 | 
	
		
			
				|  |  | +    <>
 | 
	
		
			
				|  |  | +      <div className="library-menu-items-container__header">
 | 
	
		
			
				|  |  | +        {t("library.search.heading")}
 | 
	
		
			
				|  |  | +        {!isLoading && (
 | 
	
		
			
				|  |  | +          <div className="library-menu-items-container__header__hint">
 | 
	
		
			
				|  |  | +            <kbd>esc</kbd> to clear
 | 
	
		
			
				|  |  | +          </div>
 | 
	
		
			
				|  |  | +        )}
 | 
	
		
			
				|  |  | +      </div>
 | 
	
		
			
				|  |  | +      {filteredItems.length > 0 ? (
 | 
	
		
			
				|  |  | +        <LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +          <LibraryMenuSection
 | 
	
		
			
				|  |  | +            itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | +            items={filteredItems}
 | 
	
		
			
				|  |  | +            onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | +            onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | +            onClick={onItemClick}
 | 
	
		
			
				|  |  | +            isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | +            svgCache={svgCache}
 | 
	
		
			
				|  |  | +          />
 | 
	
		
			
				|  |  | +        </LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | +      ) : (
 | 
	
		
			
				|  |  | +        <div className="library-menu-items__no-items">
 | 
	
		
			
				|  |  | +          <div className="library-menu-items__no-items__hint">
 | 
	
		
			
				|  |  | +            {t("library.search.noResults")}
 | 
	
		
			
				|  |  | +          </div>
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +      )}
 | 
	
		
			
				|  |  | +    </>
 | 
	
		
			
				|  |  | +  );
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    return (
 | 
	
		
			
				|  |  |      <div
 | 
	
		
			
				|  |  |        className="library-menu-items-container"
 | 
	
	
		
			
				|  | @@ -229,127 +357,58 @@ export default function LibraryMenuItems({
 | 
	
		
			
				|  |  |            : { borderBottom: 0 }
 | 
	
		
			
				|  |  |        }
 | 
	
		
			
				|  |  |      >
 | 
	
		
			
				|  |  | -      {!isLibraryEmpty && (
 | 
	
		
			
				|  |  | +      <div className="library-menu-items-header">
 | 
	
		
			
				|  |  | +        {!IS_LIBRARY_EMPTY && (
 | 
	
		
			
				|  |  | +          <TextField
 | 
	
		
			
				|  |  | +            ref={searchInputRef}
 | 
	
		
			
				|  |  | +            type="search"
 | 
	
		
			
				|  |  | +            className={clsx("library-menu-items-container__search", {
 | 
	
		
			
				|  |  | +              hideCancelButton: !device.editor.isMobile,
 | 
	
		
			
				|  |  | +            })}
 | 
	
		
			
				|  |  | +            placeholder={t("library.search.inputPlaceholder")}
 | 
	
		
			
				|  |  | +            value={searchInputValue}
 | 
	
		
			
				|  |  | +            onChange={(value) => setSearchInputValue(value)}
 | 
	
		
			
				|  |  | +          />
 | 
	
		
			
				|  |  | +        )}
 | 
	
		
			
				|  |  |          <LibraryDropdownMenu
 | 
	
		
			
				|  |  |            selectedItems={selectedItems}
 | 
	
		
			
				|  |  |            onSelectItems={onSelectItems}
 | 
	
		
			
				|  |  |            className="library-menu-dropdown-container--in-heading"
 | 
	
		
			
				|  |  |          />
 | 
	
		
			
				|  |  | -      )}
 | 
	
		
			
				|  |  | +      </div>
 | 
	
		
			
				|  |  |        <Stack.Col
 | 
	
		
			
				|  |  |          className="library-menu-items-container__items"
 | 
	
		
			
				|  |  |          align="start"
 | 
	
		
			
				|  |  |          gap={1}
 | 
	
		
			
				|  |  |          style={{
 | 
	
		
			
				|  |  |            flex: publishedItems.length > 0 ? 1 : "0 1 auto",
 | 
	
		
			
				|  |  | -          marginBottom: 0,
 | 
	
		
			
				|  |  | +          margin: IS_LIBRARY_EMPTY ? "auto" : 0,
 | 
	
		
			
				|  |  |          }}
 | 
	
		
			
				|  |  |          ref={libraryContainerRef}
 | 
	
		
			
				|  |  |        >
 | 
	
		
			
				|  |  | -        <>
 | 
	
		
			
				|  |  | -          {!isLibraryEmpty && (
 | 
	
		
			
				|  |  | -            <div className="library-menu-items-container__header">
 | 
	
		
			
				|  |  | -              {t("labels.personalLib")}
 | 
	
		
			
				|  |  | -            </div>
 | 
	
		
			
				|  |  | -          )}
 | 
	
		
			
				|  |  | -          {isLoading && (
 | 
	
		
			
				|  |  | -            <div
 | 
	
		
			
				|  |  | -              style={{
 | 
	
		
			
				|  |  | -                position: "absolute",
 | 
	
		
			
				|  |  | -                top: "var(--container-padding-y)",
 | 
	
		
			
				|  |  | -                right: "var(--container-padding-x)",
 | 
	
		
			
				|  |  | -                transform: "translateY(50%)",
 | 
	
		
			
				|  |  | -              }}
 | 
	
		
			
				|  |  | -            >
 | 
	
		
			
				|  |  | -              <Spinner />
 | 
	
		
			
				|  |  | -            </div>
 | 
	
		
			
				|  |  | -          )}
 | 
	
		
			
				|  |  | -          {!pendingElements.length && !unpublishedItems.length ? (
 | 
	
		
			
				|  |  | -            <div className="library-menu-items__no-items">
 | 
	
		
			
				|  |  | -              <div className="library-menu-items__no-items__label">
 | 
	
		
			
				|  |  | -                {t("library.noItems")}
 | 
	
		
			
				|  |  | -              </div>
 | 
	
		
			
				|  |  | -              <div className="library-menu-items__no-items__hint">
 | 
	
		
			
				|  |  | -                {publishedItems.length > 0
 | 
	
		
			
				|  |  | -                  ? t("library.hint_emptyPrivateLibrary")
 | 
	
		
			
				|  |  | -                  : t("library.hint_emptyLibrary")}
 | 
	
		
			
				|  |  | -              </div>
 | 
	
		
			
				|  |  | -            </div>
 | 
	
		
			
				|  |  | -          ) : (
 | 
	
		
			
				|  |  | -            <LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | -              {pendingElements.length > 0 && (
 | 
	
		
			
				|  |  | -                <LibraryMenuSection
 | 
	
		
			
				|  |  | -                  itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | -                  items={[{ id: null, elements: pendingElements }]}
 | 
	
		
			
				|  |  | -                  onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | -                  onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | -                  onClick={onAddToLibraryClick}
 | 
	
		
			
				|  |  | -                  isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | -                  svgCache={svgCache}
 | 
	
		
			
				|  |  | -                />
 | 
	
		
			
				|  |  | -              )}
 | 
	
		
			
				|  |  | -              <LibraryMenuSection
 | 
	
		
			
				|  |  | -                itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | -                items={unpublishedItems}
 | 
	
		
			
				|  |  | -                onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | -                onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | -                onClick={onItemClick}
 | 
	
		
			
				|  |  | -                isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | -                svgCache={svgCache}
 | 
	
		
			
				|  |  | -              />
 | 
	
		
			
				|  |  | -            </LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | -          )}
 | 
	
		
			
				|  |  | -        </>
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -        <>
 | 
	
		
			
				|  |  | -          {(publishedItems.length > 0 ||
 | 
	
		
			
				|  |  | -            pendingElements.length > 0 ||
 | 
	
		
			
				|  |  | -            unpublishedItems.length > 0) && (
 | 
	
		
			
				|  |  | -            <div className="library-menu-items-container__header library-menu-items-container__header--excal">
 | 
	
		
			
				|  |  | -              {t("labels.excalidrawLib")}
 | 
	
		
			
				|  |  | -            </div>
 | 
	
		
			
				|  |  | -          )}
 | 
	
		
			
				|  |  | -          {publishedItems.length > 0 ? (
 | 
	
		
			
				|  |  | -            <LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | -              <LibraryMenuSection
 | 
	
		
			
				|  |  | -                itemsRenderedPerBatch={itemsRenderedPerBatch}
 | 
	
		
			
				|  |  | -                items={publishedItems}
 | 
	
		
			
				|  |  | -                onItemSelectToggle={onItemSelectToggle}
 | 
	
		
			
				|  |  | -                onItemDrag={onItemDrag}
 | 
	
		
			
				|  |  | -                onClick={onItemClick}
 | 
	
		
			
				|  |  | -                isItemSelected={isItemSelected}
 | 
	
		
			
				|  |  | -                svgCache={svgCache}
 | 
	
		
			
				|  |  | -              />
 | 
	
		
			
				|  |  | -            </LibraryMenuSectionGrid>
 | 
	
		
			
				|  |  | -          ) : unpublishedItems.length > 0 ? (
 | 
	
		
			
				|  |  | -            <div
 | 
	
		
			
				|  |  | -              style={{
 | 
	
		
			
				|  |  | -                margin: "1rem 0",
 | 
	
		
			
				|  |  | -                display: "flex",
 | 
	
		
			
				|  |  | -                flexDirection: "column",
 | 
	
		
			
				|  |  | -                alignItems: "center",
 | 
	
		
			
				|  |  | -                justifyContent: "center",
 | 
	
		
			
				|  |  | -                width: "100%",
 | 
	
		
			
				|  |  | -                fontSize: ".9rem",
 | 
	
		
			
				|  |  | -              }}
 | 
	
		
			
				|  |  | -            >
 | 
	
		
			
				|  |  | -              {t("library.noItems")}
 | 
	
		
			
				|  |  | -            </div>
 | 
	
		
			
				|  |  | -          ) : null}
 | 
	
		
			
				|  |  | -        </>
 | 
	
		
			
				|  |  | +        {isLoading && (
 | 
	
		
			
				|  |  | +          <div
 | 
	
		
			
				|  |  | +            style={{
 | 
	
		
			
				|  |  | +              position: "absolute",
 | 
	
		
			
				|  |  | +              top: "var(--container-padding-y)",
 | 
	
		
			
				|  |  | +              right: "var(--container-padding-x)",
 | 
	
		
			
				|  |  | +              transform: "translateY(50%)",
 | 
	
		
			
				|  |  | +            }}
 | 
	
		
			
				|  |  | +          >
 | 
	
		
			
				|  |  | +            <Spinner />
 | 
	
		
			
				|  |  | +          </div>
 | 
	
		
			
				|  |  | +        )}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -        {showBtn && (
 | 
	
		
			
				|  |  | +        {JSX_whenNotSearching}
 | 
	
		
			
				|  |  | +        {JSX_whenSearching}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        {IS_LIBRARY_EMPTY && (
 | 
	
		
			
				|  |  |            <LibraryMenuControlButtons
 | 
	
		
			
				|  |  |              style={{ padding: "16px 0", width: "100%" }}
 | 
	
		
			
				|  |  |              id={id}
 | 
	
		
			
				|  |  |              libraryReturnUrl={libraryReturnUrl}
 | 
	
		
			
				|  |  |              theme={theme}
 | 
	
		
			
				|  |  | -          >
 | 
	
		
			
				|  |  | -            <LibraryDropdownMenu
 | 
	
		
			
				|  |  | -              selectedItems={selectedItems}
 | 
	
		
			
				|  |  | -              onSelectItems={onSelectItems}
 | 
	
		
			
				|  |  | -            />
 | 
	
		
			
				|  |  | -          </LibraryMenuControlButtons>
 | 
	
		
			
				|  |  | +          />
 | 
	
		
			
				|  |  |          )}
 | 
	
		
			
				|  |  |        </Stack.Col>
 | 
	
		
			
				|  |  |      </div>
 |