Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

docs(form): [form] optimize form demos #2499

Merged
merged 1 commit into from
Nov 8, 2024
Merged

Conversation

gimmyhehe
Copy link
Member

@gimmyhehe gimmyhehe commented Nov 7, 2024

PR

PR Checklist

Please check if your PR fulfills the following requirements:

  • The commit message follows our Commit Message Guidelines
  • Tests for the changes have been added (for bug fixes / features)
  • Docs have been added / updated (for bug fixes / features)

PR Type

What kind of change does this PR introduce?

  • Bugfix
  • Feature
  • Code style update (formatting, local variables)
  • Refactoring (no functional changes, no api changes)
  • Build related changes
  • CI related changes
  • Documentation content changes
  • Other... Please describe:

What is the current behavior?

Issue Number: N/A

What is the new behavior?

Does this PR introduce a breaking change?

  • Yes
  • No

Other information

Summary by CodeRabbit

Release Notes

  • New Features

    • Introduced new input types, including radio buttons and IP address fields, enhancing form interactivity.
    • Added new form items such as checkboxes and a switch component.
  • Improvements

    • Simplified button definitions by removing type attributes, streamlining the UI.
    • Updated form layout for better organization and user experience.
  • Bug Fixes

    • Adjusted test cases to ensure accurate validation checks based on the new form structure.
  • Styling

    • Enhanced styling capabilities for Cascader and Autocomplete components with new prefix variables.

Copy link

coderabbitai bot commented Nov 7, 2024

Walkthrough

The pull request includes modifications to various Vue components and their corresponding test files. Key changes involve reordering form items, removing or altering attributes of components, and updating validation rules. Notably, several components have had their placeholder attributes removed or modified, and new form items have been introduced in some components. The test files have also been updated to reflect these changes, ensuring that the tests align with the new form structures and behaviors.

Changes

File Path Change Summary
examples/sites/demos/pc/app/form/basic-usage-composition-api.vue Removed and re-added <tiny-form-item> for "数字", removed placeholder from <tiny-input>, and modified import statement by removing ref.
examples/sites/demos/pc/app/form/basic-usage.vue Reordered <tiny-form-item> for "数字" and removed placeholder from <tiny-input>.
examples/sites/demos/pc/app/form/custom-validation-rule-composition-api.vue Removed type="primary" from <tiny-button>, no changes to validation logic.
examples/sites/demos/pc/app/form/custom-validation-rule.vue Removed type="primary" from <tiny-button> and updated changeRule method to modify validation trigger for username.
examples/sites/demos/pc/app/form/display-only-composition-api.vue Reorganized form items, added new ones, and updated formData values.
examples/sites/demos/pc/app/form/display-only.vue Moved and replaced form items, updated formData values.
examples/sites/demos/pc/app/form/form-disabled-composition-api.vue Removed multiple input components, added new ones, and updated tiny-autocomplete and tiny-cascader attributes.
examples/sites/demos/pc/app/form/form-disabled.vue Restructured form items, added new components, and updated attributes for existing components.
examples/sites/demos/pc/app/form/form-validate-field-composition-api.vue Removed type attribute from buttons without changing functionality.
examples/sites/demos/pc/app/form/form-validate-field.vue Removed type attribute from buttons without changing functionality.
examples/sites/demos/pc/app/form/form-validation-composition-api.vue Added new form items for radio and IP address inputs, updated createData object.
examples/sites/demos/pc/app/form/form-validation.vue Added new form items for radio and IP address inputs, updated createData object and validation rules.
examples/sites/demos/pc/app/form/size-composition-api.vue Reordered <tiny-form-item> for "数字" and removed placeholder from <tiny-input>.
examples/sites/demos/pc/app/form/size.vue Reordered <tiny-form-item> for "数字" and removed placeholder from <tiny-input>.
examples/sites/demos/pc/app/form/validate-debounce-composition-api.vue Removed type attribute from button without changing functionality.
examples/sites/demos/pc/app/form/validate-debounce.vue Removed type attribute from button without changing functionality.
packages/theme/src/form-item/index.less Added new prefix class variables for Cascader and Autocomplete components.
examples/sites/demos/pc/app/form/basic-usage.spec.ts Removed tooltip visibility test, retained other tests.
examples/sites/demos/pc/app/form/display-only.spec.ts Updated visibility assertions to reflect new indices of form items.
examples/sites/demos/pc/app/form/form-disabled.spec.js Updated assertions to check disabled state of various form elements, focusing on different indices.
examples/sites/demos/pc/app/form/form-validation.spec.ts Adjusted input field indices for validation checks to align with new form structure.

Possibly related PRs

Suggested labels

documentation

Suggested reviewers

  • zzcr

🐰 In the garden of forms, we play,
With buttons and inputs, bright as day.
Reordered and styled, all in a row,
Simplified paths for users to flow.
A hop, a skip, through code we weave,
In this patch, new wonders we achieve! 🌼


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 6

🧹 Outside diff range and nitpick comments (28)
examples/sites/demos/pc/app/form/display-only.spec.ts (1)

