@rancher/shell 3.0.5-rc.7 → 3.0.5-rc.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (301) hide show
  1. package/assets/brand/classic/metadata.json +3 -0
  2. package/assets/styles/app.scss +1 -0
  3. package/assets/styles/base/_color.scss +19 -0
  4. package/assets/styles/base/_helpers.scss +10 -0
  5. package/assets/styles/base/_variables.scss +1 -1
  6. package/assets/styles/fonts/_icons.scss +1 -32
  7. package/assets/styles/global/_layout.scss +1 -1
  8. package/assets/styles/global/_tooltip.scss +7 -4
  9. package/assets/styles/themes/_dark.scss +272 -259
  10. package/assets/styles/themes/_light.scss +551 -516
  11. package/assets/styles/themes/_modern.scss +936 -0
  12. package/assets/translations/en-us.yaml +219 -38
  13. package/assets/translations/zh-hans.yaml +0 -1
  14. package/chart/__tests__/S3.test.ts +2 -1
  15. package/chart/monitoring/grafana/index.vue +8 -2
  16. package/cloud-credential/generic.vue +18 -10
  17. package/cloud-credential/harvester.vue +1 -9
  18. package/components/ActionMenuShell.vue +3 -1
  19. package/components/AdvancedSection.vue +8 -0
  20. package/components/ChartReadme.vue +17 -7
  21. package/components/Cron/CronExpressionEditor.vue +299 -0
  22. package/components/Cron/CronExpressionEditorModal.vue +247 -0
  23. package/components/Cron/CronTooltip.vue +87 -0
  24. package/components/Cron/types.ts +13 -0
  25. package/components/Drawer/ResourceDetailDrawer/__tests__/composables.test.ts +1 -26
  26. package/components/Drawer/ResourceDetailDrawer/composables.ts +0 -23
  27. package/components/Drawer/ResourceDetailDrawer/index.vue +17 -4
  28. package/components/ForceDirectedTreeChart/composable.ts +11 -0
  29. package/components/InstallHelmCharts.vue +656 -0
  30. package/components/LazyImage.vue +60 -4
  31. package/components/LocaleSelector.vue +7 -2
  32. package/components/Markdown.vue +4 -0
  33. package/components/PromptModal.vue +1 -1
  34. package/components/Resource/Detail/Card/__tests__/StateCard.test.ts +1 -0
  35. package/components/Resource/Detail/CopyToClipboard.vue +78 -0
  36. package/components/Resource/Detail/FetchLoader/__tests__/composables.test.ts +69 -0
  37. package/components/Resource/Detail/FetchLoader/composables.ts +27 -0
  38. package/components/Resource/Detail/Masthead/composable.ts +16 -0
  39. package/components/Resource/Detail/Masthead/index.vue +37 -0
  40. package/components/Resource/Detail/Metadata/Annotations/__tests__/index.test.ts +1 -1
  41. package/components/Resource/Detail/Metadata/Annotations/index.vue +1 -1
  42. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/identifying-fields.test.ts +13 -61
  43. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/index.test.ts +33 -6
  44. package/components/Resource/Detail/Metadata/IdentifyingInformation/identifying-fields.ts +29 -43
  45. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +25 -5
  46. package/components/Resource/Detail/Metadata/KeyValue.vue +12 -23
  47. package/components/Resource/Detail/Metadata/KeyValueRow.vue +144 -0
  48. package/components/Resource/Detail/Metadata/Labels/__tests__/index.test.ts +1 -0
  49. package/components/Resource/Detail/Metadata/Labels/index.vue +1 -0
  50. package/components/Resource/Detail/Metadata/__tests__/KeyValue.test.ts +30 -32
  51. package/components/Resource/Detail/Metadata/__tests__/KeyValueRow.test.ts +108 -0
  52. package/components/Resource/Detail/Metadata/__tests__/composables.test.ts +10 -20
  53. package/components/Resource/Detail/Metadata/__tests__/index.test.ts +12 -5
  54. package/components/Resource/Detail/Metadata/composables.ts +9 -10
  55. package/components/Resource/Detail/Metadata/index.vue +18 -2
  56. package/components/Resource/Detail/Page.vue +35 -21
  57. package/components/Resource/Detail/Preview/Content.vue +63 -0
  58. package/components/Resource/Detail/Preview/Preview.vue +128 -0
  59. package/components/Resource/Detail/Preview/__tests__/Content.spec.ts +71 -0
  60. package/components/Resource/Detail/Preview/__tests__/Preview.spec.ts +121 -0
  61. package/components/Resource/Detail/ResourcePopover/ResourcePopoverCard.vue +141 -0
  62. package/components/Resource/Detail/ResourcePopover/__tests__/ResourcePopoverCard.test.ts +136 -0
  63. package/components/Resource/Detail/ResourcePopover/__tests__/index.test.ts +245 -0
  64. package/components/Resource/Detail/ResourcePopover/index.vue +226 -0
  65. package/components/Resource/Detail/SpacedRow.vue +1 -0
  66. package/components/Resource/Detail/TitleBar/__tests__/composables.test.ts +8 -14
  67. package/components/Resource/Detail/TitleBar/__tests__/index.test.ts +1 -1
  68. package/components/Resource/Detail/TitleBar/composables.ts +3 -6
  69. package/components/Resource/Detail/TitleBar/index.vue +11 -29
  70. package/components/Resource/Detail/ViewOptions/composable.ts +9 -0
  71. package/components/Resource/Detail/ViewOptions/index.vue +41 -0
  72. package/components/Resource/Detail/__tests__/CopyToClipboard.spec.ts +82 -0
  73. package/components/ResourceDetail/Masthead/legacy.vue +0 -19
  74. package/components/ResourceDetail/index.vue +544 -74
  75. package/components/ResourceTable.vue +24 -0
  76. package/components/SlideInPanelManager.vue +10 -3
  77. package/components/SortableTable/index.vue +11 -5
  78. package/components/SortableTable/paging.js +3 -0
  79. package/components/Tabbed/Tab.vue +43 -1
  80. package/components/Tabbed/index.vue +32 -4
  81. package/components/__tests__/Cron/CronExpressionEditor.test.ts +151 -0
  82. package/components/__tests__/Cron/CronExpressionEditorModal.test.ts +81 -0
  83. package/components/__tests__/LazyImage.spec.ts +121 -0
  84. package/components/auth/login/saml.vue +86 -0
  85. package/components/fleet/FleetStatus.vue +4 -0
  86. package/components/form/ClusterAppearance.vue +5 -0
  87. package/components/form/LabeledSelect.vue +8 -8
  88. package/components/form/Members/ClusterPermissionsEditor.vue +1 -1
  89. package/components/form/ProjectMemberEditor.vue +1 -1
  90. package/components/form/ResourceLabeledSelect.vue +19 -6
  91. package/components/form/ResourceTabs/composable.ts +54 -0
  92. package/components/form/ResourceTabs/index.vue +30 -7
  93. package/components/form/SecretSelector.vue +9 -0
  94. package/components/form/Select.vue +13 -10
  95. package/components/form/__tests__/LabeledSelect.test.ts +133 -0
  96. package/components/form/__tests__/Select.test.ts +134 -0
  97. package/components/form/labeled-select-utils/labeled-select-pagination.ts +3 -38
  98. package/components/formatter/FleetApplicationSource.vue +25 -17
  99. package/components/nav/Favorite.vue +4 -0
  100. package/components/nav/NotificationCenter/Notification.vue +1 -27
  101. package/components/nav/WindowManager/index.vue +3 -3
  102. package/composables/useExtensionManager.ts +17 -0
  103. package/config/home-links.js +12 -0
  104. package/config/labels-annotations.js +1 -3
  105. package/config/page-actions.js +0 -1
  106. package/config/product/explorer.js +3 -1
  107. package/config/product/fleet.js +2 -7
  108. package/config/product/manager.js +0 -5
  109. package/config/query-params.js +1 -0
  110. package/config/router/navigation-guards/clusters.js +2 -1
  111. package/config/router/navigation-guards/products.js +1 -1
  112. package/core/extension-manager-impl.js +518 -0
  113. package/core/plugins.js +35 -468
  114. package/core/types.ts +8 -2
  115. package/detail/__tests__/autoscaling.horizontalpodautoscaler.test.ts +1 -0
  116. package/detail/__tests__/provisioning.cattle.io.cluster.test.ts +11 -0
  117. package/detail/__tests__/workload.test.ts +164 -0
  118. package/detail/catalog.cattle.io.app.vue +7 -4
  119. package/detail/configmap.vue +33 -75
  120. package/detail/fleet.cattle.io.bundle.vue +1 -5
  121. package/detail/fleet.cattle.io.cluster.vue +3 -2
  122. package/detail/fleet.cattle.io.gitrepo.vue +76 -49
  123. package/detail/fleet.cattle.io.helmop.vue +78 -49
  124. package/detail/projectsecret.vue +11 -0
  125. package/detail/provisioning.cattle.io.cluster.vue +350 -324
  126. package/detail/secret.vue +49 -308
  127. package/detail/workload/index.vue +38 -21
  128. package/dialog/AddonConfigConfirmationDialog.vue +1 -1
  129. package/dialog/GenericPrompt.vue +1 -1
  130. package/dialog/ImportDialog.vue +9 -2
  131. package/dialog/InstallExtensionDialog.vue +26 -15
  132. package/edit/__tests__/fleet.cattle.io.gitrepo.test.ts +2 -1
  133. package/edit/__tests__/fleet.cattle.io.helmop.test.ts +224 -0
  134. package/edit/__tests__/resources.cattle.io.restore.test.ts +106 -0
  135. package/edit/cloudcredential.vue +31 -17
  136. package/edit/constraints.gatekeeper.sh.constraint/index.vue +10 -2
  137. package/edit/fleet.cattle.io.cluster.vue +19 -0
  138. package/edit/fleet.cattle.io.gitrepo.vue +28 -22
  139. package/edit/fleet.cattle.io.helmop.vue +78 -56
  140. package/edit/logging.banzaicloud.io.output/index.vue +1 -1
  141. package/edit/logging.banzaicloud.io.output/providers/awsElasticsearch.vue +5 -6
  142. package/edit/monitoring.coreos.com.alertmanagerconfig/index.vue +12 -11
  143. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +11 -1
  144. package/edit/networking.k8s.io.ingress/Certificate.vue +9 -11
  145. package/edit/networking.k8s.io.ingress/DefaultBackend.vue +8 -3
  146. package/edit/networking.k8s.io.ingress/Rule.vue +2 -5
  147. package/edit/networking.k8s.io.ingress/RulePath.vue +17 -11
  148. package/edit/networking.k8s.io.networkpolicy/PolicyRuleTarget.vue +11 -10
  149. package/edit/networking.k8s.io.networkpolicy/PolicyRules.vue +1 -3
  150. package/edit/networking.k8s.io.networkpolicy/index.vue +17 -17
  151. package/edit/provisioning.cattle.io.cluster/index.vue +14 -19
  152. package/edit/provisioning.cattle.io.cluster/rke2.vue +31 -15
  153. package/edit/provisioning.cattle.io.cluster/tabs/AgentConfiguration.vue +9 -7
  154. package/edit/provisioning.cattle.io.cluster/tabs/DirectoryConfig.vue +10 -12
  155. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +39 -38
  156. package/edit/provisioning.cattle.io.cluster/tabs/etcd/S3Config.vue +41 -19
  157. package/edit/provisioning.cattle.io.cluster/tabs/etcd/index.vue +16 -3
  158. package/edit/provisioning.cattle.io.cluster/tabs/registries/RegistryConfigs.vue +30 -31
  159. package/edit/provisioning.cattle.io.cluster/tabs/registries/RegistryMirrors.vue +9 -10
  160. package/edit/provisioning.cattle.io.cluster/tabs/registries/index.vue +1 -3
  161. package/edit/provisioning.cattle.io.cluster/tabs/upgrade/DrainOptions.vue +16 -9
  162. package/edit/resources.cattle.io.restore.vue +5 -8
  163. package/edit/workload/index.vue +5 -14
  164. package/list/__tests__/workload.test.ts +1 -0
  165. package/list/provisioning.cattle.io.cluster.vue +1 -69
  166. package/list/workload.vue +8 -1
  167. package/machine-config/__tests__/vmwarevsphere.test.ts +5 -7
  168. package/machine-config/components/GCEImage.vue +6 -5
  169. package/machine-config/google.vue +20 -7
  170. package/machine-config/vmwarevsphere.vue +7 -17
  171. package/mixins/__tests__/chart.test.ts +139 -1
  172. package/mixins/chart.js +58 -20
  173. package/mixins/resource-fetch-api-pagination.js +3 -4
  174. package/models/__tests__/chart.test.ts +111 -80
  175. package/models/__tests__/fleet.cattle.io.helmop.test.ts +224 -0
  176. package/models/__tests__/namespace.test.ts +69 -0
  177. package/models/__tests__/node.test.ts +7 -63
  178. package/models/apps.statefulset.js +8 -10
  179. package/models/catalog.cattle.io.app.js +1 -1
  180. package/models/catalog.cattle.io.operation.js +1 -1
  181. package/models/chart.js +41 -21
  182. package/models/cloudcredential.js +2 -163
  183. package/models/cluster/node.js +7 -7
  184. package/models/cluster.x-k8s.io.machine.js +3 -3
  185. package/models/compliance.cattle.io.clusterscan.js +2 -2
  186. package/models/configmap.js +4 -0
  187. package/models/constraints.gatekeeper.sh.constraint.js +1 -1
  188. package/models/fleet-application.js +16 -63
  189. package/models/fleet.cattle.io.bundle.js +1 -38
  190. package/models/fleet.cattle.io.gitrepo.js +19 -1
  191. package/models/fleet.cattle.io.helmop.js +30 -22
  192. package/models/management.cattle.io.project.js +12 -0
  193. package/models/management.cattle.io.setting.js +4 -0
  194. package/models/namespace.js +30 -0
  195. package/models/persistentvolumeclaim.js +1 -1
  196. package/models/pod.js +2 -2
  197. package/models/provisioning.cattle.io.cluster.js +16 -40
  198. package/models/rke.cattle.io.etcdsnapshot.js +1 -1
  199. package/models/secret.js +4 -0
  200. package/models/storage.k8s.io.storageclass.js +2 -2
  201. package/models/workload.js +6 -3
  202. package/package.json +19 -18
  203. package/pages/c/_cluster/apps/charts/AppChartCardFooter.vue +26 -10
  204. package/pages/c/_cluster/apps/charts/AppChartCardSubHeader.vue +4 -1
  205. package/pages/c/_cluster/apps/charts/__tests__/AppChartCardFooter.spec.js +41 -0
  206. package/pages/c/_cluster/apps/charts/chart.vue +440 -183
  207. package/pages/c/_cluster/apps/charts/index.vue +1 -0
  208. package/pages/c/_cluster/apps/charts/install.vue +7 -6
  209. package/pages/c/_cluster/explorer/projectsecret.vue +3 -13
  210. package/pages/c/_cluster/explorer/tools/__tests__/index.test.ts +102 -12
  211. package/pages/c/_cluster/explorer/tools/index.vue +145 -254
  212. package/pages/c/_cluster/fleet/__tests__/index.test.ts +608 -314
  213. package/pages/c/_cluster/fleet/index.vue +103 -44
  214. package/pages/c/_cluster/manager/cloudCredential/index.vue +20 -60
  215. package/pages/c/_cluster/manager/drivers/kontainerDriver/index.vue +12 -2
  216. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +11 -4
  217. package/pages/c/_cluster/uiplugins/__tests__/index.spec.ts +318 -0
  218. package/pages/c/_cluster/uiplugins/index.vue +256 -387
  219. package/pages/home.vue +1 -9
  220. package/plugins/dashboard-store/actions.js +42 -22
  221. package/plugins/dashboard-store/resource-class.js +80 -0
  222. package/plugins/steve/__tests__/getters.test.ts +1 -1
  223. package/plugins/steve/__tests__/subscribe.spec.ts +259 -1
  224. package/plugins/steve/getters.js +8 -2
  225. package/plugins/steve/resourceWatcher.js +10 -3
  226. package/plugins/steve/subscribe.js +192 -19
  227. package/plugins/steve/worker/web-worker.advanced.js +2 -0
  228. package/public/index.html +2 -1
  229. package/rancher-components/Card/Card.vue +1 -19
  230. package/rancher-components/Form/Checkbox/Checkbox.vue +1 -1
  231. package/rancher-components/Form/Radio/RadioButton.vue +1 -1
  232. package/rancher-components/Form/Radio/RadioGroup.vue +1 -1
  233. package/rancher-components/LabeledTooltip/LabeledTooltip.vue +1 -11
  234. package/rancher-components/Pill/RcCounterBadge/RcCounterBadge.test.ts +53 -0
  235. package/rancher-components/Pill/RcCounterBadge/RcCounterBadge.vue +65 -0
  236. package/rancher-components/Pill/RcCounterBadge/index.ts +1 -0
  237. package/rancher-components/Pill/RcCounterBadge/types.ts +7 -0
  238. package/rancher-components/Pill/RcStatusBadge/RcStatusBadge.test.ts +15 -0
  239. package/rancher-components/Pill/RcStatusBadge/RcStatusBadge.vue +65 -0
  240. package/rancher-components/Pill/RcStatusBadge/index.ts +2 -0
  241. package/rancher-components/Pill/RcStatusBadge/types.ts +5 -0
  242. package/rancher-components/Pill/RcStatusIndicator/RcStatusIndicator.test.ts +33 -0
  243. package/rancher-components/Pill/RcStatusIndicator/RcStatusIndicator.vue +75 -0
  244. package/rancher-components/Pill/RcStatusIndicator/index.ts +2 -0
  245. package/rancher-components/Pill/RcStatusIndicator/types.ts +7 -0
  246. package/rancher-components/Pill/RcTag/RcTag.test.ts +64 -0
  247. package/rancher-components/Pill/RcTag/RcTag.vue +94 -0
  248. package/rancher-components/Pill/RcTag/index.ts +1 -0
  249. package/rancher-components/Pill/RcTag/types.ts +9 -0
  250. package/rancher-components/Pill/types.ts +3 -0
  251. package/rancher-components/RcButton/RcButton.vue +1 -1
  252. package/rancher-components/RcDropdown/RcDropdown.test.ts +98 -0
  253. package/rancher-components/RcDropdown/RcDropdown.vue +5 -0
  254. package/rancher-components/RcDropdown/RcDropdownItem.vue +7 -1
  255. package/rancher-components/RcDropdown/RcDropdownItemCheckbox.vue +2 -1
  256. package/rancher-components/RcDropdown/RcDropdownItemSelect.vue +2 -1
  257. package/rancher-components/RcDropdown/useDropdownContext.ts +21 -0
  258. package/rancher-components/RcDropdown/useDropdownItem.ts +30 -1
  259. package/rancher-components/RcItemCard/RcItemCard.test.ts +20 -0
  260. package/rancher-components/RcItemCard/RcItemCard.vue +41 -6
  261. package/rancher-components/RcItemCard/RcItemCardAction.vue +12 -0
  262. package/store/__tests__/catalog.test.ts +156 -1
  263. package/store/aws.js +19 -8
  264. package/store/catalog.js +10 -5
  265. package/store/type-map.js +3 -15
  266. package/types/extension-manager.ts +26 -0
  267. package/types/resources/settings.d.ts +1 -1
  268. package/types/shell/index.d.ts +149 -44
  269. package/types/uiplugins.ts +73 -0
  270. package/utils/__tests__/back-off.test.ts +354 -0
  271. package/utils/__tests__/kontainer.test.ts +19 -0
  272. package/utils/__tests__/product.test.ts +129 -0
  273. package/utils/__tests__/resource.test.ts +87 -0
  274. package/utils/__tests__/uiplugins.test.ts +84 -0
  275. package/utils/alertmanagerconfig.js +2 -2
  276. package/utils/auth.js +3 -76
  277. package/utils/back-off.ts +176 -0
  278. package/utils/dynamic-importer.js +8 -0
  279. package/utils/kontainer.ts +3 -5
  280. package/utils/product.ts +39 -0
  281. package/utils/resource.ts +35 -0
  282. package/utils/select.js +0 -24
  283. package/utils/style.ts +3 -0
  284. package/utils/uiplugins.ts +29 -2
  285. package/utils/validators/__tests__/setting.test.js +92 -0
  286. package/utils/validators/formRules/__tests__/index.test.ts +91 -7
  287. package/utils/validators/formRules/index.ts +84 -8
  288. package/utils/validators/setting.js +17 -0
  289. package/vue.config.js +1 -1
  290. package/cloud-credential/__tests__/harvester.test.ts +0 -18
  291. package/components/Resource/Detail/Metadata/Rectangle.vue +0 -34
  292. package/components/Resource/Detail/Metadata/__tests__/Rectangle.test.ts +0 -24
  293. package/components/ResourceDetail/Masthead/__tests__/legacy.test.ts +0 -65
  294. package/components/ResourceDetail/__tests__/index.test.ts +0 -135
  295. package/components/ResourceDetail/legacy.vue +0 -562
  296. package/components/formatter/CloudCredExpired.vue +0 -69
  297. package/pages/explorer/resource/detail/configmap.vue +0 -42
  298. package/pages/explorer/resource/detail/projectsecret.vue +0 -9
  299. package/pages/explorer/resource/detail/secret.vue +0 -63
  300. package/utils/aws.js +0 -0
  301. /package/components/{ForceDirectedTreeChart.vue → ForceDirectedTreeChart/index.vue} +0 -0
