视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
vue构建一个自动建站项目
2020-11-27 19:58:31 责编:小采
文档
 这次给大家带来vue构建一个自动建站项目,vue构建一个自动建站项目的注意事项有哪些,下面就是实战案例,一起来看一下。

写在前面

之前一直用Jquery+Jquery-ui来做这个项目,那个时候没有设计稿,没有项目需求,就因为BOSS一句话,要做这样的东西,当时就...好吧!我承认,其实已经习惯了,无所谓了(也是无奈,哎)!!!

在之后的一段时间里,做了一个demo出来,BOSS很满意了,所以自己接下来就慢慢做吧,差不多两三个月吧,就闷头做这个,后来项目上线了,当然因为产品的不完善,还是有点问题了!

不过基本能满足公司的需求了,能编辑的都可以编辑,组件的background(包括背景图片) color border box-shadow margin padding width height 对齐方式(字体和组件内部元素) border-radius font(font-size/font-family)等等这些基础的都可以随心变更,当然考虑到可能满足不了公司的使用,就加了一个自定义样式的功能,而这个只有懂前端的人才能使用了,没办法,需求永远赶不上变化,这样保险一点。因为大家都知道,需求的满足和变更永远跑在现成需求的前面

除了这些基础的可更改,各个组件的特有可变更的功能也基本齐全的,比如轮播图图片变更,轮播方式,控制是否轮播等等这些功能,这里就不一一介绍了

包括后来,因为有组件内部个别元素不能修改,又增加了[绑定修改]功能,就是这个功能选中之后,在视图界面,选中需要修改的元素,便可以进行修改了,这个功能还是有点意思的

说了这么多,其实当时因为做的仓促,存储的时候存的是HTML,大家不要鄙视(要脸0.0),这个也是我心里一直的梗,最近加上BOSS重新提出了一些想法,有蛮多东西要加,思前想后,决定将项目重构一下

考虑到vue响应式与基本是纯数据操作,所以决定使用vue重新构建这个项目。

开发准备

1、使用vue-cli,下载下来配置好的东西
2、因为中间牵涉了拖拽生成组件的操作,所以使用了vuedraggable和sortablejs。

安装vuedraggable sortablejs

npm install vuedraggable
npm install sortablejs

项目中我们只需要引入vuedraggable就可以了,牵涉了sortablejs东西的时候,vuedraggable会去自己加载调用sortablejs里面的方法的,这个就不是我们需要关注的(你如果想了解,可以自己去看看);

3、安装vuex,因为里面牵涉到了大量的数据交互,很多组件都需要一些公用的数据,不使用vuex去管理,将会为开发带来更多不必要的麻烦;

安装vuex

npm install --save vuex

4、因为没有设计稿的缘故,所以大胆使用了第三方UI库 element-ui;

element-ui官网地址

安装elememt

npm install element-ui
//为什么是element-ui而不是element?因为当时npm上已经有了element包了(我当时还觉得挺有意思的,0.0 好冷啊!!!)

5、axios安装,后面与后台数据交互会用到

安装axios

npm install --save axios

差不多准备工作就这些了,接下来我们看项目实施;

项目开始

1、各种文件的配置

-> main.js中文件的配置

图片中都有解释,应该可以看的懂的;

-> 侧边栏拖拽组件数据的配置

因为文件太长,所以删掉了一些,这里就是一个简单的格式,仅供参考,不作为标准;

在组件当中,存在一个布局的问题,所以要有布局组件,让组件可以放到布局组建中,这样才更加的灵活

-> vuexjs 状态管理中的js配置

说明:

1、因为用户在拖拽之后要实时保存到sessionStorage中, 所以再初始的时候要到sessionStroage中去取数据,防止突然刷新页面,还没有保存到数据库中,用户刚刚编辑的数据全部丢失的情况;
2、这里说明一下,可能考虑到用于已经提交了数据,所以用户关闭窗口之后,再次进来的时候,要结合后台给出的用户之前的数据,一起存储到sessionStorage中去,相信这一点大家肯定想的到的,这里善意提醒一下 0.0;
3、我这这里暂时放了四个参数,图中都有说明,我主要是将基本编辑做成了一个组件,会根据用户点击时哪个组件,而重新渲染数据给到编辑组件,从而可以实时对应到点击的组件去编辑;
4、editShow的作用就是控制编辑组件显示与否的,主要删除组件的时候,让编辑组件隐藏一下;点击其他组件的显示一下;

基本的配置就这些了,接下来就是真正的开发了;

2、项目开发开始

-> app.vue文件中该怎么写?

