diff --git a/docs/custom-formkit-input/README.md b/docs/custom-formkit-input/README.md
index 60631d87..e4455119 100644
--- a/docs/custom-formkit-input/README.md
+++ b/docs/custom-formkit-input/README.md
@@ -12,6 +12,7 @@
   - 参数
     1. language: 目前支持 `yaml`, `html`, `css`, `javascript`, `json`
     2. height: 编辑器高度,如:`100px`
+- `attachment`: 附件选择
 - `menuCheckbox`:选择一组菜单
 - `menuRadio`:选择一个菜单
 - `menuItemSelect`:选择菜单项
diff --git a/src/formkit/formkit.config.ts b/src/formkit/formkit.config.ts
index 2c0fc28e..78de33b8 100644
--- a/src/formkit/formkit.config.ts
+++ b/src/formkit/formkit.config.ts
@@ -4,6 +4,7 @@ import { createAutoAnimatePlugin } from "@formkit/addons";
 import { zh } from "@formkit/i18n";
 import type { DefaultConfigOptions } from "@formkit/vue";
 import { form } from "./inputs/form";
+import { attachment } from "./inputs/attachment";
 import { code } from "./inputs/code";
 import { menuCheckbox } from "./inputs/menu-checkbox";
 import { menuRadio } from "./inputs/menu-radio";
