bootstrap-wysiwyg结合ajax实现图片上传实时刷新功能


Posted in Javascript onMay 27, 2016

最近由于项目需求,要实现一个前端文本编辑框,附带图片上传实时查看的功能。比较了网上的几款插件,首先是百度的UEitor,发现该框架过于庞大,一个小框架引入如此多的文件并不是我想看到的;其次是jQuery的easyUI,虽然个人版的是免费的,但是项目属于公司业务,似乎用商业版的框架并不妥。考虑到项目的前端主要就是在bootstrap的基础上构建起来的,最终选用了bootstrap-wysiwyg插件,它非常的精简,轻巧而且扩展性强。

引入bootstrap-wysiwyg并且实现文本编辑功能十分的便捷,但是,我注意到,图片上传是用fileapi实现的。对于大多数网站,虽然用FileApi实现无上传预览用户体验非常好,但是真正存入数据库的时候,我们还是希望能够存储图片的在服务器的静态路径,而并非字符串化的图片。简而言之,我们需要对bootstrap-wysiwyg(以下简称WY)做稍许改写。

首先我们来观察下页面上图片控件,其它的控件略过,查一下源码,很容易发现如下代码:

<div class="btn-group">
 <a class="btn" title="Insert picture (or just drag & drop)" id="pictureBtn">
 <i class="icon-picture"></i></a>
  <input type="file" data-role="magic-overlay" 
   data-target="#pictureBtn"
  data-edit="insertImage" />
</div>

做一下说明,data-role,data-target属性是bootstrap中预定义的事件,在这里我们可以理解为布局相关,不用考虑。关键点来了,第三个属性data-edit,bootstrap中并没有这一事件,观察bootstrap-wysiwyg.js,可以发现这样一些代码:

