@rancher/shell 3.0.5-rc.8 → 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 (171) hide show
  1. package/assets/styles/base/_color.scss +4 -1
  2. package/assets/styles/global/_tooltip.scss +7 -4
  3. package/assets/styles/themes/_dark.scss +11 -0
  4. package/assets/styles/themes/_light.scss +13 -1
  5. package/assets/styles/themes/_modern.scss +22 -0
  6. package/assets/translations/en-us.yaml +136 -14
  7. package/assets/translations/zh-hans.yaml +0 -1
  8. package/chart/monitoring/grafana/index.vue +8 -2
  9. package/components/ActionMenuShell.vue +3 -1
  10. package/components/Cron/CronExpressionEditor.vue +299 -0
  11. package/components/Cron/CronExpressionEditorModal.vue +247 -0
  12. package/components/Cron/CronTooltip.vue +87 -0
  13. package/components/Cron/types.ts +13 -0
  14. package/components/ForceDirectedTreeChart/composable.ts +11 -0
  15. package/components/PromptModal.vue +1 -1
  16. package/components/Resource/Detail/Card/__tests__/StateCard.test.ts +1 -0
  17. package/components/Resource/Detail/CopyToClipboard.vue +78 -0
  18. package/components/Resource/Detail/FetchLoader/__tests__/composables.test.ts +69 -0
  19. package/components/Resource/Detail/FetchLoader/composables.ts +27 -0
  20. package/components/Resource/Detail/Metadata/Annotations/__tests__/index.test.ts +1 -1
  21. package/components/Resource/Detail/Metadata/Annotations/index.vue +1 -1
  22. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/identifying-fields.test.ts +13 -61
  23. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/index.test.ts +33 -6
  24. package/components/Resource/Detail/Metadata/IdentifyingInformation/identifying-fields.ts +24 -38
  25. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +25 -5
  26. package/components/Resource/Detail/Metadata/KeyValue.vue +12 -23
  27. package/components/Resource/Detail/Metadata/KeyValueRow.vue +144 -0
  28. package/components/Resource/Detail/Metadata/Labels/__tests__/index.test.ts +1 -0
  29. package/components/Resource/Detail/Metadata/Labels/index.vue +1 -0
  30. package/components/Resource/Detail/Metadata/__tests__/KeyValue.test.ts +30 -32
  31. package/components/Resource/Detail/Metadata/__tests__/KeyValueRow.test.ts +108 -0
  32. package/components/Resource/Detail/Metadata/__tests__/composables.test.ts +0 -3
  33. package/components/Resource/Detail/Metadata/__tests__/index.test.ts +12 -5
  34. package/components/Resource/Detail/Metadata/composables.ts +1 -4
  35. package/components/Resource/Detail/Metadata/index.vue +1 -0
  36. package/components/Resource/Detail/Preview/Content.vue +63 -0
  37. package/components/Resource/Detail/Preview/Preview.vue +128 -0
  38. package/components/Resource/Detail/Preview/__tests__/Content.spec.ts +71 -0
  39. package/components/Resource/Detail/Preview/__tests__/Preview.spec.ts +121 -0
  40. package/components/Resource/Detail/ResourcePopover/ResourcePopoverCard.vue +141 -0
  41. package/components/Resource/Detail/ResourcePopover/__tests__/ResourcePopoverCard.test.ts +136 -0
  42. package/components/Resource/Detail/ResourcePopover/__tests__/index.test.ts +245 -0
  43. package/components/Resource/Detail/ResourcePopover/index.vue +226 -0
  44. package/components/Resource/Detail/SpacedRow.vue +1 -0
  45. package/components/Resource/Detail/TitleBar/__tests__/composables.test.ts +0 -5
  46. package/components/Resource/Detail/TitleBar/__tests__/index.test.ts +1 -1
  47. package/components/Resource/Detail/TitleBar/composables.ts +1 -3
  48. package/components/Resource/Detail/TitleBar/index.vue +2 -29
  49. package/components/Resource/Detail/ViewOptions/composable.ts +9 -0
  50. package/components/Resource/Detail/ViewOptions/index.vue +41 -0
  51. package/components/Resource/Detail/__tests__/CopyToClipboard.spec.ts +82 -0
  52. package/components/ResourceDetail/Masthead/legacy.vue +0 -19
  53. package/components/ResourceDetail/index.vue +1 -26
  54. package/components/ResourceTable.vue +24 -0
  55. package/components/SortableTable/index.vue +7 -1
  56. package/components/SortableTable/paging.js +3 -0
  57. package/components/Tabbed/Tab.vue +43 -1
  58. package/components/Tabbed/index.vue +3 -1
  59. package/components/__tests__/Cron/CronExpressionEditor.test.ts +151 -0
  60. package/components/__tests__/Cron/CronExpressionEditorModal.test.ts +81 -0
  61. package/components/auth/login/saml.vue +86 -0
  62. package/components/form/LabeledSelect.vue +8 -8
  63. package/components/form/ResourceTabs/composable.ts +54 -0
  64. package/components/form/ResourceTabs/index.vue +10 -7
  65. package/components/form/Select.vue +13 -10
  66. package/components/form/__tests__/LabeledSelect.test.ts +133 -0
  67. package/components/form/__tests__/Select.test.ts +134 -0
  68. package/composables/useExtensionManager.ts +17 -0
  69. package/config/home-links.js +12 -0
  70. package/config/labels-annotations.js +0 -1
  71. package/config/page-actions.js +0 -1
  72. package/config/product/explorer.js +3 -1
  73. package/config/product/fleet.js +2 -7
  74. package/config/product/manager.js +0 -5
  75. package/config/query-params.js +1 -0
  76. package/config/router/navigation-guards/clusters.js +2 -1
  77. package/config/router/navigation-guards/products.js +1 -1
  78. package/core/extension-manager-impl.js +518 -0
  79. package/core/plugins.js +35 -468
  80. package/core/types.ts +8 -2
  81. package/detail/__tests__/autoscaling.horizontalpodautoscaler.test.ts +1 -0
  82. package/detail/catalog.cattle.io.app.vue +7 -4
  83. package/detail/fleet.cattle.io.bundle.vue +1 -5
  84. package/detail/fleet.cattle.io.cluster.vue +3 -2
  85. package/detail/fleet.cattle.io.gitrepo.vue +76 -49
  86. package/detail/fleet.cattle.io.helmop.vue +78 -49
  87. package/dialog/AddonConfigConfirmationDialog.vue +1 -1
  88. package/dialog/GenericPrompt.vue +1 -1
  89. package/dialog/ImportDialog.vue +9 -2
  90. package/dialog/InstallExtensionDialog.vue +18 -10
  91. package/edit/__tests__/fleet.cattle.io.gitrepo.test.ts +2 -1
  92. package/edit/__tests__/resources.cattle.io.restore.test.ts +106 -0
  93. package/edit/cloudcredential.vue +31 -17
  94. package/edit/constraints.gatekeeper.sh.constraint/index.vue +10 -2
  95. package/edit/fleet.cattle.io.cluster.vue +19 -0
  96. package/edit/fleet.cattle.io.gitrepo.vue +23 -16
  97. package/edit/monitoring.coreos.com.alertmanagerconfig/index.vue +12 -11
  98. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +11 -1
  99. package/edit/provisioning.cattle.io.cluster/index.vue +14 -19
  100. package/edit/provisioning.cattle.io.cluster/rke2.vue +11 -3
  101. package/edit/resources.cattle.io.restore.vue +5 -8
  102. package/list/__tests__/workload.test.ts +1 -0
  103. package/list/workload.vue +8 -1
  104. package/machine-config/components/GCEImage.vue +6 -5
  105. package/machine-config/google.vue +11 -6
  106. package/mixins/__tests__/chart.test.ts +139 -1
  107. package/mixins/chart.js +58 -18
  108. package/models/__tests__/namespace.test.ts +69 -0
  109. package/models/apps.statefulset.js +8 -10
  110. package/models/chart.js +5 -1
  111. package/models/fleet-application.js +16 -46
  112. package/models/fleet.cattle.io.bundle.js +1 -38
  113. package/models/fleet.cattle.io.gitrepo.js +4 -0
  114. package/models/fleet.cattle.io.helmop.js +4 -0
  115. package/models/management.cattle.io.project.js +12 -0
  116. package/models/namespace.js +30 -0
  117. package/models/workload.js +3 -0
  118. package/package.json +10 -10
  119. package/pages/c/_cluster/apps/charts/AppChartCardFooter.vue +26 -11
  120. package/pages/c/_cluster/apps/charts/chart.vue +29 -20
  121. package/pages/c/_cluster/apps/charts/index.vue +1 -0
  122. package/pages/c/_cluster/apps/charts/install.vue +6 -5
  123. package/pages/c/_cluster/explorer/tools/__tests__/index.test.ts +102 -12
  124. package/pages/c/_cluster/explorer/tools/index.vue +145 -254
  125. package/pages/c/_cluster/manager/cloudCredential/index.vue +18 -1
  126. package/pages/c/_cluster/manager/drivers/kontainerDriver/index.vue +12 -2
  127. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +1 -1
  128. package/pages/c/_cluster/uiplugins/__tests__/index.spec.ts +318 -0
  129. package/pages/c/_cluster/uiplugins/index.vue +221 -363
  130. package/pages/home.vue +1 -9
  131. package/plugins/dashboard-store/resource-class.js +49 -0
  132. package/public/index.html +2 -1
  133. package/rancher-components/Card/Card.vue +1 -1
  134. package/rancher-components/Form/Checkbox/Checkbox.vue +1 -1
  135. package/rancher-components/Form/Radio/RadioButton.vue +1 -1
  136. package/rancher-components/Form/Radio/RadioGroup.vue +1 -1
  137. package/rancher-components/LabeledTooltip/LabeledTooltip.vue +1 -11
  138. package/rancher-components/Pill/RcCounterBadge/RcCounterBadge.test.ts +53 -0
  139. package/rancher-components/Pill/RcCounterBadge/RcCounterBadge.vue +65 -0
  140. package/rancher-components/Pill/RcCounterBadge/index.ts +1 -0
  141. package/rancher-components/Pill/RcCounterBadge/types.ts +7 -0
  142. package/rancher-components/Pill/RcStatusBadge/RcStatusBadge.vue +1 -1
  143. package/rancher-components/Pill/RcStatusBadge/index.ts +1 -1
  144. package/rancher-components/Pill/RcStatusIndicator/RcStatusIndicator.vue +3 -3
  145. package/rancher-components/Pill/RcStatusIndicator/types.ts +1 -1
  146. package/rancher-components/Pill/RcTag/RcTag.test.ts +64 -0
  147. package/rancher-components/Pill/RcTag/RcTag.vue +94 -0
  148. package/rancher-components/Pill/RcTag/index.ts +1 -0
  149. package/rancher-components/Pill/RcTag/types.ts +9 -0
  150. package/rancher-components/Pill/types.ts +1 -0
  151. package/rancher-components/RcItemCard/RcItemCard.vue +1 -0
  152. package/rancher-components/RcItemCard/RcItemCardAction.vue +12 -0
  153. package/store/__tests__/catalog.test.ts +63 -0
  154. package/store/catalog.js +2 -2
  155. package/store/type-map.js +3 -15
  156. package/types/extension-manager.ts +26 -0
  157. package/types/shell/index.d.ts +121 -16
  158. package/utils/__tests__/product.test.ts +129 -0
  159. package/utils/__tests__/resource.test.ts +87 -0
  160. package/utils/alertmanagerconfig.js +2 -2
  161. package/utils/auth.js +3 -76
  162. package/utils/product.ts +39 -0
  163. package/utils/resource.ts +35 -0
  164. package/utils/select.js +0 -24
  165. package/utils/validators/formRules/__tests__/index.test.ts +3 -0
  166. package/utils/validators/formRules/index.ts +2 -1
  167. package/vue.config.js +1 -1
  168. package/components/Resource/Detail/Metadata/Rectangle.vue +0 -34
  169. package/components/Resource/Detail/Metadata/__tests__/Rectangle.test.ts +0 -24
  170. package/components/ResourceDetail/Masthead/__tests__/legacy.test.ts +0 -65
  171. /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>
