NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name OLD ADC Monitor Improver // @description Various ADC ticket monitor display improvements. // @author mileswilford // @version 3.2.0 // @require http://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.11.2/moment.min.js // @include http://monitor.drafthouse.com/monitor.aspx* // @include https://monitor.drafthouse.com/monitor.aspx* // ==/UserScript== $(document).ready(function() { var ADC_MI_VERSION = "3.2.0"; var STAGING_MODE = true; /* CONFIGURABLES ---------------------------------------------------------------- */ var titleReplaces = [ ['Drafthouse Films: ', ''], ['2D ', ''], ['3D ', '3D ', 'show-3d'], ['PRIVATE EVENT 59 min', 'PCE/TECH', 'pce-tech'], ['PRIVATE EVENT 29 min', 'PCE/TECH', 'pce-tech'], ['Private Party', 'Private Party', 'pce-tech'], ['PCE Tech Check', 'PCE Tech Check', 'pce-tech'], ['Girlie Night: ', ''], ['Action Pack: ' , ''], ['Mondo x Chiller: ', ''], ['Advance Victory Screening: ', ''], ['Victory Screening: ', ''], ['Kids Camp: ', 'Kids: '], ['Promo Screening: ', ''] ]; var virtualScreens = { '0002' : 3, '0007' : 11, "0102" : 8, "0103" : 8, '0901' : 7, "1401" : 9 }; /* END CONFIGURABLES ----------------------------------------------------------- */ // Parses times into miliseconds, conscientious of the fact that times before 5AM are tomorrow. function timeParse(timeString) { var timeFormat = "h:mm A"; var parsedTimeMoment = moment(timeString, timeFormat); if (parsedTimeMoment.hour() < 5) { parsedTimeMoment.add(1, 'days'); } return parsedTimeMoment.toDate().getTime(); } function getURLParameter(param) { var query = window.location.search.substring(1); var vars = query.split("&"); for (var i = 0; i < vars.length ; i++) { var pair = vars[i].split("="); if (pair[0] == param){ return pair[1]; } } return false; } // Returns the time as a numerical percent // Parses the time if it is fed a string function getTimePercent(time) { if (typeof time == 'string') { time = timeParse(time); } return ((time - firstAndLastTime[0])/(firstAndLastTime[1] - firstAndLastTime[0]) * 100); } // Add version number for reference and lines toggle $('body').append($("<div>").text("ADC MI " + ADC_MI_VERSION).css({ "position" : "fixed", "bottom" : 0, "right" : 0, "opacity" : 0.3 }).prepend("<label id='adcmi-gridlines'><input type='checkbox'/> Toggle Lines </label>")); // Add timer block $('body').append($("<div>").text("30").css({ "position": "fixed", "bottom" : 0, "left" : 0, "opacity" : 0.6 }).addClass('timer-block')); // Add scheduling mode toggle $('body').append($('<div><label id="adcmi-scheduling"><input type="checkbox" /> Scheduling Mode </label>').css({ "position" : "fixed", "bottom" : 0, "left" : "4em", "opacity" : 0.6 })); $('body').append($('<div id="adcmi-scheduling-info">').css({ "position" : "absolute", "top" : 0, "right" : 0 })); // The written instructions $('.theater').after($('<div>').text("Dark lines represent check drop time in the show. Dark area represents time elapsed. Pink areas represent 30-minute preshows.") .css('text-align', 'center')); // Gets me the first and last times of the day in ms form in an array firstAndLastTime = (function () { var alltimes = []; $('.timestart, .timeend').each(function() { var theTime = $(this).text(); alltimes.push(timeParse(theTime)); }); alltimes.sort(); return [alltimes[0], alltimes[alltimes.length-1]]; })(); // Moves left point 30 minutes left to allow for 1st preshow firstAndLastTime[0] -= 1000 * 60 * 30; // Some front-end cleanup and prep for my absolute positioning // Fade the virtual screen if (virtualScreens[getURLParameter('siteid')]) { $('.screen:nth-of-type(' + virtualScreens[getURLParameter('siteid')] + ')') .css({'opacity' : 0.35, 'border' : 0}); } // Hide some stuff that is useless or rendered obsolete by this script $('.layout').hide(); // Remove now-redundant intermissions $('.intermission').remove(); $('.format').remove(); // Provide a box-space and adjust borders for screens $('.screen').css({ 'position' : 'relative', 'height' : 92 }); // Fixes firefox display bug related to box-sizing $('.seats').css({ 'position' : 'relative', 'top' : '-4px' }); // Will later set the positions of the screenings $('.screen > span').attr('style', '').css({ 'position' : 'absolute', 'top' : 0, 'bottom' : 0, 'border': 0 }).each(function() { var $this = $(this); var seats = $this.find('.seats').text().split(' '); var numSold = parseInt(seats[0], 10); var numLeft = parseInt(seats.pop(), 10); var seatingPercent = (numSold / (numSold + numLeft)); var red; var green; var blue; var yellowPoint = 0.6; if (seatingPercent < yellowPoint) { // red = 255 * (seatingPercent / yellowPoint); // green = 255; // blue = 0; red = 150; green = 150; blue = 150; } else { red = 255; green = 255 * Math.pow((1 - seatingPercent)/(1 - yellowPoint), 1); blue = 0; } $after = $('<div>').css({ 'position' : 'absolute', 'top' : 0, 'bottom' : 0, 'left' : 0, 'right' : 0, 'z-index' : -5 }); $after.css('background-color', 'rgba(' + Math.round(red) + ', ' + Math.round(green) + ', ' + Math.round(blue) + ', .3)'); $this.append($after); $this.css('background-color', 'white'); }); $('div').css('border-color', 'transparent'); // Provide a box-space for the count meter $('.theater').css({'position' : 'relative', 'overflow' : 'hidden'}) // Add theater numbers .prepend($('<div>').addClass('theater-count').css({ 'position' : 'absolute', 'top' : 0, 'bottom' : 0 })); // Format titles and execute titleReplaces $('.title').css('text-align', 'center').each(function() { var title = $(this).text(); for (var i = 0; i < titleReplaces.length; i++) { if (titleReplaces[i][2] && title.indexOf(titleReplaces[i][0]) != -1) { // Later, these classes are hooks to run additional changes $(this).addClass(titleReplaces[i][2]); } title = title.replace(titleReplaces[i][0], titleReplaces[i][1]); } $(this).text(title); }); // Make sure Not On Sale shows are always the bottom of the stack $('.notonsale').each(function() { $(this).parent().css({ 'z-index': 0, 'background-color' : '#DDF' }); }); // Add theater numbers var screenCounter = 0; $('.screen').each(function() { screenCounter++; $(this).prepend($('<div>').text(screenCounter).css({ 'position' : 'absolute', 'top' : 0, 'bottom' : 0, 'left' : 0, 'font-size' : '4em' })); }); // Absolutely position each show's left and right sides $('.screen > span').each(function() { var startTime = $(this).find('.timestart').text(); var endTime = $(this).find('.timeend').text(); $(this).css({ 'left' : getTimePercent(startTime) + '%', 'right' : (100 - getTimePercent(endTime)) + '%' }); }); // Get the current time in ms, conscientious of the fact that times before 5AM are tomorrow. var now = (function() { var current = moment(); if (current.hour() < 5) { current.add(1, 'days'); } return current; })(); // Create current time meter edge and position it absolutely // It does NOT need to move automatically - the monitor auto-refreshes too quickly for it to matter // First get the current date as YYYYMMDD var nowString = (function() { var current = moment(now); var year = current.year().toString(); var month = (function() { var month = current.month() + 1; if (month < 10) { return "0" + month.toString(); } else { return month.toString(); } })(); var day = (function() { var day = current.date(); if (day < 10) { return "0" + day; } else { return day.toString(); } })(); return year + month + day; })(); // Check the URL parameters to make sure it is looking at current day var dateParam = getURLParameter('date'); if (dateParam == nowString || !dateParam) { // Insert the meter $('.theater').append($('<div>').css({ 'background-color' : 'rgba(0, 0, 0, 0.15)', 'pointer-events' : 'none', 'padding-top' : $('.theater').height(), 'position' : 'absolute', 'top' : 0, 'bottom' : 0, 'left' : 0, 'right' : 100-getTimePercent(now.toDate().getTime()) + '%', 'z-index' : 1000 })); } // Creates a check-drop line marker and position it absolutely. // This must be done in pixels, which means the window will need to reload after it resizes. function minutesToPixels(minutes) { var fourtyMinMs = minutes * 60 * 1000; var fourtyMinPercent = getTimePercent(firstAndLastTime[0] + fourtyMinMs); var theaterWidth = $('.theater').width(); return theaterWidth * (fourtyMinPercent/100); } $('.screen > span').append($('<div>').addClass('timedrop-line').css({ 'background-color' : 'rgba(0, 0, 0, 0.4)', 'position' : 'absolute', 'top' : '1.8em', 'bottom' : 0, 'right' : minutesToPixels(40) + 'px', 'width' : '2px', 'z-index' : 1000 })).prepend($('<div>').addClass('preshow-marker').css({ 'position' : 'absolute', 'top' : 1, 'bottom' : 1, 'left' : -minutesToPixels(30)-1, 'width' : minutesToPixels(30)-1, 'background-color' : '#FDE', 'opacity' : 0.5 })); $(window).resize(function() { $('.timedrop-line').css('right', minutesToPixels(40) + 'px'); $('.preshow-marker').css({ 'left' : -minutesToPixels(30)-1, 'width' : minutesToPixels(30)-1 }) }); // In this module, execute special show-based title-replace scripts (function() { $('.pce-tech').parent().css("overflow", "hidden").find('.timedrop, .timedrop-line').remove(); $('.show-3d').parent().append($('<div>').css({ 'background-image' : 'url()', 'background-position' : 'center center', 'background-repeat' : 'no-repeat', 'background-opacity' : 0.5, 'pointer-events' : 'none', 'opacity' : 0.5, 'position' : 'absolute', 'top' : 0, 'bottom' : 0, 'left' : 0, 'right' : 0 })); })(); // Buttons and configurations module (function() { function toggleBorders(borderType) { $('.adcmi-screen').css('border-top', "5px solid " + borderType); } $('#adcmi-gridlines').click(function() { if ($(this).find('input').is(':checked')) { toggleBorders('black'); if (localStorage) { localStorage.setItem('showGridlines', 'true'); } } else { toggleBorders('transparent'); if (localStorage) { localStorage.setItem('showGridlines', 'false'); } } }); if (localStorage && localStorage.getItem('showGridlines') == 'true') { $('#adcmi-gridlines').click(); toggleBorders('black'); } function toggleSchedulingMode(onOff) { if (onOff) { $('.notonsale').parent().find('.preshow-marker').show(); $('#adcmi-scheduling-info').show(); } else { $('.notonsale').parent().find('.preshow-marker').hide(); $('#adcmi-scheduling-info').hide(); } } $('#adcmi-scheduling').click(function() { if ($(this).find('input').is(':checked')) { toggleSchedulingMode(true); if (localStorage) { localStorage.setItem('schedulingMode', 'true'); } } else { toggleSchedulingMode(false); if (localStorage) { localStorage.setItem('schedulingMode', 'false'); } } }); if (localStorage && localStorage.getItem('schedulingMode') == 'true') { $('#adcmi-scheduling').find('input').prop('checked', true); } else { toggleSchedulingMode(false); } })(); // Finally, make sure the site is reloading correctly in case the native auto reload isn't working. // Do this once every 30 seconds // TODO: reload in background? (function() { var timerCount = 30; var timerExpires = 0; // 35 seconds var $timerBlock = $('.timer-block'); function refreshTimer() { timerCount--; if (timerCount <= timerExpires) { location.reload(); } else { $timerBlock.text(timerCount); } } if (!STAGING_MODE) {setInterval(refreshTimer, 1000);} })(); });