/* NEW BSD LICENSE {{{ Copyright (c) 2010, anekos. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. The names of the authors may not be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ################################################################################### # http://sourceforge.jp/projects/opensource/wiki/licenses%2Fnew_BSD_license # # に参考になる日本語訳がありますが、有効なのは上記英文となります。 # ################################################################################### }}} */ // PLUGIN_INFO {{{ let PLUGIN_INFO = extension manager アドオン管理 extension manager アドオンを管理します。 1.1.0 anekos new BSD License (Please read the source code comments of this plugin) 修正BSDライセンス (ソースコードのコメントを参照してください) http://svn.coderepos.org/share/lang/javascript/vimperator-plugins/trunk/extensions-manager.js 2.3 2.3 ; // }}} // INFO {{{ let INFO = <> anekos New BSD License

Extention manager death yohooo

:extstate :extstate sub-command name extension-names...

Store or restore current extensions state with name.

The following sub-commands are interpreted.

if the extension-names arguments are specified, this command is only for these extensions.
  • store
  • restore

anekos New BSD License

extention manager

:extstate :extstate sub-command name extension-names...

拡張の有効無効状態を name で保存復帰します。 extension-names で拡張名を指定しておくと、指定された拡張のみが保存復帰の対象になります。

以下の sub-commands があります。

store
保存
restore
復帰

:extbisect :extbisect sub-command

問題のある拡張などをあぶり出すためのコマンドです。 二分探索ぽい方法よって、効率的に困った拡張を探し出します。 自動的に拡張の有効無効を切り替えていくので、かなり楽が出来ると思います。

作業手順。 (ok fail start を実行すると自動的に再起動します)

  1. ":extbisect start" で開始
  2. 問題が起きていないかテスト
  3. 起きていなければ":extbisect ok"、起きていれば":extbisect fail"を実行
  4. 再起動するので、2 を再び繰り返す。

問題のある拡張がなんであるか、確定したら ok/fail したときにメッセージが出ます。 メッセージを確認したら、":extbisect reset" で拡張の状態を元に戻してください。

テストを行うことで、エラーで Firefox が終了してしまう場合は、再起動後に ok / fail を実行してください。

; // }}} (function () { let states = storage.newMap('plugins-extman-states', {store: true}); let bisect = storage.newMap('plugins-extman-bisect', {store: true}); states.modify = bisect.modify = function (name, func) this.set(name, func(this.get(name))); function xabled (id, enable) services.get("extensionManager")[enable ? 'enableItem' : 'disableItem'](id); function xabledWithName (name, enable) xabled(liberator.getExtension(name), enable); function cmp (s1, s2) (s1.toLowerCase() == s2.toLowerCase()); function extFilter (exts, names) (names && names.length) ? exts.filter(function (ext) names.some(function (name) cmp(name, ext.name))) : exts; function store (name, targets) { let es = extFilter(liberator.extensions, targets); states.set(name, {extensions: es, date: new Date()}); } function slash (ary, index) [ary.slice(0, index), ary.slice(index)]; function isVimp (ext) (ext.name == 'Vimperator'); store('last'); let extState = { store: function (name, targets) { function done () liberator.echo('extensions state was stored to "' + name + '".'); if (!name) return liberator.echoerr('too few arguments'); if (states.get(name)) { commandline.input( 'overwrite? [y/n]', function (answer) (answer.toLowerCase() == 'y' ? (store(name, targets, true), done()) : liberator.echo('canceled')) ); } else { store(name, targets, true); done(); } }, restore: function (name, targets) { if (!name) return liberator.echoerr('too few arguments'); let state = states.get(name); if (!state) return liberator.echoerr('"' + name + '" has not been stored.'); let es = state.extensions; if (targets.length) es = es.filter(function (ext) targets.some(function (t) t == ext.name)); es.forEach(function (ext) xabled(ext.id, ext.enabled)); liberator.echo('extensions state was restored from "' + name + '".'); }, flush: function () { states.save(); }, __noSuchMethod__: function (name) { liberator.echoerr(name + ' is not valid sub-command'); } }; let extBisect = { start: function () { if (this.__notReady(false)) return; let targets = liberator.extensions.filter(function (ext) !isVimp(ext) && ext.enabled); bisect.set('store', liberator.extensions); bisect.set('state', 'started'); let ([a, b] = slash(targets, targets.length / 2)) { bisect.set('yet', a); bisect.set('current', b); } bisect.save(); this.__reflectCurrent(); liberator.restart(); }, ok: function () { if (this.__notReady(true)) return; if (this.__finished(true)) return; bisect.modify('yet', function (value) { let [a, b] = slash(value, value.length / 2); bisect.set('current', a); return b; }); this.__reflectCurrent(); liberator.restart(); }, fail: function () { if (this.__notReady(true)) return; if (this.__finished(false)) return; bisect.modify('current', function (value) { let [a, b] = slash(value, value.length / 2); bisect.set('yet', a); return b; }); this.__reflectCurrent(); liberator.restart(); }, show: function () { function f (ext) liberator.echo(' ' + ext.name); liberator.echo('<>'); bisect.get('current').forEach(f) liberator.echo('<>'); bisect.get('yet').forEach(f) }, reset: function () { if (this.__notReady(true)) return; bisect.set('state', ''); bisect.get('store').forEach(function (ext) xabled(ext.id, ext.enabled)); bisect.save(); liberator.echo('extensions were reset'); liberator.restart(); }, __notReady: function (started) { if (!!(bisect.get('state') == 'started') == !!started) return false; liberator.echoerr('extbisect has ' + (started ? 'not ' : '') + 'been started.'); return true; }, __finished: function (ok) { function answer (ext) { liberator.echo(util.escapeString(ext.name) + ' is the criminal!!'); return true; } let current = bisect.get('current'); let yet = bisect.get('yet'); if (ok && yet.length <= 1) return answer(yet[0]); if (!ok && current.length <= 1) return answer(current[0]); return false; }, __reflectCurrent: function () { let current = bisect.get('current'); liberator.extensions.forEach( function (ext) (xabled(ext.id, isVimp(ext) || current.some(function (e) e.id == ext.id))) ); }, __noSuchMethod__: function (name) { liberator.echoerr(name + ' is not valid sub-command'); } }; commands.addUserCommand( ['exts[tate]'], 'store / restore extensions state (enabled / disabled).', function (args) { let [cmd,] = args; extState[cmd](args[1], args.slice(2)); }, { completer: function (context, args) { if (args.length == 1) { context.title = ['sub command', 'description']; context.completions = [ ['store', 'store current state'], ['restore', 'restore current state'], ['flush', 'flush'] ]; } else if (args.length == 2) { context.title = ['name']; context.completions = [[name, state.date] for ([name, state] in states)]; } else if (args.length >= 3) { let exts = liberator.extensions; context.title = ['name']; context.quote = ['', util.escapeString, '']; context.completions = [[ext.name, ext.description] for each (ext in exts)]; } } }, true ); commands.addUserCommand( ['extbisect'], 'bisectrrrrrrrrrrr', function (args) { let [cmd,] = args; extBisect[cmd](); }, { completer: function (context, args) { if (args.length == 1) { context.title = ['sub command', 'description']; context.completions = [ ['start', 'start bisect'], ['ok', ''], ['fail', ''], ['reset', ''], ['show', ''] ]; } } }, true ); states.save(); })(); // vim:sw=2 ts=2 et si fdm=marker: 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336
/*
 * ==VimperatorPlugin==
 * @name            youtubeamp.js
 * @description     this script gives you keyboard oprations for YouTube.com.
 * @description-ja  YouTube のプレーヤーをキーボードで操作できるようにする。
 * @author          janus_wel <janus_wel@fb3.so-net.ne.jp>
 * @version         0.12
 * @minversion      2.0pre 2008/10/16
 * ==/VimperatorPlugin==
 *
 * LICENSE
 *   New BSD License
 *
 * USAGE
 *   :ytinfo
 *     プレーヤーに関しての情報を表示する。今のところバージョンだけ。
 *   :ytpause
 *     再生 / 一時停止を切り替える。
 *   :ytmute
 *     音声あり / なしを切り替える。
 *   :ytsize
 *     最大化 / ノーマルを切り替える。動いてない。
 *   :ytseek [position]
 *     指定した場所にシークする。秒数で指定が可能。
 *     指定なしの場合一番最初にシークする。
 *   :ytseek! delta
 *     現在の位置から delta 分離れた所にシークする。秒数で指定が可能。
 *     マイナスを指定すると戻る。指定なしの場合変化しない。
 *   :ytvolume [volume]
 *     ボリュームを設定する。 0 ~ 100 が指定できる。
 *     指定なしの場合 100 にセットする。
 *   :ytvolume! delta
 *     ボリュームを現在の値から変更する。 -100 ~ +100 を指定可能。
 *     指定なしの場合変化しない。
 *
 * SEE ALSO
 *   http://code.google.com/apis/youtube/js_api_reference.html
 *
 * HISTORY
 *   2008/10/07 ver. 0.10   - initial written.
 * */

(function() {

Function.prototype.bind = function(object) {
    var __method = this;
    return function() {
        return __method.apply(object, arguments);
    };
};

// class definition
// YouTubePlayerController Class
function YouTubePlayerController() {
    this.initialize.apply(this, arguments);
}
YouTubePlayerController.prototype = {
    initialize: function() {
        this.fuller = this._changeToFull.bind(this);
    },

    constants: {
        VERSION: '0.12',

        CARDINAL_NUMBER: 10,

        YOUTUBE_DOMAIN: '.youtube.jp',
        YOUTUBE_URL:    '^http://[^.]+\\.youtube\\.com/',
        WATCH_URL:      'http://[^.]+\\.youtube\\.com/watch',
        WATCH_PAGE:     1,

        PLAYER_NODE_ID: 'movie_player',

        STATE_PLAYING: 1,

        SIZE_WIDTH_DEFAULT:  480,
        SIZE_HEIGHT_DEFAULT: 385,

        NAME_PLAYER_VERSION: 'PLAYER_VERSION',

        SEEKTO_DEFAULT:   0,
        SEEKBY_DEFAULT:   0,
        VOLUMETO_DEFAULT: 100,
        VOLUMEBY_DEFAULT: 0,

        HIDE_NODES: [
            'old-masthead',
            'watch-vid-title',
            'watch-other-vids',
            'old-footer',
            'copyright',
            'watch-main-area',
            'watch-comments-stats',
            'watch-video-response',
            'chrome-promo',
            'watch-video-quality-setting',
        ],
    },

    getControllerVersion: function() { return this.constants.VERSION; },

    pagecheck: function() {
        if(this.getURL().match(this.constants.WATCH_URL)) return this.constants.WATCH_PAGE;
        throw new Error('current tab is not watch page on youtube.com');
    },

    getURL: function() { return liberator.modules.buffer.URL; },

    _player: function() {
        if(this.pagecheck() === this.constants.WATCH_PAGE) {
            let player = this._getElementById(this.constants.PLAYER_NODE_ID);
            if(! player) throw new Error('player is not found');

            return player;
        }
        return null;
    },

    togglePlay: function() {
        var p = this._player();
        (p.getPlayerState() !== this.constants.STATE_PLAYING)
            ? p.playVideo()
            : p.pauseVideo();
    },

    toggleMute: function() {
        var p = this._player();
        p.isMuted() ? p.unMute() : p.mute();
    },

    toggleSize: function() {
        var p = this._player();
        (p.width == this.constants.SIZE_WIDTH_DEFAULT && p.height == this.constants.SIZE_HEIGHT_DEFAULT)
            ? this._fullSize()
            : this._normalSize();
    },

    _changeToFull: function() {
        var p = this._player();
        setTimeout(function() {
            p.width = content.innerWidth;
            p.height = content.innerHeight;
        }, 0);
    },

    _getElementById: function(id) {
        var e = window.content.document.getElementById(id);
        if(!e) return null;

        return e.wrappedJSObject
            ? e.wrappedJSObject
            : e;
    },

    _fullSize: function() {
        var b = this._getElementById('baseDiv');
        this.defMargin = b.style.margin;
        this.defPadding = b.style.padding;
        this.defWidth = b.style.width;
        b.style.margin = 0;
        b.style.padding = 0;
        b.style.width = '100%';

        for(let i=0, max=this.constants.HIDE_NODES.length ; i<max ; ++i) {
            let h = this._getElementById(this.constants.HIDE_NODES[i]);
            if(h) { h.style.display = 'none'; }
        }

        this._changeToFull();

        window.addEventListener(
            'resize',
            this.fuller,
            false
        );
    },

    _normalSize: function() {
        var b = this._getElementById('baseDiv');
        b.style.margin  = this.defMargin;
        b.style.padding = this.defPadding;
        b.style.width   = this.defWidth;

        for(let i=0, max=this.constants.HIDE_NODES.length ; i<max ; ++i) {
            let h = this._getElementById(this.constants.HIDE_NODES[i]);
            if(h) { h.style.display = 'block'; }
        }

        var p = this._player();
        p.width = this.constants.SIZE_WIDTH_DEFAULT;
        p.height = this.constants.SIZE_HEIGHT_DEFAULT;

        window.removeEventListener(
            'resize',
            this.fuller,
            false
        );
    },

    seekTo: function(position) {
        if(position) {
            if(position.match(/^(\d+):(\d+)$/)) {
                position = parseInt(RegExp.$1, this.constants.CARDINAL_NUMBER) * 60
                    + parseInt(RegExp.$2, this.constants.CARDINAL_NUMBER);
            }
            if(isNaN(position)) throw new Error('assign unsigned number : seekTo()');
        }
        else position = this.constants.SEEKTO_DEFAULT;

        var p = this._player();
        p.seekTo(position);
    },

    seekBy: function(delta) {
        if(delta) {
            if(isNaN(delta)) throw new Error('assign signed number : seekBy()');
        }
        else delta = this.constants.SEEKBY_DEFAULT;

        var p = this._player();
        var position = p.getCurrentTime();
        position += parseInt(delta, this.constants.CARDINAL_NUMBER);

        p.seekTo(position);
    },

    volumeTo: function(volume) {
        if(volume) {
            if(isNaN(volume)) throw new Error('assign unsigned number : volumeTo()');
        }
        else volume = this.constants.VOLUMETO_DEFAULT;

        var p = this._player();
        p.setVolume(volume);
    },

    volumeBy: function(delta) {
        if(delta) {
            if(isNaN(delta)) throw new Error('assign signed number : volumeBy()');
        }
        else delta = this.constants.VOLUMEBY_DEFAULT;

        var p = this._player();
        var volume = p.getVolume();
        volume += parseInt(delta, this.constants.CARDINAL_NUMBER);

        p.setVolume(volume);
    },
};

// global object
var controller = new YouTubePlayerController();

// command register
liberator.modules.commands.addUserCommand(
    ['ytinfo'],
    'display player information',
    function() {
        try {
            let info = [
                'controller version : ' + controller.getControllerVersion(),
            ].join('\n');
            liberator.echo(info, liberator.modules.commandline.FORCE_MULTILINE);
        }
        catch(e) { liberator.echoerr(e); }
    },
    {}
);

liberator.modules.commands.addUserCommand(
    ['ytpause'],
    'toggle play / pause',
    function() {
        try      { controller.togglePlay(); }
        catch(e) { liberator.echoerr(e); }
    },
    {}
);

liberator.modules.commands.addUserCommand(
    ['ytmute'],
    'toggle mute',
    function() {
        try      { controller.toggleMute(); }
        catch(e) { liberator.echoerr(e); }
    },
    {}
);

liberator.modules.commands.addUserCommand(
    ['ytseek'],
    'controll seek bar',
    function(args, special) {
        try {
            let arg = (args.arguments.length > 1)
                ? args.arguments[0].toString()
                : args.string;
            special ? controller.seekBy(arg) : controller.seekTo(arg);
        }
        catch(e) { liberator.echoerr(e); }
    },
    {
        bang: true,
    }
);

liberator.modules.commands.addUserCommand(
    ['ytvolume'],
    'controll volume',
    function(args, special) {
        try {
            let arg = (args.arguments.length > 1)
                ? args.arguments[0].toString()
                : args.string;
            special ? controller.volumeBy(arg) : controller.volumeTo(arg);
        }
        catch(e) { liberator.echoerr(e); }
    },
    {
        bang: true,
    }
);

liberator.modules.commands.addUserCommand(
    ['ytsize'],
    'toggle video size',
    function() {
        try      { controller.toggleSize(); }
        catch(e) { liberator.echoerr(e); }
    },
    {}
);

})()

// vim: set sw=4 ts=4 et;