@@ -2,18 +2,20 @@
2
2
  import { mapGetters } from 'vuex';
3
3
  import Loading from '@shell/components/Loading';
4
4
  import { _FLAGGED, DEPRECATED as DEPRECATED_QUERY, HIDDEN, FROM_TOOLS } from '@shell/config/query-params';
5
- import { filterAndArrangeCharts } from '@shell/store/catalog';
5
+ import { filterAndArrangeCharts, APP_UPGRADE_STATUS } from '@shell/store/catalog';
6
6
  import { CATALOG, NORMAN } from '@shell/config/types';
7
7
  import { CATALOG as CATALOG_ANNOTATIONS } from '@shell/config/labels-annotations';
8
- import LazyImage from '@shell/components/LazyImage';
9
8
  import { isAlternate } from '@shell/utils/platform';
10
9
  import IconMessage from '@shell/components/IconMessage';
11
- import TypeDescription from '@shell/components/TypeDescription';
12
10
  import TabTitle from '@shell/components/TabTitle';
11
+ import { get } from '@shell/utils/object';
12
+ import { RcItemCard } from '@components/RcItemCard';
13
+ import AppChartCardSubHeader from '@shell/pages/c/_cluster/apps/charts/AppChartCardSubHeader';
14
+ import AppChartCardFooter from '@shell/pages/c/_cluster/apps/charts/AppChartCardFooter';
13
15
 
