├── .jshintrc ├── vendor └── twitter-bootstrap │ ├── img │ ├── glyphicons-halflings.png │ └── glyphicons-halflings-white.png │ ├── css │ ├── bootstrap-responsive.min.css │ └── bootstrap-responsive.css │ └── js │ ├── bootstrap.min.js │ └── bootstrap.js ├── README.md ├── random-ai.js ├── nginx.conf ├── count-ai.js ├── online-game-list.html ├── online.css ├── online.html ├── app.css ├── online-game-detail.html ├── index.html ├── app.js ├── online.js └── othello.js /.jshintrc: -------------------------------------------------------------------------------- 1 | { 2 | "bitwise": false, 3 | "curly": false 4 | } 5 | -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/img/glyphicons-halflings.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kana/othello-js/HEAD/vendor/twitter-bootstrap/img/glyphicons-halflings.png -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/img/glyphicons-halflings-white.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kana/othello-js/HEAD/vendor/twitter-bootstrap/img/glyphicons-halflings-white.png -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Othello JS 2 | 3 | A quick and dirty implementation of [Othello (Reversi)](http://en.wikipedia.org/wiki/Reversi). 4 | [Try it online](https://kana.github.io/othello-js/). 5 | -------------------------------------------------------------------------------- /random-ai.js: -------------------------------------------------------------------------------- 1 | othello.registerAI({ 2 | findTheBestMove: function (gameTree) { 3 | return gameTree.moves[Math.floor(Math.random() * gameTree.moves.length)]; 4 | } 5 | }); 6 | // vim: expandtab softtabstop=2 shiftwidth=2 foldmethod=marker 7 | -------------------------------------------------------------------------------- /nginx.conf: -------------------------------------------------------------------------------- 1 | daemon off; 2 | worker_processes 1; 3 | 4 | events { 5 | worker_connections 1024; 6 | } 7 | 8 | http { 9 | include /usr/local/etc/nginx/mime.types; 10 | default_type application/octet-stream; 11 | 12 | server { 13 | listen 8080; 14 | 15 | location / { 16 | root .; 17 | autoindex on; 18 | } 19 | } 20 | } 21 | 22 | # vim: filetype=conf expandtab softtabstop=2 shiftwidth=2 23 | -------------------------------------------------------------------------------- /count-ai.js: -------------------------------------------------------------------------------- 1 | (function () { 2 | var O = othello; 3 | 4 | function sum(ns) { 5 | return ns.reduce(function (t, n) {return t + n;}); 6 | } 7 | 8 | function scoreBoard(board, player) { 9 | var opponent = O.nextPlayer(player); 10 | return sum($.map(board, function (v) {return v == player;})) - 11 | sum($.map(board, function (v) {return v == opponent;})); 12 | } 13 | 14 | O.registerAI({ 15 | findTheBestMove: function (gameTree) { 16 | var scores = 17 | gameTree.moves.map(function (m) { 18 | return scoreBoard(O.force(m.gameTreePromise).board, gameTree.player); 19 | }); 20 | var maxScore = Math.max.apply(null, scores); 21 | return gameTree.moves[scores.indexOf(maxScore)] 22 | } 23 | }); 24 | })(); 25 | // vim: expandtab softtabstop=2 shiftwidth=2 foldmethod=marker 26 | -------------------------------------------------------------------------------- /online-game-list.html: -------------------------------------------------------------------------------- 1 |
2 | Start a new game 4 |
5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 |
BlackWhiteStateStarted at
Watch{{g.blackId ? '@' + g.blackName : '-'}}{{g.whiteId ? '@' + g.whiteName : '-'}}{{g.state}}{{g.created_at | date:'yyyy-MM-dd HH:mm:ss Z'}}
28 | -------------------------------------------------------------------------------- /online.css: -------------------------------------------------------------------------------- 1 | .block {margin-top: 1em;} 2 | .block:first-child {margin-top: 0em;} 3 | 4 | h1, .profile {display: inline-block;} 5 | .profile {margin-left: 4em; vertical-align: super;} 6 | 7 | #console, #result {text-align: center; position: relative; left: -1em;} 8 | #moves li {display: inline; float: left; margin-right: 1ex;} 9 | #outline > li {list-style-type: none;} 10 | 11 | #game-board > table .header 12 | { 13 | text-align: center; 14 | font-weight: bolder; 15 | padding: 0 1ex; 16 | } 17 | 18 | #game-board > table .cell 19 | { 20 | background: #090; 21 | border: 1px solid #ccc; 22 | padding: 0; 23 | margin: 0; 24 | line-height: 0; 25 | } 26 | #game-board > table.playing .cell.attackable {cursor: pointer;} 27 | #game-board > table.playing .cell.attackable:hover {background: #cc0;} 28 | 29 | #game-board > table .cell > .disc 30 | { 31 | display: inline-block; 32 | width: 2em; 33 | height: 2em; 34 | border-radius: 1em; 35 | margin: 0.25em; 36 | } 37 | #game-board > table .cell.white > .disc {background: #fff;} 38 | #game-board > table .cell.black > .disc {background: #000;} 39 | #game-board > table.playing .cell.attackable > .disc {opacity: 0.2;} 40 | #game-board > table.playing .cell.attackable:hover > .disc {opacity: 1.0;} 41 | -------------------------------------------------------------------------------- /online.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | OthelloOnline 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 |
17 |

OthelloOnline

18 |
19 | 20 | @{{user.name}} 21 | 22 | 23 | 24 | 25 | 26 |
27 |
28 |
29 | 30 | 31 | -------------------------------------------------------------------------------- /app.css: -------------------------------------------------------------------------------- 1 | #main.container 2 | { 3 | text-align: center; 4 | } 5 | 6 | .disabled 7 | { 8 | opacity: 0.5; 9 | } 10 | 11 | #message 12 | { 13 | margin-top: 1em; 14 | } 15 | 16 | #preference-pane 17 | { 18 | margin-top: 2em; 19 | } 20 | 21 | #player-type-forms 22 | { 23 | display: inline-block; 24 | } 25 | #black-player-level, 26 | #white-player-level 27 | { 28 | width: auto; 29 | } 30 | #swap-player-types-button 31 | { 32 | position: relative; 33 | top: -2em; 34 | } 35 | 36 | #add-new-ai-button 37 | { 38 | display: inline-block; 39 | margin-bottom: 10px; 40 | } 41 | 42 | #game-board > table 43 | { 44 | margin: 0 auto; 45 | } 46 | 47 | #game-board > table th 48 | { 49 | margin: 0; 50 | padding: 0.125em 0.25em; 51 | line-height: 100%; 52 | } 53 | 54 | #game-board > table .cell 55 | { 56 | background: #090; 57 | border: 1px solid #ccc; 58 | padding: 0; 59 | margin: 0; 60 | line-height: 0; 61 | } 62 | #game-board > table .cell.attackable 63 | { 64 | cursor: pointer; 65 | } 66 | #game-board > table .cell.attackable:hover 67 | { 68 | background: #cc0; 69 | } 70 | 71 | #game-board > table .cell > .disc 72 | { 73 | display: inline-block; 74 | width: 2em; 75 | height: 2em; 76 | border-radius: 1em; 77 | margin: 0.25em; 78 | } 79 | #game-board > table .cell.white > .disc 80 | { 81 | background: #fff; 82 | } 83 | #game-board > table .cell.black > .disc 84 | { 85 | background: #000; 86 | } 87 | #game-board > table .cell.attackable > .disc 88 | { 89 | opacity: 0.2; 90 | } 91 | #game-board > table .cell.attackable:hover > .disc 92 | { 93 | opacity: 1.0; 94 | } 95 | -------------------------------------------------------------------------------- /online-game-detail.html: -------------------------------------------------------------------------------- 1 |
2 | Return to the game list 3 |
4 | 5 |
6 |
7 |
8 | 9 | 10 | 18 | 19 |
15 | 16 | {{cell.name}} 17 |
20 |
21 |
22 | 23 | The winner is black. 24 | The winner is white. 25 | The game ends in a draw. 26 | 27 |
28 |
29 |
Current player: {{gameTree.player}}
30 |
Choose a move.
31 |
32 | 37 |
38 |
39 |
40 |
41 | 78 |
79 |
80 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | Othello JS 6 | 7 | 8 | 9 | 10 | 11 | 24 | 25 |
26 |
27 |
Current player: -
28 |
29 |
30 |
31 |
32 |
33 | 54 | 75 |
76 | 77 |
78 | 79 | 80 |
81 |
82 | 86 |
87 | 88 |
89 |
90 | 91 | 92 | 93 | 94 | 95 | 96 | 97 | -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | (function (O) { 2 | 'use strict'; 3 | 4 | // UI {{{1 5 | 6 | function drawGameBoard(board, player, moves) { 7 | var ss = []; 8 | var attackable = []; 9 | moves.forEach(function (m) { 10 | if (!m.isPassingMove) 11 | attackable[O.ix(m.x, m.y)] = true; 12 | }); 13 | 14 | ss.push(''); 15 | for (var y = -1; y < O.N; y++) { 16 | ss.push(''); 17 | for (var x = -1; x < O.N; x++) { 18 | if (0 <= y && 0 <= x) { 19 | ss.push(''); 30 | } else if (0 <= x && y === -1) { 31 | ss.push(''); 32 | } else if (x === -1 && 0 <= y) { 33 | ss.push(''); 34 | } else /* if (x === -1 && y === -1) */ { 35 | ss.push(''); 36 | } 37 | } 38 | ss.push(''); 39 | } 40 | ss.push('
'); 28 | ss.push(''); 29 | ss.push('' + String.fromCharCode('a'.charCodeAt(0)+x) + '' + (y + 1) + '
'); 41 | 42 | $('#game-board').html(ss.join('')); 43 | $('#current-player-name').text(player); 44 | } 45 | 46 | function resetUI() { 47 | $('#console').empty(); 48 | $('#message').empty(); 49 | } 50 | 51 | function setUpUIToChooseMove(gameTree) { 52 | $('#message').text('Choose your move.'); 53 | gameTree.moves.forEach(function (m, i) { 54 | if (m.isPassingMove) { 55 | $('#console').append( 56 | $('') 57 | .val(O.nameMove(m)) 58 | .click(function () { 59 | shiftToNewGameTree(O.force(m.gameTreePromise)); 60 | }) 61 | ); 62 | } else { 63 | $('#cell_' + m.x + '_' + m.y) 64 | .click(function () { 65 | shiftToNewGameTree(O.force(m.gameTreePromise)); 66 | }); 67 | } 68 | }); 69 | } 70 | 71 | function setUpUIToReset() { 72 | resetGame(); 73 | if ($('#repeat-games:checked').length) 74 | startNewGame(); 75 | } 76 | 77 | var minimumDelayForAI = 500; // milliseconds 78 | function chooseMoveByAI(gameTree, ai) { 79 | $('#message').text('Now thinking...'); 80 | setTimeout( 81 | function () { 82 | var start = Date.now(); 83 | var newGameTree = O.force(ai.findTheBestMove(gameTree).gameTreePromise); 84 | var end = Date.now(); 85 | var delta = end - start; 86 | setTimeout( 87 | function () { 88 | shiftToNewGameTree(newGameTree); 89 | }, 90 | Math.max(minimumDelayForAI - delta, 1) 91 | ); 92 | }, 93 | 1 94 | ); 95 | } 96 | 97 | function showWinner(board) { 98 | var r = O.judge(board); 99 | $('#message').text( 100 | r === 0 ? 101 | 'The game ends in a draw.' : 102 | 'The winner is ' + (r === 1 ? O.BLACK : O.WHITE) + '.' 103 | ); 104 | } 105 | 106 | var playerTable = {}; 107 | 108 | function makePlayer(playerType) { 109 | if (playerType === 'human') { 110 | return setUpUIToChooseMove; 111 | } else { 112 | var ai = O.makeAI(playerType); 113 | return function (gameTree) { 114 | chooseMoveByAI(gameTree, ai); 115 | }; 116 | } 117 | } 118 | 119 | function blackPlayerType() { 120 | return $('#black-player-type').val(); 121 | } 122 | 123 | function whitePlayerType() { 124 | return $('#white-player-type').val(); 125 | } 126 | 127 | function swapPlayerTypes() { 128 | var t = $('#black-player-type').val(); 129 | $('#black-player-type').val($('#white-player-type').val()).change(); 130 | $('#white-player-type').val(t).change(); 131 | } 132 | 133 | function shiftToNewGameTree(gameTree) { 134 | drawGameBoard(gameTree.board, gameTree.player, gameTree.moves); 135 | resetUI(); 136 | if (gameTree.moves.length === 0) { 137 | showWinner(gameTree.board); 138 | recordStat(gameTree.board); 139 | if ($('#repeat-games:checked').length) 140 | showStat(); 141 | setUpUIToReset(); 142 | } else { 143 | playerTable[gameTree.player](gameTree); 144 | } 145 | } 146 | 147 | var stats = {}; 148 | 149 | function recordStat(board) { 150 | var s = stats[[blackPlayerType(), whitePlayerType()]] || {b: 0, w: 0, d: 0}; 151 | var r = O.judge(board); 152 | if (r === 1) 153 | s.b++; 154 | if (r === 0) 155 | s.d++; 156 | if (r === -1) 157 | s.w++; 158 | stats[[blackPlayerType(), whitePlayerType()]] = s; 159 | } 160 | 161 | function showStat() { 162 | var s = stats[[blackPlayerType(), whitePlayerType()]]; 163 | $('#stats').text('Black: ' + s.b + ', White: ' + s.w + ', Draw: ' + s.d); 164 | } 165 | 166 | function resetGame() { 167 | $('#preference-pane :input:not(#repeat-games)') 168 | .removeClass('disabled') 169 | .removeAttr('disabled'); 170 | } 171 | 172 | function startNewGame() { 173 | $('#preference-pane :input:not(#repeat-games)') 174 | .addClass('disabled') 175 | .attr('disabled', 'disabled'); 176 | playerTable[O.BLACK] = makePlayer(blackPlayerType()); 177 | playerTable[O.WHITE] = makePlayer(whitePlayerType()); 178 | shiftToNewGameTree(O.makeInitialGameTree()); 179 | } 180 | 181 | 182 | 183 | 184 | // Startup {{{1 185 | 186 | $('#start-button').click(function () {startNewGame();}); 187 | $('#add-new-ai-button').click(function () {O.addNewAI();}); 188 | $('#swap-player-types-button').click(function () {swapPlayerTypes();}); 189 | resetGame(); 190 | drawGameBoard(O.makeInitialGameBoard(), '-', []); 191 | 192 | 193 | 194 | 195 | //}}} 196 | })(othello); 197 | // vim: expandtab softtabstop=2 shiftwidth=2 foldmethod=marker 198 | -------------------------------------------------------------------------------- /online.js: -------------------------------------------------------------------------------- 1 | angular.module('OthelloOnline', ['ngRoute', 'firebase']) 2 | .value('fbUrl', 'https://othelloonline.firebaseio.com/') 3 | .service('fbRef', function (fbUrl) { 4 | return new Firebase(fbUrl) 5 | }) 6 | .service('fbAuth', function ($q, $firebase, $firebaseAuth, fbRef) { 7 | function simplifyAuthData(auth) { 8 | return { 9 | id: auth.uid, 10 | name: auth.twitter.username 11 | }; 12 | } 13 | var user; 14 | return function (mode) { 15 | if (mode === 'signOut') { 16 | user = null; 17 | $firebaseAuth(fbRef).$unauth(); 18 | return; 19 | } 20 | 21 | if (user) 22 | return $q.when(user); 23 | 24 | var authObj = $firebaseAuth(fbRef); 25 | var auth = authObj.$getAuth(); 26 | if (auth) { 27 | user = simplifyAuthData(auth); 28 | return $q.when(user); 29 | } 30 | 31 | if (mode === 'check') 32 | return $q.when(null); 33 | 34 | var deferred = $q.defer(); 35 | authObj.$authWithOAuthPopup('twitter').then(function (auth) { 36 | user = simplifyAuthData(auth); 37 | deferred.resolve(user); 38 | }).catch(function (error) { 39 | console.log(error); 40 | alert(error); 41 | }); 42 | return deferred.promise; 43 | } 44 | }) 45 | .service('GameOutlines', function ($q, fbFetch) { 46 | var self = this; 47 | self.fetch = function () { 48 | if (self.gameOutlines) 49 | return $q.when(self.gameOutlines); 50 | 51 | return fbFetch('gameOutlines', 'Array').then(function (data) { 52 | self.gameOutlines = data; 53 | return data; 54 | }); 55 | }; 56 | }) 57 | .service('fbFetch', function ($q, $firebase, fbRef) { 58 | return function (path, opt_type) { 59 | var type = opt_type || 'Object'; 60 | var deferred = $q.defer(); 61 | var ref = fbRef.child(path); 62 | var $ref = $firebase(ref); 63 | ref.on('value', function (snapshot) { 64 | deferred.resolve($ref['$as' + type]()); 65 | }); 66 | return deferred.promise; 67 | }; 68 | }) 69 | .service('fbCache', function ($q, fbFetch) { 70 | var memo = {}; 71 | return function (path, opt_type) { 72 | if (memo[path]) 73 | return $q.when(memo[path]); 74 | return fbFetch(path, opt_type).then(function (data) { 75 | memo[path] = data; 76 | return data; 77 | }); 78 | }; 79 | }) 80 | .config(function ($routeProvider) { 81 | $routeProvider 82 | .when('/games', { 83 | controller: 'GameList', 84 | templateUrl: 'online-game-list.html', 85 | resolve: { 86 | gameOutlines: function (GameOutlines) { 87 | return GameOutlines.fetch(); 88 | } 89 | } 90 | }) 91 | .when('/games/new', { 92 | controller: 'GameCreation', 93 | templateUrl: 'online-game-detail.html' 94 | }) 95 | .when('/games/:gameId', { 96 | controller: 'GameDetail', 97 | templateUrl: 'online-game-detail.html', 98 | resolve: { 99 | gameOutline: function ($route, fbFetch) { 100 | return fbFetch('gameOutlines/' + $route.current.params.gameId); 101 | }, 102 | moves: function ($route, fbFetch) { 103 | return fbFetch( 104 | 'gameDetails/' + $route.current.params.gameId + '/moves', 105 | 'Array' 106 | ); 107 | } 108 | } 109 | }) 110 | .otherwise({ 111 | redirectTo: '/games' 112 | }); 113 | }) 114 | .controller('Base', function ($scope, fbAuth) { 115 | function fetchAndBindUser(user) { 116 | $scope.user = user; 117 | return user; 118 | } 119 | fbAuth('check').then(fetchAndBindUser); 120 | $scope.signIn = function () { 121 | return fbAuth('signIn').then(fetchAndBindUser); 122 | }; 123 | $scope.signOut = function () { 124 | $scope.user = null; 125 | return fbAuth('signOut'); 126 | }; 127 | }) 128 | .controller('GameList', function ($scope, gameOutlines) { 129 | $scope.games = gameOutlines; 130 | }) 131 | .controller('GameCreation', function ($scope, fbRef, $location) { 132 | // NB: Firebase does not save "empty" objects by design. 133 | var go = fbRef.child('gameOutlines').push({ 134 | // blackId: null, 135 | // blackName: null, 136 | // whiteId: null, 137 | // whiteName: null, 138 | state: 'preparing', 139 | created_at: Firebase.ServerValue.TIMESTAMP 140 | }); 141 | // var gd = fbRef.child('gameDetails').child(go.key()).set({ 142 | // moves: [] 143 | // }); 144 | $location.path('/games/' + go.key()); 145 | }) 146 | .controller('GameDetail', function ($scope, gameOutline, moves) { 147 | var O = othello; 148 | $scope.O = othello; 149 | 150 | // gameDetails/$game_id/moves is directly watched, because it is troublesome 151 | // to deal with empty moves by watching gameDetails/$game_id. 152 | $scope.outline = gameOutline; 153 | $scope.moves = moves; 154 | 155 | $scope.join = function (color) { 156 | $scope.signIn().then(function (user) { 157 | $scope.outline[color + 'Id'] = user.id; 158 | $scope.outline[color + 'Name'] = user.name; 159 | if ($scope.outline.blackId && $scope.outline.whiteId) 160 | $scope.outline.state = 'playing'; 161 | $scope.outline.$save(); 162 | }); 163 | }; 164 | $scope.leave = function (color) { 165 | $scope.outline[color + 'Id'] = null; 166 | $scope.outline[color + 'Name'] = null; 167 | $scope.outline.$save(); 168 | }; 169 | // TODO: Add UI to replay the game if it is finished. 170 | 171 | function play(moveName) { 172 | var validMoveNames = 173 | $scope.gameTree.moves.map(function (m) {return O.nameMove(m);}); 174 | var i = validMoveNames.indexOf(moveName); 175 | if (0 <= i) { 176 | $scope.gameTree = O.force($scope.gameTree.moves[i].gameTreePromise); 177 | } else { 178 | throw new Error( 179 | 'Error: Unexpected move "' + moveName + '" is chosen\n' + 180 | 'but valid moves are ' + validMoveNames.join(', ') + '.' 181 | ); 182 | } 183 | } 184 | function visualizedBoardFrom(gameTree) { 185 | var player = gameTree.player; 186 | var board = gameTree.board; 187 | var attackable = []; 188 | gameTree.moves.forEach(function (m) { 189 | if (!m.isPassingMove) 190 | attackable[O.ix(m.x, m.y)] = m; 191 | }); 192 | 193 | var newBoard = []; 194 | for (var y = -1; y < O.N; y++) { 195 | var row = []; 196 | for (var x = -1; x < O.N; x++) { 197 | if (0 <= y && 0 <= x) { 198 | var a = attackable[O.ix(x, y)]; 199 | var b = board[O.ix(x, y)]; 200 | row.push({ 201 | cell: true, 202 | black: player === O.BLACK && a || b == O.BLACK, 203 | white: player === O.WHITE && a || b == O.WHITE, 204 | attackable: a 205 | }); 206 | } else if (0 <= x && y === -1) { 207 | row.push({ 208 | header: true, 209 | name: String.fromCharCode('a'.charCodeAt(0) + x) 210 | }); 211 | } else if (x === -1 && 0 <= y) { 212 | row.push({ 213 | header: true, 214 | name: y + 1 215 | }); 216 | } else /* if (x === -1 && y === -1) */ { 217 | row.push({}); 218 | } 219 | } 220 | newBoard.push(row); 221 | } 222 | return newBoard; 223 | } 224 | 225 | $scope.$watch('gameTree', function () { 226 | $scope.visualizedBoard = visualizedBoardFrom($scope.gameTree); 227 | }); 228 | $scope.gameTree = O.makeInitialGameTree(); 229 | $scope.moves.$watch(function (e) { 230 | if (e.event == 'child_added') { 231 | play($scope.moves[$scope.moves.length - 1].$value); 232 | } 233 | }); 234 | 235 | $scope.isMyTurn = function () { 236 | return $scope.user.id === $scope.outline[$scope.gameTree.player + 'Id']; 237 | }; 238 | $scope.isPassingMove = function (move) { 239 | return move.isPassingMove; 240 | }; 241 | $scope.choose = function (move) { 242 | var moveName = O.nameMove(move); 243 | play(moveName); 244 | $scope.moves.$add(moveName); 245 | if ($scope.gameTree.moves.length === 0) { 246 | $scope.outline.state = 'finished'; 247 | $scope.outline.$save(); 248 | } 249 | }; 250 | }); 251 | // vim: expandtab softtabstop=2 shiftwidth=2 foldmethod=marker 252 | -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/css/bootstrap-responsive.min.css: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap Responsive v2.3.1 3 | * 4 | * Copyright 2012 Twitter, Inc 5 | * Licensed under the Apache License v2.0 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * 8 | * Designed and built with all the love in the world @twitter by @mdo and @fat. 9 | */.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;line-height:0;content:""}.clearfix:after{clear:both}.hide-text{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.input-block-level{display:block;width:100%;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}@-ms-viewport{width:device-width}.hidden{display:none;visibility:hidden}.visible-phone{display:none!important}.visible-tablet{display:none!important}.hidden-desktop{display:none!important}.visible-desktop{display:inherit!important}@media(min-width:768px) and (max-width:979px){.hidden-desktop{display:inherit!important}.visible-desktop{display:none!important}.visible-tablet{display:inherit!important}.hidden-tablet{display:none!important}}@media(max-width:767px){.hidden-desktop{display:inherit!important}.visible-desktop{display:none!important}.visible-phone{display:inherit!important}.hidden-phone{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:inherit!important}.hidden-print{display:none!important}}@media(min-width:1200px){.row{margin-left:-30px;*zoom:1}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:30px}.container,.navbar-static-top .container,.navbar-fixed-top .container,.navbar-fixed-bottom .container{width:1170px}.span12{width:1170px}.span11{width:1070px}.span10{width:970px}.span9{width:870px}.span8{width:770px}.span7{width:670px}.span6{width:570px}.span5{width:470px}.span4{width:370px}.span3{width:270px}.span2{width:170px}.span1{width:70px}.offset12{margin-left:1230px}.offset11{margin-left:1130px}.offset10{margin-left:1030px}.offset9{margin-left:930px}.offset8{margin-left:830px}.offset7{margin-left:730px}.offset6{margin-left:630px}.offset5{margin-left:530px}.offset4{margin-left:430px}.offset3{margin-left:330px}.offset2{margin-left:230px}.offset1{margin-left:130px}.row-fluid{width:100%;*zoom:1}.row-fluid:before,.row-fluid:after{display:table;line-height:0;content:""}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;float:left;width:100%;min-height:30px;margin-left:2.564102564102564%;*margin-left:2.5109110747408616%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .controls-row [class*="span"]+[class*="span"]{margin-left:2.564102564102564%}.row-fluid .span12{width:100%;*width:99.94680851063829%}.row-fluid .span11{width:91.45299145299145%;*width:91.39979996362975%}.row-fluid .span10{width:82.90598290598291%;*width:82.8527914166212%}.row-fluid .span9{width:74.35897435897436%;*width:74.30578286961266%}.row-fluid .span8{width:65.81196581196582%;*width:65.75877432260411%}.row-fluid .span7{width:57.26495726495726%;*width:57.21176577559556%}.row-fluid .span6{width:48.717948717948715%;*width:48.664757228587014%}.row-fluid .span5{width:40.17094017094017%;*width:40.11774868157847%}.row-fluid .span4{width:31.623931623931625%;*width:31.570740134569924%}.row-fluid .span3{width:23.076923076923077%;*width:23.023731587561375%}.row-fluid .span2{width:14.52991452991453%;*width:14.476723040552828%}.row-fluid .span1{width:5.982905982905983%;*width:5.929714493544281%}.row-fluid .offset12{margin-left:105.12820512820512%;*margin-left:105.02182214948171%}.row-fluid .offset12:first-child{margin-left:102.56410256410257%;*margin-left:102.45771958537915%}.row-fluid .offset11{margin-left:96.58119658119658%;*margin-left:96.47481360247316%}.row-fluid .offset11:first-child{margin-left:94.01709401709402%;*margin-left:93.91071103837061%}.row-fluid .offset10{margin-left:88.03418803418803%;*margin-left:87.92780505546462%}.row-fluid .offset10:first-child{margin-left:85.47008547008548%;*margin-left:85.36370249136206%}.row-fluid .offset9{margin-left:79.48717948717949%;*margin-left:79.38079650845607%}.row-fluid .offset9:first-child{margin-left:76.92307692307693%;*margin-left:76.81669394435352%}.row-fluid .offset8{margin-left:70.94017094017094%;*margin-left:70.83378796144753%}.row-fluid .offset8:first-child{margin-left:68.37606837606839%;*margin-left:68.26968539734497%}.row-fluid .offset7{margin-left:62.393162393162385%;*margin-left:62.28677941443899%}.row-fluid .offset7:first-child{margin-left:59.82905982905982%;*margin-left:59.72267685033642%}.row-fluid .offset6{margin-left:53.84615384615384%;*margin-left:53.739770867430444%}.row-fluid .offset6:first-child{margin-left:51.28205128205128%;*margin-left:51.175668303327875%}.row-fluid .offset5{margin-left:45.299145299145295%;*margin-left:45.1927623204219%}.row-fluid .offset5:first-child{margin-left:42.73504273504273%;*margin-left:42.62865975631933%}.row-fluid .offset4{margin-left:36.75213675213675%;*margin-left:36.645753773413354%}.row-fluid .offset4:first-child{margin-left:34.18803418803419%;*margin-left:34.081651209310785%}.row-fluid .offset3{margin-left:28.205128205128204%;*margin-left:28.0987452264048%}.row-fluid .offset3:first-child{margin-left:25.641025641025642%;*margin-left:25.53464266230224%}.row-fluid .offset2{margin-left:19.65811965811966%;*margin-left:19.551736679396257%}.row-fluid .offset2:first-child{margin-left:17.094017094017094%;*margin-left:16.98763411529369%}.row-fluid .offset1{margin-left:11.11111111111111%;*margin-left:11.004728132387708%}.row-fluid .offset1:first-child{margin-left:8.547008547008547%;*margin-left:8.440625568285142%}input,textarea,.uneditable-input{margin-left:0}.controls-row [class*="span"]+[class*="span"]{margin-left:30px}input.span12,textarea.span12,.uneditable-input.span12{width:1156px}input.span11,textarea.span11,.uneditable-input.span11{width:1056px}input.span10,textarea.span10,.uneditable-input.span10{width:956px}input.span9,textarea.span9,.uneditable-input.span9{width:856px}input.span8,textarea.span8,.uneditable-input.span8{width:756px}input.span7,textarea.span7,.uneditable-input.span7{width:656px}input.span6,textarea.span6,.uneditable-input.span6{width:556px}input.span5,textarea.span5,.uneditable-input.span5{width:456px}input.span4,textarea.span4,.uneditable-input.span4{width:356px}input.span3,textarea.span3,.uneditable-input.span3{width:256px}input.span2,textarea.span2,.uneditable-input.span2{width:156px}input.span1,textarea.span1,.uneditable-input.span1{width:56px}.thumbnails{margin-left:-30px}.thumbnails>li{margin-left:30px}.row-fluid .thumbnails{margin-left:0}}@media(min-width:768px) and (max-width:979px){.row{margin-left:-20px;*zoom:1}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:20px}.container,.navbar-static-top .container,.navbar-fixed-top .container,.navbar-fixed-bottom .container{width:724px}.span12{width:724px}.span11{width:662px}.span10{width:600px}.span9{width:538px}.span8{width:476px}.span7{width:414px}.span6{width:352px}.span5{width:290px}.span4{width:228px}.span3{width:166px}.span2{width:104px}.span1{width:42px}.offset12{margin-left:764px}.offset11{margin-left:702px}.offset10{margin-left:640px}.offset9{margin-left:578px}.offset8{margin-left:516px}.offset7{margin-left:454px}.offset6{margin-left:392px}.offset5{margin-left:330px}.offset4{margin-left:268px}.offset3{margin-left:206px}.offset2{margin-left:144px}.offset1{margin-left:82px}.row-fluid{width:100%;*zoom:1}.row-fluid:before,.row-fluid:after{display:table;line-height:0;content:""}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;float:left;width:100%;min-height:30px;margin-left:2.7624309392265194%;*margin-left:2.709239449864817%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .controls-row [class*="span"]+[class*="span"]{margin-left:2.7624309392265194%}.row-fluid .span12{width:100%;*width:99.94680851063829%}.row-fluid .span11{width:91.43646408839778%;*width:91.38327259903608%}.row-fluid .span10{width:82.87292817679558%;*width:82.81973668743387%}.row-fluid .span9{width:74.30939226519337%;*width:74.25620077583166%}.row-fluid .span8{width:65.74585635359117%;*width:65.69266486422946%}.row-fluid .span7{width:57.18232044198895%;*width:57.12912895262725%}.row-fluid .span6{width:48.61878453038674%;*width:48.56559304102504%}.row-fluid .span5{width:40.05524861878453%;*width:40.00205712942283%}.row-fluid .span4{width:31.491712707182323%;*width:31.43852121782062%}.row-fluid .span3{width:22.92817679558011%;*width:22.87498530621841%}.row-fluid .span2{width:14.3646408839779%;*width:14.311449394616199%}.row-fluid .span1{width:5.801104972375691%;*width:5.747913483013988%}.row-fluid .offset12{margin-left:105.52486187845304%;*margin-left:105.41847889972962%}.row-fluid .offset12:first-child{margin-left:102.76243093922652%;*margin-left:102.6560479605031%}.row-fluid .offset11{margin-left:96.96132596685082%;*margin-left:96.8549429881274%}.row-fluid .offset11:first-child{margin-left:94.1988950276243%;*margin-left:94.09251204890089%}.row-fluid .offset10{margin-left:88.39779005524862%;*margin-left:88.2914070765252%}.row-fluid .offset10:first-child{margin-left:85.6353591160221%;*margin-left:85.52897613729868%}.row-fluid .offset9{margin-left:79.8342541436464%;*margin-left:79.72787116492299%}.row-fluid .offset9:first-child{margin-left:77.07182320441989%;*margin-left:76.96544022569647%}.row-fluid .offset8{margin-left:71.2707182320442%;*margin-left:71.16433525332079%}.row-fluid .offset8:first-child{margin-left:68.50828729281768%;*margin-left:68.40190431409427%}.row-fluid .offset7{margin-left:62.70718232044199%;*margin-left:62.600799341718584%}.row-fluid .offset7:first-child{margin-left:59.94475138121547%;*margin-left:59.838368402492065%}.row-fluid .offset6{margin-left:54.14364640883978%;*margin-left:54.037263430116376%}.row-fluid .offset6:first-child{margin-left:51.38121546961326%;*margin-left:51.27483249088986%}.row-fluid .offset5{margin-left:45.58011049723757%;*margin-left:45.47372751851417%}.row-fluid .offset5:first-child{margin-left:42.81767955801105%;*margin-left:42.71129657928765%}.row-fluid .offset4{margin-left:37.01657458563536%;*margin-left:36.91019160691196%}.row-fluid .offset4:first-child{margin-left:34.25414364640884%;*margin-left:34.14776066768544%}.row-fluid .offset3{margin-left:28.45303867403315%;*margin-left:28.346655695309746%}.row-fluid .offset3:first-child{margin-left:25.69060773480663%;*margin-left:25.584224756083227%}.row-fluid .offset2{margin-left:19.88950276243094%;*margin-left:19.783119783707537%}.row-fluid .offset2:first-child{margin-left:17.12707182320442%;*margin-left:17.02068884448102%}.row-fluid .offset1{margin-left:11.32596685082873%;*margin-left:11.219583872105325%}.row-fluid .offset1:first-child{margin-left:8.56353591160221%;*margin-left:8.457152932878806%}input,textarea,.uneditable-input{margin-left:0}.controls-row [class*="span"]+[class*="span"]{margin-left:20px}input.span12,textarea.span12,.uneditable-input.span12{width:710px}input.span11,textarea.span11,.uneditable-input.span11{width:648px}input.span10,textarea.span10,.uneditable-input.span10{width:586px}input.span9,textarea.span9,.uneditable-input.span9{width:524px}input.span8,textarea.span8,.uneditable-input.span8{width:462px}input.span7,textarea.span7,.uneditable-input.span7{width:400px}input.span6,textarea.span6,.uneditable-input.span6{width:338px}input.span5,textarea.span5,.uneditable-input.span5{width:276px}input.span4,textarea.span4,.uneditable-input.span4{width:214px}input.span3,textarea.span3,.uneditable-input.span3{width:152px}input.span2,textarea.span2,.uneditable-input.span2{width:90px}input.span1,textarea.span1,.uneditable-input.span1{width:28px}}@media(max-width:767px){body{padding-right:20px;padding-left:20px}.navbar-fixed-top,.navbar-fixed-bottom,.navbar-static-top{margin-right:-20px;margin-left:-20px}.container-fluid{padding:0}.dl-horizontal dt{float:none;width:auto;clear:none;text-align:left}.dl-horizontal dd{margin-left:0}.container{width:auto}.row-fluid{width:100%}.row,.thumbnails{margin-left:0}.thumbnails>li{float:none;margin-left:0}[class*="span"],.uneditable-input[class*="span"],.row-fluid [class*="span"]{display:block;float:none;width:100%;margin-left:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.span12,.row-fluid .span12{width:100%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="offset"]:first-child{margin-left:0}.input-large,.input-xlarge,.input-xxlarge,input[class*="span"],select[class*="span"],textarea[class*="span"],.uneditable-input{display:block;width:100%;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.input-prepend input,.input-append input,.input-prepend input[class*="span"],.input-append input[class*="span"]{display:inline-block;width:auto}.controls-row [class*="span"]+[class*="span"]{margin-left:0}.modal{position:fixed;top:20px;right:20px;left:20px;width:auto;margin:0}.modal.fade{top:-100px}.modal.fade.in{top:20px}}@media(max-width:480px){.nav-collapse{-webkit-transform:translate3d(0,0,0)}.page-header h1 small{display:block;line-height:20px}input[type="checkbox"],input[type="radio"]{border:1px solid #ccc}.form-horizontal .control-label{float:none;width:auto;padding-top:0;text-align:left}.form-horizontal .controls{margin-left:0}.form-horizontal .control-list{padding-top:0}.form-horizontal .form-actions{padding-right:10px;padding-left:10px}.media .pull-left,.media .pull-right{display:block;float:none;margin-bottom:10px}.media-object{margin-right:0;margin-left:0}.modal{top:10px;right:10px;left:10px}.modal-header .close{padding:10px;margin:-10px}.carousel-caption{position:static}}@media(max-width:979px){body{padding-top:0}.navbar-fixed-top,.navbar-fixed-bottom{position:static}.navbar-fixed-top{margin-bottom:20px}.navbar-fixed-bottom{margin-top:20px}.navbar-fixed-top .navbar-inner,.navbar-fixed-bottom .navbar-inner{padding:5px}.navbar .container{width:auto;padding:0}.navbar .brand{padding-right:10px;padding-left:10px;margin:0 0 0 -5px}.nav-collapse{clear:both}.nav-collapse .nav{float:none;margin:0 0 10px}.nav-collapse .nav>li{float:none}.nav-collapse .nav>li>a{margin-bottom:2px}.nav-collapse .nav>.divider-vertical{display:none}.nav-collapse .nav .nav-header{color:#777;text-shadow:none}.nav-collapse .nav>li>a,.nav-collapse .dropdown-menu a{padding:9px 15px;font-weight:bold;color:#777;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px}.nav-collapse .btn{padding:4px 10px 4px;font-weight:normal;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px}.nav-collapse .dropdown-menu li+li a{margin-bottom:2px}.nav-collapse .nav>li>a:hover,.nav-collapse .nav>li>a:focus,.nav-collapse .dropdown-menu a:hover,.nav-collapse .dropdown-menu a:focus{background-color:#f2f2f2}.navbar-inverse .nav-collapse .nav>li>a,.navbar-inverse .nav-collapse .dropdown-menu a{color:#999}.navbar-inverse .nav-collapse .nav>li>a:hover,.navbar-inverse .nav-collapse .nav>li>a:focus,.navbar-inverse .nav-collapse .dropdown-menu a:hover,.navbar-inverse .nav-collapse .dropdown-menu a:focus{background-color:#111}.nav-collapse.in .btn-group{padding:0;margin-top:5px}.nav-collapse .dropdown-menu{position:static;top:auto;left:auto;display:none;float:none;max-width:none;padding:0;margin:0 15px;background-color:transparent;border:0;-webkit-border-radius:0;-moz-border-radius:0;border-radius:0;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none}.nav-collapse .open>.dropdown-menu{display:block}.nav-collapse .dropdown-menu:before,.nav-collapse .dropdown-menu:after{display:none}.nav-collapse .dropdown-menu .divider{display:none}.nav-collapse .nav>li>.dropdown-menu:before,.nav-collapse .nav>li>.dropdown-menu:after{display:none}.nav-collapse .navbar-form,.nav-collapse .navbar-search{float:none;padding:10px 15px;margin:10px 0;border-top:1px solid #f2f2f2;border-bottom:1px solid #f2f2f2;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);-moz-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1)}.navbar-inverse .nav-collapse .navbar-form,.navbar-inverse .nav-collapse .navbar-search{border-top-color:#111;border-bottom-color:#111}.navbar .nav-collapse .nav.pull-right{float:none;margin-left:0}.nav-collapse,.nav-collapse.collapse{height:0;overflow:hidden}.navbar .btn-navbar{display:block}.navbar-static .navbar-inner{padding-right:10px;padding-left:10px}}@media(min-width:980px){.nav-collapse.collapse{height:auto!important;overflow:visible!important}} 10 | -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/js/bootstrap.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap.js by @fat & @mdo 3 | * Copyright 2012 Twitter, Inc. 4 | * http://www.apache.org/licenses/LICENSE-2.0.txt 5 | */ 6 | !function(e){"use strict";e(function(){e.support.transition=function(){var e=function(){var e=document.createElement("bootstrap"),t={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"},n;for(n in t)if(e.style[n]!==undefined)return t[n]}();return e&&{end:e}}()})}(window.jQuery),!function(e){"use strict";var t='[data-dismiss="alert"]',n=function(n){e(n).on("click",t,this.close)};n.prototype.close=function(t){function s(){i.trigger("closed").remove()}var n=e(this),r=n.attr("data-target"),i;r||(r=n.attr("href"),r=r&&r.replace(/.*(?=#[^\s]*$)/,"")),i=e(r),t&&t.preventDefault(),i.length||(i=n.hasClass("alert")?n:n.parent()),i.trigger(t=e.Event("close"));if(t.isDefaultPrevented())return;i.removeClass("in"),e.support.transition&&i.hasClass("fade")?i.on(e.support.transition.end,s):s()};var r=e.fn.alert;e.fn.alert=function(t){return this.each(function(){var r=e(this),i=r.data("alert");i||r.data("alert",i=new n(this)),typeof t=="string"&&i[t].call(r)})},e.fn.alert.Constructor=n,e.fn.alert.noConflict=function(){return e.fn.alert=r,this},e(document).on("click.alert.data-api",t,n.prototype.close)}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.button.defaults,n)};t.prototype.setState=function(e){var t="disabled",n=this.$element,r=n.data(),i=n.is("input")?"val":"html";e+="Text",r.resetText||n.data("resetText",n[i]()),n[i](r[e]||this.options[e]),setTimeout(function(){e=="loadingText"?n.addClass(t).attr(t,t):n.removeClass(t).removeAttr(t)},0)},t.prototype.toggle=function(){var e=this.$element.closest('[data-toggle="buttons-radio"]');e&&e.find(".active").removeClass("active"),this.$element.toggleClass("active")};var n=e.fn.button;e.fn.button=function(n){return this.each(function(){var r=e(this),i=r.data("button"),s=typeof n=="object"&&n;i||r.data("button",i=new t(this,s)),n=="toggle"?i.toggle():n&&i.setState(n)})},e.fn.button.defaults={loadingText:"loading..."},e.fn.button.Constructor=t,e.fn.button.noConflict=function(){return e.fn.button=n,this},e(document).on("click.button.data-api","[data-toggle^=button]",function(t){var n=e(t.target);n.hasClass("btn")||(n=n.closest(".btn")),n.button("toggle")})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.$indicators=this.$element.find(".carousel-indicators"),this.options=n,this.options.pause=="hover"&&this.$element.on("mouseenter",e.proxy(this.pause,this)).on("mouseleave",e.proxy(this.cycle,this))};t.prototype={cycle:function(t){return t||(this.paused=!1),this.interval&&clearInterval(this.interval),this.options.interval&&!this.paused&&(this.interval=setInterval(e.proxy(this.next,this),this.options.interval)),this},getActiveIndex:function(){return this.$active=this.$element.find(".item.active"),this.$items=this.$active.parent().children(),this.$items.index(this.$active)},to:function(t){var n=this.getActiveIndex(),r=this;if(t>this.$items.length-1||t<0)return;return this.sliding?this.$element.one("slid",function(){r.to(t)}):n==t?this.pause().cycle():this.slide(t>n?"next":"prev",e(this.$items[t]))},pause:function(t){return t||(this.paused=!0),this.$element.find(".next, .prev").length&&e.support.transition.end&&(this.$element.trigger(e.support.transition.end),this.cycle(!0)),clearInterval(this.interval),this.interval=null,this},next:function(){if(this.sliding)return;return this.slide("next")},prev:function(){if(this.sliding)return;return this.slide("prev")},slide:function(t,n){var r=this.$element.find(".item.active"),i=n||r[t](),s=this.interval,o=t=="next"?"left":"right",u=t=="next"?"first":"last",a=this,f;this.sliding=!0,s&&this.pause(),i=i.length?i:this.$element.find(".item")[u](),f=e.Event("slide",{relatedTarget:i[0],direction:o});if(i.hasClass("active"))return;this.$indicators.length&&(this.$indicators.find(".active").removeClass("active"),this.$element.one("slid",function(){var t=e(a.$indicators.children()[a.getActiveIndex()]);t&&t.addClass("active")}));if(e.support.transition&&this.$element.hasClass("slide")){this.$element.trigger(f);if(f.isDefaultPrevented())return;i.addClass(t),i[0].offsetWidth,r.addClass(o),i.addClass(o),this.$element.one(e.support.transition.end,function(){i.removeClass([t,o].join(" ")).addClass("active"),r.removeClass(["active",o].join(" ")),a.sliding=!1,setTimeout(function(){a.$element.trigger("slid")},0)})}else{this.$element.trigger(f);if(f.isDefaultPrevented())return;r.removeClass("active"),i.addClass("active"),this.sliding=!1,this.$element.trigger("slid")}return s&&this.cycle(),this}};var n=e.fn.carousel;e.fn.carousel=function(n){return this.each(function(){var r=e(this),i=r.data("carousel"),s=e.extend({},e.fn.carousel.defaults,typeof n=="object"&&n),o=typeof n=="string"?n:s.slide;i||r.data("carousel",i=new t(this,s)),typeof n=="number"?i.to(n):o?i[o]():s.interval&&i.pause().cycle()})},e.fn.carousel.defaults={interval:5e3,pause:"hover"},e.fn.carousel.Constructor=t,e.fn.carousel.noConflict=function(){return e.fn.carousel=n,this},e(document).on("click.carousel.data-api","[data-slide], [data-slide-to]",function(t){var n=e(this),r,i=e(n.attr("data-target")||(r=n.attr("href"))&&r.replace(/.*(?=#[^\s]+$)/,"")),s=e.extend({},i.data(),n.data()),o;i.carousel(s),(o=n.attr("data-slide-to"))&&i.data("carousel").pause().to(o).cycle(),t.preventDefault()})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.collapse.defaults,n),this.options.parent&&(this.$parent=e(this.options.parent)),this.options.toggle&&this.toggle()};t.prototype={constructor:t,dimension:function(){var e=this.$element.hasClass("width");return e?"width":"height"},show:function(){var t,n,r,i;if(this.transitioning||this.$element.hasClass("in"))return;t=this.dimension(),n=e.camelCase(["scroll",t].join("-")),r=this.$parent&&this.$parent.find("> .accordion-group > .in");if(r&&r.length){i=r.data("collapse");if(i&&i.transitioning)return;r.collapse("hide"),i||r.data("collapse",null)}this.$element[t](0),this.transition("addClass",e.Event("show"),"shown"),e.support.transition&&this.$element[t](this.$element[0][n])},hide:function(){var t;if(this.transitioning||!this.$element.hasClass("in"))return;t=this.dimension(),this.reset(this.$element[t]()),this.transition("removeClass",e.Event("hide"),"hidden"),this.$element[t](0)},reset:function(e){var t=this.dimension();return this.$element.removeClass("collapse")[t](e||"auto")[0].offsetWidth,this.$element[e!==null?"addClass":"removeClass"]("collapse"),this},transition:function(t,n,r){var i=this,s=function(){n.type=="show"&&i.reset(),i.transitioning=0,i.$element.trigger(r)};this.$element.trigger(n);if(n.isDefaultPrevented())return;this.transitioning=1,this.$element[t]("in"),e.support.transition&&this.$element.hasClass("collapse")?this.$element.one(e.support.transition.end,s):s()},toggle:function(){this[this.$element.hasClass("in")?"hide":"show"]()}};var n=e.fn.collapse;e.fn.collapse=function(n){return this.each(function(){var r=e(this),i=r.data("collapse"),s=e.extend({},e.fn.collapse.defaults,r.data(),typeof n=="object"&&n);i||r.data("collapse",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.collapse.defaults={toggle:!0},e.fn.collapse.Constructor=t,e.fn.collapse.noConflict=function(){return e.fn.collapse=n,this},e(document).on("click.collapse.data-api","[data-toggle=collapse]",function(t){var n=e(this),r,i=n.attr("data-target")||t.preventDefault()||(r=n.attr("href"))&&r.replace(/.*(?=#[^\s]+$)/,""),s=e(i).data("collapse")?"toggle":n.data();n[e(i).hasClass("in")?"addClass":"removeClass"]("collapsed"),e(i).collapse(s)})}(window.jQuery),!function(e){"use strict";function r(){e(t).each(function(){i(e(this)).removeClass("open")})}function i(t){var n=t.attr("data-target"),r;n||(n=t.attr("href"),n=n&&/#/.test(n)&&n.replace(/.*(?=#[^\s]*$)/,"")),r=n&&e(n);if(!r||!r.length)r=t.parent();return r}var t="[data-toggle=dropdown]",n=function(t){var n=e(t).on("click.dropdown.data-api",this.toggle);e("html").on("click.dropdown.data-api",function(){n.parent().removeClass("open")})};n.prototype={constructor:n,toggle:function(t){var n=e(this),s,o;if(n.is(".disabled, :disabled"))return;return s=i(n),o=s.hasClass("open"),r(),o||s.toggleClass("open"),n.focus(),!1},keydown:function(n){var r,s,o,u,a,f;if(!/(38|40|27)/.test(n.keyCode))return;r=e(this),n.preventDefault(),n.stopPropagation();if(r.is(".disabled, :disabled"))return;u=i(r),a=u.hasClass("open");if(!a||a&&n.keyCode==27)return n.which==27&&u.find(t).focus(),r.click();s=e("[role=menu] li:not(.divider):visible a",u);if(!s.length)return;f=s.index(s.filter(":focus")),n.keyCode==38&&f>0&&f--,n.keyCode==40&&f').appendTo(document.body),this.$backdrop.click(this.options.backdrop=="static"?e.proxy(this.$element[0].focus,this.$element[0]):e.proxy(this.hide,this)),i&&this.$backdrop[0].offsetWidth,this.$backdrop.addClass("in");if(!t)return;i?this.$backdrop.one(e.support.transition.end,t):t()}else!this.isShown&&this.$backdrop?(this.$backdrop.removeClass("in"),e.support.transition&&this.$element.hasClass("fade")?this.$backdrop.one(e.support.transition.end,t):t()):t&&t()}};var n=e.fn.modal;e.fn.modal=function(n){return this.each(function(){var r=e(this),i=r.data("modal"),s=e.extend({},e.fn.modal.defaults,r.data(),typeof n=="object"&&n);i||r.data("modal",i=new t(this,s)),typeof n=="string"?i[n]():s.show&&i.show()})},e.fn.modal.defaults={backdrop:!0,keyboard:!0,show:!0},e.fn.modal.Constructor=t,e.fn.modal.noConflict=function(){return e.fn.modal=n,this},e(document).on("click.modal.data-api",'[data-toggle="modal"]',function(t){var n=e(this),r=n.attr("href"),i=e(n.attr("data-target")||r&&r.replace(/.*(?=#[^\s]+$)/,"")),s=i.data("modal")?"toggle":e.extend({remote:!/#/.test(r)&&r},i.data(),n.data());t.preventDefault(),i.modal(s).one("hide",function(){n.focus()})})}(window.jQuery),!function(e){"use strict";var t=function(e,t){this.init("tooltip",e,t)};t.prototype={constructor:t,init:function(t,n,r){var i,s,o,u,a;this.type=t,this.$element=e(n),this.options=this.getOptions(r),this.enabled=!0,o=this.options.trigger.split(" ");for(a=o.length;a--;)u=o[a],u=="click"?this.$element.on("click."+this.type,this.options.selector,e.proxy(this.toggle,this)):u!="manual"&&(i=u=="hover"?"mouseenter":"focus",s=u=="hover"?"mouseleave":"blur",this.$element.on(i+"."+this.type,this.options.selector,e.proxy(this.enter,this)),this.$element.on(s+"."+this.type,this.options.selector,e.proxy(this.leave,this)));this.options.selector?this._options=e.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()},getOptions:function(t){return t=e.extend({},e.fn[this.type].defaults,this.$element.data(),t),t.delay&&typeof t.delay=="number"&&(t.delay={show:t.delay,hide:t.delay}),t},enter:function(t){var n=e.fn[this.type].defaults,r={},i;this._options&&e.each(this._options,function(e,t){n[e]!=t&&(r[e]=t)},this),i=e(t.currentTarget)[this.type](r).data(this.type);if(!i.options.delay||!i.options.delay.show)return i.show();clearTimeout(this.timeout),i.hoverState="in",this.timeout=setTimeout(function(){i.hoverState=="in"&&i.show()},i.options.delay.show)},leave:function(t){var n=e(t.currentTarget)[this.type](this._options).data(this.type);this.timeout&&clearTimeout(this.timeout);if(!n.options.delay||!n.options.delay.hide)return n.hide();n.hoverState="out",this.timeout=setTimeout(function(){n.hoverState=="out"&&n.hide()},n.options.delay.hide)},show:function(){var t,n,r,i,s,o,u=e.Event("show");if(this.hasContent()&&this.enabled){this.$element.trigger(u);if(u.isDefaultPrevented())return;t=this.tip(),this.setContent(),this.options.animation&&t.addClass("fade"),s=typeof this.options.placement=="function"?this.options.placement.call(this,t[0],this.$element[0]):this.options.placement,t.detach().css({top:0,left:0,display:"block"}),this.options.container?t.appendTo(this.options.container):t.insertAfter(this.$element),n=this.getPosition(),r=t[0].offsetWidth,i=t[0].offsetHeight;switch(s){case"bottom":o={top:n.top+n.height,left:n.left+n.width/2-r/2};break;case"top":o={top:n.top-i,left:n.left+n.width/2-r/2};break;case"left":o={top:n.top+n.height/2-i/2,left:n.left-r};break;case"right":o={top:n.top+n.height/2-i/2,left:n.left+n.width}}this.applyPlacement(o,s),this.$element.trigger("shown")}},applyPlacement:function(e,t){var n=this.tip(),r=n[0].offsetWidth,i=n[0].offsetHeight,s,o,u,a;n.offset(e).addClass(t).addClass("in"),s=n[0].offsetWidth,o=n[0].offsetHeight,t=="top"&&o!=i&&(e.top=e.top+i-o,a=!0),t=="bottom"||t=="top"?(u=0,e.left<0&&(u=e.left*-2,e.left=0,n.offset(e),s=n[0].offsetWidth,o=n[0].offsetHeight),this.replaceArrow(u-r+s,s,"left")):this.replaceArrow(o-i,o,"top"),a&&n.offset(e)},replaceArrow:function(e,t,n){this.arrow().css(n,e?50*(1-e/t)+"%":"")},setContent:function(){var e=this.tip(),t=this.getTitle();e.find(".tooltip-inner")[this.options.html?"html":"text"](t),e.removeClass("fade in top bottom left right")},hide:function(){function i(){var t=setTimeout(function(){n.off(e.support.transition.end).detach()},500);n.one(e.support.transition.end,function(){clearTimeout(t),n.detach()})}var t=this,n=this.tip(),r=e.Event("hide");this.$element.trigger(r);if(r.isDefaultPrevented())return;return n.removeClass("in"),e.support.transition&&this.$tip.hasClass("fade")?i():n.detach(),this.$element.trigger("hidden"),this},fixTitle:function(){var e=this.$element;(e.attr("title")||typeof e.attr("data-original-title")!="string")&&e.attr("data-original-title",e.attr("title")||"").attr("title","")},hasContent:function(){return this.getTitle()},getPosition:function(){var t=this.$element[0];return e.extend({},typeof t.getBoundingClientRect=="function"?t.getBoundingClientRect():{width:t.offsetWidth,height:t.offsetHeight},this.$element.offset())},getTitle:function(){var e,t=this.$element,n=this.options;return e=t.attr("data-original-title")||(typeof n.title=="function"?n.title.call(t[0]):n.title),e},tip:function(){return this.$tip=this.$tip||e(this.options.template)},arrow:function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")},validate:function(){this.$element[0].parentNode||(this.hide(),this.$element=null,this.options=null)},enable:function(){this.enabled=!0},disable:function(){this.enabled=!1},toggleEnabled:function(){this.enabled=!this.enabled},toggle:function(t){var n=t?e(t.currentTarget)[this.type](this._options).data(this.type):this;n.tip().hasClass("in")?n.hide():n.show()},destroy:function(){this.hide().$element.off("."+this.type).removeData(this.type)}};var n=e.fn.tooltip;e.fn.tooltip=function(n){return this.each(function(){var r=e(this),i=r.data("tooltip"),s=typeof n=="object"&&n;i||r.data("tooltip",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.tooltip.Constructor=t,e.fn.tooltip.defaults={animation:!0,placement:"top",selector:!1,template:'
',trigger:"hover focus",title:"",delay:0,html:!1,container:!1},e.fn.tooltip.noConflict=function(){return e.fn.tooltip=n,this}}(window.jQuery),!function(e){"use strict";var t=function(e,t){this.init("popover",e,t)};t.prototype=e.extend({},e.fn.tooltip.Constructor.prototype,{constructor:t,setContent:function(){var e=this.tip(),t=this.getTitle(),n=this.getContent();e.find(".popover-title")[this.options.html?"html":"text"](t),e.find(".popover-content")[this.options.html?"html":"text"](n),e.removeClass("fade top bottom left right in")},hasContent:function(){return this.getTitle()||this.getContent()},getContent:function(){var e,t=this.$element,n=this.options;return e=(typeof n.content=="function"?n.content.call(t[0]):n.content)||t.attr("data-content"),e},tip:function(){return this.$tip||(this.$tip=e(this.options.template)),this.$tip},destroy:function(){this.hide().$element.off("."+this.type).removeData(this.type)}});var n=e.fn.popover;e.fn.popover=function(n){return this.each(function(){var r=e(this),i=r.data("popover"),s=typeof n=="object"&&n;i||r.data("popover",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.popover.Constructor=t,e.fn.popover.defaults=e.extend({},e.fn.tooltip.defaults,{placement:"right",trigger:"click",content:"",template:'

'}),e.fn.popover.noConflict=function(){return e.fn.popover=n,this}}(window.jQuery),!function(e){"use strict";function t(t,n){var r=e.proxy(this.process,this),i=e(t).is("body")?e(window):e(t),s;this.options=e.extend({},e.fn.scrollspy.defaults,n),this.$scrollElement=i.on("scroll.scroll-spy.data-api",r),this.selector=(this.options.target||(s=e(t).attr("href"))&&s.replace(/.*(?=#[^\s]+$)/,"")||"")+" .nav li > a",this.$body=e("body"),this.refresh(),this.process()}t.prototype={constructor:t,refresh:function(){var t=this,n;this.offsets=e([]),this.targets=e([]),n=this.$body.find(this.selector).map(function(){var n=e(this),r=n.data("target")||n.attr("href"),i=/^#\w/.test(r)&&e(r);return i&&i.length&&[[i.position().top+(!e.isWindow(t.$scrollElement.get(0))&&t.$scrollElement.scrollTop()),r]]||null}).sort(function(e,t){return e[0]-t[0]}).each(function(){t.offsets.push(this[0]),t.targets.push(this[1])})},process:function(){var e=this.$scrollElement.scrollTop()+this.options.offset,t=this.$scrollElement[0].scrollHeight||this.$body[0].scrollHeight,n=t-this.$scrollElement.height(),r=this.offsets,i=this.targets,s=this.activeTarget,o;if(e>=n)return s!=(o=i.last()[0])&&this.activate(o);for(o=r.length;o--;)s!=i[o]&&e>=r[o]&&(!r[o+1]||e<=r[o+1])&&this.activate(i[o])},activate:function(t){var n,r;this.activeTarget=t,e(this.selector).parent(".active").removeClass("active"),r=this.selector+'[data-target="'+t+'"],'+this.selector+'[href="'+t+'"]',n=e(r).parent("li").addClass("active"),n.parent(".dropdown-menu").length&&(n=n.closest("li.dropdown").addClass("active")),n.trigger("activate")}};var n=e.fn.scrollspy;e.fn.scrollspy=function(n){return this.each(function(){var r=e(this),i=r.data("scrollspy"),s=typeof n=="object"&&n;i||r.data("scrollspy",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.scrollspy.Constructor=t,e.fn.scrollspy.defaults={offset:10},e.fn.scrollspy.noConflict=function(){return e.fn.scrollspy=n,this},e(window).on("load",function(){e('[data-spy="scroll"]').each(function(){var t=e(this);t.scrollspy(t.data())})})}(window.jQuery),!function(e){"use strict";var t=function(t){this.element=e(t)};t.prototype={constructor:t,show:function(){var t=this.element,n=t.closest("ul:not(.dropdown-menu)"),r=t.attr("data-target"),i,s,o;r||(r=t.attr("href"),r=r&&r.replace(/.*(?=#[^\s]*$)/,""));if(t.parent("li").hasClass("active"))return;i=n.find(".active:last a")[0],o=e.Event("show",{relatedTarget:i}),t.trigger(o);if(o.isDefaultPrevented())return;s=e(r),this.activate(t.parent("li"),n),this.activate(s,s.parent(),function(){t.trigger({type:"shown",relatedTarget:i})})},activate:function(t,n,r){function o(){i.removeClass("active").find("> .dropdown-menu > .active").removeClass("active"),t.addClass("active"),s?(t[0].offsetWidth,t.addClass("in")):t.removeClass("fade"),t.parent(".dropdown-menu")&&t.closest("li.dropdown").addClass("active"),r&&r()}var i=n.find("> .active"),s=r&&e.support.transition&&i.hasClass("fade");s?i.one(e.support.transition.end,o):o(),i.removeClass("in")}};var n=e.fn.tab;e.fn.tab=function(n){return this.each(function(){var r=e(this),i=r.data("tab");i||r.data("tab",i=new t(this)),typeof n=="string"&&i[n]()})},e.fn.tab.Constructor=t,e.fn.tab.noConflict=function(){return e.fn.tab=n,this},e(document).on("click.tab.data-api",'[data-toggle="tab"], [data-toggle="pill"]',function(t){t.preventDefault(),e(this).tab("show")})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.typeahead.defaults,n),this.matcher=this.options.matcher||this.matcher,this.sorter=this.options.sorter||this.sorter,this.highlighter=this.options.highlighter||this.highlighter,this.updater=this.options.updater||this.updater,this.source=this.options.source,this.$menu=e(this.options.menu),this.shown=!1,this.listen()};t.prototype={constructor:t,select:function(){var e=this.$menu.find(".active").attr("data-value");return this.$element.val(this.updater(e)).change(),this.hide()},updater:function(e){return e},show:function(){var t=e.extend({},this.$element.position(),{height:this.$element[0].offsetHeight});return this.$menu.insertAfter(this.$element).css({top:t.top+t.height,left:t.left}).show(),this.shown=!0,this},hide:function(){return this.$menu.hide(),this.shown=!1,this},lookup:function(t){var n;return this.query=this.$element.val(),!this.query||this.query.length"+t+""})},render:function(t){var n=this;return t=e(t).map(function(t,r){return t=e(n.options.item).attr("data-value",r),t.find("a").html(n.highlighter(r)),t[0]}),t.first().addClass("active"),this.$menu.html(t),this},next:function(t){var n=this.$menu.find(".active").removeClass("active"),r=n.next();r.length||(r=e(this.$menu.find("li")[0])),r.addClass("active")},prev:function(e){var t=this.$menu.find(".active").removeClass("active"),n=t.prev();n.length||(n=this.$menu.find("li").last()),n.addClass("active")},listen:function(){this.$element.on("focus",e.proxy(this.focus,this)).on("blur",e.proxy(this.blur,this)).on("keypress",e.proxy(this.keypress,this)).on("keyup",e.proxy(this.keyup,this)),this.eventSupported("keydown")&&this.$element.on("keydown",e.proxy(this.keydown,this)),this.$menu.on("click",e.proxy(this.click,this)).on("mouseenter","li",e.proxy(this.mouseenter,this)).on("mouseleave","li",e.proxy(this.mouseleave,this))},eventSupported:function(e){var t=e in this.$element;return t||(this.$element.setAttribute(e,"return;"),t=typeof this.$element[e]=="function"),t},move:function(e){if(!this.shown)return;switch(e.keyCode){case 9:case 13:case 27:e.preventDefault();break;case 38:e.preventDefault(),this.prev();break;case 40:e.preventDefault(),this.next()}e.stopPropagation()},keydown:function(t){this.suppressKeyPressRepeat=~e.inArray(t.keyCode,[40,38,9,13,27]),this.move(t)},keypress:function(e){if(this.suppressKeyPressRepeat)return;this.move(e)},keyup:function(e){switch(e.keyCode){case 40:case 38:case 16:case 17:case 18:break;case 9:case 13:if(!this.shown)return;this.select();break;case 27:if(!this.shown)return;this.hide();break;default:this.lookup()}e.stopPropagation(),e.preventDefault()},focus:function(e){this.focused=!0},blur:function(e){this.focused=!1,!this.mousedover&&this.shown&&this.hide()},click:function(e){e.stopPropagation(),e.preventDefault(),this.select(),this.$element.focus()},mouseenter:function(t){this.mousedover=!0,this.$menu.find(".active").removeClass("active"),e(t.currentTarget).addClass("active")},mouseleave:function(e){this.mousedover=!1,!this.focused&&this.shown&&this.hide()}};var n=e.fn.typeahead;e.fn.typeahead=function(n){return this.each(function(){var r=e(this),i=r.data("typeahead"),s=typeof n=="object"&&n;i||r.data("typeahead",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.typeahead.defaults={source:[],items:8,menu:'',item:'
  • ',minLength:1},e.fn.typeahead.Constructor=t,e.fn.typeahead.noConflict=function(){return e.fn.typeahead=n,this},e(document).on("focus.typeahead.data-api",'[data-provide="typeahead"]',function(t){var n=e(this);if(n.data("typeahead"))return;n.typeahead(n.data())})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.options=e.extend({},e.fn.affix.defaults,n),this.$window=e(window).on("scroll.affix.data-api",e.proxy(this.checkPosition,this)).on("click.affix.data-api",e.proxy(function(){setTimeout(e.proxy(this.checkPosition,this),1)},this)),this.$element=e(t),this.checkPosition()};t.prototype.checkPosition=function(){if(!this.$element.is(":visible"))return;var t=e(document).height(),n=this.$window.scrollTop(),r=this.$element.offset(),i=this.options.offset,s=i.bottom,o=i.top,u="affix affix-top affix-bottom",a;typeof i!="object"&&(s=o=i),typeof o=="function"&&(o=i.top()),typeof s=="function"&&(s=i.bottom()),a=this.unpin!=null&&n+this.unpin<=r.top?!1:s!=null&&r.top+this.$element.height()>=t-s?"bottom":o!=null&&n<=o?"top":!1;if(this.affixed===a)return;this.affixed=a,this.unpin=a=="bottom"?r.top-n:null,this.$element.removeClass(u).addClass("affix"+(a?"-"+a:""))};var n=e.fn.affix;e.fn.affix=function(n){return this.each(function(){var r=e(this),i=r.data("affix"),s=typeof n=="object"&&n;i||r.data("affix",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.affix.Constructor=t,e.fn.affix.defaults={offset:0},e.fn.affix.noConflict=function(){return e.fn.affix=n,this},e(window).on("load",function(){e('[data-spy="affix"]').each(function(){var t=e(this),n=t.data();n.offset=n.offset||{},n.offsetBottom&&(n.offset.bottom=n.offsetBottom),n.offsetTop&&(n.offset.top=n.offsetTop),t.affix(n)})})}(window.jQuery); -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/css/bootstrap-responsive.css: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap Responsive v2.3.1 3 | * 4 | * Copyright 2012 Twitter, Inc 5 | * Licensed under the Apache License v2.0 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * 8 | * Designed and built with all the love in the world @twitter by @mdo and @fat. 9 | */ 10 | 11 | .clearfix { 12 | *zoom: 1; 13 | } 14 | 15 | .clearfix:before, 16 | .clearfix:after { 17 | display: table; 18 | line-height: 0; 19 | content: ""; 20 | } 21 | 22 | .clearfix:after { 23 | clear: both; 24 | } 25 | 26 | .hide-text { 27 | font: 0/0 a; 28 | color: transparent; 29 | text-shadow: none; 30 | background-color: transparent; 31 | border: 0; 32 | } 33 | 34 | .input-block-level { 35 | display: block; 36 | width: 100%; 37 | min-height: 30px; 38 | -webkit-box-sizing: border-box; 39 | -moz-box-sizing: border-box; 40 | box-sizing: border-box; 41 | } 42 | 43 | @-ms-viewport { 44 | width: device-width; 45 | } 46 | 47 | .hidden { 48 | display: none; 49 | visibility: hidden; 50 | } 51 | 52 | .visible-phone { 53 | display: none !important; 54 | } 55 | 56 | .visible-tablet { 57 | display: none !important; 58 | } 59 | 60 | .hidden-desktop { 61 | display: none !important; 62 | } 63 | 64 | .visible-desktop { 65 | display: inherit !important; 66 | } 67 | 68 | @media (min-width: 768px) and (max-width: 979px) { 69 | .hidden-desktop { 70 | display: inherit !important; 71 | } 72 | .visible-desktop { 73 | display: none !important ; 74 | } 75 | .visible-tablet { 76 | display: inherit !important; 77 | } 78 | .hidden-tablet { 79 | display: none !important; 80 | } 81 | } 82 | 83 | @media (max-width: 767px) { 84 | .hidden-desktop { 85 | display: inherit !important; 86 | } 87 | .visible-desktop { 88 | display: none !important; 89 | } 90 | .visible-phone { 91 | display: inherit !important; 92 | } 93 | .hidden-phone { 94 | display: none !important; 95 | } 96 | } 97 | 98 | .visible-print { 99 | display: none !important; 100 | } 101 | 102 | @media print { 103 | .visible-print { 104 | display: inherit !important; 105 | } 106 | .hidden-print { 107 | display: none !important; 108 | } 109 | } 110 | 111 | @media (min-width: 1200px) { 112 | .row { 113 | margin-left: -30px; 114 | *zoom: 1; 115 | } 116 | .row:before, 117 | .row:after { 118 | display: table; 119 | line-height: 0; 120 | content: ""; 121 | } 122 | .row:after { 123 | clear: both; 124 | } 125 | [class*="span"] { 126 | float: left; 127 | min-height: 1px; 128 | margin-left: 30px; 129 | } 130 | .container, 131 | .navbar-static-top .container, 132 | .navbar-fixed-top .container, 133 | .navbar-fixed-bottom .container { 134 | width: 1170px; 135 | } 136 | .span12 { 137 | width: 1170px; 138 | } 139 | .span11 { 140 | width: 1070px; 141 | } 142 | .span10 { 143 | width: 970px; 144 | } 145 | .span9 { 146 | width: 870px; 147 | } 148 | .span8 { 149 | width: 770px; 150 | } 151 | .span7 { 152 | width: 670px; 153 | } 154 | .span6 { 155 | width: 570px; 156 | } 157 | .span5 { 158 | width: 470px; 159 | } 160 | .span4 { 161 | width: 370px; 162 | } 163 | .span3 { 164 | width: 270px; 165 | } 166 | .span2 { 167 | width: 170px; 168 | } 169 | .span1 { 170 | width: 70px; 171 | } 172 | .offset12 { 173 | margin-left: 1230px; 174 | } 175 | .offset11 { 176 | margin-left: 1130px; 177 | } 178 | .offset10 { 179 | margin-left: 1030px; 180 | } 181 | .offset9 { 182 | margin-left: 930px; 183 | } 184 | .offset8 { 185 | margin-left: 830px; 186 | } 187 | .offset7 { 188 | margin-left: 730px; 189 | } 190 | .offset6 { 191 | margin-left: 630px; 192 | } 193 | .offset5 { 194 | margin-left: 530px; 195 | } 196 | .offset4 { 197 | margin-left: 430px; 198 | } 199 | .offset3 { 200 | margin-left: 330px; 201 | } 202 | .offset2 { 203 | margin-left: 230px; 204 | } 205 | .offset1 { 206 | margin-left: 130px; 207 | } 208 | .row-fluid { 209 | width: 100%; 210 | *zoom: 1; 211 | } 212 | .row-fluid:before, 213 | .row-fluid:after { 214 | display: table; 215 | line-height: 0; 216 | content: ""; 217 | } 218 | .row-fluid:after { 219 | clear: both; 220 | } 221 | .row-fluid [class*="span"] { 222 | display: block; 223 | float: left; 224 | width: 100%; 225 | min-height: 30px; 226 | margin-left: 2.564102564102564%; 227 | *margin-left: 2.5109110747408616%; 228 | -webkit-box-sizing: border-box; 229 | -moz-box-sizing: border-box; 230 | box-sizing: border-box; 231 | } 232 | .row-fluid [class*="span"]:first-child { 233 | margin-left: 0; 234 | } 235 | .row-fluid .controls-row [class*="span"] + [class*="span"] { 236 | margin-left: 2.564102564102564%; 237 | } 238 | .row-fluid .span12 { 239 | width: 100%; 240 | *width: 99.94680851063829%; 241 | } 242 | .row-fluid .span11 { 243 | width: 91.45299145299145%; 244 | *width: 91.39979996362975%; 245 | } 246 | .row-fluid .span10 { 247 | width: 82.90598290598291%; 248 | *width: 82.8527914166212%; 249 | } 250 | .row-fluid .span9 { 251 | width: 74.35897435897436%; 252 | *width: 74.30578286961266%; 253 | } 254 | .row-fluid .span8 { 255 | width: 65.81196581196582%; 256 | *width: 65.75877432260411%; 257 | } 258 | .row-fluid .span7 { 259 | width: 57.26495726495726%; 260 | *width: 57.21176577559556%; 261 | } 262 | .row-fluid .span6 { 263 | width: 48.717948717948715%; 264 | *width: 48.664757228587014%; 265 | } 266 | .row-fluid .span5 { 267 | width: 40.17094017094017%; 268 | *width: 40.11774868157847%; 269 | } 270 | .row-fluid .span4 { 271 | width: 31.623931623931625%; 272 | *width: 31.570740134569924%; 273 | } 274 | .row-fluid .span3 { 275 | width: 23.076923076923077%; 276 | *width: 23.023731587561375%; 277 | } 278 | .row-fluid .span2 { 279 | width: 14.52991452991453%; 280 | *width: 14.476723040552828%; 281 | } 282 | .row-fluid .span1 { 283 | width: 5.982905982905983%; 284 | *width: 5.929714493544281%; 285 | } 286 | .row-fluid .offset12 { 287 | margin-left: 105.12820512820512%; 288 | *margin-left: 105.02182214948171%; 289 | } 290 | .row-fluid .offset12:first-child { 291 | margin-left: 102.56410256410257%; 292 | *margin-left: 102.45771958537915%; 293 | } 294 | .row-fluid .offset11 { 295 | margin-left: 96.58119658119658%; 296 | *margin-left: 96.47481360247316%; 297 | } 298 | .row-fluid .offset11:first-child { 299 | margin-left: 94.01709401709402%; 300 | *margin-left: 93.91071103837061%; 301 | } 302 | .row-fluid .offset10 { 303 | margin-left: 88.03418803418803%; 304 | *margin-left: 87.92780505546462%; 305 | } 306 | .row-fluid .offset10:first-child { 307 | margin-left: 85.47008547008548%; 308 | *margin-left: 85.36370249136206%; 309 | } 310 | .row-fluid .offset9 { 311 | margin-left: 79.48717948717949%; 312 | *margin-left: 79.38079650845607%; 313 | } 314 | .row-fluid .offset9:first-child { 315 | margin-left: 76.92307692307693%; 316 | *margin-left: 76.81669394435352%; 317 | } 318 | .row-fluid .offset8 { 319 | margin-left: 70.94017094017094%; 320 | *margin-left: 70.83378796144753%; 321 | } 322 | .row-fluid .offset8:first-child { 323 | margin-left: 68.37606837606839%; 324 | *margin-left: 68.26968539734497%; 325 | } 326 | .row-fluid .offset7 { 327 | margin-left: 62.393162393162385%; 328 | *margin-left: 62.28677941443899%; 329 | } 330 | .row-fluid .offset7:first-child { 331 | margin-left: 59.82905982905982%; 332 | *margin-left: 59.72267685033642%; 333 | } 334 | .row-fluid .offset6 { 335 | margin-left: 53.84615384615384%; 336 | *margin-left: 53.739770867430444%; 337 | } 338 | .row-fluid .offset6:first-child { 339 | margin-left: 51.28205128205128%; 340 | *margin-left: 51.175668303327875%; 341 | } 342 | .row-fluid .offset5 { 343 | margin-left: 45.299145299145295%; 344 | *margin-left: 45.1927623204219%; 345 | } 346 | .row-fluid .offset5:first-child { 347 | margin-left: 42.73504273504273%; 348 | *margin-left: 42.62865975631933%; 349 | } 350 | .row-fluid .offset4 { 351 | margin-left: 36.75213675213675%; 352 | *margin-left: 36.645753773413354%; 353 | } 354 | .row-fluid .offset4:first-child { 355 | margin-left: 34.18803418803419%; 356 | *margin-left: 34.081651209310785%; 357 | } 358 | .row-fluid .offset3 { 359 | margin-left: 28.205128205128204%; 360 | *margin-left: 28.0987452264048%; 361 | } 362 | .row-fluid .offset3:first-child { 363 | margin-left: 25.641025641025642%; 364 | *margin-left: 25.53464266230224%; 365 | } 366 | .row-fluid .offset2 { 367 | margin-left: 19.65811965811966%; 368 | *margin-left: 19.551736679396257%; 369 | } 370 | .row-fluid .offset2:first-child { 371 | margin-left: 17.094017094017094%; 372 | *margin-left: 16.98763411529369%; 373 | } 374 | .row-fluid .offset1 { 375 | margin-left: 11.11111111111111%; 376 | *margin-left: 11.004728132387708%; 377 | } 378 | .row-fluid .offset1:first-child { 379 | margin-left: 8.547008547008547%; 380 | *margin-left: 8.440625568285142%; 381 | } 382 | input, 383 | textarea, 384 | .uneditable-input { 385 | margin-left: 0; 386 | } 387 | .controls-row [class*="span"] + [class*="span"] { 388 | margin-left: 30px; 389 | } 390 | input.span12, 391 | textarea.span12, 392 | .uneditable-input.span12 { 393 | width: 1156px; 394 | } 395 | input.span11, 396 | textarea.span11, 397 | .uneditable-input.span11 { 398 | width: 1056px; 399 | } 400 | input.span10, 401 | textarea.span10, 402 | .uneditable-input.span10 { 403 | width: 956px; 404 | } 405 | input.span9, 406 | textarea.span9, 407 | .uneditable-input.span9 { 408 | width: 856px; 409 | } 410 | input.span8, 411 | textarea.span8, 412 | .uneditable-input.span8 { 413 | width: 756px; 414 | } 415 | input.span7, 416 | textarea.span7, 417 | .uneditable-input.span7 { 418 | width: 656px; 419 | } 420 | input.span6, 421 | textarea.span6, 422 | .uneditable-input.span6 { 423 | width: 556px; 424 | } 425 | input.span5, 426 | textarea.span5, 427 | .uneditable-input.span5 { 428 | width: 456px; 429 | } 430 | input.span4, 431 | textarea.span4, 432 | .uneditable-input.span4 { 433 | width: 356px; 434 | } 435 | input.span3, 436 | textarea.span3, 437 | .uneditable-input.span3 { 438 | width: 256px; 439 | } 440 | input.span2, 441 | textarea.span2, 442 | .uneditable-input.span2 { 443 | width: 156px; 444 | } 445 | input.span1, 446 | textarea.span1, 447 | .uneditable-input.span1 { 448 | width: 56px; 449 | } 450 | .thumbnails { 451 | margin-left: -30px; 452 | } 453 | .thumbnails > li { 454 | margin-left: 30px; 455 | } 456 | .row-fluid .thumbnails { 457 | margin-left: 0; 458 | } 459 | } 460 | 461 | @media (min-width: 768px) and (max-width: 979px) { 462 | .row { 463 | margin-left: -20px; 464 | *zoom: 1; 465 | } 466 | .row:before, 467 | .row:after { 468 | display: table; 469 | line-height: 0; 470 | content: ""; 471 | } 472 | .row:after { 473 | clear: both; 474 | } 475 | [class*="span"] { 476 | float: left; 477 | min-height: 1px; 478 | margin-left: 20px; 479 | } 480 | .container, 481 | .navbar-static-top .container, 482 | .navbar-fixed-top .container, 483 | .navbar-fixed-bottom .container { 484 | width: 724px; 485 | } 486 | .span12 { 487 | width: 724px; 488 | } 489 | .span11 { 490 | width: 662px; 491 | } 492 | .span10 { 493 | width: 600px; 494 | } 495 | .span9 { 496 | width: 538px; 497 | } 498 | .span8 { 499 | width: 476px; 500 | } 501 | .span7 { 502 | width: 414px; 503 | } 504 | .span6 { 505 | width: 352px; 506 | } 507 | .span5 { 508 | width: 290px; 509 | } 510 | .span4 { 511 | width: 228px; 512 | } 513 | .span3 { 514 | width: 166px; 515 | } 516 | .span2 { 517 | width: 104px; 518 | } 519 | .span1 { 520 | width: 42px; 521 | } 522 | .offset12 { 523 | margin-left: 764px; 524 | } 525 | .offset11 { 526 | margin-left: 702px; 527 | } 528 | .offset10 { 529 | margin-left: 640px; 530 | } 531 | .offset9 { 532 | margin-left: 578px; 533 | } 534 | .offset8 { 535 | margin-left: 516px; 536 | } 537 | .offset7 { 538 | margin-left: 454px; 539 | } 540 | .offset6 { 541 | margin-left: 392px; 542 | } 543 | .offset5 { 544 | margin-left: 330px; 545 | } 546 | .offset4 { 547 | margin-left: 268px; 548 | } 549 | .offset3 { 550 | margin-left: 206px; 551 | } 552 | .offset2 { 553 | margin-left: 144px; 554 | } 555 | .offset1 { 556 | margin-left: 82px; 557 | } 558 | .row-fluid { 559 | width: 100%; 560 | *zoom: 1; 561 | } 562 | .row-fluid:before, 563 | .row-fluid:after { 564 | display: table; 565 | line-height: 0; 566 | content: ""; 567 | } 568 | .row-fluid:after { 569 | clear: both; 570 | } 571 | .row-fluid [class*="span"] { 572 | display: block; 573 | float: left; 574 | width: 100%; 575 | min-height: 30px; 576 | margin-left: 2.7624309392265194%; 577 | *margin-left: 2.709239449864817%; 578 | -webkit-box-sizing: border-box; 579 | -moz-box-sizing: border-box; 580 | box-sizing: border-box; 581 | } 582 | .row-fluid [class*="span"]:first-child { 583 | margin-left: 0; 584 | } 585 | .row-fluid .controls-row [class*="span"] + [class*="span"] { 586 | margin-left: 2.7624309392265194%; 587 | } 588 | .row-fluid .span12 { 589 | width: 100%; 590 | *width: 99.94680851063829%; 591 | } 592 | .row-fluid .span11 { 593 | width: 91.43646408839778%; 594 | *width: 91.38327259903608%; 595 | } 596 | .row-fluid .span10 { 597 | width: 82.87292817679558%; 598 | *width: 82.81973668743387%; 599 | } 600 | .row-fluid .span9 { 601 | width: 74.30939226519337%; 602 | *width: 74.25620077583166%; 603 | } 604 | .row-fluid .span8 { 605 | width: 65.74585635359117%; 606 | *width: 65.69266486422946%; 607 | } 608 | .row-fluid .span7 { 609 | width: 57.18232044198895%; 610 | *width: 57.12912895262725%; 611 | } 612 | .row-fluid .span6 { 613 | width: 48.61878453038674%; 614 | *width: 48.56559304102504%; 615 | } 616 | .row-fluid .span5 { 617 | width: 40.05524861878453%; 618 | *width: 40.00205712942283%; 619 | } 620 | .row-fluid .span4 { 621 | width: 31.491712707182323%; 622 | *width: 31.43852121782062%; 623 | } 624 | .row-fluid .span3 { 625 | width: 22.92817679558011%; 626 | *width: 22.87498530621841%; 627 | } 628 | .row-fluid .span2 { 629 | width: 14.3646408839779%; 630 | *width: 14.311449394616199%; 631 | } 632 | .row-fluid .span1 { 633 | width: 5.801104972375691%; 634 | *width: 5.747913483013988%; 635 | } 636 | .row-fluid .offset12 { 637 | margin-left: 105.52486187845304%; 638 | *margin-left: 105.41847889972962%; 639 | } 640 | .row-fluid .offset12:first-child { 641 | margin-left: 102.76243093922652%; 642 | *margin-left: 102.6560479605031%; 643 | } 644 | .row-fluid .offset11 { 645 | margin-left: 96.96132596685082%; 646 | *margin-left: 96.8549429881274%; 647 | } 648 | .row-fluid .offset11:first-child { 649 | margin-left: 94.1988950276243%; 650 | *margin-left: 94.09251204890089%; 651 | } 652 | .row-fluid .offset10 { 653 | margin-left: 88.39779005524862%; 654 | *margin-left: 88.2914070765252%; 655 | } 656 | .row-fluid .offset10:first-child { 657 | margin-left: 85.6353591160221%; 658 | *margin-left: 85.52897613729868%; 659 | } 660 | .row-fluid .offset9 { 661 | margin-left: 79.8342541436464%; 662 | *margin-left: 79.72787116492299%; 663 | } 664 | .row-fluid .offset9:first-child { 665 | margin-left: 77.07182320441989%; 666 | *margin-left: 76.96544022569647%; 667 | } 668 | .row-fluid .offset8 { 669 | margin-left: 71.2707182320442%; 670 | *margin-left: 71.16433525332079%; 671 | } 672 | .row-fluid .offset8:first-child { 673 | margin-left: 68.50828729281768%; 674 | *margin-left: 68.40190431409427%; 675 | } 676 | .row-fluid .offset7 { 677 | margin-left: 62.70718232044199%; 678 | *margin-left: 62.600799341718584%; 679 | } 680 | .row-fluid .offset7:first-child { 681 | margin-left: 59.94475138121547%; 682 | *margin-left: 59.838368402492065%; 683 | } 684 | .row-fluid .offset6 { 685 | margin-left: 54.14364640883978%; 686 | *margin-left: 54.037263430116376%; 687 | } 688 | .row-fluid .offset6:first-child { 689 | margin-left: 51.38121546961326%; 690 | *margin-left: 51.27483249088986%; 691 | } 692 | .row-fluid .offset5 { 693 | margin-left: 45.58011049723757%; 694 | *margin-left: 45.47372751851417%; 695 | } 696 | .row-fluid .offset5:first-child { 697 | margin-left: 42.81767955801105%; 698 | *margin-left: 42.71129657928765%; 699 | } 700 | .row-fluid .offset4 { 701 | margin-left: 37.01657458563536%; 702 | *margin-left: 36.91019160691196%; 703 | } 704 | .row-fluid .offset4:first-child { 705 | margin-left: 34.25414364640884%; 706 | *margin-left: 34.14776066768544%; 707 | } 708 | .row-fluid .offset3 { 709 | margin-left: 28.45303867403315%; 710 | *margin-left: 28.346655695309746%; 711 | } 712 | .row-fluid .offset3:first-child { 713 | margin-left: 25.69060773480663%; 714 | *margin-left: 25.584224756083227%; 715 | } 716 | .row-fluid .offset2 { 717 | margin-left: 19.88950276243094%; 718 | *margin-left: 19.783119783707537%; 719 | } 720 | .row-fluid .offset2:first-child { 721 | margin-left: 17.12707182320442%; 722 | *margin-left: 17.02068884448102%; 723 | } 724 | .row-fluid .offset1 { 725 | margin-left: 11.32596685082873%; 726 | *margin-left: 11.219583872105325%; 727 | } 728 | .row-fluid .offset1:first-child { 729 | margin-left: 8.56353591160221%; 730 | *margin-left: 8.457152932878806%; 731 | } 732 | input, 733 | textarea, 734 | .uneditable-input { 735 | margin-left: 0; 736 | } 737 | .controls-row [class*="span"] + [class*="span"] { 738 | margin-left: 20px; 739 | } 740 | input.span12, 741 | textarea.span12, 742 | .uneditable-input.span12 { 743 | width: 710px; 744 | } 745 | input.span11, 746 | textarea.span11, 747 | .uneditable-input.span11 { 748 | width: 648px; 749 | } 750 | input.span10, 751 | textarea.span10, 752 | .uneditable-input.span10 { 753 | width: 586px; 754 | } 755 | input.span9, 756 | textarea.span9, 757 | .uneditable-input.span9 { 758 | width: 524px; 759 | } 760 | input.span8, 761 | textarea.span8, 762 | .uneditable-input.span8 { 763 | width: 462px; 764 | } 765 | input.span7, 766 | textarea.span7, 767 | .uneditable-input.span7 { 768 | width: 400px; 769 | } 770 | input.span6, 771 | textarea.span6, 772 | .uneditable-input.span6 { 773 | width: 338px; 774 | } 775 | input.span5, 776 | textarea.span5, 777 | .uneditable-input.span5 { 778 | width: 276px; 779 | } 780 | input.span4, 781 | textarea.span4, 782 | .uneditable-input.span4 { 783 | width: 214px; 784 | } 785 | input.span3, 786 | textarea.span3, 787 | .uneditable-input.span3 { 788 | width: 152px; 789 | } 790 | input.span2, 791 | textarea.span2, 792 | .uneditable-input.span2 { 793 | width: 90px; 794 | } 795 | input.span1, 796 | textarea.span1, 797 | .uneditable-input.span1 { 798 | width: 28px; 799 | } 800 | } 801 | 802 | @media (max-width: 767px) { 803 | body { 804 | padding-right: 20px; 805 | padding-left: 20px; 806 | } 807 | .navbar-fixed-top, 808 | .navbar-fixed-bottom, 809 | .navbar-static-top { 810 | margin-right: -20px; 811 | margin-left: -20px; 812 | } 813 | .container-fluid { 814 | padding: 0; 815 | } 816 | .dl-horizontal dt { 817 | float: none; 818 | width: auto; 819 | clear: none; 820 | text-align: left; 821 | } 822 | .dl-horizontal dd { 823 | margin-left: 0; 824 | } 825 | .container { 826 | width: auto; 827 | } 828 | .row-fluid { 829 | width: 100%; 830 | } 831 | .row, 832 | .thumbnails { 833 | margin-left: 0; 834 | } 835 | .thumbnails > li { 836 | float: none; 837 | margin-left: 0; 838 | } 839 | [class*="span"], 840 | .uneditable-input[class*="span"], 841 | .row-fluid [class*="span"] { 842 | display: block; 843 | float: none; 844 | width: 100%; 845 | margin-left: 0; 846 | -webkit-box-sizing: border-box; 847 | -moz-box-sizing: border-box; 848 | box-sizing: border-box; 849 | } 850 | .span12, 851 | .row-fluid .span12 { 852 | width: 100%; 853 | -webkit-box-sizing: border-box; 854 | -moz-box-sizing: border-box; 855 | box-sizing: border-box; 856 | } 857 | .row-fluid [class*="offset"]:first-child { 858 | margin-left: 0; 859 | } 860 | .input-large, 861 | .input-xlarge, 862 | .input-xxlarge, 863 | input[class*="span"], 864 | select[class*="span"], 865 | textarea[class*="span"], 866 | .uneditable-input { 867 | display: block; 868 | width: 100%; 869 | min-height: 30px; 870 | -webkit-box-sizing: border-box; 871 | -moz-box-sizing: border-box; 872 | box-sizing: border-box; 873 | } 874 | .input-prepend input, 875 | .input-append input, 876 | .input-prepend input[class*="span"], 877 | .input-append input[class*="span"] { 878 | display: inline-block; 879 | width: auto; 880 | } 881 | .controls-row [class*="span"] + [class*="span"] { 882 | margin-left: 0; 883 | } 884 | .modal { 885 | position: fixed; 886 | top: 20px; 887 | right: 20px; 888 | left: 20px; 889 | width: auto; 890 | margin: 0; 891 | } 892 | .modal.fade { 893 | top: -100px; 894 | } 895 | .modal.fade.in { 896 | top: 20px; 897 | } 898 | } 899 | 900 | @media (max-width: 480px) { 901 | .nav-collapse { 902 | -webkit-transform: translate3d(0, 0, 0); 903 | } 904 | .page-header h1 small { 905 | display: block; 906 | line-height: 20px; 907 | } 908 | input[type="checkbox"], 909 | input[type="radio"] { 910 | border: 1px solid #ccc; 911 | } 912 | .form-horizontal .control-label { 913 | float: none; 914 | width: auto; 915 | padding-top: 0; 916 | text-align: left; 917 | } 918 | .form-horizontal .controls { 919 | margin-left: 0; 920 | } 921 | .form-horizontal .control-list { 922 | padding-top: 0; 923 | } 924 | .form-horizontal .form-actions { 925 | padding-right: 10px; 926 | padding-left: 10px; 927 | } 928 | .media .pull-left, 929 | .media .pull-right { 930 | display: block; 931 | float: none; 932 | margin-bottom: 10px; 933 | } 934 | .media-object { 935 | margin-right: 0; 936 | margin-left: 0; 937 | } 938 | .modal { 939 | top: 10px; 940 | right: 10px; 941 | left: 10px; 942 | } 943 | .modal-header .close { 944 | padding: 10px; 945 | margin: -10px; 946 | } 947 | .carousel-caption { 948 | position: static; 949 | } 950 | } 951 | 952 | @media (max-width: 979px) { 953 | body { 954 | padding-top: 0; 955 | } 956 | .navbar-fixed-top, 957 | .navbar-fixed-bottom { 958 | position: static; 959 | } 960 | .navbar-fixed-top { 961 | margin-bottom: 20px; 962 | } 963 | .navbar-fixed-bottom { 964 | margin-top: 20px; 965 | } 966 | .navbar-fixed-top .navbar-inner, 967 | .navbar-fixed-bottom .navbar-inner { 968 | padding: 5px; 969 | } 970 | .navbar .container { 971 | width: auto; 972 | padding: 0; 973 | } 974 | .navbar .brand { 975 | padding-right: 10px; 976 | padding-left: 10px; 977 | margin: 0 0 0 -5px; 978 | } 979 | .nav-collapse { 980 | clear: both; 981 | } 982 | .nav-collapse .nav { 983 | float: none; 984 | margin: 0 0 10px; 985 | } 986 | .nav-collapse .nav > li { 987 | float: none; 988 | } 989 | .nav-collapse .nav > li > a { 990 | margin-bottom: 2px; 991 | } 992 | .nav-collapse .nav > .divider-vertical { 993 | display: none; 994 | } 995 | .nav-collapse .nav .nav-header { 996 | color: #777777; 997 | text-shadow: none; 998 | } 999 | .nav-collapse .nav > li > a, 1000 | .nav-collapse .dropdown-menu a { 1001 | padding: 9px 15px; 1002 | font-weight: bold; 1003 | color: #777777; 1004 | -webkit-border-radius: 3px; 1005 | -moz-border-radius: 3px; 1006 | border-radius: 3px; 1007 | } 1008 | .nav-collapse .btn { 1009 | padding: 4px 10px 4px; 1010 | font-weight: normal; 1011 | -webkit-border-radius: 4px; 1012 | -moz-border-radius: 4px; 1013 | border-radius: 4px; 1014 | } 1015 | .nav-collapse .dropdown-menu li + li a { 1016 | margin-bottom: 2px; 1017 | } 1018 | .nav-collapse .nav > li > a:hover, 1019 | .nav-collapse .nav > li > a:focus, 1020 | .nav-collapse .dropdown-menu a:hover, 1021 | .nav-collapse .dropdown-menu a:focus { 1022 | background-color: #f2f2f2; 1023 | } 1024 | .navbar-inverse .nav-collapse .nav > li > a, 1025 | .navbar-inverse .nav-collapse .dropdown-menu a { 1026 | color: #999999; 1027 | } 1028 | .navbar-inverse .nav-collapse .nav > li > a:hover, 1029 | .navbar-inverse .nav-collapse .nav > li > a:focus, 1030 | .navbar-inverse .nav-collapse .dropdown-menu a:hover, 1031 | .navbar-inverse .nav-collapse .dropdown-menu a:focus { 1032 | background-color: #111111; 1033 | } 1034 | .nav-collapse.in .btn-group { 1035 | padding: 0; 1036 | margin-top: 5px; 1037 | } 1038 | .nav-collapse .dropdown-menu { 1039 | position: static; 1040 | top: auto; 1041 | left: auto; 1042 | display: none; 1043 | float: none; 1044 | max-width: none; 1045 | padding: 0; 1046 | margin: 0 15px; 1047 | background-color: transparent; 1048 | border: none; 1049 | -webkit-border-radius: 0; 1050 | -moz-border-radius: 0; 1051 | border-radius: 0; 1052 | -webkit-box-shadow: none; 1053 | -moz-box-shadow: none; 1054 | box-shadow: none; 1055 | } 1056 | .nav-collapse .open > .dropdown-menu { 1057 | display: block; 1058 | } 1059 | .nav-collapse .dropdown-menu:before, 1060 | .nav-collapse .dropdown-menu:after { 1061 | display: none; 1062 | } 1063 | .nav-collapse .dropdown-menu .divider { 1064 | display: none; 1065 | } 1066 | .nav-collapse .nav > li > .dropdown-menu:before, 1067 | .nav-collapse .nav > li > .dropdown-menu:after { 1068 | display: none; 1069 | } 1070 | .nav-collapse .navbar-form, 1071 | .nav-collapse .navbar-search { 1072 | float: none; 1073 | padding: 10px 15px; 1074 | margin: 10px 0; 1075 | border-top: 1px solid #f2f2f2; 1076 | border-bottom: 1px solid #f2f2f2; 1077 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1078 | -moz-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1079 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1080 | } 1081 | .navbar-inverse .nav-collapse .navbar-form, 1082 | .navbar-inverse .nav-collapse .navbar-search { 1083 | border-top-color: #111111; 1084 | border-bottom-color: #111111; 1085 | } 1086 | .navbar .nav-collapse .nav.pull-right { 1087 | float: none; 1088 | margin-left: 0; 1089 | } 1090 | .nav-collapse, 1091 | .nav-collapse.collapse { 1092 | height: 0; 1093 | overflow: hidden; 1094 | } 1095 | .navbar .btn-navbar { 1096 | display: block; 1097 | } 1098 | .navbar-static .navbar-inner { 1099 | padding-right: 10px; 1100 | padding-left: 10px; 1101 | } 1102 | } 1103 | 1104 | @media (min-width: 980px) { 1105 | .nav-collapse.collapse { 1106 | height: auto !important; 1107 | overflow: visible !important; 1108 | } 1109 | } 1110 | -------------------------------------------------------------------------------- /othello.js: -------------------------------------------------------------------------------- 1 | var othello = {}; 2 | 3 | (function () { 4 | 'use strict'; 5 | 6 | // Utilities {{{1 7 | 8 | function memoize(f) { 9 | var memo = {}; 10 | var first = 0; 11 | var second = 0; 12 | return function () { 13 | if (arguments[0] === 'stat') 14 | return [first, second]; 15 | var key = JSON.stringify(arguments); 16 | if (memo[key] === undefined) { 17 | memo[key] = f.apply(this, arguments); 18 | first++; 19 | } else { 20 | second++; 21 | } 22 | return memo[key]; 23 | }; 24 | } 25 | 26 | function delay(expressionAsFunction) { 27 | var result; 28 | var isEvaluated = false; 29 | 30 | return function () { 31 | if (!isEvaluated) { 32 | result = expressionAsFunction(); 33 | isEvaluated = true; 34 | } 35 | return result; 36 | }; 37 | } 38 | 39 | function force(promise) { 40 | return promise(); 41 | } 42 | 43 | function sum(ns) { 44 | return ns.reduce(function (t, n) {return t + n;}); 45 | } 46 | 47 | function random(n) { 48 | return Math.floor(Math.random() * n); 49 | } 50 | 51 | 52 | 53 | 54 | // Core logic {{{1 55 | 56 | var m = location.href.match(/\?n=(\d+)$/); 57 | var N = m === null ? 8 : parseInt(m[1]); 58 | 59 | var EMPTY = 'empty'; 60 | var WHITE = 'white'; 61 | var BLACK = 'black'; 62 | 63 | function ix(x, y) { 64 | return x + y * N; 65 | } 66 | 67 | function makeInitialGameBoard() { 68 | var board = []; 69 | 70 | for (var x = 0; x < N; x++) 71 | for (var y = 0; y < N; y++) 72 | board[ix(x, y)] = EMPTY; 73 | 74 | var x2 = N >> 1; 75 | var y2 = N >> 1; 76 | board[ix(x2 - 1, y2 - 1)] = WHITE; 77 | board[ix(x2 - 1, y2 - 0)] = BLACK; 78 | board[ix(x2 - 0, y2 - 1)] = BLACK; 79 | board[ix(x2 - 0, y2 - 0)] = WHITE; 80 | 81 | return board; 82 | } 83 | 84 | function makeInitialGameTree() { 85 | return makeGameTree(makeInitialGameBoard(), BLACK, false, 1); 86 | } 87 | 88 | function makeGameTree(board, player, wasPassed, nest) { 89 | return { 90 | board: board, 91 | player: player, 92 | moves: listPossibleMoves(board, player, wasPassed, nest) 93 | }; 94 | } 95 | 96 | function listPossibleMoves(board, player, wasPassed, nest) { 97 | return completePassingMove( 98 | listAttackingMoves(board, player, nest), 99 | board, 100 | player, 101 | wasPassed, 102 | nest 103 | ); 104 | } 105 | 106 | function completePassingMove(attackingMoves, board, player, wasPassed, nest) { 107 | if (0 < attackingMoves.length) 108 | return attackingMoves; 109 | else if (!wasPassed) 110 | return [{ 111 | isPassingMove: true, 112 | gameTreePromise: delay(function () { 113 | return makeGameTree(board, nextPlayer(player), true, nest + 1); 114 | }) 115 | }]; 116 | else 117 | return []; 118 | } 119 | 120 | function listAttackingMovesN(board, player, nest) { 121 | var moves = []; 122 | 123 | for (var y = 0; y < N; y++) { 124 | for (var x = 0; x < N; x++) { 125 | var vulnerableCells = listVulnerableCells(board, x, y, player); 126 | if (canAttack(vulnerableCells)) { 127 | moves.push({ 128 | x: x, 129 | y: y, 130 | gameTreePromise: (function (x, y, vulnerableCells) { 131 | return delay(function () { 132 | return makeGameTree( 133 | makeAttackedBoard(board, x, y, vulnerableCells, player), 134 | nextPlayer(player), 135 | false, 136 | nest + 1 137 | ); 138 | }); 139 | })(x, y, vulnerableCells) 140 | }); 141 | } 142 | } 143 | } 144 | 145 | return moves; 146 | } 147 | 148 | function listAttackingMoves8(board, player, nest) { 149 | return listAttackableCells(board, player).map(function (c) { 150 | var x = c & 0x07; 151 | var y = c >> 3; 152 | return { 153 | x: x, 154 | y: y, 155 | gameTreePromise: delay(function () { 156 | var vulnerableCells = listVulnerableCells(board, x, y, player); 157 | return makeGameTree( 158 | makeAttackedBoard(board, x, y, vulnerableCells, player), 159 | nextPlayer(player), 160 | false, 161 | nest + 1 162 | ); 163 | }) 164 | }; 165 | }); 166 | } 167 | 168 | var listAttackingMoves = N === 8 ? listAttackingMoves8 : listAttackingMovesN; 169 | 170 | function nextPlayer(player) { 171 | return player === BLACK ? WHITE : BLACK; 172 | } 173 | 174 | function canAttack(vulnerableCells) { 175 | return vulnerableCells.length; 176 | } 177 | 178 | function makeAttackedBoard(board, x, y, vulnerableCells, player) { 179 | var newBoard = board.slice(); 180 | newBoard[ix(x, y)] = player; 181 | for (var i = 0; i < vulnerableCells.length; i++) 182 | newBoard[vulnerableCells[i]] = player; 183 | return newBoard; 184 | } 185 | 186 | function listVulnerableCells(board, x, y, player) { 187 | var vulnerableCells = []; 188 | 189 | if (board[ix(x, y)] !== EMPTY) 190 | return vulnerableCells; 191 | 192 | var opponent = nextPlayer(player); 193 | for (var dx = -1; dx <= 1; dx++) { 194 | for (var dy = -1; dy <= 1; dy++) { 195 | if (dx === 0 && dy === 0) 196 | continue; 197 | for (var i = 1; i < N; i++) { 198 | var nx = x + i * dx; 199 | var ny = y + i * dy; 200 | if (nx < 0 || N <= nx || ny < 0 || N <= ny) 201 | break; 202 | var cell = board[ix(nx, ny)]; 203 | if (cell === player && 2 <= i) { 204 | for (var j = 1; j < i; j++) 205 | vulnerableCells.push(ix(x + j * dx, y + j * dy)); 206 | break; 207 | } 208 | if (cell !== opponent) 209 | break; 210 | } 211 | } 212 | } 213 | 214 | return vulnerableCells; 215 | } 216 | 217 | function judge(board) { 218 | var n = {}; 219 | n[BLACK] = 0; 220 | n[WHITE] = 0; 221 | n[EMPTY] = 0; 222 | for (var i = 0; i < board.length; i++) 223 | n[board[i]]++; 224 | 225 | if (n[BLACK] > n[WHITE]) 226 | return 1; 227 | if (n[BLACK] < n[WHITE]) 228 | return -1; 229 | return 0; 230 | } 231 | 232 | function nameMove(move) { 233 | if (move.isPassingMove) 234 | return 'Pass'; 235 | else 236 | return 'abcdefgh'[move.x] + '12345678'[move.y]; 237 | } 238 | 239 | 240 | 241 | 242 | // Core logic: Bit board {{{1 243 | // 244 | // Naming conventions: 245 | // b = black 246 | // w = white 247 | // o = offense 248 | // d = defense 249 | // e = empty 250 | // a = attackable 251 | // u = upper half of a board 252 | // l = lower half of a board 253 | // 254 | // Assumption: N = 8 255 | 256 | var N2 = N >> 1; 257 | 258 | function listAttackableCells(board, player) { 259 | var bb = makeBitBoard(board); 260 | var ou = player === BLACK ? bb.blackUpper : bb.whiteUpper; 261 | var ol = player === BLACK ? bb.blackLower : bb.whiteLower; 262 | var du = player === BLACK ? bb.whiteUpper : bb.blackUpper; 263 | var dl = player === BLACK ? bb.whiteLower : bb.blackLower; 264 | var eu = ~(ou | du); 265 | var el = ~(ol | dl); 266 | var au = 0; 267 | var al = 0; 268 | var at; 269 | 270 | at = listAttackableBitsAtUp(ou, ol, du, dl, eu, el); 271 | au |= at.upper; 272 | al |= at.lower; 273 | 274 | at = listAttackableBitsAtRightUp(ou, ol, du, dl, eu, el); 275 | au |= at.upper; 276 | al |= at.lower; 277 | 278 | au |= listAttackableBitsAtRight(ou, du, eu); 279 | al |= listAttackableBitsAtRight(ol, dl, el); 280 | 281 | at = listAttackableBitsAtRightDown(ou, ol, du, dl, eu, el); 282 | au |= at.upper; 283 | al |= at.lower; 284 | 285 | at = listAttackableBitsAtDown(ou, ol, du, dl, eu, el); 286 | au |= at.upper; 287 | al |= at.lower; 288 | 289 | at = listAttackableBitsAtLeftDown(ou, ol, du, dl, eu, el); 290 | au |= at.upper; 291 | al |= at.lower; 292 | 293 | au |= listAttackableBitsAtLeft(ou, du, eu); 294 | al |= listAttackableBitsAtLeft(ol, dl, el); 295 | 296 | at = listAttackableBitsAtLeftUp(ou, ol, du, dl, eu, el); 297 | au |= at.upper; 298 | al |= at.lower; 299 | 300 | return cellPositionsFromBitBoard(au, al); 301 | } 302 | 303 | function makeBitBoard(board) { 304 | // MSB LSB 305 | // 1a 1b 1c 1d 1e 1f 1g 1h MSB 306 | // 2a 2b 2c 2d 2e 2f 2g 2h 307 | // upper 3a 3b 3c 3d 3e 3f 3g 3h 308 | // 4a 4b 4c 4d 4e 4f 4g 4h 309 | // bit board = + = 310 | // 5a 5b 5c 5d 5e 5f 5g 5h 311 | // lower 6a 6b 6c 6d 6e 6f 6g 6h 312 | // 7a 7b 7c 7d 7e 7f 7g 7h 313 | // 8a 8b 8c 8d 8e 8f 8g 8h LSB 314 | var bu = 0; 315 | var bl = 0; 316 | var wu = 0; 317 | var wl = 0; 318 | var nu = N2 - 1; 319 | var nl = N - 1; 320 | var n = N - 1; 321 | for (var y = 0; y < N; y++) { 322 | for (var x = 0; x < N; x++) { 323 | if (y < N2) { 324 | var i = ix(x, y); 325 | bu |= (board[i] === BLACK ? 1 : 0) << (n-x) << ((nu-y) * N); 326 | wu |= (board[i] === WHITE ? 1 : 0) << (n-x) << ((nu-y) * N); 327 | } else { 328 | var j = ix(x, y); 329 | bl |= (board[j] === BLACK ? 1 : 0) << (n-x) << ((nl-y) * N); 330 | wl |= (board[j] === WHITE ? 1 : 0) << (n-x) << ((nl-y) * N); 331 | } 332 | } 333 | } 334 | return { 335 | blackUpper: bu, 336 | blackLower: bl, 337 | whiteUpper: wu, 338 | whiteLower: wl 339 | }; 340 | } 341 | 342 | function cellPositionsFromBitBoard(au, al) { 343 | var positions = []; 344 | 345 | for (var yu = 0; yu < N2 && au; yu++) { 346 | for (var xu = 0; xu < N && au; xu++) { 347 | if (au & 0x80000000) 348 | positions.push(ix(xu, yu)); 349 | au <<= 1; 350 | } 351 | } 352 | 353 | for (var yl = N2; yl < N && al; yl++) { 354 | for (var xl = 0; xl < N && al; xl++) { 355 | if (al & 0x80000000) 356 | positions.push(ix(xl, yl)); 357 | al <<= 1; 358 | } 359 | } 360 | 361 | return positions; 362 | } 363 | 364 | function shiftUp(u, l) { 365 | return (u << N) | 366 | (l >>> (N * (N2 - 1))); 367 | } 368 | 369 | function shiftDown(u, l) { 370 | return (l >>> N) | 371 | ((u & 0x000000ff) << (N * (N2 - 1))); 372 | } 373 | 374 | function listAttackableBitsAtUp(ou, ol, _du, _dl, eu, el) { 375 | var du = _du & 0x00ffffff; 376 | var dl = _dl & 0xffffff00; 377 | var tu = du & shiftUp(ou, ol); 378 | var tl = dl & shiftUp(ol, 0); 379 | tu |= du & shiftUp(tu, tl); 380 | tl |= dl & shiftUp(tl, 0); 381 | tu |= du & shiftUp(tu, tl); 382 | tl |= dl & shiftUp(tl, 0); 383 | tu |= du & shiftUp(tu, tl); 384 | tl |= dl & shiftUp(tl, 0); 385 | tu |= du & shiftUp(tu, tl); 386 | tl |= dl & shiftUp(tl, 0); 387 | tu |= du & shiftUp(tu, tl); 388 | tl |= dl & shiftUp(tl, 0); 389 | return { 390 | upper: eu & shiftUp(tu, tl), 391 | lower: el & shiftUp(tl, 0) 392 | }; 393 | } 394 | 395 | function listAttackableBitsAtRightUp(ou, ol, _du, _dl, eu, el) { 396 | var du = _du & 0x007e7e7e; 397 | var dl = _dl & 0x7e7e7e00; 398 | var tu = du & (shiftUp(ou, ol) >>> 1); 399 | var tl = dl & (shiftUp(ol, 0) >>> 1); 400 | tu |= du & (shiftUp(tu, tl) >>> 1); 401 | tl |= dl & (shiftUp(tl, 0) >>> 1); 402 | tu |= du & (shiftUp(tu, tl) >>> 1); 403 | tl |= dl & (shiftUp(tl, 0) >>> 1); 404 | tu |= du & (shiftUp(tu, tl) >>> 1); 405 | tl |= dl & (shiftUp(tl, 0) >>> 1); 406 | tu |= du & (shiftUp(tu, tl) >>> 1); 407 | tl |= dl & (shiftUp(tl, 0) >>> 1); 408 | tu |= du & (shiftUp(tu, tl) >>> 1); 409 | tl |= dl & (shiftUp(tl, 0) >>> 1); 410 | return { 411 | upper: eu & (shiftUp(tu, tl) >>> 1), 412 | lower: el & (shiftUp(tl, 0) >>> 1) 413 | }; 414 | } 415 | 416 | function listAttackableBitsAtRight(o, _d, e) { 417 | var d = _d & 0x7e7e7e7e; 418 | var t = d & (o >>> 1); 419 | t |= d & (t >>> 1); 420 | t |= d & (t >>> 1); 421 | t |= d & (t >>> 1); 422 | t |= d & (t >>> 1); 423 | t |= d & (t >>> 1); 424 | return e & (t >>> 1); 425 | } 426 | 427 | function listAttackableBitsAtRightDown(ou, ol, _du, _dl, eu, el) { 428 | var du = _du & 0x007e7e7e; 429 | var dl = _dl & 0x7e7e7e00; 430 | var tl = dl & (shiftDown(ou, ol) >>> 1); 431 | var tu = du & (shiftDown(0, ou) >>> 1); 432 | tl |= dl & (shiftDown(tu, tl) >>> 1); 433 | tu |= du & (shiftDown(0, tu) >>> 1); 434 | tl |= dl & (shiftDown(tu, tl) >>> 1); 435 | tu |= du & (shiftDown(0, tu) >>> 1); 436 | tl |= dl & (shiftDown(tu, tl) >>> 1); 437 | tu |= du & (shiftDown(0, tu) >>> 1); 438 | tl |= dl & (shiftDown(tu, tl) >>> 1); 439 | tu |= du & (shiftDown(0, tu) >>> 1); 440 | tl |= dl & (shiftDown(tu, tl) >>> 1); 441 | tu |= du & (shiftDown(0, tu) >>> 1); 442 | return { 443 | upper: eu & (shiftDown(0, tu) >>> 1), 444 | lower: el & (shiftDown(tu, tl) >>> 1) 445 | }; 446 | } 447 | 448 | function listAttackableBitsAtDown(ou, ol, _du, _dl, eu, el) { 449 | var du = _du & 0x00ffffff; 450 | var dl = _dl & 0xffffff00; 451 | var tl = dl & shiftDown(ou, ol); 452 | var tu = du & shiftDown(0, ou); 453 | tl |= dl & shiftDown(tu, tl); 454 | tu |= du & shiftDown(0, tu); 455 | tl |= dl & shiftDown(tu, tl); 456 | tu |= du & shiftDown(0, tu); 457 | tl |= dl & shiftDown(tu, tl); 458 | tu |= du & shiftDown(0, tu); 459 | tl |= dl & shiftDown(tu, tl); 460 | tu |= du & shiftDown(0, tu); 461 | tl |= dl & shiftDown(tu, tl); 462 | tu |= du & shiftDown(0, tu); 463 | return { 464 | upper: eu & shiftDown(0, tu), 465 | lower: el & shiftDown(tu, tl) 466 | }; 467 | } 468 | 469 | function listAttackableBitsAtLeftDown(ou, ol, _du, _dl, eu, el) { 470 | var du = _du & 0x007e7e7e; 471 | var dl = _dl & 0x7e7e7e00; 472 | var tl = dl & (shiftDown(ou, ol) << 1); 473 | var tu = du & (shiftDown(0, ou) << 1); 474 | tl |= dl & (shiftDown(tu, tl) << 1); 475 | tu |= du & (shiftDown(0, tu) << 1); 476 | tl |= dl & (shiftDown(tu, tl) << 1); 477 | tu |= du & (shiftDown(0, tu) << 1); 478 | tl |= dl & (shiftDown(tu, tl) << 1); 479 | tu |= du & (shiftDown(0, tu) << 1); 480 | tl |= dl & (shiftDown(tu, tl) << 1); 481 | tu |= du & (shiftDown(0, tu) << 1); 482 | tl |= dl & (shiftDown(tu, tl) << 1); 483 | tu |= du & (shiftDown(0, tu) << 1); 484 | return { 485 | upper: eu & (shiftDown(0, tu) << 1), 486 | lower: el & (shiftDown(tu, tl) << 1) 487 | }; 488 | } 489 | 490 | function listAttackableBitsAtLeft(o, _d, e) { 491 | var d = _d & 0x7e7e7e7e; 492 | var t = d & (o << 1); 493 | t |= d & (t << 1); 494 | t |= d & (t << 1); 495 | t |= d & (t << 1); 496 | t |= d & (t << 1); 497 | t |= d & (t << 1); 498 | return e & (t << 1); 499 | } 500 | 501 | function listAttackableBitsAtLeftUp(ou, ol, _du, _dl, eu, el) { 502 | var du = _du & 0x007e7e7e; 503 | var dl = _dl & 0x7e7e7e00; 504 | var tu = du & (shiftUp(ou, ol) << 1); 505 | var tl = dl & (shiftUp(ol, 0) << 1); 506 | tu |= du & (shiftUp(tu, tl) << 1); 507 | tl |= dl & (shiftUp(tl, 0) << 1); 508 | tu |= du & (shiftUp(tu, tl) << 1); 509 | tl |= dl & (shiftUp(tl, 0) << 1); 510 | tu |= du & (shiftUp(tu, tl) << 1); 511 | tl |= dl & (shiftUp(tl, 0) << 1); 512 | tu |= du & (shiftUp(tu, tl) << 1); 513 | tl |= dl & (shiftUp(tl, 0) << 1); 514 | tu |= du & (shiftUp(tu, tl) << 1); 515 | tl |= dl & (shiftUp(tl, 0) << 1); 516 | return { 517 | upper: eu & (shiftUp(tu, tl) << 1), 518 | lower: el & (shiftUp(tl, 0) << 1) 519 | }; 520 | } 521 | 522 | 523 | 524 | 525 | // AI {{{1 526 | 527 | var aiMakers = { 528 | mcts: makeMonteCarloTreeSearchBasedAI, 529 | pmc: makePrimitiveMonteCarloBasedAI 530 | }; 531 | 532 | function makeAI(playerType) { 533 | if (playerType in externalAITable) { 534 | return externalAITable[playerType]; 535 | } else { 536 | var tokens = playerType.split('-'); 537 | var aiType = tokens[0]; 538 | var level = parseInt(tokens[1]); 539 | var extras = tokens.slice(2); 540 | var scorePosition = scorePositions[aiType]; 541 | if (scorePosition !== undefined) { 542 | return makeScoreBasedAI({ 543 | level: level, 544 | scorePosition: scorePosition 545 | }); 546 | } else { 547 | return aiMakers[aiType]({ 548 | level: level, 549 | extras: extras 550 | }); 551 | } 552 | } 553 | } 554 | 555 | 556 | 557 | 558 | // AI: Weight table based + alpha-beta pruning {{{1 559 | 560 | function makeScorePositionWith(weightTable) { 561 | var wt = weightTable; 562 | return function (gameTree, player) { 563 | var board = gameTree.board; 564 | var opponent = nextPlayer(player); 565 | var ct = {}; 566 | ct[player] = 1; 567 | ct[opponent] = -1; 568 | ct[EMPTY] = 0; 569 | var s = 0; 570 | for (var i = 0; i < board.length; i++) 571 | s += ct[board[i]] * wt[i]; 572 | return s; 573 | }; 574 | } 575 | 576 | var scorePositions = { 577 | simpleCount: makeScorePositionWith((function () { 578 | var t = []; 579 | for (var x = 0; x < N; x++) 580 | for (var y = 0; y < N; y++) 581 | t[ix(x, y)] = 1; 582 | return t; 583 | })()), 584 | basic: makeScorePositionWith((function () { 585 | var t = []; 586 | for (var x = 0; x < N; x++) 587 | for (var y = 0; y < N; y++) 588 | t[ix(x, y)] = 589 | (x === 0 || x === N - 1 ? 10 : 1) * 590 | (y === 0 || y === N - 1 ? 10 : 1); 591 | return t; 592 | })()), 593 | better: makeScorePositionWith((function () { 594 | var t = []; 595 | for (var x = 0; x < N; x++) 596 | for (var y = 0; y < N; y++) 597 | t[ix(x, y)] = 598 | (x === 0 || x === N - 1 ? 10 : 1) * 599 | (y === 0 || y === N - 1 ? 10 : 1); 600 | t[ix(0, 1)] = t[ix(0, N - 2)] = t[ix(N - 1, 1)] = t[ix(N - 1, N - 2)] = 601 | t[ix(1, 0)] = t[ix(N - 2, 0)] = t[ix(1, N - 1)] = t[ix(N - 2, N - 1)] = 0; 602 | return t; 603 | })()), 604 | edgesAndCorners: makeScorePositionWith((function () { 605 | var t = []; 606 | for (var x = 0; x < N; x++) 607 | for (var y = 0; y < N; y++) 608 | t[ix(x, y)] = 0; 609 | for (var x = 2; x < N - 2; x++) { 610 | t[ix(x, 0)] = 10; 611 | t[ix(x, N - 1)] = 10; 612 | } 613 | for (var y = 2; y < N - 2; y++) { 614 | t[ix(0, y)] = 10; 615 | t[ix(N - 1, y)] = 10; 616 | } 617 | t[ix(0, 1 )] = t[ix(1, 0 )] = t[ix(1, 1 )] = 618 | t[ix(N - 1, 1 )] = t[ix(N - 2, 0 )] = t[ix(N - 2, 1 )] = 619 | t[ix(1, N - 1)] = t[ix(0, N - 2)] = t[ix(1, N - 2)] = 620 | t[ix(N - 2, N - 1)] = t[ix(N - 1, N - 2)] = t[ix(N - 2, N - 2)] = -1; 621 | 622 | t[ix(0, 0)] = t[ix(0, N - 1)] = 623 | t[ix(N - 1, 0)] = t[ix(N - 1, N - 1)] = 100; 624 | return t; 625 | })()), 626 | moveCount: function (gameTree, player) { 627 | return gameTree.actualMoveCount * (gameTree.player == player ? 1 : -1); 628 | }, 629 | moveCountAndPositions: function (gameTree, player) { 630 | return scorePositions.moveCount(gameTree, player) + 631 | scorePositions.edgesAndCorners(gameTree, player); 632 | } 633 | }; 634 | 635 | function makeScoreBasedAI(config) { 636 | return { 637 | findTheBestMove: function (gameTree) { 638 | var ratings = calculateMaxRatings( 639 | limitGameTreeWithFeasibleDepth(gameTree, config.level), 640 | gameTree.player, 641 | Number.MIN_VALUE, 642 | Number.MAX_VALUE, 643 | config.scorePosition 644 | ); 645 | var maxRating = Math.max.apply(null, ratings); 646 | return gameTree.moves[ratings.indexOf(maxRating)]; 647 | } 648 | }; 649 | } 650 | 651 | function limitGameTreeWithFeasibleDepth(gameTree, maxBoards) { 652 | return limitGameTreeDepth( 653 | gameTree, 654 | estimateFeasibleDepth(gameTree, maxBoards) 655 | ); 656 | } 657 | 658 | function estimateFeasibleDepth(gameTree, maxBoards) { 659 | var oldApproxBoards = 1; 660 | var newApproxBoards = 1; 661 | var depth = 0; 662 | while (newApproxBoards <= maxBoards && 1 <= gameTree.moves.length) { 663 | oldApproxBoards = newApproxBoards; 664 | newApproxBoards *= gameTree.moves.length; 665 | depth += 1; 666 | gameTree = force(gameTree.moves[0].gameTreePromise); 667 | } 668 | var oldDiff = oldApproxBoards - maxBoards; 669 | var newDiff = newApproxBoards - maxBoards; 670 | return Math.abs(newDiff) - Math.abs(oldDiff) <= 0 ? depth : depth - 1; 671 | } 672 | 673 | function limitGameTreeDepth(gameTree, depth) { 674 | return { 675 | board: gameTree.board, 676 | player: gameTree.player, 677 | moves: depth === 0 ? [] : gameTree.moves.map(function (m) { 678 | return { 679 | isPassingMove: m.isPassingMove, 680 | x: m.x, 681 | y: m.y, 682 | gameTreePromise: delay(function () { 683 | return limitGameTreeDepth(force(m.gameTreePromise), depth - 1); 684 | }) 685 | }; 686 | }), 687 | actualMoveCount: gameTree.moves.length 688 | }; 689 | } 690 | 691 | function ratePosition(gameTree, player, scorePosition) { 692 | if (1 <= gameTree.moves.length) { 693 | var choose = gameTree.player === player ? Math.max : Math.min; 694 | return choose.apply(null, calculateRatings(gameTree, player, scorePosition)); 695 | } else { 696 | return scorePosition(gameTree, player); 697 | } 698 | } 699 | 700 | function calculateRatings(gameTree, player, scorePosition) { 701 | return gameTree.moves.map(function (m) { 702 | return ratePosition(force(m.gameTreePromise), player, scorePosition); 703 | }); 704 | } 705 | 706 | function ratePositionWithAlphaBetaPruning(gameTree, player, lowerLimit, upperLimit, scorePosition) { 707 | if (1 <= gameTree.moves.length) { 708 | var judge = 709 | gameTree.player === player ? 710 | Math.max : 711 | Math.min; 712 | var rate = 713 | gameTree.player === player ? 714 | calculateMaxRatings : 715 | calculateMinRatings; 716 | return judge.apply(null, rate(gameTree, player, lowerLimit, upperLimit, scorePosition)); 717 | } else { 718 | return scorePosition(gameTree, player); 719 | } 720 | } 721 | 722 | function calculateMaxRatings(gameTree, player, lowerLimit, upperLimit, scorePosition) { 723 | var ratings = []; 724 | var newLowerLimit = lowerLimit; 725 | for (var i = 0; i < gameTree.moves.length; i++) { 726 | var r = ratePositionWithAlphaBetaPruning( 727 | force(gameTree.moves[i].gameTreePromise), 728 | player, 729 | newLowerLimit, 730 | upperLimit, 731 | scorePosition 732 | ); 733 | ratings.push(r); 734 | if (upperLimit <= r) 735 | break; 736 | newLowerLimit = Math.max(r, newLowerLimit); 737 | } 738 | return ratings; 739 | } 740 | 741 | function calculateMinRatings(gameTree, player, lowerLimit, upperLimit, scorePosition) { 742 | var ratings = []; 743 | var newUpperLimit = upperLimit; 744 | for (var i = 0; i < gameTree.moves.length; i++) { 745 | var r = ratePositionWithAlphaBetaPruning( 746 | force(gameTree.moves[i].gameTreePromise), 747 | player, 748 | upperLimit, 749 | newUpperLimit, 750 | scorePosition 751 | ); 752 | ratings.push(r); 753 | if (r <= lowerLimit) 754 | break; 755 | newUpperLimit = Math.min(r, newUpperLimit); 756 | } 757 | return ratings; 758 | } 759 | 760 | 761 | 762 | 763 | // AI: Monte Carlo Tree Search {{{1 764 | 765 | function makeMonteCarloTreeSearchBasedAI(options) { 766 | return { 767 | findTheBestMove: function (gameTree) { 768 | return tryMonteCarloTreeSearch(gameTree, options.level, options.extras.length > 0); 769 | } 770 | }; 771 | } 772 | 773 | function tryMonteCarloTreeSearch(rootGameTree, maxTries, isBroken) { 774 | var root = new Node(rootGameTree, null, null); 775 | 776 | for (var i = 0; i < maxTries; i++) { 777 | var node = root; 778 | 779 | while (node.untriedMoves.length === 0 && node.childNodes.length !== 0) 780 | node = node.selectChild(rootGameTree.player, isBroken); 781 | 782 | if (node.untriedMoves.length !== 0) 783 | node = node.expandChild(); 784 | 785 | var won = node.simulate(rootGameTree.player); 786 | 787 | node.backpropagate(won); 788 | } 789 | 790 | var vs = root.childNodes.map(function (n) {return n.visits;}); 791 | return root.childNodes[vs.indexOf(Math.max.apply(null, vs))].move; 792 | } 793 | 794 | function Node(gameTree, parentNode, move) { 795 | this.gameTree = gameTree; 796 | this.parentNode = parentNode; 797 | this.move = move; 798 | this.childNodes = []; 799 | this.wins = 0; 800 | this.visits = 0; 801 | this.untriedMoves = gameTree.moves.slice(); 802 | } 803 | 804 | Node.prototype.selectChild = function (rootGamePlayer, isBroken) { 805 | var currentPlayer = this.gameTree.player; 806 | var totalVisits = this.visits; 807 | var values = this.childNodes.map(function (n) { 808 | var winPercentage = n.wins / n.visits; 809 | if (!isBroken && currentPlayer != rootGamePlayer) { 810 | winPercentage = (n.visits - n.wins) / n.visits 811 | } 812 | return winPercentage + 813 | Math.sqrt(2 * Math.log(totalVisits) / n.visits); 814 | }); 815 | return this.childNodes[values.indexOf(Math.max.apply(null, values))]; 816 | }; 817 | 818 | Node.prototype.expandChild = function () { 819 | var i = random(this.untriedMoves.length); 820 | var move = this.untriedMoves.splice(i, 1)[0]; 821 | var child = new Node(force(move.gameTreePromise), this, move); 822 | this.childNodes.push(child); 823 | return child; 824 | }; 825 | 826 | Node.prototype.simulate = function (player) { 827 | var gameTree = this.gameTree; 828 | while (gameTree.moves.length !== 0) { 829 | var i = random(gameTree.moves.length); 830 | gameTree = force(gameTree.moves[i].gameTreePromise); 831 | } 832 | return judge(gameTree.board) * (player === BLACK ? 1 : -1) / 2 + 0.5; 833 | }; 834 | 835 | Node.prototype.backpropagate = function (result) { 836 | for (var node = this; node !== null; node = node.parentNode) 837 | node.update(result); 838 | }; 839 | 840 | Node.prototype.update = function (won) { 841 | this.wins += won; 842 | this.visits += 1; 843 | }; 844 | 845 | Node.prototype.visualize = function (indent) { 846 | indent = indent || 0; 847 | var ss = []; 848 | var i; 849 | ss.push('\n'); 850 | for (i = 0; i < indent; i++) 851 | ss.push('| '); 852 | ss.push('W='); ss.push(this.wins); 853 | ss.push('/'); 854 | ss.push('V='); ss.push(this.visits); 855 | ss.push('/'); 856 | ss.push('U='); ss.push(this.untriedMoves.length); 857 | for (i = 0; i < this.childNodes.length; i++) 858 | ss.push(this.childNodes[i].visualize(indent + 1)); 859 | return ss.join(''); 860 | }; 861 | 862 | 863 | 864 | 865 | // AI: Primitive Monte Carlo {{{1 866 | 867 | function makePrimitiveMonteCarloBasedAI(options) { 868 | return { 869 | findTheBestMove: function (gameTree) { 870 | return tryPrimitiveMonteCarloSimulation( 871 | gameTree, 872 | options.level, 873 | options.extras[0] 874 | ); 875 | } 876 | }; 877 | } 878 | 879 | function tryPrimitiveMonteCarloSimulation(rootGameTree, maxTries, iterStyle) { 880 | var moveCount = rootGameTree.moves.length; 881 | var lastMove = rootGameTree.moves[moveCount - 1]; 882 | var scores = rootGameTree.moves.map(function (m) { 883 | var s = 0; 884 | var eachTries = 885 | iterStyle === 'm' 886 | ? maxTries 887 | : Math.floor(maxTries / moveCount) 888 | + (m === lastMove ? maxTries % moveCount : 0); 889 | for (var i = 0; i < eachTries; i++) 890 | s += simulateRandomGame(m, rootGameTree.player); 891 | return s; 892 | }); 893 | var maxScore = Math.max.apply(null, scores); 894 | return rootGameTree.moves[scores.indexOf(maxScore)]; 895 | } 896 | 897 | function simulateRandomGame(move, player) { 898 | var gt = force(move.gameTreePromise); 899 | while (gt.moves.length !== 0) 900 | gt = force(gt.moves[random(gt.moves.length)].gameTreePromise); 901 | return judge(gt.board) * (player === BLACK ? 1 : -1); 902 | } 903 | 904 | 905 | 906 | 907 | // API {{{1 908 | 909 | var externalAITable = {}; 910 | 911 | var lastAIType; 912 | 913 | function registerAI(ai) { 914 | externalAITable[lastAIType] = ai; 915 | } 916 | 917 | 918 | function addNewAI() { 919 | var aiUrl = $('#new-ai-url').val(); 920 | var originalLabel = $('#add-new-ai-button').text(); 921 | if (externalAITable[aiUrl] === undefined) { 922 | lastAIType = aiUrl; 923 | $('#add-new-ai-button').text('Loading...').prop('disabled', true); 924 | $.getScript(aiUrl, function () { 925 | $('#black-player-type, #white-player-type').append( 926 | '' 927 | ); 928 | $('#white-player-type').val(aiUrl).change(); 929 | $('#add-new-ai-button').text(originalLabel).removeProp('disabled'); 930 | }); 931 | } else { 932 | $('#add-new-ai-button').text('Already loaded').prop('disabled', true); 933 | setTimeout( 934 | function () { 935 | $('#add-new-ai-button').text(originalLabel).removeProp('disabled'); 936 | }, 937 | 1000 938 | ); 939 | } 940 | } 941 | 942 | 943 | 944 | 945 | // Public API {{{1 946 | 947 | othello.force = force; 948 | othello.delay = delay; 949 | othello.EMPTY = EMPTY; 950 | othello.WHITE = WHITE; 951 | othello.BLACK = BLACK; 952 | othello.nextPlayer = nextPlayer; 953 | othello.registerAI = registerAI; 954 | othello.N = N; 955 | othello.ix = ix; 956 | othello.makeInitialGameBoard = makeInitialGameBoard; 957 | othello.judge = judge; 958 | othello.addNewAI = addNewAI; 959 | othello.makeAI = makeAI; 960 | othello.makeInitialGameTree = makeInitialGameTree; 961 | othello.nameMove = nameMove; 962 | 963 | 964 | 965 | 966 | // }}} 967 | })(); 968 | // vim: expandtab softtabstop=2 shiftwidth=2 foldmethod=marker 969 | -------------------------------------------------------------------------------- /vendor/twitter-bootstrap/js/bootstrap.js: -------------------------------------------------------------------------------- 1 | /* =================================================== 2 | * bootstrap-transition.js v2.3.1 3 | * http://twitter.github.com/bootstrap/javascript.html#transitions 4 | * =================================================== 5 | * Copyright 2012 Twitter, Inc. 6 | * 7 | * Licensed under the Apache License, Version 2.0 (the "License"); 8 | * you may not use this file except in compliance with the License. 9 | * You may obtain a copy of the License at 10 | * 11 | * http://www.apache.org/licenses/LICENSE-2.0 12 | * 13 | * Unless required by applicable law or agreed to in writing, software 14 | * distributed under the License is distributed on an "AS IS" BASIS, 15 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 16 | * See the License for the specific language governing permissions and 17 | * limitations under the License. 18 | * ========================================================== */ 19 | 20 | 21 | !function ($) { 22 | 23 | "use strict"; // jshint ;_; 24 | 25 | 26 | /* CSS TRANSITION SUPPORT (http://www.modernizr.com/) 27 | * ======================================================= */ 28 | 29 | $(function () { 30 | 31 | $.support.transition = (function () { 32 | 33 | var transitionEnd = (function () { 34 | 35 | var el = document.createElement('bootstrap') 36 | , transEndEventNames = { 37 | 'WebkitTransition' : 'webkitTransitionEnd' 38 | , 'MozTransition' : 'transitionend' 39 | , 'OTransition' : 'oTransitionEnd otransitionend' 40 | , 'transition' : 'transitionend' 41 | } 42 | , name 43 | 44 | for (name in transEndEventNames){ 45 | if (el.style[name] !== undefined) { 46 | return transEndEventNames[name] 47 | } 48 | } 49 | 50 | }()) 51 | 52 | return transitionEnd && { 53 | end: transitionEnd 54 | } 55 | 56 | })() 57 | 58 | }) 59 | 60 | }(window.jQuery);/* ========================================================== 61 | * bootstrap-alert.js v2.3.1 62 | * http://twitter.github.com/bootstrap/javascript.html#alerts 63 | * ========================================================== 64 | * Copyright 2012 Twitter, Inc. 65 | * 66 | * Licensed under the Apache License, Version 2.0 (the "License"); 67 | * you may not use this file except in compliance with the License. 68 | * You may obtain a copy of the License at 69 | * 70 | * http://www.apache.org/licenses/LICENSE-2.0 71 | * 72 | * Unless required by applicable law or agreed to in writing, software 73 | * distributed under the License is distributed on an "AS IS" BASIS, 74 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 75 | * See the License for the specific language governing permissions and 76 | * limitations under the License. 77 | * ========================================================== */ 78 | 79 | 80 | !function ($) { 81 | 82 | "use strict"; // jshint ;_; 83 | 84 | 85 | /* ALERT CLASS DEFINITION 86 | * ====================== */ 87 | 88 | var dismiss = '[data-dismiss="alert"]' 89 | , Alert = function (el) { 90 | $(el).on('click', dismiss, this.close) 91 | } 92 | 93 | Alert.prototype.close = function (e) { 94 | var $this = $(this) 95 | , selector = $this.attr('data-target') 96 | , $parent 97 | 98 | if (!selector) { 99 | selector = $this.attr('href') 100 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7 101 | } 102 | 103 | $parent = $(selector) 104 | 105 | e && e.preventDefault() 106 | 107 | $parent.length || ($parent = $this.hasClass('alert') ? $this : $this.parent()) 108 | 109 | $parent.trigger(e = $.Event('close')) 110 | 111 | if (e.isDefaultPrevented()) return 112 | 113 | $parent.removeClass('in') 114 | 115 | function removeElement() { 116 | $parent 117 | .trigger('closed') 118 | .remove() 119 | } 120 | 121 | $.support.transition && $parent.hasClass('fade') ? 122 | $parent.on($.support.transition.end, removeElement) : 123 | removeElement() 124 | } 125 | 126 | 127 | /* ALERT PLUGIN DEFINITION 128 | * ======================= */ 129 | 130 | var old = $.fn.alert 131 | 132 | $.fn.alert = function (option) { 133 | return this.each(function () { 134 | var $this = $(this) 135 | , data = $this.data('alert') 136 | if (!data) $this.data('alert', (data = new Alert(this))) 137 | if (typeof option == 'string') data[option].call($this) 138 | }) 139 | } 140 | 141 | $.fn.alert.Constructor = Alert 142 | 143 | 144 | /* ALERT NO CONFLICT 145 | * ================= */ 146 | 147 | $.fn.alert.noConflict = function () { 148 | $.fn.alert = old 149 | return this 150 | } 151 | 152 | 153 | /* ALERT DATA-API 154 | * ============== */ 155 | 156 | $(document).on('click.alert.data-api', dismiss, Alert.prototype.close) 157 | 158 | }(window.jQuery);/* ============================================================ 159 | * bootstrap-button.js v2.3.1 160 | * http://twitter.github.com/bootstrap/javascript.html#buttons 161 | * ============================================================ 162 | * Copyright 2012 Twitter, Inc. 163 | * 164 | * Licensed under the Apache License, Version 2.0 (the "License"); 165 | * you may not use this file except in compliance with the License. 166 | * You may obtain a copy of the License at 167 | * 168 | * http://www.apache.org/licenses/LICENSE-2.0 169 | * 170 | * Unless required by applicable law or agreed to in writing, software 171 | * distributed under the License is distributed on an "AS IS" BASIS, 172 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 173 | * See the License for the specific language governing permissions and 174 | * limitations under the License. 175 | * ============================================================ */ 176 | 177 | 178 | !function ($) { 179 | 180 | "use strict"; // jshint ;_; 181 | 182 | 183 | /* BUTTON PUBLIC CLASS DEFINITION 184 | * ============================== */ 185 | 186 | var Button = function (element, options) { 187 | this.$element = $(element) 188 | this.options = $.extend({}, $.fn.button.defaults, options) 189 | } 190 | 191 | Button.prototype.setState = function (state) { 192 | var d = 'disabled' 193 | , $el = this.$element 194 | , data = $el.data() 195 | , val = $el.is('input') ? 'val' : 'html' 196 | 197 | state = state + 'Text' 198 | data.resetText || $el.data('resetText', $el[val]()) 199 | 200 | $el[val](data[state] || this.options[state]) 201 | 202 | // push to event loop to allow forms to submit 203 | setTimeout(function () { 204 | state == 'loadingText' ? 205 | $el.addClass(d).attr(d, d) : 206 | $el.removeClass(d).removeAttr(d) 207 | }, 0) 208 | } 209 | 210 | Button.prototype.toggle = function () { 211 | var $parent = this.$element.closest('[data-toggle="buttons-radio"]') 212 | 213 | $parent && $parent 214 | .find('.active') 215 | .removeClass('active') 216 | 217 | this.$element.toggleClass('active') 218 | } 219 | 220 | 221 | /* BUTTON PLUGIN DEFINITION 222 | * ======================== */ 223 | 224 | var old = $.fn.button 225 | 226 | $.fn.button = function (option) { 227 | return this.each(function () { 228 | var $this = $(this) 229 | , data = $this.data('button') 230 | , options = typeof option == 'object' && option 231 | if (!data) $this.data('button', (data = new Button(this, options))) 232 | if (option == 'toggle') data.toggle() 233 | else if (option) data.setState(option) 234 | }) 235 | } 236 | 237 | $.fn.button.defaults = { 238 | loadingText: 'loading...' 239 | } 240 | 241 | $.fn.button.Constructor = Button 242 | 243 | 244 | /* BUTTON NO CONFLICT 245 | * ================== */ 246 | 247 | $.fn.button.noConflict = function () { 248 | $.fn.button = old 249 | return this 250 | } 251 | 252 | 253 | /* BUTTON DATA-API 254 | * =============== */ 255 | 256 | $(document).on('click.button.data-api', '[data-toggle^=button]', function (e) { 257 | var $btn = $(e.target) 258 | if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn') 259 | $btn.button('toggle') 260 | }) 261 | 262 | }(window.jQuery);/* ========================================================== 263 | * bootstrap-carousel.js v2.3.1 264 | * http://twitter.github.com/bootstrap/javascript.html#carousel 265 | * ========================================================== 266 | * Copyright 2012 Twitter, Inc. 267 | * 268 | * Licensed under the Apache License, Version 2.0 (the "License"); 269 | * you may not use this file except in compliance with the License. 270 | * You may obtain a copy of the License at 271 | * 272 | * http://www.apache.org/licenses/LICENSE-2.0 273 | * 274 | * Unless required by applicable law or agreed to in writing, software 275 | * distributed under the License is distributed on an "AS IS" BASIS, 276 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 277 | * See the License for the specific language governing permissions and 278 | * limitations under the License. 279 | * ========================================================== */ 280 | 281 | 282 | !function ($) { 283 | 284 | "use strict"; // jshint ;_; 285 | 286 | 287 | /* CAROUSEL CLASS DEFINITION 288 | * ========================= */ 289 | 290 | var Carousel = function (element, options) { 291 | this.$element = $(element) 292 | this.$indicators = this.$element.find('.carousel-indicators') 293 | this.options = options 294 | this.options.pause == 'hover' && this.$element 295 | .on('mouseenter', $.proxy(this.pause, this)) 296 | .on('mouseleave', $.proxy(this.cycle, this)) 297 | } 298 | 299 | Carousel.prototype = { 300 | 301 | cycle: function (e) { 302 | if (!e) this.paused = false 303 | if (this.interval) clearInterval(this.interval); 304 | this.options.interval 305 | && !this.paused 306 | && (this.interval = setInterval($.proxy(this.next, this), this.options.interval)) 307 | return this 308 | } 309 | 310 | , getActiveIndex: function () { 311 | this.$active = this.$element.find('.item.active') 312 | this.$items = this.$active.parent().children() 313 | return this.$items.index(this.$active) 314 | } 315 | 316 | , to: function (pos) { 317 | var activeIndex = this.getActiveIndex() 318 | , that = this 319 | 320 | if (pos > (this.$items.length - 1) || pos < 0) return 321 | 322 | if (this.sliding) { 323 | return this.$element.one('slid', function () { 324 | that.to(pos) 325 | }) 326 | } 327 | 328 | if (activeIndex == pos) { 329 | return this.pause().cycle() 330 | } 331 | 332 | return this.slide(pos > activeIndex ? 'next' : 'prev', $(this.$items[pos])) 333 | } 334 | 335 | , pause: function (e) { 336 | if (!e) this.paused = true 337 | if (this.$element.find('.next, .prev').length && $.support.transition.end) { 338 | this.$element.trigger($.support.transition.end) 339 | this.cycle(true) 340 | } 341 | clearInterval(this.interval) 342 | this.interval = null 343 | return this 344 | } 345 | 346 | , next: function () { 347 | if (this.sliding) return 348 | return this.slide('next') 349 | } 350 | 351 | , prev: function () { 352 | if (this.sliding) return 353 | return this.slide('prev') 354 | } 355 | 356 | , slide: function (type, next) { 357 | var $active = this.$element.find('.item.active') 358 | , $next = next || $active[type]() 359 | , isCycling = this.interval 360 | , direction = type == 'next' ? 'left' : 'right' 361 | , fallback = type == 'next' ? 'first' : 'last' 362 | , that = this 363 | , e 364 | 365 | this.sliding = true 366 | 367 | isCycling && this.pause() 368 | 369 | $next = $next.length ? $next : this.$element.find('.item')[fallback]() 370 | 371 | e = $.Event('slide', { 372 | relatedTarget: $next[0] 373 | , direction: direction 374 | }) 375 | 376 | if ($next.hasClass('active')) return 377 | 378 | if (this.$indicators.length) { 379 | this.$indicators.find('.active').removeClass('active') 380 | this.$element.one('slid', function () { 381 | var $nextIndicator = $(that.$indicators.children()[that.getActiveIndex()]) 382 | $nextIndicator && $nextIndicator.addClass('active') 383 | }) 384 | } 385 | 386 | if ($.support.transition && this.$element.hasClass('slide')) { 387 | this.$element.trigger(e) 388 | if (e.isDefaultPrevented()) return 389 | $next.addClass(type) 390 | $next[0].offsetWidth // force reflow 391 | $active.addClass(direction) 392 | $next.addClass(direction) 393 | this.$element.one($.support.transition.end, function () { 394 | $next.removeClass([type, direction].join(' ')).addClass('active') 395 | $active.removeClass(['active', direction].join(' ')) 396 | that.sliding = false 397 | setTimeout(function () { that.$element.trigger('slid') }, 0) 398 | }) 399 | } else { 400 | this.$element.trigger(e) 401 | if (e.isDefaultPrevented()) return 402 | $active.removeClass('active') 403 | $next.addClass('active') 404 | this.sliding = false 405 | this.$element.trigger('slid') 406 | } 407 | 408 | isCycling && this.cycle() 409 | 410 | return this 411 | } 412 | 413 | } 414 | 415 | 416 | /* CAROUSEL PLUGIN DEFINITION 417 | * ========================== */ 418 | 419 | var old = $.fn.carousel 420 | 421 | $.fn.carousel = function (option) { 422 | return this.each(function () { 423 | var $this = $(this) 424 | , data = $this.data('carousel') 425 | , options = $.extend({}, $.fn.carousel.defaults, typeof option == 'object' && option) 426 | , action = typeof option == 'string' ? option : options.slide 427 | if (!data) $this.data('carousel', (data = new Carousel(this, options))) 428 | if (typeof option == 'number') data.to(option) 429 | else if (action) data[action]() 430 | else if (options.interval) data.pause().cycle() 431 | }) 432 | } 433 | 434 | $.fn.carousel.defaults = { 435 | interval: 5000 436 | , pause: 'hover' 437 | } 438 | 439 | $.fn.carousel.Constructor = Carousel 440 | 441 | 442 | /* CAROUSEL NO CONFLICT 443 | * ==================== */ 444 | 445 | $.fn.carousel.noConflict = function () { 446 | $.fn.carousel = old 447 | return this 448 | } 449 | 450 | /* CAROUSEL DATA-API 451 | * ================= */ 452 | 453 | $(document).on('click.carousel.data-api', '[data-slide], [data-slide-to]', function (e) { 454 | var $this = $(this), href 455 | , $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7 456 | , options = $.extend({}, $target.data(), $this.data()) 457 | , slideIndex 458 | 459 | $target.carousel(options) 460 | 461 | if (slideIndex = $this.attr('data-slide-to')) { 462 | $target.data('carousel').pause().to(slideIndex).cycle() 463 | } 464 | 465 | e.preventDefault() 466 | }) 467 | 468 | }(window.jQuery);/* ============================================================= 469 | * bootstrap-collapse.js v2.3.1 470 | * http://twitter.github.com/bootstrap/javascript.html#collapse 471 | * ============================================================= 472 | * Copyright 2012 Twitter, Inc. 473 | * 474 | * Licensed under the Apache License, Version 2.0 (the "License"); 475 | * you may not use this file except in compliance with the License. 476 | * You may obtain a copy of the License at 477 | * 478 | * http://www.apache.org/licenses/LICENSE-2.0 479 | * 480 | * Unless required by applicable law or agreed to in writing, software 481 | * distributed under the License is distributed on an "AS IS" BASIS, 482 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 483 | * See the License for the specific language governing permissions and 484 | * limitations under the License. 485 | * ============================================================ */ 486 | 487 | 488 | !function ($) { 489 | 490 | "use strict"; // jshint ;_; 491 | 492 | 493 | /* COLLAPSE PUBLIC CLASS DEFINITION 494 | * ================================ */ 495 | 496 | var Collapse = function (element, options) { 497 | this.$element = $(element) 498 | this.options = $.extend({}, $.fn.collapse.defaults, options) 499 | 500 | if (this.options.parent) { 501 | this.$parent = $(this.options.parent) 502 | } 503 | 504 | this.options.toggle && this.toggle() 505 | } 506 | 507 | Collapse.prototype = { 508 | 509 | constructor: Collapse 510 | 511 | , dimension: function () { 512 | var hasWidth = this.$element.hasClass('width') 513 | return hasWidth ? 'width' : 'height' 514 | } 515 | 516 | , show: function () { 517 | var dimension 518 | , scroll 519 | , actives 520 | , hasData 521 | 522 | if (this.transitioning || this.$element.hasClass('in')) return 523 | 524 | dimension = this.dimension() 525 | scroll = $.camelCase(['scroll', dimension].join('-')) 526 | actives = this.$parent && this.$parent.find('> .accordion-group > .in') 527 | 528 | if (actives && actives.length) { 529 | hasData = actives.data('collapse') 530 | if (hasData && hasData.transitioning) return 531 | actives.collapse('hide') 532 | hasData || actives.data('collapse', null) 533 | } 534 | 535 | this.$element[dimension](0) 536 | this.transition('addClass', $.Event('show'), 'shown') 537 | $.support.transition && this.$element[dimension](this.$element[0][scroll]) 538 | } 539 | 540 | , hide: function () { 541 | var dimension 542 | if (this.transitioning || !this.$element.hasClass('in')) return 543 | dimension = this.dimension() 544 | this.reset(this.$element[dimension]()) 545 | this.transition('removeClass', $.Event('hide'), 'hidden') 546 | this.$element[dimension](0) 547 | } 548 | 549 | , reset: function (size) { 550 | var dimension = this.dimension() 551 | 552 | this.$element 553 | .removeClass('collapse') 554 | [dimension](size || 'auto') 555 | [0].offsetWidth 556 | 557 | this.$element[size !== null ? 'addClass' : 'removeClass']('collapse') 558 | 559 | return this 560 | } 561 | 562 | , transition: function (method, startEvent, completeEvent) { 563 | var that = this 564 | , complete = function () { 565 | if (startEvent.type == 'show') that.reset() 566 | that.transitioning = 0 567 | that.$element.trigger(completeEvent) 568 | } 569 | 570 | this.$element.trigger(startEvent) 571 | 572 | if (startEvent.isDefaultPrevented()) return 573 | 574 | this.transitioning = 1 575 | 576 | this.$element[method]('in') 577 | 578 | $.support.transition && this.$element.hasClass('collapse') ? 579 | this.$element.one($.support.transition.end, complete) : 580 | complete() 581 | } 582 | 583 | , toggle: function () { 584 | this[this.$element.hasClass('in') ? 'hide' : 'show']() 585 | } 586 | 587 | } 588 | 589 | 590 | /* COLLAPSE PLUGIN DEFINITION 591 | * ========================== */ 592 | 593 | var old = $.fn.collapse 594 | 595 | $.fn.collapse = function (option) { 596 | return this.each(function () { 597 | var $this = $(this) 598 | , data = $this.data('collapse') 599 | , options = $.extend({}, $.fn.collapse.defaults, $this.data(), typeof option == 'object' && option) 600 | if (!data) $this.data('collapse', (data = new Collapse(this, options))) 601 | if (typeof option == 'string') data[option]() 602 | }) 603 | } 604 | 605 | $.fn.collapse.defaults = { 606 | toggle: true 607 | } 608 | 609 | $.fn.collapse.Constructor = Collapse 610 | 611 | 612 | /* COLLAPSE NO CONFLICT 613 | * ==================== */ 614 | 615 | $.fn.collapse.noConflict = function () { 616 | $.fn.collapse = old 617 | return this 618 | } 619 | 620 | 621 | /* COLLAPSE DATA-API 622 | * ================= */ 623 | 624 | $(document).on('click.collapse.data-api', '[data-toggle=collapse]', function (e) { 625 | var $this = $(this), href 626 | , target = $this.attr('data-target') 627 | || e.preventDefault() 628 | || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') //strip for ie7 629 | , option = $(target).data('collapse') ? 'toggle' : $this.data() 630 | $this[$(target).hasClass('in') ? 'addClass' : 'removeClass']('collapsed') 631 | $(target).collapse(option) 632 | }) 633 | 634 | }(window.jQuery);/* ============================================================ 635 | * bootstrap-dropdown.js v2.3.1 636 | * http://twitter.github.com/bootstrap/javascript.html#dropdowns 637 | * ============================================================ 638 | * Copyright 2012 Twitter, Inc. 639 | * 640 | * Licensed under the Apache License, Version 2.0 (the "License"); 641 | * you may not use this file except in compliance with the License. 642 | * You may obtain a copy of the License at 643 | * 644 | * http://www.apache.org/licenses/LICENSE-2.0 645 | * 646 | * Unless required by applicable law or agreed to in writing, software 647 | * distributed under the License is distributed on an "AS IS" BASIS, 648 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 649 | * See the License for the specific language governing permissions and 650 | * limitations under the License. 651 | * ============================================================ */ 652 | 653 | 654 | !function ($) { 655 | 656 | "use strict"; // jshint ;_; 657 | 658 | 659 | /* DROPDOWN CLASS DEFINITION 660 | * ========================= */ 661 | 662 | var toggle = '[data-toggle=dropdown]' 663 | , Dropdown = function (element) { 664 | var $el = $(element).on('click.dropdown.data-api', this.toggle) 665 | $('html').on('click.dropdown.data-api', function () { 666 | $el.parent().removeClass('open') 667 | }) 668 | } 669 | 670 | Dropdown.prototype = { 671 | 672 | constructor: Dropdown 673 | 674 | , toggle: function (e) { 675 | var $this = $(this) 676 | , $parent 677 | , isActive 678 | 679 | if ($this.is('.disabled, :disabled')) return 680 | 681 | $parent = getParent($this) 682 | 683 | isActive = $parent.hasClass('open') 684 | 685 | clearMenus() 686 | 687 | if (!isActive) { 688 | $parent.toggleClass('open') 689 | } 690 | 691 | $this.focus() 692 | 693 | return false 694 | } 695 | 696 | , keydown: function (e) { 697 | var $this 698 | , $items 699 | , $active 700 | , $parent 701 | , isActive 702 | , index 703 | 704 | if (!/(38|40|27)/.test(e.keyCode)) return 705 | 706 | $this = $(this) 707 | 708 | e.preventDefault() 709 | e.stopPropagation() 710 | 711 | if ($this.is('.disabled, :disabled')) return 712 | 713 | $parent = getParent($this) 714 | 715 | isActive = $parent.hasClass('open') 716 | 717 | if (!isActive || (isActive && e.keyCode == 27)) { 718 | if (e.which == 27) $parent.find(toggle).focus() 719 | return $this.click() 720 | } 721 | 722 | $items = $('[role=menu] li:not(.divider):visible a', $parent) 723 | 724 | if (!$items.length) return 725 | 726 | index = $items.index($items.filter(':focus')) 727 | 728 | if (e.keyCode == 38 && index > 0) index-- // up 729 | if (e.keyCode == 40 && index < $items.length - 1) index++ // down 730 | if (!~index) index = 0 731 | 732 | $items 733 | .eq(index) 734 | .focus() 735 | } 736 | 737 | } 738 | 739 | function clearMenus() { 740 | $(toggle).each(function () { 741 | getParent($(this)).removeClass('open') 742 | }) 743 | } 744 | 745 | function getParent($this) { 746 | var selector = $this.attr('data-target') 747 | , $parent 748 | 749 | if (!selector) { 750 | selector = $this.attr('href') 751 | selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7 752 | } 753 | 754 | $parent = selector && $(selector) 755 | 756 | if (!$parent || !$parent.length) $parent = $this.parent() 757 | 758 | return $parent 759 | } 760 | 761 | 762 | /* DROPDOWN PLUGIN DEFINITION 763 | * ========================== */ 764 | 765 | var old = $.fn.dropdown 766 | 767 | $.fn.dropdown = function (option) { 768 | return this.each(function () { 769 | var $this = $(this) 770 | , data = $this.data('dropdown') 771 | if (!data) $this.data('dropdown', (data = new Dropdown(this))) 772 | if (typeof option == 'string') data[option].call($this) 773 | }) 774 | } 775 | 776 | $.fn.dropdown.Constructor = Dropdown 777 | 778 | 779 | /* DROPDOWN NO CONFLICT 780 | * ==================== */ 781 | 782 | $.fn.dropdown.noConflict = function () { 783 | $.fn.dropdown = old 784 | return this 785 | } 786 | 787 | 788 | /* APPLY TO STANDARD DROPDOWN ELEMENTS 789 | * =================================== */ 790 | 791 | $(document) 792 | .on('click.dropdown.data-api', clearMenus) 793 | .on('click.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() }) 794 | .on('click.dropdown-menu', function (e) { e.stopPropagation() }) 795 | .on('click.dropdown.data-api' , toggle, Dropdown.prototype.toggle) 796 | .on('keydown.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown) 797 | 798 | }(window.jQuery); 799 | /* ========================================================= 800 | * bootstrap-modal.js v2.3.1 801 | * http://twitter.github.com/bootstrap/javascript.html#modals 802 | * ========================================================= 803 | * Copyright 2012 Twitter, Inc. 804 | * 805 | * Licensed under the Apache License, Version 2.0 (the "License"); 806 | * you may not use this file except in compliance with the License. 807 | * You may obtain a copy of the License at 808 | * 809 | * http://www.apache.org/licenses/LICENSE-2.0 810 | * 811 | * Unless required by applicable law or agreed to in writing, software 812 | * distributed under the License is distributed on an "AS IS" BASIS, 813 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 814 | * See the License for the specific language governing permissions and 815 | * limitations under the License. 816 | * ========================================================= */ 817 | 818 | 819 | !function ($) { 820 | 821 | "use strict"; // jshint ;_; 822 | 823 | 824 | /* MODAL CLASS DEFINITION 825 | * ====================== */ 826 | 827 | var Modal = function (element, options) { 828 | this.options = options 829 | this.$element = $(element) 830 | .delegate('[data-dismiss="modal"]', 'click.dismiss.modal', $.proxy(this.hide, this)) 831 | this.options.remote && this.$element.find('.modal-body').load(this.options.remote) 832 | } 833 | 834 | Modal.prototype = { 835 | 836 | constructor: Modal 837 | 838 | , toggle: function () { 839 | return this[!this.isShown ? 'show' : 'hide']() 840 | } 841 | 842 | , show: function () { 843 | var that = this 844 | , e = $.Event('show') 845 | 846 | this.$element.trigger(e) 847 | 848 | if (this.isShown || e.isDefaultPrevented()) return 849 | 850 | this.isShown = true 851 | 852 | this.escape() 853 | 854 | this.backdrop(function () { 855 | var transition = $.support.transition && that.$element.hasClass('fade') 856 | 857 | if (!that.$element.parent().length) { 858 | that.$element.appendTo(document.body) //don't move modals dom position 859 | } 860 | 861 | that.$element.show() 862 | 863 | if (transition) { 864 | that.$element[0].offsetWidth // force reflow 865 | } 866 | 867 | that.$element 868 | .addClass('in') 869 | .attr('aria-hidden', false) 870 | 871 | that.enforceFocus() 872 | 873 | transition ? 874 | that.$element.one($.support.transition.end, function () { that.$element.focus().trigger('shown') }) : 875 | that.$element.focus().trigger('shown') 876 | 877 | }) 878 | } 879 | 880 | , hide: function (e) { 881 | e && e.preventDefault() 882 | 883 | var that = this 884 | 885 | e = $.Event('hide') 886 | 887 | this.$element.trigger(e) 888 | 889 | if (!this.isShown || e.isDefaultPrevented()) return 890 | 891 | this.isShown = false 892 | 893 | this.escape() 894 | 895 | $(document).off('focusin.modal') 896 | 897 | this.$element 898 | .removeClass('in') 899 | .attr('aria-hidden', true) 900 | 901 | $.support.transition && this.$element.hasClass('fade') ? 902 | this.hideWithTransition() : 903 | this.hideModal() 904 | } 905 | 906 | , enforceFocus: function () { 907 | var that = this 908 | $(document).on('focusin.modal', function (e) { 909 | if (that.$element[0] !== e.target && !that.$element.has(e.target).length) { 910 | that.$element.focus() 911 | } 912 | }) 913 | } 914 | 915 | , escape: function () { 916 | var that = this 917 | if (this.isShown && this.options.keyboard) { 918 | this.$element.on('keyup.dismiss.modal', function ( e ) { 919 | e.which == 27 && that.hide() 920 | }) 921 | } else if (!this.isShown) { 922 | this.$element.off('keyup.dismiss.modal') 923 | } 924 | } 925 | 926 | , hideWithTransition: function () { 927 | var that = this 928 | , timeout = setTimeout(function () { 929 | that.$element.off($.support.transition.end) 930 | that.hideModal() 931 | }, 500) 932 | 933 | this.$element.one($.support.transition.end, function () { 934 | clearTimeout(timeout) 935 | that.hideModal() 936 | }) 937 | } 938 | 939 | , hideModal: function () { 940 | var that = this 941 | this.$element.hide() 942 | this.backdrop(function () { 943 | that.removeBackdrop() 944 | that.$element.trigger('hidden') 945 | }) 946 | } 947 | 948 | , removeBackdrop: function () { 949 | this.$backdrop && this.$backdrop.remove() 950 | this.$backdrop = null 951 | } 952 | 953 | , backdrop: function (callback) { 954 | var that = this 955 | , animate = this.$element.hasClass('fade') ? 'fade' : '' 956 | 957 | if (this.isShown && this.options.backdrop) { 958 | var doAnimate = $.support.transition && animate 959 | 960 | this.$backdrop = $('