NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name The GM_config Unit Test // @namespace sizzlemctwizzle // @description Attempts to test every aspect of GM_config. // @license MIT // @version 2.3.0.4 // @grant GM_getValue // @grant GM_setValue // @grant GM.getValue // @grant GM.setValue // @require https://openuserjs.org/src/libs/sizzle/GM_config.min.js // @include https://openuserjs.org/scripts/sizzle/The_GM_config_Unit_Test // @match https://openuserjs.org/scripts/sizzle/The_GM_config_Unit_Test // @homepageURL https://github.com/sizzlemctwizzle/UserScripts/blob/master/gm_config_unit_test.user.js // @updateURL https://openuserjs.org/meta/sizzle/The_GM_config_Unit_Test.meta.js // ==/UserScript== // ==OpenUserJS== // @author sizzle // @collaborator Marti // ==/OpenUserJS== /* jshint esversion: 8 */ /* globals GM_config, GM_configStruct */ var fieldDefs = { 'name': { 'section': [GM_config.create('Personal Info About Yourself'), 'We need this info to do stuff'], 'label': GM_config.create('Name'), 'type': 'text', 'default': 'Joe Simmons' }, 'birthday': { 'label': 'Date of Birth', 'type': 'date', 'format': 'dd/mm/yyyy' }, 'age': { 'label': 'Age', 'type': 'unsigned int', 'default': 19 }, 'gender': { 'options': ['Male', 'Female'], 'label': 'Gender', 'type': 'radio', 'default': 'Male' }, 'income': { 'labelPos': 'right', 'label': 'Income', 'type': 'float', 'default': 50000.0 }, 'status': { 'label': 'Married', 'labelPos': 'above', 'type': 'checkbox', 'default': false }, 'work': { 'label': 'Job', 'type': 'select', 'labelPos': 'below', 'options': ['Carpenter', 'Truck Driver', 'Porn Star'], 'default': 'Truck Driver' }, 'bunchOtext': { 'label': 'Bunch of Text', 'type': 'textarea', 'default': 'I actually did\'t realize we had this field until recently...' }, 'magic': { 'label': 'Magic Button', 'type': 'button', 'click': function() { alert('Magic works!'); } }, 'upperLeft': { 'section': 'Check some boxes', 'type': 'checkbox', 'default': false }, 'upperLeftMiddle': { 'type': 'checkbox', 'default': false }, 'upperRightMiddle': { 'type': 'checkbox', 'default': false }, 'upperRight': { 'type': 'checkbox', 'default': false }, 'middleLeft': { 'section': [], 'type': 'checkbox', 'default': false }, 'middleLeftMiddle': { 'type': 'checkbox', 'default': false }, 'middleRightMiddle': { 'type': 'checkbox', 'default': false }, 'middleRight': { 'type': 'checkbox', 'default': false }, 'bottomLeft': { 'section': [], 'type': 'checkbox', 'default': false }, 'bottomLeftMiddle': { 'type': 'checkbox', 'default': false }, 'bottomRightMiddle': { 'type': 'checkbox', 'default': false }, 'bottomRight': { 'type': 'checkbox', 'default': false }, 'labelLess': { 'section': GM_config.create('New Section'), 'type': 'text', 'default': 'This value is not saved.', 'save': false }, 'alertTextField': { 'label': 'Alert Text', 'type': 'button', 'click': function() { var value = GM_config.fields['labelLess'].toValue(); if (value != null) { alert(value); GM_config.fields['labelLess'].value = 'Value changed.'; GM_config.fields['labelLess'].reload(); } else alert('null field has already been changed and forgotten.'); } }, 'customCSS': { 'label': 'Enter CSS', 'type': 'text', 'save': false, 'default': '' }, 'validCSS': { 'type': 'hidden', 'default': '', } }; GM_config.init( { id: 'GM_config', title: 'Configurable Options Script', fields: fieldDefs, css: '#GM_config_section_1 .config_var, #GM_config_section_2 .config_var, #GM_config_section_3 .config_var { margin: 5% !important;display: inline !important; }', events: { open: function(doc) { doc.getElementById('GM_config_section_header_1').className = 'field_label'; var customCSS = GM_config.fields['customCSS'].node; var validCSS = GM_config.fields['validCSS'].node; customCSS.value = validCSS.value; customCSS.addEventListener('change', function () { if(/\w+\s*\{\s*\w+\s*:\s*\w+[\s|\S]*\}/.test(customCSS.value)) validCSS.value = customCSS.value; }, false); }, save: function(values) { // All the values that aren't saved are passed to this function // for (i in values) alert(values[i]); } }, 'types': { 'date': { 'default': null, toNode: function(configId) { var field = this.settings, value = this.value, id = this.id, create = this.create, format = (field.format || 'mm/dd/yyyy').split('/'), slash = null, retNode = create('div', { className: 'config_var', id: configId + '_' + id + '_var', title: field.title || '' }); // Save the format array to the field object so // it's easier to hack externally this.format = format; // Create the field lable retNode.appendChild(create('label', { innerHTML: field.label, id: configId + '_' + id + '_field_label', for: configId + '_field_' + id, className: 'field_label' })); // Create the inputs for each part of the date value = value ? value.split('/') : this['default']; for (var i = 0, len = format.length; i < len; ++i) { var props = { id: configId + '_field_' + id + '_' + format[i], type: 'text', size: format[i].length, value: value ? value[i] : '', onkeydown: function(e) { var input = e.target; if (input.value.length >= input.size) input.value = input.value.substr(0, input.size - 1); } }; // Jump to the next input once one is complete // This saves the user a little work if (i < format.length - 1) { slash = create(' / '); props.onkeyup = function(e) { var input = e.target, inputs = input.parentNode.getElementsByTagName('input'), num = 0; for (; num < inputs.length && input != inputs[num]; ++num); if (input.value.length >= input.size) inputs[num + 1].focus(); }; } else slash = null; // Actually create and append the input element retNode.appendChild(create('input', props)); if (slash) retNode.appendChild(slash); } return retNode; }, toValue: function() { var rval = null; if (this.wrapper) { var inputs = this.wrapper.getElementsByTagName('input'); rval = ''; // Join the field values together seperated by slashes for (var i = 0, len = inputs.length; i < len; ++i) { // Don't save values that aren't numbers if (isNaN(Number(inputs[i].value))) { alert('Date is invalid'); return null; } rval += inputs[i].value + (i < len - 1 ? '/' : ''); } } // We are just returning a string to be saved // If you want to use this value you'll want a Date object return rval; }, reset: function() { // Empty all the input fields if (this.wrapper) { var inputs = this.wrapper.getElementsByTagName('input'); for (var i = 0, len = inputs.length; i < len; ++i) inputs[i].value = ''; } } } } }); (async () => { // Retrieve language setting var lang = await GM_config.getValue('lang', 'en'); // Fields in different languages var langDefs = { 'en': // Fields in English { 'lang': { 'label': 'Choose Language', 'type': 'select', 'options': ['en', 'de'], 'save': false // This field's value will NOT be saved } }, 'de': // Fields in German { 'lang': { 'label': 'Sprache wählen', 'type': 'select', 'options': ['en', 'de'], 'save': false // This field's value will NOT be saved } } }; // Use field definitions for the stored language var fields = langDefs[lang]; // The title for the settings panel in different languages var titles = { 'en': 'Translations Dialog', 'de': 'Übersetzungen Dialog' }; var title = titles[lang]; // Translations for the buttons and reset link var saveButton = {'en': 'Save', 'de': 'Speichern'}; var closeButton = {'en': 'Close', 'de': 'Schließen'}; var resetLink = { 'en': 'Reset fields to default values', 'de': 'Felder zurücksetzen auf Standardwerte' }; var gmc_trans = new GM_configStruct( { 'id': 'GM_config_trans', // The id used for this instance of GM_config 'title': title, 'fields': fields, // Fields object 'events': { 'init': function() { // You must manually set an unsaved value this.fields['lang'].value = lang; }, 'open': function (doc) { // translate the buttons var config = this; doc.getElementById(config.id + '_saveBtn').textContent = saveButton[lang]; doc.getElementById(config.id + '_closeBtn').textContent = closeButton[lang]; doc.getElementById(config.id + '_resetLink').textContent = resetLink[lang]; }, 'save': function(values) { // All unsaved values are passed to save for (var i in values) { if (i == 'lang' && values[i] != lang) { var config = this; lang = values[i]; // Use field definitions for the chosen language fields = langDefs[lang]; config.fields['lang'].value = lang; // Use the title for the chose language title = titles[lang]; // Re-initialize GM_config for the language change config.init({ 'id': config.id, title: title, 'fields': fields }); // Refresh the config panel for the new language change config.close(); config.open(); // Save the chosen language for next time config.setValue('lang', lang); } } } } }); GM_config.init( { id: 'GM_config', 'events': { 'init': function() { GM_config.open(); } }, fields: { 'extra': { 'label': 'Extra Field', 'type': 'text', 'default': 'This field was added with a second call to init()' }, 'openTrans': { 'label': 'Open Translation Demo', 'type': 'button', 'click': function() { GM_config.close(); gmc_trans.open(); } } } }); })();