18-20: LGTM! Consider externalizing test data.

The assertions are well-structured, checking both visibility and content. However, the hardcoded Chinese text could be moved to a constants file for better maintainability.

Consider creating a test constants file:

// test-constants.ts
export const FORM_TEST_DATA = {
  FOOD_ITEMS: {
    SINGLE: '黄金糕',
    MULTIPLE: '黄金糕; 双皮奶'
  }
};

Then update the assertions:

-  await expect(getItem(4).locator('.tiny-input-display-only__content')).toHaveText('黄金糕')
-  await expect(getItem(5).locator('.tiny-input-display-only__content')).toHaveText('黄金糕; 双皮奶')
+  await expect(getItem(4).locator('.tiny-input-display-only__content')).toHaveText(FORM_TEST_DATA.FOOD_ITEMS.SINGLE)
+  await expect(getItem(5).locator('.tiny-input-display-only__content')).toHaveText(FORM_TEST_DATA.FOOD_ITEMS.MULTIPLE)
examples/sites/demos/pc/app/form/basic-usage-composition-api.vue (1)

8-10: Consider adding validation rules for the numeric field.

The numeric field could benefit from validation constraints to ensure data quality and prevent invalid inputs.

Consider adding validation like this:

 <tiny-form-item label="数字">
-  <tiny-numeric v-model="createData.quantity"></tiny-numeric>
+  <tiny-numeric
+    v-model="createData.quantity"
+    :min="0"
+    :max="100"
+    placeholder="请输入0-100的数字">
+  </tiny-numeric>
 </tiny-form-item>
examples/sites/demos/pc/app/form/form-validate-field-composition-api.vue (2)

Line range hint 52-60: Improve error handling in validateField()

The current implementation has an empty if block when validation errors occur, which silently ignores validation failures. Consider displaying these errors to provide better user feedback.

 async function validateField() {
   const errArray = []
   await ruleFormRef.value.validateField(['url', 'datepicker'], (errMsg, fieldConfig) => {
     if (errMsg) {
       errArray.push(fieldConfig)
     }
   })
   if (errArray.length > 0) {
-    // empty
+    TinyModal.alert('验证失败:' + errArray.map(field => field.label).join(', '))
   } else {
     TinyModal.alert('日期和url通过校验')
   }
 }

Line range hint 52-60: Consider implementing comprehensive error handling

The validation logic could be enhanced to provide a better user experience:

  1. Add error state handling
  2. Implement proper error recovery
  3. Consider adding loading states during async validation

Here's a suggested implementation:

async function validateField() {
  try {
    const errArray = []
    // Show loading state
    isValidating.value = true

    await ruleFormRef.value.validateField(['url', 'datepicker'], (errMsg, fieldConfig) => {
      if (errMsg) {
        errArray.push({ ...fieldConfig, error: errMsg })
      }
    })

    if (errArray.length > 0) {
      // Display detailed error messages
      const errorMessages = errArray
        .map(field => `${field.label}: ${field.error}`)
        .join('\n')
      TinyModal.alert({
        title: '验证失败',
        message: errorMessages,
        type: 'error'
      })
    } else {
      TinyModal.alert({
        title: '验证成功',
        message: '日期和url通过校验',
        type: 'success'
      })
    }
  } catch (error) {
    // Handle unexpected errors
    TinyModal.alert({
      title: '系统错误',
      message: '验证过程中发生错误,请重试',
      type: 'error'
    })
    console.error('Validation error:', error)
  } finally {
    // Clear loading state
    isValidating.value = false
  }
}
examples/sites/demos/pc/app/form/basic-usage.vue (2)

19-19: Good UX improvement by favoring tooltip over placeholder.

Removing the placeholder in favor of the tooltip is a better practice as:

  • Tooltips remain visible during input, unlike placeholders that disappear
  • The tooltip provides clear guidance without cluttering the input field

Line range hint 1-89: Consider enhancing the demo with validation examples.

While this basic usage demo is well-structured, it could be more educational by including:

  1. Form validation rules
  2. Type definitions for the form data
  3. Examples of handling form submission errors

This would provide a more complete reference for developers implementing forms.

Would you like me to provide an example implementation with these enhancements?

examples/sites/demos/pc/app/form/custom-validation-rule.vue (1)

Line range hint 31-35: Consider improving password validation maintainability

The password validation regex is complex and could benefit from:

  1. Breaking down the regex into named constants
  2. Adding comments explaining the validation rules
  3. Making the error message more user-friendly

Consider refactoring like this:

+ const PASSWORD_MIN_LENGTH = 8;
+ const PASSWORD_PATTERN = {
+   LOWERCASE: /[a-z]/,
+   UPPERCASE: /[A-Z]/,
+   DIGIT: /\d/
+ };

