Movie Dip - find movies to double dip (watch back-to-back)

On the Google Movie Showtimes, select the movies you are interested in seeing to find out which ones you can see back-to-back at the same theater.

// ==UserScript==
// @name        Movie Dip - find movies to double dip (watch back-to-back)
// @namespace   driver8.net
// @description On the Google Movie Showtimes, select the movies you are interested in seeing to find out which ones you can see back-to-back at the same theater.
// @version     0.5
// @grant       GM_addStyle
// @require     https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/sprintf/0.0.7/sprintf.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/jquery.tablesorter/2.17.5/js/jquery.tablesorter.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.7.0/moment.min.js
// @match		*://*.google.com/movies*
// ==/UserScript==

// im not vry good @ javascript sry

var AND_BUTTS = false;
var NUM_DIPS = 2;
var FRONT_TOLERANCE = 5;
var END_TOLERANCE = 40;
var FADE_SPEED = 200;
var PRETTY = true;
var LOGGING = false;

function log(msg) {
    LOGGING && console.log(msg);
}

function Film(title, id, length, times) {
    this.title = title;
    this.id = id;
    this.len = length;
    this.selected = false;
    this.checkBox = null;
    this.div = null;

    this.times = times;
}
Film.prototype.constructor = Film;


function Theater(name, id, num, films) {
    this.name = name;
    this.id = id;
    this.films = films;
    this.num = num;
    this.activeFilms = [];
    this.tableDiv = null;
}
Theater.prototype.constructor = Theater;

Theater.prototype.addFilm = function(newFilm) {
    this.films.push(newFilm);
};

Theater.prototype.toggleFilm = function(film) {
    if (this.activeFilms.contains(film)) {
        this.activeFilms.remove(film);
    } else this.activeFilms.push(film);
};

$.tablesorter.addParser({
    // set a unique id
    id: 'showtime',
    is: function(s) {
        // return false so this parser is not auto detected
        return false;
    },
    format: function(s, table, cell) {
        // format your data for normalization
        var sortNum = $(cell).data('sortNum');
        return sortNum;
    },
    // set type, either numeric or text
    type: 'numeric'
});


// Set things up
var theaters = [];
var theaterMap = {};
var tableMap = {};
// Gather list of theaters
var $allTheaters = $("div.theater");
var $settingsDiv = $('<div id="mdSettingsDiv"><b>Min. wait: </b><input type="text" class="mdSettings" id="mdFrontTolerance" name="FT" value="'+ FRONT_TOLERANCE +'" /> ' +
    '<b>Max. wait: </b><input type="text" class="mdSettings" id="mdEndTolerance" name="ET" value="'+ END_TOLERANCE +'" />' +
    '<b>Num. dips: </b><select class="mdSettings" id="mdNumDips" name="ND"><option value="2" selected>2</option><option value="3">3</option><option value="4">4</option>' +
    '<option value="5">5</option><option value="6">6</option><option value="7">7</option><option value="8">8</option></select>');
$("#results table:first").remove(); // just took up space at the top
$("#movie_results").before($settingsDiv);
$("input.mdSettings").blur(selectionsChanged);
$("#mdNumDips").change(selectionsChanged);