<template>
 <!--用的element-ui-->
 <el-container>
 <el-aside>
 <Draggable class="app-aside-drag" :options="dragOption">
 <p class="app-aside-list" 
 v-for="(dragList,index) in dragData" 
 :type="dragList.type" 
 :key="dragList.type">
 <p class="aside-item-body">
 <i class="aside-item-ele"></i>
 <span class="aside-item-ele">{{ list.title }}</span>
 </p>
 </p>
 </Draggable>
 <el-aside>
 <el-main class="app-main">
 <section class="app-phone">
 <p class="app-phone-header">
 <span class="phone-camera"></span>
 <span class="phone-ls"></span>
 </p>
 <!--页面view区 -->
 <Sort class="app-phone-body"></Sort>
 <p class="app-phone-footer">
 <button class="app-phone-menu">RS</button>
 </p>
 </section>
 </el-main>
 <el-aside class="app-right">
 <!--组件编辑区域-->
 <BaseEdit></BaseEdit>
 </el-aside>
 </el-container> 
</template>
<script>
import DragApi from "@/dragapi/dragapi.js";
import Draggable from "vuedraggable";
import Sort from "@/view/Sort";
import BaseEdit from "@/view/BaseEdit";
export default {
 name: 'app',
 data(){
 return{
 dragData: {},
 dragOption: {
 group: {
 name: 'components', //这个很重要,其他的与之能产生关联的拖拽框就靠这name 一定要一致
 pull: 'clone', 
 put: false
 },
 sort: false //默然为true。这里我们只需要他拖拽,无需能拖动排序
 }
 }
 },
 components: {
 Draggable,
 Sort,
 BaseEdit
 },
 created(){
 //侧边栏拖拽列表数据
 //这里我只写了组件的数据进来,布局的暂时没放
 this.dragData = DragApi.configList[1].content;
 }
}
</script>

-> 来看看sort view视图区域组件

<template>
 <Draggable :options="sortOption"
 @sort="onSort"
 @add="onAdd"
 class="app-sort">
 <!-- ui组件 -->
 <!--这里不懂的人,可以去vue官网看看动态组件-->
 <p v-for="(appUi,index) in sortApi" //循环组件
 :is="appUi.component" //根据存在的组件渲染出来
 :content="appUi.content"
 :oStyle="appUi.style"
 :editPartShow="appUi.editPartShow"
 :aIndex="index"
 //组件想要点击生效,只需要@click.native就行了
 @click.native="getIndex(index)"
 //key值一定要给出来,不然相同组件的排序可能会不成功
 :key="appUi.content.code">
 </p>
 </Draggable>
</template>
<script>
 //利用vuex 辅助函数来操作vuexjs中的数据
 import { mapState,mapMutations } from 'vuex';
 //拖拽插件引入
 import Draggable from 'vuedraggable';
 //各个组件引入
 import Carousel from "@/components/Carousel.vue";
 import Btn from "@/components/Btn.vue";
 export default {
 name: 'Sort',
 components: {
 Draggable,Btn,Carousel
 },
 data(){
 return {
 sortOption: {
 group: {
 name: 'components', //前面说的name,在这里就起了作用,不一样,是不能放入的
 pull: true,
 put: true
 },
 sort: true,
 animation: 300 //给了个动画,看起来舒服些
 }
 }
 },
 computed:{
 ...mapState(['editIndex','sortApi']),
 },
 watch:{
 sortApi:{
 handler(newVal,oldVal){
 window.sessionStorage.setItem('localData',JSON.stringify(newVal));
 },
 deep: true
 }
 },
 methods:{
 ...mapMutations(['sortCp','addCp','setStyle','setCommon']),
 onSort(res){ //排序产生的事件
 if(res.from === res.to){
 this.sortCp(res);
 }
 },
 onAdd(res){//组件增加产生的事件
 this.addCp(res);
 },
 getIndex(index){
 this.setCommon({index: index,flag: true});
 }
 }
 }
</script>

-> 再来看看编辑组件

<template>
 <transition name="slide-right">
 <p v-if="sortApi.length > 0 && editShow === true">
 //组件特有编辑
 <el-tabs v-model="activeName">
 <el-tab-pane label="组件设置" name="first">
 <p v-for="(appUi,index) in sortApi"
 :is="appUi.component+'Edit'"
 :content="appUi.content"
 :oStyle="appUi.style"
 :editPartShow="appUi.editPartShow"
 :aIndex="index"
 :currentIndex="editIndex"
 :key="appUi.content.code">
 </p>
 </el-tab-pane>
 <el-tab-pane label="样式设置" name="second">
 //公共样式编辑
 <el-collapse v-model="colorPicker.name" class="base-edit" accordion>
 <el-collapse-item class="tititt" :title="colorPicker.type" :name="colorPicker.type">
 <el-form ref="form" :model="colorPicker" size="mini">
 <el-form-item class="cui-inline-reset"
 v-for="(item,index) in colorPicker.content"
 :label="item.title"
 :key="item.style">
 <el-color-picker
 //在element-ui框架中,有很多@change @active-change事件,直接写事件发现不能传入参数,
 //当然,办法总比问题多,我们换成一下这种写法就行了,他的默然参数写在前面
 //这里颜色拾取器 返回的是实时的颜色值
 //我这里主要想传一个对应的style
 @active-change=" (value) => setStyle(value,item.style)"
 v-model="sortApi[editIndex].style[item.style]"
 show-alpha>
 </el-color-picker>
 <span class="black-text-shadow"
 :style="{color: sortApi[editIndex].style[item.style]}">
 {{ sortApi[editIndex].style[item.style] }}
 </span>
 </el-form-item>
 </el-form>
 </el-collapse-item>
 </el-collapse>
 </el-tab-pane>
 </el-tabs>
 </p>
 </transition>
