├── .gitignore
├── pom.xml
├── redeploy-tomcat.sh
└── src
├── main
├── java
│ └── org
│ │ └── springframework
│ │ └── samples
│ │ └── portfolio
│ │ ├── Portfolio.java
│ │ ├── PortfolioPosition.java
│ │ ├── config
│ │ ├── DispatcherServletInitializer.java
│ │ └── WebConfig.java
│ │ ├── service
│ │ ├── Quote.java
│ │ ├── StockQuoteGenerator.java
│ │ └── StockService.java
│ │ └── web
│ │ └── PortfolioController.java
├── resources
│ └── log4j.xml
└── webapp
│ ├── assets
│ ├── css
│ │ ├── bootstrap-responsive.css
│ │ ├── bootstrap.css
│ │ ├── bootstrap.min.css
│ │ └── portfolio.css
│ ├── img
│ │ ├── glyphicons-halflings-white.png
│ │ └── glyphicons-halflings.png
│ └── js
│ │ ├── bootstrap.js
│ │ ├── jquery-1.8.3.js
│ │ ├── jquery.loadTemplate-0.5.4.js
│ │ ├── knockout-2.3.0.js
│ │ ├── portfolio-viewModel.js
│ │ ├── sockjs-0.3.4.js
│ │ └── stomp.js
│ └── index.html
└── test
└── resources
└── log4j.xml
/.gitignore:
--------------------------------------------------------------------------------
1 | target
2 | .project
3 | .classpath
4 | .settings
5 | *.iml
6 | /.idea/
7 | bin
8 | .gradle
9 |
--------------------------------------------------------------------------------
/pom.xml:
--------------------------------------------------------------------------------
1 |
3 | 4.0.0
4 | org.springframework.samples
5 | spring-websocket-portfolio
6 | war
7 | 1.0.0-SNAPSHOT
8 |
9 |
10 | 4.0.0.BUILD-SNAPSHOT
11 | 8.0-SNAPSHOT
12 |
13 |
14 |
15 |
16 | org.springframework
17 | spring-context
18 | ${org.springframework-version}
19 |
20 |
21 | commons-logging
22 | commons-logging
23 |
24 |
25 |
26 |
27 | org.springframework
28 | spring-web
29 | ${org.springframework-version}
30 |
31 |
32 | org.springframework
33 | spring-webmvc
34 | ${org.springframework-version}
35 |
36 |
37 | org.springframework
38 | spring-websocket
39 | ${org.springframework-version}
40 |
41 |
42 |
43 | reactor
44 | reactor-core
45 | 1.0.0.BUILD-SNAPSHOT
46 |
47 |
48 | reactor
49 | reactor-tcp
50 | 1.0.0.BUILD-SNAPSHOT
51 |
52 |
53 |
54 | javax.websocket
55 | javax.websocket-api
56 | 1.0
57 | provided
58 |
59 |
60 | javax.servlet
61 | javax.servlet-api
62 | 3.1-b09
63 | provided
64 |
65 |
66 |
67 |
68 |
69 | com.fasterxml.jackson.core
70 | jackson-databind
71 | 2.2.1
72 |
73 |
74 |
75 | org.slf4j
76 | slf4j-api
77 | 1.6.4
78 |
79 |
80 | org.slf4j
81 | jcl-over-slf4j
82 | 1.6.4
83 | runtime
84 |
85 |
86 | org.slf4j
87 | slf4j-log4j12
88 | 1.6.4
89 | runtime
90 |
91 |
92 | log4j
93 | log4j
94 | 1.2.16
95 | runtime
96 |
97 |
98 |
99 | commons-logging
100 | commons-logging
101 | 1.1
102 |
103 |
104 | javax.servlet
105 | servlet-api
106 |
107 |
108 |
109 |
110 |
111 | junit
112 | junit
113 | 4.8.2
114 | test
115 |
116 |
117 |
118 | org.apache.tomcat
119 | tomcat-websocket
120 | ${tomcat-version}
121 | test
122 |
123 |
124 |
125 |
126 |
127 |
128 | spring-snapshots
129 | http://repo.springsource.org/snapshot
130 | true
131 | false
132 |
133 |
134 | tomcat-snapshots
135 | https://repository.apache.org/content/repositories/snapshots
136 | true
137 | false
138 |
139 |
140 | java-net
141 | https://maven.java.net/content/repositories/releases
142 |
143 |
144 |
145 |
146 |
147 | apache.snapshots
148 | http://repository.apache.org/content/groups/snapshots-group/
149 |
150 |
151 |
152 |
153 | ${project.artifactId}
154 |
155 |
156 | org.apache.maven.plugins
157 | maven-compiler-plugin
158 | 2.3.2
159 |
160 | 1.6
161 | 1.6
162 |
163 |
164 |
165 | org.apache.maven.plugins
166 | maven-war-plugin
167 | 2.2
168 |
169 | false
170 |
171 |
172 |
173 | org.apache.maven.plugins
174 | maven-resources-plugin
175 | 2.5
176 |
177 | UTF-8
178 |
179 |
180 |
181 | org.apache.maven.plugins
182 | maven-eclipse-plugin
183 | 2.8
184 |
185 | true
186 | false
187 | 2.0
188 |
189 |
190 |
191 | org.eclipse.jetty
192 | jetty-maven-plugin
193 | 9.0.3.v20130506
194 |
195 |
196 | /spring-sockjs-protocol-webapp
197 |
198 |
199 |
200 |
201 |
202 |
203 |
--------------------------------------------------------------------------------
/redeploy-tomcat.sh:
--------------------------------------------------------------------------------
1 | set -v
2 |
3 | mvn -DskipTests clean package
4 |
5 | # Change the line below to the location of Tomcat built from trunk
6 | TOMCAT=~/dev/sources/apache/tomcat/trunk/output/build
7 |
8 | rm -rf $TOMCAT/webapps/spring-websocket-portfolio*
9 |
10 | cp target/spring-websocket-portfolio.war $TOMCAT/webapps/
11 |
12 | $TOMCAT/bin/shutdown.sh
13 | sleep 3
14 |
15 | $TOMCAT/bin/startup.sh
16 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/Portfolio.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio;
2 |
3 | import java.util.ArrayList;
4 | import java.util.List;
5 |
6 |
7 | public class Portfolio {
8 |
9 | private final List positions = new ArrayList();
10 |
11 |
12 | public PortfolioPosition[] getPositions() {
13 | return this.positions.toArray(new PortfolioPosition[this.positions.size()]);
14 | }
15 |
16 | public void addPosition(PortfolioPosition position) {
17 | this.positions.add(position);
18 | }
19 |
20 | }
21 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/PortfolioPosition.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio;
2 |
3 | public class PortfolioPosition {
4 |
5 | private final String company;
6 |
7 | private final String ticker;
8 |
9 | private final double price;
10 |
11 | private final int shares;
12 |
13 |
14 | public PortfolioPosition(String company, String ticker, double price, int shares) {
15 | this.company = company;
16 | this.ticker = ticker;
17 | this.price = price;
18 | this.shares = shares;
19 | }
20 |
21 | public String getCompany() {
22 | return this.company;
23 | }
24 |
25 | public String getTicker() {
26 | return this.ticker;
27 | }
28 |
29 | public double getPrice() {
30 | return this.price;
31 | }
32 |
33 | public int getShares() {
34 | return this.shares;
35 | }
36 |
37 | }
38 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/config/DispatcherServletInitializer.java:
--------------------------------------------------------------------------------
1 | /*
2 | * Copyright 2002-2013 the original author or authors.
3 | *
4 | * Licensed under the Apache License, Version 2.0 (the "License");
5 | * you may not use this file except in compliance with the License.
6 | * You may obtain a copy of the License at
7 | *
8 | * http://www.apache.org/licenses/LICENSE-2.0
9 | *
10 | * Unless required by applicable law or agreed to in writing, software
11 | * distributed under the License is distributed on an "AS IS" BASIS,
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | * See the License for the specific language governing permissions and
14 | * limitations under the License.
15 | */
16 |
17 | package org.springframework.samples.portfolio.config;
18 |
19 | import javax.servlet.ServletRegistration.Dynamic;
20 |
21 | import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
22 |
23 | public class DispatcherServletInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
24 |
25 |
26 | @Override
27 | protected Class>[] getRootConfigClasses() {
28 | return null;
29 | }
30 |
31 | @Override
32 | protected Class>[] getServletConfigClasses() {
33 | return new Class>[] { WebConfig.class };
34 | }
35 |
36 | @Override
37 | protected String[] getServletMappings() {
38 | return new String[] { "/" };
39 | }
40 |
41 | @Override
42 | protected void customizeRegistration(Dynamic registration) {
43 | registration.setInitParameter("dispatchOptionsRequest", "true");
44 | }
45 |
46 | }
47 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/config/WebConfig.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio.config;
2 |
3 | import java.util.Collections;
4 |
5 | import org.springframework.context.annotation.Bean;
6 | import org.springframework.context.annotation.ComponentScan;
7 | import org.springframework.context.annotation.Configuration;
8 | import org.springframework.messaging.SubscribableChannel;
9 | import org.springframework.messaging.channel.PublishSubscribeChannel;
10 | import org.springframework.messaging.converter.MappingJackson2MessageConverter;
11 | import org.springframework.messaging.converter.MessageConverter;
12 | import org.springframework.scheduling.annotation.EnableScheduling;
13 | import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
14 | import org.springframework.web.HttpRequestHandler;
15 | import org.springframework.web.messaging.service.broker.SimpleBrokerWebMessageHandler;
16 | import org.springframework.web.messaging.service.method.AnnotationWebMessageHandler;
17 | import org.springframework.web.messaging.stomp.support.StompWebSocketHandler;
18 | import org.springframework.web.messaging.support.WebMessagingTemplate;
19 | import org.springframework.web.servlet.config.annotation.DefaultServletHandlerConfigurer;
20 | import org.springframework.web.servlet.config.annotation.EnableWebMvc;
21 | import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
22 | import org.springframework.web.servlet.handler.SimpleUrlHandlerMapping;
23 | import org.springframework.web.socket.sockjs.SockJsService;
24 | import org.springframework.web.socket.sockjs.support.DefaultSockJsService;
25 | import org.springframework.web.socket.sockjs.support.SockJsHttpRequestHandler;
26 |
27 |
28 | @Configuration
29 | @EnableWebMvc
30 | @EnableScheduling
31 | @ComponentScan(basePackages="org.springframework.samples")
32 | public class WebConfig extends WebMvcConfigurerAdapter {
33 |
34 | private final MessageConverter> messageConverter = new MappingJackson2MessageConverter();
35 |
36 | @Bean
37 | public SimpleUrlHandlerMapping handlerMapping() {
38 |
39 | SockJsService sockJsService = new DefaultSockJsService(taskScheduler());
40 | HttpRequestHandler requestHandler = new SockJsHttpRequestHandler(sockJsService, stompWebSocketHandler());
41 |
42 | SimpleUrlHandlerMapping hm = new SimpleUrlHandlerMapping();
43 | hm.setOrder(-1);
44 | hm.setUrlMap(Collections.singletonMap("/portfolio/**", requestHandler));
45 | return hm;
46 | }
47 |
48 | @Bean
49 | public StompWebSocketHandler stompWebSocketHandler() {
50 | StompWebSocketHandler handler = new StompWebSocketHandler(inboundChannel());
51 | outboundChannel().subscribe(handler);
52 | return handler;
53 | }
54 |
55 | @Bean
56 | public AnnotationWebMessageHandler annotationMessageHandler() {
57 | AnnotationWebMessageHandler handler = new AnnotationWebMessageHandler(inboundChannel(), outboundChannel());
58 | handler.setMessageConverter(this.messageConverter);
59 | inboundChannel().subscribe(handler);
60 | return handler;
61 | }
62 |
63 | @Bean
64 | public SimpleBrokerWebMessageHandler simpleBrokerMessageHandler() {
65 | SimpleBrokerWebMessageHandler handler = new SimpleBrokerWebMessageHandler(outboundChannel());
66 | inboundChannel().subscribe(handler);
67 | return handler;
68 | }
69 |
70 | @Bean
71 | public SubscribableChannel inboundChannel() {
72 | return new PublishSubscribeChannel();
73 | }
74 |
75 | @Bean
76 | public SubscribableChannel outboundChannel() {
77 | return new PublishSubscribeChannel();
78 | }
79 |
80 | @Bean
81 | public WebMessagingTemplate messagingTemplate() {
82 | WebMessagingTemplate template = new WebMessagingTemplate(inboundChannel());
83 | template.setMessageConverter(this.messageConverter);
84 | return template;
85 | }
86 |
87 |
88 | @Bean
89 | public ThreadPoolTaskScheduler taskScheduler() {
90 | ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
91 | taskScheduler.setThreadNamePrefix("SockJS-");
92 | taskScheduler.setPoolSize(4);
93 | return taskScheduler;
94 | }
95 |
96 |
97 | // Allow serving HTML files through the default Servlet
98 |
99 | @Override
100 | public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
101 | configurer.enable();
102 | }
103 |
104 | }
105 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/service/Quote.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio.service;
2 |
3 | import java.math.BigDecimal;
4 |
5 |
6 | public class Quote {
7 |
8 | private final String ticker;
9 |
10 | private final BigDecimal price;
11 |
12 |
13 | public Quote(String ticker, BigDecimal price) {
14 | this.ticker = ticker;
15 | this.price = price;
16 | }
17 |
18 |
19 | public String getTicker() {
20 | return this.ticker;
21 | }
22 |
23 | public BigDecimal getPrice() {
24 | return this.price;
25 | }
26 |
27 | @Override
28 | public String toString() {
29 | return "Quote [ticker=" + this.ticker + ", this.price=" + price + "]";
30 | }
31 |
32 | }
33 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/service/StockQuoteGenerator.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio.service;
2 |
3 | import java.math.BigDecimal;
4 | import java.math.MathContext;
5 | import java.util.ArrayList;
6 | import java.util.Collections;
7 | import java.util.Iterator;
8 | import java.util.List;
9 | import java.util.Map;
10 | import java.util.Random;
11 | import java.util.concurrent.ConcurrentHashMap;
12 |
13 |
14 | public class StockQuoteGenerator {
15 |
16 | private static final MathContext mathContext = new MathContext(2);
17 |
18 | private final Random random = new Random();
19 |
20 | private final Map quotes = new ConcurrentHashMap();
21 |
22 | private final List tickers;
23 |
24 | private Iterator tickersIterator;
25 |
26 |
27 | public StockQuoteGenerator() {
28 |
29 | this.quotes.put("CTXS", "24.30");
30 | this.quotes.put("DELL", "13.03");
31 | this.quotes.put("EMC", "24.13");
32 | this.quotes.put("GOOG", "893.49");
33 | this.quotes.put("MSFT", "34.21");
34 | this.quotes.put("ORCL", "31.22");
35 | this.quotes.put("RHT", "48.30");
36 | this.quotes.put("VMW", "66.98");
37 |
38 | this.tickers = new ArrayList(this.quotes.keySet());
39 | this.tickersIterator = this.tickers.iterator();
40 | }
41 |
42 | public Quote nextQuote() {
43 | if (!this.tickersIterator.hasNext()) {
44 | Collections.shuffle(this.tickers);
45 | this.tickersIterator = this.tickers.iterator();
46 | }
47 | String ticker = this.tickersIterator.next();
48 | BigDecimal price = getPrice(ticker);
49 | return new Quote(ticker, price);
50 | }
51 |
52 | private BigDecimal getPrice(String ticker) {
53 | BigDecimal seedPrice = new BigDecimal(this.quotes.get(ticker), mathContext);
54 | double range = seedPrice.multiply(new BigDecimal(0.02)).doubleValue();
55 | BigDecimal priceChange = new BigDecimal(String.valueOf(this.random.nextDouble() * range), mathContext);
56 | return seedPrice.add(priceChange);
57 | }
58 |
59 |
60 | public static void main(String[] args) {
61 | StockQuoteGenerator gen = new StockQuoteGenerator();
62 | for (int i=0 ; i < gen.quotes.size(); i++) {
63 | System.out.println(gen.nextQuote());
64 | }
65 | System.out.println("");
66 | for (int i=0 ; i < gen.quotes.size(); i++) {
67 | System.out.println(gen.nextQuote());
68 | }
69 | }
70 |
71 | }
72 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/service/StockService.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio.service;
2 |
3 | import org.springframework.beans.factory.annotation.Autowired;
4 | import org.springframework.scheduling.annotation.Scheduled;
5 | import org.springframework.stereotype.Component;
6 | import org.springframework.web.messaging.support.WebMessagingTemplate;
7 |
8 |
9 | @Component
10 | public class StockService {
11 |
12 | private final WebMessagingTemplate messagingTemplate;
13 |
14 | private final StockQuoteGenerator quoteGenerator = new StockQuoteGenerator();
15 |
16 |
17 | @Autowired
18 | public StockService(WebMessagingTemplate messagingTemplate) {
19 | this.messagingTemplate = messagingTemplate;
20 | }
21 |
22 |
23 | @Scheduled(fixedDelay=2000)
24 | public void sendQuotes() {
25 | Quote quote = this.quoteGenerator.nextQuote();
26 | this.messagingTemplate.convertAndSend("/topic/PRICE.STOCK.NASDAQ." + quote.getTicker(), quote);
27 | }
28 |
29 | }
30 |
--------------------------------------------------------------------------------
/src/main/java/org/springframework/samples/portfolio/web/PortfolioController.java:
--------------------------------------------------------------------------------
1 | package org.springframework.samples.portfolio.web;
2 |
3 | import java.io.IOException;
4 | import java.util.ArrayList;
5 | import java.util.List;
6 |
7 | import org.springframework.beans.factory.annotation.Autowired;
8 | import org.springframework.samples.portfolio.PortfolioPosition;
9 | import org.springframework.stereotype.Controller;
10 | import org.springframework.web.messaging.annotation.SubscribeEvent;
11 | import org.springframework.web.messaging.support.WebMessagingTemplate;
12 |
13 |
14 | @Controller
15 | public class PortfolioController {
16 |
17 | private final WebMessagingTemplate messagingTemplate;
18 |
19 |
20 | @Autowired
21 | public PortfolioController(WebMessagingTemplate messagingTemplate) {
22 | this.messagingTemplate = messagingTemplate;
23 | }
24 |
25 |
26 | @SubscribeEvent("/positions")
27 | public List getPortfolios() throws IOException {
28 | List positions = new ArrayList();
29 | positions.add(new PortfolioPosition("Citrix Systems, Inc.", "CTXS", 24.30, 75));
30 | positions.add(new PortfolioPosition("Dell Inc.", "DELL", 13.44, 50));
31 | positions.add(new PortfolioPosition("EMC Corporation", "EMC", 24.30, 75));
32 | positions.add(new PortfolioPosition("Google Inc", "GOOG", 905.09, 5));
33 | positions.add(new PortfolioPosition("Microsoft", "MSFT", 34.15, 33));
34 | positions.add(new PortfolioPosition("VMware, Inc.", "VMW", 65.58, 23));
35 | return positions;
36 | }
37 |
38 | }
39 |
--------------------------------------------------------------------------------
/src/main/resources/log4j.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/src/main/webapp/assets/css/bootstrap-responsive.css:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap Responsive v2.3.2
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 |
--------------------------------------------------------------------------------
/src/main/webapp/assets/css/portfolio.css:
--------------------------------------------------------------------------------
1 |
2 | body {
3 | padding-top: 40px;
4 | padding-bottom: 40px;
5 | background-color: #f5f5f5;
6 | }
7 |
8 | .main-content {
9 | max-width: 800px;
10 | padding: 19px 29px 29px;
11 | margin: 0 auto 20px;
12 | background-color: #fff;
13 | border: 1px solid #e5e5e5;
14 | -webkit-border-radius: 5px;
15 | -moz-border-radius: 5px;
16 | border-radius: 5px;
17 | -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, .05);
18 | -moz-box-shadow: 0 1px 2px rgba(0, 0, 0, .05);
19 | box-shadow: 0 1px 2px rgba(0, 0, 0, .05);
20 | }
21 |
22 | .table th.number, .table td.number {
23 | text-align: right;
24 | }
25 |
26 | .table tfoot {
27 | font-weight: bold;
28 | }
--------------------------------------------------------------------------------
/src/main/webapp/assets/img/glyphicons-halflings-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rwinch/spring-websocket-portfolio/d125d0c0fec94d4293378aa9dfc91e8e148bd997/src/main/webapp/assets/img/glyphicons-halflings-white.png
--------------------------------------------------------------------------------
/src/main/webapp/assets/img/glyphicons-halflings.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/rwinch/spring-websocket-portfolio/d125d0c0fec94d4293378aa9dfc91e8e148bd997/src/main/webapp/assets/img/glyphicons-halflings.png
--------------------------------------------------------------------------------
/src/main/webapp/assets/js/bootstrap.js:
--------------------------------------------------------------------------------
1 | /* ===================================================
2 | * bootstrap-transition.js v2.3.2
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.2
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.2
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.2
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.2
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.2
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 | if ('ontouchstart' in document.documentElement) {
689 | // if mobile we we use a backdrop because click events don't delegate
690 | $('').insertBefore($(this)).on('click', clearMenus)
691 | }
692 | $parent.toggleClass('open')
693 | }
694 |
695 | $this.focus()
696 |
697 | return false
698 | }
699 |
700 | , keydown: function (e) {
701 | var $this
702 | , $items
703 | , $active
704 | , $parent
705 | , isActive
706 | , index
707 |
708 | if (!/(38|40|27)/.test(e.keyCode)) return
709 |
710 | $this = $(this)
711 |
712 | e.preventDefault()
713 | e.stopPropagation()
714 |
715 | if ($this.is('.disabled, :disabled')) return
716 |
717 | $parent = getParent($this)
718 |
719 | isActive = $parent.hasClass('open')
720 |
721 | if (!isActive || (isActive && e.keyCode == 27)) {
722 | if (e.which == 27) $parent.find(toggle).focus()
723 | return $this.click()
724 | }
725 |
726 | $items = $('[role=menu] li:not(.divider):visible a', $parent)
727 |
728 | if (!$items.length) return
729 |
730 | index = $items.index($items.filter(':focus'))
731 |
732 | if (e.keyCode == 38 && index > 0) index-- // up
733 | if (e.keyCode == 40 && index < $items.length - 1) index++ // down
734 | if (!~index) index = 0
735 |
736 | $items
737 | .eq(index)
738 | .focus()
739 | }
740 |
741 | }
742 |
743 | function clearMenus() {
744 | $('.dropdown-backdrop').remove()
745 | $(toggle).each(function () {
746 | getParent($(this)).removeClass('open')
747 | })
748 | }
749 |
750 | function getParent($this) {
751 | var selector = $this.attr('data-target')
752 | , $parent
753 |
754 | if (!selector) {
755 | selector = $this.attr('href')
756 | selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
757 | }
758 |
759 | $parent = selector && $(selector)
760 |
761 | if (!$parent || !$parent.length) $parent = $this.parent()
762 |
763 | return $parent
764 | }
765 |
766 |
767 | /* DROPDOWN PLUGIN DEFINITION
768 | * ========================== */
769 |
770 | var old = $.fn.dropdown
771 |
772 | $.fn.dropdown = function (option) {
773 | return this.each(function () {
774 | var $this = $(this)
775 | , data = $this.data('dropdown')
776 | if (!data) $this.data('dropdown', (data = new Dropdown(this)))
777 | if (typeof option == 'string') data[option].call($this)
778 | })
779 | }
780 |
781 | $.fn.dropdown.Constructor = Dropdown
782 |
783 |
784 | /* DROPDOWN NO CONFLICT
785 | * ==================== */
786 |
787 | $.fn.dropdown.noConflict = function () {
788 | $.fn.dropdown = old
789 | return this
790 | }
791 |
792 |
793 | /* APPLY TO STANDARD DROPDOWN ELEMENTS
794 | * =================================== */
795 |
796 | $(document)
797 | .on('click.dropdown.data-api', clearMenus)
798 | .on('click.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
799 | .on('click.dropdown.data-api' , toggle, Dropdown.prototype.toggle)
800 | .on('keydown.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
801 |
802 | }(window.jQuery);
803 | /* =========================================================
804 | * bootstrap-modal.js v2.3.2
805 | * http://twitter.github.com/bootstrap/javascript.html#modals
806 | * =========================================================
807 | * Copyright 2012 Twitter, Inc.
808 | *
809 | * Licensed under the Apache License, Version 2.0 (the "License");
810 | * you may not use this file except in compliance with the License.
811 | * You may obtain a copy of the License at
812 | *
813 | * http://www.apache.org/licenses/LICENSE-2.0
814 | *
815 | * Unless required by applicable law or agreed to in writing, software
816 | * distributed under the License is distributed on an "AS IS" BASIS,
817 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
818 | * See the License for the specific language governing permissions and
819 | * limitations under the License.
820 | * ========================================================= */
821 |
822 |
823 | !function ($) {
824 |
825 | "use strict"; // jshint ;_;
826 |
827 |
828 | /* MODAL CLASS DEFINITION
829 | * ====================== */
830 |
831 | var Modal = function (element, options) {
832 | this.options = options
833 | this.$element = $(element)
834 | .delegate('[data-dismiss="modal"]', 'click.dismiss.modal', $.proxy(this.hide, this))
835 | this.options.remote && this.$element.find('.modal-body').load(this.options.remote)
836 | }
837 |
838 | Modal.prototype = {
839 |
840 | constructor: Modal
841 |
842 | , toggle: function () {
843 | return this[!this.isShown ? 'show' : 'hide']()
844 | }
845 |
846 | , show: function () {
847 | var that = this
848 | , e = $.Event('show')
849 |
850 | this.$element.trigger(e)
851 |
852 | if (this.isShown || e.isDefaultPrevented()) return
853 |
854 | this.isShown = true
855 |
856 | this.escape()
857 |
858 | this.backdrop(function () {
859 | var transition = $.support.transition && that.$element.hasClass('fade')
860 |
861 | if (!that.$element.parent().length) {
862 | that.$element.appendTo(document.body) //don't move modals dom position
863 | }
864 |
865 | that.$element.show()
866 |
867 | if (transition) {
868 | that.$element[0].offsetWidth // force reflow
869 | }
870 |
871 | that.$element
872 | .addClass('in')
873 | .attr('aria-hidden', false)
874 |
875 | that.enforceFocus()
876 |
877 | transition ?
878 | that.$element.one($.support.transition.end, function () { that.$element.focus().trigger('shown') }) :
879 | that.$element.focus().trigger('shown')
880 |
881 | })
882 | }
883 |
884 | , hide: function (e) {
885 | e && e.preventDefault()
886 |
887 | var that = this
888 |
889 | e = $.Event('hide')
890 |
891 | this.$element.trigger(e)
892 |
893 | if (!this.isShown || e.isDefaultPrevented()) return
894 |
895 | this.isShown = false
896 |
897 | this.escape()
898 |
899 | $(document).off('focusin.modal')
900 |
901 | this.$element
902 | .removeClass('in')
903 | .attr('aria-hidden', true)
904 |
905 | $.support.transition && this.$element.hasClass('fade') ?
906 | this.hideWithTransition() :
907 | this.hideModal()
908 | }
909 |
910 | , enforceFocus: function () {
911 | var that = this
912 | $(document).on('focusin.modal', function (e) {
913 | if (that.$element[0] !== e.target && !that.$element.has(e.target).length) {
914 | that.$element.focus()
915 | }
916 | })
917 | }
918 |
919 | , escape: function () {
920 | var that = this
921 | if (this.isShown && this.options.keyboard) {
922 | this.$element.on('keyup.dismiss.modal', function ( e ) {
923 | e.which == 27 && that.hide()
924 | })
925 | } else if (!this.isShown) {
926 | this.$element.off('keyup.dismiss.modal')
927 | }
928 | }
929 |
930 | , hideWithTransition: function () {
931 | var that = this
932 | , timeout = setTimeout(function () {
933 | that.$element.off($.support.transition.end)
934 | that.hideModal()
935 | }, 500)
936 |
937 | this.$element.one($.support.transition.end, function () {
938 | clearTimeout(timeout)
939 | that.hideModal()
940 | })
941 | }
942 |
943 | , hideModal: function () {
944 | var that = this
945 | this.$element.hide()
946 | this.backdrop(function () {
947 | that.removeBackdrop()
948 | that.$element.trigger('hidden')
949 | })
950 | }
951 |
952 | , removeBackdrop: function () {
953 | this.$backdrop && this.$backdrop.remove()
954 | this.$backdrop = null
955 | }
956 |
957 | , backdrop: function (callback) {
958 | var that = this
959 | , animate = this.$element.hasClass('fade') ? 'fade' : ''
960 |
961 | if (this.isShown && this.options.backdrop) {
962 | var doAnimate = $.support.transition && animate
963 |
964 | this.$backdrop = $('')
965 | .appendTo(document.body)
966 |
967 | this.$backdrop.click(
968 | this.options.backdrop == 'static' ?
969 | $.proxy(this.$element[0].focus, this.$element[0])
970 | : $.proxy(this.hide, this)
971 | )
972 |
973 | if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
974 |
975 | this.$backdrop.addClass('in')
976 |
977 | if (!callback) return
978 |
979 | doAnimate ?
980 | this.$backdrop.one($.support.transition.end, callback) :
981 | callback()
982 |
983 | } else if (!this.isShown && this.$backdrop) {
984 | this.$backdrop.removeClass('in')
985 |
986 | $.support.transition && this.$element.hasClass('fade')?
987 | this.$backdrop.one($.support.transition.end, callback) :
988 | callback()
989 |
990 | } else if (callback) {
991 | callback()
992 | }
993 | }
994 | }
995 |
996 |
997 | /* MODAL PLUGIN DEFINITION
998 | * ======================= */
999 |
1000 | var old = $.fn.modal
1001 |
1002 | $.fn.modal = function (option) {
1003 | return this.each(function () {
1004 | var $this = $(this)
1005 | , data = $this.data('modal')
1006 | , options = $.extend({}, $.fn.modal.defaults, $this.data(), typeof option == 'object' && option)
1007 | if (!data) $this.data('modal', (data = new Modal(this, options)))
1008 | if (typeof option == 'string') data[option]()
1009 | else if (options.show) data.show()
1010 | })
1011 | }
1012 |
1013 | $.fn.modal.defaults = {
1014 | backdrop: true
1015 | , keyboard: true
1016 | , show: true
1017 | }
1018 |
1019 | $.fn.modal.Constructor = Modal
1020 |
1021 |
1022 | /* MODAL NO CONFLICT
1023 | * ================= */
1024 |
1025 | $.fn.modal.noConflict = function () {
1026 | $.fn.modal = old
1027 | return this
1028 | }
1029 |
1030 |
1031 | /* MODAL DATA-API
1032 | * ============== */
1033 |
1034 | $(document).on('click.modal.data-api', '[data-toggle="modal"]', function (e) {
1035 | var $this = $(this)
1036 | , href = $this.attr('href')
1037 | , $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) //strip for ie7
1038 | , option = $target.data('modal') ? 'toggle' : $.extend({ remote:!/#/.test(href) && href }, $target.data(), $this.data())
1039 |
1040 | e.preventDefault()
1041 |
1042 | $target
1043 | .modal(option)
1044 | .one('hide', function () {
1045 | $this.focus()
1046 | })
1047 | })
1048 |
1049 | }(window.jQuery);
1050 | /* ===========================================================
1051 | * bootstrap-tooltip.js v2.3.2
1052 | * http://twitter.github.com/bootstrap/javascript.html#tooltips
1053 | * Inspired by the original jQuery.tipsy by Jason Frame
1054 | * ===========================================================
1055 | * Copyright 2012 Twitter, Inc.
1056 | *
1057 | * Licensed under the Apache License, Version 2.0 (the "License");
1058 | * you may not use this file except in compliance with the License.
1059 | * You may obtain a copy of the License at
1060 | *
1061 | * http://www.apache.org/licenses/LICENSE-2.0
1062 | *
1063 | * Unless required by applicable law or agreed to in writing, software
1064 | * distributed under the License is distributed on an "AS IS" BASIS,
1065 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1066 | * See the License for the specific language governing permissions and
1067 | * limitations under the License.
1068 | * ========================================================== */
1069 |
1070 |
1071 | !function ($) {
1072 |
1073 | "use strict"; // jshint ;_;
1074 |
1075 |
1076 | /* TOOLTIP PUBLIC CLASS DEFINITION
1077 | * =============================== */
1078 |
1079 | var Tooltip = function (element, options) {
1080 | this.init('tooltip', element, options)
1081 | }
1082 |
1083 | Tooltip.prototype = {
1084 |
1085 | constructor: Tooltip
1086 |
1087 | , init: function (type, element, options) {
1088 | var eventIn
1089 | , eventOut
1090 | , triggers
1091 | , trigger
1092 | , i
1093 |
1094 | this.type = type
1095 | this.$element = $(element)
1096 | this.options = this.getOptions(options)
1097 | this.enabled = true
1098 |
1099 | triggers = this.options.trigger.split(' ')
1100 |
1101 | for (i = triggers.length; i--;) {
1102 | trigger = triggers[i]
1103 | if (trigger == 'click') {
1104 | this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
1105 | } else if (trigger != 'manual') {
1106 | eventIn = trigger == 'hover' ? 'mouseenter' : 'focus'
1107 | eventOut = trigger == 'hover' ? 'mouseleave' : 'blur'
1108 | this.$element.on(eventIn + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
1109 | this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
1110 | }
1111 | }
1112 |
1113 | this.options.selector ?
1114 | (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
1115 | this.fixTitle()
1116 | }
1117 |
1118 | , getOptions: function (options) {
1119 | options = $.extend({}, $.fn[this.type].defaults, this.$element.data(), options)
1120 |
1121 | if (options.delay && typeof options.delay == 'number') {
1122 | options.delay = {
1123 | show: options.delay
1124 | , hide: options.delay
1125 | }
1126 | }
1127 |
1128 | return options
1129 | }
1130 |
1131 | , enter: function (e) {
1132 | var defaults = $.fn[this.type].defaults
1133 | , options = {}
1134 | , self
1135 |
1136 | this._options && $.each(this._options, function (key, value) {
1137 | if (defaults[key] != value) options[key] = value
1138 | }, this)
1139 |
1140 | self = $(e.currentTarget)[this.type](options).data(this.type)
1141 |
1142 | if (!self.options.delay || !self.options.delay.show) return self.show()
1143 |
1144 | clearTimeout(this.timeout)
1145 | self.hoverState = 'in'
1146 | this.timeout = setTimeout(function() {
1147 | if (self.hoverState == 'in') self.show()
1148 | }, self.options.delay.show)
1149 | }
1150 |
1151 | , leave: function (e) {
1152 | var self = $(e.currentTarget)[this.type](this._options).data(this.type)
1153 |
1154 | if (this.timeout) clearTimeout(this.timeout)
1155 | if (!self.options.delay || !self.options.delay.hide) return self.hide()
1156 |
1157 | self.hoverState = 'out'
1158 | this.timeout = setTimeout(function() {
1159 | if (self.hoverState == 'out') self.hide()
1160 | }, self.options.delay.hide)
1161 | }
1162 |
1163 | , show: function () {
1164 | var $tip
1165 | , pos
1166 | , actualWidth
1167 | , actualHeight
1168 | , placement
1169 | , tp
1170 | , e = $.Event('show')
1171 |
1172 | if (this.hasContent() && this.enabled) {
1173 | this.$element.trigger(e)
1174 | if (e.isDefaultPrevented()) return
1175 | $tip = this.tip()
1176 | this.setContent()
1177 |
1178 | if (this.options.animation) {
1179 | $tip.addClass('fade')
1180 | }
1181 |
1182 | placement = typeof this.options.placement == 'function' ?
1183 | this.options.placement.call(this, $tip[0], this.$element[0]) :
1184 | this.options.placement
1185 |
1186 | $tip
1187 | .detach()
1188 | .css({ top: 0, left: 0, display: 'block' })
1189 |
1190 | this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
1191 |
1192 | pos = this.getPosition()
1193 |
1194 | actualWidth = $tip[0].offsetWidth
1195 | actualHeight = $tip[0].offsetHeight
1196 |
1197 | switch (placement) {
1198 | case 'bottom':
1199 | tp = {top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}
1200 | break
1201 | case 'top':
1202 | tp = {top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}
1203 | break
1204 | case 'left':
1205 | tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}
1206 | break
1207 | case 'right':
1208 | tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}
1209 | break
1210 | }
1211 |
1212 | this.applyPlacement(tp, placement)
1213 | this.$element.trigger('shown')
1214 | }
1215 | }
1216 |
1217 | , applyPlacement: function(offset, placement){
1218 | var $tip = this.tip()
1219 | , width = $tip[0].offsetWidth
1220 | , height = $tip[0].offsetHeight
1221 | , actualWidth
1222 | , actualHeight
1223 | , delta
1224 | , replace
1225 |
1226 | $tip
1227 | .offset(offset)
1228 | .addClass(placement)
1229 | .addClass('in')
1230 |
1231 | actualWidth = $tip[0].offsetWidth
1232 | actualHeight = $tip[0].offsetHeight
1233 |
1234 | if (placement == 'top' && actualHeight != height) {
1235 | offset.top = offset.top + height - actualHeight
1236 | replace = true
1237 | }
1238 |
1239 | if (placement == 'bottom' || placement == 'top') {
1240 | delta = 0
1241 |
1242 | if (offset.left < 0){
1243 | delta = offset.left * -2
1244 | offset.left = 0
1245 | $tip.offset(offset)
1246 | actualWidth = $tip[0].offsetWidth
1247 | actualHeight = $tip[0].offsetHeight
1248 | }
1249 |
1250 | this.replaceArrow(delta - width + actualWidth, actualWidth, 'left')
1251 | } else {
1252 | this.replaceArrow(actualHeight - height, actualHeight, 'top')
1253 | }
1254 |
1255 | if (replace) $tip.offset(offset)
1256 | }
1257 |
1258 | , replaceArrow: function(delta, dimension, position){
1259 | this
1260 | .arrow()
1261 | .css(position, delta ? (50 * (1 - delta / dimension) + "%") : '')
1262 | }
1263 |
1264 | , setContent: function () {
1265 | var $tip = this.tip()
1266 | , title = this.getTitle()
1267 |
1268 | $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
1269 | $tip.removeClass('fade in top bottom left right')
1270 | }
1271 |
1272 | , hide: function () {
1273 | var that = this
1274 | , $tip = this.tip()
1275 | , e = $.Event('hide')
1276 |
1277 | this.$element.trigger(e)
1278 | if (e.isDefaultPrevented()) return
1279 |
1280 | $tip.removeClass('in')
1281 |
1282 | function removeWithAnimation() {
1283 | var timeout = setTimeout(function () {
1284 | $tip.off($.support.transition.end).detach()
1285 | }, 500)
1286 |
1287 | $tip.one($.support.transition.end, function () {
1288 | clearTimeout(timeout)
1289 | $tip.detach()
1290 | })
1291 | }
1292 |
1293 | $.support.transition && this.$tip.hasClass('fade') ?
1294 | removeWithAnimation() :
1295 | $tip.detach()
1296 |
1297 | this.$element.trigger('hidden')
1298 |
1299 | return this
1300 | }
1301 |
1302 | , fixTitle: function () {
1303 | var $e = this.$element
1304 | if ($e.attr('title') || typeof($e.attr('data-original-title')) != 'string') {
1305 | $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
1306 | }
1307 | }
1308 |
1309 | , hasContent: function () {
1310 | return this.getTitle()
1311 | }
1312 |
1313 | , getPosition: function () {
1314 | var el = this.$element[0]
1315 | return $.extend({}, (typeof el.getBoundingClientRect == 'function') ? el.getBoundingClientRect() : {
1316 | width: el.offsetWidth
1317 | , height: el.offsetHeight
1318 | }, this.$element.offset())
1319 | }
1320 |
1321 | , getTitle: function () {
1322 | var title
1323 | , $e = this.$element
1324 | , o = this.options
1325 |
1326 | title = $e.attr('data-original-title')
1327 | || (typeof o.title == 'function' ? o.title.call($e[0]) : o.title)
1328 |
1329 | return title
1330 | }
1331 |
1332 | , tip: function () {
1333 | return this.$tip = this.$tip || $(this.options.template)
1334 | }
1335 |
1336 | , arrow: function(){
1337 | return this.$arrow = this.$arrow || this.tip().find(".tooltip-arrow")
1338 | }
1339 |
1340 | , validate: function () {
1341 | if (!this.$element[0].parentNode) {
1342 | this.hide()
1343 | this.$element = null
1344 | this.options = null
1345 | }
1346 | }
1347 |
1348 | , enable: function () {
1349 | this.enabled = true
1350 | }
1351 |
1352 | , disable: function () {
1353 | this.enabled = false
1354 | }
1355 |
1356 | , toggleEnabled: function () {
1357 | this.enabled = !this.enabled
1358 | }
1359 |
1360 | , toggle: function (e) {
1361 | var self = e ? $(e.currentTarget)[this.type](this._options).data(this.type) : this
1362 | self.tip().hasClass('in') ? self.hide() : self.show()
1363 | }
1364 |
1365 | , destroy: function () {
1366 | this.hide().$element.off('.' + this.type).removeData(this.type)
1367 | }
1368 |
1369 | }
1370 |
1371 |
1372 | /* TOOLTIP PLUGIN DEFINITION
1373 | * ========================= */
1374 |
1375 | var old = $.fn.tooltip
1376 |
1377 | $.fn.tooltip = function ( option ) {
1378 | return this.each(function () {
1379 | var $this = $(this)
1380 | , data = $this.data('tooltip')
1381 | , options = typeof option == 'object' && option
1382 | if (!data) $this.data('tooltip', (data = new Tooltip(this, options)))
1383 | if (typeof option == 'string') data[option]()
1384 | })
1385 | }
1386 |
1387 | $.fn.tooltip.Constructor = Tooltip
1388 |
1389 | $.fn.tooltip.defaults = {
1390 | animation: true
1391 | , placement: 'top'
1392 | , selector: false
1393 | , template: ''
1394 | , trigger: 'hover focus'
1395 | , title: ''
1396 | , delay: 0
1397 | , html: false
1398 | , container: false
1399 | }
1400 |
1401 |
1402 | /* TOOLTIP NO CONFLICT
1403 | * =================== */
1404 |
1405 | $.fn.tooltip.noConflict = function () {
1406 | $.fn.tooltip = old
1407 | return this
1408 | }
1409 |
1410 | }(window.jQuery);
1411 | /* ===========================================================
1412 | * bootstrap-popover.js v2.3.2
1413 | * http://twitter.github.com/bootstrap/javascript.html#popovers
1414 | * ===========================================================
1415 | * Copyright 2012 Twitter, Inc.
1416 | *
1417 | * Licensed under the Apache License, Version 2.0 (the "License");
1418 | * you may not use this file except in compliance with the License.
1419 | * You may obtain a copy of the License at
1420 | *
1421 | * http://www.apache.org/licenses/LICENSE-2.0
1422 | *
1423 | * Unless required by applicable law or agreed to in writing, software
1424 | * distributed under the License is distributed on an "AS IS" BASIS,
1425 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1426 | * See the License for the specific language governing permissions and
1427 | * limitations under the License.
1428 | * =========================================================== */
1429 |
1430 |
1431 | !function ($) {
1432 |
1433 | "use strict"; // jshint ;_;
1434 |
1435 |
1436 | /* POPOVER PUBLIC CLASS DEFINITION
1437 | * =============================== */
1438 |
1439 | var Popover = function (element, options) {
1440 | this.init('popover', element, options)
1441 | }
1442 |
1443 |
1444 | /* NOTE: POPOVER EXTENDS BOOTSTRAP-TOOLTIP.js
1445 | ========================================== */
1446 |
1447 | Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype, {
1448 |
1449 | constructor: Popover
1450 |
1451 | , setContent: function () {
1452 | var $tip = this.tip()
1453 | , title = this.getTitle()
1454 | , content = this.getContent()
1455 |
1456 | $tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
1457 | $tip.find('.popover-content')[this.options.html ? 'html' : 'text'](content)
1458 |
1459 | $tip.removeClass('fade top bottom left right in')
1460 | }
1461 |
1462 | , hasContent: function () {
1463 | return this.getTitle() || this.getContent()
1464 | }
1465 |
1466 | , getContent: function () {
1467 | var content
1468 | , $e = this.$element
1469 | , o = this.options
1470 |
1471 | content = (typeof o.content == 'function' ? o.content.call($e[0]) : o.content)
1472 | || $e.attr('data-content')
1473 |
1474 | return content
1475 | }
1476 |
1477 | , tip: function () {
1478 | if (!this.$tip) {
1479 | this.$tip = $(this.options.template)
1480 | }
1481 | return this.$tip
1482 | }
1483 |
1484 | , destroy: function () {
1485 | this.hide().$element.off('.' + this.type).removeData(this.type)
1486 | }
1487 |
1488 | })
1489 |
1490 |
1491 | /* POPOVER PLUGIN DEFINITION
1492 | * ======================= */
1493 |
1494 | var old = $.fn.popover
1495 |
1496 | $.fn.popover = function (option) {
1497 | return this.each(function () {
1498 | var $this = $(this)
1499 | , data = $this.data('popover')
1500 | , options = typeof option == 'object' && option
1501 | if (!data) $this.data('popover', (data = new Popover(this, options)))
1502 | if (typeof option == 'string') data[option]()
1503 | })
1504 | }
1505 |
1506 | $.fn.popover.Constructor = Popover
1507 |
1508 | $.fn.popover.defaults = $.extend({} , $.fn.tooltip.defaults, {
1509 | placement: 'right'
1510 | , trigger: 'click'
1511 | , content: ''
1512 | , template: ''
1513 | })
1514 |
1515 |
1516 | /* POPOVER NO CONFLICT
1517 | * =================== */
1518 |
1519 | $.fn.popover.noConflict = function () {
1520 | $.fn.popover = old
1521 | return this
1522 | }
1523 |
1524 | }(window.jQuery);
1525 | /* =============================================================
1526 | * bootstrap-scrollspy.js v2.3.2
1527 | * http://twitter.github.com/bootstrap/javascript.html#scrollspy
1528 | * =============================================================
1529 | * Copyright 2012 Twitter, Inc.
1530 | *
1531 | * Licensed under the Apache License, Version 2.0 (the "License");
1532 | * you may not use this file except in compliance with the License.
1533 | * You may obtain a copy of the License at
1534 | *
1535 | * http://www.apache.org/licenses/LICENSE-2.0
1536 | *
1537 | * Unless required by applicable law or agreed to in writing, software
1538 | * distributed under the License is distributed on an "AS IS" BASIS,
1539 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1540 | * See the License for the specific language governing permissions and
1541 | * limitations under the License.
1542 | * ============================================================== */
1543 |
1544 |
1545 | !function ($) {
1546 |
1547 | "use strict"; // jshint ;_;
1548 |
1549 |
1550 | /* SCROLLSPY CLASS DEFINITION
1551 | * ========================== */
1552 |
1553 | function ScrollSpy(element, options) {
1554 | var process = $.proxy(this.process, this)
1555 | , $element = $(element).is('body') ? $(window) : $(element)
1556 | , href
1557 | this.options = $.extend({}, $.fn.scrollspy.defaults, options)
1558 | this.$scrollElement = $element.on('scroll.scroll-spy.data-api', process)
1559 | this.selector = (this.options.target
1560 | || ((href = $(element).attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
1561 | || '') + ' .nav li > a'
1562 | this.$body = $('body')
1563 | this.refresh()
1564 | this.process()
1565 | }
1566 |
1567 | ScrollSpy.prototype = {
1568 |
1569 | constructor: ScrollSpy
1570 |
1571 | , refresh: function () {
1572 | var self = this
1573 | , $targets
1574 |
1575 | this.offsets = $([])
1576 | this.targets = $([])
1577 |
1578 | $targets = this.$body
1579 | .find(this.selector)
1580 | .map(function () {
1581 | var $el = $(this)
1582 | , href = $el.data('target') || $el.attr('href')
1583 | , $href = /^#\w/.test(href) && $(href)
1584 | return ( $href
1585 | && $href.length
1586 | && [[ $href.position().top + (!$.isWindow(self.$scrollElement.get(0)) && self.$scrollElement.scrollTop()), href ]] ) || null
1587 | })
1588 | .sort(function (a, b) { return a[0] - b[0] })
1589 | .each(function () {
1590 | self.offsets.push(this[0])
1591 | self.targets.push(this[1])
1592 | })
1593 | }
1594 |
1595 | , process: function () {
1596 | var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
1597 | , scrollHeight = this.$scrollElement[0].scrollHeight || this.$body[0].scrollHeight
1598 | , maxScroll = scrollHeight - this.$scrollElement.height()
1599 | , offsets = this.offsets
1600 | , targets = this.targets
1601 | , activeTarget = this.activeTarget
1602 | , i
1603 |
1604 | if (scrollTop >= maxScroll) {
1605 | return activeTarget != (i = targets.last()[0])
1606 | && this.activate ( i )
1607 | }
1608 |
1609 | for (i = offsets.length; i--;) {
1610 | activeTarget != targets[i]
1611 | && scrollTop >= offsets[i]
1612 | && (!offsets[i + 1] || scrollTop <= offsets[i + 1])
1613 | && this.activate( targets[i] )
1614 | }
1615 | }
1616 |
1617 | , activate: function (target) {
1618 | var active
1619 | , selector
1620 |
1621 | this.activeTarget = target
1622 |
1623 | $(this.selector)
1624 | .parent('.active')
1625 | .removeClass('active')
1626 |
1627 | selector = this.selector
1628 | + '[data-target="' + target + '"],'
1629 | + this.selector + '[href="' + target + '"]'
1630 |
1631 | active = $(selector)
1632 | .parent('li')
1633 | .addClass('active')
1634 |
1635 | if (active.parent('.dropdown-menu').length) {
1636 | active = active.closest('li.dropdown').addClass('active')
1637 | }
1638 |
1639 | active.trigger('activate')
1640 | }
1641 |
1642 | }
1643 |
1644 |
1645 | /* SCROLLSPY PLUGIN DEFINITION
1646 | * =========================== */
1647 |
1648 | var old = $.fn.scrollspy
1649 |
1650 | $.fn.scrollspy = function (option) {
1651 | return this.each(function () {
1652 | var $this = $(this)
1653 | , data = $this.data('scrollspy')
1654 | , options = typeof option == 'object' && option
1655 | if (!data) $this.data('scrollspy', (data = new ScrollSpy(this, options)))
1656 | if (typeof option == 'string') data[option]()
1657 | })
1658 | }
1659 |
1660 | $.fn.scrollspy.Constructor = ScrollSpy
1661 |
1662 | $.fn.scrollspy.defaults = {
1663 | offset: 10
1664 | }
1665 |
1666 |
1667 | /* SCROLLSPY NO CONFLICT
1668 | * ===================== */
1669 |
1670 | $.fn.scrollspy.noConflict = function () {
1671 | $.fn.scrollspy = old
1672 | return this
1673 | }
1674 |
1675 |
1676 | /* SCROLLSPY DATA-API
1677 | * ================== */
1678 |
1679 | $(window).on('load', function () {
1680 | $('[data-spy="scroll"]').each(function () {
1681 | var $spy = $(this)
1682 | $spy.scrollspy($spy.data())
1683 | })
1684 | })
1685 |
1686 | }(window.jQuery);/* ========================================================
1687 | * bootstrap-tab.js v2.3.2
1688 | * http://twitter.github.com/bootstrap/javascript.html#tabs
1689 | * ========================================================
1690 | * Copyright 2012 Twitter, Inc.
1691 | *
1692 | * Licensed under the Apache License, Version 2.0 (the "License");
1693 | * you may not use this file except in compliance with the License.
1694 | * You may obtain a copy of the License at
1695 | *
1696 | * http://www.apache.org/licenses/LICENSE-2.0
1697 | *
1698 | * Unless required by applicable law or agreed to in writing, software
1699 | * distributed under the License is distributed on an "AS IS" BASIS,
1700 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1701 | * See the License for the specific language governing permissions and
1702 | * limitations under the License.
1703 | * ======================================================== */
1704 |
1705 |
1706 | !function ($) {
1707 |
1708 | "use strict"; // jshint ;_;
1709 |
1710 |
1711 | /* TAB CLASS DEFINITION
1712 | * ==================== */
1713 |
1714 | var Tab = function (element) {
1715 | this.element = $(element)
1716 | }
1717 |
1718 | Tab.prototype = {
1719 |
1720 | constructor: Tab
1721 |
1722 | , show: function () {
1723 | var $this = this.element
1724 | , $ul = $this.closest('ul:not(.dropdown-menu)')
1725 | , selector = $this.attr('data-target')
1726 | , previous
1727 | , $target
1728 | , e
1729 |
1730 | if (!selector) {
1731 | selector = $this.attr('href')
1732 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
1733 | }
1734 |
1735 | if ( $this.parent('li').hasClass('active') ) return
1736 |
1737 | previous = $ul.find('.active:last a')[0]
1738 |
1739 | e = $.Event('show', {
1740 | relatedTarget: previous
1741 | })
1742 |
1743 | $this.trigger(e)
1744 |
1745 | if (e.isDefaultPrevented()) return
1746 |
1747 | $target = $(selector)
1748 |
1749 | this.activate($this.parent('li'), $ul)
1750 | this.activate($target, $target.parent(), function () {
1751 | $this.trigger({
1752 | type: 'shown'
1753 | , relatedTarget: previous
1754 | })
1755 | })
1756 | }
1757 |
1758 | , activate: function ( element, container, callback) {
1759 | var $active = container.find('> .active')
1760 | , transition = callback
1761 | && $.support.transition
1762 | && $active.hasClass('fade')
1763 |
1764 | function next() {
1765 | $active
1766 | .removeClass('active')
1767 | .find('> .dropdown-menu > .active')
1768 | .removeClass('active')
1769 |
1770 | element.addClass('active')
1771 |
1772 | if (transition) {
1773 | element[0].offsetWidth // reflow for transition
1774 | element.addClass('in')
1775 | } else {
1776 | element.removeClass('fade')
1777 | }
1778 |
1779 | if ( element.parent('.dropdown-menu') ) {
1780 | element.closest('li.dropdown').addClass('active')
1781 | }
1782 |
1783 | callback && callback()
1784 | }
1785 |
1786 | transition ?
1787 | $active.one($.support.transition.end, next) :
1788 | next()
1789 |
1790 | $active.removeClass('in')
1791 | }
1792 | }
1793 |
1794 |
1795 | /* TAB PLUGIN DEFINITION
1796 | * ===================== */
1797 |
1798 | var old = $.fn.tab
1799 |
1800 | $.fn.tab = function ( option ) {
1801 | return this.each(function () {
1802 | var $this = $(this)
1803 | , data = $this.data('tab')
1804 | if (!data) $this.data('tab', (data = new Tab(this)))
1805 | if (typeof option == 'string') data[option]()
1806 | })
1807 | }
1808 |
1809 | $.fn.tab.Constructor = Tab
1810 |
1811 |
1812 | /* TAB NO CONFLICT
1813 | * =============== */
1814 |
1815 | $.fn.tab.noConflict = function () {
1816 | $.fn.tab = old
1817 | return this
1818 | }
1819 |
1820 |
1821 | /* TAB DATA-API
1822 | * ============ */
1823 |
1824 | $(document).on('click.tab.data-api', '[data-toggle="tab"], [data-toggle="pill"]', function (e) {
1825 | e.preventDefault()
1826 | $(this).tab('show')
1827 | })
1828 |
1829 | }(window.jQuery);/* =============================================================
1830 | * bootstrap-typeahead.js v2.3.2
1831 | * http://twitter.github.com/bootstrap/javascript.html#typeahead
1832 | * =============================================================
1833 | * Copyright 2012 Twitter, Inc.
1834 | *
1835 | * Licensed under the Apache License, Version 2.0 (the "License");
1836 | * you may not use this file except in compliance with the License.
1837 | * You may obtain a copy of the License at
1838 | *
1839 | * http://www.apache.org/licenses/LICENSE-2.0
1840 | *
1841 | * Unless required by applicable law or agreed to in writing, software
1842 | * distributed under the License is distributed on an "AS IS" BASIS,
1843 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1844 | * See the License for the specific language governing permissions and
1845 | * limitations under the License.
1846 | * ============================================================ */
1847 |
1848 |
1849 | !function($){
1850 |
1851 | "use strict"; // jshint ;_;
1852 |
1853 |
1854 | /* TYPEAHEAD PUBLIC CLASS DEFINITION
1855 | * ================================= */
1856 |
1857 | var Typeahead = function (element, options) {
1858 | this.$element = $(element)
1859 | this.options = $.extend({}, $.fn.typeahead.defaults, options)
1860 | this.matcher = this.options.matcher || this.matcher
1861 | this.sorter = this.options.sorter || this.sorter
1862 | this.highlighter = this.options.highlighter || this.highlighter
1863 | this.updater = this.options.updater || this.updater
1864 | this.source = this.options.source
1865 | this.$menu = $(this.options.menu)
1866 | this.shown = false
1867 | this.listen()
1868 | }
1869 |
1870 | Typeahead.prototype = {
1871 |
1872 | constructor: Typeahead
1873 |
1874 | , select: function () {
1875 | var val = this.$menu.find('.active').attr('data-value')
1876 | this.$element
1877 | .val(this.updater(val))
1878 | .change()
1879 | return this.hide()
1880 | }
1881 |
1882 | , updater: function (item) {
1883 | return item
1884 | }
1885 |
1886 | , show: function () {
1887 | var pos = $.extend({}, this.$element.position(), {
1888 | height: this.$element[0].offsetHeight
1889 | })
1890 |
1891 | this.$menu
1892 | .insertAfter(this.$element)
1893 | .css({
1894 | top: pos.top + pos.height
1895 | , left: pos.left
1896 | })
1897 | .show()
1898 |
1899 | this.shown = true
1900 | return this
1901 | }
1902 |
1903 | , hide: function () {
1904 | this.$menu.hide()
1905 | this.shown = false
1906 | return this
1907 | }
1908 |
1909 | , lookup: function (event) {
1910 | var items
1911 |
1912 | this.query = this.$element.val()
1913 |
1914 | if (!this.query || this.query.length < this.options.minLength) {
1915 | return this.shown ? this.hide() : this
1916 | }
1917 |
1918 | items = $.isFunction(this.source) ? this.source(this.query, $.proxy(this.process, this)) : this.source
1919 |
1920 | return items ? this.process(items) : this
1921 | }
1922 |
1923 | , process: function (items) {
1924 | var that = this
1925 |
1926 | items = $.grep(items, function (item) {
1927 | return that.matcher(item)
1928 | })
1929 |
1930 | items = this.sorter(items)
1931 |
1932 | if (!items.length) {
1933 | return this.shown ? this.hide() : this
1934 | }
1935 |
1936 | return this.render(items.slice(0, this.options.items)).show()
1937 | }
1938 |
1939 | , matcher: function (item) {
1940 | return ~item.toLowerCase().indexOf(this.query.toLowerCase())
1941 | }
1942 |
1943 | , sorter: function (items) {
1944 | var beginswith = []
1945 | , caseSensitive = []
1946 | , caseInsensitive = []
1947 | , item
1948 |
1949 | while (item = items.shift()) {
1950 | if (!item.toLowerCase().indexOf(this.query.toLowerCase())) beginswith.push(item)
1951 | else if (~item.indexOf(this.query)) caseSensitive.push(item)
1952 | else caseInsensitive.push(item)
1953 | }
1954 |
1955 | return beginswith.concat(caseSensitive, caseInsensitive)
1956 | }
1957 |
1958 | , highlighter: function (item) {
1959 | var query = this.query.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&')
1960 | return item.replace(new RegExp('(' + query + ')', 'ig'), function ($1, match) {
1961 | return '' + match + ''
1962 | })
1963 | }
1964 |
1965 | , render: function (items) {
1966 | var that = this
1967 |
1968 | items = $(items).map(function (i, item) {
1969 | i = $(that.options.item).attr('data-value', item)
1970 | i.find('a').html(that.highlighter(item))
1971 | return i[0]
1972 | })
1973 |
1974 | items.first().addClass('active')
1975 | this.$menu.html(items)
1976 | return this
1977 | }
1978 |
1979 | , next: function (event) {
1980 | var active = this.$menu.find('.active').removeClass('active')
1981 | , next = active.next()
1982 |
1983 | if (!next.length) {
1984 | next = $(this.$menu.find('li')[0])
1985 | }
1986 |
1987 | next.addClass('active')
1988 | }
1989 |
1990 | , prev: function (event) {
1991 | var active = this.$menu.find('.active').removeClass('active')
1992 | , prev = active.prev()
1993 |
1994 | if (!prev.length) {
1995 | prev = this.$menu.find('li').last()
1996 | }
1997 |
1998 | prev.addClass('active')
1999 | }
2000 |
2001 | , listen: function () {
2002 | this.$element
2003 | .on('focus', $.proxy(this.focus, this))
2004 | .on('blur', $.proxy(this.blur, this))
2005 | .on('keypress', $.proxy(this.keypress, this))
2006 | .on('keyup', $.proxy(this.keyup, this))
2007 |
2008 | if (this.eventSupported('keydown')) {
2009 | this.$element.on('keydown', $.proxy(this.keydown, this))
2010 | }
2011 |
2012 | this.$menu
2013 | .on('click', $.proxy(this.click, this))
2014 | .on('mouseenter', 'li', $.proxy(this.mouseenter, this))
2015 | .on('mouseleave', 'li', $.proxy(this.mouseleave, this))
2016 | }
2017 |
2018 | , eventSupported: function(eventName) {
2019 | var isSupported = eventName in this.$element
2020 | if (!isSupported) {
2021 | this.$element.setAttribute(eventName, 'return;')
2022 | isSupported = typeof this.$element[eventName] === 'function'
2023 | }
2024 | return isSupported
2025 | }
2026 |
2027 | , move: function (e) {
2028 | if (!this.shown) return
2029 |
2030 | switch(e.keyCode) {
2031 | case 9: // tab
2032 | case 13: // enter
2033 | case 27: // escape
2034 | e.preventDefault()
2035 | break
2036 |
2037 | case 38: // up arrow
2038 | e.preventDefault()
2039 | this.prev()
2040 | break
2041 |
2042 | case 40: // down arrow
2043 | e.preventDefault()
2044 | this.next()
2045 | break
2046 | }
2047 |
2048 | e.stopPropagation()
2049 | }
2050 |
2051 | , keydown: function (e) {
2052 | this.suppressKeyPressRepeat = ~$.inArray(e.keyCode, [40,38,9,13,27])
2053 | this.move(e)
2054 | }
2055 |
2056 | , keypress: function (e) {
2057 | if (this.suppressKeyPressRepeat) return
2058 | this.move(e)
2059 | }
2060 |
2061 | , keyup: function (e) {
2062 | switch(e.keyCode) {
2063 | case 40: // down arrow
2064 | case 38: // up arrow
2065 | case 16: // shift
2066 | case 17: // ctrl
2067 | case 18: // alt
2068 | break
2069 |
2070 | case 9: // tab
2071 | case 13: // enter
2072 | if (!this.shown) return
2073 | this.select()
2074 | break
2075 |
2076 | case 27: // escape
2077 | if (!this.shown) return
2078 | this.hide()
2079 | break
2080 |
2081 | default:
2082 | this.lookup()
2083 | }
2084 |
2085 | e.stopPropagation()
2086 | e.preventDefault()
2087 | }
2088 |
2089 | , focus: function (e) {
2090 | this.focused = true
2091 | }
2092 |
2093 | , blur: function (e) {
2094 | this.focused = false
2095 | if (!this.mousedover && this.shown) this.hide()
2096 | }
2097 |
2098 | , click: function (e) {
2099 | e.stopPropagation()
2100 | e.preventDefault()
2101 | this.select()
2102 | this.$element.focus()
2103 | }
2104 |
2105 | , mouseenter: function (e) {
2106 | this.mousedover = true
2107 | this.$menu.find('.active').removeClass('active')
2108 | $(e.currentTarget).addClass('active')
2109 | }
2110 |
2111 | , mouseleave: function (e) {
2112 | this.mousedover = false
2113 | if (!this.focused && this.shown) this.hide()
2114 | }
2115 |
2116 | }
2117 |
2118 |
2119 | /* TYPEAHEAD PLUGIN DEFINITION
2120 | * =========================== */
2121 |
2122 | var old = $.fn.typeahead
2123 |
2124 | $.fn.typeahead = function (option) {
2125 | return this.each(function () {
2126 | var $this = $(this)
2127 | , data = $this.data('typeahead')
2128 | , options = typeof option == 'object' && option
2129 | if (!data) $this.data('typeahead', (data = new Typeahead(this, options)))
2130 | if (typeof option == 'string') data[option]()
2131 | })
2132 | }
2133 |
2134 | $.fn.typeahead.defaults = {
2135 | source: []
2136 | , items: 8
2137 | , menu: ''
2138 | , item: ''
2139 | , minLength: 1
2140 | }
2141 |
2142 | $.fn.typeahead.Constructor = Typeahead
2143 |
2144 |
2145 | /* TYPEAHEAD NO CONFLICT
2146 | * =================== */
2147 |
2148 | $.fn.typeahead.noConflict = function () {
2149 | $.fn.typeahead = old
2150 | return this
2151 | }
2152 |
2153 |
2154 | /* TYPEAHEAD DATA-API
2155 | * ================== */
2156 |
2157 | $(document).on('focus.typeahead.data-api', '[data-provide="typeahead"]', function (e) {
2158 | var $this = $(this)
2159 | if ($this.data('typeahead')) return
2160 | $this.typeahead($this.data())
2161 | })
2162 |
2163 | }(window.jQuery);
2164 | /* ==========================================================
2165 | * bootstrap-affix.js v2.3.2
2166 | * http://twitter.github.com/bootstrap/javascript.html#affix
2167 | * ==========================================================
2168 | * Copyright 2012 Twitter, Inc.
2169 | *
2170 | * Licensed under the Apache License, Version 2.0 (the "License");
2171 | * you may not use this file except in compliance with the License.
2172 | * You may obtain a copy of the License at
2173 | *
2174 | * http://www.apache.org/licenses/LICENSE-2.0
2175 | *
2176 | * Unless required by applicable law or agreed to in writing, software
2177 | * distributed under the License is distributed on an "AS IS" BASIS,
2178 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
2179 | * See the License for the specific language governing permissions and
2180 | * limitations under the License.
2181 | * ========================================================== */
2182 |
2183 |
2184 | !function ($) {
2185 |
2186 | "use strict"; // jshint ;_;
2187 |
2188 |
2189 | /* AFFIX CLASS DEFINITION
2190 | * ====================== */
2191 |
2192 | var Affix = function (element, options) {
2193 | this.options = $.extend({}, $.fn.affix.defaults, options)
2194 | this.$window = $(window)
2195 | .on('scroll.affix.data-api', $.proxy(this.checkPosition, this))
2196 | .on('click.affix.data-api', $.proxy(function () { setTimeout($.proxy(this.checkPosition, this), 1) }, this))
2197 | this.$element = $(element)
2198 | this.checkPosition()
2199 | }
2200 |
2201 | Affix.prototype.checkPosition = function () {
2202 | if (!this.$element.is(':visible')) return
2203 |
2204 | var scrollHeight = $(document).height()
2205 | , scrollTop = this.$window.scrollTop()
2206 | , position = this.$element.offset()
2207 | , offset = this.options.offset
2208 | , offsetBottom = offset.bottom
2209 | , offsetTop = offset.top
2210 | , reset = 'affix affix-top affix-bottom'
2211 | , affix
2212 |
2213 | if (typeof offset != 'object') offsetBottom = offsetTop = offset
2214 | if (typeof offsetTop == 'function') offsetTop = offset.top()
2215 | if (typeof offsetBottom == 'function') offsetBottom = offset.bottom()
2216 |
2217 | affix = this.unpin != null && (scrollTop + this.unpin <= position.top) ?
2218 | false : offsetBottom != null && (position.top + this.$element.height() >= scrollHeight - offsetBottom) ?
2219 | 'bottom' : offsetTop != null && scrollTop <= offsetTop ?
2220 | 'top' : false
2221 |
2222 | if (this.affixed === affix) return
2223 |
2224 | this.affixed = affix
2225 | this.unpin = affix == 'bottom' ? position.top - scrollTop : null
2226 |
2227 | this.$element.removeClass(reset).addClass('affix' + (affix ? '-' + affix : ''))
2228 | }
2229 |
2230 |
2231 | /* AFFIX PLUGIN DEFINITION
2232 | * ======================= */
2233 |
2234 | var old = $.fn.affix
2235 |
2236 | $.fn.affix = function (option) {
2237 | return this.each(function () {
2238 | var $this = $(this)
2239 | , data = $this.data('affix')
2240 | , options = typeof option == 'object' && option
2241 | if (!data) $this.data('affix', (data = new Affix(this, options)))
2242 | if (typeof option == 'string') data[option]()
2243 | })
2244 | }
2245 |
2246 | $.fn.affix.Constructor = Affix
2247 |
2248 | $.fn.affix.defaults = {
2249 | offset: 0
2250 | }
2251 |
2252 |
2253 | /* AFFIX NO CONFLICT
2254 | * ================= */
2255 |
2256 | $.fn.affix.noConflict = function () {
2257 | $.fn.affix = old
2258 | return this
2259 | }
2260 |
2261 |
2262 | /* AFFIX DATA-API
2263 | * ============== */
2264 |
2265 | $(window).on('load', function () {
2266 | $('[data-spy="affix"]').each(function () {
2267 | var $spy = $(this)
2268 | , data = $spy.data()
2269 |
2270 | data.offset = data.offset || {}
2271 |
2272 | data.offsetBottom && (data.offset.bottom = data.offsetBottom)
2273 | data.offsetTop && (data.offset.top = data.offsetTop)
2274 |
2275 | $spy.affix(data)
2276 | })
2277 | })
2278 |
2279 |
2280 | }(window.jQuery);
--------------------------------------------------------------------------------
/src/main/webapp/assets/js/jquery.loadTemplate-0.5.4.js:
--------------------------------------------------------------------------------
1 | (function ($) {
2 | "use strict";
3 | var templates = {},
4 | queue = {},
5 | formatters = {};
6 |
7 | function loadTemplate(template, data, options) {
8 | var $that = this,
9 | $template,
10 | settings,
11 | isFile;
12 |
13 | data = data || {};
14 |
15 | settings = $.extend({
16 | // These are the defaults.
17 | overwriteCache: false,
18 | complete: null,
19 | success: null,
20 | error: function () {
21 | $(this).each(function () {
22 | $(this).html(settings.errorMessage);
23 | });
24 | },
25 | errorMessage: "There was an error loading the template.",
26 | paged: false,
27 | pageNo: 1,
28 | elemPerPage: 10
29 | }, options);
30 |
31 | if ($.type(data) === "array") {
32 | return processArray.call(this, template, data, settings);
33 | }
34 |
35 | if (!containsSlashes(template)) {
36 | $template = $(template);
37 | }
38 |
39 | isFile = settings.isFile || (typeof settings.isFile === "undefined" && (typeof $template === "undefined" || $template.length === 0));
40 |
41 | if (isFile && !settings.overwriteCache && templates[template]) {
42 | prepareTemplateFromCache(template, $that, data, settings);
43 | } else if (isFile && !settings.overwriteCache && templates.hasOwnProperty(template)) {
44 | addToQueue(template, $that, data, settings);
45 | } else if (isFile) {
46 | loadAndPrepareTemplate(template, $that, data, settings);
47 | } else {
48 | loadTemplateFromDocument($template, $that, data, settings);
49 | }
50 | return this;
51 | }
52 |
53 | function addTemplateFormatter(key, formatter) {
54 | if (formatter) {
55 | formatters[key] = formatter;
56 | } else {
57 | formatters = $.extend(formatters, key);
58 | }
59 | }
60 |
61 | function containsSlashes(str) {
62 | return typeof str === "string" && str.indexOf("/") > -1;
63 | }
64 |
65 | function processArray(template, data, options) {
66 | var $that = this,
67 | todo = data.length,
68 | done = 0,
69 | newOptions;
70 |
71 | options = options || {};
72 |
73 | if (options.paged) {
74 | var startNo = (options.pageNo - 1) * options.elemPerPage;
75 | data = data.slice(startNo, startNo + options.elemPerPage);
76 | }
77 |
78 | newOptions = $.extend(
79 | {},
80 | options,
81 | {
82 | complete: function () {
83 | $that.append(this.html());
84 | done++;
85 | if (done === todo) {
86 | if (options && typeof options.complete === "function") {
87 | options.complete();
88 | }
89 | }
90 | }
91 | }
92 | );
93 |
94 | $that.html("");
95 |
96 | $(data).each(function () {
97 | var $div = $("");
98 | loadTemplate.call($div, template, this, newOptions);
99 | });
100 |
101 | return this;
102 | }
103 |
104 | function addToQueue(template, selection, data, settings) {
105 | if (queue[template]) {
106 | queue[template].push({ data: data, selection: selection, settings: settings });
107 | } else {
108 | queue[template] = [{ data: data, selection: selection, settings: settings}];
109 | }
110 | }
111 |
112 | function prepareTemplateFromCache(template, selection, data, settings) {
113 | var $templateContainer = templates[template].clone();
114 |
115 | prepareTemplate.call(selection, $templateContainer, data, settings.complete);
116 | if (typeof settings.success === "function") {
117 | settings.success();
118 | }
119 | }
120 |
121 | function loadAndPrepareTemplate(template, selection, data, settings) {
122 | var $templateContainer = $("");
123 |
124 | templates[template] = null;
125 | $templateContainer.load(template, function (responseText, textStatus) {
126 | if (textStatus === "error") {
127 | handleTemplateLoadingError(template, selection, data, settings);
128 | } else {
129 | handleTemplateLoadingSuccess($templateContainer, template, selection, data, settings);
130 | }
131 | });
132 | }
133 |
134 | function loadTemplateFromDocument($template, selection, data, settings) {
135 | var $templateContainer = $("");
136 |
137 | if ($template.is("script")) {
138 | $template = $.parseHTML($.trim($template.html()));
139 | }
140 |
141 | $templateContainer.html($template);
142 | prepareTemplate.call(selection, $templateContainer, data, settings.complete);
143 |
144 | if (typeof settings.success === "function") {
145 | settings.success();
146 | }
147 | }
148 |
149 | function prepareTemplate(template, data, complete) {
150 | bindData(template, data);
151 |
152 | $(this).each(function () {
153 | $(this).html(template.html());
154 | });
155 |
156 | if (typeof complete === "function") {
157 | complete.call($(this));
158 | }
159 | }
160 |
161 | function handleTemplateLoadingError(template, selection, data, settings) {
162 | var value;
163 |
164 | if (typeof settings.error === "function") {
165 | settings.error.call(selection);
166 | }
167 |
168 | $(queue[template]).each(function (key, value) {
169 | if (typeof value.settings.error === "function") {
170 | value.settings.error.call(value.selection);
171 | }
172 | });
173 |
174 | if (typeof settings.complete === "function") {
175 | settings.complete.call(selection);
176 | }
177 |
178 | while (queue[template] && (value = queue[template].shift())) {
179 | if (typeof value.settings.complete === "function") {
180 | value.settings.complete.call(value.selection);
181 | }
182 | }
183 |
184 | if (typeof queue[template] !== 'undefined' && queue[template].length > 0) {
185 | queue[template] = [];
186 | }
187 | }
188 |
189 | function handleTemplateLoadingSuccess($templateContainer, template, selection, data, settings) {
190 | var value;
191 |
192 | templates[template] = $templateContainer.clone();
193 | prepareTemplate.call(selection, $templateContainer, data, settings.complete);
194 |
195 | if (typeof settings.success === "function") {
196 | settings.success.call(selection);
197 | }
198 |
199 | while (queue[template] && (value = queue[template].shift())) {
200 | prepareTemplate.call(value.selection, templates[template].clone(), value.data, value.settings.complete);
201 | if (typeof value.settings.success === "function") {
202 | value.settings.success.call(value.selection);
203 | }
204 | }
205 | }
206 |
207 | function bindData(template, data) {
208 | data = data || {};
209 |
210 | processElements("data-content", template, data, function ($elem, value) {
211 | $elem.html(applyFormatters($elem, value, "content"));
212 | });
213 |
214 | processElements("data-content-append", template, data, function ($elem, value) {
215 | $elem.append(applyFormatters($elem, value, "content"));
216 | });
217 |
218 | processElements("data-content-prepend", template, data, function ($elem, value) {
219 | $elem.prepend(applyFormatters($elem, value, "content"));
220 | });
221 |
222 | processElements("data-src", template, data, function ($elem, value) {
223 | $elem.attr("src", applyFormatters($elem, value, "src"));
224 | }, function ($elem) {
225 | $elem.remove();
226 | });
227 |
228 | processElements("data-alt", template, data, function ($elem, value) {
229 | $elem.attr("alt", applyFormatters($elem, value, "alt"));
230 | });
231 |
232 | processElements("data-link", template, data, function ($elem, value) {
233 | var $linkElem = $("");
234 | $linkElem.attr("href", applyFormatters($elem, value, "link"));
235 | $linkElem.html($elem.html());
236 | $elem.html($linkElem);
237 | });
238 |
239 | processElements("data-link-wrap", template, data, function ($elem, value) {
240 | var $linkElem = $("");
241 | $linkElem.attr("href", applyFormatters($elem, value, "link-wrap"));
242 | $elem.wrap($linkElem);
243 | });
244 |
245 | processElements("data-options", template, data, function ($elem, value) {
246 | $(value).each(function () {
247 | var $option = $("");
248 | $option.attr('value', this).text(this).appendTo($elem);
249 | });
250 | });
251 |
252 | processAllElements(template, data);
253 | }
254 |
255 | function processElements(attribute, template, data, dataBindFunction, noDataFunction) {
256 | $("[" + attribute + "]", template).each(function () {
257 | var $this = $(this),
258 | param = $this.attr(attribute),
259 | value = getValue(data, param);
260 |
261 | $this.removeAttr(attribute);
262 |
263 | if (value && dataBindFunction) {
264 | dataBindFunction($this, value);
265 | } else if (noDataFunction) {
266 | noDataFunction($this);
267 | }
268 | });
269 | return;
270 | }
271 |
272 | function processAllElements(template, data) {
273 | $("[data-template-bind]", template).each(function () {
274 | var $this = $(this),
275 | param = $.parseJSON($this.attr("data-template-bind"));
276 |
277 | $this.removeAttr("data-template-bind");
278 |
279 | $(param).each(function () {
280 | var value;
281 |
282 | if (typeof (this.value) === 'object') {
283 | value = getValue(data, this.value.data);
284 | } else {
285 | value = getValue(data, this.value);
286 | }
287 | if (typeof value !== "undefined" && this.attribute) {
288 | switch (this.attribute) {
289 | case "content":
290 | $this.html(applyDataBindFormatters(value, this));
291 | break;
292 | case "contentAppend":
293 | $this.append(applyDataBindFormatters(value, this));
294 | break;
295 | case "contentPrepend":
296 | $this.prepend(applyDataBindFormatters(value, this));
297 | break;
298 | case "options":
299 | var optionsData = this;
300 | $(value).each(function () {
301 | var $option = $("");
302 | $option.attr('value', this[optionsData.value.value]).text(applyDataBindFormatters(this[optionsData.value.content], optionsData)).appendTo($this);
303 | });
304 | break;
305 | default:
306 | $this.attr(this.attribute, applyDataBindFormatters(value, this));
307 | }
308 | }
309 | });
310 | });
311 | }
312 |
313 | function applyDataBindFormatters(value, data) {
314 | if (data.formatter && formatters[data.formatter]) {
315 | return formatters[data.formatter](value, data.formatOptions);
316 | }
317 | return value;
318 | }
319 |
320 | function getValue(data, param) {
321 | var paramParts = param.split('.'),
322 | part,
323 | value = data;
324 |
325 | while ((part = paramParts.shift()) && typeof value !== "undefined") {
326 | value = value[part];
327 | }
328 |
329 | return value;
330 | }
331 |
332 | function applyFormatters($elem, value, attr) {
333 | var formatterTarget = $elem.attr("data-format-target"),
334 | formatter;
335 |
336 | if (formatterTarget === attr || (!formatterTarget && attr === "content")) {
337 | formatter = $elem.attr("data-format");
338 | if (formatter && typeof formatters[formatter] === "function") {
339 | var formatOptions = $elem.attr("data-format-options");
340 | return formatters[formatter](value, formatOptions);
341 | }
342 | }
343 |
344 | return value;
345 | }
346 |
347 | $.fn.loadTemplate = loadTemplate;
348 | $.addTemplateFormatter = addTemplateFormatter;
349 |
350 | })(jQuery);
--------------------------------------------------------------------------------
/src/main/webapp/assets/js/knockout-2.3.0.js:
--------------------------------------------------------------------------------
1 | // Knockout JavaScript library v2.3.0
2 | // (c) Steven Sanderson - http://knockoutjs.com/
3 | // License: MIT (http://www.opensource.org/licenses/mit-license.php)
4 |
5 | (function() {function F(q){return function(){return q}};(function(q){var w=this||(0,eval)("this"),s=w.document,H=w.navigator,t=w.jQuery,y=w.JSON;(function(q){"function"===typeof require&&"object"===typeof exports&&"object"===typeof module?q(module.exports||exports):"function"===typeof define&&define.amd?define(["exports"],q):q(w.ko={})})(function(C){function G(b,c,d,f){a.d[b]={init:function(b){a.a.f.set(b,I,{});return{controlsDescendantBindings:!0}},update:function(b,e,m,h,k){m=a.a.f.get(b,I);e=a.a.c(e());h=!d!==!e;var l=!m.fb;if(l||c||h!==m.vb)l&&(m.fb=
6 | a.a.Oa(a.e.childNodes(b),!0)),h?(l||a.e.P(b,a.a.Oa(m.fb)),a.Ja(f?f(k,e):k,b)):a.e.ba(b),m.vb=h}};a.g.S[b]=!1;a.e.L[b]=!0}function J(b,c,d){d&&c!==a.h.n(b)&&a.h.W(b,c);c!==a.h.n(b)&&a.q.I(a.a.Ga,null,[b,"change"])}var a="undefined"!==typeof C?C:{};a.b=function(b,c){for(var d=b.split("."),f=a,g=0;ga.a.k(e,b[c])&&e.push(b[c]);return e},Z:function(a,b){a=a||[];for(var e=[],c=0,d=a.length;cd?c&&b.push(e):c||b.splice(d,1)},
10 | extend:function(a,b){if(b)for(var e in b)b.hasOwnProperty(e)&&(a[e]=b[e]);return a},w:b,oa:function(b){for(;b.firstChild;)a.removeNode(b.firstChild)},Mb:function(b){b=a.a.N(b);for(var e=s.createElement("div"),c=0,d=b.length;ce?a.setAttribute("selected",b):a.selected=b},F:function(a){return null===a||a===q?"":a.trim?a.trim():a.toString().replace(/^[\s\xa0]+|[\s\xa0]+$/g,"")},Wb:function(b,e){for(var c=[],d=(b||"").split(e),g=0,f=d.length;ga.length?!1:a.substring(0,b.length)===
12 | b},yb:function(a,b){if(b.compareDocumentPosition)return 16==(b.compareDocumentPosition(a)&16);for(;null!=a;){if(a==b)return!0;a=a.parentNode}return!1},aa:function(b){return a.a.yb(b,b.ownerDocument)},pb:function(b){return!!a.a.La(b,a.a.aa)},u:function(a){return a&&a.tagName&&a.tagName.toLowerCase()},o:function(b,d,k){var f=e&&g[d];if(f||"undefined"==typeof t)if(f||"function"!=typeof b.addEventListener)if("undefined"!=typeof b.attachEvent){var n=function(a){k.call(b,a)},p="on"+d;b.attachEvent(p,n);
13 | a.a.C.ia(b,function(){b.detachEvent(p,n)})}else throw Error("Browser doesn't support addEventListener or attachEvent");else b.addEventListener(d,k,!1);else{if(c(b,d)){var r=k;k=function(a,b){var e=this.checked;b&&(this.checked=!0!==b.sb);r.call(this,a);this.checked=e}}t(b).bind(d,k)}},Ga:function(a,b){if(!a||!a.nodeType)throw Error("element must be a DOM node when calling triggerEvent");if("undefined"!=typeof t){var e=[];c(a,b)&&e.push({sb:a.checked});t(a).trigger(b,e)}else if("function"==typeof s.createEvent)if("function"==
14 | typeof a.dispatchEvent)e=s.createEvent(f[b]||"HTMLEvents"),e.initEvent(b,!0,!0,w,0,0,0,0,0,!1,!1,!1,!1,0,a),a.dispatchEvent(e);else throw Error("The supplied element doesn't support dispatchEvent");else if("undefined"!=typeof a.fireEvent)c(a,b)&&(a.checked=!0!==a.checked),a.fireEvent("on"+b);else throw Error("Browser doesn't support triggering events");},c:function(b){return a.T(b)?b():b},ya:function(b){return a.T(b)?b.t():b},ga:function(b,e,c){if(e){var d=/\S+/g,g=b.className.match(d)||[];a.a.p(e.match(d),
15 | function(b){a.a.ja(g,b,c)});b.className=g.join(" ")}},ib:function(b,e){var c=a.a.c(e);if(null===c||c===q)c="";var d=a.e.firstChild(b);!d||3!=d.nodeType||a.e.nextSibling(d)?a.e.P(b,[s.createTextNode(c)]):d.data=c;a.a.Bb(b)},gb:function(a,b){a.name=b;if(7>=e)try{a.mergeAttributes(s.createElement(""),!1)}catch(c){}},Bb:function(a){9<=e&&(a=1==a.nodeType?a:a.parentNode,a.style&&(a.style.zoom=a.style.zoom))},zb:function(a){if(e){var b=a.style.width;a.style.width=0;a.style.width=
16 | b}},Qb:function(b,e){b=a.a.c(b);e=a.a.c(e);for(var c=[],d=b;d<=e;d++)c.push(d);return c},N:function(a){for(var b=[],e=0,c=a.length;e",""]||!d.indexOf("
",""]||(!d.indexOf(" | ","
"]||[0,"",""];b="ignored"+d[1]+b+d[2]+"
";for("function"==typeof w.innerShiv?
25 | c.appendChild(w.innerShiv(b)):c.innerHTML=b;d[0]--;)c=c.lastChild;c=a.a.N(c.lastChild.childNodes)}return c};a.a.fa=function(b,c){a.a.oa(b);c=a.a.c(c);if(null!==c&&c!==q)if("string"!=typeof c&&(c=c.toString()),"undefined"!=typeof t)t(b).html(c);else for(var d=a.a.xa(c),f=0;fe;e++)b=b();return b})};a.toJSON=function(b,c,e){b=a.lb(b);return a.a.Ca(b,c,e)};d.prototype={save:function(b,c){var e=a.a.k(this.keys,
40 | b);0<=e?this.Ha[e]=c:(this.keys.push(b),this.Ha.push(c))},get:function(b){b=a.a.k(this.keys,b);return 0<=b?this.Ha[b]:q}}})();a.b("toJS",a.lb);a.b("toJSON",a.toJSON);(function(){a.h={n:function(b){switch(a.a.u(b)){case "option":return!0===b.__ko__hasDomDataOptionValue__?a.a.f.get(b,a.d.options.wa):7>=a.a.ca?b.getAttributeNode("value")&&b.getAttributeNode("value").specified?b.value:b.text:b.value;case "select":return 0<=b.selectedIndex?a.h.n(b.options[b.selectedIndex]):q;default:return b.value}},W:function(b,
41 | c){switch(a.a.u(b)){case "option":switch(typeof c){case "string":a.a.f.set(b,a.d.options.wa,q);"__ko__hasDomDataOptionValue__"in b&&delete b.__ko__hasDomDataOptionValue__;b.value=c;break;default:a.a.f.set(b,a.d.options.wa,c),b.__ko__hasDomDataOptionValue__=!0,b.value="number"===typeof c?c:""}break;case "select":""===c&&(c=q);if(null===c||c===q)b.selectedIndex=-1;for(var d=b.options.length-1;0<=d;d--)if(a.h.n(b.options[d])==c){b.selectedIndex=d;break}1e.length)return[];"{"===e.charAt(0)&&(e=e.substring(1,e.length-
43 | 1));c=[];for(var d=null,f,k=0;k=a.a.ca&&c in K?(c=K[c],e?b.removeAttribute(c):b[c]=d):e||b.setAttribute(c,d.toString());"name"===c&&a.a.gb(b,e?"":d.toString())})}};a.d.checked={init:function(b,c,d){a.a.o(b,"click",function(){var f;if("checkbox"==b.type)f=b.checked;else if("radio"==b.type&&b.checked)f=
58 | b.value;else return;var g=c(),e=a.a.c(g);"checkbox"==b.type&&e instanceof Array?a.a.ja(g,b.value,b.checked):a.g.ha(g,d,"checked",f,!0)});"radio"!=b.type||b.name||a.d.uniqueName.init(b,F(!0))},update:function(b,c){var d=a.a.c(c());"checkbox"==b.type?b.checked=d instanceof Array?0<=a.a.k(d,b.value):d:"radio"==b.type&&(b.checked=b.value==d)}};a.d.css={update:function(b,c){var d=a.a.c(c());"object"==typeof d?a.a.w(d,function(c,d){d=a.a.c(d);a.a.ga(b,c,d)}):(d=String(d||""),a.a.ga(b,b.__ko__cssValue,!1),
59 | b.__ko__cssValue=d,a.a.ga(b,d,!0))}};a.d.enable={update:function(b,c){var d=a.a.c(c());d&&b.disabled?b.removeAttribute("disabled"):d||b.disabled||(b.disabled=!0)}};a.d.disable={update:function(b,c){a.d.enable.update(b,function(){return!a.a.c(c())})}};a.d.event={init:function(b,c,d,f){var g=c()||{};a.a.w(g,function(e){"string"==typeof e&&a.a.o(b,e,function(b){var g,k=c()[e];if(k){var l=d();try{var n=a.a.N(arguments);n.unshift(f);g=k.apply(f,n)}finally{!0!==g&&(b.preventDefault?b.preventDefault():b.returnValue=
60 | !1)}!1===l[e+"Bubble"]&&(b.cancelBubble=!0,b.stopPropagation&&b.stopPropagation())}})})}};a.d.foreach={Ya:function(b){return function(){var c=b(),d=a.a.ya(c);if(!d||"number"==typeof d.length)return{foreach:c,templateEngine:a.D.sa};a.a.c(c);return{foreach:d.data,as:d.as,includeDestroyed:d.includeDestroyed,afterAdd:d.afterAdd,beforeRemove:d.beforeRemove,afterRender:d.afterRender,beforeMove:d.beforeMove,afterMove:d.afterMove,templateEngine:a.D.sa}}},init:function(b,c){return a.d.template.init(b,a.d.foreach.Ya(c))},
61 | update:function(b,c,d,f,g){return a.d.template.update(b,a.d.foreach.Ya(c),d,f,g)}};a.g.S.foreach=!1;a.e.L.foreach=!0;a.d.hasfocus={init:function(b,c,d){function f(e){b.__ko_hasfocusUpdating=!0;var f=b.ownerDocument;if("activeElement"in f){var g;try{g=f.activeElement}catch(l){g=f.body}e=g===b}f=c();a.g.ha(f,d,"hasfocus",e,!0);b.__ko_hasfocusLastValue=e;b.__ko_hasfocusUpdating=!1}var g=f.bind(null,!0),e=f.bind(null,!1);a.a.o(b,"focus",g);a.a.o(b,"focusin",g);a.a.o(b,"blur",e);a.a.o(b,"focusout",e)},
62 | update:function(b,c){var d=!!a.a.c(c());b.__ko_hasfocusUpdating||b.__ko_hasfocusLastValue===d||(d?b.focus():b.blur(),a.q.I(a.a.Ga,null,[b,d?"focusin":"focusout"]))}};a.d.hasFocus=a.d.hasfocus;a.d.html={init:function(){return{controlsDescendantBindings:!0}},update:function(b,c){a.a.fa(b,c())}};var I="__ko_withIfBindingData";G("if");G("ifnot",!1,!0);G("with",!0,!1,function(a,c){return a.createChildContext(c)});a.d.options={init:function(b){if("select"!==a.a.u(b))throw Error("options binding applies only to SELECT elements");
63 | for(;0a.a.ca?0:b.nodes)?b.nodes():null;if(c)return a.a.N(c.cloneNode(!0).childNodes);b=b.text();return a.a.xa(b)};a.D.sa=new a.D;a.Ba(a.D.sa);a.b("nativeTemplateEngine",a.D);(function(){a.ua=function(){var a=this.Ib=function(){if("undefined"==typeof t||!t.tmpl)return 0;try{if(0<=t.tmpl.tag.tmpl.open.toString().indexOf("__"))return 2}catch(a){}return 1}();this.renderTemplateSource=
86 | function(b,f,g){g=g||{};if(2>a)throw Error("Your version of jQuery.tmpl is too old. Please upgrade to jQuery.tmpl 1.0.0pre or later.");var e=b.data("precompiled");e||(e=b.text()||"",e=t.template(null,"{{ko_with $item.koBindingContext}}"+e+"{{/ko_with}}"),b.data("precompiled",e));b=[f.$data];f=t.extend({koBindingContext:f},g.templateOptions);f=t.tmpl(e,b,f);f.appendTo(s.createElement("div"));t.fragments={};return f};this.createJavaScriptEvaluatorBlock=function(a){return"{{ko_code ((function() { return "+
87 | a+" })()) }}"};this.addTemplate=function(a,b){s.write("
60 |
61 |
62 |
63 |
64 |
65 |
66 |
97 |
98 |
112 |