Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with HTTPS or Subversion.

Download ZIP
Browse files

change domains

  • Loading branch information...
commit 743c7e92296e26b09d623540a52016e7128d9f10 1 parent 3173c1f
Moritz Tolxdorff authored
1  README.md
View
@@ -0,0 +1 @@
+## Hangout-Lower-Third
1  TOS.txt
View
@@ -0,0 +1 @@
+coming soon
268 html5slider.js
View
@@ -0,0 +1,268 @@
+/*
+html5slider - a JS implementation of <input type=range> for Firefox 4 and up
+https://github.com/fryn/html5slider
+
+Copyright (c) 2010-2011 Frank Yan, <http://frankyan.com>
+
+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.
+*/
+
+(function() {
+
+// test for native support
+var test = document.createElement('input');
+try {
+ test.type = 'range';
+ if (test.type == 'range')
+ return;
+} catch (e) {
+ return;
+}
+
+// test for required property support
+if (!document.mozSetImageElement || !('MozAppearance' in test.style))
+ return;
+
+var scale;
+var isMac = navigator.platform == 'MacIntel';
+var thumb = {
+ radius: isMac ? 9 : 6,
+ width: isMac ? 22 : 12,
+ height: isMac ? 16 : 20
+};
+var track = '-moz-linear-gradient(top, transparent ' + (isMac ?
+ '6px, #999 6px, #999 7px, #ccc 9px, #bbb 11px, #bbb 12px, transparent 12px' :
+ '9px, #999 9px, #bbb 10px, #fff 11px, transparent 11px') +
+ ', transparent)';
+var styles = {
+ 'min-width': thumb.width + 'px',
+ 'min-height': thumb.height + 'px',
+ 'max-height': thumb.height + 'px',
+ padding: 0,
+ border: 0,
+ 'border-radius': 0,
+ cursor: 'default',
+ 'text-indent': '-999999px' // -moz-user-select: none; breaks mouse capture
+};
+var onChange = document.createEvent('HTMLEvents');
+onChange.initEvent('change', true, false);
+
+if (document.readyState == 'loading')
+ document.addEventListener('DOMContentLoaded', initialize, true);
+else
+ initialize();
+
+function initialize() {
+ // create initial sliders
+ Array.forEach(document.querySelectorAll('input[type=range]'), transform);
+ // create sliders on-the-fly
+ document.addEventListener('DOMNodeInserted', onNodeInserted, true);
+}
+
+function onNodeInserted(e) {
+ check(e.target);
+ if (e.target.querySelectorAll)
+ Array.forEach(e.target.querySelectorAll('input'), check);
+}
+
+function check(input, async) {
+ if (input.localName != 'input' || input.type == 'range');
+ else if (input.getAttribute('type') == 'range')
+ transform(input);
+ else if (!async)
+ setTimeout(check, 0, input, true);
+}
+
+function transform(slider) {
+
+ var isValueSet, areAttrsSet, isChanged, isClick, prevValue, rawValue, prevX;
+ var min, max, step, range, value = slider.value;
+
+ // lazily create shared slider affordance
+ if (!scale) {
+ scale = document.body.appendChild(document.createElement('hr'));
+ style(scale, {
+ '-moz-appearance': isMac ? 'scale-horizontal' : 'scalethumb-horizontal',
+ display: 'block',
+ visibility: 'visible',
+ opacity: 1,
+ position: 'fixed',
+ top: '-999999px'
+ });
+ document.mozSetImageElement('__sliderthumb__', scale);
+ }
+
+ // reimplement value and type properties
+ var getValue = function() { return '' + value; };
+ var setValue = function setValue(val) {
+ value = '' + val;
+ isValueSet = true;
+ draw();
+ delete slider.value;
+ slider.value = value;
+ slider.__defineGetter__('value', getValue);
+ slider.__defineSetter__('value', setValue);
+ };
+ slider.__defineGetter__('value', getValue);
+ slider.__defineSetter__('value', setValue);
+ slider.__defineGetter__('type', function() { return 'range'; });
+
+ // sync properties with attributes
+ ['min', 'max', 'step'].forEach(function(prop) {
+ if (slider.hasAttribute(prop))
+ areAttrsSet = true;
+ slider.__defineGetter__(prop, function() {
+ return this.hasAttribute(prop) ? this.getAttribute(prop) : '';
+ });
+ slider.__defineSetter__(prop, function(val) {
+ val === null ? this.removeAttribute(prop) : this.setAttribute(prop, val);
+ });
+ });
+
+ // initialize slider
+ slider.readOnly = true;
+ style(slider, styles);
+ update();
+
+ slider.addEventListener('DOMAttrModified', function(e) {
+ // note that value attribute only sets initial value
+ if (e.attrName == 'value' && !isValueSet) {
+ value = e.newValue;
+ draw();
+ }
+ else if (~['min', 'max', 'step'].indexOf(e.attrName)) {
+ update();
+ areAttrsSet = true;
+ }
+ }, true);
+
+ slider.addEventListener('mousedown', onDragStart, true);
+ slider.addEventListener('keydown', onKeyDown, true);
+ slider.addEventListener('focus', onFocus, true);
+ slider.addEventListener('blur', onBlur, true);
+
+ function onDragStart(e) {
+ isClick = true;
+ setTimeout(function() { isClick = false; }, 0);
+ if (e.button || !range)
+ return;
+ var width = parseFloat(getComputedStyle(this, 0).width);
+ var multiplier = (width - thumb.width) / range;
+ if (!multiplier)
+ return;
+ // distance between click and center of thumb
+ var dev = e.clientX - this.getBoundingClientRect().left - thumb.width / 2 -
+ (value - min) * multiplier;
+ // if click was not on thumb, move thumb to click location
+ if (Math.abs(dev) > thumb.radius) {
+ isChanged = true;
+ this.value -= -dev / multiplier;
+ }
+ rawValue = value;
+ prevX = e.clientX;
+ this.addEventListener('mousemove', onDrag, true);
+ this.addEventListener('mouseup', onDragEnd, true);
+ }
+
+ function onDrag(e) {
+ var width = parseFloat(getComputedStyle(this, 0).width);
+ var multiplier = (width - thumb.width) / range;
+ if (!multiplier)
+ return;
+ rawValue += (e.clientX - prevX) / multiplier;
+ prevX = e.clientX;
+ isChanged = true;
+ this.value = rawValue;
+ }
+
+ function onDragEnd() {
+ this.removeEventListener('mousemove', onDrag, true);
+ this.removeEventListener('mouseup', onDragEnd, true);
+ }
+
+ function onKeyDown(e) {
+ if (e.keyCode > 36 && e.keyCode < 41) { // 37-40: left, up, right, down
+ onFocus.call(this);
+ isChanged = true;
+ this.value = value + (e.keyCode == 38 || e.keyCode == 39 ? step : -step);
+ }
+ }
+
+ function onFocus() {
+ if (!isClick)
+ this.style.boxShadow = !isMac ? '0 0 0 2px #fb0' :
+ '0 0 2px 1px -moz-mac-focusring, inset 0 0 1px -moz-mac-focusring';
+ }
+
+ function onBlur() {
+ this.style.boxShadow = '';
+ }
+
+ // determines whether value is valid number in attribute form
+ function isAttrNum(value) {
+ return !isNaN(value) && +value == parseFloat(value);
+ }
+
+ // validates min, max, and step attributes and redraws
+ function update() {
+ min = isAttrNum(slider.min) ? +slider.min : 0;
+ max = isAttrNum(slider.max) ? +slider.max : 100;
+ if (max < min)
+ max = min > 100 ? min : 100;
+ step = isAttrNum(slider.step) && slider.step > 0 ? +slider.step : 1;
+ range = max - min;
+ draw(true);
+ }
+
+ // recalculates value property
+ function calc() {
+ if (!isValueSet && !areAttrsSet)
+ value = slider.getAttribute('value');
+ if (!isAttrNum(value))
+ value = (min + max) / 2;;
+ // snap to step intervals (WebKit sometimes does not - bug?)
+ value = Math.round((value - min) / step) * step + min;
+ if (value < min)
+ value = min;
+ else if (value > max)
+ value = min + ~~(range / step) * step;
+ }
+
+ // renders slider using CSS background ;)
+ function draw(attrsModified) {
+ calc();
+ if (isChanged && value != prevValue)
+ slider.dispatchEvent(onChange);
+ isChanged = false;
+ if (!attrsModified && value == prevValue)
+ return;
+ prevValue = value;
+ var position = range ? (value - min) / range * 100 : 0;
+ var bg = '-moz-element(#__sliderthumb__) ' + position + '% no-repeat, ';
+ style(slider, { background: bg + track });
+ }
+
+}
+
+function style(element, styles) {
+ for (var prop in styles)
+ element.style.setProperty(prop, styles[prop], 'important');
+}
+
+})();
BIN  i/header.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
BIN  i/volumecontrol_16x16.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
BIN  i/volumecontrol_220x140.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
BIN  i/volumecontrol_32x32.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
BIN  i/volumecontrol_icon.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
BIN  i/volumecontrol_icon_dev.png
View
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
1  privacy.txt
View
@@ -0,0 +1 @@
+coming soon
198 style.css
View
@@ -0,0 +1,198 @@
+#container #header {
+ background-color: #f3f3f3;
+ height: 28px;
+ border-bottom: 1px solid #d9d9d9;
+ padding: 0 10px;
+ line-height: 24px;
+ font-weight: bold;
+ font-size: 11px;
+ vertical-align: middle;
+}
+
+#container #header .icon_header {
+ background-image: url();
+ height: 16px;
+ width: 16px;
+ position: absolute;
+ top: 5px;
+ left: 5px;
+}
+
+#container #header .header_title {
+ margin-left: 18px;
+ line-height: 28px;
+}
+
+#container #body .button {
+ position: relative;
+ top: -20px;
+ left: 10px;
+ background-color: #fff;
+ border:1px solid transparent;
+ display:inline-block;
+ color:#2e2e2e;
+ height:18px;
+ width: 18px;
+ font-weight:bold;
+ padding:4px 8px;
+ text-decoration:none;
+ background-image: url();
+ background-repeat: no repeat;
+}
+
+#container #body .button:hover {
+ -webkit-box-shadow: inset 0 1px 2px rgba(0,0,0,.1);
+ box-shadow: inset 0 1px 2px rgba(0,0,0,.1);
+ -webkit-border-radius: 2px;
+ border-radius: 2px;
+ background-color: #f8f8f8;
+ border-color: #c6c6c6;
+ background-image: url();
+ background-repeat: no repeat;
+}
+
+#container #body .button.muted{
+ position: relative;
+ top: -20px;
+ left: 10px;
+ -webkit-border-radius: 2px;
+ border-radius: 2px;
+ background-color: #DD4B49;
+ border:1px solid #B13C2E;
+ color:#2e2e2e;
+ background-image: url();
+ background-repeat: no repeat;
+}
+
+#container #body {
+ background-color: #fff;
+ overflow-x: auto;
+}
+
+#container #body #participants {
+ list-style: none;
+ margin: 0px;
+ padding: 10px;
+}
+
+#container #body #participants li {
+ height: 40px;
+}
+
+#container #body #participants li img{
+ width: 32px;
+ height: 32px;
+ padding: 10px;
+}
+
+#container #body #participants li input[type="range"] {
+ -webkit-appearance: none;
+ background-color: #d9d9d9;
+ height: 3px;
+ position: relative;
+ top: -25px;
+ margin-left: 5px;
+ width: 185px;
+ border: 1px solid #d6d6d6;
+}
+
+#container #body #participants li input[type="range"]::-webkit-slider-thumb {
+ -webkit-appearance: none;
+ position: relative;
+ top: -1px;
+ z-index: 1;
+ width: 7px;
+ height: 18px;
+ -webkit-border-radius: 3px;
+ -moz-border-radius: 3px;
+ border-radius: 3px;
+ background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#cfcfcf), color-stop(50%,#bbbbbb), color-stop(51%,#acacac), color-stop(100%,#909090));
+}
+
+#container #footer {
+ background-color: #f3f3f3;
+ border-right: 1px solid #d9d9d9;
+ height: 40px;
+ border-top: 1px solid #d9d9d9;
+}
+
+#container #footer .ltbutton {
+-webkit-border-radius: 2px;
+ border-radius: 2px;
+ background-color:#ededed;
+ border:1px solid #dcdcdc;
+ display:inline-block;
+ color:#2e2e2e;
+ font-family:arial;
+ font-size:15px;
+ font-weight:bold;
+ padding:4px 8px;
+ text-decoration:none;
+ float: right;
+ position: relative;
+ right: 5px;
+ top: 5px;
+}
+#container #footer .ltbutton:hover {
+-webkit-box-shadow: inset 0 1px 2px rgba(0,0,0,.1);
+ box-shadow: inset 0 1px 2px rgba(0,0,0,.1);
+-webkit-border-radius: 2px;
+ border-radius: 2px;
+ background-color:#dfdfdf;
+ color: #DD4B49;
+}
+
+#container #footer .ltbutton_muted {
+-webkit-border-radius: 2px;
+ border-radius: 2px;
+ background-color:#DD4B49;
+ border:1px solid #B13C2E;
+ display:inline-block;
+ color:#fff;
+ font-family:arial;
+ font-size:15px;
+ font-weight:bold;
+ padding:4px 8px;
+ text-decoration:none;
+}
+#container #footer .ltbutton_muted:hover {
+ background-color:#dd4b49;
+ color: #fff;
+}
+
+#container #footer .footer_note {
+ font-size:10px;
+ font-family:verdana;
+ color:#2e2e2e;
+ float:left;
+ position: relative;
+ left: 5px;
+ top: 15px;
+}
+
+#container #footer a {
+ font-size:10px;
+ font-family:verdana;
+ color:#2e2e2e;
+ text-decoration:none;
+}
+#container #footer a:hover {
+ text-decoration:underline;
+ padding-bottom: 5px;
+ margin-bottom: 5px;
+}
+
+#container #body #participants .gain {
+ position: absolute;
+ top: 10px;
+ left: 52px;
+ width: 3px;
+ height: 32px;
+ border: 0px solid #d9d9d9;
+ vertical-align: bottom;
+}
+#container #body #participants .gain_level {
+ position: relative;
+ bottom:0px;
+ width: 3px;
+}
1  support.txt
View
@@ -0,0 +1 @@
+coming soon
439 volumecontrol.js
View
@@ -0,0 +1,439 @@
+(function(){
+ /*
+ * Volume Control for Google+ Hangouts
+ * Copyright 2012 Moritz Tolxdorff
+ * Version: 1.0.3
+ * Release date: 06.04.2012
+ * Developers:
+ ** Moritz Tolxdorff
+ ** Robert Pitt
+ *
+ * Thanks:
+ ** Robert Pitt
+ */
+
+
+ /**
+ * @ApplicationController
+ * @constructor
+ */
+ function ApplicationController(){
+ if(!gapi){
+ throw "gapi not loaded!";
+ }
+
+ /**
+ * @ApplicationController.DEBUGGING - defines if debugging is enabled
+ * @private
+ * @type {boolean}
+ */
+ this.DEBUGGING = false;
+
+ /**
+ * @ApplicationController.maxHeight - defines the maximum window height
+ * @public
+ * @const
+ * @type {Number}
+ */
+ this.maxHeight = $(window).height();
+
+ /**
+ * @ApplicationController.globalShow - defines the initial state of globalShow
+ * @private
+ * @type {boolean}
+ */
+ this.globalMuted = false;
+
+ /**
+ * @ApplicationController.volumeColorEnum - defines colors for volume levels
+ * @public
+ * @enum
+ * @type {object}
+ */
+ this.volumeColorEnum = {
+ 1 : "#11b012",
+ 2 : "#4f9022",
+ 3 : "#847430",
+ 4 : "#a6623a",
+ 5 : "#cf4c44"
+ }
+
+ /*
+ * Bind gapi events when API is ready
+ */
+ gapi.hangout.onApiReady.add(this.onApiReady.bind(this));
+ gapi.hangout.onParticipantsChanged.add(this.onParticipantsChanged.bind(this));
+ gapi.hangout.av.onVolumesChanged.add(this.onVolumesChanged.bind(this));
+
+ /*
+ * Bind window events when window size has changed
+ */
+ $(window).resize(this.onWindowResize.bind(this));
+ }
+
+ /**
+ * @onParticipantsChanged - Fired participants changed
+ * @private
+ * @param evt {gapi.hangout.ParticipantsChangedEvent}
+ */
+ ApplicationController.prototype.onParticipantsChanged = function(evt){
+ this.generateControlls();
+ if(this.globalMuted == true){
+ var p = gapi.hangout.getParticipants();
+ for(var i=0; i<p.length; i++) {
+ gapi.hangout.av.setParticipantAudioLevel(p[i].id, 0)
+ }
+ }
+ }
+
+ /**
+ * @onVolumesChanged - Fired when volume levels change
+ * @private
+ * @param evt {gapi.hangout.av.VolumesChangedEvent}
+ */
+ ApplicationController.prototype.onVolumesChanged = function(evt){
+ var map = {};
+ var items = $("#participants li");
+
+ for(var i = 0; i < items.length; i++){
+ map[items[i].id.replace("participant_","")] = items[i];
+ }
+
+ for(var id in evt.volumes){
+ var level = parseInt(evt.volumes[id]);
+ if(id in map){
+ $(".gain_level", map[id]).css({"height": (6.4 * level), "margin-top": (32 - (level * 6.4))});
+
+ if(level in this.volumeColorEnum){
+ $(".gain_level", map[id]).css({"background-color": this.volumeColorEnum[level]});
+ }
+ }
+ }
+ }
+
+ /**
+ * @onWindowResize - Fired when window resizes
+ * @private
+ * @param evt {jQueryEventObject}
+ */
+ ApplicationController.prototype.onWindowResize = function(evt){
+ this.log("Window resized");
+ this.maxHeight = $(window).height();
+ this.scale();
+ }
+
+ /**
+ * @buildDOM - Building the DOM structure
+ * @private
+ */
+ ApplicationController.prototype.buildDOM = function(){
+ this.log("Building DOM");
+
+ /*
+ * Creates an empty div element
+ */
+ var div = this.createElement("div");
+
+ /*
+ * Creates an empty span element
+ */
+ var span = this.createElement("span");
+
+ /*
+ * Creates an empty a element
+ */
+ var a = this.createElement("a", {"target": "_blank"});
+
+ /*
+ * Create pane header
+ */
+ var header = div.clone().attr({"id": "header"});
+
+ /*
+ * Append icon and title to header
+ */
+ header.append(span.clone().attr({"class": "icon_header"}));
+ header.append(span.clone().attr({"class": "header_title"}).html("Volume Control v1.0.3"));
+
+ /*
+ * Create pane body
+ */
+ var body = div.clone().attr({"id": "body"}).css({"height": (this.maxHeight-70)+"px"});
+
+ /*
+ * Create the ul element for the template list and append it to the body
+ */
+ var ul = this.createElement("ul", {"id": "participants"}).appendTo(body);
+
+ /*
+ * Create the footer Div
+ */
+ var footer = div.clone().attr({"id": "footer"});
+
+ /*
+ * Create On/Off button and append it to the footer
+ */
+ var button = this.createElement("button", {"class": "ltbutton"}).text("Mute Hangout").appendTo(footer);
+
+ /*
+ * Append footer note to footer
+ */
+ footer.append(span.clone().attr({"class": "footer_note"}).html("&copy 2012 ").append(a.clone().attr({"href": "https://plus.google.com/117596712775912423303"}).html("Moritz")).append(" &amp; ").append(a.clone().attr({"href": "https://plus.google.com/110106586947414476573"}).html("Robert")));
+
+ /*
+ * Bind click event to the On/Off button
+ */
+ button.click(this.toggleMute.bind(this));
+
+ /*
+ * Append DOM structure to container
+ */
+ jQuery("#container").append(header, body, footer);
+ }
+
+ /**
+ * @scale - Scales the body for different resolutions
+ * @public
+ */
+ ApplicationController.prototype.scale = function(){
+ /*
+ * Set the maximum height of the body minus header, input div and footer
+ */
+ jQuery("#body").height(this.maxHeight-70);
+ }
+
+ /**
+ * @onSliderChange - Fired when a slider input[type=range] is moved
+ * @private
+ * @param evt {jQueryEventObject}
+ */
+ ApplicationController.prototype.onSliderChange = function(evt){
+ var li = jQuery(evt.target.parentNode);
+ var data = li.data("participant");
+ var level = parseInt(evt.target.value);
+ /* Function for future use when the API changes
+ if(level < 1){
+ level = parseFloat(level).toFixed(1);
+ gapi.hangout.av.setParticipantAudioLevel(data.id, [level,level]);
+ return;
+ }*/
+
+ /*
+ * Setting the audio level for a participant of which the slider was changed
+ */
+ gapi.hangout.av.setParticipantAudioLevel(data.id, [level,level]);
+
+ /*
+ * Switch between button styles depending on the mute state
+ */
+ level === 0 ? jQuery(".button",li).addClass("muted") : jQuery(".button",li).removeClass("muted");
+ }
+
+ /**
+ * @muteAllParticipants - mutes all participants in the hangout
+ * @public
+ * @param volume {int}
+ */
+ ApplicationController.prototype.setVolumeForAllParticipants = function(volume){
+ p = gapi.hangout.getParticipants();
+ for(i = 0; i < p.length; i++) {
+ gapi.hangout.av.setParticipantAudioLevel(p[i].id, volume)
+ }
+ }
+
+ /**
+ * @toggleMute - Fired when #button is clicked
+ * @public
+ * @see ApplicationController.buildDOM
+ */
+ ApplicationController.prototype.toggleMute = function(){
+ var muteButtons = jQuery("#participants li .button");
+ var rangeSliders = jQuery("#participants li input[type=range]");
+ if(this.globalMuted === false){
+ this.setVolumeForAllParticipants(0);
+
+ muteButtons.addClass("muted");
+ rangeSliders.val(0);
+ gapi.hangout.layout.displayNotice("You've muted the whole hangout. This only applies for yourself!");
+ gapi.hangout.av.setMicrophoneMute(true);
+ gapi.hangout.av.setCameraMute(true);
+ jQuery("#footer .ltbutton").text("Unmute Hangout").addClass("ltbutton_muted");
+ muteButtons.attr({"disabled": "disabled"});
+ rangeSliders.attr({"disabled": "disabled"});
+ this.globalMuted = true;
+ }else{
+ /* Commented for testing
+ gapi.hangout.av.setMicrophoneMute(false);
+ gapi.hangout.av.setCameraMute(false);
+ */
+ this.setVolumeForAllParticipants(1);
+
+ muteButtons.removeClass("muted");
+ rangeSliders.val(1);
+ gapi.hangout.layout.displayNotice("You've unmuted the whole hangout.");
+ gapi.hangout.av.clearMicrophoneMute();
+ gapi.hangout.av.clearCameraMute();
+ jQuery("#footer .ltbutton").text("Mute Hangout").removeClass("ltbutton_muted");
+ muteButtons.removeAttr("disabled");
+ rangeSliders.removeAttr("disabled");
+ this.globalMuted = false;
+ }
+ }
+
+ /**
+ * @toggleMuteParticipant - Fired when the participant mute button is clicked
+ * @public
+ * @param evt {jQueryEventObject}
+ */
+ ApplicationController.prototype.toggleMuteParticipant = function(evt){
+ var li = jQuery(evt.target.parentNode);
+
+ if(jQuery(evt.target).hasClass("muted")){
+ jQuery("input[type=range]",li)[0].value = 1;
+ jQuery(evt.target).removeClass("muted");
+ }else{
+ jQuery("input[type=range]",li)[0].value = 0;
+ jQuery(evt.target).addClass("muted");
+ }
+
+ jQuery("input[type=range]",li).trigger("change");
+ }
+
+ /**
+ * @generateControlls - Generates the slider and mute controlls for the participants
+ * @private
+ */
+ ApplicationController.prototype.generateControlls = function(){
+ var uid = gapi.hangout.getParticipantId();
+ var p = gapi.hangout.getParticipants();
+ var ul = jQuery("#participants");
+
+ /*
+ * Creates an empty div element
+ */
+ var div = this.createElement("div");
+
+ jQuery("li",ul).remove();
+
+ /*
+ * Loop through all participants
+ */
+ for(i = 0; i < p.length; i++) {
+ /*
+ * Ignore this itteration for current user
+ */
+ if(p[i].id == uid){
+ continue;
+ }
+
+ var cUser = p[i];
+
+ /*
+ * Setting all needed variables for participant volume level, placeHolder image and audio levels
+ */
+ var volume_level = gapi.hangout.av.getParticipantVolume(cUser.id);
+ var placeholderImage = "";
+ var levels = gapi.hangout.av.getParticipantAudioLevel(cUser.id);
+
+ /*
+ * Creating li element for each participant
+ */
+ var li = this.createElement("li", {"id": "participant_" + cUser.id}).data("participant",cUser);
+
+ /*
+ * Creating outter gain div for the volume bar
+ */
+ var gain = div.clone().attr({"class": "gain"}).css({"top": ( 9 + i * 40 ) + "px"}).appendTo(li);
+
+ /*
+ * Creating the inner gain_level div for the color changing volume bar
+ */
+ var gain_level = div.clone().attr({"class": "gain_level"}).appendTo(gain);
+
+ /*
+ * Creating the profile image
+ */
+ var image = this.createElement("img", {"src": placeholderImage, "title": "Unknown"}).appendTo(li);
+
+ /*
+ * Creating the slider element
+ */
+ var slider = this.createElement("input", {"type": "range", "min": "0", "max": "10", "step": "1", "value": gapi.hangout.av.getParticipantAudioLevel(cUser.id)[0]}).appendTo(li);
+
+ /*
+ * Creating the participant mute button
+ */
+ var muteButton = this.createElement("input", {"type": "button", "class": "button", "value": ""});
+
+ /*
+ * Checking if the slider is below 0.1 or 0 to set the button to muted
+ */
+ if(levels[0] < 0.1 || levels[0] === 0){
+ muteButton.addClass("muted");
+ }
+
+ /*
+ * Binding the change and click event to elements
+ */
+ slider.bind("change", this.onSliderChange.bind(this));
+ muteButton.click(this.toggleMuteParticipant.bind(this));
+
+ /*
+ * If a participant is using the app use their profile picture and name instead of the placholders
+ */
+ if(cUser.person){
+ image.attr({"src": cUser.person.image.url, "title": cUser.person.displayName})
+ }
+
+ /*
+ * Appending the mute button to the li element and the li to the ul element
+ */
+ li.append(muteButton);
+ ul.append(li);
+ }
+ }
+
+ /**
+ * @createElement - Creates a new element
+ * @public
+ * @param type {String}
+ * @param attr {Object}
+ */
+ ApplicationController.prototype.createElement = function(type, attr){
+ return jQuery("<" + type + ">").attr(attr || {});
+ }
+
+ /**
+ * @log - Writes to console.log if DEBUGGING is enabled
+ * this.log(...)
+ * @private
+ * @param {Mixed}
+ */
+ ApplicationController.prototype.log = function(){
+ if(this.DEBUGGING === true){
+ console.log(Array.prototype.slice.call(arguments))
+ }
+ }
+
+ /**
+ * @onApiReady - Fired by gapi when it's ready
+ * @private
+ * @param event {gapi.hangout.apiReadyEvent}
+ */
+ ApplicationController.prototype.onApiReady = function(event){
+ if(event.isApiReady){
+ try {
+ this.buildDOM();
+ this.scale();
+ this.generateControlls();
+ console.log("Volume Control loaded!");
+ }
+ catch(err) {
+ console.log(err);
+ }
+ }
+ }
+
+ // Export instantiated ApplicationController to main window
+ window["appController"] = new ApplicationController();
+})()
44 volumecontrol.xml
View
@@ -0,0 +1,44 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<Module>
+ <!--
+ * Volume Control for Google+ Hangouts
+ * Copyright 2012 Moritz Tolxdorff
+ * Version: 1.0.3
+ * Release date: 06.04.2012
+ * Developers:
+ ** Moritz Tolxdorff
+ ** Robert Pitt
+ *
+ * Thanks:
+ ** Robert Pitt
+ -->
+ <ModulePrefs title="VolumeControl">
+ <Require feature="rpc"/>
+ <Require feature="views"/>
+ </ModulePrefs>
+ <Content type="html">
+ <![CDATA[
+ <script src="https://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js"></script>
+ <script src="//hangoutsapi.talkgadget.google.com/hangouts/api/hangout.js?v=1.0"></script>
+ <link rel="stylesheet" type="text/css" href="//tolxdorff.appspot.com/a/volume/style.css" />
+ <!--[if lt IE 9]>
+ <script src="//html5shiv.googlecode.com/svn/trunk/html5.js"></script>
+ <![endif]-->
+ <div id="container">
+ </div>
+ <script src="//tolxdorff.appspot.com/a/volume/html5slider.js"></script>
+ <script src="//tolxdorff.appspot.com/a/volume/volumecontrol.js"></script>
+ <script type="text/javascript">
+ var _gaq = _gaq || [];
+ _gaq.push(['_setAccount', 'UA-30670305-1']);
+ _gaq.push(['_trackPageview']);
+
+ (function() {
+ var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
+ ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
+ var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s);
+ })();
+ </script>
+ ]]>
+ </Content>
+</Module>
Please sign in to comment.
Something went wrong with that request. Please try again.