@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
@@ -1,7 +1,6 @@
1
1
  <script>
2
2
  import CreateEditView from '@shell/mixins/create-edit-view';
3
3
  import LabeledSelect from '@shell/components/form/LabeledSelect';
4
- import { Banner } from '@components/Banner';
5
4
 
6
5
  import { get, set } from '@shell/utils/object';
7
6
  import { MANAGEMENT, VIRTUAL_HARVESTER_PROVIDER } from '@shell/config/types';
@@ -9,7 +8,7 @@ import { MANAGEMENT, VIRTUAL_HARVESTER_PROVIDER } from '@shell/config/types';
9
8
  export default {
10
9
  emits: ['validationChanged'],
11
10
 
12
- components: { LabeledSelect, Banner },
11
+ components: { LabeledSelect },
13
12
  mixins: [CreateEditView],
14
13
 
15
14
  async fetch() {
@@ -100,13 +99,6 @@ export default {
100
99
 
101
100
  <template>
102
101
  <div>
103
- <div class="row mb-10">
104
- <Banner
105
- color="warning"
106
- label-key="cluster.credential.harvester.tokenExpirationWarning"
107
- data-testid="harvester-token-expiration-warning-banner"
108
- />
109
- </div>
110
102
  <div class="row mb-10">
111
103
  <div
112
104
  class="col span-6"
@@ -28,7 +28,7 @@ const openChanged = (event: boolean) => {
28
28
  }
29
29
  };
30
30
 
31
- const emit = defineEmits<{(event: string, payload: any): void}>();
31
+ const emit = defineEmits<{(event: string, payload: any): void;(event: 'action-invoked'): void;}>();
32
32
  const route = useRoute();
33
33
 
34
34
  const execute = (action: any, event: MouseEvent, args?: any) => {
@@ -36,6 +36,8 @@ const execute = (action: any, event: MouseEvent, args?: any) => {
36
36
  return;
37
37
  }
38
38
 
39
+ emit('action-invoked');
40
+
39
41
  // this will come from extensions...
40
42
  if (action.invoke) {
41
43
  const fn = action.invoke;
@@ -32,9 +32,17 @@ export default {
32
32
  v-t="show ? 'generic.hideAdvanced' : 'generic.showAdvanced'"
33
33
  class="hand block"
34
34
  :class="{'mb-10': show}"
35
+ tabindex="0"
36
+ @keydown.enter.space.prevent="toggle"
35
37
  @click="toggle"
36
38
  />
37
39
 
38
40
  <slot v-if="show" />
39
41
  </div>
40
42
  </template>
43
+
44
+ <style lang="scss" scoped>
45
+ .hand.block {
46
+ max-width: fit-content;
47
+ }
48
+ </style>
@@ -8,6 +8,14 @@ export default {
8
8
  type: Object,
9
9
  required: true
10
10
  },
11
+ showAppReadme: {
12
+ type: Boolean,
13
+ default: true
14
+ },
15
+ hideReadmeFirstTitle: {
16
+ type: Boolean,
17
+ default: true
18
+ }
11
19
  },
12
20
  data() {
13
21
  return {
@@ -30,13 +38,13 @@ export default {
30
38
  <div class="wrapper">
31
39
  <div class="chart-readmes">
32
40
  <Markdown
33
- v-if="appReadme"
41
+ v-if="showAppReadme && appReadme"
34
42
  v-model:value="appReadme"
35
- class="md md-desc mb-20"
43
+ :class="[hideReadmeFirstTitle ? 'hidden-first-title' : '', 'md', 'md-desc', 'mb-20']"
36
44
  @loaded="appReadmeLoaded = true"
37
45
  />
38
46
  <h1
39
- v-if="appReadme && readme && appReadmeLoaded && readmeLoaded"
47
+ v-if="showAppReadme && appReadme && readme && appReadmeLoaded && readmeLoaded"
40
48
  class="pt-10"
41
49
  >
42
50
  {{ t('catalog.install.appReadmeTitle') }}
@@ -45,6 +53,7 @@ export default {
45
53
  v-if="readme"
46
54
  v-model:value="readme"
47
55
  class="md md-desc"
56
+ :class="[hideReadmeFirstTitle ? 'hidden-first-title' : '', 'md', 'md-desc']"
48
57
  @loaded="readmeLoaded = true"
49
58
  />
50
59
  </div>
@@ -88,13 +97,14 @@ export default {
88
97
  word-break: break-word;
89
98
  }
90
99
 
91
- :deep() > h1:first-of-type {
92
- display: none;
93
- }
94
-
95
100
  :deep() p {
96
101
  margin-bottom: 0.5em;
97
102
  }
98
103
  }
104
+ .hidden-first-title {
105
+ :deep() > h1:first-of-type {
106
+ display: none;
107
+ }
108
+ }
99
109
 
100
110
  </style>
@@ -0,0 +1,299 @@
1
+ <script setup lang="ts">
2
+ import {
3
+ reactive, computed, watch, ref, nextTick
4
+ } from 'vue';
5
+ import { useStore } from 'vuex';
6
+ import { isValidCron } from 'cron-validator';
7
+ import cronstrue from 'cronstrue';
8
+ import { createPopper, Instance as PopperInstance } from '@popperjs/core';
9
+ import { useI18n } from '@shell/composables/useI18n';
10
+ import LabeledInput from '@components/Form/LabeledInput/LabeledInput.vue';
11
+ import CronTooltip from './CronTooltip.vue';
12
+ import type { TooltipSection, CronField } from './types';
13
+ import { cronFields } from './types';
14
+
15
+ const props = defineProps<{
16
+ /**
17
+ * Initial cron expression string.
18
+ */
19
+ cronExpression?: string;
20
+ }>();
21
+
22
+ // eslint-disable-next-line
23
+ const emit = defineEmits<{
24
+ (e: 'update:isValid', value: boolean): void;
25
+ (e: 'update:readableCron', value: string): void;
26
+ (e: 'update:cronExpression', value: string): void;
27
+ }>();
28
+
29
+ const store = useStore();
30
+ const { t } = useI18n(store);
31
+ const fields: CronField[] = cronFields;
32
+
33
+ const fieldLabels: Record<CronField, string> = {
34
+ minute: 'component.cron.expressionEditor.label.minute',
35
+ hour: 'component.cron.expressionEditor.label.hour',
36
+ dayOfMonth: 'component.cron.expressionEditor.label.dayOfMonth',
37
+ month: 'component.cron.expressionEditor.label.month',
38
+ dayOfWeek: 'component.cron.expressionEditor.label.dayOfWeek',
39
+ };
40
+
41
+ function makeFieldRecord<T>(value: T): Record<CronField, T> {
42
+ return cronFields.reduce((acc, f) => {
43
+ acc[f] = value;
44
+
45
+ return acc;
46
+ }, {} as Record<CronField, T>);
47
+ }
48
+
49
+ function parseCronToFields(expr: string): Record<CronField, string> {
50
+ const parts = expr?.trim().split(' ') || [];
51
+ const record = makeFieldRecord('');
52
+
53
+ fields.forEach((f, idx) => {
54
+ record[f] = parts[idx] || '';
55
+ });
56
+
57
+ return record;
58
+ }
59
+
60
+ const cronValues = reactive<Record<CronField, string>>(parseCronToFields(props.cronExpression || '* * * * *'));
61
+ const errors = reactive<Record<CronField, boolean>>(makeFieldRecord(false));
62
+ const focusedField = reactive<Record<CronField, boolean>>(makeFieldRecord(false));
63
+ const rootRef = ref<HTMLElement | null>(null);
64
+ const wrapperRefs: Record<CronField, HTMLElement | null> = makeFieldRecord(null);
65
+ const tooltipRefs: Record<CronField, HTMLElement | null> = makeFieldRecord(null);
66
+ const popperInstances: Record<CronField, PopperInstance | null> = makeFieldRecord(null);
67
+
68
+ const tooltipData: Record<CronField, TooltipSection[]> = {
69
+ minute: [
70
+ {
71
+ type: 'rules',
72
+ items: [
73
+ { value: '*', descKey: 'component.cron.expressionEditor.minute.any' },
74
+ { value: '1,5', descKey: 'component.cron.expressionEditor.minute.at1and5' },
75
+ { value: '1-5', descKey: 'component.cron.expressionEditor.minute.range' },
76
+ { value: '*/5', descKey: 'component.cron.expressionEditor.minute.every5' },
77
+ { value: '8/5', descKey: 'component.cron.expressionEditor.minute.start8' },
78
+ ]
79
+ },
80
+ {
81
+ type: 'explanation',
82
+ items: [
83
+ { descKey: 'component.cron.expressionEditor.minute.allowed' },
84
+ ]
85
+ }
86
+ ],
87
+ hour: [
88
+ {
89
+ type: 'rules',
90
+ items: [
91
+ { value: '*', descKey: 'component.cron.expressionEditor.hour.any' },
92
+ { value: '1,5', descKey: 'component.cron.expressionEditor.hour.at1and5' },
93
+ { value: '1-5', descKey: 'component.cron.expressionEditor.hour.range' },
94
+ { value: '*/5', descKey: 'component.cron.expressionEditor.hour.every5' },
95
+ { value: '8/5', descKey: 'component.cron.expressionEditor.hour.start8' },
96
+ ]
97
+ },
98
+ {
99
+ type: 'explanation',
100
+ items: [
101
+ { descKey: 'component.cron.expressionEditor.hour.allowed' },
102
+ ]
103
+ }
104
+ ],
105
+ dayOfMonth: [
106
+ {
107
+ type: 'rules',
108
+ items: [
109
+ { value: '*', descKey: 'component.cron.expressionEditor.dayOfMonth.any' },
110
+ { value: '?', descKey: 'component.cron.expressionEditor.dayOfMonth.omit' },
111
+ { value: '1,5', descKey: 'component.cron.expressionEditor.dayOfMonth.1and5' },
112
+ { value: '1-5', descKey: 'component.cron.expressionEditor.dayOfMonth.range' },
113
+ { value: '*/5', descKey: 'component.cron.expressionEditor.dayOfMonth.every5' },
114
+ { value: '8/5', descKey: 'component.cron.expressionEditor.dayOfMonth.start8' },
115
+ ]
116
+ },
117
+ {
118
+ type: 'explanation',
119
+ items: [
120
+ { descKey: 'component.cron.expressionEditor.dayOfMonth.allowed' },
121
+ ]
122
+ }
123
+ ],
124
+ month: [
125
+ {
126
+ type: 'rules',
127
+ items: [
128
+ { value: '*', descKey: 'component.cron.expressionEditor.month.any' },
129
+ { value: '1,5', descKey: 'component.cron.expressionEditor.month.1and5' },
130
+ { value: '1-5', descKey: 'component.cron.expressionEditor.month.range' },
131
+ { value: '*/2', descKey: 'component.cron.expressionEditor.month.every2' },
132
+ { value: '3/2', descKey: 'component.cron.expressionEditor.month.start3' },
133
+ ]
134
+ },
135
+ {
136
+ type: 'explanation',
137
+ items: [
138
+ { descKey: 'component.cron.expressionEditor.month.allowed' },
139
+ { descKey: 'component.cron.expressionEditor.month.alias' },
140
+ ]
141
+ }
142
+ ],
143
+ dayOfWeek: [
144
+ {
145
+ type: 'rules',
146
+ items: [
147
+ { value: '*', descKey: 'component.cron.expressionEditor.dayOfWeek.any' },
148
+ { value: '?', descKey: 'component.cron.expressionEditor.dayOfWeek.omit' },
149
+ { value: '1,5', descKey: 'component.cron.expressionEditor.dayOfWeek.1and5' },
150
+ { value: '1-5', descKey: 'component.cron.expressionEditor.dayOfWeek.range' },
151
+ ]
152
+ },
153
+ {
154
+ type: 'explanation',
155
+ items: [
156
+ { descKey: 'component.cron.expressionEditor.dayOfWeek.allowed' },
157
+ { descKey: 'component.cron.expressionEditor.dayOfWeek.alias' },
158
+ ]
159
+ }
160
+ ],
161
+ };
162
+
163
+ const validateField = (field: CronField, value: string) => {
164
+ if (!value) {
165
+ errors[field] = true;
166
+
167
+ return;
168
+ }
169
+
170
+ const exprMap: Record<CronField, string> = {
171
+ minute: `${ value } * * * *`,
172
+ hour: `* ${ value } * * *`,
173
+ dayOfMonth: `* * ${ value } * *`,
174
+ month: `* * * ${ value } *`,
175
+ dayOfWeek: `* * * * ${ value }`,
176
+ };
177
+
178
+ errors[field] = !isValidCron(exprMap[field], {
179
+ alias: true,
180
+ allowBlankDay: true,
181
+ allowSevenAsSunday: true,
182
+ });
183
+ };
184
+
185
+ fields.forEach((f) => validateField(f, cronValues[f]));
186
+
187
+ const isValid = computed(() => !Object.values(errors).some(Boolean));
188
+ const expression = computed(() => fields.map((f) => cronValues[f]).join(' '));
189
+ const readableCron = computed(() => {
190
+ if (!isValid.value) return t('component.cron.expressionEditor.invalidCronExpression');
191
+ try {
192
+ return cronstrue.toString(expression.value);
193
+ } catch {
194
+ return t('component.cron.expressionEditor.invalidCronExpression');
195
+ }
196
+ });
197
+
198
+ watch(cronValues, () => {
199
+ emit('update:cronExpression', expression.value);
200
+ emit('update:readableCron', readableCron.value);
201
+ emit('update:isValid', isValid.value);
202
+ }, { deep: true, immediate: true });
203
+
204
+ const handleInput = (field: CronField, val: string) => {
205
+ cronValues[field] = val;
206
+ validateField(field, val);
207
+ };
208
+
209
+ const handleFocus = async(field: CronField) => {
210
+ focusedField[field] = true;
211
+ await nextTick();
212
+ if (wrapperRefs[field] && tooltipRefs[field]) {
213
+ popperInstances[field] = createPopper(wrapperRefs[field], tooltipRefs[field], {
214
+ placement: 'bottom-start',
215
+ modifiers: [
216
+ { name: 'flip', options: { fallbackPlacements: ['top-start', 'bottom-end'] } },
217
+ { name: 'preventOverflow', options: { boundary: rootRef.value || document.body, padding: 4 } },
218
+ { name: 'offset', options: { offset: [0, 4] } },
219
+ ],
220
+ });
221
+ }
222
+ };
223
+
224
+ const handleBlur = (field: CronField) => {
225
+ focusedField[field] = false;
226
+ popperInstances[field]?.destroy();
227
+ popperInstances[field] = null;
228
+ };
229
+ </script>
230
+
231
+ <template>
232
+ <div
233
+ ref="rootRef"
234
+ class="cron-edit"
235
+ v-bind="$attrs"
236
+ >
237
+ <div class="cron-row">
238
+ <div
239
+ v-for="field in fields"
240
+ :key="field"
241
+ :ref="el => wrapperRefs[field] = el as HTMLElement"
242
+ class="input-wrapper"
243
+ >
244
+ <LabeledInput
245
+ :label="t(fieldLabels[field])"
246
+ :value="cronValues[field]"
247
+ :status="errors[field] ? 'error' : undefined"
248
+ :tooltip="errors[field] ? t('component.cron.expressionEditor.invalidValue') : ''"
249
+ :aria-invalid="!!errors[field]"
250
+ :aria-label="t('component.cron.expressionEditor.a11y.examples', { label: t(fieldLabels[field]) })"
251
+ :aria-describedby="`tooltip-${field}`"
252
+ @update:value="val => handleInput(field, val)"
253
+ @focus="() => handleFocus(field)"
254
+ @blur="() => handleBlur(field)"
255
+ />
256
+ <div
257
+ v-show="focusedField[field]"
258
+ :id="`tooltip-${field}`"
259
+ :ref="el => tooltipRefs[field] = el as HTMLElement"
260
+ role="tooltip"
261
+ class="cron-tooltip-wrapper"
262
+ >
263
+ <CronTooltip :sections="tooltipData[field]" />
264
+ </div>
265
+ </div>
266
+ </div>
267
+ </div>
268
+ </template>
269
+
270
+ <style scoped lang="scss">
271
+ $input-max-width: 110px;
272
+
273
+ .cron-row {
274
+ display: flex;
275
+ justify-content: center;
276
+ flex-wrap: wrap;
277
+ gap: 8px;
278
+ }
279
+
280
+ .input-wrapper {
281
+ max-width: $input-max-width;
282
+ flex: 1 1 auto;
283
+
284
+ .label {
285
+ color: var(--label-secondary);
286
+ font-size: 12px;
287
+ }
288
+ }
289
+
290
+ .cron-tooltip-wrapper {
291
+ padding: 16px;
292
+ background: var(--body-bg);
293
+ border: 1px solid var(--border);
294
+ border-radius: var(--border-radius-lg);
295
+ box-shadow: 0 2px 8px var(--shadow);
296
+ display: inline-block;
297
+ z-index: 2;
298
+ }
299
+ </style>
@@ -0,0 +1,247 @@
1
+ <script setup lang="ts">
2
+ import {
3
+ ref, watch, onMounted, onBeforeUnmount, nextTick
4
+ } from 'vue';
5
+ import { useStore } from 'vuex';
6
+ import { useI18n } from '@shell/composables/useI18n';
7
+ import AppModal from '@shell/components/AppModal.vue';
8
+ import CronExpressionEditor from './CronExpressionEditor.vue';
9
+
10
+ const props = defineProps<{
11
+ /**
12
+ * Initial cron expression string.
13
+ */
14
+ cronExpression?: string;
15
+ /**
16
+ * Controls whether the cron editor modal is visible.
17
+ */
18
+ show: boolean;
19
+ }>();
20
+
21
+ // eslint-disable-next-line
22
+ const emit = defineEmits<{
23
+ (e: 'update:cronExpression', value: string): void;
24
+ (e: 'update:show', value: boolean): void;
25
+ (e: 'update:readableCron', value: string): void;
26
+ }>();
27
+
28
+ const store = useStore();
29
+ const { t } = useI18n(store);
30
+
31
+ const localCron = ref(props.cronExpression ?? '* * * * *');
32
+ const localShow = ref(props.show);
33
+ const readableCron = ref('');
34
+ const isCronValid = ref(true);
35
+
36
+ const cronInfoRef = ref<HTMLElement | null>(null);
37
+ const modalBodyRef = ref<HTMLElement | null>(null);
38
+ const modalWidth = ref('600px');
39
+
40
+ const wildcards = [
41
+ { symbol: '*', desc: 'component.cron.expressionEditorModal.wildcards.anyValue' },
42
+ { symbol: 'X,Y', desc: 'component.cron.expressionEditorModal.wildcards.xAndY' },
43
+ { symbol: 'X-Y', desc: 'component.cron.expressionEditorModal.wildcards.fromXtoY' },
44
+ { symbol: '*/X', desc: 'component.cron.expressionEditorModal.wildcards.everyX' },
45
+ { symbol: 'Y/X', desc: 'component.cron.expressionEditorModal.wildcards.everyXStartingY' },
46
+ { symbol: 'Y-Z/X', desc: 'component.cron.expressionEditorModal.wildcards.everyXFromYtoZ' },
47
+ ];
48
+
49
+ const examples = [
50
+ { cron: '0 0 * * *', desc: 'component.cron.expressionEditorModal.examples.dailyMidnight' },
51
+ { cron: '0 */5 * * *', desc: 'component.cron.expressionEditorModal.examples.every5Hours' },
52
+ { cron: '45 17 1 * *', desc: 'component.cron.expressionEditorModal.examples.day1At1745' },
53
+ { cron: '30 8/1 * * 1-5', desc: 'component.cron.expressionEditorModal.examples.weekdaysAt0830' },
54
+ { cron: '0 */1 * 3,4,5 *', desc: 'component.cron.expressionEditorModal.examples.marchToMayHourly' },
55
+ { cron: '0 9-17/4 * * *', desc: 'component.cron.expressionEditorModal.examples.every4Hours9to17' },
56
+ ];
57
+
58
+ const closeModal = () => emit('update:show', false);
59
+ const confirmCron = () => {
60
+ if (!isCronValid.value) return;
61
+ emit('update:cronExpression', localCron.value);
62
+ emit('update:readableCron', readableCron.value);
63
+ closeModal();
64
+ };
65
+
66
+ // dynamically update modal width based on content
67
+ const updateWidth = () => {
68
+ if (!modalBodyRef.value || !cronInfoRef.value) return;
69
+
70
+ const bodyStyle = getComputedStyle(modalBodyRef.value);
71
+ const padding = parseFloat(bodyStyle.paddingLeft) + parseFloat(bodyStyle.paddingRight);
72
+ const extraBuffer = 10;
73
+ const contentWidth = cronInfoRef.value.scrollWidth + padding + extraBuffer;
74
+
75
+ // limit width to 90% of viewport
76
+ modalWidth.value = `${ Math.min(contentWidth, window.innerWidth * 0.9) }px`;
77
+ };
78
+
79
+ watch(() => props.cronExpression, (val) => {
80
+ if (val !== undefined) localCron.value = val;
81
+ });
82
+ watch(() => props.show, (val) => {
83
+ localShow.value = val;
84
+
85
+ if (val) {
86
+ // reset cron to prop when modal opens
87
+ localCron.value = props.cronExpression ?? '* * * * *';
88
+ nextTick(updateWidth);
89
+ }
90
+ });
91
+
92
+ onMounted(() => {
93
+ nextTick(updateWidth);
94
+ window.addEventListener('resize', updateWidth);
95
+ });
96
+
97
+ onBeforeUnmount(() => {
98
+ window.removeEventListener('resize', updateWidth);
99
+ });
100
+ </script>
101
+
102
+ <template>
103
+ <AppModal
104
+ v-if="localShow"
105
+ :width="modalWidth"
106
+ name="cron-editor-modal"
107
+ custom-class="cron-editor-modal"
108
+ aria-labelledby="cron-editor-title"
109
+ aria-describedby="cron-editor-desc"
110
+ trigger-focus-trap
111
+ @close="closeModal"
112
+ >
113
+ <div
114
+ ref="modalBodyRef"
115
+ class="modal-body"
116
+ >
117
+ <h4 id="cron-editor-title">
118
+ {{ t('component.cron.expressionEditorModal.title') }}
119
+ </h4>
120
+ <p
121
+ id="cron-editor-desc"
122
+ class="description"
123
+ >
124
+ {{ t('component.cron.expressionEditorModal.description') }}
125
+ </p>
126
+
127
+ <div
128
+ class="readableCron"
129
+ aria-live="polite"
130
+ >
131
+ {{ readableCron }}
132
+ </div>
133
+
134
+ <CronExpressionEditor
135
+ v-model:cron-expression="localCron"
136
+ v-model:readable-cron="readableCron"
137
+ v-model:is-valid="isCronValid"
138
+ class="custom-cron-editor"
139
+ />
140
+
141
+ <div
142
+ ref="cronInfoRef"
143
+ class="cron-info"
144
+ >
145
+ <div class="cron-wildcards">
146
+ <h5>{{ t('component.cron.expressionEditorModal.wildcards.title') }}</h5>
147
+ <ul>
148
+ <li
149
+ v-for="(item, idx) in wildcards"
150
+ :key="idx"
151
+ >
152
+ <span class="symbol">{{ item.symbol }}</span>
153
+ <span class="desc">{{ t(item.desc) }}</span>
154
+ </li>
155
+ </ul>
156
+ </div>
157
+
158
+ <div class="cron-examples">
159
+ <h5>{{ t('component.cron.expressionEditorModal.examples.title') }}</h5>
160
+ <ul>
161
+ <li
162
+ v-for="(ex, idx) in examples"
163
+ :key="idx"
164
+ >
165
+ <span class="symbol">{{ ex.cron }}</span>
166
+ <span class="desc">{{ t(ex.desc) }}</span>
167
+ </li>
168
+ </ul>
169
+ </div>
170
+ </div>
171
+ </div>
172
+
173
+ <div class="modal-footer">
174
+ <button
175
+ class="btn btn-sm role-secondary"
176
+ @click="closeModal"
177
+ >
178
+ {{ t('generic.cancel') }}
179
+ </button>
180
+ <button
181
+ class="btn btn-sm role-primary ml-10"
182
+ :disabled="!isCronValid"
183
+ @click="confirmCron"
184
+ >
185
+ {{ t('generic.confirm') }}
186
+ </button>
187
+ </div>
188
+ </AppModal>
189
+ </template>
190
+
191
+ <style scoped lang="scss">
192
+ :global(#modals .cron-editor-modal) {
193
+ border-radius: var(--border-radius-lg);
194
+ }
195
+
196
+ .modal-body {
197
+ padding: 20px 20px 8px;
198
+
199
+ .description {
200
+ margin: 16px 0;
201
+ }
202
+
203
+ .readableCron {
204
+ padding: 16px;
205
+ background-color: var(--disabled-banner-bg);
206
+ }
207
+
208
+ .custom-cron-editor {
209
+ margin: 64px auto;
210
+ max-width: 600px;
211
+ }
212
+
213
+ .cron-info {
214
+ display: flex;
215
+ gap: 52px;
216
+ flex-wrap: nowrap;
217
+ overflow-x: auto;
218
+
219
+ ul {
220
+ list-style: none;
221
+ padding: 0;
222
+ margin: 16px 0;
223
+ display: grid;
224
+ grid-template-columns: max-content 1fr;
225
+ gap: 8px 10px;
226
+
227
+ li {
228
+ display: contents;
229
+ white-space: nowrap;
230
+ color: var(--input-label);
231
+ font-size: 12px;
232
+ }
233
+
234
+ .symbol {
235
+ color: var(--body-text);
236
+ }
237
+ }
238
+ }
239
+ }
240
+
241
+ .modal-footer {
242
+ border-top: 1px solid var(--border);
243
+ display: flex;
244
+ padding: 10px 20px;
245
+ justify-content: flex-end;
246
+ }
247
+ </style>