1. 效果演示

2. 主要知识点

  • 使用slot分发内容
  • 动态组件
  • 组件通信
  • 实例的生命周期
  • 表单

3. 遇到的问题

  • bus 通信 第一次 $on 监听不到

    1. // 解决bus 第一次通信 $on 监听不到
    2. this.$nextTick(function () {
    3. if (_this.totalSize - 1 == _this.order) {
    4. bus.$emit('submiteDisabled', disabledStatus)
    5. } else {
    6. bus.$emit('nextStepDisabled', disabledStatus)
    7. }
    8. })
  • bus 通信 $on 多次触发

    1. // bus传值之后要进行销毁, 尤其是跳转页面进行使用的时候
    2. beforeDestroy() {
    3. if (this.totalSize - 1 == this.order) {
    4. bus.$off('submiteDisabled')
    5. } else {
    6. bus.$off('nextStepDisabled')
    7. }
    8. },

4. 代码整理

  • HTML

    1. <!DOCTYPE html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <title>答题卡</title>
    6. <link rel="stylesheet" href="./css/style.css">
    7. </head>
    8. <body>
    9. <div id="app">
    10. <main-component v-for="(question, index) in questions" :key="index" :title="question.title" :order="index"
    11. v-if="index === currentOrder" :content="question.chooses">
    12. <template slot="title" slot-scope="props">
    13. <span>{{props.order}}. {{props.title}}</span>
    14. </template>
    15. <template slot="content" slot-scope="props">
    16. <component :is="props.type"
    17. :datas="question.chooses.values"
    18. :selected = "question.chooses.selected"
    19. :order = "index"
    20. :total-size = "questions.length"
    21. v-if="props.type != ''"></component>
    22. <textarea-component :value="question.chooses.value"
    23. :order="index"
    24. :total-size = "questions.length" v-else></textarea-component>
    25. </template>
    26. <template slot="functionalDomain">
    27. <button-component :now-order = "currentOrder"
    28. :total-elements = "questions.length"
    29. :next-step-disabled = "true"
    30. ></button-component>
    31. </template>
    32. </main-component>
    33. </div>
    34. <script src="./js/vue.js"></script>
    35. <script src="./js/index.js"></script>
    36. </body>
    37. </html>
  • JS

    1. var bus = new Vue();
    2. Vue.component('main-component', {
    3. props: {
    4. order: Number,
    5. title: String,
    6. content: {
    7. type: Object
    8. }
    9. },
    10. template: '<div class="main">\
    11. <div class="title">\
    12. <slot name="title" :title="mytitle" :order="myorder + 1"></slot>\
    13. </div>\
    14. <div class="content">\
    15. <slot name="content" :type="typeSelect"></slot>\
    16. </div>\
    17. <div class="footer">\
    18. <slot name="functionalDomain" ></slot>\
    19. </div>\
    20. </div>',
    21. data: function () {
    22. return {
    23. mytitle: this.title,
    24. myorder: this.order
    25. }
    26. },
    27. computed: {
    28. typeSelect: function () {
    29. let type_co = this.content.type;
    30. if ('radio' == type_co || 'checkbox' == type_co) {
    31. return type_co + '-component';
    32. }else{
    33. return '';
    34. }
    35. }
    36. },
    37. });
    38. Vue.component('radio-component', {
    39. props:{
    40. selected: String,
    41. datas:Array,
    42. order: Number,
    43. totalSize: Number
    44. },
    45. template: '<div>\
    46. <label v-for="(item, index) in mydatas" :key="index"><input type="radio" :value="item.value" v-model="results" />{{item.name}}</label>\
    47. </div>',
    48. data:function (){
    49. return {
    50. results: this.selected,
    51. mydatas: this.datas
    52. }
    53. },
    54. methods: {
    55. updatNextStepStatus: function (val) {
    56. let disabledStatus = true;
    57. let _this = this;
    58. if (val != '') {
    59. disabledStatus = false;
    60. }
    61. // 解决bus 第一次通信 $on 监听不到
    62. this.$nextTick(function () {
    63. if (_this.totalSize - 1 == _this.order) {
    64. bus.$emit('submiteDisabled', disabledStatus)
    65. } else {
    66. bus.$emit('nextStepDisabled', disabledStatus)
    67. }
    68. })
    69. }
    70. },
    71. watch: {
    72. results: function (val) {
    73. this.updatNextStepStatus(val);
    74. bus.$emit('valueChange', this.order, val)
    75. },
    76. selected: function (val) {
    77. this.results = val
    78. }
    79. },
    80. mounted() {
    81. this.updatNextStepStatus(this.results);
    82. },
    83. beforeDestroy() {
    84. if (this.totalSize - 1 == this.order) {
    85. bus.$off('submiteDisabled')
    86. } else {
    87. bus.$off('nextStepDisabled')
    88. }
    89. },
    90. });
    91. Vue.component('checkbox-component', {
    92. props: {
    93. selected: Array,
    94. datas: Array,
    95. order: Number,
    96. totalSize: Number
    97. },
    98. template: '<div>\
    99. <label v-for="(item, index) in mydatas" :key="index"><input type="checkbox" :value="item.value" v-model="results" />{{item.name}}</label>\
    100. </div>',
    101. data: function () {
    102. return {
    103. results: this.selected,
    104. mydatas: this.datas
    105. }
    106. },
    107. methods: {
    108. updatNextStepStatus: function (newValue, oldValue) {
    109. let disabledStatus = true;
    110. let _this = this;
    111. if (oldValue.length < 2 && newValue.length < 2) {
    112. return;
    113. } else if (oldValue.length >= 2 && newValue.length < 2) {
    114. disabledStatus = true;
    115. } else if (newValue.length >= 3){
    116. alert("多选,最多选择3个选项")
    117. }else{
    118. disabledStatus = false;
    119. }
    120. // 解决bus 第一次通信 $on 监听不到
    121. this.$nextTick(function () {
    122. if (_this.totalSize - 1 == _this.order) {
    123. bus.$emit('submiteDisabled', disabledStatus)
    124. } else {
    125. bus.$emit('nextStepDisabled', disabledStatus)
    126. }
    127. })
    128. }
    129. },
    130. watch: {
    131. results: function (newValue, oldValue) {
    132. this.updatNextStepStatus(newValue, oldValue);
    133. bus.$emit('valueChange', this.order, newValue)
    134. },
    135. selected: function (val) {
    136. this.results = val
    137. }
    138. },
    139. mounted() {
    140. this.updatNextStepStatus(this.results,[]);
    141. },
    142. beforeDestroy() {
    143. if (this.totalSize - 1 == this.order) {
    144. bus.$off('submiteDisabled')
    145. } else {
    146. bus.$off('nextStepDisabled')
    147. }
    148. },
    149. });
    150. Vue.component('textarea-component', {
    151. props: {
    152. value: String,
    153. order: Number,
    154. totalSize: Number
    155. },
    156. template: '<textarea v-model="results" placeholder="不少于100字"></textarea>',
    157. data: function () {
    158. return {
    159. results: this.value,
    160. }
    161. },
    162. methods: {
    163. updatNextStepStatus: function (newValue, oldValue) {
    164. let disabledStatus = true;
    165. let _this = this;
    166. if (newValue.length > 0 && newValue.length <= 5) {
    167. disabledStatus = false;
    168. }else{
    169. disabledStatus = true;
    170. }
    171. // 解决bus 第一次通信 $on 监听不到
    172. this.$nextTick(function () {
    173. if (_this.totalSize - 1 == _this.order) {
    174. bus.$emit('submiteDisabled', disabledStatus)
    175. }else{
    176. bus.$emit('nextStepDisabled', disabledStatus)
    177. }
    178. })
    179. }
    180. },
    181. watch: {
    182. results: function (newValue, oldValue) {
    183. this.updatNextStepStatus(newValue, '');
    184. bus.$emit('valueChange', this.order, newValue)
    185. },
    186. value: function (val) {
    187. this.results = val
    188. }
    189. },
    190. mounted() {
    191. this.updatNextStepStatus(this.results, '');
    192. },
    193. beforeDestroy() {
    194. if (this.totalSize - 1 == this.order) {
    195. bus.$off('submiteDisabled')
    196. }else{
    197. bus.$off('nextStepDisabled')
    198. }
    199. },
    200. });
    201. Vue.component('button-component', {
    202. props: {
    203. basicClasses: {
    204. type: Object,
    205. default: function () {
    206. return {
    207. 'button-white': true
    208. }
    209. }
    210. },
    211. nextStepClasses: {
    212. type: Object,
    213. default: function () {
    214. return {
    215. 'button-primary': true
    216. }
    217. }
    218. },
    219. nextStepDisabled:{
    220. type: Boolean,
    221. default: false
    222. },
    223. nowOrder: Number,
    224. totalElements: Number
    225. },
    226. template: '<div>\
    227. <button class="btn" :disabled="nextStepStatus" :class="primary" \
    228. v-if="order != total - 1" @click="handleNextStep">下一步</button>\
    229. <button class="btn" :disabled="submiteStatus" :class="primary" \
    230. v-if="order == total - 1" @click="handleSubmit">提交</button>\
    231. <button class="btn" :disabled="false" :class="basic"\
    232. v-if= "order != 0" @click="handleBackStep">上一步</button>\
    233. <button class="btn" :disabled="false" @click="handleReset" :class="basic">重置</button>\
    234. </div>',
    235. data() {
    236. return {
    237. basic: this.basicClasses,
    238. primary: this.nextStepClasses,
    239. nextStepStatus: this.nextStepDisabled,
    240. order: this.nowOrder,
    241. total: this.totalElements,
    242. submiteStatus: true
    243. }
    244. },
    245. methods: {
    246. handleNextStep: function () {
    247. bus.$emit('nextStep')
    248. },
    249. handleBackStep: function () {
    250. bus.$emit('backStep')
    251. },
    252. handleSubmit: function () {
    253. bus.$emit('submit')
    254. },
    255. handleReset: function () {
    256. let _this = this;
    257. bus.$emit('reset', _this.order)
    258. }
    259. },
    260. mounted() {
    261. let _this = this;
    262. bus.$on('nextStepDisabled', function (status) {
    263. _this.nextStepStatus = status
    264. });
    265. bus.$on('submiteDisabled', function (status) {
    266. _this.submiteStatus = status
    267. })
    268. },
    269. });
    270. var app = new Vue({
    271. el: "#app",
    272. data: {
    273. currentOrder: 0,
    274. questions: [
    275. {
    276. title: '请问您的性别是?',
    277. chooses: {
    278. type: 'radio',
    279. values: [{ 'name': '男', 'value': '1' }, { 'name': '女', 'value': '2' }, { 'name': '保密', 'value': '3' }],
    280. selected: ''
    281. }
    282. },
    283. {
    284. title: '请选择您的兴趣爱好?',
    285. chooses: {
    286. type: 'checkbox',
    287. values: [{ 'name': '看书', 'value': 'read book' },
    288. { 'name': '游泳', 'value': 'swim' },
    289. { 'name': '跑步', 'value': 'run' },
    290. { 'name': '看电影', 'value': 'see movie' }],
    291. selected: []
    292. }
    293. },
    294. {
    295. title: '请介绍一下你自己',
    296. chooses: {
    297. type: 'text',
    298. value: ''
    299. }
    300. },
    301. ]
    302. },
    303. methods: {
    304. handleNext: function () {
    305. let current = this.currentOrder;
    306. if (++current >= this.questions.length) {
    307. return;
    308. }else{
    309. this.currentOrder += 1;
    310. }
    311. },
    312. handleBack: function () {
    313. let currentOrder = this.currentOrder;
    314. if (-- currentOrder < 0) {
    315. return;
    316. }else{
    317. this.currentOrder -= 1;
    318. }
    319. },
    320. handleReset: function (order, val) {
    321. let currentQuestion = this.questions[order].chooses;
    322. if (currentQuestion.type == 'radio') {
    323. currentQuestion.selected = val === undefined ? '': val;
    324. } else if (currentQuestion.type == 'checkbox') {
    325. currentQuestion.selected = val === undefined ? [] : val;
    326. } else {
    327. currentQuestion.value = val === undefined ? '' : val
    328. }
    329. }
    330. },
    331. mounted: function () {
    332. let _this = this;
    333. bus.$on('nextStep', function (msg) {
    334. let current = _this.currentOrder;
    335. if (++current >= _this.questions.length) {
    336. return;
    337. } else {
    338. _this.currentOrder += 1;
    339. }
    340. });
    341. bus.$on('backStep', function (msg) {
    342. let currentOrder = _this.currentOrder;
    343. if (--currentOrder < 0) {
    344. return;
    345. } else {
    346. _this.currentOrder -= 1;
    347. }
    348. });
    349. bus.$on('submit', function (msg) {
    350. console.log('提交')
    351. });
    352. bus.$on('reset', function (order) {
    353. _this.handleReset(order)
    354. });
    355. bus.$on('valueChange', function (order, val) {
    356. _this.handleReset(order, val)
    357. });
    358. }
    359. })
  • CSS

    1. [v-cloak]{
    2. display: none;
    3. }
    4. .btn {
    5. border: none;
    6. outline:none;
    7. color: white;
    8. padding: 7px 25px;
    9. text-align: center;
    10. text-decoration: none;
    11. display: inline-block;
    12. font-size: 16px;
    13. margin: 4px 2px;
    14. cursor: pointer;
    15. }
    16. .button-white {
    17. background-color: rgb(217, 219, 223);
    18. color: #000000;
    19. }
    20. .button-white:active {
    21. background-color: rgb(134, 149, 179);
    22. color: #000000;
    23. }
    24. .button-primary {
    25. background-color: rgb(39, 126, 228);
    26. border: none;
    27. color: white;
    28. padding: 7px 25px;
    29. text-align: center;
    30. text-decoration: none;
    31. display: inline-block;
    32. font-size: 16px;
    33. margin: 4px 2px;
    34. cursor: pointer;
    35. }
    36. .button-primary:active {
    37. background-color: rgb(185, 171, 31);
    38. }
    39. .button-primary:disabled {
    40. background-color: rgb(115, 122, 130);
    41. }