$allTheaters.each(function(idx) {
    var $theaterDiv = $(this);
    var theaterName = $theaterDiv.find("h2.name a").text();
    var theaterId = $theaterDiv.children("div.desc:first").attr("id");

// For each theater, gather list of movies
    var $movieDivs = $theaterDiv.find("div.movie");

    var newFilms = [];
    $movieDivs.each(function(idx2) {
        var $movieDiv = $(this);
        var movieId = theaterId + "_" + idx2;

// For each movie, gather names, lengths and showtimes
        var $movieNameA = $movieDiv.find("div.name a");
        var movieName = $movieNameA.text();
        AND_BUTTS && $movieNameA.text(movieName + " and Butts");

        var $newCheckbox = $('<input type="checkbox" />');
        $newCheckbox.data("theater", theaterId);
        $newCheckbox.data("movie", idx2);
        $newCheckbox.attr("id", theaterId + "-" + idx2);
        $newCheckbox.change(movieCheckboxChange);

        $movieNameA.before($newCheckbox);

// Cute stuff
        $movieDiv.click(function(event) {
            event.target.tagName !== "INPUT" && $newCheckbox.click();
            return true;
        });

// Find movie length
        var movieLength = 90;
        var lengthMatches = $movieDiv.find("span.info").text().match(/\D*(?:(\d+)h)?\D*(\d+)m/);
        if (lengthMatches) {
            var hours = parseInt(lengthMatches[1]);
            var mins = parseInt(lengthMatches[2]);
            mins = mins >= 0 ? mins : 90;
            mins += hours > 0 ? hours * 60 : 0;
            movieLength = mins;
        }


// Deal with showtimes
        var newTimes = [];
        var $movieTimes = $movieDiv.find("div.times > span");

        var am1 = -1, am2 = -1, pm1 = -1;
        $movieTimes.each(function(idx) {
            var timeText = $(this).text();
            var matches = timeText.match(/(\d+):(\d+)\s*(am|pm)?/);
            if (matches[3] === 'am') {
                if (am1 == -1) {
                    am1 = idx;
                } else {
                    am2 = idx;
                }
            } else if (matches[3] === 'pm') {
                pm1 = idx;
            }
        });

        $movieTimes.each(function(idx) {
            var matches = $(this).text().match(/((\d+):(\d+))\s*(am|pm)?/);
            var timeStr = matches[1];
            var momTime = moment();
			if (pm1 == -1) {
				momTime = moment(timeStr + 'am', 'hh:mma');
            } else if (am1 < pm1) {
                if (idx <= am1) {
                    momTime = moment(timeStr + 'am', 'hh:mma');
                } else if (idx > pm1) {
                    momTime = moment(timeStr + 'am', 'hh:mma');
                    momTime.add(1, 'days');
                } else {
                    momTime = moment(timeStr + 'pm', 'hh:mma');
                }
            } else {
                if (idx <= pm1) {
                    momTime = moment(timeStr + 'pm', 'hh:mma');
                } else {
                    momTime = moment(timeStr + 'am', 'hh:mma');
                    momTime.add(1, 'days');
                }
            }
            newTimes.push(momTime);
        });

        var newFilm = new Film(movieName, movieId, movieLength, newTimes);
        newFilm.checkBox = $newCheckbox;
        newFilm.div = $movieDiv;
        newFilms[idx2] = newFilm;

    });

    var newTheater = new Theater(theaterName, theaterId, idx, newFilms);

    theaters.push(newTheater);
    theaterMap[theaterId] = newTheater;
});

function movieDivEnter(event) {
    var id = event.data.id;
    $('table.mdTable td.title_cell, table.mdTable td.time_cell')
        .filter(function () {
            return $(this).data("movieId") === id;
        })
        .addClass('times_table_highlight');
}

function movieDivLeave() {
        $('table.mdTable td.times_table_highlight').removeClass('times_table_highlight');
}

// Runs whenever a checkbox is checked or unchecked
function movieCheckboxChange(event) {
    var $checkBox = $(event.target);
    var theaterId = $checkBox.data("theater");
    var filmNum = $checkBox.data("movie");
    var theater = theaterMap[theaterId];
    var film = theater.films[filmNum];
    var $movieDiv = film.div;

    film.selected = $checkBox.prop("checked");
    if (film.selected) {
        theater.activeFilms.push(film);
        $movieDiv.addClass("mdSelected");
        $movieDiv.mouseenter({'id': film.id}, movieDivEnter);
        $movieDiv.mouseleave(movieDivLeave);
    } else {
        theater.activeFilms = theater.activeFilms.filter(function(item) {
            return item.id !== film.id;
        });
        $movieDiv.removeClass("mdSelected");
        $movieDiv.off("mouseenter");
        $movieDiv.off("mouseleave");
    }

    selectionsChanged(event, theater);
    $movieDiv.mouseenter();
    return false;
}

// Runs whenever there is a change in selected movies (with theater ID) or in the settings (w/o ID)
function selectionsChanged(event, theater) {
    if (theater) {
        checkTheater(theater)
    } else {
        $.each(theaters, function () {
            checkTheater(this);
        });
    }
    return false;
}

// Check theater for selections. If there are enough selections, makeTable.
function checkTheater(theater) {
    if (theater.activeFilms.length >= NUM_DIPS) {
        var $newTable = makeTable(theater.activeFilms, theater.id);

        if ($newTable) {
            $newTable.hide();
            $newTable.children("table").tablesorter({
                theme: 'blue',
                headers: {
                    1: {
                        sorter:'showtime'
                    },
                    3: {
                        sorter:'showtime'
                    },
                    5: {
                        sorter:'showtime'
                    },
                    7: {
                        sorter:'showtime'
                    },
                    9: {
                        sorter:'showtime'
                    },
                    11: {
                        sorter:'showtime'
                    }
                }
            });
            $('#' + theater.id).next().after($newTable);

            if (theater.tableDiv !== null) {
                theater.tableDiv.remove();
                theater.tableDiv = $newTable;
                theater.tableDiv.show();
            } else {
                theater.tableDiv = $newTable;
                theater.tableDiv.show(FADE_SPEED);

            }
        } else if (theater.tableDiv !== null) {
            theater.tableDiv.hide(FADE_SPEED, function() { theater.tableDiv.remove(); theater.tableDiv = null; });
        }
    } else {
        if (theater.tableDiv !== null) {
            theater.tableDiv.hide(FADE_SPEED, function() { theater.tableDiv.remove(); theater.tableDiv = null; });
        }
    }
}