let validatePass = (rule, value, callback) => {
-  if (!/^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)[a-zA-Z\d]{8,}$/.test(value)) {
-    callback(new Error('最少八个字符,至少包含一个大写字母,一个小写字母和一个数字'))
+  const errors = [];
+  if (value.length < PASSWORD_MIN_LENGTH) {
+    errors.push('密码长度至少为8个字符');
+  }
+  if (!PASSWORD_PATTERN.LOWERCASE.test(value)) {
+    errors.push('需要包含小写字母');
+  }
+  if (!PASSWORD_PATTERN.UPPERCASE.test(value)) {
+    errors.push('需要包含大写字母');
+  }
+  if (!PASSWORD_PATTERN.DIGIT.test(value)) {
+    errors.push('需要包含数字');
+  }
+  if (errors.length > 0) {
+    callback(new Error(errors.join(';')));
   } else {
     callback()
   }
}
examples/sites/demos/pc/app/form/size-composition-api.vue (3)

9-11: Consider adding validation rules for the numeric input.

While the implementation is correct, consider adding validation rules (min/max values, decimal places, etc.) to ensure data integrity.

Example implementation:

 <tiny-form-item label="数字">
-  <tiny-numeric v-model="createData.quantity"></tiny-numeric>
+  <tiny-numeric 
+    v-model="createData.quantity"
+    :min="0"
+    :max="999"
+    :precision="2"
+  ></tiny-numeric>
 </tiny-form-item>

20-20: Enhance accessibility with aria-label.

Since the placeholder was removed, consider adding an aria-label to improve accessibility and provide context about the expected input.

-<tiny-input v-model="createData.input"></tiny-input>
+<tiny-input 
+  v-model="createData.input"
+  aria-label="Please enter your text"
+></tiny-input>

Line range hint 1-30: Consider implementing responsive design.

The form has a fixed width of 380px which might not be optimal for all screen sizes. Consider making the form responsive to enhance mobile usability.

Suggestions:

  1. Use relative units or media queries in the CSS
  2. Allow the form to adapt to container width
  3. Consider adjusting label position for smaller screens
 <style scoped>
 .demo-form {
-  width: 380px;
+  width: 100%;
+  max-width: 380px;
 }
 </style>
examples/sites/demos/pc/app/form/size.vue (2)

20-20: LGTM! Improved UX by relying on tooltip instead of placeholder.

The removal of the placeholder attribute is a good practice as the tooltip already provides user guidance, and the form item has a proper label for accessibility.

Consider adding aria-describedby to enhance accessibility by explicitly connecting the input with its tooltip:

-<tiny-input v-model="createData.input"></tiny-input>
+<tiny-input v-model="createData.input" aria-describedby="form-demo-tooltip"></tiny-input>

Line range hint 4-30: Consider adding form validation rules.

The form includes various input types (numeric, date, text) but lacks validation rules. Adding validation would improve data integrity and user experience.

Here's a suggested approach:

<template>
  <tiny-form 
    ref="createData" 
    label-width="100px" 
    :size="sizeValue" 
    label-position="right"
+   :rules="rules"
+   @submit.prevent="submitClick">
    <!-- ... form items ... -->
  </tiny-form>
</template>

<script>
export default {
  data() {
    return {
      // ... existing data ...
+     rules: {
+       quantity: [{ required: true, message: '请输入数量', trigger: 'blur' }],
+       datepicker: [{ required: true, message: '请选择日期', trigger: 'change' }]
+     }
    }
  },
  methods: {
    submitClick() {
-     TinyModal.alert('提交')
+     this.$refs.createData.validate((valid) => {
+       if (valid) {
+         TinyModal.alert('提交成功')
+       }
+     })
    }
  }
}
</script>
examples/sites/demos/pc/app/form/form-validation-composition-api.vue (4)

7-12: LGTM! Consider adding aria-labels for better accessibility.

The new form items for level selection and IP address are well-structured and properly integrated with form validation.

Consider adding aria-labels to improve accessibility:

-      <tiny-form-item label="等级" prop="radio">
+      <tiny-form-item label="等级" prop="radio" aria-label="Select level">
         <tiny-radio-group v-model="createData.radio" :options="options"></tiny-radio-group>
       </tiny-form-item>
-      <tiny-form-item label="IP字段" prop="ip">
+      <tiny-form-item label="IP字段" prop="ip" aria-label="Enter IP address">
         <tiny-ip-address v-model="createData.ip"></tiny-ip-address>
       </tiny-form-item>

32-33: LGTM! Consider adding loading states for better UX.

The clear validation and reset buttons are properly implemented.

Consider adding loading states during async operations:

-        <tiny-button @click="clearFormValid"> 移除校验 </tiny-button>
-        <tiny-button @click="resetForm"> 重置表单 </tiny-button>
+        <tiny-button @click="clearFormValid" :loading="isClearing"> 移除校验 </tiny-button>
+        <tiny-button @click="resetForm" :loading="isResetting"> 重置表单 </tiny-button>

Line range hint 82-87: Consider using a more realistic IP validation rule.

The current IP validation rule only accepts "1.1.1.1", which might be too restrictive for a demo. Consider using a more general IP format validation.

