Firefox扩展插件开发笔记(二) - 编写内嵌js脚本代码

google连不上,我的翻译插件不能用了,找了很久又没有找到其他的替代品,就像着自己能不能做一个利用baudi API的翻版,另外考虑到一些加强功能,可以对选中的文本进行wiki查询并显示wiki的结果。于是开始开发准备。明确下想法,首先打开网页时,将自己写的一个脚本(content.js)嵌入到网页中,该脚本产生一个隐藏的按钮和面板,并监听鼠标的抬起事件,当抬起并有选中文本的时候,将悬浮按钮显示,点击按钮时,获取选中的文本内容,发送翻译请求到baidu 翻译或者请求wiki页面,回传,将内容显示在面板中,搞定!

导航目录

  1. Javascript选取文本
  2. 构建Add-on代码,注入脚本到页面中

我自己做程序的时候都喜欢先把各个环节具体的做法确定下来,特别是一些关键的需要考虑的地方,这个扩展主要有以下几个考虑的地方:1.如何利用javascript获取选中文本(之前真不懂)2.如何发送翻译请求 3.如何搞定面板的显示布局。一步步来,这里先解决第一个问题

js如何选取文本

网上查了一下解决方案,在Firefox, Google Chrome, Safari, Opera中,可以用 window.getSelection(),而在在IE下,可以用 document.selection.createRange().text,我们只需要前者。

    //获取选择文本内容
    function getSelecctTxt() {
        console.log(window.getSelection());
        return window.getSelection() .toString();
    }

判断是否有选取的文本,其中的my-ad-btn就是我在js中生成的隐藏按钮div的id

    $('body') .mouseup(setHandler);
    //判断是否选取文字,处理鼠标抬起事件,显示按钮
    function setHandler(ev) {
        $('#my-add-btn') .css(
            'display','none'
        );

        ev = ev || window.event;
        if (getSelecctTxt() == '' || ev.which != 1) {
            isShow = false;
            $('.content-div').hide();
            return ;
        }
        alert("up");
        $('#my-add-btn') .css({
            top: ev.pageY + 9,
            left: ev.pageX + 9,
            display: 'inline-block',
            lineHeight: '18px'
        });
        //alert(getSelectTxt());
    }

接着再在监听按扭事件,本来想发请求到media-wiki获取wiki的json数据然后显示的,结果看了一晚上的英文API使用手册,最后终于明白如何获得对特定单词的wiki解释数据,但一看数据还需要写代码解析wiki语法规则,表示太懒了于是暴力内嵌了一个iframe直接加载对应解释的网页。
代码中wikiframe是显示wiki页面的iframe的id,trans-div是显示翻译结果的div(以后获得了的翻译结果就放在这个div里显示)的id。准备把这两个结果页面做成选项卡的样子,可以切换查看翻译结果或者wiki页面,这里先注释掉。

    //绑定点击按钮事件
    $('#my-add-btn') .click(function (e) {
        $('#my-add-btn') .css({
            display: 'none'
        });

        //$('#wikiframe').attr('src','https://en.wikipedia.org/w/index.php?useformat=mobile&title=firefox');
        //$('.content-div') .show();
        //isShow = true;
        //$('#wikiframe').css('height','0px');
        var str = getSelecctTxt();
        $('#trans-div').text(getSelecctTxt());
          console.log("selecttext:"+getSelecctTxt());
        return false;
    });

另外当用户选择的是一个词的时候将翻译成详细的词典解释,也就是需要发送请求到百度词典API,句子的话发送请求百度翻译API,这里使用正则表达式进行判断:

    //绑定点击按钮事件
    $('#my-add-btn') .click(function (e) {
        $('#my-add-btn') .css({
            display: 'none'
        });

        //$('#wikiframe').attr('src','https://en.wikipedia.org/w/index.php?useformat=mobile&title=firefox');
        //$('.content-div') .show();
        //isShow = true;
        //$('#wikiframe').css('height','0px');
        var str = getSelecctTxt();
        if(-1==(str.search(/^\s?([a-zA-Z-]{1,20}|[\u4E00-\u9FA5]{1,4})\s?$/i))){
            alert(str+" =不是单词");
        }else{
           alert(str+"=单词");
        }
        $('#trans-div').text(getSelecctTxt());
          console.log("selecttext:"+getSelecctTxt());
        return false;
    });

代码再整合一下,变成下面的样子(请忽视那一大块乱码,那是图片的base64编码):

var imageurl = 'url("")';

/*判断脚本是否在frame中执行*/