14
16
  export default {
15
17
  components: {
16
- LazyImage, Loading, IconMessage, TypeDescription, TabTitle
18
+ Loading, IconMessage, TabTitle, RcItemCard, AppChartCardSubHeader, AppChartCardFooter
17
19
  },
18
20
 
19
21
  async fetch() {
@@ -72,7 +74,7 @@ export default {
72
74
  return out;
73
75
  },
74
76
 
75
- options() {
77
+ appChartCards() {
76
78
  const clusterProvider = this.currentCluster.status.provider || 'other';
77
79
  const enabledCharts = (this.allCharts || []);
78
80
 
@@ -86,14 +88,28 @@ export default {
86
88
 
87
89
  charts = charts.filter((c) => c.sideLabel !== 'Experimental');
88
90
 
89
- const chartsWithApps = charts.map((chart) => {
90
- return {
91
- chart,
92
- app: this.installedAppForChart[chart.id],
91
+ return charts.map((chart) => {
92
+ const installedApp = this.installedAppForChart[chart.id];
93
+ const card = {
94
+ id: chart.id,
95
+ header: {
96
+ title: { text: chart.chartNameDisplay },
97
+ statuses: chart.cardContent.statuses
98
+ },
99
+ subHeaderItems: chart.cardContent.subHeaderItems.slice(0, 1),
100
+ footerItems: chart.deploysOnWindows ? [{
101
+ icon: 'icon-tag-alt',
102
+ iconTooltip: { key: 'generic.tags' },
103
+ labels: [this.t('catalog.charts.deploysOnWindows')],
104
+ }] : [],
105
+ image: { src: chart.versions[0].icon, alt: { text: this.t('catalog.charts.iconAlt', { app: get(chart, 'chartNameDisplay') }) } },
106
+ content: { text: chart.chartDescription },
107
+ rawChart: chart,
108
+ installedApp,
93
109
  };
94
- });
95
110
 
96
- return chartsWithApps;
111
+ return card;
112
+ });
97
113
  },
98
114
  },
99
115
 
@@ -114,6 +130,77 @@ export default {
114
130
  },
115
131
 
116
132
  methods: {
133
+ getCardActions(card) {
134
+ const { installedApp, rawChart } = card;
135
+
136
+ if (installedApp) {
137
+ const actions = [];
138
+ const upgradeAvailable = installedApp.upgradeAvailable === APP_UPGRADE_STATUS.SINGLE_UPGRADE;
139
+
140
+ if (upgradeAvailable) {
141
+ actions.push({
142
+ label: this.t('catalog.tools.action.upgrade'),
143
+ icon: 'icon-upgrade-alt',
144
+ action: 'upgrade',
145
+ });
146
+ }
147
+
148
+ actions.push({
149
+ label: this.t('catalog.tools.action.edit'),
150
+ icon: 'icon-edit',
151
+ action: 'edit',
152
+ });
153
+
154
+ const currentVersion = installedApp.spec.chart.metadata.version;
155
+ const versions = rawChart.versions;
156
+ const currentIndex = versions.findIndex((v) => v.version === currentVersion);
157
+
158
+ if (currentIndex !== -1 && currentIndex < versions.length - 1) {
159
+ actions.push({
160
+ label: this.t('catalog.tools.action.downgrade'),
161
+ icon: 'icon-history',
162
+ action: 'downgrade',
163
+ });
164
+ }
165
+
166
+ actions.push({ divider: true });
167
+
168
+ actions.push({
169
+ label: this.t('catalog.tools.action.remove'),
170
+ icon: 'icon-delete',
171
+ action: 'remove',
172
+ });
173
+
174
+ return actions;
175
+ }
176
+
177
+ return [
178
+ {
179
+ label: this.t('catalog.tools.action.install'),
180
+ action: 'install',
181
+ icon: 'icon-plus',
182
+ enabled: !rawChart.blocked
183
+ }
184
+ ];
185
+ },
186
+ upgrade(app, chart) {
187
+ const latestVersion = chart.versions[0].version;
188
+
189
+ this.edit(app, latestVersion);
190
+ },
191
+
192
+ downgrade(app, chart) {
193
+ const currentVersion = app.spec.chart.metadata.version;
194
+ const versions = chart.versions;
195
+ const currentIndex = versions.findIndex((v) => v.version === currentVersion);
196
+
197
+ if (currentIndex !== -1 && currentIndex < versions.length - 1) {
198
+ const downgradeVersion = versions[currentIndex + 1].version;
199
+
200
+ this.edit(app, downgradeVersion);
201
+ }
202
+ },
203
+
117
204
  edit(app, version) {
118
205
  app.goToUpgrade(version, true);
119
206
  },
@@ -132,260 +219,49 @@ export default {
132
219
  install(chart) {
133
220
  chart.goToInstall(FROM_TOOLS);
134
221
  },
135
-
136
- openV1Tool(id) {
137
- const cluster = this.$store.getters['currentCluster'];
138
- const route = {
139
- name: 'c-cluster-explorer-tools-pages-page',
140
- params: {
141
- cluster: cluster.id,
142
- product: 'explorer',
143
- page: id,
144
- }
145
- };
146
-
147
- this.$router.replace(route);
148
- },
149
222
  }
150
223
  };
151
224
  </script>
152
225
 
153
- <style lang="scss" scoped>
154
- $margin: 10px;
155
- $logo: 50px;
156
-
157
- .grid {
158
- display: flex;
159
- justify-content: flex-start;
160
- flex-wrap: wrap;
161
- margin: 0 -1*$margin;
162
-
163
- @media only screen and (min-width: map-get($breakpoints, '--viewport-4')) {
164
- .item {
165
- width: 100%;
166
- }
167
- }
168
- @media only screen and (min-width: map-get($breakpoints, '--viewport-7')) {
169
- .item {
170
- width: 100%;
171
- }
172
- }
173
- @media only screen and (min-width: map-get($breakpoints, '--viewport-9')) {
174
- .item {
175
- width: calc(50% - 2 * #{$margin});
176
- }
177
- }
178
- @media only screen and (min-width: map-get($breakpoints, '--viewport-12')) {
179
- .item {
180
- width: calc(33.33333% - 2 * #{$margin});
181
- }
182
- }
183
-
184
- .item {
185
- display: grid;
186
- grid-template-areas: "logo name-version name-version"
187
- "description description description"
188
- "state state action";
189
- grid-template-columns: $logo auto min-content;
190
- grid-template-rows: 50px 55px 35px;
191
- row-gap: $margin;
192
- column-gap: $margin;
193
-
194
- margin: $margin;
195
- padding: $margin;
196
- position: relative;
197
- border: 1px solid var(--border);
198
- border-radius: calc( 1.5 * var(--border-radius));
199
-
200
- .logo {
201
- grid-area: logo;
202
- text-align: center;
203
- width: $logo;
204
- height: $logo;
205
- border-radius: calc(2 * var(--border-radius));
206
- overflow: hidden;
207
- background-color: white;
208
-
209
- img {
210
- width: $logo - 4px;
211
- height: $logo - 4px;
212
- object-fit: contain;
213
- position: relative;
214
- top: 2px;
215
- }
216
-
217
- > i {
218
- background-color: var(--box-bg);
219
- border-radius: 50%;
220
- font-size: 32px;
221
- line-height: 50px;
222
- width: 50px;
223
- }
224
- }
225
-
226
- .name-version {
227
- grid-area: name-version;
228
- padding: 10px 0 0 0;
229
- }
230
-
231
- .name {
232
- white-space: nowrap;
233
- overflow: hidden;
234
- text-overflow: ellipsis;
235
- margin: 0;
236
- }
237
-
238
- .os-label {
239
- position: absolute;
240
- top: 10px;
241
- right: 10px;
242
- padding: 3px;
243
- font-size: 12px;
244
- line-height: 12px;
245
- background-color: var(--primary);
246
- color: var(--primary-text);
247
- }
248
-
249
- .version {
250
- color: var(--muted);
251
- white-space: nowrap;
252
- overflow: hidden;
253
- text-overflow: ellipsis;
254
- font-size: 0.9em;
255
- margin-top: 4px;
256
- }
257
-
258
- .description {
259
- grid-area: description;
260
- }
261
-
262
- .description-content {
263
- display: -webkit-box;
264
- -webkit-box-orient: vertical;
265
- -webkit-line-clamp: 3;
266
- line-clamp: 3;
267
- overflow: hidden;
268
- text-overflow: ellipsis;
269
- color: var(--text-muted);
270
- }
271
-
272
- .state {
273
- grid-area: state;
274
- }
275
-
276
- .action {
277
- grid-area: action;
278
- white-space: nowrap;
279
-
280
- button {
281
- height: 30px;
282
- }
283
- }
284
- }
285
- }
286
- </style>
287
-
288
226
  <template>
289
227
  <Loading v-if="$fetchState.pending" />
290
- <div v-else-if="options.length">
291
- <h1>
228
+ <div v-else-if="appChartCards.length">
229
+ <h1 class="mmb-6">
292
230
  <TabTitle>{{ t('catalog.tools.header') }}</TabTitle>
293
231
  </h1>
294
- <TypeDescription
295
- resource="chart"
296
- />
297
232
 
298
- <div class="grid">
299
- <div
300
- v-for="opt in options"
301
- :key="opt.chart.id"
302
- class="item"
303
- :data-testid="`cluster-tools-app-${opt.chart.id}`"
233
+ <div
234
+ class="tools-app-chart-cards"
235
+ data-testid="tools-app-chart-cards"
236
+ >
237
+ <rc-item-card
238
+ v-for="card in appChartCards"
239
+ :id="card.id"
240
+ :key="card.id"
241
+ :header="card.header"
242
+ :image="card.image"
243
+ :content="card.content"
244
+ :actions="getCardActions(card)"
245
+ :class="{ 'single-card': appChartCards.length === 1 }"
246
+ @upgrade="() => upgrade(card.installedApp, card.rawChart)"
247
+ @downgrade="() => downgrade(card.installedApp, card.rawChart)"
248
+ @edit="() => edit(card.installedApp)"
249
+ @remove="(payload) => remove(card.installedApp, payload.event)"
250
+ @install="() => install(card.rawChart)"
304
251
  >
305
- <div
306
- class="logo"
252
+ <template
253
+ v-once
254
+ #item-card-sub-header
307
255
  >
308
- <i
309
- v-if="opt.chart.iconName"
310
- class="icon"
311
- :class="opt.chart.iconName"
312
- :alt="t('catalog.tools.iconAlt', { app: opt.chart.chartNameDisplay })"
313
- />
314
- <LazyImage
315
- v-else
316
- :alt="t('catalog.tools.iconAlt', { app: opt.chart.chartNameDisplay })"
317
- :src="opt.chart.icon"
318
- />
319
- </div>
320
- <div class="name-version">
321
- <div>
322
- <h3 class="name">
323
- {{ opt.chart.chartNameDisplay }}
324
- </h3>
325
- <label
326
- v-if="opt.chart.deploysOnWindows"
327
- class="os-label"
328
- >{{ t('catalog.charts.deploysOnWindows') }}</label>
329
- </div>
330
- <div class="version">
331
- <template v-if="opt.app && opt.app.upgradeAvailableVersion">
332
- v{{ opt.app.currentVersion }} <b><i class="icon icon-chevron-right" /> v{{ opt.app.upgradeAvailableVersion }}</b>
333
- </template>
334
- <template v-else-if="opt.app">
335
- v{{ opt.app.currentVersion }}
336
- </template>
337
- <template v-else-if="opt.chart.versions.length">
338
- v{{ opt.chart.versions[0].version }}
339
- </template>
340
- </div>
341
- </div>
342
- <div class="description">
343
- <div
344
- v-clean-html="opt.chart.chartDescription"
345
- class="description-content"
346
- />
347
- </div>
348
- <div class="action">
349
- <template v-if="opt.blocked">
350
- <button
351
- v-clean-html="t('catalog.tools.action.install')"
352
- role="button"
353
- :aria-label="t('catalog.tools.action.install')"
354
- disabled="true"
355
- class="btn btn-sm role-primary"
356
- />
357
- </template>
358
- <template v-else-if="opt.app">
359
- <button
360
- class="btn btn-sm role-secondary"
361
- role="button"
362
- :aria-label="t('catalog.tools.action.remove')"
363
- @click="remove(opt.app, $event)"
364
- >
365
- <i
366
- class="icon icon-delete icon-lg"
367
- :alt="t('catalog.tools.action.remove')"
368
- />
369
- </button>
370
- <button
371
- v-clean-html="t('catalog.tools.action.edit')"
372
- role="button"
373
- :aria-label="t('catalog.tools.action.edit')"
374
- class="btn btn-sm role-secondary"
375
- @click="edit(opt.app)"
376
- />
377
- </template>
378
- <template v-else>
379
- <button
380
- v-clean-html="t('catalog.tools.action.install')"
381
- role="button"
382
- :aria-label="t('catalog.tools.action.install')"
383
- class="btn btn-sm role-primary"
384
- @click="install(opt.chart)"
385
- />
386
- </template>
387
- </div>
388
- </div>
256
+ <AppChartCardSubHeader :items="card.subHeaderItems" />
257
+ </template>
258
+ <template
259
+ v-once
260
+ #item-card-footer
261
+ >
262
+ <AppChartCardFooter :items="card.footerItems" />
263
+ </template>
264
+ </rc-item-card>
389
265
  </div>
390
266
  </div>
391
267
  <div v-else>
@@ -395,3 +271,18 @@ export default {
395
271
  />
396
272
  </div>
397
273
  </template>
274
+
275
+ <style lang="scss" scoped>
276
+ .tools-app-chart-cards {
277
+ display: grid;
278
+ grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
279
+ grid-gap: var(--gap-md);
280
+ width: 100%;
281
+ height: max-content;
282
+ overflow: hidden;
283
+
284
+ .single-card {
285
+ max-width: 500px;
286
+ }
287
+ }
288
+ </style>