<!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"
}