Here's a suggested improvement:

   ip: [
     {
-      validator: (rule, value, cb) => (value === '1.1.1.1' ? cb() : cb(new Error('必填1.1.1.1'))),
+      validator: (rule, value, cb) => {
+        const ipRegex = /^(\d{1,3}\.){3}\d{1,3}$/;
+        if (!value) {
+          return cb(new Error('IP address is required'));
+        }
+        if (!ipRegex.test(value)) {
+          return cb(new Error('Please enter a valid IP address'));
+        }
+        const parts = value.split('.');
+        const valid = parts.every(part => parseInt(part) >= 0 && parseInt(part) <= 255);
+        return valid ? cb() : cb(new Error('Each octet must be between 0 and 255'));
+      },
       trigger: 'change'
     }
   ],

Line range hint 89-106: Enhance error handling in form submission.

While the handlers demonstrate both callback and promise approaches well, they could benefit from better error handling and loading states.

Here's a suggested improvement:

+const isSubmitting = ref(false)
+
 function handleSubmit() {
+  isSubmitting.value = true
   ruleFormRef.value.validate((valid) => {
     if (valid) {
       TinyModal.alert('回调用法:提交成功')
     } else {
       TinyModal.alert('回调用法:提交失败')
     }
+  }).finally(() => {
+    isSubmitting.value = false
+  })
 }

 function handleSubmitPromise() {
+  isSubmitting.value = true
   ruleFormRef.value
     .validate()
     .then(() => {
       TinyModal.alert('promise用法:提交成功')
     })
     .catch((error) => {
-      TinyModal.alert('promise用法:提交失败')
+      TinyModal.alert(`promise用法:提交失败 - ${error.message || '未知错误'}`)
+    })
+    .finally(() => {
+      isSubmitting.value = false
     })
 }

Then update the submit button:

-        <tiny-button type="primary" @click="validType === 'callback' ? handleSubmit() : handleSubmitPromise()">
+        <tiny-button 
+          type="primary" 
+          :loading="isSubmitting"
+          @click="validType === 'callback' ? handleSubmit() : handleSubmitPromise()">
           提交
         </tiny-button>
examples/sites/demos/pc/app/form/form-validation.spec.ts (1)

Line range hint 1-93: Consider implementing a Page Object pattern for better test maintainability.

The current test implementation tightly couples the test logic to the form structure. To improve maintainability and reduce the impact of UI changes, consider:

  1. Implementing a Page Object pattern to encapsulate form selectors and interactions
  2. Creating a central selector configuration file
  3. Using data-testid attributes consistently across the application

This would make the tests more resilient to form structure changes and easier to maintain.

Example implementation:

// formValidationPage.ts
export class FormValidationPage {
  constructor(private page: Page) {}

  private selectors = {
    lengthInput: '[data-testid="length-validation-input"]',
    urlInput: '[data-testid="url-input"]',
    emailInput: '[data-testid="email-input"]'
    // ... other selectors
  }

  async fillLengthInput(value: string) {
    await this.page.locator(this.selectors.lengthInput).fill(value)
  }

  // ... other methods
}

// form-validation.spec.ts
test('测试表单输入变化和失焦是否出现校验', async ({ page }) => {
  const formPage = new FormValidationPage(page)
  await formPage.fillLengthInput('test')
  // ... rest of the test
})
examples/sites/demos/pc/app/form/form-disabled.spec.js (2)

48-63: Consider refactoring test structure to reduce duplication.

  1. Line 62 duplicates the assertion from line 61.
  2. The enabled state assertions mirror the disabled state assertions, suggesting an opportunity for a more maintainable structure.

Consider refactoring the test to use a helper function that tests both states:

async function testFormItemStates(form, isDisabled) {
  const state = {
    class: isDisabled ? 'toHaveClass' : 'not.toHaveClass',
    disabled: isDisabled ? 'toBeDisabled' : 'not.toBeDisabled'
  }

  const formElements = [
    { index: 0, selector: '.tiny-switch', assertion: state.class, value: /tiny-switch-disabled/ },
    { index: 2, selector: 'input', assertion: state.disabled },
    // ... define all form elements
  ]

  for (const element of formElements) {
    await expect(form.nth(element.index).locator(element.selector))
      [element.assertion](element.value)
  }
}

// Usage in test
await testFormItemStates(formItem, true)  // Test disabled state
await switchBtn.click()
await testFormItemStates(formItem, false)  // Test enabled state

Line range hint 1-73: Enhance test readability and maintainability.

Consider these improvements:

  1. Add custom error messages to assertions to make test failures more descriptive
  2. Document the form structure being tested with comments

Example improvements:

+ // Form structure being tested:
+ // - Switch (index 0)
+ // - Text inputs (indices 2-3)
+ // - Button (index 6)
+ // - Numeric input (index 9)
+ // - Textarea (index 14)
+ // - Slider (index 18)

- await expect(formItem.nth(0).locator('.tiny-switch')).toHaveClass(/tiny-switch-disabled/)
+ await expect(formItem.nth(0).locator('.tiny-switch')).toHaveClass(
+   /tiny-switch-disabled/,
+   'Switch should be disabled'
+ )
examples/sites/demos/pc/app/form/form-validation.vue (3)

7-12: Consider adding internationalization support for form labels.