// Figures out which selected films can be multi-dipped
function makeTable(films, tId) {
    log("making table");
    var selected_films = films;
    var matches = [];
    var front_tolerance = parseInt($("#mdFrontTolerance").prop("value"));
    var end_tolerance = parseInt($("#mdEndTolerance").prop("value"));
    var num_dips = parseInt($("#mdNumDips").prop("value"));

    for (var i = 0, len = selected_films.length; i < len; i++) {
        var film1 = selected_films[i];
        var times1 = film1.times;
        var len1 = film1.len;
        for (var j = i + 1; j < len; j++) {
            var film2 = selected_films[j];
            if ( !(film1.title.indexOf(film2.title) == 0 && film1.title.lastIndexOf("3D") > 0) &&
                !(film2.title.indexOf(film1.title) == 0 && film2.title.lastIndexOf("3D") > 0)) {
                var times2 = film2.times;
                var len2 = film2.len;

                times1.forEach(function (val1) {
                    times2.forEach(function (val2) {
                        var match;
                        var diff2 = val1.diff(val2, 'minutes');
                        var diff1 = val2.diff(val1, 'minutes');

                        if (diff1 - len1 >= front_tolerance && diff1 - len1 <= end_tolerance) {
                            match = {
                                m: [film1, film2],
                                t: [val1, val2],
                                wait: [diff1 - len1]
                            };
                            matches.push(match);
                        } else if (diff2 - len2 >= front_tolerance && diff2 - len2 <= end_tolerance) {
                            match = {
                                m: [film2, film1],
                                t: [val2, val1],
                                wait: [diff2 - len2]
                            };
                            matches.push(match);
                        }
                    });
                });
            }
        }
    }

    log('num dips: ' + num_dips);
    var new_matches;
    var dipnum = 2;
    while (dipnum < num_dips && matches.length > 0) {
        new_matches = [];
        $.each(matches, function() {
            var this_match = this;
            $.each(selected_films, function() {
                var film2 = this;
                var nogood = false;
                $.each(this_match.m, function() {
                    if (    this.title === film2.title ||
                            (this.title.indexOf(film2.title) == 0 && this.title.lastIndexOf("3D") > 0) ||
                            (film2.title.indexOf(this.title) == 0 && film2.title.lastIndexOf("3D") > 0)
                    ) {
                        nogood = true;
                        return false; // break
                    }
                });
                if (nogood) return true; // continue

                var film1 = this_match.m[dipnum - 1];
                var val1 = this_match.t[dipnum - 1];
                var len1 = film1.len;
                var times2 = film2.times;

                times2.forEach(function (val2) {
                    var diff1 = val2.diff(val1, 'minutes');

                    if (diff1 - len1 >= front_tolerance && diff1 - len1 <= end_tolerance) {
                        var new_match = {};
                        new_match.m = this_match.m.concat(film2);
                        new_match.t = this_match.t.concat(val2);
                        new_match.wait = this_match.wait.concat(diff1 - len1);
                        new_matches.push(new_match);
                    }
                });
            });
        });

        dipnum++;
        console.log("new matches: " + new_matches.length);
        matches = new_matches;
    }


    if (matches.length > 0) {
        console.log(matches.length + " matches");
        var html = "";

        // Sort the matched pairs of movies by the first movie's starting time.
        matches.sort(function(a, b) {
            var diff = a.t[0].diff(b.t[0], 'minutes');
            return diff;
        });

        // Create the html table
        html = '<div class="mdTableDiv" id="' + tId + 'mdTableDiv"> <table id="' + tId + 'mdTable" class="tablesorter mdTable"> <thead> <tr> ';
        for (var mnum = 1; mnum <= num_dips; mnum++) {
            html += '<th>Movie ' + mnum + '</th> <th>Time ' + mnum + '</th>';
            if (mnum > 1) {
                html += '<th>Wait</th>';
            }
        }
        html += ' </tr> </thead> <tbody>';
        var fmt = 'hh:mm A';

        matches.forEach(function(val) {
            var s = '<tr> ';
            for (mnum = 0; mnum < num_dips; mnum++) {
                s += '<td class="title_cell" data-movie-id="' + val.m[mnum].id + '">' + val.m[mnum].title +
                    '</td> <td class="time_cell" data-movie-id="' + val.m[mnum].id + '" data-sort-num="' +
                    val.t[mnum].valueOf() + '"><b>' + val.t[mnum].format(fmt) + '</b></td> ';
                if (mnum > 0) {
                    s+= '<td class="wait_cell"><b>' + val.wait[mnum - 1] + '</b> min wait</td> ';
                }
            }
            s += '</tr>\n';

            html += s;
        });

        html += "</tbody> </table> </div>";

        return $(html);

    } else return null;
}


