@cqa-lib/cqa-ui 1.1.281 → 1.1.283

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.
@@ -31,8 +31,13 @@ export class CreateStepGroupComponent {
31
31
  ? 'Group 1 selected step into a container'
32
32
  : `Group ${n} selected steps into a container`;
33
33
  }
34
- /** Display label for a step in the "Steps to group" list (same pattern as normal-step display). */
34
+ /** Display label for a step in the "Steps to group" list (same as main step list - use action when available). */
35
35
  getStepDisplayLabel(step, index) {
36
+ // Prefer step.action - it matches the full display in the step list (e.g. "Enter @|password| in the Email")
37
+ const rawAction = step.action != null && String(step.action).trim() !== '' ? String(step.action) : '';
38
+ if (rawAction) {
39
+ return this.stripHtml(rawAction);
40
+ }
36
41
  if (isNormalStepConfig(step)) {
37
42
  return this.getNormalStepLabel(step);
38
43
  }
@@ -56,6 +61,15 @@ export class CreateStepGroupComponent {
56
61
  }
57
62
  return `Step ${index + 1}`;
58
63
  }
64
+ /** Strip HTML tags for plain-text display (action may contain spans for badges). */
65
+ stripHtml(html) {
66
+ if (typeof document !== 'undefined') {
67
+ const tmp = document.createElement('div');
68
+ tmp.innerHTML = html;
69
+ return (tmp.textContent || tmp.innerText || '').trim();
70
+ }
71
+ return html.replace(/<[^>]*>/g, '').replace(/&nbsp;/g, ' ').trim();
72
+ }
59
73
  getNormalStepLabel(step) {
60
74
  const params = step.parameters ?? [];
61
75
  const getParam = (name) => params.find((p) => p.name?.toLowerCase() === name.toLowerCase());
@@ -111,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
111
125
  }], cancelled: [{
112
126
  type: Output
113
127
  }] } });
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-step-group.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.ts","../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { FormBuilder, FormGroup, Validators } from '@angular/forms';\nimport {\n  TestCaseStepConfig,\n  NormalStepConfig,\n  TestCaseEventType,\n  isNormalStepConfig,\n} from '../test-case-step.models';\n\nexport interface CreateStepGroupFormData {\n  groupName: string;\n}\n\n/**\n * Create Step Group panel/modal for Test Case Details.\n * Follows the same structure and validation pattern as Test Data Modal and Loop Step.\n * Displays selected steps and allows naming the new group; emits createGroup with group name\n * so the host can create the step group and replace selected steps with it.\n */\n@Component({\n  selector: 'cqa-create-step-group',\n  templateUrl: './create-step-group.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class CreateStepGroupComponent {\n  @Input() stepsToGroup: TestCaseStepConfig[] = [];\n\n  @Output() createGroup = new EventEmitter<CreateStepGroupFormData>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  form: FormGroup;\n\n  constructor(private readonly fb: FormBuilder) {\n    this.form = this.fb.group({\n      groupName: ['', [Validators.required, Validators.minLength(1)]],\n    });\n  }\n\n  get stepsCount(): number {\n    return this.stepsToGroup?.length ?? 0;\n  }\n\n  get subtitleText(): string {\n    const n = this.stepsCount;\n    return n === 1\n      ? 'Group 1 selected step into a container'\n      : `Group ${n} selected steps into a container`;\n  }\n\n  /** Display label for a step in the \"Steps to group\" list (same pattern as normal-step display). */\n  getStepDisplayLabel(step: TestCaseStepConfig, index: number): string {\n    if (isNormalStepConfig(step)) {\n      return this.getNormalStepLabel(step);\n    }\n    if ('groupName' in step && step.groupName) {\n      return step.groupName;\n    }\n    if ('condition' in step && step.condition) {\n      return step.condition;\n    }\n    if ('loopType' in step) {\n      const loop = step as { loopType: string; testDataProfile?: string; condition?: string };\n      if (loop.loopType === 'for' && loop.testDataProfile) {\n        return `For loop: ${loop.testDataProfile}`;\n      }\n      if (loop.loopType === 'while' && loop.condition) {\n        return `While: ${loop.condition}`;\n      }\n    }\n    if (step.description) {\n      return step.description;\n    }\n    return `Step ${index + 1}`;\n  }\n\n  private getNormalStepLabel(step: NormalStepConfig): string {\n    const params = step.parameters ?? [];\n    const getParam = (name: string) =>\n      params.find((p) => p.name?.toLowerCase() === name.toLowerCase());\n    const val = (p: { value?: string; displayValue?: string } | undefined) =>\n      p?.displayValue ?? p?.value ?? '';\n\n    switch (step.eventType as TestCaseEventType) {\n      case 'navigate': {\n        const url = getParam('url');\n        return url ? `Navigate to ${val(url)}` : 'Navigate';\n      }\n      case 'ai-agent': {\n        const instructions = getParam('instructions') ?? getParam('description');\n        return val(instructions) || 'AI Agent step';\n      }\n      case 'type':\n        return val(getParam('text')) || 'Type';\n      case 'click':\n        return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';\n      case 'verify':\n        return val(getParam('description')) || 'Verify';\n      case 'custom': {\n        const desc = getParam('description') ?? getParam('action');\n        return val(desc) || 'Custom step';\n      }\n      default:\n        return val(getParam('description')) || val(params[0]) || 'Step';\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateGroup(): void {\n    if (this.form.invalid || this.stepsCount === 0) {\n      this.form.markAllAsTouched();\n      return;\n    }\n    const groupName = (this.form.get('groupName')?.value ?? '').trim();\n    if (!groupName) {\n      this.form.get('groupName')?.setErrors({ required: true });\n      return;\n    }\n    this.createGroup.emit({ groupName });\n  }\n}\n","<div\n  class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n  <!-- Header: title + close (X) -->\n  <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n      <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0\">\n        Create step group\n      </h2>\n      <p class=\"cqa-text-[14px] cqa-leading-[20px] cqa-text-[#64748B] cqa-m-0\">\n        {{ subtitleText }}\n      </p>\n    </div>\n    <button\n      type=\"button\"\n      (click)=\"onCancel()\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n      title=\"Close\"\n      aria-label=\"Close\">\n      <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n        <path d=\"M18 6L6 18M6 6l12 12\" />\n      </svg>\n    </button>\n  </div>\n\n  <!-- Group name * -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#161617]\">\n      Group name <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <cqa-custom-input\n      placeholder=\"e.g., Authentication flow\"\n      [value]=\"form.get('groupName')?.value\"\n      [fullWidth]=\"true\"\n      size=\"md\"\n      (valueChange)=\"form.get('groupName')?.setValue($event); form.get('groupName')?.updateValueAndValidity()\">\n    </cqa-custom-input>\n    <p *ngIf=\"form.get('groupName')?.invalid && form.get('groupName')?.touched\" class=\"cqa-text-xs cqa-text-red-500 cqa-m-0\">\n      Group name is required.\n    </p>\n  </div>\n\n  <!-- Steps to group (N) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n      Steps to group ({{ stepsCount }})\n    </label>\n    <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-overflow-y-auto cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n      <div\n        *ngFor=\"let step of stepsToGroup; let i = index\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-rounded-[4px] cqa-py-[4px] cqa-px-[8px] cqa-bg-[rgba(216,217,252,0.3)]\">\n        <!-- Numbered badge -->\n        <span\n          class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#3F43EE] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n          {{ i + 1 }}\n        </span>\n        <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#0B0B0C] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n          {{ getStepDisplayLabel(step, i) }}\n        </span>\n      </div>\n    </div>\n  </div>\n\n  <!-- Actions: Cancel | Create group -->\n  <div class=\"cqa-flex cqa-items-stretch cqa-w-full cqa-gap-3\">\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"outlined\"\n        btnSize=\"lg\"\n        text=\"Cancel\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"\n        (clicked)=\"onCancel()\">\n      </cqa-button>\n    </div>\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"filled\"\n        btnSize=\"lg\"\n        text=\"Create group\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE] cqa-bg-[#3F43EE]'\"\n        (clicked)=\"onCreateGroup()\">\n      </cqa-button>\n    </div>\n  </div>\n</div>\n"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-step-group.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.ts","../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { FormBuilder, FormGroup, Validators } from '@angular/forms';\nimport {\n  TestCaseStepConfig,\n  NormalStepConfig,\n  TestCaseEventType,\n  isNormalStepConfig,\n} from '../test-case-step.models';\n\nexport interface CreateStepGroupFormData {\n  groupName: string;\n}\n\n/**\n * Create Step Group panel/modal for Test Case Details.\n * Follows the same structure and validation pattern as Test Data Modal and Loop Step.\n * Displays selected steps and allows naming the new group; emits createGroup with group name\n * so the host can create the step group and replace selected steps with it.\n */\n@Component({\n  selector: 'cqa-create-step-group',\n  templateUrl: './create-step-group.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class CreateStepGroupComponent {\n  @Input() stepsToGroup: TestCaseStepConfig[] = [];\n\n  @Output() createGroup = new EventEmitter<CreateStepGroupFormData>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  form: FormGroup;\n\n  constructor(private readonly fb: FormBuilder) {\n    this.form = this.fb.group({\n      groupName: ['', [Validators.required, Validators.minLength(1)]],\n    });\n  }\n\n  get stepsCount(): number {\n    return this.stepsToGroup?.length ?? 0;\n  }\n\n  get subtitleText(): string {\n    const n = this.stepsCount;\n    return n === 1\n      ? 'Group 1 selected step into a container'\n      : `Group ${n} selected steps into a container`;\n  }\n\n  /** Display label for a step in the \"Steps to group\" list (same as main step list - use action when available). */\n  getStepDisplayLabel(step: TestCaseStepConfig, index: number): string {\n    // Prefer step.action - it matches the full display in the step list (e.g. \"Enter @|password| in the Email\")\n    const rawAction = step.action != null && String(step.action).trim() !== '' ? String(step.action) : '';\n    if (rawAction) {\n      return this.stripHtml(rawAction);\n    }\n    if (isNormalStepConfig(step)) {\n      return this.getNormalStepLabel(step);\n    }\n    if ('groupName' in step && step.groupName) {\n      return step.groupName;\n    }\n    if ('condition' in step && step.condition) {\n      return step.condition;\n    }\n    if ('loopType' in step) {\n      const loop = step as { loopType: string; testDataProfile?: string; condition?: string };\n      if (loop.loopType === 'for' && loop.testDataProfile) {\n        return `For loop: ${loop.testDataProfile}`;\n      }\n      if (loop.loopType === 'while' && loop.condition) {\n        return `While: ${loop.condition}`;\n      }\n    }\n    if (step.description) {\n      return step.description;\n    }\n    return `Step ${index + 1}`;\n  }\n\n  /** Strip HTML tags for plain-text display (action may contain spans for badges). */\n  private stripHtml(html: string): string {\n    if (typeof document !== 'undefined') {\n      const tmp = document.createElement('div');\n      tmp.innerHTML = html;\n      return (tmp.textContent || tmp.innerText || '').trim();\n    }\n    return html.replace(/<[^>]*>/g, '').replace(/&nbsp;/g, ' ').trim();\n  }\n\n  private getNormalStepLabel(step: NormalStepConfig): string {\n    const params = step.parameters ?? [];\n    const getParam = (name: string) =>\n      params.find((p) => p.name?.toLowerCase() === name.toLowerCase());\n    const val = (p: { value?: string; displayValue?: string } | undefined) =>\n      p?.displayValue ?? p?.value ?? '';\n\n    switch (step.eventType as TestCaseEventType) {\n      case 'navigate': {\n        const url = getParam('url');\n        return url ? `Navigate to ${val(url)}` : 'Navigate';\n      }\n      case 'ai-agent': {\n        const instructions = getParam('instructions') ?? getParam('description');\n        return val(instructions) || 'AI Agent step';\n      }\n      case 'type':\n        return val(getParam('text')) || 'Type';\n      case 'click':\n        return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';\n      case 'verify':\n        return val(getParam('description')) || 'Verify';\n      case 'custom': {\n        const desc = getParam('description') ?? getParam('action');\n        return val(desc) || 'Custom step';\n      }\n      default:\n        return val(getParam('description')) || val(params[0]) || 'Step';\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateGroup(): void {\n    if (this.form.invalid || this.stepsCount === 0) {\n      this.form.markAllAsTouched();\n      return;\n    }\n    const groupName = (this.form.get('groupName')?.value ?? '').trim();\n    if (!groupName) {\n      this.form.get('groupName')?.setErrors({ required: true });\n      return;\n    }\n    this.createGroup.emit({ groupName });\n  }\n}\n","<div\n  class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n  <!-- Header: title + close (X) -->\n  <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n      <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0\">\n        Create step group\n      </h2>\n      <p class=\"cqa-text-[14px] cqa-leading-[20px] cqa-text-[#64748B] cqa-m-0\">\n        {{ subtitleText }}\n      </p>\n    </div>\n    <button\n      type=\"button\"\n      (click)=\"onCancel()\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n      title=\"Close\"\n      aria-label=\"Close\">\n      <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n        <path d=\"M18 6L6 18M6 6l12 12\" />\n      </svg>\n    </button>\n  </div>\n\n  <!-- Group name * -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#161617]\">\n      Group name <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <cqa-custom-input\n      placeholder=\"e.g., Authentication flow\"\n      [value]=\"form.get('groupName')?.value\"\n      [fullWidth]=\"true\"\n      size=\"md\"\n      (valueChange)=\"form.get('groupName')?.setValue($event); form.get('groupName')?.updateValueAndValidity()\">\n    </cqa-custom-input>\n    <p *ngIf=\"form.get('groupName')?.invalid && form.get('groupName')?.touched\" class=\"cqa-text-xs cqa-text-red-500 cqa-m-0\">\n      Group name is required.\n    </p>\n  </div>\n\n  <!-- Steps to group (N) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n      Steps to group ({{ stepsCount }})\n    </label>\n    <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-overflow-y-auto cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n      <div\n        *ngFor=\"let step of stepsToGroup; let i = index\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-rounded-[4px] cqa-py-[4px] cqa-px-[8px] cqa-bg-[rgba(216,217,252,0.3)]\">\n        <!-- Numbered badge -->\n        <span\n          class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#3F43EE] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n          {{ i + 1 }}\n        </span>\n        <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#0B0B0C] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n          {{ getStepDisplayLabel(step, i) }}\n        </span>\n      </div>\n    </div>\n  </div>\n\n  <!-- Actions: Cancel | Create group -->\n  <div class=\"cqa-flex cqa-items-stretch cqa-w-full cqa-gap-3\">\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"outlined\"\n        btnSize=\"lg\"\n        text=\"Cancel\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"\n        (clicked)=\"onCancel()\">\n      </cqa-button>\n    </div>\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"filled\"\n        btnSize=\"lg\"\n        text=\"Create group\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE] cqa-bg-[#3F43EE]'\"\n        (clicked)=\"onCreateGroup()\">\n      </cqa-button>\n    </div>\n  </div>\n</div>\n"]}