obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
353 lines (336 loc) • 39.1 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initCjs() {
const globalThisRecord = globalThis;
globalThisRecord['__name'] ??= name;
const originalRequire = require;
if (originalRequire && !originalRequire.__isPatched) {
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
require = Object.assign(
(id) => requirePatched(id),
originalRequire,
{
__isPatched: true
}
);
}
const newFuncs = {
__extractDefault() {
return extractDefault;
},
process() {
const browserProcess = {
browser: true,
cwd() {
return '/';
},
env: {},
platform: 'android'
};
return browserProcess;
}
};
for (const key of Object.keys(newFuncs)) {
globalThisRecord[key] ??= newFuncs[key]?.();
}
function name(obj) {
return obj;
}
function extractDefault(module) {
return module && module.__esModule && 'default' in module ? module.default : module;
}
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'obsidian',
'@codemirror/autocomplete',
'@codemirror/collab',
'@codemirror/commands',
'@codemirror/language',
'@codemirror/lint',
'@codemirror/search',
'@codemirror/state',
'@codemirror/text',
'@codemirror/view',
'@lezer/common',
'@lezer/lr',
'@lezer/highlight'];
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'@codemirror/closebrackets',
'@codemirror/comment',
'@codemirror/fold',
'@codemirror/gutter',
'@codemirror/highlight',
'@codemirror/history',
'@codemirror/matchbrackets',
'@codemirror/panel',
'@codemirror/rangeset',
'@codemirror/rectangular-selection',
'@codemirror/stream-parser',
'@codemirror/tooltip'];
function requirePatched(id) {
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
return originalRequire?.(id);
}
// eslint-disable-next-line @typescript-eslint/no-deprecated, @typescript-eslint/no-unnecessary-condition -- We need access to app here which might not be available yet.
if (globalThis?.app?.isMobile) {
if (id === 'process' || id === 'node:process') {
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
return globalThis.process;
}
} else {
const module = originalRequire?.(id);
if (module) {
return extractDefault(module);
}
}
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
return {};
}
})();
;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var Dataview_exports = {};
__export(Dataview_exports, {
getRenderedContainer: () => getRenderedContainer,
insertCodeBlock: () => insertCodeBlock,
reloadCurrentFileCache: () => reloadCurrentFileCache,
renderIframe: () => renderIframe,
renderPaginatedList: () => renderPaginatedList,
renderPaginatedTable: () => renderPaginatedTable
});
module.exports = __toCommonJS(Dataview_exports);
var import_Async = require('../Async.cjs');
var import_Error = require('../Error.cjs');
var import_FileSystem = require('./FileSystem.cjs');
var import_i18n = require('./i18n/i18n.cjs');
var import_ResourceUrl = require('./ResourceUrl.cjs');
async function reloadCurrentFileCache(dv) {
await window.DataviewAPI?.index.reload((0, import_FileSystem.getFile)(dv.app, dv.current().file.path));
}
const paginationCss = `
.pagination .page-link.disabled {
pointer-events: none;
color: gray;
}
.pagination .page-link {
margin: 0 5px;
cursor: pointer;
text-decoration: none;
color: blue;
}
.pagination .page-link:hover:not(.disabled) {
text-decoration: underline;
}
.pagination .page-link.current {
font-weight: bold;
text-decoration: underline;
}
.pagination select,
.pagination input {
margin: 0 5px;
}
`;
async function getRenderedContainer(dv, renderer) {
const oldContainer = dv.container;
const tempContainer = dv.paragraph("");
dv.container = tempContainer;
dv.container.empty();
try {
await renderer();
} catch (e) {
dv.paragraph(`\u274C${(0, import_Error.errorToString)(e)}`);
} finally {
dv.container = oldContainer;
tempContainer.remove();
}
return tempContainer;
}
function insertCodeBlock(dv, language, code) {
const MIN_FENCE_LENGTH = 3;
const fenceRegExp = new RegExp(`^\`{${String(MIN_FENCE_LENGTH)},}`, "gm");
const fenceMatches = code.matchAll(fenceRegExp);
const fenceLengths = Array.from(fenceMatches).map((fenceMatch) => fenceMatch[0].length);
const maxFenceLength = Math.max(0, ...fenceLengths);
const resultFenceLength = Math.max(MIN_FENCE_LENGTH, maxFenceLength + 1);
const resultFence = "`".repeat(resultFenceLength);
dv.paragraph(`${resultFence}${language}
${code}
${resultFence}`);
}
function renderIframe(options) {
const {
dv,
height = "600px",
relativePathOrFile,
width = "100%"
} = options;
dv.el("iframe", "", {
attr: {
height,
src: (0, import_ResourceUrl.relativePathToResourceUrl)(dv.app, (0, import_FileSystem.getPath)(dv.app, relativePathOrFile), dv.current().file.path),
width
}
});
}
async function renderPaginatedList(options) {
const {
dv,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
itemsPerPageOptions = [10, 20, 50, 100],
rows
} = options;
await renderPaginated({
dv,
itemsPerPageOptions,
renderer: async (rowsForOnePage) => {
await dv.list(rowsForOnePage);
},
rows
});
}
async function renderPaginatedTable(options) {
const {
dv,
headers,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
itemsPerPageOptions = [10, 20, 50, 100],
rows
} = options;
await renderPaginated({
dv,
itemsPerPageOptions,
renderer: async (rowsForOnePage) => {
await dv.table(headers, rowsForOnePage);
},
rows
});
}
async function renderPaginated(options) {
const SECOND_PAGE_NUMBER = 2;
const MORE_PAGE_NUMBER = 3;
const {
dv,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
itemsPerPageOptions = [10, 20, 50, 100],
rows
} = options;
if (rows.length === 0) {
dv.paragraph("No items found");
return;
}
const container = dv.container;
let itemsPerPage = itemsPerPageOptions[0] ?? (0, import_Error.throwExpression)(new Error("Items per page options are empty"));
let totalPages = Math.ceil(rows.length / itemsPerPage);
await renderPage(1);
function createPaginationControls(pageNumber) {
const paginationDiv = container.createDiv({ cls: "pagination" });
const paginationRow1Div = paginationDiv.createDiv();
createPageLink("First", 1, pageNumber === 1);
createPageLink("Prev", pageNumber - 1, pageNumber === 1);
if (pageNumber > MORE_PAGE_NUMBER) {
paginationRow1Div.createSpan({ text: "..." });
}
for (let i = Math.max(1, pageNumber - SECOND_PAGE_NUMBER); i <= Math.min(totalPages, pageNumber + SECOND_PAGE_NUMBER); i++) {
const pageLink = createPageLink(String(i), i, i === pageNumber);
if (i === pageNumber) {
pageLink.addClass("current");
}
}
if (pageNumber < totalPages - SECOND_PAGE_NUMBER) {
paginationRow1Div.createSpan({ text: "..." });
}
createPageLink("Next", pageNumber + 1, pageNumber === totalPages);
createPageLink("Last", totalPages, pageNumber === totalPages);
const paginationRow2Div = paginationDiv.createDiv();
paginationRow2Div.createSpan({ text: ` ${(0, import_i18n.t)(($) => $.obsidianDevUtils.dataview.itemsPerPage)} ` });
const itemsPerPageSelect = paginationRow2Div.createEl("select");
itemsPerPageOptions.forEach((option) => {
itemsPerPageSelect.createEl("option", { text: String(option), value: String(option) });
});
itemsPerPageSelect.value = String(itemsPerPage);
itemsPerPageSelect.addEventListener(
"change",
(0, import_Async.convertAsyncToSync)(async () => {
itemsPerPage = parseInt(itemsPerPageSelect.value, 10);
totalPages = Math.ceil(rows.length / itemsPerPage);
await renderPage(1);
})
);
paginationRow2Div.createSpan({ text: ` ${(0, import_i18n.t)(($) => $.obsidianDevUtils.dataview.jumpToPage)} ` });
const jumpToPageInput = paginationRow2Div.createEl("input", { attr: { max: totalPages, min: 1 }, type: "number" });
jumpToPageInput.addEventListener(
"keydown",
(0, import_Async.convertAsyncToSync)(async (event) => {
if (event.key === "Enter") {
const page = parseInt(jumpToPageInput.value, 10);
if (page >= 1 && page <= totalPages) {
await renderPage(page);
}
}
})
);
paginationRow2Div.createSpan({ text: (0, import_i18n.t)(($) => $.obsidianDevUtils.dataview.pageHeader, { pageNumber, totalItems: rows.length, totalPages }) });
function createPageLink(text, currentPageNumber, disabled = false) {
const link = paginationRow1Div.createEl("a", { cls: "page-link", href: `#${String(currentPageNumber)}`, text });
if (disabled) {
link.addClass("disabled");
link.onclick = (event) => {
event.preventDefault();
};
} else {
link.addEventListener(
"click",
(0, import_Async.convertAsyncToSync)(async (event) => {
event.preventDefault();
await renderPage(currentPageNumber);
})
);
}
return link;
}
}
async function renderPage(pageNumber) {
container.empty();
container.createEl("style", { text: paginationCss });
const startIndex = (pageNumber - 1) * itemsPerPage;
const endIndex = startIndex + itemsPerPage;
const rowsForCurrentPage = rows.slice(startIndex, endIndex);
const oldContainer = dv.container;
dv.container = container;
try {
await options.renderer(rowsForCurrentPage);
} catch (e) {
dv.paragraph(`\u274C${(0, import_Error.errorToString)(e)}`);
} finally {
dv.container = oldContainer;
}
createPaginationControls(pageNumber);
}
}
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
getRenderedContainer,
insertCodeBlock,
reloadCurrentFileCache,
renderIframe,
renderPaginatedList,
renderPaginatedTable
});
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Dataview.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with Dataview in Obsidian.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport type { DataviewInlineApi as DataviewInlineApiOriginal } from './@types/Dataview/api/inline-api.d.ts';\nimport type {\n  DataArray,\n  DataviewApi,\n  SMarkdownPage\n} from './@types/Dataview/index.d.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\n\nimport { convertAsyncToSync } from '../Async.ts';\nimport {\n  errorToString,\n  throwExpression\n} from '../Error.ts';\nimport {\n  getFile,\n  getPath\n} from './FileSystem.ts';\nimport { t } from './i18n/i18n.ts';\nimport { relativePathToResourceUrl } from './ResourceUrl.ts';\n\n/**\n * Export DateTime and Link types from the Dataview API.\n */\nexport type {\n  DateTime,\n  Link\n} from './@types/Dataview/index.d.ts';\n\ndeclare global {\n  /**\n   * A {@link DataviewApi} object represents the API for interacting with Dataview in Obsidian.\n   */\n  // eslint-disable-next-line vars-on-top -- It is a `var` in module declaration. ESLint mistakenly confuses it with `var` as a variable declaration.\n  var DataviewAPI: DataviewApi | undefined;\n}\n\n/**\n * A combined page type, which includes the front matter and the SMarkdownPage.\n */\nexport type CombinedPage<CustomFrontmatter = unknown> = CombinedFrontmatter<CustomFrontmatter> & SMarkdownPage;\n\n/**\n * Extended interface for the Dataview Inline API, providing additional methods for custom page types and array handling.\n *\n * @typeParam CustomPage - The type of the custom page. Defaults to `SMarkdownPage`.\n */\nexport interface DataviewInlineApi extends DataviewInlineApiOriginal {\n  /**\n   * Wraps an array of items into a {@link DataArray} object.\n   *\n   * @typeParam T - The type of the items in the array.\n   * @param arr - The array of items to wrap.\n   * @returns A {@link DataArray} containing the items.\n   */\n  array<T>(arr: T[]): DataArray<T>;\n\n  /**\n   * Retrieves the current page, with an optional custom page type.\n   *\n   * @typeParam CustomPage - The type of the custom page. Defaults to `SMarkdownPage`.\n   * @returns The current page.\n   */\n  current<CustomFrontmatter = unknown>(): CombinedPage<CustomFrontmatter>;\n\n  /**\n   * Retrieves pages based on an optional query, with an optional custom page type.\n   *\n   * @typeParam CustomPage - The type of the custom page. Defaults to `SMarkdownPage`.\n   * @param query - An optional string query to filter the pages.\n   * @returns A {@link DataArray} of pages matching the query.\n   */\n  pages<CustomFrontmatter = unknown>(query?: string): DataArray<CombinedPage<CustomFrontmatter>>;\n\n  /**\n   * Creates a paragraph HTML element with the provided text and optional DOM element options.\n   *\n   * @param text - The content of the paragraph.\n   * @param options - Optional DOM element options, including an optional container.\n   * @returns The created HTML paragraph element.\n   */\n  paragraph(\n    text: unknown,\n    options?: DomElementInfoWithContainer\n  ): HTMLParagraphElement;\n}\n\n/**\n * DomElementInfo with an optional container.\n */\nexport type DomElementInfoWithContainer = { container?: HTMLElement } & DomElementInfo;\n\n/**\n * A combined file type, which includes the front matter and the SMarkdownFile.\n */\nexport type PageFile = SMarkdownPage['file'];\n\n/**\n * List of page files.\n */\nexport type PageFiles = ArrayOrDataArray<PageFile>;\n\n/**\n * Reloads the current file cache using the Dataview API.\n *\n * @param dv - The DataviewInlineApi instance.\n * @returns A {@link Promise} that resolves when the cache is reloaded.\n */\nexport async function reloadCurrentFileCache(dv: DataviewInlineApi): Promise<void> {\n  await window.DataviewAPI?.index.reload(getFile(dv.app, dv.current().file.path));\n}\n\nconst paginationCss = `\n.pagination .page-link.disabled {\n  pointer-events: none;\n  color: gray;\n}\n\n.pagination .page-link {\n  margin: 0 5px;\n  cursor: pointer;\n  text-decoration: none;\n  color: blue;\n}\n\n.pagination .page-link:hover:not(.disabled) {\n  text-decoration: underline;\n}\n.pagination .page-link.current {\n  font-weight: bold;\n  text-decoration: underline;\n}\n\n.pagination select,\n.pagination input {\n  margin: 0 5px;\n}\n`;\n\n/**\n * Array or DataArray type.\n */\nexport type ArrayOrDataArray<T> = DataArray<T> | T[];\n\n/**\n * Options for {@link renderIframe}.\n */\nexport interface RenderIframeOptions {\n  /**\n   * A {@link DataviewInlineApi} instance.\n   */\n  dv: DataviewInlineApi;\n\n  /**\n   * A height of the iframe.\n   */\n  height?: string;\n\n  /**\n   * A relative path to the resource to be displayed in the iframe.\n   */\n  relativePathOrFile: PathOrFile;\n\n  /**\n   * A width of the iframe.\n   */\n  width?: string;\n}\n\n/**\n * Options for {@link renderPaginatedList}.\n */\nexport interface RenderPaginatedListOptions<T> {\n  /**\n   * A {@link DataviewInlineApi} instance.\n   */\n  dv: DataviewInlineApi;\n\n  /**\n   * Options for items per page. Defaults to `[10, 20, 50, 100]`.\n   */\n  itemsPerPageOptions?: number[];\n\n  /**\n   * A list of items to paginate.\n   */\n  rows: ArrayOrDataArray<T>;\n}\n\n/**\n * Options for {@link renderPaginated}.\n */\nexport interface RenderPaginatedOptions<T> {\n  /**\n   * A {@link DataviewInlineApi} instance.\n   */\n  dv: DataviewInlineApi;\n\n  /**\n   * Options for items per page.\n   */\n  itemsPerPageOptions?: number[];\n\n  /**\n   * Display the paginated content.\n   *\n   * @param rowsForOnePage - The rows to render.\n   * @returns A {@link Promise} that resolves when the content is rendered.\n   */\n  renderer(rowsForOnePage: ArrayOrDataArray<T>): Promisable<void>;\n\n  /**\n   * Rows to paginate.\n   */\n  rows: ArrayOrDataArray<T>;\n}\n\n/**\n * Options for {@link renderPaginatedTable}.\n */\nexport interface RenderPaginatedTableOptions<T> {\n  /**\n   * A {@link DataviewInlineApi} instance.\n   */\n  dv: DataviewInlineApi;\n\n  /**\n   * A headers of the table.\n   */\n  headers: string[];\n\n  /**\n   * Options for items per page. Defaults to `[10, 20, 50, 100]`.\n   */\n  itemsPerPageOptions?: number[];\n\n  /**\n   * Rows of the table to paginate.\n   */\n  rows: ArrayOrDataArray<T>;\n}\n\n/**\n * Renders the content using the provided renderer function in a temporary container,\n * and then returns the container.\n *\n * @param dv - The DataviewInlineApi instance.\n * @param renderer - The function responsible for rendering the content.\n * @returns A {@link Promise} that resolves to the HTML paragraph element\n * that was used as the temporary container.\n */\nexport async function getRenderedContainer(dv: DataviewInlineApi, renderer: () => Promisable<void>): Promise<HTMLParagraphElement> {\n  const oldContainer = dv.container;\n  const tempContainer = dv.paragraph('');\n  dv.container = tempContainer;\n  dv.container.empty();\n\n  try {\n    await renderer();\n  } catch (e) {\n    dv.paragraph(`\u274C${errorToString(e)}`);\n  } finally {\n    // eslint-disable-next-line require-atomic-updates -- Yes, it is a potential race condition, but I don't an elegant way to fix it.\n    dv.container = oldContainer;\n    tempContainer.remove();\n  }\n\n  return tempContainer;\n}\n\n/**\n * Inserts a code block into the specified Dataview instance using the provided language and code.\n *\n * @param dv - The DataviewInlineApi instance to insert the code block into.\n * @param language - The language identifier for the code block.\n * @param code - The code content to be inserted into the code block.\n */\nexport function insertCodeBlock(dv: DataviewInlineApi, language: string, code: string): void {\n  const MIN_FENCE_LENGTH = 3;\n  const fenceRegExp = new RegExp(`^\\`{${String(MIN_FENCE_LENGTH)},}`, 'gm');\n  const fenceMatches = code.matchAll(fenceRegExp);\n  const fenceLengths = Array.from(fenceMatches).map((fenceMatch) => fenceMatch[0].length);\n  const maxFenceLength = Math.max(0, ...fenceLengths);\n  const resultFenceLength = Math.max(MIN_FENCE_LENGTH, maxFenceLength + 1);\n  const resultFence = '`'.repeat(resultFenceLength);\n\n  dv.paragraph(`${resultFence}${language}\n${code}\n${resultFence}`);\n}\n\n/**\n * Renders an iframe in the Dataview container with the specified relative path, width, and height.\n *\n * @param options - The options for rendering the iframe.\n */\nexport function renderIframe(options: RenderIframeOptions): void {\n  const {\n    dv,\n    height = '600px',\n    relativePathOrFile,\n    width = '100%'\n  } = options;\n  dv.el('iframe', '', {\n    attr: {\n      height,\n      src: relativePathToResourceUrl(dv.app, getPath(dv.app, relativePathOrFile), dv.current().file.path),\n      width\n    }\n  });\n}\n\n/**\n * Renders a paginated list using the provided DataviewInlineApi instance.\n *\n * @typeParam T - The type of items in the list.\n *\n * @param options - The options for rendering the paginated list.\n *\n * @returns A {@link Promise} that resolves when the list is rendered.\n */\nexport async function renderPaginatedList<T>(options: RenderPaginatedListOptions<T>): Promise<void> {\n  const {\n    dv,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    itemsPerPageOptions = [10, 20, 50, 100],\n    rows\n  } = options;\n  await renderPaginated({\n    dv,\n    itemsPerPageOptions,\n    renderer: async (rowsForOnePage: ArrayOrDataArray<T>): Promise<void> => {\n      await dv.list(rowsForOnePage);\n    },\n    rows\n  });\n}\n\n/**\n * Renders a paginated table using the provided DataviewInlineApi instance.\n *\n * @typeParam T - The type of items in the table rows.\n *\n * @param options - The options for rendering the paginated table.\n *\n * @returns A {@link Promise} that resolves when the table is rendered.\n */\nexport async function renderPaginatedTable<T extends unknown[]>(options: RenderPaginatedTableOptions<T>): Promise<void> {\n  const {\n    dv,\n    headers,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    itemsPerPageOptions = [10, 20, 50, 100],\n    rows\n  } = options;\n  await renderPaginated({\n    dv,\n    itemsPerPageOptions,\n    renderer: async (rowsForOnePage: ArrayOrDataArray<T>): Promise<void> => {\n      await dv.table(headers, rowsForOnePage);\n    },\n    rows\n  });\n}\n\n/**\n * Helper function to render paginated content using the specified renderer.\n *\n * @typeParam T - The type of items to paginate.\n *\n * @param options - The options for rendering the paginated content.\n *\n * @returns A {@link Promise} that resolves when the content is rendered.\n */\nasync function renderPaginated<T>(options: RenderPaginatedOptions<T>): Promise<void> {\n  const SECOND_PAGE_NUMBER = 2;\n  const MORE_PAGE_NUMBER = 3;\n  const {\n    dv,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    itemsPerPageOptions = [10, 20, 50, 100],\n    rows\n  } = options;\n  if (rows.length === 0) {\n    dv.paragraph('No items found');\n    return;\n  }\n  const container = dv.container;\n  let itemsPerPage = itemsPerPageOptions[0] ?? throwExpression(new Error('Items per page options are empty'));\n  let totalPages = Math.ceil(rows.length / itemsPerPage);\n  await renderPage(1);\n\n  function createPaginationControls(pageNumber: number): void {\n    const paginationDiv = container.createDiv({ cls: 'pagination' });\n    const paginationRow1Div = paginationDiv.createDiv();\n\n    createPageLink('First', 1, pageNumber === 1);\n    createPageLink('Prev', pageNumber - 1, pageNumber === 1);\n\n    if (pageNumber > MORE_PAGE_NUMBER) {\n      paginationRow1Div.createSpan({ text: '...' });\n    }\n\n    for (let i = Math.max(1, pageNumber - SECOND_PAGE_NUMBER); i <= Math.min(totalPages, pageNumber + SECOND_PAGE_NUMBER); i++) {\n      const pageLink = createPageLink(String(i), i, i === pageNumber);\n      if (i === pageNumber) {\n        pageLink.addClass('current');\n      }\n    }\n\n    if (pageNumber < totalPages - SECOND_PAGE_NUMBER) {\n      paginationRow1Div.createSpan({ text: '...' });\n    }\n\n    createPageLink('Next', pageNumber + 1, pageNumber === totalPages);\n    createPageLink('Last', totalPages, pageNumber === totalPages);\n\n    const paginationRow2Div = paginationDiv.createDiv();\n\n    paginationRow2Div.createSpan({ text: ` ${t(($) => $.obsidianDevUtils.dataview.itemsPerPage)} ` });\n\n    const itemsPerPageSelect = paginationRow2Div.createEl('select');\n    itemsPerPageOptions.forEach((option: number): void => {\n      itemsPerPageSelect.createEl('option', { text: String(option), value: String(option) });\n    });\n    itemsPerPageSelect.value = String(itemsPerPage);\n    itemsPerPageSelect.addEventListener(\n      'change',\n      convertAsyncToSync(async (): Promise<void> => {\n        itemsPerPage = parseInt(itemsPerPageSelect.value, 10);\n        totalPages = Math.ceil(rows.length / itemsPerPage);\n        await renderPage(1);\n      })\n    );\n\n    paginationRow2Div.createSpan({ text: ` ${t(($) => $.obsidianDevUtils.dataview.jumpToPage)} ` });\n\n    const jumpToPageInput = paginationRow2Div.createEl('input', { attr: { max: totalPages, min: 1 }, type: 'number' });\n    jumpToPageInput.addEventListener(\n      'keydown',\n      convertAsyncToSync(async (event: KeyboardEvent): Promise<void> => {\n        if (event.key === 'Enter') {\n          const page = parseInt(jumpToPageInput.value, 10);\n          if (page >= 1 && page <= totalPages) {\n            await renderPage(page);\n          }\n        }\n      })\n    );\n\n    paginationRow2Div.createSpan({ text: t(($) => $.obsidianDevUtils.dataview.pageHeader, { pageNumber, totalItems: rows.length, totalPages }) });\n\n    function createPageLink(text: string, currentPageNumber: number, disabled = false): HTMLAnchorElement {\n      const link = paginationRow1Div.createEl('a', { cls: 'page-link', href: `#${String(currentPageNumber)}`, text });\n      if (disabled) {\n        link.addClass('disabled');\n        link.onclick = (event: MouseEvent): void => {\n          event.preventDefault();\n        };\n      } else {\n        link.addEventListener(\n          'click',\n          convertAsyncToSync(async (event: MouseEvent): Promise<void> => {\n            event.preventDefault();\n            await renderPage(currentPageNumber);\n          })\n        );\n      }\n      return link;\n    }\n  }\n\n  async function renderPage(pageNumber: number): Promise<void> {\n    container.empty();\n    // eslint-disable-next-line obsidianmd/no-forbidden-elements -- We need to create a style element to apply the pagination CSS.\n    container.createEl('style', { text: paginationCss });\n\n    const startIndex = (pageNumber - 1) * itemsPerPage;\n    const endIndex = startIndex + itemsPerPage;\n    const rowsForCurrentPage = rows.slice(startIndex, endIndex);\n\n    const oldContainer = dv.container;\n\n    dv.container = container;\n    try {\n      await options.renderer(rowsForCurrentPage);\n    } catch (e) {\n      dv.paragraph(`\u274C${errorToString(e)}`);\n    } finally {\n      // eslint-disable-next-line require-atomic-updates -- Yes, it is a potential race condition, but I don't an elegant way to fix it.\n      dv.container = oldContainer;\n    }\n\n    createPaginationControls(pageNumber);\n  }\n}\n"],
  "mappings": "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",
  "names": []
}