// Add custom CSS
var userStyles0 =
    "div.mdDiv { display: inline-block; !important}" +
    "div.movie:hover { background-color: #FFF8D4;}" +
    "a.mdCheck { display: inline-block; border: 2px solid #588220; border-radius: 8px; background-color: #588220; color: #D8FFA8; " +
    "padding: 2px 5px; margin-right: 5px; margin-bottom: 5px; line-height: 24px; !important}" +
    "a.mdCheck:link { text-decoration: none; }" +
    "div.mdSelected { background-color: #CCFFBB; !important}" +
    "div.mdSelected:hover { background-color: #99FF88 }" +
    ".theater .showtimes { margin-bottom: 0px; !important}" +
    "div.theater { margin-bottom: 40px; }" +
    ".mdTable td.wait_cell { color: #888888; !important }" +
    ".mdTable td.times_table_highlight { background-color: #CCFFBB; !important }" +
    "#mdSettingsDiv b { font-size: 1.1em; }" +
    "#mdSettingsDiv input { margin-right: 5px; font-size: 1em; }" +
    "#mdSettingsDiv select { margin-right: 5px; font-size: 1em; width: 4em; }" +
    "";

var userStyles3 = "" +
    ".tablesorter-blue {" +
        "	width: auto;" +
        "	background-color: #fff;" +
        "	margin: 10px 0 15px;" +
        "	text-align: left;" +
        "	border-spacing: 0;" +
        "	border: #cdcdcd 1px solid;" +
        "	border-width: 1px 0 0 1px;" +
        "}" +
        ".tablesorter-blue th," +
        ".tablesorter-blue td {" +
        "	border: #cdcdcd 1px solid;" +
        "	border-width: 0 1px 1px 0;" +
        "}" +
        "" +
        "/* header */" +
        ".tablesorter-blue th," +
        ".tablesorter-blue thead td {" +
        "	font: bold 12px/18px Arial, Sans-serif;" +
        "	color: #000;" +
        "	background-color: #99bfe6;" +
        "	border-collapse: collapse;" +
        "	padding: 6px;" +
        "	text-shadow: 0 1px 0 rgba(204, 204, 204, 0.7);" +
        "}" +
        ".tablesorter-blue tbody td," +
        ".tablesorter-blue tfoot th," +
        ".tablesorter-blue tfoot td {" +
        "	padding: 6px;" +
        "	vertical-align: top;" +
        "}" +
        ".tablesorter-blue .header," +
        ".tablesorter-blue .tablesorter-header {" +
        "	/* black (unsorted) double arrow */" +
        "	background-image: url();" +
        "	/* white (unsorted) double arrow */" +
        "	/* background-image: url(); */" +
        "	/* image */" +
        "	/* background-image: url(images/black-unsorted.gif); */" +
        "	background-repeat: no-repeat;" +
        "	background-position: center right;" +
        "	padding: 6px 18px 6px 6px;" +
        "	white-space: normal;" +
        "	cursor: pointer;" +
        "}" +
        ".tablesorter-blue .headerSortUp," +
        ".tablesorter-blue .tablesorter-headerSortUp," +
        ".tablesorter-blue .tablesorter-headerAsc {" +
        "	background-color: #9fbfdf;" +
        "	/* black asc arrow */" +
        "	background-image: url();" +
        "	/* white asc arrow */" +
        "	/* background-image: url(); */" +
        "	/* image */" +
        "	/* background-image: url(images/black-asc.gif); */" +
        "}" +
        ".tablesorter-blue .headerSortDown," +
        ".tablesorter-blue .tablesorter-headerSortDown," +
        ".tablesorter-blue .tablesorter-headerDesc {" +
        "	background-color: #8cb3d9;" +
        "	/* black desc arrow */" +
        "	background-image: url();" +
        "	/* white desc arrow */" +
        "	/* background-image: url(); */" +
        "	/* image */" +
        "	/* background-image: url(images/black-desc.gif); */" +
        "}" +
        ".tablesorter-blue thead .sorter-false {" +
        "	background-image: none;" +
        "	cursor: default;" +
        "	padding: 6px;" +
        "}" +
        "" +
        "/* tfoot */" +
        ".tablesorter-blue tfoot .tablesorter-headerSortUp," +
        ".tablesorter-blue tfoot .tablesorter-headerSortDown," +
        ".tablesorter-blue tfoot .tablesorter-headerAsc," +
        ".tablesorter-blue tfoot .tablesorter-headerDesc {" +
        "	/* remove sort arrows from footer */" +
        "	background-image: none;" +
        "}" +
        "" +
        "/* tbody */" +
        ".tablesorter-blue td {" +
        "	color: #3d3d3d;" +
        "	background-color: #fff;" +
        "	padding: 6px;" +
        "	vertical-align: top;" +
        "}" +
        "" +
        "/* hovered row colors" +
        " you'll need to add additional lines for" +
        " rows with more than 2 child rows" +
        " */" +
        ".tablesorter-blue tbody > tr:hover > td," +
        ".tablesorter-blue tbody > tr:hover + tr.tablesorter-childRow > td," +
        ".tablesorter-blue tbody > tr:hover + tr.tablesorter-childRow + tr.tablesorter-childRow > td," +
        ".tablesorter-blue tbody > tr.even:hover > td," +
        ".tablesorter-blue tbody > tr.even:hover + tr.tablesorter-childRow > td," +
        ".tablesorter-blue tbody > tr.even:hover + tr.tablesorter-childRow + tr.tablesorter-childRow > td {" +
        "	background: #d9d9d9;" +
        "}" +
        ".tablesorter-blue tbody > tr.odd:hover > td," +
        ".tablesorter-blue tbody > tr.odd:hover + tr.tablesorter-childRow > td," +
        ".tablesorter-blue tbody > tr.odd:hover + tr.tablesorter-childRow + tr.tablesorter-childRow > td {" +
        "	background: #bfbfbf;" +
        "}" +
        "" +
        "/* table processing indicator */" +
        ".tablesorter-blue .tablesorter-processing {" +
        "	background-position: center center !important;" +
        "	background-repeat: no-repeat !important;" +
        "	/* background-image: url(../addons/pager/icons/loading.gif) !important; */" +
        "	background-image: url('') !important;" +
        "}" +
        "" +
        "/* Zebra Widget - row alternating colors */" +
        ".tablesorter-blue tbody tr.odd td {" +
        "	background-color: #ebf2fa;" +
        "}" +
        ".tablesorter-blue tbody tr.even td {" +
        "	background-color: #fff;" +
        "}" +
        "" +
        "/* Column Widget - column sort colors */" +
        ".tablesorter-blue td.primary," +
        ".tablesorter-blue tr.odd td.primary {" +
        "	background-color: #99b3e6;" +
        "}" +
        ".tablesorter-blue tr.even td.primary {" +
        "	background-color: #c2d1f0;" +
        "}" +
        ".tablesorter-blue td.secondary," +
        ".tablesorter-blue tr.odd td.secondary {" +
        "	background-color: #c2d1f0;" +
        "}" +
        ".tablesorter-blue tr.even td.secondary {" +
        "	background-color: #d6e0f5;" +
        "}" +
        ".tablesorter-blue td.tertiary," +
        ".tablesorter-blue tr.odd td.tertiary {" +
        "	background-color: #d6e0f5;" +
        "}" +
        ".tablesorter-blue tr.even td.tertiary {" +
        "	background-color: #ebf0fa;" +
        "}" +
        "";

GM_addStyle(userStyles3);
GM_addStyle(userStyles0);


/// REMOVE GOOGLE LINK TRACKING ///
// http://google.com/url?q=http://www.imdb.com/title/tt2294449/&sa=X&oi=moviesi&ii=0&usg=AFQjCNGscoN77bL-4GPf_VHQGumRZwQtAg
// http://google.com/url?q=http://www.youtube.com/watch%3Fv%3DVQqUOvQKPBg&sa=X&oi=movies&ii=0&usg=AFQjCNE3UMhKGsCv8WI8pgS-LMhnI2e0fA

$('a[href^="/url?q="]').each(function() {
    this.href = this.href.replace(/^https?:\/\/google\.com\/url\?q=/, '');
    this.href = this.href.replace(/&(?:sa|oi|ii|usg)=.*/, '');
    this.href = decodeURIComponent(this.href);
    this.href = this.href.replace(/afid=goog&?/, '');
    this.href = this.href.replace(/source=google&?/, '');
});