</template>
<script>
 import { mapState,mapMutations } from 'vuex';
 //这里我将组建特有的编辑栏,写成了一个组件,为什么不写在相应的组件一起了?
 //这里必须说明一下,主要是我没有想到方法,让他在同一组件内分离出来,单独将dom结构放在编辑栏这里,如果有大神知道
 //还望不吝赐教
 import BtnEdit from "@/components/BtnEdit.vue";
 
 export default{
 name: 'BaseEdit',
 components: {
 BtnEdit
 },
 data(){
 return{
 colorPicker: {
 type: '颜色设置',
 name: 'Picker',
 content:[
 {
 title: '背景颜色',
 style: 'background'
 },
 {
 title: '字体颜色',
 style: 'color'
 }
 ]
 
 },
 activeName: 'first'
 }
 },
 
 computed:{
 ...mapState(['editIndex','sortApi','editShow'])
 },
 methods:{
 setStyle(value,style){
 //根据上面传入的style属性,实时改变现有的值
 this.$set(this.sortApi[this.editIndex].style,style,value);
 }
 }
 }
</script>

-> 选出一个组件来看看里面是怎么配置的

//按钮组件,其实里面很简单
//组件的对应的编辑组件,里面内容和这个也差不多,下面就不写了
<template>
 <p class="btn-box ui-sortable" :data-code="content.code">
 <el-button class="ui-btn"
 :style="oStyle">
 {{ content.text }}
 </el-button>
 //因为每个组件都有删除功能,所以写成了一个组件
 <DeleteCp :aIndex="aIndex"></DeleteCp>
 </p>
</template>
<script>
 import DeleteCp from "@/components/DeleteCp";
 export default {
 name: 'Btn',
 props: { //父组件传入的参数
 content: Object,
 oStyle: Object,
 aIndex: Number
 },
 components: {
 DeleteCp
 },
 data(){
 return{
 btnModel: 'btn-model'
 }
 }
 }
</script>

->最后来看看删除组件吧

<template>
 <p class="delete-compontent-box">
 <p class="el-icon-delete remove-component" @click.stop="dailogStatu"></p>
 <el-dialog
 title="提示"
 :visible.sync="dialogVisible"
 :append-to-body="appendToBody"
 width="430px">
 <p class="el-message-boxcontent">
 <p class="el-message-boxstatus el-icon-warning"></p>
 <p class="el-message-boxmessage dialog-message">此操作将删除该模块, 是否继续?</p>
 </p>
 <span slot="footer" class="dialog-footer">
 <el-button @click="dialogVisible = false" size="small">取 消</el-button>
 <el-button type="primary" @click="onRemove(aIndex)" size="small">确 定</el-button>
 </span>
 </el-dialog>
 </p>
</template>
<script>
 import { mapMutations } from "vuex";
 export default {
 name: 'oText',
 props: {
 aIndex: Number
 },
 data(){
 return{
 //这两个参数是弹框的参数
 dialogVisible: false,
 appendToBody: true 
 }
 },
 methods:{
 ...mapMutations(['deleteCp','setCommon']),
 dailogStatu(){
 //主要是控制弹窗出来,并且显示该组件对应的编辑栏
 this.dialogVisible = true;
 this.setCommon({flag: true,index: this.aIndex})
 },
 onRemove(index){
 //点击确定删除对应的组件
 let flag = false;
 this.deleteCp(index);
 this.dialogVisible = false;
 this.$message({
 message: '该模块已删除 !',
 type: 'success'
 });
 this.setCommon({flag: false,index: 0})
 }
 }
 }
</script>


-> 来看看效果图吧

效果图展示

结束语

好了,今天写了很多了,最后我们来梳理一下思路:

1、首先配置左侧的拖拽组件
2、配置vuex中的数据
3、app.vue中配置
4、编辑组件的配置
5、各种数据的传递与依赖

其实每个项目,都需要一个清晰的路线,这样才能很好的开发下去,所以我的建议是,在拿到项目的时候,千万不要一股脑的去写,一定要想好怎么做,以及突发事情的发生(比如突来的需求变更),这样既方便了我们自己,也方便了后来维护的人,也阻止了不必要的麻烦

谢谢大家的耐心的阅读,毕竟这只是一个大概的介绍,肯定存在很多不足,如果大家有建议,欢迎留言交流,也希望大家多多支持脚本之家。

相信看了本文案例你已经掌握了方法,更多精彩请关注Gxl网其它相关文章!

推荐阅读:

JS如何做出公共子序列

datepicker插件监听输入框

ejsExcel模板在Vue.js中的使用

D3.js创建物流地图

下载本文
显示全文
专题