基于Metronic的Bootstrap開發框架經驗總結(17)-- 使用 summernote插件實現HTML文檔的編輯和圖片插入操作

在很多場合,我們需要在線編輯HTML內容,然后在頁面上或者其他終端上(如小程序、APP應用等)顯示,編輯HTML內容的插件有很多,本篇介紹基于Bootstrap的 summernote插件實現HTML文檔的編輯和圖片插入操作,這個控件的使用非常方便,并且用戶群也很大。
Summernote 是一個簡單靈活,所見即所得(WYSIWYG)的編輯器,Summernote是一個輕量級、靈活基于Bootstrap和jQuery的HTML文本編輯器,擁有強大的API配置功能,多國語言支持支持Bootstrap2.x和3.0,支持視頻和圖片上傳以及代碼的高亮顯示,多種后臺語言版本示例以及多主體支持,在瀏覽器兼容方面,支持IE9+以及現代的瀏覽器Chrome,Firefox,Safari等,在移動端應該有不錯的表現,還是很不錯的,值得使用。
該插件是開源的,官網地址:http://summernote.org/,GitHub地址:https://github.com/summernote/summernote/、安裝調整都很方便。

1、Summernote的簡單使用

使用方法:
1)、加載JS和CSS

<!-- include libraries(jQuery, bootstrap) -->
<link  rel="stylesheet">
<script src="http://cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.js"></script> 
<script src="http://netdna.bootstrapcdn.com/bootstrap/3.3.5/js/bootstrap.js"></script> 

<!-- include summernote css/js-->
<link  rel="stylesheet">
<script src="http://cdnjs.cloudflare.com/ajax/libs/summernote/0.8.7/summernote.js"></script>

基于MVC的Asp.net應用程序中,我們一般在BundleConfig.cs里面初始化我們的引入JS文件,如下所示。

            //添加對bootstrap-summernote的支持
            css_metronic.Include("~/Content/metronic/assets/global/plugins/bootstrap-summernote/summernote.css");
            js_metronic.Include("~/Content/metronic/assets/global/plugins/bootstrap-summernote/summernote.min.js");
            js_metronic.Include("~/Content/metronic/assets/global/plugins/bootstrap-summernote/lang/summernote-zh-CN.min.js");
            bundles.Add(css_metronic);
            bundles.Add(js_metronic);

2)、編寫HTML內容

在HTML頁面代碼增加下面的標記。

<div id="summernote">Hello Summernote</div>

3)、初始化調用

簡單的初始化代碼如下所示。

$(document).ready(function() {
  $('#summernote').summernote();
});

如果增加控件高度的定義,則如下所示。

$('#summernote').summernote({
  height: 300,   
  focus: true    
});

而詳細的初始化界面,一般還包括語言國際化、圖片上傳等處理代碼,如下所示

function initEditor() {
    $('#Note').summernote({
        lang: 'zh-CN',       // default: 'en-US'
        height: 600,         // set editor height                
        minHeight: null,    // set minimum height of editor
        maxHeight: null,    // set maximum height of editor
        focus: true,         // set focus to editable area after initializing summe
        callbacks: {
            onImageUpload: function (files) { //the onImageUpload API  
                img = sendFile(files[0]);
            }
        }
    });
}

4)、獲取內容

var markupStr = $('#summernote').summernote('code');

5)、設置內容

設置內容和獲取內容類似,在code后面增加需要寫入的HTML內容即可。

var markupStr = 'hello world';
$('#summernote').summernote('code', markupStr);

2、 summernote插件在實際項目中的使用

上面介紹了常規 summernote插件的使用過程,一般情況下,我們使用它除了編輯HTML內容外,還需要對圖片上傳進行處理,以確保我們可以直接把文件上傳到后臺文件系統里面去,而不是把它們轉換為Base64的編碼放在頁面內。

一般的初始化代碼如下所示。