5. 升级

  • 组件重复内容很多,可合并

调查问卷WebApp的更多相关文章

  1. "琳琅满屋"调查问卷 心得体会及结果分析

    ·关于心得体会       当时小组提出这个校园二手交易市场的时候,就确定了对象范围,仅仅是面向在校大学生,而且在我们之前就已经有了很多成功的商品交易的例子可以让我们去借鉴,再加上我们或多或少的有过网 ...

  2. JavasScript实现调查问卷插件

    原文:JavasScript实现调查问卷插件 鄙人屌丝程序猿一枚,闲来无事,想尝试攻城师是感觉,于是乎搞了点小玩意.用js实现调查问卷,实现了常规的题型,单选,多选,排序,填空,矩阵等. 遂开源贴出来 ...

  3. 关于“Durian”调查问卷的心得体会

    这周我们做了项目着手前的客户需求调查,主要以调查问卷的方式进行.其实做问卷调查并不是想象中的那么简单,首先要确定问卷调查的内容,每一个问题都要经过深思熟虑,字字斟酌,既要切合问卷主要目的,又要简洁扼要 ...

  4. 从Adobe调查问卷看原型设计工具大战

    近年国内外原型设计工具新品频出,除了拥趸众多的老牌Axure在RP 8之后没有什么大的动作,大家都拼了命地在出新品.今天 inVision 的 Craft 出了 2.0 的预告视频,明天 Adobe ...

  5. Scrum立会报告+燃尽图(十一月十七日总第二十五次):设计调查问卷;修复上一阶段bug

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2284 项目地址:https://git.coding.net/zhang ...

  6. <问吧>调查问卷心得体会

    <问吧>调查问卷心得与体会 在这之前,我们已经组成了一个六个人的小团队---“走廊奔跑队”,我们这次做的这个项目的名称是:问吧.在项目实施之前,我们必做的一步就是需求分析,目的就是充分了解 ...

  7. android 实现调查问卷-单选-多选

    非常久没写东西了.今天来总结下有关android调查问卷的需求实现. 转载请加地址:http://blog.csdn.net/jing110fei/article/details/46618229 先 ...

  8. 自动化测试调查问卷送《QTP自动化测试最佳实践》

    自动化测试调查问卷送<QTP自动化测试最佳实践> http://automationqa.com/forum.php?mod=viewthread&tid=2308&fro ...

  9. HDU - 6344 2018百度之星资格赛 1001调查问卷(状压dp)

    调查问卷  Accepts: 1289  Submissions: 5642  Time Limit: 6500/6000 MS (Java/Others)  Memory Limit: 262144 ...

