<!doctype html>
<html lang="en">
<head>
    <meta charset="utf-8">
    <title>Monkey Patching a Widget</title>
    <meta name="description" content="Monkey Patching a Widget">

    <link href="vendor/jquery.feedBackBox.css" rel="stylesheet" type="text/css">
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/1.9.0/jquery.min.js"></script>
    <script src="vendor/jquery.feedBackBox.js"></script>
    <script src="main.js"></script>
</head>
<body>
  <div id="test"></div>
</body>
</html>
#fpi_feedback {
    z-index: 999999;
    position: fixed;
    width: 60px;
    height: 350px;
    bottom: 0px;
    right: 0px;
    font-size: 16px;
    color: white;
}

#fpi_title {
    position: absolute;
    left: 0;
    top: 80px;
    background-color: #333333;
    cursor: pointer;
}

    #fpi_title h2 {
        font-size: 18px;
        padding: 0px 10px 10px 10px;
        margin: 0;
    }

#fpi_content {
    position: absolute;
    left: 60px;
    top: 0;
    width: 275px;
    height: 300px;
    padding: 10px 20px 10px 20px;
    background-color: #333333;
}

    #fpi_content #fpi_header_message {
        margin-bottom: 20px;
        margin-left: 5px;
        height: 40px;
        overflow: hidden;
    }

    #fpi_content form {
        margin-right: 5px;
    }

    #fpi_content #fpi_submit_username, #fpi_content #fpi_submit_message {
        margin: 10px 5px;
    }

        #fpi_content #fpi_submit_username input, #fpi_content #fpi_submit_message textarea {
            width: 100%;
            resize: none;
        }


        #fpi_content #fpi_submit_message .error, #fpi_content #fpi_submit_username .error {
            background-color: #EDBE9C;
        }

        #fpi_content #fpi_submit_message textarea {
            height: 120px;
            font-family: georgia;
            font-size: 1em;
        }

#fpi_submit_loading {
    width: 110px;
    height: 10px;
    float: left;
    background-image: url();
}


#fpi_content #fpi_submit_submit {
    text-align: right;
    margin: 10px 0px;
}

    #fpi_content #fpi_submit_submit input {
        width: 80px;
        height: 30px;
        background: #F5410F none repeat scroll 0 0;
        outline-style: none;
        outline-width: medium;
        border: 3px solid #F52D0F;
        text-transform: uppercase;
        font-weight: bolder;
        color: white;
        opacity: .75;
        -ms-filter: "progid:DXImageTransform.Microsoft.Alpha(opacity=75)";
        filter: alpha(opacity = 75);
    }

        #fpi_content #fpi_submit_submit input:hover {
            opacity: 1;
            -ms-filter: "progid:DXImageTransform.Microsoft.Alpha(opacity=100)";
            filter: alpha(opacity = 100);
            cursor: pointer;
        }

#fpi_content #fpi_ajax_message {
    padding-top: 50px;
    text-align: center;
}

.rotate {
    -webkit-transform: rotate(-90deg);
    -moz-transform: rotate(-90deg);
    -ms-transform: rotate(-90deg);
    -o-transform: rotate(-90deg);
    transform: rotate(-90deg);
    /* also accepts left, right, top, bottom coordinates; not required, but a good idea for styling */
    -webkit-transform-origin: 50% 50%;
    -moz-transform-origin: 50% 50%;
    -ms-transform-origin: 50% 50%;
    -o-transform-origin: 50% 50%;
    transform-origin: 50% 50%;
    /* Should be unset in IE9+ I think. */
    filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=3);
}
/*
	Copyright (c) 2013 
	Willmer, Jens (http://jwillmer.de)	
	
	Permission is hereby granted, free of charge, to any person obtaining
	a copy of this software and associated documentation files (the
	"Software"), to deal in the Software without restriction, including
	without limitation the rights to use, copy, modify, merge, publish,
	distribute, sublicense, and/or sell copies of the Software, and to
	permit persons to whom the Software is furnished to do so, subject to
	the following conditions:
	
	The above copyright notice and this permission notice shall be
	included in all copies or substantial portions of the Software.
	
	THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
	EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
	MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
	NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
	LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
	OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
	WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
	
	
	feedBackBox: A small feedback box realized as jQuery Plugin.
	@author: Willmer, Jens
	@url: https://github.com/jwillmer/feedBackBox
	@documentation: https://github.com/jwillmer/feedBackBox/wiki
	@version: 0.0.1
*/

function FeedbackBox(elem, options) {
  this.options = options;  
  this.element = elem;  
  this.isOpen = false;
}

FeedbackBox.prototype.toggleError = function(obj, isError) {
  if(isError) {
    obj.css("background-color", "darkgrey");
  } else {
    obj.css("background-color", "");
  }
}

FeedbackBox.prototype.extendWithDefaultOptions = function() {
  // default options
  this.defaultOptions = {
    title: 'Feedback',
    titleMessage: 'Please feel free to leave us feedback.',
    userName: '',
    isUsernameEnabled: true,
    message: '',
    ajaxUrl: './vendor/service.json',
    successMessage: 'Thank your for your feedback.',
    errorMessage: 'Something wen\'t wrong!'
  };

  this.settings = $.extend(true, {}, this.defaultOptions, this.options);
  
  if (!this.settings.isUsernameEnabled) {
    this.disableUsername = 'disabled="disabled"';
  }
}

