fix: stabilize market metrics hydration and sorting

This commit is contained in:
admin123 2026-04-23 14:35:14 +08:00
parent a51c6f7bf2
commit 233de28713
7 changed files with 362 additions and 45 deletions

View File

@ -61,6 +61,7 @@ const SORTABLE_MARKET_FIELDS = [
type RowOrderTarget = {
container: HTMLElement;
mode: "css" | "dom";
node: HTMLElement;
};
@ -106,12 +107,7 @@ export function readMarketPageSignature(root: ParentNode): string {
document
?.querySelector(".el-pagination .number.active, .xt-pagination .number.active")
?.textContent?.trim() ?? "";
const table = syncMarketTable(root);
const authorIds =
table?.rows
.map((row) => row.authorId)
.filter((authorId) => Boolean(authorId))
.join("|") ?? "";
const authorIds = readRawAuthorIds(root).join("|");
return `${explicitPageIndex || activePageIndex}::${authorIds}`;
}
@ -209,6 +205,9 @@ export function applyRowOrder(
orderedAuthorIds: string[]
): void {
const rowById = new Map(table.rows.map((rowDom) => [rowDom.authorId, rowDom]));
const orderByAuthorId = new Map(
orderedAuthorIds.map((authorId, index) => [authorId, index])
);
orderedAuthorIds.forEach((authorId) => {
const rowDom = rowById.get(authorId);
@ -216,7 +215,17 @@ export function applyRowOrder(
return;
}
rowDom.orderTargets.forEach(({ container, node }) => {
rowDom.orderTargets.forEach(({ container, mode, node }) => {
const visualOrder = orderByAuthorId.get(authorId) ?? orderedAuthorIds.length;
if (mode === "css") {
container.dataset.marketOrderMode = "css";
container.style.display = "flex";
container.style.flexDirection = "column";
node.style.order = String(visualOrder);
return;
}
container.dataset.marketOrderMode = "dom";
container.appendChild(node);
});
});
@ -283,6 +292,7 @@ function syncSyntheticMarketTable(root: ParentNode): MarketTableDom | null {
orderTargets: [
{
container: body,
mode: "dom",
node: row
}
],
@ -324,6 +334,75 @@ function syncDivGridMarketTable(root: ParentNode): MarketTableDom | null {
return null;
}
function readRawAuthorIds(root: ParentNode): string[] {
const document = getOwnerDocument(root);
const syntheticAuthorIds = readSyntheticAuthorIds(root);
if (syntheticAuthorIds && syntheticAuthorIds.length > 0) {
return syntheticAuthorIds;
}
const divGridAuthorIds = readDivGridAuthorIds(root);
if (divGridAuthorIds && divGridAuthorIds.length > 0) {
return divGridAuthorIds;
}
if (!document) {
return [];
}
return readSerializedMarketRows(document)
.map((row) => row.authorId)
.filter((authorId) => Boolean(authorId));
}
function readSyntheticAuthorIds(root: ParentNode): string[] | null {
const body = root.querySelector("[data-market-body]") as HTMLElement | null;
if (!body) {
return null;
}
return Array.from(body.querySelectorAll("[data-market-row]"))
.map((row) =>
row instanceof HTMLElement ? row.dataset.authorId ?? "" : ""
)
.filter((authorId) => Boolean(authorId));
}
function readDivGridAuthorIds(root: ParentNode): string[] | null {
const document = getOwnerDocument(root);
if (!document) {
return null;
}
const marketRoot = document.querySelector(".base-author-list");
if (!(marketRoot instanceof document.defaultView!.HTMLElement)) {
return null;
}
const bodySection = Array.from(marketRoot.querySelectorAll(".section-wrapper")).find(
(section): section is HTMLElement =>
section instanceof document.defaultView!.HTMLElement &&
!section.classList.contains("sticky-header")
);
const authorSection = bodySection
? Array.from(bodySection.children).find(
(child): child is HTMLElement =>
child instanceof document.defaultView!.HTMLElement &&
child.querySelector(".content-column .content-cell")
) ?? null
: null;
const authorColumn = authorSection
? getDirectContentColumns(authorSection)[0] ?? null
: null;
if (!authorColumn) {
return null;
}
return getDirectContentCells(authorColumn)
.map((cell) => extractAuthorId(cell))
.filter((authorId) => Boolean(authorId));
}
function syncDivGridRoot(root: HTMLElement): MarketTableDom | null {
const headerSection = root.querySelector(
".section-wrapper.sticky-header"
@ -520,6 +599,7 @@ function syncDivGridRoot(root: HTMLElement): MarketTableDom | null {
return {
container,
mode: "css",
node: cell
};
})

View File

@ -10,6 +10,7 @@ interface ExportRangeControllerOptions {
onProgress?: (state: { currentPage: number; totalPages?: number }) => void;
prepareCurrentPageForExport(): Promise<void>;
readCurrentPageRecords(): MarketRecord[];
readCurrentPageRowCount(): number;
window: Window;
}
@ -26,13 +27,10 @@ export function createExportRangeController(options: ExportRangeControllerOption
currentPage,
totalPages: target.mode === "count" ? target.pageCount : undefined
});
const currentPageReady = await waitForCurrentPageReady(expectedMinimumRowCount);
if (!currentPageReady) {
const currentPageRecords = await preparePageRecords(expectedMinimumRowCount);
if (!currentPageRecords) {
throw new Error(`${currentPage} 页加载超时,请稍后重试`);
}
await options.prepareCurrentPageForExport();
const currentPageRecords = options.readCurrentPageRecords();
currentPageRecords.forEach((record) => {
const existingRecord = mergedRecords.get(record.authorId);
mergedRecords.set(record.authorId, mergeMarketRecord(existingRecord, record));
@ -63,6 +61,33 @@ export function createExportRangeController(options: ExportRangeControllerOption
}
};
async function preparePageRecords(
expectedMinimumRowCount: number | undefined
): Promise<MarketRecord[] | null> {
for (let attempt = 0; attempt < 4; attempt += 1) {
const currentPageReady = await waitForCurrentPageReady();
if (!currentPageReady) {
return null;
}
await options.prepareCurrentPageForExport();
const currentPageRecords = options.readCurrentPageRecords();
if (
currentPageRecords.length > 0 &&
(
typeof expectedMinimumRowCount !== "number" ||
expectedMinimumRowCount <= 0 ||
isCurrentPageTerminal() ||
currentPageRecords.length >= expectedMinimumRowCount
)
) {
return currentPageRecords;
}
}
return null;
}
async function waitForPageChange(previousSignature: string): Promise<boolean> {
const previousPageState = parsePageSignature(previousSignature);
@ -82,9 +107,7 @@ export function createExportRangeController(options: ExportRangeControllerOption
return false;
}
async function waitForCurrentPageReady(
expectedMinimumRowCount: number | undefined
): Promise<boolean> {
async function waitForCurrentPageReady(): Promise<boolean> {
let stableAttemptCount = 0;
let lastReadyFingerprint = "";
@ -101,17 +124,6 @@ export function createExportRangeController(options: ExportRangeControllerOption
continue;
}
if (
typeof expectedMinimumRowCount === "number" &&
expectedMinimumRowCount > 0 &&
!pageState.isTerminalPage &&
pageState.rowCount < expectedMinimumRowCount
) {
stableAttemptCount = 0;
lastReadyFingerprint = "";
continue;
}
const readyFingerprint = [
pageState.pageToken,
pageState.authorIds,
@ -146,9 +158,13 @@ export function createExportRangeController(options: ExportRangeControllerOption
authorIds: pageSignature.authorIds,
isTerminalPage: isPageControlDisabled(nextPageControl),
pageToken: pageSignature.pageToken,
rowCount: options.readCurrentPageRecords().length
rowCount: options.readCurrentPageRowCount()
};
}
function isCurrentPageTerminal(): boolean {
return isPageControlDisabled(findNextPageControl(options.document));
}
}
function parsePageSignature(signature: string): {

View File

@ -89,7 +89,7 @@ function compareRateSortRecords(
const rightLowerBound = parseRateLowerBound(rightValue ?? null);
if (leftLowerBound == null && rightLowerBound == null) {
return 0;
return compareRecordIdentity(leftRecord, rightRecord);
}
if (leftLowerBound == null) {
@ -107,7 +107,11 @@ function compareRateSortRecords(
}
const tieBreak = compareRateValues(leftValue, rightValue);
return sort.direction === "asc" ? tieBreak : -tieBreak;
if (tieBreak !== 0) {
return sort.direction === "asc" ? tieBreak : -tieBreak;
}
return compareRecordIdentity(leftRecord, rightRecord);
}
function compareBackendMetricRecords(
@ -120,7 +124,7 @@ function compareBackendMetricRecords(
const rightValue = parseBackendMetricValue(rightRecord.backendMetrics?.[field]);
if (leftValue == null && rightValue == null) {
return 0;
return compareRecordIdentity(leftRecord, rightRecord);
}
if (leftValue == null) {
@ -131,7 +135,11 @@ function compareBackendMetricRecords(
return -1;
}
return sort.direction === "asc" ? leftValue - rightValue : rightValue - leftValue;
if (leftValue !== rightValue) {
return sort.direction === "asc" ? leftValue - rightValue : rightValue - leftValue;
}
return compareRecordIdentity(leftRecord, rightRecord);
}
function parseBackendMetricValue(value: string | null | undefined): number | null {
@ -156,3 +164,15 @@ function isRateSortField(
field === "personalVideoAfterSearchRate"
);
}
function compareRecordIdentity(
leftRecord: MarketRecord,
rightRecord: MarketRecord
): number {
const authorIdCompare = leftRecord.authorId.localeCompare(rightRecord.authorId);
if (authorIdCompare !== 0) {
return authorIdCompare;
}
return leftRecord.authorName.localeCompare(rightRecord.authorName);
}

View File

@ -3,6 +3,7 @@ import { createBatchPayload, type BatchPayload } from "./batch-payload";
import {
applyRowOrder,
applyRowVisibility,
readMarketPageSignature,
renderMarketRowState,
syncPluginSortHeaders,
syncMarketTable,
@ -97,8 +98,14 @@ export function createMarketController(options: CreateMarketControllerOptions) {
let activeSort: MarketSortState | undefined;
let isSyncRunning = false;
let isSyncScheduled = false;
let lastKnownPageSignature = "";
let needsResync = false;
const observer = mutationObserverFactory(() => {
const nextPageSignature = readMarketPageSignature(options.document);
if (nextPageSignature === lastKnownPageSignature) {
return;
}
scheduleSync();
});
const observationRoot = options.document.body ?? options.document.documentElement;
@ -374,6 +381,7 @@ export function createMarketController(options: CreateMarketControllerOptions) {
const records = getVisibleOrderedRecords(table);
applyRowVisibility(table, new Set(records.map((record) => record.authorId)));
applyRowOrder(table, records.map((record) => record.authorId));
lastKnownPageSignature = readMarketPageSignature(options.document);
});
}
@ -719,6 +727,7 @@ export function createMarketController(options: CreateMarketControllerOptions) {
try {
await hydrateCurrentPage();
applyCurrentView();
lastKnownPageSignature = readMarketPageSignature(options.document);
} finally {
isSyncRunning = false;
if (needsResync) {

View File

@ -121,4 +121,67 @@ describe("filter-sort-controller", () => {
expect(result.map((record) => record.authorId)).toEqual(["b", "a", "c"]);
});
test("keeps equal rate buckets in a deterministic order across repeated sorts", () => {
const records: MarketRecord[] = [
{
authorId: "b",
authorName: "Beta",
status: "success",
rates: {
singleVideoAfterSearchRate: "0.5% - 1%"
}
},
{
authorId: "a",
authorName: "Alpha",
status: "success",
rates: {
singleVideoAfterSearchRate: "0.5% - 1%"
}
},
{
authorId: "d",
authorName: "Delta",
status: "success",
rates: {
singleVideoAfterSearchRate: "0.25% - 0.5%"
}
},
{
authorId: "c",
authorName: "Gamma",
status: "success",
rates: {
singleVideoAfterSearchRate: "0.25% - 0.5%"
}
}
];
const firstResult = applyFilterAndSort(records, {
sort: {
direction: "desc",
field: "singleVideoAfterSearchRate"
}
});
const secondResult = applyFilterAndSort([...records].reverse(), {
sort: {
direction: "desc",
field: "singleVideoAfterSearchRate"
}
});
expect(firstResult.map((record) => record.authorId)).toEqual([
"a",
"b",
"c",
"d"
]);
expect(secondResult.map((record) => record.authorId)).toEqual([
"a",
"b",
"c",
"d"
]);
});
});

View File

@ -511,6 +511,86 @@ describe("market-content-entry", () => {
]);
});
test("rehydrates real rows after serialized market rows arrive later", async () => {
document.body.innerHTML = buildRealMarketFixtureWithoutAuthorIds([
{
authorName: "达人 A",
price21To60s: "¥450,000"
},
{
authorName: "达人 B",
price21To60s: "¥20,000"
}
]);
const loadAuthorMetrics = vi.fn(async (authorId: string) => ({
success: true as const,
rates:
authorId === "111"
? {
singleVideoAfterSearchRate: "0.02%",
personalVideoAfterSearchRate: "0.03% - 0.2%"
}
: {
singleVideoAfterSearchRate: "0.5%-1%",
personalVideoAfterSearchRate: "0.01% - 0.1%"
}
}));
const { createMarketController } = await import("../src/content/market/index");
const controller = trackController(createMarketController({
document,
loadAuthorMetrics,
window
}));
await controller.ready;
expect(loadAuthorMetrics).not.toHaveBeenCalled();
expect(readDivPluginRowTexts(0)).toEqual(["", "", "", "", "", "", "", ""]);
expect(readDivPluginRowTexts(1)).toEqual(["", "", "", "", "", "", "", ""]);
document.documentElement.setAttribute(
"data-sces-market-rows",
JSON.stringify([
{
authorId: "111",
authorName: "达人 A",
singleVideoAfterSearchRate: "0.02%"
},
{
authorId: "222",
authorName: "达人 B"
}
])
);
await flushWithTimers();
await flushWithTimers();
expect(loadAuthorMetrics).toHaveBeenCalledTimes(2);
expect(readDivPluginRowTexts(0)).toEqual([
"0.02%",
"0.03% - 0.2%",
"",
"",
"",
"",
"",
""
]);
expect(readDivPluginRowTexts(1)).toEqual([
"0.5% - 1%",
"0.01% - 0.1%",
"",
"",
"",
"",
"",
""
]);
});
test("applying plugin filters hides non-matching current-page rows without a full scan", async () => {
document.body.innerHTML = buildMarketFixture();
const resultStore = createMarketResultStore();
@ -3113,28 +3193,47 @@ function readRowOrder() {
}
function readDivAuthorOrder() {
return Array.from(
document.querySelectorAll('[data-testid^="author-cell-"] a'),
(link) => link.textContent?.trim() ?? ""
const authorColumn = document.querySelector(
'[data-testid="author-section"] .content-column'
);
return readVisualCells(authorColumn).map(
(cell) => cell.querySelector("a")?.textContent?.trim() ?? ""
);
}
function readDivRightRowTexts(rowIndex: number) {
return Array.from(
document.querySelectorAll('[data-testid="right-section"] > .content-column'),
(column) =>
column.querySelectorAll(".content-cell")[rowIndex]?.textContent?.trim() ?? ""
(column) => readVisualCells(column as Element)[rowIndex]?.textContent?.trim() ?? ""
);
}
function readDivPluginRowTexts(rowIndex: number) {
return Array.from(
document.querySelectorAll('[data-testid="plugin-section"] > .content-column'),
(column) =>
column.querySelectorAll(".content-cell")[rowIndex]?.textContent?.trim() ?? ""
(column) => readVisualCells(column as Element)[rowIndex]?.textContent?.trim() ?? ""
);
}
function readVisualCells(root: Element | null): HTMLElement[] {
if (!root) {
return [];
}
return Array.from(root.querySelectorAll(":scope > .content-cell"))
.filter((cell): cell is HTMLElement => cell instanceof HTMLElement)
.sort((left, right) => {
const leftOrder = Number(left.style.order || "0");
const rightOrder = Number(right.style.order || "0");
if (leftOrder !== rightOrder) {
return leftOrder - rightOrder;
}
const cells = Array.from(root.querySelectorAll(":scope > .content-cell"));
return cells.indexOf(left) - cells.indexOf(right);
});
}
function trackController<T extends { dispose?: () => void }>(controller: T): T {
if (controller.dispose) {
disposers.push(() => controller.dispose?.());

View File

@ -553,6 +553,7 @@ describe("market-dom-sync", () => {
expect(table.rows[0].rates).toEqual({
singleVideoAfterSearchRate: "0.02%"
});
expect(readMarketPageSignature(document)).toContain("::111|222");
});
test("finds the real next-page button in Xingtu pagination", () => {
@ -581,6 +582,16 @@ describe("market-dom-sync", () => {
expect(isPageControlDisabled(nextControl)).toBe(false);
expect(readMarketPageSignature(document)).toContain("1::111|222");
});
test("reads market page signature without mutating the page", () => {
document.body.innerHTML = buildRealMarketGridFixture();
const signature = readMarketPageSignature(document);
expect(signature).toContain("::111|222");
expect(document.querySelector('[data-testid="plugin-header"]')).toBeNull();
expect(document.querySelector('[data-testid="plugin-section"]')).toBeNull();
});
});
function buildRealMarketGridFixture() {
@ -898,16 +909,14 @@ function readPluginHeaderTexts() {
function readRightRowTexts(rowIndex: number) {
return Array.from(
document.querySelectorAll('[data-testid="right-section"] > .content-column'),
(column) =>
column.querySelectorAll(".content-cell")[rowIndex]?.textContent?.trim() ?? ""
(column) => readVisualCells(column as Element)[rowIndex]?.textContent?.trim() ?? ""
);
}
function readPluginRowTexts(rowIndex: number) {
return Array.from(
document.querySelectorAll('[data-testid="plugin-section"] > .content-column'),
(column) =>
column.querySelectorAll(".content-cell")[rowIndex]?.textContent?.trim() ?? ""
(column) => readVisualCells(column as Element)[rowIndex]?.textContent?.trim() ?? ""
);
}
@ -918,9 +927,11 @@ function readScrollHintText() {
}
function readAuthorNames() {
return Array.from(
document.querySelectorAll('[data-testid="author-section"] .content-cell a'),
(link) => link.textContent?.trim() ?? ""
const authorColumn = document.querySelector(
'[data-testid="author-section"] .content-column'
);
return readVisualCells(authorColumn).map(
(cell) => cell.querySelector("a")?.textContent?.trim() ?? ""
);
}
@ -937,3 +948,22 @@ function readRightActionHiddenStates() {
(cell) => (cell as HTMLElement).hidden
);
}
function readVisualCells(root: Element | null): HTMLElement[] {
if (!root) {
return [];
}
return Array.from(root.querySelectorAll(":scope > .content-cell"))
.filter((cell): cell is HTMLElement => cell instanceof HTMLElement)
.sort((left, right) => {
const leftOrder = Number(left.style.order || "0");
const rightOrder = Number(right.style.order || "0");
if (leftOrder !== rightOrder) {
return leftOrder - rightOrder;
}
const cells = Array.from(root.querySelectorAll(":scope > .content-cell"));
return cells.indexOf(left) - cells.indexOf(right);
});
}