随机推荐

  1. PAT 甲级 1028 List Sorting (25 分)(排序,简单题)

    1028 List Sorting (25 分)   Excel can sort records according to any column. Now you are supposed to i ...

  2. Delphi实现树型结构具体实例

    unit Unit1;interfaceuses  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, ...

  3. .Netcore 2.0 Ocelot Api网关教程(7)- 限流

    本文介绍Ocelot中的限流,限流允许Api网关控制一段时间内特定api的总访问次数.限流的使用非常简单,只需要添加配置即可. 1.添加限流 修改 configuration.json 配置文件,对  ...

  4. Intellij IDEA 激活到2100年教程

    1.先看效果   2.下载 JetbrainsCrack-4.2-release-enc.jar,提取码:9os6.并将下载后的文件复制到$ {InstallDir} / bin,$ {Install ...

  5. 【miscellaneous】使用Google语音识别引擎(Google Speech API)[3月5日修改]

    原文:http://blog.csdn.net/dlangu0393/article/details/7214728#comments 近期重写本文,暂时禁止评论. 最近在使用Qt编写一个客户端程序的 ...

  6. 【Qt开发】【ARM-Linux开发】 QT在ARM上显示字体的问题

    在PC机上利用QT开发的应用程序在设置字体时,在PC上运行,可根据自己的设置,字体随之变大或变小.而移植到ARM上运行时发现,显示字体与所设置的字体不用,字体普遍偏小.经过上网搜索发现,是环境变量字库 ...

  7. 三角箭头 css实现

    效果图: 1.html 代码 <div>较昨日 <span class="dot-up"></span> 20%</div> < ...

  8. 编写一个自定义事件类,包含on/off/emit/once方法

    function Event() { this._events = {}; } Event.prototype.on = function(type, fn) { if (!this._events[ ...

  9. kafka2.10集群搭建(一)

    一.kafka集群搭建 1.上传解压 2.配置文件的配置 1.修改 server.properties文件 broker.id=11 #192.168.199.11 #21 一般使用ip后三位 lis ...

  10. 使用jbc查询数据封装成对象的工具类

    适用于获取Connection对象的util package com.briup.myDataSource; import java.io.FileReader; import java.io.Inp ...