function initEditor() {
    $('#Note').summernote({
        lang: 'zh-CN',       // default: 'en-US'
        height: 600,         // set editor height                
        minHeight: null,    // set minimum height of editor
        maxHeight: null,    // set maximum height of editor
        focus: true,         // set focus to editable area after initializing summe
        callbacks: {
            onImageUpload: function (files) { //the onImageUpload API  
                img = sendFile(files[0]);
            }
        }
    });
}

其中的sendFile的函數如下所示,主要是調用附件管理模塊進行文件的存儲。

//提交文件到服務器處理
function sendFile(file) {
    data = new FormData();
    data.append("file", file);
    //增加額外的參數
    data.append("folder", '商品信息');
    data.append("guid", $("#ID").val());

    $.ajax({
        data: data,
        type: "POST",
        url: "/FileUpload/Upload",
        cache: false,
        contentType: false,
        processData: false,
        success: function (json) {
            var data = $.parseJSON(json);
            var url = data.urls[0];
            $("#Note").summernote('insertImage', url, 'image name'); // the insertImage API  
        }
    });
}

一般來說,我們的文件為了在多個應用之間共享處理,一般建議以FTP方式進行處理,這方面可以參考隨筆《在附件管理模塊中增加對FTP 上傳和預覽的支持》,FTP上傳文件可以使用FluentFTP這個組件(GitHub地址:https://github.com/hgupta9/FluentFTP ),這個是一個應用很廣,功能很強大的FTP組件。
上傳附件到服務器上的后臺控制器代碼如下所示,主要是構建文件信息,上傳后返回對應的URL,然后就可以在 summernote插件上顯示圖片了。

/// <summary>
/// 上傳附件到服務器上
/// </summary>
/// <param name="fileData">附件信息</param>
/// <param name="guid">附件組GUID</param>
/// <param name="folder">指定的上傳目錄</param>
/// <returns></returns>
[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Upload(string guid, string folder)
{
    //如果需要修改字段顯示,則參考下面代碼處理
    dynamic obj = new ExpandoObject();
    List<string> urls = new List<string>();
    var result = new CommonResult();

    HttpFileCollectionBase files = HttpContext.Request.Files;
    if (files != null)
    {
        foreach (string key in files.Keys)
        {
            try
            {
                #region MyRegion
                HttpPostedFileBase fileData = files[key];
                if (fileData != null)
                {
                    HttpContext.Request.ContentEncoding = Encoding.GetEncoding("UTF-8");
                    HttpContext.Response.ContentEncoding = Encoding.GetEncoding("UTF-8");
                    HttpContext.Response.Charset = "UTF-8";

                    string fileName = Path.GetFileName(fileData.FileName);      //原始文件名稱
                    string fileExtension = Path.GetExtension(fileName);         //文件擴展名

                    FileUploadInfo info = new FileUploadInfo();
                    info.FileData = ReadFileBytes(fileData);
                    if (info.FileData != null)
                    {
                        info.FileSize = info.FileData.Length;
                    }
                    info.Category = folder;
                    info.FileName = fileName;
                    info.FileExtend = fileExtension;
                    info.AttachmentGUID = guid;

                    info.AddTime = DateTime.Now;
                    info.Editor = CurrentUser.Name;//登錄人
                    //info.Owner_ID = OwerId;//所屬主表記錄ID

                    result = BLLFactory<FileUpload>.Instance.Upload(info);
                    if (!result.Success)
                    {
                        LogTextHelper.Error("上傳文件失敗:" + result.ErrorMessage);
                    }
                    else
                    {
                        //返回具體路徑地址
                        string serverRealPath = info.BasePath.UriCombine(info.SavePath);
                        if (!Path.IsPathRooted(info.BasePath) &&
                            !info.BasePath.StartsWith("http://") &&
                            !info.BasePath.StartsWith("https://"))
                        {
                            //如果是相對目錄,加上當前程序的目錄才能定位文件地址
                            var url = HttpContext.Request.Url;
                            var baseurl = url.AbsoluteUri.Replace(url.PathAndQuery, "");
                            serverRealPath = baseurl.UriCombine(serverRealPath).Replace('\\', '/');
                        }
                        urls.Add(serverRealPath);
                    }
                }
                #endregion
            }
            catch (Exception ex)
            {
                result.ErrorMessage = ex.Message;
                LogTextHelper.Error(ex);
            }
        }
        obj.urls = urls;
    }
    else
    {
        result.ErrorMessage = "fileData對象為空";
    }

    var newResult = new { Success = result.Success, ErrorMessage = result.ErrorMessage, urls = urls };
    return ToJsonContent(newResult);
}

界面效果如我在隨筆介紹的一樣《微信小程序結合后臺數據管理實現商品數據的動態展示、維護》。
案例對商品的詳細信息的富文本進行編輯,來進行圖片文字的編輯處理,如下界面所示。


當我們插入圖片的時候,彈出一個對話框界面,選擇文件上傳后返回URL顯示在SummerNote插件上。

上傳文件成功后,通過下面的代碼插入一個圖片,如下代碼。

$.ajax({
    data: data,
    type: "POST",
    url: "/FileUpload/Upload",
    cache: false,
    contentType: false,
    processData: false,
    success: function (json) {
        var data = $.parseJSON(json);
        var url = data.urls[0];
        $("#Note").summernote('insertImage', url, 'image name'); // the insertImage API  
    }
});

以上就是我在實際的Bootstrap框架項目中使用 summernote插件實現HTML文檔的編輯和圖片插入操作,整體性還是比較容易上手的,經驗供大家借鑒。
其他相關隨筆可以參考閱讀下:
結合bootstrap fileinput插件和Bootstrap-table表格插件,實現文件上傳、預覽、提交的導入Excel數據操作流程
基于Metronic的Bootstrap開發框架經驗總結(16)-- 使用插件bootstrap-table實現表格記錄的查詢、分頁、排序等處理
基于Metronic的Bootstrap開發框架經驗總結(15)-- 更新使用Metronic 4.75版本
從開發框架提高開發效率說起
基于Metronic的Bootstrap開發框架經驗總結(14)--條碼和二維碼的生成及打印處理
基于Metronic的Bootstrap開發框架經驗總結(13)--頁面鏈接收藏夾功能的實現2
基于Metronic的Bootstrap開發框架經驗總結(12)--頁面鏈接收藏夾功能的實現
基于Metronic的Bootstrap開發框架經驗總結(11)--頁面菜單的幾種呈現方式
基于Metronic的Bootstrap開發框架經驗總結(10)--優化Bootstrap圖標管理
在MVC控制器里面使用dynamic和ExpandoObject,實現數據轉義的輸出
基于Metronic的Bootstrap開發框架經驗總結(9)--實現Web頁面內容的打印預覽和保存操作
基于Metronic的Bootstrap開發框架經驗總結(8)--框架功能總體界面介紹
基于Metronic的Bootstrap開發框架經驗總結(7)--數據的導入、導出及附件的查看處理
基于Metronic的Bootstrap開發框架經驗總結(6)--對話框及提示框的處理和優化
基于Metronic的Bootstrap開發框架經驗總結(5)--Bootstrap文件上傳插件File Input的使用
基于Metronic的Bootstrap開發框架經驗總結(4)--Bootstrap圖標的提取和利用
基于Metronic的Bootstrap開發框架經驗總結(3)--下拉列表Select2插件的使用
基于Metronic的Bootstrap開發框架經驗總結(2)--列表分頁處理和插件JSTree的使用
基于Metronic的Bootstrap開發框架經驗總結(1)-框架總覽及菜單模塊的處理

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,565評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,115評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 177,577評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,514評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,234評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,621評論 1 326
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,641評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,822評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,380評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,128評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,319評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,879評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,548評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,970評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,229評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,048評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,285評論 2 376

推薦閱讀更多精彩內容