Skip to content

Commit

Permalink
update mobile ad position logic for beta fronts and refactor ad inser…
Browse files Browse the repository at this point in the history
…tion logic into clearer rule definitions
  • Loading branch information
cemms1 committed Jan 8, 2025
1 parent a9e611a commit b654dc4
Showing 1 changed file with 49 additions and 26 deletions.
75 changes: 49 additions & 26 deletions dotcom-rendering/src/lib/getFrontsAdPositions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,10 @@ type GroupedCounts = {
standard: number;
};

type AdCandidate = Pick<DCRCollectionType, 'collectionType' | 'config'>;
export type AdCandidate = Pick<
DCRCollectionType,
'collectionType' | 'containerLevel'
>;

const getMerchHighPosition = (collectionCount: number): number =>
collectionCount >= 4 ? 2 : 0;
Expand All @@ -37,25 +40,37 @@ const isBeforeThrasher = (index: number, collections: AdCandidate[]) =>
const isMostViewedContainer = (collection: AdCandidate) =>
collection.collectionType === 'news/most-popular';

const isSecondaryLevelContainer = (collection: AdCandidate | undefined) =>
collection?.config.containerLevel === 'Secondary';

const isBeforeASecondaryLevelContainer = (
const isBeforeSecondaryLevelContainer = (
index: number,
collections: AdCandidate[],
) => isSecondaryLevelContainer(collections[index + 1]);

/** Detemines whether an ad can be inserted above the current container on mobile */
const shouldInsertAd =
(merchHighPosition: number) =>
(collection: AdCandidate, index: number, collections: AdCandidate[]) =>
!(
isFirstContainerAndThrasher(collection.collectionType, index) ||
isMerchHighPosition(index, merchHighPosition) ||
isBeforeThrasher(index, collections) ||
isMostViewedContainer(collection) ||
isBeforeASecondaryLevelContainer(index, collections)
);
) => collections[index + 1]?.containerLevel === 'Secondary';

const hasSecondaryLevelContainers = (collections: AdCandidate[]) =>
!!collections.find((c) => c.containerLevel === 'Secondary');

/** Determines whether an ad can be inserted below the current container on mobile */
const canInsertAd =
(merchHighPosition: number, hasSecondaryContainers: boolean) =>
(collection: AdCandidate, index: number, collections: AdCandidate[]) => {
/** Base rules are separated from beta ones to avoid changing ad placement logic for fronts without beta containers */
const conditions = [
!isFirstContainerAndThrasher(collection.collectionType, index),
!isMerchHighPosition(index, merchHighPosition),
!isBeforeThrasher(index, collections),
!isMostViewedContainer(collection),
];

/** Beta rules exist for fronts with primary and secondary level containers */
const betaConditions = [
// Allow insertion after first container and prevent insertion before a secondary level container
index === 0 || !isBeforeSecondaryLevelContainer(index, collections),
];

// Ad insertion is possible if every condition is met
return hasSecondaryContainers
? [...conditions, ...betaConditions].every(Boolean)
: conditions.every(Boolean);
};

const isEvenIndex = (_collection: unknown, index: number): boolean =>
index % 2 === 0;
Expand All @@ -66,18 +81,26 @@ const isEvenIndex = (_collection: unknown, index: number): boolean =>
* b. next to the merchandising-high ad slot
* c. before a thrasher
* d. after the Most Viewed container.
* e. before a secondary level container
* After we've filtered out the containers next to which we can insert an ad,
* we take every other container, up to a maximum of 10, for targeting MPU insertion.
* we take every other container (if no secondary level containers exist),
* up to a maximum of 10, for targeting MPU insertion.
*/
const getMobileAdPositions = (collections: AdCandidate[]): number[] => {
const merchHighPosition = getMerchHighPosition(collections.length);

return collections
.filter(shouldInsertAd(merchHighPosition))
.filter(isEvenIndex)
.map((collection: AdCandidate) => collections.indexOf(collection))
.filter((adPosition: number) => adPosition !== -1)
.slice(0, MAX_FRONTS_MOBILE_ADS);
const hasSecondaryContainers = hasSecondaryLevelContainers(collections);

return (
collections
.filter(canInsertAd(merchHighPosition, hasSecondaryContainers))
// Use every other ad position if the front has no secondary containers
.filter((c, i) =>
hasSecondaryContainers ? true : isEvenIndex(c, i),
)
.map((collection: AdCandidate) => collections.indexOf(collection))
.filter((adPosition: number) => adPosition !== -1)
.slice(0, MAX_FRONTS_MOBILE_ADS)
);
};

/**
Expand Down

0 comments on commit b654dc4

Please sign in to comment.