Bootstrap

uniapp(接入智谱Ai完整示例)

说明

带聊天框、带复制,其他自定义

<template>
  <view class="container box-border">
    <view @click="closeOutside">
      <wd-drop-menu>
        <wd-drop-menu-item v-model="selectLLM" :options="optionsLLM" @change="LLMChage" />
        <wd-drop-menu-item title="次数" ref="dropMenu">
          <view class="p-3">剩余可使用次数:9999</view>
        </wd-drop-menu-item>
      </wd-drop-menu>
    </view>
    <scroll-view
      scroll-y
      class="w-[96%] m-auto rounded-2 box-border mt-4 p-3 bg-white min-h-[20vh] max-h-[76vh]"
    >
      <view class="text-[#999] text-[13px] pb-3" v-if="chatData.length == 0">
        请在下方输入框输入内容进行使用
      </view>
      <!-- 返回数据内容 -->
      <view class="list">
        <view v-for="(item, index) in chatData" :key="index">
          <!-- 用户数据 -->
          <view class="right text-right" v-if="item.type == 'user'">
            <view class="flex items-center justify-end">
              <view class="text-[14px] font-bold pr-1">用户1</view>
              <wd-img
                :width="36"
                :height="36"
                round
                src="/images/20231218/202312181827416f1553423.png"
              />
            </view>
            <view>{{ item.content }}</view>
          </view>
          <!-- Ai 回复 -->
          <view class="left mb-[40rpx]" v-else>
            <view class="flex items-center">
              <wd-img
                :width="36"
                :height="36"
                round
                src="/uploads/images/20231221/2023122119253887c6f3341.png"
              />
              <view class="text-[14px] font-bold pl-1">美育云AI</view>
            </view>
            <view
              class="bg-[#f5f5f5] p-2 mt-1 rounded-1"
              v-html="item.content.length > 0 ? item.content : responseData"
            ></view>
            <view
              class="flex items-center mt-1"
              v-if="item.content.length > 0"
              @click="copyText(item.content)"
            >
              <wd-icon name="edit-1" size="22px" color="#999"></wd-icon>
              <view class="pl-1 text-#999">复制</view>
            </view>
          </view>
        </view>
      </view>
    </scroll-view>

    <!-- 输入框,发送按钮 -->
    <view class="w-full mt-4 p-3 box-border fixed bottom-5">
      <wd-input
        v-model="inputText"
        placeholder="请输入内容"
        class="w-full m-auto rounded-[10px]"
        no-border
        custom-input-class="inputTextClass"
      >
        <template #suffix>
          <view v-if="!isLoading">
            <wd-button type="primary" @click="handleClick">发送</wd-button>
          </view>
          <view class="text-[13px] py-2 text-#999 pr-2" v-else>加载中...</view>
        </template>
      </wd-input>
    </view>
  </view>
</template>

<script lang="ts" setup>
import { marked } from 'marked'
import { ref } from 'vue'
import { useQueue } from 'wot-design-uni'
const { closeOutside } = useQueue()

const selectLLM = ref<number>(0)

const optionsLLM = ref<Record<string, any>>([
  { label: '智谱清言', value: 0 },
  { label: '通义千问', value: 1 },
])

//  定义完整聊天数据
const chatData = ref<Record<string, any>[]>([])

//  定义提问次数
const askTimes = ref<number>(0)
// 定义模型切换
function LLMChage({ value }) {
  console.log(value)
}

// 点击复制
function copyText(text: string) {
  //  去除html标签
  text = text.replace(/<[^>]+>/g, '')
  uni.setClipboardData({
    data: text,
    success: function () {
      uni.showToast({
        title: '复制成功',
        icon: 'none',
      })
    },
  })
}

//  输入文本
const inputText = ref<String>('')

//  定义响应数据
const responseData = ref<String>('')

//  定义点击事件
const handleClick = () => {
  //  插入一条聊天数据
  chatData.value.push({
    type: 'user',
    content: inputText.value,
  })
  sendRequest()
}

//  是否加载中
const isLoading = ref<Boolean>(false)
//  定义请求
const sendRequest = () => {
  //  插入一条数据
  chatData.value.push({
    type: 'ai',
    content: '',
  })

  const xhr = new XMLHttpRequest()

  isLoading.value = true

  // 打开 POST 请求
  xhr.open('POST', 'https://open.bigmodel.cn/api/paas/v4/chat/completions', true)

  // 设置自定义请求头
  xhr.setRequestHeader('Authorization', 'Bearer 你自己的key')
  xhr.setRequestHeader('Content-Type', 'application/json')

  // 发送 POST 请求,带参数
  xhr.send(
    JSON.stringify({
      model: 'glm-4-plus',
      messages: [{ role: 'user', content: inputText.value }],
      stream: true,
    }),
  )

  // 监听流数据
  xhr.onreadystatechange = async () => {

    /**
 *  @author: xl
 *  @date: 2025/01/12
 *  @description: 智谱清言语言模型接入 wechat:suiyi01001
 *  @version: 1.0.0
 */

    //  说明有数据返回
    if (xhr.readyState === 3) {
      //   分割数据
      const jsonStringsArray = xhr.responseText
        .split('\n')
        .filter((line) => line.startsWith('data: '))

      //   循环分割的数据
      let dataText = ''
      for (var i = 0; i < jsonStringsArray.length; i++) {
        if (jsonStringsArray[i].slice(6) == '[DONE]') {
          console.log('SSE 已关闭连接')
          setTimeout(() => {
            isLoading.value = false
            // 修改聊天数据
            console.log('responseData: ', chatData.value, chatData.value[chatData.value.length - 1])
            // chatData.value.push({
            //   type: 'ai',
            //   content: responseData.value,
            // })
            chatData.value[chatData.value.length - 1].content = responseData.value
          }, 1000)
        } else {
          const jsonObject = JSON.parse(jsonStringsArray[i].slice(6)) // 剔除"data:"前缀
          // console.log('jsonObject: ', jsonObject);
          const choices = jsonObject.choices // 提取choices数组
          // 存在数据输出结果
          if (jsonObject.choices && jsonObject.choices[0].delta) {
            dataText += jsonObject.choices[0].delta.content
          }
        }
      }
      //  输出最终数据
      responseData.value = await marked(dataText)
    }
  }

  // 监听错误
  xhr.onerror = (err) => {
    console.error('SSE 错误:', err)
  }
}
</script>

<style lang="scss" scoped>
page {
  background-color: #f5f5f5;
}
</style>

;