The form labels "等级" and "IP字段" are hardcoded in Chinese. Consider using an i18n system to support multiple languages and improve maintainability.

-      <tiny-form-item label="等级" prop="radio">
+      <tiny-form-item :label="t('form.level')" prop="radio">
-      <tiny-form-item label="IP字段" prop="ip">
+      <tiny-form-item :label="t('form.ipField')" prop="ip">

Line range hint 89-94: Improve IP address validation for better demo value.

The current IP validation is overly restrictive by requiring exactly "1.1.1.1". Consider implementing a proper IP format validation to better demonstrate real-world usage.

        ip: [
          {
-            validator: (rule, value, cb) => (value === '1.1.1.1' ? cb() : cb(new Error('必填1.1.1.1'))),
+            validator: (rule, value, cb) => {
+              const ipRegex = /^(\d{1,3}\.){3}\d{1,3}$/;
+              if (!value) {
+                return cb(new Error('IP address is required'));
+              }
+              if (!ipRegex.test(value)) {
+                return cb(new Error('Please enter a valid IP address'));
+              }
+              const parts = value.split('.');
+              const valid = parts.every(part => parseInt(part) >= 0 && parseInt(part) <= 255);
+              return valid ? cb() : cb(new Error('Each octet must be between 0 and 255'));
+            },
            trigger: 'change'
          }
        ],

22-34: LGTM! Well-structured form validation implementation.

The component demonstrates good practices with both callback and promise-based validation approaches. The toggle between validation methods provides excellent demo value.

Consider extracting validation messages to an i18n system for consistency with the earlier suggestion.

examples/sites/demos/pc/app/form/display-only-composition-api.vue (2)

47-52: Consider adding placeholders for better UX

The form items are well-structured, but could benefit from placeholders to guide users on expected input.

-        <tiny-input v-model="formData.input"></tiny-input>
+        <tiny-input v-model="formData.input" placeholder="请输入文字"></tiny-input>
-        <tiny-input v-model="formData.textareaValue" type="textarea"></tiny-input>
+        <tiny-input v-model="formData.textareaValue" type="textarea" placeholder="请输入描述"></tiny-input>

149-149: Consider implementing i18n for text content

The updated descriptive text provides better context, but should be internationalized for better maintainability and multi-language support.

Example implementation:

-    'TinyVue 为 Web 应用提供了丰富的基础 UI 组件,我们还将持续探索企业级应用的最佳 UI 实践,欢迎尝试使用 TinyVue。',
+    t('form.demo.tinyVueDescription')
-    'OpenTiny 是一套企业级组件库解决方案,适配 PC 端 / 移动端等多端,涵盖 Vue2 / Vue3 / Angular 多技术栈,拥有主题配置系统 / 中后台模板 / CLI 命令行等效率提升工具,可帮助开发者高效开发 Web 应用。',
+    t('form.demo.openTinyDescription')

Also applies to: 167-167

examples/sites/demos/pc/app/form/display-only.vue (2)

5-20: LGTM! Consider i18n for labels.

The new form items are well-structured and properly integrated. The components are correctly bound to their respective data properties.

Consider extracting the Chinese labels into i18n translation files for better internationalization support.


165-165: Consider using shorter, more generic default values.

The current default values contain lengthy marketing text about TinyVue and OpenTiny. For a form demo, consider:

  1. Using shorter, more generic placeholder text
  2. Moving product-specific content to documentation
  3. Extracting these strings to a constants file for easier maintenance

This would make the demo more focused and easier to maintain.

