2022-03-06 13:52:19 -05:00
|
|
|
import EmberObject from "@ember/object";
|
2019-05-07 19:52:17 -04:00
|
|
|
import AdComponent from "discourse/plugins/discourse-adplugin/discourse/components/ad-component";
|
2020-03-23 04:40:29 -04:00
|
|
|
import discourseComputed, { observes } from "discourse-common/utils/decorators";
|
2022-03-06 13:52:19 -05:00
|
|
|
import { isBlank } from "@ember/utils";
|
2019-04-18 17:52:59 -04:00
|
|
|
|
2022-03-06 13:52:19 -05:00
|
|
|
const adConfig = EmberObject.create({
|
2019-04-18 17:52:59 -04:00
|
|
|
"google-adsense": {
|
2019-05-07 19:52:17 -04:00
|
|
|
settingPrefix: "adsense", // settings follow naming convention
|
|
|
|
enabledSetting: "adsense_publisher_code",
|
2020-09-04 07:24:14 -04:00
|
|
|
nthPost: "adsense_nth_post_code",
|
2019-04-18 17:52:59 -04:00
|
|
|
},
|
|
|
|
"google-dfp-ad": {
|
2019-05-07 19:52:17 -04:00
|
|
|
settingPrefix: "dfp", // settings follow naming convention
|
|
|
|
enabledSetting: "dfp_publisher_id",
|
2020-09-04 07:24:14 -04:00
|
|
|
nthPost: "dfp_nth_post_code",
|
2019-04-18 17:52:59 -04:00
|
|
|
},
|
|
|
|
"amazon-product-links": {
|
|
|
|
settingPrefix: "amazon",
|
2019-05-07 19:52:17 -04:00
|
|
|
enabledSetting: false,
|
|
|
|
nthPost: "amazon_nth_post_code",
|
2019-04-18 17:52:59 -04:00
|
|
|
desktop: {
|
|
|
|
"topic-list-top": "amazon_topic_list_top_src_code",
|
|
|
|
"post-bottom": "amazon_post_bottom_src_code",
|
|
|
|
"topic-above-post-stream": "amazon_topic_above_post_stream_src_code",
|
2020-09-04 07:24:14 -04:00
|
|
|
"topic-above-suggested": "amazon_topic_above_suggested_src_code",
|
2019-04-18 17:52:59 -04:00
|
|
|
},
|
|
|
|
mobile: {
|
|
|
|
"topic-list-top": "amazon_mobile_topic_list_top_src_code",
|
|
|
|
"post-bottom": "amazon_mobile_post_bottom_src_code",
|
|
|
|
"topic-above-post-stream":
|
|
|
|
"amazon_mobile_topic_above_post_stream_src_code",
|
2020-09-04 07:24:14 -04:00
|
|
|
"topic-above-suggested": "amazon_mobile_topic_above_suggested_src_code",
|
|
|
|
},
|
2019-04-18 17:52:59 -04:00
|
|
|
},
|
|
|
|
"carbonads-ad": {
|
|
|
|
settingPrefix: "carbonads",
|
2019-05-07 19:52:17 -04:00
|
|
|
enabledSetting: "carbonads_serve_id",
|
2019-04-18 17:52:59 -04:00
|
|
|
desktop: {
|
|
|
|
"topic-list-top": "carbonads_topic_list_top_enabled",
|
|
|
|
"post-bottom": false,
|
|
|
|
"topic-above-post-stream": "carbonads_above_post_stream_enabled",
|
2020-09-04 07:24:14 -04:00
|
|
|
"topic-above-suggested": false,
|
|
|
|
},
|
2019-09-16 14:26:17 -04:00
|
|
|
},
|
|
|
|
"adbutler-ad": {
|
|
|
|
settingPrefix: "adbutler",
|
|
|
|
enabledSetting: "adbutler_publisher_id",
|
|
|
|
desktop: {
|
|
|
|
"topic-list-top": "adbutler_topic_list_top_zone_id",
|
|
|
|
"post-bottom": "adbutler_post_bottom_zone_id",
|
|
|
|
"topic-above-post-stream": "adbutler_topic_above_post_stream_zone_id",
|
2020-09-04 07:24:14 -04:00
|
|
|
"topic-above-suggested": "adbutler_topic_above_suggested_zone_id",
|
2019-09-16 14:26:17 -04:00
|
|
|
},
|
|
|
|
mobile: {
|
|
|
|
"topic-list-top": "adbutler_mobile_topic_list_top_zone_id",
|
|
|
|
"post-bottom": "adbutler_mobile_post_bottom_zone_id",
|
|
|
|
"topic-above-post-stream":
|
|
|
|
"adbutler_mobile_topic_above_post_stream_zone_id",
|
2020-09-04 07:24:14 -04:00
|
|
|
"topic-above-suggested": "adbutler_mobile_topic_above_suggested_zone_id",
|
|
|
|
},
|
|
|
|
},
|
2019-04-18 17:52:59 -04:00
|
|
|
});
|
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
const displayCounts = {
|
|
|
|
houseAds: 0,
|
2020-09-04 07:24:14 -04:00
|
|
|
allAds: 0,
|
2019-05-07 19:52:17 -04:00
|
|
|
};
|
2019-05-03 13:26:16 -04:00
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
export default AdComponent.extend({
|
|
|
|
needsUpdate: false,
|
2022-07-07 07:53:29 -04:00
|
|
|
tagName: "",
|
2019-05-03 13:26:16 -04:00
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
/**
|
|
|
|
* For a given ad placement and optionally a post number if in between posts,
|
|
|
|
* list all ad network names that are configured to show there.
|
|
|
|
*/
|
2022-07-07 07:53:29 -04:00
|
|
|
@discourseComputed("placement", "postNumber", "indexNumber")
|
|
|
|
availableAdTypes(placement, postNumber, indexNumber) {
|
2019-05-07 19:52:17 -04:00
|
|
|
let types = [];
|
|
|
|
const houseAds = this.site.get("house_creatives"),
|
|
|
|
placeUnderscored = placement.replace(/-/g, "_");
|
|
|
|
|
|
|
|
if (houseAds && houseAds.settings) {
|
|
|
|
const adsForSlot = houseAds.settings[placeUnderscored];
|
|
|
|
|
2022-07-07 07:53:29 -04:00
|
|
|
const adAvailable =
|
|
|
|
Object.keys(houseAds.creatives).length > 0 && !isBlank(adsForSlot);
|
|
|
|
|
|
|
|
// postNumber and indexNumber are both null for topic-list-top, topic-above-post-stream,
|
|
|
|
// and topic-above-suggested placements. Assume we want to place an ad outside the topic list.
|
|
|
|
const notPlacingBetweenTopics = !postNumber && !indexNumber;
|
|
|
|
|
|
|
|
const canBePlacedInBetweenTopics =
|
|
|
|
placeUnderscored === "topic_list_between" &&
|
|
|
|
this.isNthTopicListItem(
|
|
|
|
parseInt(houseAds.settings.after_nth_topic, 10)
|
|
|
|
);
|
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
if (
|
2022-07-07 07:53:29 -04:00
|
|
|
adAvailable &&
|
|
|
|
(notPlacingBetweenTopics ||
|
|
|
|
this.isNthPost(parseInt(houseAds.settings.after_nth_post, 10)) ||
|
|
|
|
canBePlacedInBetweenTopics)
|
2019-05-07 19:52:17 -04:00
|
|
|
) {
|
|
|
|
types.push("house-ad");
|
|
|
|
}
|
2019-04-18 17:52:59 -04:00
|
|
|
}
|
|
|
|
|
2020-09-04 07:24:14 -04:00
|
|
|
Object.keys(adConfig).forEach((adNetwork) => {
|
2019-04-18 17:52:59 -04:00
|
|
|
const config = adConfig[adNetwork];
|
|
|
|
let settingNames = null,
|
|
|
|
name;
|
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
if (
|
2022-07-07 07:53:29 -04:00
|
|
|
this._isNetworkAvailable(config.enabledSetting) &&
|
|
|
|
this._shouldPlaceAdInSlot(postNumber, config.nthPost)
|
2019-05-07 19:52:17 -04:00
|
|
|
) {
|
|
|
|
if (this.site.mobileView) {
|
|
|
|
settingNames = config.mobile || config.desktop;
|
|
|
|
} else {
|
|
|
|
settingNames = config.desktop;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (settingNames) {
|
|
|
|
name = settingNames[placement];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (name === undefined) {
|
|
|
|
// follows naming convention: prefix_(mobile_)_{placement}_code
|
|
|
|
name = `${config.settingPrefix}_${
|
|
|
|
this.site.mobileView ? "mobile_" : ""
|
|
|
|
}${placeUnderscored}_code`;
|
|
|
|
}
|
2019-04-18 17:52:59 -04:00
|
|
|
|
2019-07-17 14:05:45 -04:00
|
|
|
if (
|
|
|
|
name !== false &&
|
|
|
|
this.siteSettings[name] !== false &&
|
2022-03-06 13:52:19 -05:00
|
|
|
!isBlank(this.siteSettings[name])
|
2019-07-17 14:05:45 -04:00
|
|
|
) {
|
2019-05-07 19:52:17 -04:00
|
|
|
types.push(adNetwork);
|
|
|
|
}
|
2019-04-18 17:52:59 -04:00
|
|
|
}
|
2019-05-07 19:52:17 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
return types;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When house ads are configured to alternate with other ad networks, we
|
|
|
|
* need to trigger an update of which ad component is shown after
|
|
|
|
* navigating between topic lists or topics.
|
|
|
|
*/
|
|
|
|
@observes("refreshOnChange")
|
|
|
|
changed() {
|
|
|
|
if (this.get("listLoading")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// force adComponents to be recomputed
|
|
|
|
this.notifyPropertyChange("needsUpdate");
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of the names of ad components that should be rendered
|
|
|
|
* in the given ad placement. It handles alternating between house ads
|
|
|
|
* and other ad networks.
|
|
|
|
*/
|
2020-03-23 04:40:29 -04:00
|
|
|
@discourseComputed("placement", "availableAdTypes", "needsUpdate")
|
2019-05-07 19:52:17 -04:00
|
|
|
adComponents(placement, availableAdTypes) {
|
|
|
|
if (
|
|
|
|
!availableAdTypes.includes("house-ad") ||
|
|
|
|
availableAdTypes.length === 1
|
|
|
|
) {
|
|
|
|
// Current behaviour is to allow multiple ads from different networks
|
|
|
|
// to show in the same place. We could change this to choose one somehow.
|
|
|
|
return availableAdTypes;
|
|
|
|
}
|
|
|
|
|
|
|
|
const houseAds = this.site.get("house_creatives");
|
|
|
|
let houseAdsSkipped = false;
|
2019-04-18 17:52:59 -04:00
|
|
|
|
2019-05-07 19:52:17 -04:00
|
|
|
if (houseAds.settings.house_ads_frequency === 100) {
|
|
|
|
// house always wins
|
|
|
|
return ["house-ad"];
|
|
|
|
} else if (houseAds.settings.house_ads_frequency > 0) {
|
|
|
|
// show house ads the given percent of the time
|
|
|
|
if (
|
|
|
|
displayCounts.allAds === 0 ||
|
|
|
|
(100 * displayCounts.houseAds) / displayCounts.allAds <
|
|
|
|
houseAds.settings.house_ads_frequency
|
|
|
|
) {
|
|
|
|
displayCounts.houseAds += 1;
|
|
|
|
displayCounts.allAds += 1;
|
|
|
|
return ["house-ad"];
|
|
|
|
} else {
|
|
|
|
houseAdsSkipped = true;
|
2019-04-18 17:52:59 -04:00
|
|
|
}
|
2019-05-07 19:52:17 -04:00
|
|
|
}
|
2019-04-18 17:52:59 -04:00
|
|
|
|
2020-09-04 07:24:14 -04:00
|
|
|
const networkNames = availableAdTypes.filter((x) => x !== "house-ad");
|
2019-05-07 19:52:17 -04:00
|
|
|
|
|
|
|
if (houseAdsSkipped) {
|
|
|
|
displayCounts.allAds += networkNames.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return networkNames;
|
2020-09-04 07:24:14 -04:00
|
|
|
},
|
2022-07-07 07:53:29 -04:00
|
|
|
|
|
|
|
_isNetworkAvailable(enabledNetworkSettingName) {
|
|
|
|
// False means there's no setting to enable or disable this ad network.
|
|
|
|
// Assume it's always enabled.
|
|
|
|
if (enabledNetworkSettingName === false) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return (
|
|
|
|
enabledNetworkSettingName &&
|
|
|
|
!isBlank(this.siteSettings[enabledNetworkSettingName])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_shouldPlaceAdInSlot(currentPostNumber, positionToPlace) {
|
|
|
|
return (
|
|
|
|
!currentPostNumber ||
|
|
|
|
!positionToPlace ||
|
|
|
|
this.isNthPost(parseInt(this.siteSettings[positionToPlace], 10))
|
|
|
|
);
|
|
|
|
},
|
2019-04-18 17:52:59 -04:00
|
|
|
});
|