@@ -37,6 +37,17 @@ export default {
37
37
  allCredentials: null,
38
38
  resource: NORMAN.CLOUD_CREDENTIAL,
39
39
  schema: this.$store.getters['rancher/schemaFor'](NORMAN.CLOUD_CREDENTIAL),
40
+ groupOptions: [{
41
+ tooltipKey: 'resourceTable.groupBy.none',
42
+ icon: 'icon-list-flat',
43
+ value: 'none',
44
+ }, {
45
+ tooltipKey: 'manager.cloudCredentials.list.groupBy.provider',
46
+ hideColumn: 'provider',
47
+ icon: 'icon-folder',
48
+ value: 'providerDisplay',
49
+ field: 'providerDisplay',
50
+ }],
40
51
  };
41
52
  },
42
53
 
@@ -56,6 +67,12 @@ export default {
56
67
  sort: 'publicData',
57
68
  search: 'publicData',
58
69
  formatter: 'CloudCredPublicData',
70
+ }, {
71
+ name: 'provider',
72
+ labelKey: 'manager.cloudCredentials.list.headers.provider',
73
+ value: 'providerDisplay',
74
+ sort: 'providerDisplay',
75
+ search: 'providerDisplay',
59
76
  },
60
77
  DESCRIPTION,
61
78
  AGE_NORMAN
