2021-05-21 20:22:01 +03:00
|
|
|
import Component from '@glimmer/component';
|
|
|
|
import flattenGroupedOptions from 'ghost-admin/utils/flatten-grouped-options';
|
|
|
|
import {action} from '@ember/object';
|
2022-01-19 17:29:49 +03:00
|
|
|
import {isBlank} from '@ember/utils';
|
2021-05-21 20:22:01 +03:00
|
|
|
import {inject as service} from '@ember/service';
|
2022-02-09 13:49:38 +03:00
|
|
|
import {task} from 'ember-concurrency';
|
2021-05-21 20:22:01 +03:00
|
|
|
import {tracked} from '@glimmer/tracking';
|
|
|
|
|
|
|
|
const BASE_FILTERS = ['status:free', 'status:-free'];
|
|
|
|
|
|
|
|
export default class GhMembersRecipientSelect extends Component {
|
|
|
|
@service membersUtils;
|
|
|
|
@service store;
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
@tracked forceSpecificChecked = false;
|
2021-05-21 20:22:01 +03:00
|
|
|
@tracked specificOptions = [];
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super(...arguments);
|
|
|
|
|
|
|
|
this.fetchSpecificOptionsTask.perform();
|
|
|
|
}
|
|
|
|
|
2022-05-05 13:18:41 +03:00
|
|
|
get renderInPlace() {
|
|
|
|
return this.args.renderInPlace === undefined ? true : this.args.renderInPlace;
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
get baseFilters() {
|
|
|
|
const filterItems = (this.args.filter || '').split(',');
|
|
|
|
const filterItemsArray = filterItems.filter(item => BASE_FILTERS.includes(item?.trim()));
|
|
|
|
return new Set(filterItemsArray);
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
get isFreeChecked() {
|
|
|
|
return this.baseFilters.has('status:free');
|
|
|
|
}
|
|
|
|
|
|
|
|
get isPaidChecked() {
|
|
|
|
return this.baseFilters.has('status:-free');
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
get isPaidAvailable() {
|
|
|
|
return this.membersUtils.isStripeEnabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
get specificFilters() {
|
|
|
|
const filterItems = (this.args.filter || '').split(',');
|
|
|
|
const filterItemsArray = filterItems.reject(item => isBlank(item) || BASE_FILTERS.includes(item?.trim()));
|
|
|
|
return new Set(filterItemsArray);
|
|
|
|
}
|
|
|
|
|
|
|
|
get isSpecificChecked() {
|
|
|
|
return this.forceSpecificChecked || this.specificFilters.size > 0;
|
|
|
|
}
|
|
|
|
|
2021-05-21 20:22:01 +03:00
|
|
|
get selectedSpecificOptions() {
|
|
|
|
return flattenGroupedOptions(this.specificOptions)
|
|
|
|
.filter(o => this.specificFilters.has(o.segment));
|
|
|
|
}
|
|
|
|
|
|
|
|
@action
|
2022-03-11 14:06:13 +03:00
|
|
|
toggleFilter(filter) {
|
2021-05-21 20:22:01 +03:00
|
|
|
if (this.args.disabled) {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-19 17:29:49 +03:00
|
|
|
|
|
|
|
const newBaseFilters = this.baseFilters;
|
|
|
|
newBaseFilters.has(filter) ? newBaseFilters.delete(filter) : newBaseFilters.add(filter);
|
|
|
|
|
|
|
|
this.updateFilter({newBaseFilters});
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
@action
|
2022-03-11 14:06:13 +03:00
|
|
|
toggleSpecificFilter() {
|
2021-05-21 20:22:01 +03:00
|
|
|
if (this.args.disabled) {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-19 17:29:49 +03:00
|
|
|
|
2022-01-27 14:40:11 +03:00
|
|
|
// on->off, forced with an empty filter
|
|
|
|
if (this.forceSpecificChecked && this.specificFilters.size === 0) {
|
|
|
|
this.forceSpecificChecked = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
this.forceSpecificChecked = false;
|
|
|
|
|
|
|
|
// on->off, store current filter for re-use when toggled back on
|
|
|
|
if (this.isSpecificChecked) {
|
|
|
|
this._previousSpecificFilters = this.specificFilters;
|
|
|
|
this.updateFilter({newSpecificFilters: new Set()});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// off->on, re-use stored filter
|
|
|
|
if (this._previousSpecificFilters) {
|
|
|
|
this.updateFilter({newSpecificFilters: this._previousSpecificFilters});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// off->on, display the filter selection even though the actual filter is empty
|
|
|
|
this.forceSpecificChecked = true;
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
@action
|
|
|
|
selectSpecificOptions(selectedOptions) {
|
|
|
|
if (this.args.disabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
const newSpecificFilters = new Set(selectedOptions.map(o => o.segment));
|
2023-04-04 00:45:58 +03:00
|
|
|
|
|
|
|
// If the user has deselected all options, clear the _previousSpecificFilters
|
|
|
|
// and force the specific filter to be checked so that the user can still see the options select
|
|
|
|
// Refs https://github.com/TryGhost/Team/issues/2859
|
|
|
|
if (newSpecificFilters.size === 0) {
|
|
|
|
this._previousSpecificFilters = undefined;
|
|
|
|
this.forceSpecificChecked = true;
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:29:49 +03:00
|
|
|
this.updateFilter({newSpecificFilters});
|
|
|
|
}
|
|
|
|
|
|
|
|
updateFilter({newBaseFilters, newSpecificFilters}) {
|
|
|
|
const selectedFilters = new Set([
|
|
|
|
...(newBaseFilters || this.baseFilters),
|
|
|
|
...(newSpecificFilters || this.specificFilters)
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (!this.isPaidAvailable) {
|
|
|
|
selectedFilters.delete('status:-free');
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
2022-01-19 17:29:49 +03:00
|
|
|
|
|
|
|
const newFilter = Array.from(selectedFilters).join(',') || null;
|
|
|
|
|
|
|
|
this.args.onChange?.(newFilter);
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
@task
|
|
|
|
*fetchSpecificOptionsTask() {
|
|
|
|
const options = [];
|
|
|
|
|
|
|
|
// fetch all labels w̶i̶t̶h̶ c̶o̶u̶n̶t̶s̶
|
|
|
|
// TODO: add `include: 'count.members` to query once API is fixed
|
|
|
|
const labels = yield this.store.query('label', {limit: 'all'});
|
|
|
|
|
|
|
|
if (labels.length > 0) {
|
|
|
|
const labelsGroup = {
|
|
|
|
groupName: 'Labels',
|
|
|
|
options: []
|
|
|
|
};
|
|
|
|
|
|
|
|
labels.forEach((label) => {
|
|
|
|
labelsGroup.options.push({
|
|
|
|
name: label.name,
|
|
|
|
segment: `label:${label.slug}`,
|
|
|
|
count: label.count?.members,
|
|
|
|
class: 'segment-label'
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
options.push(labelsGroup);
|
|
|
|
}
|
2022-05-11 20:11:54 +03:00
|
|
|
// fetch all tiers w̶i̶t̶h̶ c̶o̶u̶n̶t̶s̶
|
2022-05-05 19:05:47 +03:00
|
|
|
// TODO: add `include: 'count.members` to query once API supports
|
2022-05-11 20:11:54 +03:00
|
|
|
const tiers = yield this.store.query('tier', {filter: 'type:paid', limit: 'all'});
|
2021-05-21 20:22:01 +03:00
|
|
|
|
2022-05-11 20:11:54 +03:00
|
|
|
if (tiers.length > 1) {
|
2023-05-18 08:11:09 +03:00
|
|
|
const activeTiersGroup = {
|
|
|
|
groupName: 'Active tiers',
|
|
|
|
options: []
|
|
|
|
};
|
|
|
|
|
|
|
|
const archivedTiersGroup = {
|
|
|
|
groupName: 'Archived tiers',
|
2022-05-05 19:05:47 +03:00
|
|
|
options: []
|
|
|
|
};
|
2021-05-21 20:22:01 +03:00
|
|
|
|
2022-05-11 20:11:54 +03:00
|
|
|
tiers.forEach((tier) => {
|
2023-05-18 08:11:09 +03:00
|
|
|
const tierData = {
|
2022-05-11 20:11:54 +03:00
|
|
|
name: tier.name,
|
|
|
|
segment: `tier:${tier.slug}`,
|
|
|
|
count: tier.count?.members,
|
|
|
|
class: 'segment-tier'
|
2023-05-18 08:11:09 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
if (tier.active) {
|
|
|
|
activeTiersGroup.options.push(tierData);
|
|
|
|
} else {
|
|
|
|
archivedTiersGroup.options.push(tierData);
|
|
|
|
}
|
2022-05-05 19:05:47 +03:00
|
|
|
});
|
2021-05-21 20:22:01 +03:00
|
|
|
|
2023-05-18 08:11:09 +03:00
|
|
|
options.push(activeTiersGroup);
|
|
|
|
options.push(archivedTiersGroup);
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
|
2022-05-05 19:05:47 +03:00
|
|
|
this.specificOptions = options;
|
2021-05-21 20:22:01 +03:00
|
|
|
}
|
|
|
|
}
|