@@ -22,6 +23,7 @@ const config: DefaultConfigOptions = {
   plugins: [createAutoAnimatePlugin()],
   inputs: {
     form,
+    attachment,
     code,
     menuCheckbox,
     menuRadio,
diff --git a/src/formkit/inputs/attachment/AttachmentInput.vue b/src/formkit/inputs/attachment/AttachmentInput.vue
new file mode 100644
index 00000000..53acbeca
--- /dev/null
+++ b/src/formkit/inputs/attachment/AttachmentInput.vue
@@ -0,0 +1,61 @@
+<script lang="ts" setup>
+import type { FormKitFrameworkContext } from "@formkit/core";
+import { IconFolder } from "@halo-dev/components";
+import AttachmentSelectorModal from "@/modules/contents/attachments/components/AttachmentSelectorModal.vue";
+import { ref, type PropType } from "vue";
+import type { AttachmentLike } from "@halo-dev/console-shared";
+
+const props = defineProps({
+  context: {
+    type: Object as PropType<FormKitFrameworkContext>,
+    required: true,
+  },
+});
+
+const attachmentSelectorModal = ref(false);
+
+const onInput = (e: Event) => {
+  props.context.handlers.DOMInput(e);
+};
+
+const onAttachmentSelect = (attachments: AttachmentLike[]) => {
+  const urls: (string | undefined)[] = attachments.map((attachment) => {
+    if (typeof attachment === "string") {
+      return attachment;
+    }
+    if ("url" in attachment) {
+      return attachment.url;
+    }
+    if ("spec" in attachment) {
+      return attachment.status?.permalink;
+    }
+  });
+
+  if (urls.length) {
+    props.context.node.input(urls[0]);
+  }
+};
+</script>
+
+<template>
+  <input
+    :id="context.id"
+    :value="context._value"
+    :class="context.classes.input"
+    :name="context.node.name"
+    v-bind="context.attrs"
+    type="text"
+    @blur="context.handlers.blur()"
+    @input="onInput"
+  />
+  <div
+    class="group flex h-full cursor-pointer items-center border-l px-3 transition-all hover:bg-gray-100"
+    @click="attachmentSelectorModal = true"
+  >
+    <IconFolder class="h-4 w-4 text-gray-500 group-hover:text-gray-700" />
+  </div>
+  <AttachmentSelectorModal
+    v-model:visible="attachmentSelectorModal"
+    @select="onAttachmentSelect"
+  />
+</template>
diff --git a/src/formkit/inputs/attachment/index.ts b/src/formkit/inputs/attachment/index.ts
new file mode 100644
index 00000000..bdc7253e
--- /dev/null
+++ b/src/formkit/inputs/attachment/index.ts
@@ -0,0 +1,10 @@
+import { initialValue } from "@formkit/inputs";
+import { createInput } from "@formkit/vue";
+import AttachmentInput from "./AttachmentInput.vue";
+
+export const attachment = createInput(AttachmentInput, {
+  type: "input",
+  props: [],
+  forceTypeProp: "text",
+  features: [initialValue],
+});
diff --git a/src/modules/contents/attachments/components/AttachmentGroupList.vue b/src/modules/contents/attachments/components/AttachmentGroupList.vue
index 7cd2136c..7b98b332 100644
--- a/src/modules/contents/attachments/components/AttachmentGroupList.vue
+++ b/src/modules/contents/attachments/components/AttachmentGroupList.vue
@@ -98,6 +98,7 @@ onMounted(async () => {
 </script>
 <template>
   <AttachmentGroupEditingModal
+    v-if="!readonly"
     v-model:visible="editingModal"
     :group="groupToUpdate"
     @close="onEditingModalClose"
diff --git a/src/modules/contents/attachments/components/AttachmentSelectorModal.vue b/src/modules/contents/attachments/components/AttachmentSelectorModal.vue
index ad99df3b..ad82477b 100644
--- a/src/modules/contents/attachments/components/AttachmentSelectorModal.vue
+++ b/src/modules/contents/attachments/components/AttachmentSelectorModal.vue
@@ -61,6 +61,7 @@ const handleConfirm = () => {
   <VModal
     :visible="visible"
     :width="1240"
+    :mount-to-body="true"
     title="选择附件"
     height="calc(100vh - 20px)"
     @update:visible="onVisibleChange"
diff --git a/src/modules/contents/pages/components/SinglePageSettingModal.vue b/src/modules/contents/pages/components/SinglePageSettingModal.vue
index 7ed429c6..09da47f3 100644
--- a/src/modules/contents/pages/components/SinglePageSettingModal.vue
+++ b/src/modules/contents/pages/components/SinglePageSettingModal.vue
@@ -272,7 +272,7 @@ watchEffect(() => {
           <FormKit
             v-model="formState.page.spec.cover"
             label="封面图"
-            type="text"
+            type="attachment"
             name="cover"
           ></FormKit>
         </FormKit>
diff --git a/src/modules/contents/posts/categories/components/CategoryEditingModal.vue b/src/modules/contents/posts/categories/components/CategoryEditingModal.vue
index 8f7e08ae..6450340e 100644
--- a/src/modules/contents/posts/categories/components/CategoryEditingModal.vue
+++ b/src/modules/contents/posts/categories/components/CategoryEditingModal.vue
@@ -152,7 +152,8 @@ watch(
         help="需要主题适配以支持"
         name="cover"
         label="封面图"
-        type="text"
+        type="attachment"
+        validation="required"
       ></FormKit>
       <FormKit
         v-model="formState.spec.description"
diff --git a/src/modules/contents/posts/components/PostSettingModal.vue b/src/modules/contents/posts/components/PostSettingModal.vue
index 623214f0..906018d0 100644
--- a/src/modules/contents/posts/components/PostSettingModal.vue
+++ b/src/modules/contents/posts/components/PostSettingModal.vue
@@ -285,7 +285,7 @@ watchEffect(() => {
             v-model="formState.post.spec.cover"
             name="cover"
             label="封面图"
-            type="text"
+            type="attachment"
           ></FormKit>
         </FormKit>
         <!--TODO: add SEO/Metas/Inject Code form-->
diff --git a/src/modules/contents/posts/tags/components/TagEditingModal.vue b/src/modules/contents/posts/tags/components/TagEditingModal.vue
index 8c83f116..68c4a7e2 100644
--- a/src/modules/contents/posts/tags/components/TagEditingModal.vue
+++ b/src/modules/contents/posts/tags/components/TagEditingModal.vue
@@ -171,7 +171,7 @@ watch(
         name="cover"
         help="需要主题适配以支持"
         label="封面图"
-        type="text"
+        type="attachment"
       ></FormKit>
     </FormKit>
     <template #footer>
diff --git a/src/modules/system/users/components/UserEditingModal.vue b/src/modules/system/users/components/UserEditingModal.vue
index 2f0fdc17..fe0c6e3d 100644
--- a/src/modules/system/users/components/UserEditingModal.vue
+++ b/src/modules/system/users/components/UserEditingModal.vue
@@ -197,7 +197,7 @@ const handleRawModeChange = () => {
         <FormKit
           v-model="formState.spec.avatar"
           label="头像"
-          type="text"
+          type="attachment"
           name="avatar"
           validation="url"
         ></FormKit>