@@ -91,7 +108,7 @@ export default {
91
108
  :rows="rows"
92
109
  :headers="headers"
93
110
  :namespaced="false"
94
- group-by="providerDisplay"
111
+ :group-options="groupOptions"
95
112
  >
96
113
  <template #cell:id="{row}">
97
114
  {{ row.id.replace('cattle-global-data:', '') }}
@@ -1,13 +1,16 @@
1
1
  <script>
2
2
  import { NORMAN } from '@shell/config/types';
3
+ import { isAdminUser } from '@shell/store/type-map';
3
4
  import ResourceTable from '@shell/components/ResourceTable';
4
5
  import AsyncButton from '@shell/components/AsyncButton';
5
6
  import Loading from '@shell/components/Loading';
6
7
  import Masthead from '@shell/components/ResourceList/Masthead';
8
+ import Banner from '@components/Banner/Banner.vue';
9
+
7
10
  export default {
8
11
  name: 'KontainerDrivers',
9
12
  components: {
10
- ResourceTable, Loading, Masthead, AsyncButton
13
+ ResourceTable, Loading, Masthead, AsyncButton, Banner
11
14
  },
12
15
 
13
16
  async fetch() {
@@ -21,7 +24,8 @@ export default {
21
24
  resource: NORMAN.KONTAINER_DRIVER,
22
25
  schema: this.$store.getters['rancher/schemaFor'](NORMAN.KONTAINER_DRIVER),
23
26
  useQueryParamsForSimpleFiltering: false,
24
- forceUpdateLiveAndDelayed: 10
27
+ forceUpdateLiveAndDelayed: 10,
28
+ showDeprecationBanner: isAdminUser(this.$store.getters),
25
29
  };
26
30
  },
27
31
  computed: {
@@ -69,6 +73,12 @@ export default {
69
73
  />
70
74
  </template>
71
75
  </Masthead>
76
+ <Banner
77
+ v-if="showDeprecationBanner"
78
+ color="warning"
79
+ label-key="drivers.kontainer.emberDeprecationMessage"
80
+ data-testid="kontainer-driver-ember-deprecation-banner"
81
+ />
72
82
  <ResourceTable
73
83
  :schema="schema"
74
84
  :rows="rows"
@@ -223,7 +223,7 @@ export default {
223
223
  :aria-label="t('plugins.closePluginPanel')"
224
224
  tabindex="0"
225
225
  @click="hide()"
226
- @keyup.enter.space="hide()"
226
+ @keydown.enter.space="hide()"
227
227
  >
228
228
  <i class="icon icon-close" />
229
229
  </div>