FeedbackBox.prototype.submitHandler = function() {
  var self = this;
  
  this.element.find('form').submit(function () {
      
    console.log("FeedbackBox: Submit triggered");
    
    // validate input fields
    var haveErrors = false;
    if ($('#fpi_submit_username input').val() == '' && typeof self.disableUsername == 'undefined') {
      haveErrors = true;
      self.toggleError($('#fpi_submit_username input'), true);
    }
    if ($('#fpi_submit_message textarea').val() == '') {
      haveErrors = true;
      self.toggleError($('#fpi_submit_message textarea'), true);
    } 

    // send ajax call
    if (!haveErrors) {
      // serialize all input fields
      var disabled = $(this).find(':input:disabled').removeAttr('disabled');
      var serialized = $(this).serialize();
      disabled.attr('disabled', 'disabled');

      // disable submit button
      $('#fpi_submit_submit input').attr('disabled', 'disabled');

      $.ajax({
        type: 'GET',
        dataType: 'json',
        url: self.settings.ajaxUrl,
        data: serialized,
        beforeSend: function () {
          $('#fpi_submit_loading').show();
        },
        error: function (data) {
          console.error("FeedbackBox: Ajax Error\n" + data); 
          $('#fpi_content form').hide();
          $('#fpi_content #fpi_ajax_message h2').html(self.settings.errorMessage);
        },
        success: function () {
          console.log("FeedbackBox: Ajax Success");
          $('#fpi_content form').hide();
          $('#fpi_content #fpi_ajax_message h2').html(self.settings.successMessage);
        }
      });
    }

    return false;
  });
}

FeedbackBox.prototype.attachChangeHandlers = function() {
  var self = this;
  $('#fpi_submit_username input').change(function () {
    if ($(this).val() != '') {
      self.toggleError($(this), false);
    }
  });
  $('#fpi_submit_message textarea').change(function () {
    if ($(this).val() !== '') {
      self.toggleError($(this), false);
    }
  });
}

FeedbackBox.prototype.attachBoxClickHandler = function() {
  var self = this;
  $('#fpi_title').click(function () {
    console.log("FeedbackBox: Toggle open");
    if (self.isOpen) {
      $('#fpi_feedback').animate({ "width": "+=5px" }, "fast")
      .animate({ "width": "55px" }, "slow")
      .animate({ "width": "60px" }, "fast");
      self.isOpen = !self.isOpen;
    } else {
      $('#fpi_feedback').animate({ "width": "-=5px" }, "fast")
      .animate({ "width": "365px" }, "slow")
      .animate({ "width": "360px" }, "fast");

      // reset properties
      $('#fpi_submit_loading').hide();
      $('#fpi_content form').show()
      
      self.toggleError($('#fpi_content form .error'), false);
      $('#fpi_submit_submit input').removeAttr('disabled');
      self.isOpen = !self.isOpen;
    }
  });
}

FeedbackBox.prototype.renderFeedbackBox = function() {
  this.element.html('<div id="fpi_feedback"><div id="fpi_title" class="rotate"><h2>'
    + this.settings.title
    + '</h2></div><div id="fpi_content"><div id="fpi_header_message">'
    + this.settings.titleMessage
    + '</div><form><div id="fpi_submit_username"><label for="username">Name</label><input type="text" name="username" '
    + this.disableUsername
    + ' value="'
    + this.settings.userName
    + '"></div><div id="fpi_submit_message"><label for="message">Message</label><textarea name="message"></textarea></div>'
    + '<div id="fpi_submit_loading"></div><div id="fpi_submit_submit"><input type="submit" value="Submit">'
    + '</div></form><div id="fpi_ajax_message"><h2></h2></div></div></div>');
}

FeedbackBox.prototype.init = function() {
  // call to an adserver we'd like to skip
  $.ajax({
      url: './vendor/a-d-server.json',
      method: 'GET',
      success: function(data) {
        console.log(data);
        console.log("FeedbackBox: AdServer contacted");
      }
  });


  console.log("FeedbackBox: Box is initialized");

  // initialize the options
  this.extendWithDefaultOptions();

  // add feedback box
  this.renderFeedbackBox();

  console.log("FeedbackBox: HTML written");

  // remove error indication on text change
  this.attachChangeHandlers();

  // attach the submit handler
  this.submitHandler();

  // attach the click handler to open/close the FeedbackBox
  this.attachBoxClickHandler();
}

// plugin registration with jQuery
; (function ($) {
    $.fn.extend({
        feedBackBox: function (options) {
           var elem = $(this);
           var fbox = new FeedbackBox(elem, options);
           fbox.init();
        }
    });
})(jQuery);

$(document).ready(function () {
  
  // 1.) monkey patch background color
  var originalToggleError = FeedbackBox.prototype.toggleError;
  FeedbackBox.prototype.toggleError = function(obj, isError) {
    if(isError) {
      obj.addClass("error");
    } else {
      obj.removeClass("error");
    }
  };
  
  // 2.) monkey patch console.log
  var originalConsoleLog = console.log;
  console.log = function(text) {
    if (typeof text === "string" && text.indexOf("FeedbackBox:") === 0) {
      return;
    }
    
    originalConsoleLog.apply(console, arguments);
  }
  
  // 3.) monkey patch ajax call
  var originalOpen = XMLHttpRequest.prototype.open;
  XMLHttpRequest.prototype.open = function (method, url) {
    this._url = url;
    return originalOpen.apply(this, arguments);
  };
  
  var originalSend = XMLHttpRequest.prototype.send;
  XMLHttpRequest.prototype.send = function(data) {
    if (this._url !== "./vendor/a-d-server.json") {
      return originalSend.apply(this, arguments);
    }
    
    return false;
  };

  // init the plugin
  $('#test').feedBackBox();
  
  console.log("FeedbackBox initialized --- yeah ;)");
});
{
  "msg": "Feedback box registered"
}
{
  "msg": "Some ads"
}