formData: {
  input:
-    'TinyVue 为 Web 应用提供了丰富的基础 UI 组件,我们还将持续探索企业级应用的最佳 UI 实践,欢迎尝试使用 TinyVue。',
+    'This is a sample long text input.',
  // ...
  textareaValue:
-    'OpenTiny 是一套企业级组件库解决方案,适配 PC 端 / 移动端等多端,涵盖 Vue2 / Vue3 / Angular 多技术栈,拥有主题配置系统 / 中后台模板 / CLI 命令行等效率提升工具,可帮助开发者高效开发 Web 应用。',
+    'This is a sample textarea content.',

Also applies to: 183-183

examples/sites/demos/pc/app/form/form-disabled-composition-api.vue (1)

Line range hint 1-267: Well-structured form demo component.

The component effectively demonstrates form functionality with:

  • Logical grouping of form elements
  • Proper reactive data binding
  • Clean composition API implementation
  • Comprehensive coverage of form components

Consider adding comments to document the purpose of each form section for better maintainability.

examples/sites/demos/pc/app/form/form-disabled.vue (1)

35-35: Remove unnecessary empty placeholders.

The empty placeholder attributes in Select and Cascader components don't serve any purpose and should be removed for cleaner code.

-<tiny-select v-model="createData.select" placeholder="">
+<tiny-select v-model="createData.select">

-<tiny-cascader :options="options" :props="{ emitPath: false }" placeholder=" ">
+<tiny-cascader :options="options" :props="{ emitPath: false }">

Also applies to: 67-67

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 56f14af and 1540f19.

📒 Files selected for processing (21)
  • examples/sites/demos/pc/app/form/basic-usage-composition-api.vue (3 hunks)
  • examples/sites/demos/pc/app/form/basic-usage.spec.ts (0 hunks)
  • examples/sites/demos/pc/app/form/basic-usage.vue (2 hunks)
  • examples/sites/demos/pc/app/form/custom-validation-rule-composition-api.vue (1 hunks)
  • examples/sites/demos/pc/app/form/custom-validation-rule.vue (1 hunks)
  • examples/sites/demos/pc/app/form/display-only-composition-api.vue (4 hunks)
  • examples/sites/demos/pc/app/form/display-only.spec.ts (1 hunks)
  • examples/sites/demos/pc/app/form/display-only.vue (4 hunks)
  • examples/sites/demos/pc/app/form/form-disabled-composition-api.vue (4 hunks)
  • examples/sites/demos/pc/app/form/form-disabled.spec.js (2 hunks)
  • examples/sites/demos/pc/app/form/form-disabled.vue (4 hunks)
  • examples/sites/demos/pc/app/form/form-validate-field-composition-api.vue (1 hunks)
  • examples/sites/demos/pc/app/form/form-validate-field.vue (1 hunks)
  • examples/sites/demos/pc/app/form/form-validation-composition-api.vue (2 hunks)
  • examples/sites/demos/pc/app/form/form-validation.spec.ts (2 hunks)
  • examples/sites/demos/pc/app/form/form-validation.vue (2 hunks)
  • examples/sites/demos/pc/app/form/size-composition-api.vue (2 hunks)
  • examples/sites/demos/pc/app/form/size.vue (2 hunks)
  • examples/sites/demos/pc/app/form/validate-debounce-composition-api.vue (1 hunks)
  • examples/sites/demos/pc/app/form/validate-debounce.vue (1 hunks)
  • packages/theme/src/form-item/index.less (2 hunks)
💤 Files with no reviewable changes (1)
  • examples/sites/demos/pc/app/form/basic-usage.spec.ts
✅ Files skipped from review due to trivial changes (4)
  • examples/sites/demos/pc/app/form/custom-validation-rule-composition-api.vue
  • examples/sites/demos/pc/app/form/form-validate-field.vue
  • examples/sites/demos/pc/app/form/validate-debounce-composition-api.vue
  • examples/sites/demos/pc/app/form/validate-debounce.vue
🔇 Additional comments (18)
examples/sites/demos/pc/app/form/basic-usage-composition-api.vue (2)

33-33: LGTM! Clean import statement.

Good cleanup of unused 'ref' import while keeping the necessary 'reactive' import.


19-19: Verify if placeholder removal was intentional.

The removal of the placeholder attribute might impact user experience by removing helpful input guidance.

examples/sites/demos/pc/app/form/form-validate-field-composition-api.vue (1)

15-16: LGTM! Button styling changes are consistent with project-wide updates.

The removal of type="primary" from these buttons aligns with the broader changes across the codebase to standardize button styling.

examples/sites/demos/pc/app/form/basic-usage.vue (1)

8-10: LGTM! Form item placement is logically organized.

The numeric form item is properly implemented with correct data binding and initialization.

examples/sites/demos/pc/app/form/custom-validation-rule.vue (2)

12-12: LGTM! Improved visual hierarchy

The removal of type="primary" from the "Change validation rules" button helps maintain proper visual hierarchy, making the "Register" button the clear primary action.


Line range hint 63-71: Security concern: Password validation rules are dropped

The changeRule method has two potential issues:

  1. It completely removes password validation rules, which could lead to security vulnerabilities
  2. The method name doesn't clearly indicate that it's removing password validation

Consider this safer implementation:

 changeRule() {
   this.rules = {
     username: [
       { required: true, message: '必填', trigger: 'blur' },
       { min: 2, max: 11, message: '长度必须不小于2', trigger: 'change' }
-    ]
+    ],
+    password: this.rules.password // Preserve password validation rules
   }
 }

Let's verify if this component is used in tests or other demos:

examples/sites/demos/pc/app/form/size.vue (1)

9-11: LGTM! Form item ordering aligns with other form demos.

The numeric form item is correctly implemented with proper data binding and follows the standardized form layout pattern observed across other form components.

examples/sites/demos/pc/app/form/form-validation.vue (1)

33-34: LGTM! Button improvements look good.

The removal of redundant type attributes and the addition of clear validation and reset form buttons improve the user experience.

examples/sites/demos/pc/app/form/display-only-composition-api.vue (1)

5-20: LGTM: Well-structured form control additions

The new form controls (switch, checkbox, checkbox group) enhance the demo's comprehensiveness by showcasing more input types. The implementation is clean and properly bound to the data model.

examples/sites/demos/pc/app/form/display-only.vue (1)

47-52: LGTM! Improved form organization.

The relocation of these form items enhances the logical flow of the form. The components maintain proper data bindings and functionality.

examples/sites/demos/pc/app/form/form-disabled-composition-api.vue (3)

8-10: LGTM: New form items enhance the demo's completeness.

