Thursday, September 22, 2011

Script Options


// ==UserScript==
// @id             e267d0cb-8ee1-45ad-a9d3-f1254e4be0e6
// @name           Script Options
// @namespace      Takato
// @author         Takato
// @description    Settings manager for my scripts. A direct copy of PhasmaExMachina's script.
// @version        2011.09.23.4
// @exclude        *
// ==/UserScript==

Config = {
 data:null,
 callback:null,
 tempOptions:{},
 footerHtml:'<span style="font-size:.9em;">Note: You may need to refresh the page to see changes.</span>',
 reloadOnSave:false,
 init:function(settings) {
  Config.settings = settings;
 },
 preloadData:function() {
  Config.data = {};
  Config.settings = typeof(Config.tabs) != 'undefined' ? Config.tabs : Config.settings; 
  for(var tabName in Config.settings) {
   if(typeof(Config.settings[tabName].fields) == "object") {
    var fields = Config.settings[tabName].fields
    for(var fieldName in fields) {
     Config.data[fieldName] = Config.get(fieldName);
     
    }
   }
  }
 },
 close:function() {
  document.body.removeChild(Config.$('ConfigBodyWrapper'));
  document.body.removeChild(Config.$('ConfigMask'));
  window.removeEventListener('keyup', Config.keyUpHandler, true);
  if(typeof(Config.callback) == 'function')
   Config.callback();
 },
 show:function(callback) {
  Config.tempOptions = {};
  Config.settings = typeof(Config.settings) != 'undefined' ? Config.settings : Config.tabs;
  Config.callback = typeof(callback) == 'function' ? callback : null;
  if(typeof(Config.styleDrawn) == 'undefined') {    // apply styling
   GM_addStyle("\
     #ConfigMask { position:absolute; width:100%; top:0; left:0; height:100%; background-color:#000; opacity:.7; z-index:9000; } \
     #ConfigBody * { border:none; font-size:12px; color:#333; font-weight:normal !important; margin:0 !important; padding:0 !important; background:none; text-decoration:none; font-family:Helvetica Neue,Arial,Helvetica,sans-serif; line-height:1.2em; } \
     #ConfigBody { width:600px; margin:auto !important; top:30px; position:fixed; left:35%; text-align:left; background:#f9f9f9; border:1px outset #333; padding:0 !important; font-family:Arial; font-size:14px; -moz-border-radius:5px; cursor:default; z-index:9010; color:#333; padding-bottom:1em !important; } \
     #ConfigBody a { text-decoration:underline; color:#000099 !important; } \
     #ConfigBody strong, #ConfigContentBox strong { font-weight:bold !important; } \
     #ConfigBody h1 { font-size:13px; font-weight:bold !important; padding:.5em !important; border-bottom:1px solid #333; background-color:#999; margin-bottom:.75em !important; } \
     #ConfigBody h2 { font-weight:bold; margin:.5em 1em !important; } \
     #ConfigBody h1 { font-size:13px; font-weight:bold; color:#fff; text-decoration:none; } \
     #ConfigBody h1 a:hover { text-decoration:underline; } \
     #ConfigBody li { list-style-type:circle; } \
     #ConfigBody p { font-size:12px; font-weight:normal; margin-bottom:1em !important; } \
     #ConfigContentPadding { margin:0 1em !important; }\
     #ConfigTabs { margin-top:20px !important; }\
     #ConfigTabs span { border:1px solid #666; -moz-border-radius:5px 5px 0 0; padding: 2px 10px !important; position:relative; top:-2px; background-color:#ddd; cursor:pointer; }\
     #ConfigTabs span:hover { background-color:#eee; }\
     #ConfigTabs span.active { background-color:#F9F9F9; top:-1px; border-bottom:none; padding-top:3px !important; font-weight:bold; cursor:inherit; }\
     #ConfigTabs span.active:hover { background-color:#F9F9F9; }\
     #ConfigContentBox { border:1px inset #666; padding:1.5em 1em 1em !important; max-height:300px; overflow:auto; }\
     #ConfigContentBox table { width:auto !important; }\
     #ConfigContentBox td { font-weight:normal; }\
     #ConfigContentBox input { border:1px inset #666 !important; }\
     #ConfigContentBox td.fieldLabel { text-align:right !important; padding-right:.5em !important; font-weight:bold !important; }\
     #ConfigContentBox td select { border:1px inset #666; }\
     #ConfigHistory { margin:0 1em 1em 1em !important; max-height:150px; overflow-y:auto; border:1px inset #999; padding:0 1em 1em !important; width:448px; } \
     #ConfigHistory ul { margin-left:2em !important; } \
     #ConfigClose { float:right; cursor:pointer; height:14px; opacity:.5; } \
     #ConfigClose:hover { opacity:.9; } \
     #ConfigFooter { padding:1.5em 1em 0 !important; } \
     #ConfigFooter input { border:1px outset #666; padding:3px 5px 5px 20px !important; background:no-repeat 4px center #eee; -moz-border-radius:3px; cursor:pointer; width:80px; float:right; margin-left:.5em !important; } \
     #ConfigFooter input:hover { background-color:#f9f9f9; } \
     #ConfigFooter select { border:1px inset #666; }\
     #ConfigContentBox #ConfigFieldTable { width:auto !important; margin-left:2em !important; }\
     #ConfigContentBox #ConfigFieldTable td { padding-bottom:.5em !important; }"
    );
   if(typeof(Config.css) != 'undefined')     // apply user specified styles if set
    GM_addStyle(Config.css);
   Config.styleDrawn = true;
  }
  // declare and apply config background mask
  var noticeBg = document.createElement('div');       
  noticeBg.id = "ConfigMask";
  noticeBg.style.height = (unsafeWindow.scrollMaxY + unsafeWindow.innerHeight) + 'px';
  document.body.appendChild(noticeBg);
  // declare and apply config window
  var noticeWrapper = document.createElement('div');
   noticeWrapper.setAttribute('style', 'position:absolute; width:100%; top:0; left:0; z-index:9010; max-width:auto; min-width:auto; max-height:auto; min-height:auto;');
   noticeWrapper.id = "ConfigBodyWrapper";
    var notice = document.createElement('div');
    notice.id = "ConfigBody";
    var html = '<h1>\
        <img src="' + Config.icons.config + '" align="absmiddle" style="margin-top:-2px;"/>\
        ' + (typeof(Config.scriptName) == 'string' ? Config.scriptName + ' - ' : '') + ' Settings\
       </h1>\
       <div id="ConfigContentPadding">\
        <div id="ConfigTabs">';
    // draw tabls
    var i = 0;
    var firstTabId = "";
    for(var label in Config.settings) {
     var id = 'configTab' + label.replace(/\s/g, '_');
     html += '<span id="' + id + '">' + label + '</span>';
     firstTabId = i == 0 ? id : firstTabId;
     i++;
    }
    html += '</div><div id="ConfigContentBox">';   
    html += '</div>';   
    html += '</div>';   
    html += '<div id="ConfigFooter">\
       <input type="button" id="ConfigCancelButton" value="Cancel" style="background-image:url(' + Config.icons.close + ')"/>\
       <input type="button" id="ConfigCloseButton" value="Save" style="background-image:url(' + Config.icons.save + ')"/>\
       ' + Config.footerHtml + '\
      </div>';
    notice.innerHTML = html;;
   noticeWrapper.appendChild(notice);
  document.body.appendChild(noticeWrapper);
  // add tab change listeners
  for(var label in Config.settings) {
   var id = 'configTab' + label.replace(/\s/g, '_');
   Config.$(id).addEventListener('click', function() { Config.activateTab(this.id); }, false);
  }
  // add escape key press and other listener
  Config.activateTab(firstTabId);
  window.addEventListener('keyup', Config.keyUpHandler, true);
  Config.$('ConfigCloseButton').addEventListener('click', function() {
   Config.save();
   Config.close();
   if(Config.reloadOnSave)
    document.location = '';
  }, true);
  Config.$('ConfigCancelButton').addEventListener('click', Config.close, true);
 },
 //-------------------------------- "private" methods -----------------------------------------
 activateTab:function(id) {
  // deactivate current tab
  var elems = Config.$('ConfigTabs').getElementsByTagName('span');
  for(var i = 0; i < elems.length; i++) {
   elems[i].className = ''; 
  }
  // set current tab
  Config.$(id).className = 'active';
  var key = id.replace(/^configTab/, '').replace(/_/g, ' ');
  var fields = Config.settings[key].fields;
  var html = typeof(Config.settings[key].html) == 'string' ? Config.settings[key].html : '';
  html += '<table cellpadding="0" cellspacing="0" border="0" id="ConfigFieldTable">';
  for(var fieldName in fields) {
   var field = fields[fieldName];
   var type = typeof(field.type) != 'string' ? 'html' : field.type;
   var tip = typeof(field.tip) == 'string' ? field.tip : '';
   if(type != 'html')
    html += '<tr title="' + tip + '"><td colspan="' + (type == 'html' ? '2' : '1') + '" class="fieldLabel">' + 
     (typeof(field.label) == 'string' ? field.label : '') + '</td><td>';
   else
    html += '<tr>';
   switch(type) {
    case 'select':
     html += '<select id="configInput_' + fieldName + '">';
     if(typeof(field.options) == 'undefined')
      alert('Options Error: ' + fieldName + ' of type "select" is missing the "options" property');
     else {
      for(var text in field.options) {
       var val = field.options[text];
       html += '<option value="' + val + '"' + (Config.get(fieldName) == val ? ' selected' : '') + '>' + text + ' &nbsp;</option>';
      }
     }
     html += '</select>';
     break;
    case 'password':
    case 'text':
     var width = typeof(fields[fieldName].width) != 'undefined' ? (fields[fieldName].width.toString().match(/px/) ? fields[fieldName].width : fields[fieldName].width + 'px') : false;
     html += '<input id="configInput_' + fieldName + '" value="' + Config.get(fieldName) + '" style="' + (width ? 'width:' + width + ';' : '') + '" type="' + type + '"/>';
     break;
    case 'checkbox':
     html += '<input id="configInput_' + fieldName + '" type="checkbox" style="position:relative; top:2px;"' + (Config.get(fieldName) ? 'checked' : '' ) + '/>';
     break;
    case 'html':
     html += '<td colspan="2">' + field.value + '</td>';
     break;
   }
   if(type != 'html') {
    html += typeof(fields[fieldName].text) == 'string' ? ' &nbsp; - ' + fields[fieldName].text : '';
    html += '</td></tr>';
   } else
    html += '</tr>';
  }
  html += '</table>';
  // add check for updates
  if(id == "configTabAbout" && typeof(ScriptUpdater) == 'object' && typeof(ScriptUpdater.scriptId) != 'undefined') {
   html += '<p><br/><a href="javascript:void(0)" id="ConfigCheckUpdatesLink">Check for updates</a></p>';
  }
  Config.$('ConfigContentBox').innerHTML = html;
  // add event listeners
  for(var fieldName in fields) {
   switch(fields[fieldName].type) {
    case 'checkbox': 
     Config.$('configInput_' + fieldName).addEventListener('change', function() {
      Config.tempOptions[this.id.toString().match(/configInput_(.+)$/)[1]] = this.checked ? '1' : '0';                  
     }, false);
     break;
    case 'select': 
     Config.$('configInput_' + fieldName).addEventListener('change', function() {
      Config.tempOptions[this.id.toString().match(/configInput_(.+)$/)[1]] = this.value;                  
     }, false);
     break;
    case 'password':
    case 'text':
     Config.$('configInput_' + fieldName).addEventListener('keyup', function() {
      Config.tempOptions[this.id.toString().match(/configInput_(.+)$/)[1]] = this.value;                  
     }, false);
     break;
   }
  }
  if(id == "configTabAbout" && typeof(ScriptUpdater) == 'object' && typeof(ScriptUpdater.scriptId) != 'undefined') {
   $('#ConfigCheckUpdatesLink')[0].addEventListener('click', function() { ScriptUpdater.forceNotice(ScriptUpdater.scriptId, ScriptUpdater.scriptCurrentVersion); }, false);
  }
 },
 keyUpHandler:function (e) {
  if(e.keyCode == 27) { Config.close(); }
 },
 icons:{
  install:"%3D",
  config:"",
  close:"%3D%3D",
  uso:"%3D%3D",
  save:"%3D",
 },
 getField:function(key) {
  Config.settings = typeof(Config.tabs) != 'undefined' ? Config.tabs : Config.settings;
  for(var tabName in Config.settings) {
   if(typeof(Config.settings[tabName].fields) == "object") {
    var fields = Config.settings[tabName].fields
    for(var fieldName in fields)
     if(fieldName == key)
      return fields[fieldName];
   }
  }
  return false;
 },
 get:function(key) {
  if(Config.data != null && typeof(Config.data[key]) != 'undefined')
   return Config.data[key];
  else {
   var field = Config.getField(key);
   key = typeof(Config.prefix) == 'string' ? Config.prefix + key : key; 
   switch(field.type) {
    case 'checkbox':
     if(typeof(field.value) == 'undefined' || !field.value)
      return (typeof(GM_getValue(key)) != 'undefined' && GM_getValue(key).toString() == "1") ? true : false; // false by default
     else 
      return (typeof(GM_getValue(key)) != 'undefined' && GM_getValue(key).toString() == "0") ? false : true; // true by default
     break;
    case 'select':
    case 'password':
    case 'text':
     return typeof(GM_getValue(key)) == 'undefined' ? (typeof(field.value) != 'undefined' ?  field.value : '') : GM_getValue(key);
     break;
    default: 
     return 'unfound';
     return typeof(GM_getValue(key)) == 'undefined' ? false : GM_getValue(key);
   } 
  }
 },
 save:function() { 
  for(var x in Config.tempOptions)
   Config.set(x, Config.tempOptions[x]);
  Config.tempOptions = {};
 },
 set:function(key, value) {
  key = typeof(Config.prefix) == 'string' ? Config.prefix + key : key; 
  GM_setValue(key, value);
 },
 $:function(id) {
  return document.getElementById(id);
 },
};

0 comments:

Post a Comment