2025-05-02 16:11:48 +00:00
|
|
|
import React, { useContext, useEffect, useState } from "react";
|
2024-07-18 09:20:17 +00:00
|
|
|
import { InjectedRouter } from "react-router";
|
2026-03-10 18:00:14 +00:00
|
|
|
import { useQuery, useQueryClient } from "react-query";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
|
|
|
import PATHS from "router/paths";
|
2024-12-30 18:18:21 +00:00
|
|
|
import { DEFAULT_USE_QUERY_OPTIONS } from "utilities/constants";
|
2024-10-02 15:43:20 +00:00
|
|
|
import { getFileDetails, IFileDetails } from "utilities/file/fileUtils";
|
2025-02-27 15:53:34 +00:00
|
|
|
import { getPathWithQueryParams, QueryParams } from "utilities/url";
|
2026-01-19 19:36:01 +00:00
|
|
|
import softwareAPI from "services/entities/software";
|
2024-12-17 16:41:57 +00:00
|
|
|
import labelsAPI, { getCustomLabels } from "services/entities/labels";
|
2024-09-03 22:35:33 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
import { NotificationContext } from "context/notification";
|
2024-10-15 16:13:34 +00:00
|
|
|
import { AppContext } from "context/app";
|
Remove UI gating in GitOps mode for excepted entities (#42486)
<!-- Add the related story/sub-task/bug number, like Resolves #123, or
remove if NA -->
**Related issue:** Resolves #42184
# Checklist for submitter
If some of the following don't apply, delete the relevant line.
- [ ] Changes file added for user-visible changes in `changes/`,
`orbit/changes/` or `ee/fleetd-chrome/changes`.
See [Changes
files](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/guides/committing-changes.md#changes-files)
for more information.
- [ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements), JS
inline code is prevented especially for url redirects, and untrusted
data interpolated into shell scripts/commands is validated against shell
metacharacters.
- [ ] If paths of existing endpoints are modified without backwards
compatibility, checked the frontend/CLI for any necessary changes
## Testing
- [ ] Added/updated automated tests
- [ ] Where appropriate, [automated tests simulate multiple hosts and
test for host
isolation](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/reference/patterns-backend.md#unit-testing)
(updates to one hosts's records do not affect another)
- [ ] QA'd all new/changed functionality manually
For unreleased bug fixes in a release candidate, one of:
- [ ] Confirmed that the fix is not expected to adversely impact load
test results
- [ ] Alerted the release DRI if additional load testing is needed
## Database migrations
- [ ] Checked schema for all modified table for columns that will
auto-update timestamps during migration.
- [ ] Confirmed that updating the timestamps is acceptable, and will not
cause unwanted side effects.
- [ ] Ensured the correct collation is explicitly set for character
columns (`COLLATE utf8mb4_unicode_ci`).
## New Fleet configuration settings
- [ ] Setting(s) is/are explicitly excluded from GitOps
If you didn't check the box above, follow this checklist for
GitOps-enabled settings:
- [ ] Verified that the setting is exported via `fleetctl
generate-gitops`
- [ ] Verified the setting is documented in a separate PR to [the GitOps
documentation](https://github.com/fleetdm/fleet/blob/main/docs/Configuration/yaml-files.md#L485)
- [ ] Verified that the setting is cleared on the server if it is not
supplied in a YAML file (or that it is documented as being optional)
- [ ] Verified that any relevant UI is disabled when GitOps mode is
enabled
## fleetd/orbit/Fleet Desktop
- [ ] Verified compatibility with the latest released version of Fleet
(see [Must
rule](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/workflows/fleetd-development-and-release-strategy.md))
- [ ] If the change applies to only one platform, confirmed that
`runtime.GOOS` is used as needed to isolate changes
- [ ] Verified that fleetd runs on macOS, Linux and Windows
- [ ] Verified auto-update works from the released version of component
to the new version (see [tools/tuf/test](../tools/tuf/test/README.md))
<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit
## Release Notes
* **New Features**
* Added support for GitOps exceptions per entity type (labels, software,
secrets), allowing specific areas to bypass GitOps mode restrictions
when configured.
* **Bug Fixes**
* Improved GitOps mode behavior to properly respect per-entity-type
exception settings across software, labels, and secrets management.
* **Tests**
* Extended test coverage for GitOps exception handling scenarios.
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-03-31 14:10:56 +00:00
|
|
|
import useGitOpsMode from "hooks/useGitOpsMode";
|
2024-12-17 16:41:57 +00:00
|
|
|
import { ILabelSummary } from "interfaces/label";
|
2024-09-03 22:35:33 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
import FileProgressModal from "components/FileProgressModal";
|
2024-10-15 16:13:34 +00:00
|
|
|
import PremiumFeatureMessage from "components/PremiumFeatureMessage";
|
2024-12-17 16:41:57 +00:00
|
|
|
import Spinner from "components/Spinner";
|
|
|
|
|
import DataError from "components/DataError";
|
2026-02-25 23:43:23 +00:00
|
|
|
import InfoBanner from "components/InfoBanner";
|
2025-05-02 16:11:48 +00:00
|
|
|
import CategoriesEndUserExperienceModal from "pages/SoftwarePage/components/modals/CategoriesEndUserExperienceModal";
|
2024-10-15 16:13:34 +00:00
|
|
|
|
2025-05-02 16:11:48 +00:00
|
|
|
import PackageForm from "pages/SoftwarePage/components/forms/PackageForm";
|
|
|
|
|
import { IPackageFormData } from "pages/SoftwarePage/components/forms/PackageForm/PackageForm";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
import { getErrorMessage } from "./helpers";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
const baseClass = "software-custom-package";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
interface ISoftwarePackageProps {
|
|
|
|
|
currentTeamId: number;
|
2024-07-18 09:20:17 +00:00
|
|
|
router: InjectedRouter;
|
2024-10-02 15:43:20 +00:00
|
|
|
isSidePanelOpen: boolean;
|
|
|
|
|
setSidePanelOpen: (isOpen: boolean) => void;
|
2024-07-18 09:20:17 +00:00
|
|
|
}
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
const SoftwareCustomPackage = ({
|
|
|
|
|
currentTeamId,
|
2024-08-02 15:12:43 +00:00
|
|
|
router,
|
2024-10-02 15:43:20 +00:00
|
|
|
isSidePanelOpen,
|
|
|
|
|
setSidePanelOpen,
|
|
|
|
|
}: ISoftwarePackageProps) => {
|
2024-07-18 09:20:17 +00:00
|
|
|
const { renderFlash } = useContext(NotificationContext);
|
Remove UI gating in GitOps mode for excepted entities (#42486)
<!-- Add the related story/sub-task/bug number, like Resolves #123, or
remove if NA -->
**Related issue:** Resolves #42184
# Checklist for submitter
If some of the following don't apply, delete the relevant line.
- [ ] Changes file added for user-visible changes in `changes/`,
`orbit/changes/` or `ee/fleetd-chrome/changes`.
See [Changes
files](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/guides/committing-changes.md#changes-files)
for more information.
- [ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements), JS
inline code is prevented especially for url redirects, and untrusted
data interpolated into shell scripts/commands is validated against shell
metacharacters.
- [ ] If paths of existing endpoints are modified without backwards
compatibility, checked the frontend/CLI for any necessary changes
## Testing
- [ ] Added/updated automated tests
- [ ] Where appropriate, [automated tests simulate multiple hosts and
test for host
isolation](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/reference/patterns-backend.md#unit-testing)
(updates to one hosts's records do not affect another)
- [ ] QA'd all new/changed functionality manually
For unreleased bug fixes in a release candidate, one of:
- [ ] Confirmed that the fix is not expected to adversely impact load
test results
- [ ] Alerted the release DRI if additional load testing is needed
## Database migrations
- [ ] Checked schema for all modified table for columns that will
auto-update timestamps during migration.
- [ ] Confirmed that updating the timestamps is acceptable, and will not
cause unwanted side effects.
- [ ] Ensured the correct collation is explicitly set for character
columns (`COLLATE utf8mb4_unicode_ci`).
## New Fleet configuration settings
- [ ] Setting(s) is/are explicitly excluded from GitOps
If you didn't check the box above, follow this checklist for
GitOps-enabled settings:
- [ ] Verified that the setting is exported via `fleetctl
generate-gitops`
- [ ] Verified the setting is documented in a separate PR to [the GitOps
documentation](https://github.com/fleetdm/fleet/blob/main/docs/Configuration/yaml-files.md#L485)
- [ ] Verified that the setting is cleared on the server if it is not
supplied in a YAML file (or that it is documented as being optional)
- [ ] Verified that any relevant UI is disabled when GitOps mode is
enabled
## fleetd/orbit/Fleet Desktop
- [ ] Verified compatibility with the latest released version of Fleet
(see [Must
rule](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/workflows/fleetd-development-and-release-strategy.md))
- [ ] If the change applies to only one platform, confirmed that
`runtime.GOOS` is used as needed to isolate changes
- [ ] Verified that fleetd runs on macOS, Linux and Windows
- [ ] Verified auto-update works from the released version of component
to the new version (see [tools/tuf/test](../tools/tuf/test/README.md))
<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit
## Release Notes
* **New Features**
* Added support for GitOps exceptions per entity type (labels, software,
secrets), allowing specific areas to bypass GitOps mode restrictions
when configured.
* **Bug Fixes**
* Improved GitOps mode behavior to properly respect per-entity-type
exception settings across software, labels, and secrets management.
* **Tests**
* Extended test coverage for GitOps exception handling scenarios.
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-03-31 14:10:56 +00:00
|
|
|
const { isPremiumTier } = useContext(AppContext);
|
2026-03-10 18:00:14 +00:00
|
|
|
const queryClient = useQueryClient();
|
Remove UI gating in GitOps mode for excepted entities (#42486)
<!-- Add the related story/sub-task/bug number, like Resolves #123, or
remove if NA -->
**Related issue:** Resolves #42184
# Checklist for submitter
If some of the following don't apply, delete the relevant line.
- [ ] Changes file added for user-visible changes in `changes/`,
`orbit/changes/` or `ee/fleetd-chrome/changes`.
See [Changes
files](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/guides/committing-changes.md#changes-files)
for more information.
- [ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements), JS
inline code is prevented especially for url redirects, and untrusted
data interpolated into shell scripts/commands is validated against shell
metacharacters.
- [ ] If paths of existing endpoints are modified without backwards
compatibility, checked the frontend/CLI for any necessary changes
## Testing
- [ ] Added/updated automated tests
- [ ] Where appropriate, [automated tests simulate multiple hosts and
test for host
isolation](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/reference/patterns-backend.md#unit-testing)
(updates to one hosts's records do not affect another)
- [ ] QA'd all new/changed functionality manually
For unreleased bug fixes in a release candidate, one of:
- [ ] Confirmed that the fix is not expected to adversely impact load
test results
- [ ] Alerted the release DRI if additional load testing is needed
## Database migrations
- [ ] Checked schema for all modified table for columns that will
auto-update timestamps during migration.
- [ ] Confirmed that updating the timestamps is acceptable, and will not
cause unwanted side effects.
- [ ] Ensured the correct collation is explicitly set for character
columns (`COLLATE utf8mb4_unicode_ci`).
## New Fleet configuration settings
- [ ] Setting(s) is/are explicitly excluded from GitOps
If you didn't check the box above, follow this checklist for
GitOps-enabled settings:
- [ ] Verified that the setting is exported via `fleetctl
generate-gitops`
- [ ] Verified the setting is documented in a separate PR to [the GitOps
documentation](https://github.com/fleetdm/fleet/blob/main/docs/Configuration/yaml-files.md#L485)
- [ ] Verified that the setting is cleared on the server if it is not
supplied in a YAML file (or that it is documented as being optional)
- [ ] Verified that any relevant UI is disabled when GitOps mode is
enabled
## fleetd/orbit/Fleet Desktop
- [ ] Verified compatibility with the latest released version of Fleet
(see [Must
rule](https://github.com/fleetdm/fleet/blob/main/docs/Contributing/workflows/fleetd-development-and-release-strategy.md))
- [ ] If the change applies to only one platform, confirmed that
`runtime.GOOS` is used as needed to isolate changes
- [ ] Verified that fleetd runs on macOS, Linux and Windows
- [ ] Verified auto-update works from the released version of component
to the new version (see [tools/tuf/test](../tools/tuf/test/README.md))
<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit
## Release Notes
* **New Features**
* Added support for GitOps exceptions per entity type (labels, software,
secrets), allowing specific areas to bypass GitOps mode restrictions
when configured.
* **Bug Fixes**
* Improved GitOps mode behavior to properly respect per-entity-type
exception settings across software, labels, and secrets management.
* **Tests**
* Extended test coverage for GitOps exception handling scenarios.
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-03-31 14:10:56 +00:00
|
|
|
const { gitOpsModeEnabled } = useGitOpsMode("software");
|
2025-05-22 13:47:57 +00:00
|
|
|
|
2025-05-02 16:11:48 +00:00
|
|
|
const [uploadProgress, setUploadProgress] = useState(0);
|
|
|
|
|
const [uploadDetails, setUploadDetails] = useState<IFileDetails | null>(null);
|
|
|
|
|
const [
|
|
|
|
|
showPreviewEndUserExperience,
|
|
|
|
|
setShowPreviewEndUserExperience,
|
|
|
|
|
] = useState(false);
|
2025-11-07 22:30:51 +00:00
|
|
|
const [
|
|
|
|
|
isIpadOrIphoneSoftwareSource,
|
|
|
|
|
setIsIpadOrIphoneSoftwareSource,
|
|
|
|
|
] = useState(false);
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2024-12-17 16:41:57 +00:00
|
|
|
const {
|
|
|
|
|
data: labels,
|
|
|
|
|
isLoading: isLoadingLabels,
|
|
|
|
|
isError: isErrorLabels,
|
|
|
|
|
} = useQuery<ILabelSummary[], Error>(
|
|
|
|
|
["custom_labels"],
|
2025-12-30 03:28:45 +00:00
|
|
|
() =>
|
|
|
|
|
labelsAPI
|
|
|
|
|
.summary(currentTeamId)
|
|
|
|
|
.then((res) => getCustomLabels(res.labels)),
|
2024-12-17 16:41:57 +00:00
|
|
|
{
|
|
|
|
|
...DEFAULT_USE_QUERY_OPTIONS,
|
|
|
|
|
enabled: isPremiumTier,
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
2024-07-18 09:20:17 +00:00
|
|
|
useEffect(() => {
|
|
|
|
|
const beforeUnloadHandler = (e: BeforeUnloadEvent) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
// Next line with e.returnValue is included for legacy support
|
|
|
|
|
// e.g.Chrome / Edge < 119
|
|
|
|
|
e.returnValue = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// set up event listener to prevent user from leaving page while uploading
|
2024-10-02 15:43:20 +00:00
|
|
|
if (uploadDetails) {
|
2024-07-18 09:20:17 +00:00
|
|
|
addEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
} else {
|
|
|
|
|
removeEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// clean up event listener and timeout on component unmount
|
|
|
|
|
return () => {
|
|
|
|
|
removeEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
};
|
2024-10-02 15:43:20 +00:00
|
|
|
}, [uploadDetails]);
|
|
|
|
|
|
2025-11-07 22:30:51 +00:00
|
|
|
const onClickPreviewEndUserExperience = (isIosOrIpadosApp = false) => {
|
2025-05-02 16:11:48 +00:00
|
|
|
setShowPreviewEndUserExperience(!showPreviewEndUserExperience);
|
2025-11-07 22:30:51 +00:00
|
|
|
setIsIpadOrIphoneSoftwareSource(isIosOrIpadosApp);
|
2025-05-02 16:11:48 +00:00
|
|
|
};
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
const onCancel = () => {
|
|
|
|
|
router.push(
|
2025-02-27 15:53:34 +00:00
|
|
|
getPathWithQueryParams(PATHS.SOFTWARE_TITLES, {
|
2026-02-17 21:19:33 +00:00
|
|
|
fleet_id: currentTeamId,
|
2025-02-27 15:53:34 +00:00
|
|
|
})
|
2024-10-02 15:43:20 +00:00
|
|
|
);
|
|
|
|
|
};
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
const onSubmit = async (formData: IPackageFormData) => {
|
|
|
|
|
if (!formData.software) {
|
|
|
|
|
renderFlash(
|
|
|
|
|
"error",
|
|
|
|
|
`Couldn't add. Please refresh the page and try again.`
|
|
|
|
|
);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2026-02-11 17:18:32 +00:00
|
|
|
setUploadDetails(getFileDetails(formData.software));
|
2024-10-02 15:43:20 +00:00
|
|
|
|
2024-09-17 13:40:47 +00:00
|
|
|
// Note: This TODO is copied to onSaveSoftwareChanges in EditSoftwareModal
|
2024-07-18 09:20:17 +00:00
|
|
|
// TODO: confirm we are deleting the second sentence (not modifying it) for non-self-service installers
|
|
|
|
|
try {
|
2025-05-21 15:31:39 +00:00
|
|
|
const {
|
|
|
|
|
software_package: { title_id: softwarePackageTitleId },
|
|
|
|
|
} = await softwareAPI.addSoftwarePackage({
|
2024-10-02 15:43:20 +00:00
|
|
|
data: formData,
|
|
|
|
|
teamId: currentTeamId,
|
|
|
|
|
onUploadProgress: (progressEvent) => {
|
2024-10-08 19:27:55 +00:00
|
|
|
const progress = progressEvent.progress || 0;
|
|
|
|
|
// for large uploads it seems to take a bit for the server to finalize its response so we'll keep the
|
|
|
|
|
// progress bar at 97% until the server response is received
|
|
|
|
|
setUploadProgress(Math.max(progress - 0.03, 0.01));
|
2024-10-02 15:43:20 +00:00
|
|
|
},
|
|
|
|
|
});
|
2024-07-18 09:20:17 +00:00
|
|
|
|
2025-05-22 13:47:57 +00:00
|
|
|
if (!gitOpsModeEnabled) {
|
|
|
|
|
renderFlash(
|
|
|
|
|
"success",
|
|
|
|
|
<>
|
|
|
|
|
<b>{formData.software?.name}</b> successfully added.
|
|
|
|
|
{formData.selfService
|
|
|
|
|
? " The end user can install from Fleet Desktop."
|
|
|
|
|
: ""}
|
|
|
|
|
</>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-10 18:00:14 +00:00
|
|
|
queryClient.invalidateQueries({
|
|
|
|
|
queryKey: [{ scope: "software-titles" }],
|
|
|
|
|
});
|
|
|
|
|
|
2025-05-21 15:31:39 +00:00
|
|
|
const newQueryParams: QueryParams = {
|
2026-02-17 21:19:33 +00:00
|
|
|
fleet_id: currentTeamId,
|
2025-05-22 13:47:57 +00:00
|
|
|
gitops_yaml: gitOpsModeEnabled ? "true" : undefined,
|
2025-05-21 15:31:39 +00:00
|
|
|
};
|
2024-07-18 09:20:17 +00:00
|
|
|
router.push(
|
2025-05-21 15:31:39 +00:00
|
|
|
getPathWithQueryParams(
|
|
|
|
|
PATHS.SOFTWARE_TITLE_DETAILS(softwarePackageTitleId.toString()),
|
|
|
|
|
newQueryParams
|
|
|
|
|
)
|
2024-10-22 17:52:20 +00:00
|
|
|
);
|
2024-07-18 09:20:17 +00:00
|
|
|
} catch (e) {
|
2024-12-24 21:26:24 +00:00
|
|
|
renderFlash("error", getErrorMessage(e));
|
2024-07-18 09:20:17 +00:00
|
|
|
}
|
2024-10-02 15:43:20 +00:00
|
|
|
setUploadDetails(null);
|
2024-07-18 09:20:17 +00:00
|
|
|
};
|
|
|
|
|
|
2024-12-17 16:41:57 +00:00
|
|
|
const renderContent = () => {
|
|
|
|
|
if (isLoadingLabels) {
|
|
|
|
|
return <Spinner />;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isErrorLabels) {
|
2025-05-06 13:20:03 +00:00
|
|
|
return <DataError verticalPaddingSize="pad-xxxlarge" />;
|
2024-12-17 16:41:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<>
|
2026-02-25 23:43:23 +00:00
|
|
|
{gitOpsModeEnabled && (
|
2026-03-30 15:17:39 +00:00
|
|
|
<InfoBanner borderRadius="medium">
|
2026-02-25 23:43:23 +00:00
|
|
|
Add custom packages in GitOps mode so Fleet can host your software.
|
|
|
|
|
After adding, copy its SHA-256 hash into your YAML so the next
|
|
|
|
|
GitOps workflow doesn't delete it.
|
|
|
|
|
</InfoBanner>
|
|
|
|
|
)}
|
2024-12-17 16:41:57 +00:00
|
|
|
<PackageForm
|
|
|
|
|
labels={labels || []}
|
|
|
|
|
showSchemaButton={!isSidePanelOpen}
|
|
|
|
|
onClickShowSchema={() => setSidePanelOpen(true)}
|
|
|
|
|
className={`${baseClass}__package-form`}
|
|
|
|
|
onCancel={onCancel}
|
|
|
|
|
onSubmit={onSubmit}
|
2025-05-02 16:11:48 +00:00
|
|
|
onClickPreviewEndUserExperience={onClickPreviewEndUserExperience}
|
2024-12-17 16:41:57 +00:00
|
|
|
/>
|
|
|
|
|
{uploadDetails && (
|
|
|
|
|
<FileProgressModal
|
|
|
|
|
fileDetails={uploadDetails}
|
|
|
|
|
fileProgress={uploadProgress}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
2025-05-02 16:11:48 +00:00
|
|
|
{showPreviewEndUserExperience && (
|
|
|
|
|
<CategoriesEndUserExperienceModal
|
|
|
|
|
onCancel={onClickPreviewEndUserExperience}
|
2025-11-07 22:30:51 +00:00
|
|
|
isIosOrIpadosApp={isIpadOrIphoneSoftwareSource}
|
2025-05-02 16:11:48 +00:00
|
|
|
/>
|
|
|
|
|
)}
|
2024-12-17 16:41:57 +00:00
|
|
|
</>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
2024-10-15 16:13:34 +00:00
|
|
|
if (!isPremiumTier) {
|
|
|
|
|
return (
|
|
|
|
|
<PremiumFeatureMessage className={`${baseClass}__premium-message`} />
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-17 16:41:57 +00:00
|
|
|
return <div className={baseClass}>{renderContent()}</div>;
|
2024-07-18 09:20:17 +00:00
|
|
|
};
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
export default SoftwareCustomPackage;
|