Files
assistant-storefront/app/javascript/shared/components/SearchImage.vue

48 lines
1.2 KiB
Vue
Raw Normal View History

feat: add search_image and product_list content types with image upload ## 新增功能 ### 1. 新增消息类型 - 添加 search_image (17) 和 product_list (18) content_type - 支持图片搜索和商品列表消息展示 ### 2. 图片上传功能 - 添加 ImageUploadButton 组件,支持图片上传到 Mall API - 上传后发送 search_image 类型消息,图片 URL 存储在 content_attributes - 支持图片文件验证(类型、大小) ### 3. Webhook 推送优化 - 修改 webhook_listener.rb,允许 search_image 类型即使 content 为空也推送 webhook - 解决 search_image 消息不触发 webhook 的问题 ### 4. 前端组件 - 新增 SearchImage.vue 组件(widget 和 dashboard) - 新增 ProductList.vue、Logistics.vue、OrderDetail.vue、OrderList.vue 组件 - 更新 Message.vue 路由逻辑支持新的 content_type - 更新 UserMessage.vue 支持 search_image 消息显示 ### 5. API 层修改 - widget/messages_controller.rb: 允许 content_attributes 参数 - widget/base_controller.rb: 使用前端传入的 content_attributes - widget/conversation API: 支持 contentAttributes 参数传递 - conversation actions 和 helpers: 完整的 content_attributes 数据流 ### 6. Widget 测试页面优化 - 重写 /widget_tests 页面,支持游客/用户模式切换 - 登录流程:使用 reset() + setUser(),不刷新页面 - 退出流程:使用 reset() + 刷新页面 - 添加详细的日志输出和状态显示 ## 技术细节 ### Message Model ```ruby enum content_type: { # ...existing types... search_image: 17, product_list: 18 } ``` ### Webhook Listener ```ruby # Allow search_image webhook even if content is blank return if message.content.blank? && message.content_type != 'search_image' ``` ### Widget Upload Flow ``` 用户选择图片 → 上传到 Mall API → 获取图片 URL → 发送消息: { content: '', content_type: 'search_image', content_attributes: { url: '...' } } ``` Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-27 19:03:46 +08:00
<template>
<div class="search-image-wrap bg-gray-100 inline-block">
<div v-if="!imageUrl" class="text-red-500 text-xs p-2">
No image URL found. Attributes: {{ JSON.stringify(messageContentAttributes) }}
</div>
<img
v-if="imageUrl"
:src="imageUrl"
alt="搜索图片"
class="h-48 w-auto rounded-lg block"
@error="handleImageError"
@load="handleImageLoad"
/>
</div>
</template>
<script setup>
import { computed } from 'vue';
const props = defineProps({
messageContentAttributes: {
type: Object,
default: () => {},
},
message: {
type: String,
default: '',
},
});
const imageUrl = computed(() => {
console.log('[SearchImage] messageContentAttributes:', props.messageContentAttributes);
console.log('[SearchImage] message:', props.message);
const url = props.messageContentAttributes?.url || props.message || '';
console.log('[SearchImage] Computed imageUrl:', url);
return url;
});
const handleImageError = (event) => {
console.error('[SearchImage] Image failed to load:', imageUrl.value);
console.error('[SearchImage] Error event:', event);
};
const handleImageLoad = () => {
console.log('[SearchImage] Image loaded successfully:', imageUrl.value);
};
</script>