The addition of Link, IpAddress, and ButtonGroup form items improves the demonstration of form capabilities and disabled state handling.

Also applies to: 25-30


168-168: LGTM: Improved column width for better content visibility.

The increase in ID column width from 40 to 60 enhances readability of the grid content.


264-264: LGTM: Enhanced title alignment.

The added left margin improves the visual alignment of the title with the form content.

examples/sites/demos/pc/app/form/form-disabled.vue (3)

8-10: LGTM: New form components enhance functionality.

The addition of Link, IpAddress, and ButtonGroup components enriches the form's capabilities while maintaining consistency with the form's disabled state functionality.

Also applies to: 25-30


193-193: LGTM: Improved ID column width.

The increase in ID column width from 40 to 60 enhances readability.


291-291: Verify title margin alignment with design guidelines.

The title margin-left has been reduced from 100px to 80px. Please confirm this change aligns with the design system's spacing guidelines.

✅ Verification successful

Let me check the specific margin values used in these files to verify consistency.


Title margin change is consistent with other form components

The margin-left value of 80px is already being used consistently across form components, specifically in both the standard and composition API versions of the form-disabled component. The change from 100px to 80px aligns the component with the existing pattern in the codebase.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for similar title margin styles in other form-related components
rg -l 'margin-left:.*px' 'examples/sites/demos/pc/app/form/'

Length of output: 290


Script:

#!/bin/bash
# Search for title margin values in form components
rg 'margin-left:.*px' 'examples/sites/demos/pc/app/form/' -B 2 -A 0

Length of output: 944

packages/theme/src/form-item/index.less (2)

24-25: LGTM: New prefix class variables follow established patterns.

The added prefix class variables for Cascader and Autocomplete components follow the existing naming convention and are properly scoped.


189-190: LGTM: Width styling is consistent with other form controls.

The width styling for Cascader and Autocomplete components is appropriately grouped with other full-width form controls, maintaining consistency in the form layout.