if (window != top || (self.frameElement && self.frameElement.tagName == 'IFRAME')) {
    $('html,body').scrollTop(80);
    //alert('scroll');
    return ;
 }
 
 //获取选择文本内容
 function getSelecctTxt() {
    var ll = window.getSelection().toString();
    //console.err("df");
    //alert('uu');
    return window.getSelection().toString();
 }

 //添加悬浮按钮与相应CSS
 function showAddBtn() {
    $('body') .append('<a id="my-add-btn" name="my-add-btn" href="#my-add-btn"></a>');
    $('#my-add-btn') .css({
            display: 'none',
            height: '18px',
            width: '18px',
            position: 'absolute',
            zIndex: '2147483647',
            textAlign: 'center',
            lineHeight: '18px',
            textDecoration: 'none',
            font: 'bold 12px/25px Arial, sans-serif',
            color: 'rgba(62, 87, 6, 0.53)',
            background: '-moz-linear-gradient(center top , rgba(165, 205, 78, 1) 0%, rgba(107, 143, 26, 1) 100%)',
            backgroundImage: imageurl,
            backgroundSize: '18px 18px',
            opacity: '0.55',
            textShadow: '1px 1px 1px rgba(255,255,255, .22)',
            borderRadius: '18px',
            boxShadow: '1px 1px 1px rgba(0,0,0, .2), inset 1px 1px 1px rgba(255,255,255, .3)',
            transition: 'all 0.15s ease'
    });
    $('#my-add-btn') .hover(function () {
        $(this) .css({
                color: 'rgba(62, 87, 6, 1)',
                opacity: '1'
        });
        }, function () {
           $(this) .css({
                    color: 'rgba(62, 87, 6, 0.53)',
                    opacity: '0.55'
            });
        }
     );
        //如果点击在按钮上,阻止mouseup事件传播

     $('#my-add-btn') .mouseup(function (e) {
        return false
     });
  }

    /**************弹出框***************/
   //TO-DO

    showAddBtn();
    //绑定点击按钮事件
    $('#my-add-btn') .click(function (e) {
        $('#my-add-btn') .css({
            display: 'none'
        });
        var str = getSelecctTxt();
        if(-1==(str.search(/^@([a-zA-Z\u4E00-\u9FA5]{3,10}):/i))){
            alert(str+" =不是单词");
        }else{
           alert(str+"=单词");
        }
        $('#trans-div').text(getSelecctTxt());
        return false;
    });

    //判断是否选取文字,处理鼠标抬起事件,显示按钮
    function setHandler(ev) {
        $('#my-add-btn') .css(
            'display','none'
        );

        ev = ev || window.event;
        if (getSelecctTxt() == '' || ev.which != 1) {

            isShow = false;
            $('.content-div').hide();
            return ;
        }
        $('#my-add-btn') .css({
            top: ev.pageY + 9,
            left: ev.pageX + 9,
            display: 'inline-block',
            lineHeight: '18px'
        });
    }
    $('body') .mouseup(setHandler);
    $('body').mousedown(function(e){
        if(isShow) {
            if($(e.target).is('.content-div')||$(e.target).parents('.content-div').length > 0) return;
            isShow = false;
            $('.content-div').hide();
        }
    });

好了现在来构建add-on程序代码测试一下上面的的代码,懒一点的话可以把上面代码用类似油猴子的插件运行测试一下,注意需要导入jQuery哟
参考selectTextUserJs.js

嵌入content script到页面中

首先使用cfx工具初始化一下开发模板(如何运行CFX):

cfx init

这时打开lib文件夹下面lib文件的main.js,在这里写主要add-on代码,所以在这里用tabs模块将写好的javascript脚本注入到页面中,注意如果想要注入.css文件到页面中,可以使用pageMod模块:

var tabs = require("sdk/tabs"),
data = require("sdk/self").data;

//当标签页加载完成,调用tab.attach注入脚本文件
tabs.on("ready", runScript);
//装载contentScript
function runScript(tab) {
    var worker = tab.attach({
        contentScriptFile: [data.url("jquery-1.11.2.min.js"), data.url("content.js")],
    });
}

将之前的写好的脚本和jquery文件保存到add-on项目目录下的data文件夹下,使用cfx工具运行:

cfx run

如果提示没有找到firefox二进制文件,使用-b参数指定firefo.exe文件路径,不出意外的话,现在在打开的浏览器里选择一些文本就会出现一个小按钮,点击之后会显示选择的文本信息

标签: firefox, 扩展插件, add-on, content script

添加新评论