在Django+Vue3+GraphQL的Blog例子代码中引入Element-Plus UI Framework

Vue3的UI Framework中有Element-Plus、BalmUI、Quasar、PrimeVue、Ant Design Vue等UI Framework.

Element-Plus是Element-UI的Vue3版,Element-UI的使用人数的基数较大,Github上的Star数也较多,就选择了Element-Plus作为这个Blog项目的UI Framework.

UI Framework的好处就是提供了相较裸Html+CSS开发起来更好看和方便一些.

Element-Plus在国内有镜像站点,参看起来速度相对快一些. Element-Plus国内镜像站点.

代码可以从Github上取得https://github.com/magicduan/django_vue_graphql/releases/tag/base_0.2

配置Element-UI开发环境

如果在已有的项目中安装的用包管理器安装最方便了.

 npm install element-plus --save

由于对Element-Plus的组件的使用方法不是很熟练, 按照官网的说明我直接从Element-Plus提供的Vite的快速模版进行启动.

https://github.com/element-plus/element-plus-vite-starter

从Github上下载代码后展开,执行下列命令就可以启动一个Element-Plus的Demo服务了.

npm install
npm run dev

按照前一篇Blog例子代码, 修改vite.config.ts,加入端口配置,将Vue的端口设置为8080

export default defineConfig({
....
  server:{
    port:8080
  },
....

使用npm 安装vue-router、apollo-client

npm install vue-router@4
npm install --save graphql graphql-tag @apollo/client
npm install --save @vue/apollo-composable

 

迁移代码

由于是后安装的vue-router,缺省的vue-router 没有设置, 建立与Frontend相同的目录

src/router
src/views

将Blog项目中的Frontend中相关的代码Copy进入对应目录

src/router/index.ts
src/views/AllPosts.vue
src/views/AuthorView.vue
src/views/PostsByTag.vue
src/views/PostView.vue
src/componets/AuthorLink.vue
src/componets/PostLis.vue

修改顶部菜单BaseHeader.vue

Element-UI模版提供的是顶菜单+左工具栏的UI模版, src/compoments/layout/BaseHeader.vue 为顶部菜单, BaseSide.vue为左边菜单.

这里我们修改BaseHeader.vue,修改后菜单项为

MyBlog --> HomeView.vue
Posts 
   All Posts --> All Posts
   Add Post //将来实现
   Delete Post//将来实现
Tag ->根据Backend端的取得Tag的种类动态生成Tag菜单
HelloWord -->对应原Element-UI的Helloworld的例子

由于需要根据BackEnd端的Tag查询结果动态生成Tag,我们为BaseHeader.vue加入Props :tagItems

Baseheader.vue代码具体修改如下:

<script setup lang="ts" >
import { toggleDark } from '~/composables';
import { ref } from 'vue';
const props = defineProps({
  tagItems:{type:Array,required:true, default:[]}
})

const msg = ref("/hello/\"Hello Vue 3.0 + Element Plus + Vite\"")

</script>
<template>
  <el-menu class="el-menu-demo" mode="horizontal" :router= true>
    <el-menu-item index="/">My Blog</el-menu-item>
    <el-sub-menu index="2">
      <template #title>Posts</template>
      <el-menu-item index="/posts">All Posts</el-menu-item>
      <el-menu-item index="2-1">Add Post</el-menu-item>
      <el-menu-item index="2-2">Delete Post</el-menu-item>
      <el-sub-menu index="2-4">
        <template #title>...</template>
        <el-menu-item index="2-4-1">item one</el-menu-item>
        <el-menu-item index="2-4-2">item two</el-menu-item>
        <el-menu-item index="2-4-3">item three</el-menu-item>
      </el-sub-menu>
    </el-sub-menu>

    <el-sub-menu v-if="tagItems.length" index="tags" >
      <template #title>Tags</template>
      <el-menu-item v-for="tagInfo in tagItems" :index="tagInfo.path">
        {{ tagInfo.name }}
      </el-menu-item>
    </el-sub-menu>
    <el-menu-item v-else :index="tags" disabled>Tags</el-menu-item>

    <el-menu-item :index="msg" >HelloVue</el-menu-item>

    <el-menu-item h="full" @click="toggleDark()">
      <button class="border-none w-full bg-transparent cursor-pointer" style="height: var(--ep-menu-item-height);">
        <i inline-flex i="dark:ep-moon ep-sunny" />
      </button>
    </el-menu-item>
  </el-menu>
</template>

 其中需要注意的几点:

  • 将菜单项与vue-router结合起来,菜单项的index属性为router/index.ts中设置的URL path, 需要设置el-menu
    :router= true
  • HelloWorld.vue 其中需要属性msg,为了在菜单中进行配置,我们需要在index.ts中将HelloWord.vue的URL的prop设置为true
  
const router = createRouter({
...  
    {
      path: '/hello/:msg',
      name: 'HellowWorld',
      component: () => import("../components/HelloWorld.vue"),
      props: true  
    },
...

将HelloWorld.vue Component的Prop属性作为参数来传递

修改App.vue

在App.vue中加入从Backend通过GraphQL取Tags的代码, 将Tags数组传递给BaseHeader.vue,将<HelloWord .../>改为<routerview/>

<script setup lang="ts">
import HomeView from './views/HomeView.vue';
import HelloWorld from './components/HelloWorld.vue';
import gql from 'graphql-tag';
import { useQuery } from '@vue/apollo-composable';
import { computed } from '@vue/reactivity';

const {result,loading,error} =  useQuery(gql`
  query getTags{
    tags {
          name
    }
  }
`)

const tags = computed(()=>{
  let tagsArray = []
  if (result.value){
    result.value?.tags.forEach(tagName => {
      let tagInfo = {path:"/tag/"+tagName.name,name:tagName.name}
      tagsArray.push(tagInfo)
    });
  }
  return tagsArray

})

</script>
<template>
  <el-config-provider namespace="ep" >
    <BaseHeader :tagItems="tags"/>
    <div style="display: flex">
      <BaseSide />
      <div>
        <!-- <HelloWorld msg="Hello Vue 3.0 + Element Plus + Vite" @changeTags ='changTags'/>  -->
        <RouterView/> 
      </div>
    </div>
  </el-config-provider>
</template>

<style>
#app {
  text-align: center;
  color: var(--ep-text-color-primary);
}

.element-plus-logo {
  width: 50%;
}
</style>

在Backend中加入Tags的查询

backend/blog/scheme/py

class Query(graphene.ObjectType):
    ....
    tags = graphene.List(TagType)
    .....
    def resolve_tags(root,info):
        return (
            models.Tag.objects.all()
        )

 

修改main.ts加入router,Apollo相关的代码

import { DefaultApolloClient } from '@vue/apollo-composable'
import { ApolloClient, createHttpLink, InMemoryCache } from '@apollo/client/core'

// HTTP connection to the API
const httpLink = createHttpLink({
  uri: 'http://localhost:8000/graphql',
})

// Cache implementation
const cache = new InMemoryCache()

// Create the apollo client
const apolloClient = new ApolloClient({
  link: httpLink,
  cache,
})

const app = createApp({
    setup () {
      provide(DefaultApolloClient, apolloClient)
    },
  
    render: () => h(App),
  })

app.use(router)

 

至此blog相关的代码迁移完成, 可以完成Posts List等操作了,但是页面相对不太好看,下一步我们使用Element-UI的控件进行对应改造即可.

使用Element-UI改造Posts等相关的Vue

在技术打通之后,UI的改造就是细心+审美的工作了,按照我自己的意思进行了修改,具体的修改就不赘述了,直接参考代码即可.

 

原文链接:https://www.cnblogs.com/magicduan/p/17315028.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:在Django+Vue3+GraphQL的Blog例子代码中引入Element-Plus UI Framework - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • Python入门教程4. 元组基本操作 原创

    下面是详细讲解“Python入门教程4.元组基本操作原创”的完整攻略: 一、前言 本教程是Python入门教程的第四篇,主要介绍Python中元组的基本操作。 二、元组的定义 和列表(List)一样,元组(Tuple)也是一种常见的序列类型,它可以存储多个有序的元素,而且一旦创建后,它的元素就不能被修改了。元组的定义方式如下: tuple_name = (e…

    python 2023年5月14日
    00
  • 在Python中对具有多维系数的Legendre数列进行微分

    在Python中,可以使用SciPy库里的模块 special 来计算Legendre多项式,并进行微分操作。 首先,使用以下代码导入 special 模块: from scipy import special 接着,使用以下代码计算具有n阶和m阶的Legendre多项式: n = 2 m = 3 pnm = special.lpmv(m,n,0.5) # …

    python-answer 2023年3月25日
    00
  • Python基于pillow判断图片完整性的方法

    下面是详细讲解 “Python基于pillow判断图片完整性的方法” 的完整攻略。 简介 在处理图片的过程中,有时候需要判断图片是否完整。图片完整性通常指图片文件是否可以被正确地打开、读取、解压,以及其中的像素数据是否能够正常的被读取。在Python中,我们可以使用Pillow作为图片处理库来实现判断图片完整性的操作。 步骤 下面是Python基于pillo…

    python 2023年5月18日
    00
  • Python轻松写个课堂随机点名系统

    下面是详细的“Python轻松写个课堂随机点名系统”攻略: 1. 确定目的和需求 在编写程序之前,我们需要明确系统的目的和需求。点名系统的目的是随机选取课堂中的学生进行点名,方便老师进行点名操作。系统需求包括: 存储学生名单 随机选取学生 统计已点名人数 显示已点名学生名单 2. 准备工作 在编写程序之前,我们需要准备好开发环境和所需材料。开发环境可以选择A…

    python 2023年6月3日
    00
  • 详解Python高阶函数

    详解Python高阶函数攻略 什么是高阶函数 高阶函数是指可以接受其他函数作为参数或返回一个函数作为结果的函数。在Python中,函数可以被视为数据类型,也就是说,函数可以作为另一个函数的参数进行传递或作为另一个函数的返回值进行返回。 为什么要使用高阶函数 通过使用高阶函数,我们可以让我们的代码更加简洁、优雅,减少重复的代码,提高代码的可读性和复用性。 高阶…

    python 2023年6月5日
    00
  • python实现一个猜拳游戏

    下面是一个完整的Python实现猜拳游戏的攻略: 1. 需求分析 在实现猜拳游戏前,我们需要对需求做一些简单的分析: 猜拳游戏可以进行多轮; 电脑和玩家都可以选择石头、剪刀、布这三个选项; 对手的选择是随机的,玩家需要输入自己的选择; 根据猜拳规则,有获胜、失败和平局三种结果; 每轮游戏结束后需要显示对手的选择、玩家的选择和当前的游戏结果。 根据以上需求,我…

    python 2023年5月19日
    00
  • Python基于动态规划算法解决01背包问题实例

    Python基于动态规划算法解决01背包问题实例 什么是01背包问题? 01背包问题是一个经典的动态规划问题,它的基本想是在给定的一组物品中选择一物品,使得这些物品总重量不超过背包的容量,同时总值最大。 动态规划算法解决01背包问题 动态规划算法一种常用的算法思想,它的基本思想是将一个大问题解成若干个小问题,然后逐步解决这小问题,最终得到大问题的解。在决01…

    python 2023年5月14日
    00
  • Python爬虫之urllib库详解

    Python爬虫之urllib库详解 什么是urllib库 urllib库是Python内置的HTTP请求库,包含了一组简单的API,可以用来发送GET、POST、PUT、DELETE、HEAD等HTTP请求,支持处理URL、Cookie、代理、验证、浏览器标识等常见的HTTP请求需求。 urllib库的常见模块 urllib库包含了四个常用的模块,分别是:…

    python 2023年5月14日
    00
合作推广
合作推广
分享本页
返回顶部