toolbar.find('input[type=file][data-' + options.commandRole + ']')
   .change( ...
   ...
commandRole : 'edit',

也就是说,该属性其实是为了方便选择器而实现的,相当于给图片按钮添加了监听器事件。

我们接着研究一下WY图片预览的实现,第一步,就像上面代码展示的一样,监听器捕捉到fileInput的change事件,做出响应,调用insertFiles函数

restoreSelection();
if (this.type === 'file' && this.files && this.files.length > 0) {
 insertFiles(this.files);
}
saveSelection();
his.value = '';

找到insertFiles函数

insertFiles = function (files) {
  editor.focus();
  $.each(files, function (idx, fileInfo) {
   if (/^image\//.test(fileInfo.type)) {
   $.when(readFileIntoDataUrl(fileInfo)).done(function (dataUrl) {
    execCommand('insertimage', dataUrl);
   }).fail(function (e) {
    options.fileUploadError("file-reader", e);
   });
   } else {
   options.fileUploadError("unsupported-file-type", fileInfo.type);
   }
  });
  }

我们注意到它使用了jQuery的$.Deferred()方法,先调用了一个readFileIntoDataUrl方法,成功之后通过自封装的execCommand方法实现将图片输出到文本框。该图片其实就是一个<img>标签,只不过src属性是用字符串表示的图片。所以我们要做的其实是在监听器触发之后,将文件上传,获得图片的src,再把链接交给之后的execCommand方法。

由于笔者对Deferred并不是特别熟悉,所以还是采用更为通常的callback模式

观察ajaxFileUpload的调用方式:

$.ajaxFileUpload({
  url : ...,
  secureurl : false,
  fileElementId : ...,
  dataType : "json",
  success : function(obj) {
   ...
  },
  error : function() {
   ...
  }
  });

有两个必选的属性,url和fileElementId,为了保持依赖的正确性,重写ajaxFileUpload是不可取的。但是由于WY的控件是监听器实现的,所以通过函数将参数传过去是不现实的,所以我们需要自己对输入框定义一些属性来达到目的。

在fileInput中添加一些属性

<input type="file" id="pictureInput" name="picture"
   data-role="magic-overlay" data-target="#pictureBtn"
   data-edit="insertImage" action="..." />

id 用作 fileElementId,name属性也是必须的,主要是为了后台取值指名,action是图片需要提交到的url

在bootstrap-wysiwyg.js中定义一个函数名为uploadFileToServer,函数格式如下:

var uploadFileToServer = function(id, action, callback) {
 $.ajaxFileUpload({
  url : action,
  secureurl : false,
  fileElementId : id,
  dataType : 'json',
  success : function(obj) {
  if (obj.status) {
   callback(obj.imgsrc);
  } else
   options.fileUploadError("server-internal-exception",
    obj.message);
  },
  error : function() {
  options.fileUploadErroe("upload-failure", "");
  }
 });

将insertFiles方法作改写如下:

insertFiles = function(files, id, action) {
  editor.focus();
  $.each(files, function(idx, fileInfo) {
  if (/^image\//.test(fileInfo.type)) {
   /*
   * $.when(readFileIntoDataUrl(fileInfo)).done(function(dataUrl) {
   * execCommand('insertimage', dataUrl); }).fail(function(e) {
   * options.fileUploadError("file-reader", e); });
   */
   uploadFileToServer(id, action, function(src) {
   execCommand('insertimage', src);
   });
  } else {
   options.fileUploadError("unsupported-file-type",
    fileInfo.type);
  }
  });

同时对监听器做出一定的修改,以便拿到必要的属性

toolbar.find('input[type=file][data-' + options.commandRole + ']')
  .change(
   function() {
    restoreSelection();
    if (this.type === 'file' && this.files
     && this.files.length > 0) {
     insertFiles(this.files, $(this).attr('id'),
      $(this).attr('action'));
    }
    saveSelection();
    this.value = '';
    });

主要是增加了两个参数位置。

如此,改写便完成了,注意,要确保正确执行,请在控件之前引用ajaxFileUpload.js.

如果大家还想深入学习,可以点击这里进行学习,再为大家附3个精彩的专题:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript 变量命名规则
Sep 23 Javascript
javascript Array数组对象的扩展函数代码
May 22 Javascript
jquery select动态加载选择(兼容各种浏览器)
Feb 01 Javascript
MyEclipse取消验证Js的两种方法
Nov 14 Javascript
Javascript中判断对象是否为空
Jun 10 Javascript
JavaScript中的cacheStorage使用详解
Jul 29 Javascript
基于jQuery的ajax方法封装
Jul 14 Javascript
关于js二维数组和多维数组的定义声明(详解)
Oct 02 Javascript
JavaScript定义函数_动力节点Java学院整理
Jun 27 Javascript
微信小程序input框中加入小图标的实现方法
Jun 19 Javascript
vue拖拽组件使用方法详解
Dec 01 Javascript
在vue中使用vant TreeSelect分类选择组件操作
Nov 02 Javascript
极力推荐一款小巧玲珑的可视化编辑器bootstrap-wysiwyg
May 27 #Javascript
所见即所得的富文本编辑器bootstrap-wysiwyg使用方法详解
May 27 #Javascript
非常酷炫的Bootstrap图片轮播动画
May 27 #Javascript
Bootstrap安装环境配置教程分享
May 27 #Javascript
Bootstrap布局方式详解
May 27 #Javascript
使用Bootstrap框架制作查询页面的界面实例代码
May 27 #Javascript
JS组件Bootstrap Table布局详解
May 27 #Javascript
You might like
PHP上传文件时文件过大$_FILES为空的解决方法
2013/11/26 PHP
smarty模板引擎之配置文件数据和保留数据
2015/03/30 PHP
yii框架结合charjs实现统计30天数据的方法
2020/04/04 PHP
js 获取Listbox选择的值的代码
2010/04/15 Javascript
javascript之querySelector和querySelectorAll使用说明
2011/10/09 Javascript
老生常谈 js中this的指向
2016/06/30 Javascript
NodeJS基础API搭建服务器详细过程记录
2017/04/01 NodeJs
jQuery插件FusionCharts绘制的2D双面积图效果示例【附demo源码】
2017/04/11 jQuery
详解node服务器中打开html文件的两种方法
2017/09/18 Javascript
vue v-for循环重复数据无法添加问题解决方法【加track-by='索引'】
2019/03/15 Javascript
Nuxt v-bind绑定img src不显示的解决
2019/12/05 Javascript
Python中subprocess模块用法实例详解
2015/05/20 Python
使用Python的PIL模块来进行图片对比
2016/02/18 Python
python 默认参数问题的陷阱
2016/02/29 Python
python发送邮件功能实现代码
2016/07/15 Python
Python两个内置函数 locals 和globals(学习笔记)
2016/08/28 Python
Python实现多线程抓取网页功能实例详解
2017/06/08 Python
python寻找list中最大值、最小值并返回其所在位置的方法
2018/06/27 Python
对numpy中的数组条件筛选功能详解
2018/07/02 Python
Python编程深度学习绘图库之matplotlib
2018/12/28 Python
Python3实现统计单词表中每个字母出现频率的方法示例
2019/01/28 Python
解决pycharm运行程序出现卡住scanning files to index索引的问题
2019/06/27 Python
Python实现随机取一个矩阵数组的某几行
2019/11/26 Python
PyQt5 closeEvent关闭事件退出提示框原理解析
2020/01/08 Python
Python函数必须先定义,后调用说明(函数调用函数例外)
2020/06/02 Python
Python中的None与 NULL(即空字符)的区别详解
2020/09/24 Python
如何使用scrapy中的ItemLoader提取数据
2020/09/30 Python
用 Django 开发一个 Python Web API的方法步骤
2020/12/03 Python
css3旋转木马_动力节点Java学院整理
2017/07/12 HTML / CSS
全球最大的在线旅游公司:Expedia
2017/11/16 全球购物
关于VPN
2012/06/10 面试题
学术会议邀请函范文
2014/01/22 职场文书
幼教求职信
2014/03/12 职场文书
消防宣传标语大全
2015/08/03 职场文书
《浅水洼里的小鱼》教学反思
2016/02/16 职场文书
Win11运行育碧游戏总是崩溃怎么办 win11玩育碧游戏出现性能崩溃的解决办法
2022/04/06 数码科技