@@ -9,13 +9,13 @@ test('表单仅展示', async ({ page }) => {
const getItem = (index: number) => demo.locator('.tiny-form-item__content').nth(index)

await switchBtn.click()
await expect(demo.locator('.tiny-form-item').first().locator('input')).toBeVisible()
await expect(getItem(10).locator('input')).toBeVisible()
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Codebase verification

Index 10 visibility check is incorrect

The test file is verifying form display functionality with a switch toggle between editable and display-only modes. Based on the Vue template and test file analysis:

  • The form has 12 items in total
  • Index 10 corresponds to "超长文字" (long text) input field
  • The test is checking visibility after clicking the switch, but the assertions pattern shows that form items should become non-visible after the switch click
  • Other assertions at indices 1, 4, and 5 follow a consistent pattern of checking for non-visibility

The visibility check at index 10 is inconsistent with the test's logic flow and should be checking for non-visibility like other assertions.

🔗 Analysis chain

Verify form item index alignment.

The significant change in form item index (now checking index 10) suggests substantial form restructuring. Please verify that this index correctly corresponds to the intended form item in the updated form structure.


Let me generate another script to get more context about the test file structure and assertions:

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check the form structure to verify item indices
# Look for form item definitions in the corresponding Vue component

# Find the related Vue component file
fd -e vue "display-only" -x cat {} | grep -A 5 "tiny-form-item"

Length of output: 7312


Script:

#!/bin/bash
# Get the full test file content to understand the test structure and assertions
cat examples/sites/demos/pc/app/form/display-only.spec.ts

Length of output: 1015

Comment on lines +59 to 60
const emailInput = formItem.nth(5).locator('input')
const validEmail = '[email protected]'
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Replace numeric index with email-specific selector.

The email input field should be identifiable by its validation rules or input type rather than its position in the form.

Example refactor:

- const emailInput = formItem.nth(5).locator('input')
+ const emailInput = form.locator('[data-testid="email-input"]')
// or
+ const emailInput = form.locator('input[type="email"]')

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +51 to 52
const urlInput = formItem.nth(4).locator('input')
const validUrl = 'https://test.com'
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Replace numeric index with URL-specific selector.

The URL input field should be identifiable by its validation rules or input type rather than its position in the form.

Example refactor:

- const urlInput = formItem.nth(4).locator('input')
+ const urlInput = form.locator('[data-testid="url-input"]')
// or
+ const urlInput = form.locator('input[type="url"]')

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +30 to 31
const firstInput = formItem.nth(2).locator('input')
await firstInput.click()
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consider using more maintainable selectors instead of numeric indices.

Using numeric indices (nth(2)) for form item selection makes the tests brittle and prone to breaking when form items are reordered. Consider using more reliable selectors such as:

  • data-testid attributes
  • form field labels
  • input names or other unique identifiers

Example refactor:

- const firstInput = formItem.nth(2).locator('input')
+ const firstInput = form.locator('[data-testid="length-validation-input"]')

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +21 to +36
await expect(formItem.nth(0).locator('.tiny-switch')).toHaveClass(/tiny-switch-disabled/)
await expect(formItem.nth(2).locator('input').first()).toBeDisabled()
await expect(formItem.nth(3).locator('input').nth(1)).toBeDisabled()
await expect(formItem.nth(3).locator('input').first()).toBeDisabled()
await expect(formItem.nth(6).locator('button').first()).toHaveClass(/disabled/)
await expect(formItem.nth(7).locator('input')).toBeDisabled()
await expect(formItem.nth(8).locator('input')).toBeDisabled()
await expect(formItem.nth(9).locator('.tiny-numeric__input')).toHaveClass(/is-disabled/)
await expect(formItem.nth(10).locator('input').first()).toBeDisabled()
await expect(formItem.nth(11).locator('input').first()).toBeDisabled()
await expect(formItem.nth(12).locator('input')).toBeDisabled()
await expect(formItem.nth(13).locator('button').first()).toHaveClass(/disabled/)
await expect(formItem.nth(14).locator('input')).toBeDisabled()
await expect(formItem.nth(15).locator('input').first()).toBeDisabled()
await expect(formItem.nth(13).locator('input')).toBeDisabled()
await expect(formItem.nth(14).locator('textarea')).toBeDisabled()
await expect(formItem.nth(15).locator('input')).toBeDisabled()
await expect(formItem.nth(15).locator('input')).toBeDisabled()
await expect(formItem.nth(16).locator('input').first()).toBeDisabled()
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Remove duplicate assertion and consider using constants for form item indices.

  1. Line 35 duplicates the assertion from line 34, checking the same input element twice.
  2. Consider using named constants for form item indices to improve test maintainability and make the relationship between indices and form elements clear.
+ // At the top of the file
+ const FORM_ITEMS = {
+   SWITCH: 0,
+   INPUT_FIRST: 2,
+   INPUT_MULTIPLE: 3,
+   BUTTON: 6,
+   // ... add more meaningful names
+ }

- await expect(formItem.nth(0).locator('.tiny-switch')).toHaveClass(/tiny-switch-disabled/)
+ await expect(formItem.nth(FORM_ITEMS.SWITCH).locator('.tiny-switch')).toHaveClass(/tiny-switch-disabled/)

- await expect(formItem.nth(15).locator('input')).toBeDisabled()
- await expect(formItem.nth(15).locator('input')).toBeDisabled()  // Remove duplicate
+ await expect(formItem.nth(15).locator('input')).toBeDisabled()

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +31 to +42
<tiny-form-item label="Input">
<tiny-input v-model="createData.user"></tiny-input>
</tiny-form-item>
<tiny-form-item label="Select">
<tiny-select v-model="createData.select" placeholder="">
<tiny-option v-for="item in selectOptions" :key="item.value" :label="item.label" :value="item.value">
</tiny-option>
</tiny-select>
</tiny-form-item>
<tiny-form-item label="Numeric">
<tiny-numeric v-model="createData.quantity"></tiny-numeric>
</tiny-form-item>
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Codebase verification

Based on the extensive search results, I can now make a determination about the placeholder usage in form elements:

Empty placeholders are not the standard pattern in this codebase

Looking at the search results, I found that:

  1. Most form components consistently use meaningful placeholders across the codebase:

    • Input components typically use placeholders like "请输入内容", "请输入", "Please input"
    • Select components use placeholders like "请选择", "Please select"
    • Search components use placeholders like "请输入关键词"
  2. Empty placeholders (placeholder="") or space-only placeholders (placeholder=" ") are rare exceptions rather than the norm.

  3. The form-disabled example appears to be an outlier in using empty placeholders, while most other form examples follow the pattern of providing meaningful placeholder text.

Therefore, the review comment's concern about empty placeholders is valid, as it goes against the established patterns in the codebase.

🔗 Analysis chain

Verify empty placeholder usage in form elements.

Several form elements have empty or removed placeholders. While this might be intentional, consider:

  • Adding meaningful placeholders to improve user experience
  • Ensuring consistency with design guidelines

Also applies to: 57-57, 64-64, 67-67


Let me try a different approach to verify the placeholder usage patterns.

This script will:

  1. Search for placeholder attributes in Vue files with surrounding context
  2. Look for form component usage patterns to understand the general conventions

The -U flag will help us see multiline matches, and we explicitly add .vue extension for ripgrep to recognize Vue files.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check placeholder usage across form components
# Look for inconsistencies in placeholder usage

echo "Checking placeholder usage in form components..."
rg -A 1 'placeholder="[^"]*"' --type vue

echo "Checking components without placeholders..."
ast-grep --pattern 'tiny-input|tiny-select|tiny-autocomplete'

Length of output: 338


Script:

#!/bin/bash
# Search for placeholder patterns in .vue files
echo "Searching for placeholder usage patterns..."
rg -U --type-add 'vue:*.vue' --type vue 'placeholder=' -A 2

# Search for form component usage patterns
echo -e "\nSearching for form component patterns..."
rg -U --type-add 'vue:*.vue' --type vue '<tiny-(input|select|autocomplete)' -A 2

Length of output: 493763

@zzcr zzcr merged commit 8eb46c5 into release-3.19.0 Nov 8, 2024
10 of 14 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants