ميدياويكي:Gadget-Numeral-converter-toggle.js

ملاحظة: بعد الحفظ، قد يلزمك إفراغ الكاش لرؤية التغييرات.

/**
 * Numeral converter toggle
 * forked from [[mw:MediaWiki:Gadget-Numerakri.js]]
 * maintainer حبيشان
 */
mw.loader.using(['mediawiki.cookie', 'mediawiki.util', 'mediawiki.user']).then(function () {
    'use strict';
    const settingkey = 'gadget-Numeral_converter',
        msgs = {
            'numconv-turn-on-label': 'أرقام مشرقية',
            'numconv-turn-on-tooltip': 'تحويل الأرقام إلى أرقام مشرقية',
            'numconv-turn-off-label': 'أرقام مغربية',
            'numconv-turn-off-tooltip': 'إبقاء الأرقام مغربية',
        }

    var isOn = mw.loader.getState('ext.gadget.Numeral_converter') === 'ready';
        const skipclass = 'mwgadget-numconv-skip',
        maps = [
        ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '%'],
        ['٠', '١', '٢', '٣', '٤', '٥', '٦', '٧', '٨', '٩', '٪']
        ],
        rejectedTags = ['input', 'textarea', 'style', 'script', 'pre'];

    var walker;


    /**
     * @param {HTMLElement|TextNode} node
     * @return {number} NodeFilter.FILTER_* constant
     */
    function filterNode(node) {
        if (node.nodeType === Node.TEXT_NODE) {
            return NodeFilter.FILTER_ACCEPT;
        }
        var n = node.nodeName && node.nodeName.toLowerCase();
        if ( rejectedTags.indexOf(n) > -1 ||
            // node.hasAttribute('contenteditable') ||
            $(node).hasClass(skipclass)
        ) {
            // Skip this element and skip its children
            return NodeFilter.FILTER_REJECT;
        }
        // Skip this element, but check its children
        return NodeFilter.FILTER_SKIP;
    }

    /**
     * @param {TextNode} node
     */
    function handleTextNode(node) {
        var original = node.nodeValue,
            changed = original,
            // matchers = getMatchers(currentType),
            i = 0;
        for (; i <= 10; i++) {
            if (isOn) {
                changed = changed.replaceAll(maps[0][i], maps[1][i]);
            } else {
                changed = changed.replaceAll(maps[1][i], maps[0][i]);
            }
        }
        if (original !== changed) {
            node.nodeValue = changed;
        }
    }

    // https://doc.wikimedia.org/mediawiki-core/master/js/#!/api/mw-method-requestIdleCallback
    function idleWalker(deadline) {
        var el;
        if (!walker) {
            return;
        }
        while (deadline.timeRemaining() > 0) {
            el = walker.nextNode();
            if (!el) {
                // Reached the end
                walker = null;
                return;
            }
            handleTextNode(el);
        }

        // The user may interact with the page. We pause so the browser can process
        // interaction. The text handler will continue after that.
        if (walker) {
            mw.requestIdleCallback(idleWalker);
        }
    }

    function startPageConversion() {
        if (isOn) {
            $('ol:lang(ar) li, ol.references, li.references').css('list-style-type', 'arabic-indic');
        } else {
            $('ol:lang(ar) li, ol.references, li.references').css('list-style-type', 'decimal');
        }
        // If a walker is already active, replace it.
        // If no walker is active yet, start it.
        if (!walker) {
            mw.requestIdleCallback(idleWalker);
        }
        walker = document.createTreeWalker(document.body, NodeFilter.SHOW_ALL, filterNode, false);
    }


    /**
     * Save a browser cookie for 30 days, or remove it.
     * @param {string|null} value
     */
    function saveType(value) {
        mw.requestIdleCallback(function () {
            if (mw.user.isAnon()) {
                mw.cookie.set(settingkey, value, { expires: 30 * 86400 , path: '/' });
            } else {
                new mw.Api().saveOption(settingkey, value);
                mw.user.options.set(settingkey, value);
            }
            mw.storage.session.set(settingkey, value);
        });
    }

    /**
     * @return {string}
     */
    function getStoredType() {
        var value = mw.user.isAnon() ? mw.cookie.get(settingkey, '0') : mw.user.options.get(settingkey) || '0';
        var svalue = mw.storage.session.get(settingkey);

        if (svalue && svalue !== value) {
            value = svalue;
            saveType(value);
        } else if (value && Number(value)>1 ) {
            value = value === '2'? '1': '0';
            saveType(value);
        }

        return value === '1';
    }

    function togglePortlets() {
        var labelSelector;
        switch (mw.config.get('skin')) {
            case 'vector':
            case 'vector-2022':
            case 'minerva':
                labelSelector = '#pt-numconvert span:not(:empty), #pt-numconvert-sticky-header span:not(:empty)';
                break;
            default:
                labelSelector = '#pt-numconvert a';
        }
        $(labelSelector).text(getMsg('label'));
        $('#pt-numconvert a, #pt-numconvert-sticky-header a')
            .attr('title', getMsg('tooltip'));
    }

    function setHtmlClass() {
        $(document.documentElement).toggleClass('numconv-on', isOn);
    }
    
    function toggleMode() {
        isOn = !isOn;
        saveType(isOn? '1': '0');
        togglePortlets();
        setHtmlClass()
        startPageConversion();
        //if (isOn) {
        //    mw.loader.load('ext.gadget.Numeral_converter');
        // } else {
        //     startPageConversion();
        // }
    }

    function getMsg(suffix) {
        var key = 'numconv-turn-' + (isOn ? 'off' : 'on') + '-' + suffix;
        return msgs[key];
    }

    function init() {

        // Decide selected type
        isOn = getStoredType();
        setHtmlClass();
        if (isOn && mw.user.isAnon()) {
            // mw.loader.load('ext.gadget.Numeral_converter');
            startPageConversion();
        }
        $(mw.util.addPortletLink(
            mw.user.isAnon() ? 'p-user-menu-anon-editor' : 'p-personal',
            '#',
            getMsg('label'),
            'pt-numconvert',
            getMsg('tooltip'),
            null,
            mw.user.isAnon() ?
                '#pt-createaccount' :
                '#pt-preferences'
        ))
		.children().on('click', function (e) {
			e.preventDefault();
			toggleMode();
		});
    };

    $(function () {
        mw.requestIdleCallback(init);
    });
});