Cursor Usage: show Requests + Cost

Add a Cost column after Requests on Cursor Usage page while keeping the original Requests column.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey, Greasemonkey или Violentmonkey.

Для установки этого скрипта вам необходимо установить расширение, такое как Tampermonkey.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey или Violentmonkey.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey или Userscripts.

Чтобы установить этот скрипт, сначала вы должны установить расширение браузера, например Tampermonkey.

Чтобы установить этот скрипт, вы должны установить расширение — менеджер скриптов.

(у меня уже есть менеджер скриптов, дайте мне установить скрипт!)

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

(у меня уже есть менеджер стилей, дайте мне установить скрипт!)

// ==UserScript==
// @name         Cursor Usage: show Requests + Cost
// @namespace    https://cursor.com/
// @version      0.1.2
// @description  Add a Cost column after Requests on Cursor Usage page while keeping the original Requests column.
// @match        https://cursor.com/dashboard/usage*
// @match        https://cursor.com/*/dashboard/usage*
// @run-at       document-start
// @grant        none
// @license MIT
// ==/UserScript==

(function () {
  'use strict';

  const PATCH_KEY = '__tmCursorUsageRequestsAndCost';

  const keyOf = (event) =>
    [
      event?.timestamp,
      event?.model,
      event?.requestsCosts,
      event?.owningUser || '',
    ].join('|');

  const formatCost = (cents) => {
    if (typeof cents !== 'number' || Number.isNaN(cents)) return '-';
    return new Intl.NumberFormat('en-US', {
      style: 'currency',
      currency: 'USD',
      minimumFractionDigits: 2,
      maximumFractionDigits: 2,
    }).format(cents / 100);
  };

  function install() {
    if (window[PATCH_KEY]?.cleanup) {
      window[PATCH_KEY].cleanup();
    }

    const state = {
      rawByKey: new Map(),
      observer: null,
      renderTimer: null,
      originalFetch: window.fetch.bind(window),
    };

    function getFiberRoot() {
      const table = document.querySelector('.dashboard-table-scroll-container');
      if (!table) return null;

      const fiberKey = Object.keys(table).find((k) => k.startsWith('__reactFiber$'));
      if (!fiberKey) return null;

      let fiber = table[fiberKey];
      while (fiber?.return) fiber = fiber.return;
      return fiber || null;
    }

    function walkFiber(visit) {
      const root = getFiberRoot();
      if (!root) return;

      const seen = new Set();

      const dfs = (node) => {
        if (!node || seen.has(node)) return;
        seen.add(node);
        visit(node);
        dfs(node.child);
        dfs(node.sibling);
      };

      dfs(root);
    }

    function getDisplayData() {
      let data = null;

      walkFiber((node) => {
        const props = node.memoizedProps;
        if (!data && props && Array.isArray(props.data) && Array.isArray(props.columns)) {
          data = props.data;
        }
      });

      return data;
    }

    function getCustomRange() {
      let range = null;

      walkFiber((node) => {
        const props = node.memoizedProps;
        if (!range && props?.customRange?.from && props?.customRange?.to) {
          range = props.customRange;
        }
      });

      return range;
    }

    function getPageSize() {
      const btn = [...document.querySelectorAll('button')].find((el) =>
        /^Rows:\s*\d+$/i.test((el.textContent || '').trim())
      );
      const match = btn?.textContent?.match(/(\d+)/);
      return match ? Number(match[1]) : 100;
    }

    function render() {
      const table = document.querySelector('.dashboard-table-scroll-container');
      const container = table?.querySelector('.dashboard-table-container');
      const headerRow = container?.querySelector('.dashboard-table-header-row');
      const rows = [...(container?.querySelectorAll('.dashboard-table-row') || [])];
      const displayData = getDisplayData();

      if (!table || !container || !headerRow || !rows.length || !displayData?.length) {
        return;
      }

      table.setAttribute('aria-colcount', '6');

      const minWidth = parseInt(container.style.minWidth || '761', 10);
      if (!Number.isNaN(minWidth) && minWidth < 881) {
        container.style.minWidth = '881px';
      }

      let headerCell = headerRow.querySelector('[data-tm-cost-header="1"]');
      if (!headerCell) {
        headerCell = document.createElement('div');
        headerCell.setAttribute('role', 'columnheader');
        headerCell.setAttribute('data-tm-cost-header', '1');
        headerCell.className = 'dashboard-table-header dashboard-table-header-align-right';
        headerCell.style.width = '120px';
        headerCell.style.flexShrink = '0';
        headerCell.style.flexGrow = '0';

        const span = document.createElement('span');
        span.className = 'truncate min-w-0';
        span.textContent = 'Cost';
        headerCell.appendChild(span);

        const requestsHeader = [...headerRow.children].find(
          (el) => el.textContent.trim() === 'Requests'
        );

        headerRow.insertBefore(headerCell, requestsHeader?.nextSibling || null);
      }

      rows.forEach((row, index) => {
        const displayEvent = displayData[index];
        if (!displayEvent) return;

        const raw = state.rawByKey.get(keyOf(displayEvent));
        const cents = raw?.tokenUsage?.totalCents;

        let cell = row.querySelector('[data-tm-cost-cell="1"]');
        if (!cell) {
          cell = document.createElement('div');
          cell.setAttribute('role', 'cell');
          cell.setAttribute('data-tm-cost-cell', '1');
          cell.className = 'dashboard-table-cell dashboard-table-cell-align-right';
          cell.style.width = '120px';
          cell.style.flexShrink = '0';
          cell.style.flexGrow = '0';
          cell.style.color = 'var(--text-secondary)';

          const cells = [...row.querySelectorAll(':scope > [role="cell"]')];
          const requestsCell = cells[cells.length - 1];
          row.insertBefore(cell, requestsCell?.nextSibling || null);
        }

        cell.textContent = formatCost(cents);
        cell.title = typeof cents === 'number' ? `${cents.toFixed(4)} cents` : '';
      });
    }

    function scheduleRender() {
      clearTimeout(state.renderTimer);
      state.renderTimer = setTimeout(render, 0);
    }

    function recordRawUsage(payload) {
      const events = payload?.usageEventsDisplay;
      if (!Array.isArray(events)) return;

      state.rawByKey.clear();
      for (const event of events) {
        state.rawByKey.set(keyOf(event), event);
      }

      scheduleRender();
    }

    window.fetch = async (...args) => {
      const response = await state.originalFetch(...args);

      try {
        const input = args[0];
        const url = typeof input === 'string' ? input : input?.url;

        if (url && url.includes('/api/dashboard/get-filtered-usage-events')) {
          response.clone().json().then(recordRawUsage).catch(() => {});
        }
      } catch {}

      return response;
    };

    state.observer = new MutationObserver(() => scheduleRender());
    state.observer.observe(document.documentElement, {
      childList: true,
      subtree: true,
    });

    (async () => {
      try {
        const range = getCustomRange();
        if (!range?.from || !range?.to) return;

        const pageSize = getPageSize();
        const res = await state.originalFetch('/api/dashboard/get-filtered-usage-events', {
          method: 'POST',
          headers: { 'content-type': 'application/json' },
          credentials: 'same-origin',
          body: JSON.stringify({
            teamId: 0,
            startDate: String(new Date(range.from).getTime()),
            endDate: String(new Date(range.to).getTime()),
            page: 1,
            pageSize,
          }),
        });

        const payload = await res.json();
        recordRawUsage(payload);
      } catch {}
    })();

    state.cleanup = () => {
      clearTimeout(state.renderTimer);
      state.observer?.disconnect();
      window.fetch = state.originalFetch;

      document
        .querySelectorAll('[data-tm-cost-cell="1"], [data-tm-cost-header="1"]')
        .forEach((el) => el.remove());

      const table = document.querySelector('.dashboard-table-scroll-container');
      if (table) table.setAttribute('aria-colcount', '5');
    };

    window[PATCH_KEY] = state;
  }

  install();
})();